xref: /openbmc/linux/drivers/ptp/ptp_dte.c (revision 7ae9fb1b7ecbb5d85d07857943f677fd1a559b18)
12aec85b2SThomas Gleixner // SPDX-License-Identifier: GPL-2.0-only
22aec85b2SThomas Gleixner // Copyright 2017 Broadcom
38a56aa10SArun Parameswaran 
48a56aa10SArun Parameswaran #include <linux/err.h>
58a56aa10SArun Parameswaran #include <linux/io.h>
68a56aa10SArun Parameswaran #include <linux/module.h>
7ac316725SRandy Dunlap #include <linux/mod_devicetable.h>
88a56aa10SArun Parameswaran #include <linux/platform_device.h>
98a56aa10SArun Parameswaran #include <linux/ptp_clock_kernel.h>
108a56aa10SArun Parameswaran #include <linux/types.h>
118a56aa10SArun Parameswaran 
128a56aa10SArun Parameswaran #define DTE_NCO_LOW_TIME_REG	0x00
138a56aa10SArun Parameswaran #define DTE_NCO_TIME_REG	0x04
148a56aa10SArun Parameswaran #define DTE_NCO_OVERFLOW_REG	0x08
158a56aa10SArun Parameswaran #define DTE_NCO_INC_REG		0x0c
168a56aa10SArun Parameswaran 
178a56aa10SArun Parameswaran #define DTE_NCO_SUM2_MASK	0xffffffff
188a56aa10SArun Parameswaran #define DTE_NCO_SUM2_SHIFT	4ULL
198a56aa10SArun Parameswaran 
208a56aa10SArun Parameswaran #define DTE_NCO_SUM3_MASK	0xff
218a56aa10SArun Parameswaran #define DTE_NCO_SUM3_SHIFT	36ULL
228a56aa10SArun Parameswaran #define DTE_NCO_SUM3_WR_SHIFT	8
238a56aa10SArun Parameswaran 
248a56aa10SArun Parameswaran #define DTE_NCO_TS_WRAP_MASK	0xfff
258a56aa10SArun Parameswaran #define DTE_NCO_TS_WRAP_LSHIFT	32
268a56aa10SArun Parameswaran 
278a56aa10SArun Parameswaran #define DTE_NCO_INC_DEFAULT	0x80000000
288a56aa10SArun Parameswaran #define DTE_NUM_REGS_TO_RESTORE	4
298a56aa10SArun Parameswaran 
308a56aa10SArun Parameswaran /* Full wrap around is 44bits in ns (~4.887 hrs) */
318a56aa10SArun Parameswaran #define DTE_WRAP_AROUND_NSEC_SHIFT 44
328a56aa10SArun Parameswaran 
338a56aa10SArun Parameswaran /* 44 bits NCO */
343d05035eSGeert Uytterhoeven #define DTE_NCO_MAX_NS	0xFFFFFFFFFFFLL
358a56aa10SArun Parameswaran 
368a56aa10SArun Parameswaran /* 125MHz with 3.29 reg cfg */
378a56aa10SArun Parameswaran #define DTE_PPB_ADJ(ppb) (u32)(div64_u64((((u64)abs(ppb) * BIT(28)) +\
388a56aa10SArun Parameswaran 				      62500000ULL), 125000000ULL))
398a56aa10SArun Parameswaran 
408a56aa10SArun Parameswaran /* ptp dte priv structure */
418a56aa10SArun Parameswaran struct ptp_dte {
428a56aa10SArun Parameswaran 	void __iomem *regs;
438a56aa10SArun Parameswaran 	struct ptp_clock *ptp_clk;
448a56aa10SArun Parameswaran 	struct ptp_clock_info caps;
458a56aa10SArun Parameswaran 	struct device *dev;
468a56aa10SArun Parameswaran 	u32 ts_ovf_last;
478a56aa10SArun Parameswaran 	u32 ts_wrap_cnt;
488a56aa10SArun Parameswaran 	spinlock_t lock;
498a56aa10SArun Parameswaran 	u32 reg_val[DTE_NUM_REGS_TO_RESTORE];
508a56aa10SArun Parameswaran };
518a56aa10SArun Parameswaran 
dte_write_nco(void __iomem * regs,s64 ns)528a56aa10SArun Parameswaran static void dte_write_nco(void __iomem *regs, s64 ns)
538a56aa10SArun Parameswaran {
548a56aa10SArun Parameswaran 	u32 sum2, sum3;
558a56aa10SArun Parameswaran 
568a56aa10SArun Parameswaran 	sum2 = (u32)((ns >> DTE_NCO_SUM2_SHIFT) & DTE_NCO_SUM2_MASK);
578a56aa10SArun Parameswaran 	/* compensate for ignoring sum1 */
588a56aa10SArun Parameswaran 	if (sum2 != DTE_NCO_SUM2_MASK)
598a56aa10SArun Parameswaran 		sum2++;
608a56aa10SArun Parameswaran 
618a56aa10SArun Parameswaran 	/* to write sum3, bits [15:8] needs to be written */
628a56aa10SArun Parameswaran 	sum3 = (u32)(((ns >> DTE_NCO_SUM3_SHIFT) & DTE_NCO_SUM3_MASK) <<
638a56aa10SArun Parameswaran 		     DTE_NCO_SUM3_WR_SHIFT);
648a56aa10SArun Parameswaran 
658a56aa10SArun Parameswaran 	writel(0, (regs + DTE_NCO_LOW_TIME_REG));
668a56aa10SArun Parameswaran 	writel(sum2, (regs + DTE_NCO_TIME_REG));
678a56aa10SArun Parameswaran 	writel(sum3, (regs + DTE_NCO_OVERFLOW_REG));
688a56aa10SArun Parameswaran }
698a56aa10SArun Parameswaran 
dte_read_nco(void __iomem * regs)708a56aa10SArun Parameswaran static s64 dte_read_nco(void __iomem *regs)
718a56aa10SArun Parameswaran {
728a56aa10SArun Parameswaran 	u32 sum2, sum3;
738a56aa10SArun Parameswaran 	s64 ns;
748a56aa10SArun Parameswaran 
758a56aa10SArun Parameswaran 	/*
768a56aa10SArun Parameswaran 	 * ignoring sum1 (4 bits) gives a 16ns resolution, which
778a56aa10SArun Parameswaran 	 * works due to the async register read.
788a56aa10SArun Parameswaran 	 */
798a56aa10SArun Parameswaran 	sum3 = readl(regs + DTE_NCO_OVERFLOW_REG) & DTE_NCO_SUM3_MASK;
808a56aa10SArun Parameswaran 	sum2 = readl(regs + DTE_NCO_TIME_REG);
818a56aa10SArun Parameswaran 	ns = ((s64)sum3 << DTE_NCO_SUM3_SHIFT) |
828a56aa10SArun Parameswaran 		 ((s64)sum2 << DTE_NCO_SUM2_SHIFT);
838a56aa10SArun Parameswaran 
848a56aa10SArun Parameswaran 	return ns;
858a56aa10SArun Parameswaran }
868a56aa10SArun Parameswaran 
dte_write_nco_delta(struct ptp_dte * ptp_dte,s64 delta)878a56aa10SArun Parameswaran static void dte_write_nco_delta(struct ptp_dte *ptp_dte, s64 delta)
888a56aa10SArun Parameswaran {
898a56aa10SArun Parameswaran 	s64 ns;
908a56aa10SArun Parameswaran 
918a56aa10SArun Parameswaran 	ns = dte_read_nco(ptp_dte->regs);
928a56aa10SArun Parameswaran 
938a56aa10SArun Parameswaran 	/* handle wraparound conditions */
948a56aa10SArun Parameswaran 	if ((delta < 0) && (abs(delta) > ns)) {
958a56aa10SArun Parameswaran 		if (ptp_dte->ts_wrap_cnt) {
968a56aa10SArun Parameswaran 			ns += DTE_NCO_MAX_NS + delta;
978a56aa10SArun Parameswaran 			ptp_dte->ts_wrap_cnt--;
988a56aa10SArun Parameswaran 		} else {
998a56aa10SArun Parameswaran 			ns = 0;
1008a56aa10SArun Parameswaran 		}
1018a56aa10SArun Parameswaran 	} else {
1028a56aa10SArun Parameswaran 		ns += delta;
1038a56aa10SArun Parameswaran 		if (ns > DTE_NCO_MAX_NS) {
1048a56aa10SArun Parameswaran 			ptp_dte->ts_wrap_cnt++;
1058a56aa10SArun Parameswaran 			ns -= DTE_NCO_MAX_NS;
1068a56aa10SArun Parameswaran 		}
1078a56aa10SArun Parameswaran 	}
1088a56aa10SArun Parameswaran 
1098a56aa10SArun Parameswaran 	dte_write_nco(ptp_dte->regs, ns);
1108a56aa10SArun Parameswaran 
1118a56aa10SArun Parameswaran 	ptp_dte->ts_ovf_last = (ns >> DTE_NCO_TS_WRAP_LSHIFT) &
1128a56aa10SArun Parameswaran 			DTE_NCO_TS_WRAP_MASK;
1138a56aa10SArun Parameswaran }
1148a56aa10SArun Parameswaran 
dte_read_nco_with_ovf(struct ptp_dte * ptp_dte)1158a56aa10SArun Parameswaran static s64 dte_read_nco_with_ovf(struct ptp_dte *ptp_dte)
1168a56aa10SArun Parameswaran {
1178a56aa10SArun Parameswaran 	u32 ts_ovf;
1188a56aa10SArun Parameswaran 	s64 ns = 0;
1198a56aa10SArun Parameswaran 
1208a56aa10SArun Parameswaran 	ns = dte_read_nco(ptp_dte->regs);
1218a56aa10SArun Parameswaran 
1228a56aa10SArun Parameswaran 	/*Timestamp overflow: 8 LSB bits of sum3, 4 MSB bits of sum2 */
1238a56aa10SArun Parameswaran 	ts_ovf = (ns >> DTE_NCO_TS_WRAP_LSHIFT) & DTE_NCO_TS_WRAP_MASK;
1248a56aa10SArun Parameswaran 
1258a56aa10SArun Parameswaran 	/* Check for wrap around */
1268a56aa10SArun Parameswaran 	if (ts_ovf < ptp_dte->ts_ovf_last)
1278a56aa10SArun Parameswaran 		ptp_dte->ts_wrap_cnt++;
1288a56aa10SArun Parameswaran 
1298a56aa10SArun Parameswaran 	ptp_dte->ts_ovf_last = ts_ovf;
1308a56aa10SArun Parameswaran 
1318a56aa10SArun Parameswaran 	/* adjust for wraparounds */
1328a56aa10SArun Parameswaran 	ns += (s64)(BIT_ULL(DTE_WRAP_AROUND_NSEC_SHIFT) * ptp_dte->ts_wrap_cnt);
1338a56aa10SArun Parameswaran 
1348a56aa10SArun Parameswaran 	return ns;
1358a56aa10SArun Parameswaran }
1368a56aa10SArun Parameswaran 
ptp_dte_adjfine(struct ptp_clock_info * ptp,long scaled_ppm)137*e2bd9c76SJacob Keller static int ptp_dte_adjfine(struct ptp_clock_info *ptp, long scaled_ppm)
1388a56aa10SArun Parameswaran {
139*e2bd9c76SJacob Keller 	s32 ppb = scaled_ppm_to_ppb(scaled_ppm);
1408a56aa10SArun Parameswaran 	u32 nco_incr;
1418a56aa10SArun Parameswaran 	unsigned long flags;
1428a56aa10SArun Parameswaran 	struct ptp_dte *ptp_dte = container_of(ptp, struct ptp_dte, caps);
1438a56aa10SArun Parameswaran 
1448a56aa10SArun Parameswaran 	if (abs(ppb) > ptp_dte->caps.max_adj) {
1458a56aa10SArun Parameswaran 		dev_err(ptp_dte->dev, "ppb adj too big\n");
1468a56aa10SArun Parameswaran 		return -EINVAL;
1478a56aa10SArun Parameswaran 	}
1488a56aa10SArun Parameswaran 
1498a56aa10SArun Parameswaran 	if (ppb < 0)
1508a56aa10SArun Parameswaran 		nco_incr = DTE_NCO_INC_DEFAULT - DTE_PPB_ADJ(ppb);
1518a56aa10SArun Parameswaran 	else
1528a56aa10SArun Parameswaran 		nco_incr = DTE_NCO_INC_DEFAULT + DTE_PPB_ADJ(ppb);
1538a56aa10SArun Parameswaran 
1548a56aa10SArun Parameswaran 	spin_lock_irqsave(&ptp_dte->lock, flags);
1558a56aa10SArun Parameswaran 	writel(nco_incr, ptp_dte->regs + DTE_NCO_INC_REG);
1568a56aa10SArun Parameswaran 	spin_unlock_irqrestore(&ptp_dte->lock, flags);
1578a56aa10SArun Parameswaran 
1588a56aa10SArun Parameswaran 	return 0;
1598a56aa10SArun Parameswaran }
1608a56aa10SArun Parameswaran 
ptp_dte_adjtime(struct ptp_clock_info * ptp,s64 delta)1618a56aa10SArun Parameswaran static int ptp_dte_adjtime(struct ptp_clock_info *ptp, s64 delta)
1628a56aa10SArun Parameswaran {
1638a56aa10SArun Parameswaran 	unsigned long flags;
1648a56aa10SArun Parameswaran 	struct ptp_dte *ptp_dte = container_of(ptp, struct ptp_dte, caps);
1658a56aa10SArun Parameswaran 
1668a56aa10SArun Parameswaran 	spin_lock_irqsave(&ptp_dte->lock, flags);
1678a56aa10SArun Parameswaran 	dte_write_nco_delta(ptp_dte, delta);
1688a56aa10SArun Parameswaran 	spin_unlock_irqrestore(&ptp_dte->lock, flags);
1698a56aa10SArun Parameswaran 
1708a56aa10SArun Parameswaran 	return 0;
1718a56aa10SArun Parameswaran }
1728a56aa10SArun Parameswaran 
ptp_dte_gettime(struct ptp_clock_info * ptp,struct timespec64 * ts)1738a56aa10SArun Parameswaran static int ptp_dte_gettime(struct ptp_clock_info *ptp, struct timespec64 *ts)
1748a56aa10SArun Parameswaran {
1758a56aa10SArun Parameswaran 	unsigned long flags;
1768a56aa10SArun Parameswaran 	struct ptp_dte *ptp_dte = container_of(ptp, struct ptp_dte, caps);
1778a56aa10SArun Parameswaran 
1788a56aa10SArun Parameswaran 	spin_lock_irqsave(&ptp_dte->lock, flags);
1798a56aa10SArun Parameswaran 	*ts = ns_to_timespec64(dte_read_nco_with_ovf(ptp_dte));
1808a56aa10SArun Parameswaran 	spin_unlock_irqrestore(&ptp_dte->lock, flags);
1818a56aa10SArun Parameswaran 
1828a56aa10SArun Parameswaran 	return 0;
1838a56aa10SArun Parameswaran }
1848a56aa10SArun Parameswaran 
ptp_dte_settime(struct ptp_clock_info * ptp,const struct timespec64 * ts)1858a56aa10SArun Parameswaran static int ptp_dte_settime(struct ptp_clock_info *ptp,
1868a56aa10SArun Parameswaran 			     const struct timespec64 *ts)
1878a56aa10SArun Parameswaran {
1888a56aa10SArun Parameswaran 	unsigned long flags;
1898a56aa10SArun Parameswaran 	struct ptp_dte *ptp_dte = container_of(ptp, struct ptp_dte, caps);
1908a56aa10SArun Parameswaran 
1918a56aa10SArun Parameswaran 	spin_lock_irqsave(&ptp_dte->lock, flags);
1928a56aa10SArun Parameswaran 
1938a56aa10SArun Parameswaran 	/* Disable nco increment */
1948a56aa10SArun Parameswaran 	writel(0, ptp_dte->regs + DTE_NCO_INC_REG);
1958a56aa10SArun Parameswaran 
1968a56aa10SArun Parameswaran 	dte_write_nco(ptp_dte->regs, timespec64_to_ns(ts));
1978a56aa10SArun Parameswaran 
1988a56aa10SArun Parameswaran 	/* reset overflow and wrap counter */
1998a56aa10SArun Parameswaran 	ptp_dte->ts_ovf_last = 0;
2008a56aa10SArun Parameswaran 	ptp_dte->ts_wrap_cnt = 0;
2018a56aa10SArun Parameswaran 
2028a56aa10SArun Parameswaran 	/* Enable nco increment */
2038a56aa10SArun Parameswaran 	writel(DTE_NCO_INC_DEFAULT, ptp_dte->regs + DTE_NCO_INC_REG);
2048a56aa10SArun Parameswaran 
2058a56aa10SArun Parameswaran 	spin_unlock_irqrestore(&ptp_dte->lock, flags);
2068a56aa10SArun Parameswaran 
2078a56aa10SArun Parameswaran 	return 0;
2088a56aa10SArun Parameswaran }
2098a56aa10SArun Parameswaran 
ptp_dte_enable(struct ptp_clock_info * ptp,struct ptp_clock_request * rq,int on)2108a56aa10SArun Parameswaran static int ptp_dte_enable(struct ptp_clock_info *ptp,
2118a56aa10SArun Parameswaran 			    struct ptp_clock_request *rq, int on)
2128a56aa10SArun Parameswaran {
2138a56aa10SArun Parameswaran 	return -EOPNOTSUPP;
2148a56aa10SArun Parameswaran }
2158a56aa10SArun Parameswaran 
2167d47e9a2SBhumika Goyal static const struct ptp_clock_info ptp_dte_caps = {
2178a56aa10SArun Parameswaran 	.owner		= THIS_MODULE,
2188a56aa10SArun Parameswaran 	.name		= "DTE PTP timer",
2198a56aa10SArun Parameswaran 	.max_adj	= 50000000,
2208a56aa10SArun Parameswaran 	.n_ext_ts	= 0,
2218a56aa10SArun Parameswaran 	.n_pins		= 0,
2228a56aa10SArun Parameswaran 	.pps		= 0,
223*e2bd9c76SJacob Keller 	.adjfine	= ptp_dte_adjfine,
2248a56aa10SArun Parameswaran 	.adjtime	= ptp_dte_adjtime,
2258a56aa10SArun Parameswaran 	.gettime64	= ptp_dte_gettime,
2268a56aa10SArun Parameswaran 	.settime64	= ptp_dte_settime,
2278a56aa10SArun Parameswaran 	.enable		= ptp_dte_enable,
2288a56aa10SArun Parameswaran };
2298a56aa10SArun Parameswaran 
ptp_dte_probe(struct platform_device * pdev)2308a56aa10SArun Parameswaran static int ptp_dte_probe(struct platform_device *pdev)
2318a56aa10SArun Parameswaran {
2328a56aa10SArun Parameswaran 	struct ptp_dte *ptp_dte;
2338a56aa10SArun Parameswaran 	struct device *dev = &pdev->dev;
2348a56aa10SArun Parameswaran 
2358a56aa10SArun Parameswaran 	ptp_dte = devm_kzalloc(dev, sizeof(struct ptp_dte), GFP_KERNEL);
2368a56aa10SArun Parameswaran 	if (!ptp_dte)
2378a56aa10SArun Parameswaran 		return -ENOMEM;
2388a56aa10SArun Parameswaran 
239f063d58bSYueHaibing 	ptp_dte->regs = devm_platform_ioremap_resource(pdev, 0);
24037f7c66fSDing Xiang 	if (IS_ERR(ptp_dte->regs))
2418a56aa10SArun Parameswaran 		return PTR_ERR(ptp_dte->regs);
2428a56aa10SArun Parameswaran 
2438a56aa10SArun Parameswaran 	spin_lock_init(&ptp_dte->lock);
2448a56aa10SArun Parameswaran 
2458a56aa10SArun Parameswaran 	ptp_dte->dev = dev;
2468a56aa10SArun Parameswaran 	ptp_dte->caps = ptp_dte_caps;
2478a56aa10SArun Parameswaran 	ptp_dte->ptp_clk = ptp_clock_register(&ptp_dte->caps, &pdev->dev);
2488a56aa10SArun Parameswaran 	if (IS_ERR(ptp_dte->ptp_clk)) {
2498a56aa10SArun Parameswaran 		dev_err(dev,
2508a56aa10SArun Parameswaran 			"%s: Failed to register ptp clock\n", __func__);
2518a56aa10SArun Parameswaran 		return PTR_ERR(ptp_dte->ptp_clk);
2528a56aa10SArun Parameswaran 	}
2538a56aa10SArun Parameswaran 
2548a56aa10SArun Parameswaran 	platform_set_drvdata(pdev, ptp_dte);
2558a56aa10SArun Parameswaran 
2568a56aa10SArun Parameswaran 	dev_info(dev, "ptp clk probe done\n");
2578a56aa10SArun Parameswaran 
2588a56aa10SArun Parameswaran 	return 0;
2598a56aa10SArun Parameswaran }
2608a56aa10SArun Parameswaran 
ptp_dte_remove(struct platform_device * pdev)2618a56aa10SArun Parameswaran static int ptp_dte_remove(struct platform_device *pdev)
2628a56aa10SArun Parameswaran {
2638a56aa10SArun Parameswaran 	struct ptp_dte *ptp_dte = platform_get_drvdata(pdev);
2648a56aa10SArun Parameswaran 	u8 i;
2658a56aa10SArun Parameswaran 
2668a56aa10SArun Parameswaran 	ptp_clock_unregister(ptp_dte->ptp_clk);
2678a56aa10SArun Parameswaran 
2688a56aa10SArun Parameswaran 	for (i = 0; i < DTE_NUM_REGS_TO_RESTORE; i++)
2698a56aa10SArun Parameswaran 		writel(0, ptp_dte->regs + (i * sizeof(u32)));
2708a56aa10SArun Parameswaran 
2718a56aa10SArun Parameswaran 	return 0;
2728a56aa10SArun Parameswaran }
2738a56aa10SArun Parameswaran 
2748a56aa10SArun Parameswaran #ifdef CONFIG_PM_SLEEP
ptp_dte_suspend(struct device * dev)2758a56aa10SArun Parameswaran static int ptp_dte_suspend(struct device *dev)
2768a56aa10SArun Parameswaran {
277c0bfdae0SWolfram Sang 	struct ptp_dte *ptp_dte = dev_get_drvdata(dev);
2788a56aa10SArun Parameswaran 	u8 i;
2798a56aa10SArun Parameswaran 
2808a56aa10SArun Parameswaran 	for (i = 0; i < DTE_NUM_REGS_TO_RESTORE; i++) {
2818a56aa10SArun Parameswaran 		ptp_dte->reg_val[i] =
2828a56aa10SArun Parameswaran 			readl(ptp_dte->regs + (i * sizeof(u32)));
2838a56aa10SArun Parameswaran 	}
2848a56aa10SArun Parameswaran 
2858a56aa10SArun Parameswaran 	/* disable the nco */
2868a56aa10SArun Parameswaran 	writel(0, ptp_dte->regs + DTE_NCO_INC_REG);
2878a56aa10SArun Parameswaran 
2888a56aa10SArun Parameswaran 	return 0;
2898a56aa10SArun Parameswaran }
2908a56aa10SArun Parameswaran 
ptp_dte_resume(struct device * dev)2918a56aa10SArun Parameswaran static int ptp_dte_resume(struct device *dev)
2928a56aa10SArun Parameswaran {
293c0bfdae0SWolfram Sang 	struct ptp_dte *ptp_dte = dev_get_drvdata(dev);
2948a56aa10SArun Parameswaran 	u8 i;
2958a56aa10SArun Parameswaran 
2968a56aa10SArun Parameswaran 	for (i = 0; i < DTE_NUM_REGS_TO_RESTORE; i++) {
2978a56aa10SArun Parameswaran 		if ((i * sizeof(u32)) != DTE_NCO_OVERFLOW_REG)
2988a56aa10SArun Parameswaran 			writel(ptp_dte->reg_val[i],
2998a56aa10SArun Parameswaran 				(ptp_dte->regs + (i * sizeof(u32))));
3008a56aa10SArun Parameswaran 		else
3018a56aa10SArun Parameswaran 			writel(((ptp_dte->reg_val[i] &
3028a56aa10SArun Parameswaran 				DTE_NCO_SUM3_MASK) << DTE_NCO_SUM3_WR_SHIFT),
3038a56aa10SArun Parameswaran 				(ptp_dte->regs + (i * sizeof(u32))));
3048a56aa10SArun Parameswaran 	}
3058a56aa10SArun Parameswaran 
3068a56aa10SArun Parameswaran 	return 0;
3078a56aa10SArun Parameswaran }
3088a56aa10SArun Parameswaran 
3098a56aa10SArun Parameswaran static const struct dev_pm_ops ptp_dte_pm_ops = {
3108a56aa10SArun Parameswaran 	.suspend = ptp_dte_suspend,
3118a56aa10SArun Parameswaran 	.resume = ptp_dte_resume
3128a56aa10SArun Parameswaran };
3138a56aa10SArun Parameswaran 
3148a56aa10SArun Parameswaran #define PTP_DTE_PM_OPS	(&ptp_dte_pm_ops)
3158a56aa10SArun Parameswaran #else
3168a56aa10SArun Parameswaran #define PTP_DTE_PM_OPS	NULL
3178a56aa10SArun Parameswaran #endif
3188a56aa10SArun Parameswaran 
3198a56aa10SArun Parameswaran static const struct of_device_id ptp_dte_of_match[] = {
3208a56aa10SArun Parameswaran 	{ .compatible = "brcm,ptp-dte", },
3218a56aa10SArun Parameswaran 	{},
3228a56aa10SArun Parameswaran };
3238a56aa10SArun Parameswaran MODULE_DEVICE_TABLE(of, ptp_dte_of_match);
3248a56aa10SArun Parameswaran 
3258a56aa10SArun Parameswaran static struct platform_driver ptp_dte_driver = {
3268a56aa10SArun Parameswaran 	.driver = {
3278a56aa10SArun Parameswaran 		.name = "ptp-dte",
3288a56aa10SArun Parameswaran 		.pm = PTP_DTE_PM_OPS,
3298a56aa10SArun Parameswaran 		.of_match_table = ptp_dte_of_match,
3308a56aa10SArun Parameswaran 	},
3318a56aa10SArun Parameswaran 	.probe    = ptp_dte_probe,
3328a56aa10SArun Parameswaran 	.remove   = ptp_dte_remove,
3338a56aa10SArun Parameswaran };
3348a56aa10SArun Parameswaran module_platform_driver(ptp_dte_driver);
3358a56aa10SArun Parameswaran 
3368a56aa10SArun Parameswaran MODULE_AUTHOR("Broadcom");
3378a56aa10SArun Parameswaran MODULE_DESCRIPTION("Broadcom DTE PTP Clock driver");
3388a56aa10SArun Parameswaran MODULE_LICENSE("GPL v2");
339