xref: /OK3568_Linux_fs/kernel/drivers/net/ethernet/freescale/dpaa2/dpaa2-ptp.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun  * Copyright 2013-2016 Freescale Semiconductor Inc.
4*4882a593Smuzhiyun  * Copyright 2016-2018 NXP
5*4882a593Smuzhiyun  * Copyright 2020 NXP
6*4882a593Smuzhiyun  */
7*4882a593Smuzhiyun 
8*4882a593Smuzhiyun #include <linux/module.h>
9*4882a593Smuzhiyun #include <linux/of.h>
10*4882a593Smuzhiyun #include <linux/of_address.h>
11*4882a593Smuzhiyun #include <linux/msi.h>
12*4882a593Smuzhiyun #include <linux/fsl/mc.h>
13*4882a593Smuzhiyun 
14*4882a593Smuzhiyun #include "dpaa2-ptp.h"
15*4882a593Smuzhiyun 
dpaa2_ptp_enable(struct ptp_clock_info * ptp,struct ptp_clock_request * rq,int on)16*4882a593Smuzhiyun static int dpaa2_ptp_enable(struct ptp_clock_info *ptp,
17*4882a593Smuzhiyun 			    struct ptp_clock_request *rq, int on)
18*4882a593Smuzhiyun {
19*4882a593Smuzhiyun 	struct ptp_qoriq *ptp_qoriq = container_of(ptp, struct ptp_qoriq, caps);
20*4882a593Smuzhiyun 	struct fsl_mc_device *mc_dev;
21*4882a593Smuzhiyun 	struct device *dev;
22*4882a593Smuzhiyun 	u32 mask = 0;
23*4882a593Smuzhiyun 	u32 bit;
24*4882a593Smuzhiyun 	int err;
25*4882a593Smuzhiyun 
26*4882a593Smuzhiyun 	dev = ptp_qoriq->dev;
27*4882a593Smuzhiyun 	mc_dev = to_fsl_mc_device(dev);
28*4882a593Smuzhiyun 
29*4882a593Smuzhiyun 	switch (rq->type) {
30*4882a593Smuzhiyun 	case PTP_CLK_REQ_EXTTS:
31*4882a593Smuzhiyun 		switch (rq->extts.index) {
32*4882a593Smuzhiyun 		case 0:
33*4882a593Smuzhiyun 			bit = DPRTC_EVENT_ETS1;
34*4882a593Smuzhiyun 			break;
35*4882a593Smuzhiyun 		case 1:
36*4882a593Smuzhiyun 			bit = DPRTC_EVENT_ETS2;
37*4882a593Smuzhiyun 			break;
38*4882a593Smuzhiyun 		default:
39*4882a593Smuzhiyun 			return -EINVAL;
40*4882a593Smuzhiyun 		}
41*4882a593Smuzhiyun 		if (on)
42*4882a593Smuzhiyun 			extts_clean_up(ptp_qoriq, rq->extts.index, false);
43*4882a593Smuzhiyun 		break;
44*4882a593Smuzhiyun 	case PTP_CLK_REQ_PPS:
45*4882a593Smuzhiyun 		bit = DPRTC_EVENT_PPS;
46*4882a593Smuzhiyun 		break;
47*4882a593Smuzhiyun 	default:
48*4882a593Smuzhiyun 		return -EOPNOTSUPP;
49*4882a593Smuzhiyun 	}
50*4882a593Smuzhiyun 
51*4882a593Smuzhiyun 	err = dprtc_get_irq_mask(mc_dev->mc_io, 0, mc_dev->mc_handle,
52*4882a593Smuzhiyun 				 DPRTC_IRQ_INDEX, &mask);
53*4882a593Smuzhiyun 	if (err < 0) {
54*4882a593Smuzhiyun 		dev_err(dev, "dprtc_get_irq_mask(): %d\n", err);
55*4882a593Smuzhiyun 		return err;
56*4882a593Smuzhiyun 	}
57*4882a593Smuzhiyun 
58*4882a593Smuzhiyun 	if (on)
59*4882a593Smuzhiyun 		mask |= bit;
60*4882a593Smuzhiyun 	else
61*4882a593Smuzhiyun 		mask &= ~bit;
62*4882a593Smuzhiyun 
63*4882a593Smuzhiyun 	err = dprtc_set_irq_mask(mc_dev->mc_io, 0, mc_dev->mc_handle,
64*4882a593Smuzhiyun 				 DPRTC_IRQ_INDEX, mask);
65*4882a593Smuzhiyun 	if (err < 0) {
66*4882a593Smuzhiyun 		dev_err(dev, "dprtc_set_irq_mask(): %d\n", err);
67*4882a593Smuzhiyun 		return err;
68*4882a593Smuzhiyun 	}
69*4882a593Smuzhiyun 
70*4882a593Smuzhiyun 	return 0;
71*4882a593Smuzhiyun }
72*4882a593Smuzhiyun 
73*4882a593Smuzhiyun static const struct ptp_clock_info dpaa2_ptp_caps = {
74*4882a593Smuzhiyun 	.owner		= THIS_MODULE,
75*4882a593Smuzhiyun 	.name		= "DPAA2 PTP Clock",
76*4882a593Smuzhiyun 	.max_adj	= 512000,
77*4882a593Smuzhiyun 	.n_alarm	= 2,
78*4882a593Smuzhiyun 	.n_ext_ts	= 2,
79*4882a593Smuzhiyun 	.n_per_out	= 3,
80*4882a593Smuzhiyun 	.n_pins		= 0,
81*4882a593Smuzhiyun 	.pps		= 1,
82*4882a593Smuzhiyun 	.adjfine	= ptp_qoriq_adjfine,
83*4882a593Smuzhiyun 	.adjtime	= ptp_qoriq_adjtime,
84*4882a593Smuzhiyun 	.gettime64	= ptp_qoriq_gettime,
85*4882a593Smuzhiyun 	.settime64	= ptp_qoriq_settime,
86*4882a593Smuzhiyun 	.enable		= dpaa2_ptp_enable,
87*4882a593Smuzhiyun };
88*4882a593Smuzhiyun 
dpaa2_ptp_irq_handler_thread(int irq,void * priv)89*4882a593Smuzhiyun static irqreturn_t dpaa2_ptp_irq_handler_thread(int irq, void *priv)
90*4882a593Smuzhiyun {
91*4882a593Smuzhiyun 	struct ptp_qoriq *ptp_qoriq = priv;
92*4882a593Smuzhiyun 	struct ptp_clock_event event;
93*4882a593Smuzhiyun 	struct fsl_mc_device *mc_dev;
94*4882a593Smuzhiyun 	struct device *dev;
95*4882a593Smuzhiyun 	u32 status = 0;
96*4882a593Smuzhiyun 	int err;
97*4882a593Smuzhiyun 
98*4882a593Smuzhiyun 	dev = ptp_qoriq->dev;
99*4882a593Smuzhiyun 	mc_dev = to_fsl_mc_device(dev);
100*4882a593Smuzhiyun 
101*4882a593Smuzhiyun 	err = dprtc_get_irq_status(mc_dev->mc_io, 0, mc_dev->mc_handle,
102*4882a593Smuzhiyun 				   DPRTC_IRQ_INDEX, &status);
103*4882a593Smuzhiyun 	if (unlikely(err)) {
104*4882a593Smuzhiyun 		dev_err(dev, "dprtc_get_irq_status err %d\n", err);
105*4882a593Smuzhiyun 		return IRQ_NONE;
106*4882a593Smuzhiyun 	}
107*4882a593Smuzhiyun 
108*4882a593Smuzhiyun 	if (status & DPRTC_EVENT_PPS) {
109*4882a593Smuzhiyun 		event.type = PTP_CLOCK_PPS;
110*4882a593Smuzhiyun 		ptp_clock_event(ptp_qoriq->clock, &event);
111*4882a593Smuzhiyun 	}
112*4882a593Smuzhiyun 
113*4882a593Smuzhiyun 	if (status & DPRTC_EVENT_ETS1)
114*4882a593Smuzhiyun 		extts_clean_up(ptp_qoriq, 0, true);
115*4882a593Smuzhiyun 
116*4882a593Smuzhiyun 	if (status & DPRTC_EVENT_ETS2)
117*4882a593Smuzhiyun 		extts_clean_up(ptp_qoriq, 1, true);
118*4882a593Smuzhiyun 
119*4882a593Smuzhiyun 	err = dprtc_clear_irq_status(mc_dev->mc_io, 0, mc_dev->mc_handle,
120*4882a593Smuzhiyun 				     DPRTC_IRQ_INDEX, status);
121*4882a593Smuzhiyun 	if (unlikely(err)) {
122*4882a593Smuzhiyun 		dev_err(dev, "dprtc_clear_irq_status err %d\n", err);
123*4882a593Smuzhiyun 		return IRQ_NONE;
124*4882a593Smuzhiyun 	}
125*4882a593Smuzhiyun 
126*4882a593Smuzhiyun 	return IRQ_HANDLED;
127*4882a593Smuzhiyun }
128*4882a593Smuzhiyun 
dpaa2_ptp_probe(struct fsl_mc_device * mc_dev)129*4882a593Smuzhiyun static int dpaa2_ptp_probe(struct fsl_mc_device *mc_dev)
130*4882a593Smuzhiyun {
131*4882a593Smuzhiyun 	struct device *dev = &mc_dev->dev;
132*4882a593Smuzhiyun 	struct fsl_mc_device_irq *irq;
133*4882a593Smuzhiyun 	struct ptp_qoriq *ptp_qoriq;
134*4882a593Smuzhiyun 	struct device_node *node;
135*4882a593Smuzhiyun 	void __iomem *base;
136*4882a593Smuzhiyun 	int err;
137*4882a593Smuzhiyun 
138*4882a593Smuzhiyun 	ptp_qoriq = devm_kzalloc(dev, sizeof(*ptp_qoriq), GFP_KERNEL);
139*4882a593Smuzhiyun 	if (!ptp_qoriq)
140*4882a593Smuzhiyun 		return -ENOMEM;
141*4882a593Smuzhiyun 
142*4882a593Smuzhiyun 	err = fsl_mc_portal_allocate(mc_dev, 0, &mc_dev->mc_io);
143*4882a593Smuzhiyun 	if (err) {
144*4882a593Smuzhiyun 		if (err == -ENXIO)
145*4882a593Smuzhiyun 			err = -EPROBE_DEFER;
146*4882a593Smuzhiyun 		else
147*4882a593Smuzhiyun 			dev_err(dev, "fsl_mc_portal_allocate err %d\n", err);
148*4882a593Smuzhiyun 		goto err_exit;
149*4882a593Smuzhiyun 	}
150*4882a593Smuzhiyun 
151*4882a593Smuzhiyun 	err = dprtc_open(mc_dev->mc_io, 0, mc_dev->obj_desc.id,
152*4882a593Smuzhiyun 			 &mc_dev->mc_handle);
153*4882a593Smuzhiyun 	if (err) {
154*4882a593Smuzhiyun 		dev_err(dev, "dprtc_open err %d\n", err);
155*4882a593Smuzhiyun 		goto err_free_mcp;
156*4882a593Smuzhiyun 	}
157*4882a593Smuzhiyun 
158*4882a593Smuzhiyun 	ptp_qoriq->dev = dev;
159*4882a593Smuzhiyun 
160*4882a593Smuzhiyun 	node = of_find_compatible_node(NULL, NULL, "fsl,dpaa2-ptp");
161*4882a593Smuzhiyun 	if (!node) {
162*4882a593Smuzhiyun 		err = -ENODEV;
163*4882a593Smuzhiyun 		goto err_close;
164*4882a593Smuzhiyun 	}
165*4882a593Smuzhiyun 
166*4882a593Smuzhiyun 	dev->of_node = node;
167*4882a593Smuzhiyun 
168*4882a593Smuzhiyun 	base = of_iomap(node, 0);
169*4882a593Smuzhiyun 	if (!base) {
170*4882a593Smuzhiyun 		err = -ENOMEM;
171*4882a593Smuzhiyun 		goto err_put;
172*4882a593Smuzhiyun 	}
173*4882a593Smuzhiyun 
174*4882a593Smuzhiyun 	err = fsl_mc_allocate_irqs(mc_dev);
175*4882a593Smuzhiyun 	if (err) {
176*4882a593Smuzhiyun 		dev_err(dev, "MC irqs allocation failed\n");
177*4882a593Smuzhiyun 		goto err_unmap;
178*4882a593Smuzhiyun 	}
179*4882a593Smuzhiyun 
180*4882a593Smuzhiyun 	irq = mc_dev->irqs[0];
181*4882a593Smuzhiyun 	ptp_qoriq->irq = irq->msi_desc->irq;
182*4882a593Smuzhiyun 
183*4882a593Smuzhiyun 	err = request_threaded_irq(ptp_qoriq->irq, NULL,
184*4882a593Smuzhiyun 				   dpaa2_ptp_irq_handler_thread,
185*4882a593Smuzhiyun 				   IRQF_NO_SUSPEND | IRQF_ONESHOT,
186*4882a593Smuzhiyun 				   dev_name(dev), ptp_qoriq);
187*4882a593Smuzhiyun 	if (err < 0) {
188*4882a593Smuzhiyun 		dev_err(dev, "devm_request_threaded_irq(): %d\n", err);
189*4882a593Smuzhiyun 		goto err_free_mc_irq;
190*4882a593Smuzhiyun 	}
191*4882a593Smuzhiyun 
192*4882a593Smuzhiyun 	err = dprtc_set_irq_enable(mc_dev->mc_io, 0, mc_dev->mc_handle,
193*4882a593Smuzhiyun 				   DPRTC_IRQ_INDEX, 1);
194*4882a593Smuzhiyun 	if (err < 0) {
195*4882a593Smuzhiyun 		dev_err(dev, "dprtc_set_irq_enable(): %d\n", err);
196*4882a593Smuzhiyun 		goto err_free_threaded_irq;
197*4882a593Smuzhiyun 	}
198*4882a593Smuzhiyun 
199*4882a593Smuzhiyun 	err = ptp_qoriq_init(ptp_qoriq, base, &dpaa2_ptp_caps);
200*4882a593Smuzhiyun 	if (err)
201*4882a593Smuzhiyun 		goto err_free_threaded_irq;
202*4882a593Smuzhiyun 
203*4882a593Smuzhiyun 	dpaa2_phc_index = ptp_qoriq->phc_index;
204*4882a593Smuzhiyun 	dpaa2_ptp = ptp_qoriq;
205*4882a593Smuzhiyun 	dev_set_drvdata(dev, ptp_qoriq);
206*4882a593Smuzhiyun 
207*4882a593Smuzhiyun 	return 0;
208*4882a593Smuzhiyun 
209*4882a593Smuzhiyun err_free_threaded_irq:
210*4882a593Smuzhiyun 	free_irq(ptp_qoriq->irq, ptp_qoriq);
211*4882a593Smuzhiyun err_free_mc_irq:
212*4882a593Smuzhiyun 	fsl_mc_free_irqs(mc_dev);
213*4882a593Smuzhiyun err_unmap:
214*4882a593Smuzhiyun 	iounmap(base);
215*4882a593Smuzhiyun err_put:
216*4882a593Smuzhiyun 	of_node_put(node);
217*4882a593Smuzhiyun err_close:
218*4882a593Smuzhiyun 	dprtc_close(mc_dev->mc_io, 0, mc_dev->mc_handle);
219*4882a593Smuzhiyun err_free_mcp:
220*4882a593Smuzhiyun 	fsl_mc_portal_free(mc_dev->mc_io);
221*4882a593Smuzhiyun err_exit:
222*4882a593Smuzhiyun 	return err;
223*4882a593Smuzhiyun }
224*4882a593Smuzhiyun 
dpaa2_ptp_remove(struct fsl_mc_device * mc_dev)225*4882a593Smuzhiyun static int dpaa2_ptp_remove(struct fsl_mc_device *mc_dev)
226*4882a593Smuzhiyun {
227*4882a593Smuzhiyun 	struct device *dev = &mc_dev->dev;
228*4882a593Smuzhiyun 	struct ptp_qoriq *ptp_qoriq;
229*4882a593Smuzhiyun 
230*4882a593Smuzhiyun 	ptp_qoriq = dev_get_drvdata(dev);
231*4882a593Smuzhiyun 
232*4882a593Smuzhiyun 	dpaa2_phc_index = -1;
233*4882a593Smuzhiyun 	ptp_qoriq_free(ptp_qoriq);
234*4882a593Smuzhiyun 
235*4882a593Smuzhiyun 	fsl_mc_free_irqs(mc_dev);
236*4882a593Smuzhiyun 	dprtc_close(mc_dev->mc_io, 0, mc_dev->mc_handle);
237*4882a593Smuzhiyun 	fsl_mc_portal_free(mc_dev->mc_io);
238*4882a593Smuzhiyun 
239*4882a593Smuzhiyun 	return 0;
240*4882a593Smuzhiyun }
241*4882a593Smuzhiyun 
242*4882a593Smuzhiyun static const struct fsl_mc_device_id dpaa2_ptp_match_id_table[] = {
243*4882a593Smuzhiyun 	{
244*4882a593Smuzhiyun 		.vendor = FSL_MC_VENDOR_FREESCALE,
245*4882a593Smuzhiyun 		.obj_type = "dprtc",
246*4882a593Smuzhiyun 	},
247*4882a593Smuzhiyun 	{}
248*4882a593Smuzhiyun };
249*4882a593Smuzhiyun MODULE_DEVICE_TABLE(fslmc, dpaa2_ptp_match_id_table);
250*4882a593Smuzhiyun 
251*4882a593Smuzhiyun static struct fsl_mc_driver dpaa2_ptp_drv = {
252*4882a593Smuzhiyun 	.driver = {
253*4882a593Smuzhiyun 		.name = KBUILD_MODNAME,
254*4882a593Smuzhiyun 		.owner = THIS_MODULE,
255*4882a593Smuzhiyun 	},
256*4882a593Smuzhiyun 	.probe = dpaa2_ptp_probe,
257*4882a593Smuzhiyun 	.remove = dpaa2_ptp_remove,
258*4882a593Smuzhiyun 	.match_id_table = dpaa2_ptp_match_id_table,
259*4882a593Smuzhiyun };
260*4882a593Smuzhiyun 
261*4882a593Smuzhiyun module_fsl_mc_driver(dpaa2_ptp_drv);
262*4882a593Smuzhiyun 
263*4882a593Smuzhiyun MODULE_LICENSE("GPL v2");
264*4882a593Smuzhiyun MODULE_DESCRIPTION("DPAA2 PTP Clock Driver");
265