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