12b49d128SYangbo Lu // SPDX-License-Identifier: (GPL-2.0 OR MIT)
22b49d128SYangbo Lu /* Microsemi Ocelot PTP clock driver
32b49d128SYangbo Lu *
42b49d128SYangbo Lu * Copyright (c) 2017 Microsemi Corporation
52b49d128SYangbo Lu * Copyright 2020 NXP
62b49d128SYangbo Lu */
7a460513eSAndy Shevchenko #include <linux/time64.h>
8a460513eSAndy Shevchenko
9d50e41bfSVladimir Oltean #include <linux/dsa/ocelot.h>
10d50e41bfSVladimir Oltean #include <linux/ptp_classify.h>
112b49d128SYangbo Lu #include <soc/mscc/ocelot_ptp.h>
122b49d128SYangbo Lu #include <soc/mscc/ocelot_sys.h>
13d50e41bfSVladimir Oltean #include <soc/mscc/ocelot_vcap.h>
142b49d128SYangbo Lu #include <soc/mscc/ocelot.h>
15d50e41bfSVladimir Oltean #include "ocelot.h"
162b49d128SYangbo Lu
173b327726SVladimir Oltean #define OCELOT_PTP_TX_TSTAMP_TIMEOUT (5 * HZ)
183b327726SVladimir Oltean
ocelot_ptp_gettime64(struct ptp_clock_info * ptp,struct timespec64 * ts)192b49d128SYangbo Lu int ocelot_ptp_gettime64(struct ptp_clock_info *ptp, struct timespec64 *ts)
202b49d128SYangbo Lu {
212b49d128SYangbo Lu struct ocelot *ocelot = container_of(ptp, struct ocelot, ptp_info);
222b49d128SYangbo Lu unsigned long flags;
232b49d128SYangbo Lu time64_t s;
242b49d128SYangbo Lu u32 val;
252b49d128SYangbo Lu s64 ns;
262b49d128SYangbo Lu
272b49d128SYangbo Lu spin_lock_irqsave(&ocelot->ptp_clock_lock, flags);
282b49d128SYangbo Lu
292b49d128SYangbo Lu val = ocelot_read_rix(ocelot, PTP_PIN_CFG, TOD_ACC_PIN);
302b49d128SYangbo Lu val &= ~(PTP_PIN_CFG_SYNC | PTP_PIN_CFG_ACTION_MASK | PTP_PIN_CFG_DOM);
312b49d128SYangbo Lu val |= PTP_PIN_CFG_ACTION(PTP_PIN_ACTION_SAVE);
322b49d128SYangbo Lu ocelot_write_rix(ocelot, val, PTP_PIN_CFG, TOD_ACC_PIN);
332b49d128SYangbo Lu
342b49d128SYangbo Lu s = ocelot_read_rix(ocelot, PTP_PIN_TOD_SEC_MSB, TOD_ACC_PIN) & 0xffff;
352b49d128SYangbo Lu s <<= 32;
362b49d128SYangbo Lu s += ocelot_read_rix(ocelot, PTP_PIN_TOD_SEC_LSB, TOD_ACC_PIN);
372b49d128SYangbo Lu ns = ocelot_read_rix(ocelot, PTP_PIN_TOD_NSEC, TOD_ACC_PIN);
382b49d128SYangbo Lu
392b49d128SYangbo Lu spin_unlock_irqrestore(&ocelot->ptp_clock_lock, flags);
402b49d128SYangbo Lu
412b49d128SYangbo Lu /* Deal with negative values */
422b49d128SYangbo Lu if (ns >= 0x3ffffff0 && ns <= 0x3fffffff) {
432b49d128SYangbo Lu s--;
442b49d128SYangbo Lu ns &= 0xf;
452b49d128SYangbo Lu ns += 999999984;
462b49d128SYangbo Lu }
472b49d128SYangbo Lu
482b49d128SYangbo Lu set_normalized_timespec64(ts, s, ns);
492b49d128SYangbo Lu return 0;
502b49d128SYangbo Lu }
512b49d128SYangbo Lu EXPORT_SYMBOL(ocelot_ptp_gettime64);
522b49d128SYangbo Lu
ocelot_ptp_settime64(struct ptp_clock_info * ptp,const struct timespec64 * ts)532b49d128SYangbo Lu int ocelot_ptp_settime64(struct ptp_clock_info *ptp,
542b49d128SYangbo Lu const struct timespec64 *ts)
552b49d128SYangbo Lu {
562b49d128SYangbo Lu struct ocelot *ocelot = container_of(ptp, struct ocelot, ptp_info);
572b49d128SYangbo Lu unsigned long flags;
582b49d128SYangbo Lu u32 val;
592b49d128SYangbo Lu
602b49d128SYangbo Lu spin_lock_irqsave(&ocelot->ptp_clock_lock, flags);
612b49d128SYangbo Lu
622b49d128SYangbo Lu val = ocelot_read_rix(ocelot, PTP_PIN_CFG, TOD_ACC_PIN);
632b49d128SYangbo Lu val &= ~(PTP_PIN_CFG_SYNC | PTP_PIN_CFG_ACTION_MASK | PTP_PIN_CFG_DOM);
642b49d128SYangbo Lu val |= PTP_PIN_CFG_ACTION(PTP_PIN_ACTION_IDLE);
652b49d128SYangbo Lu
662b49d128SYangbo Lu ocelot_write_rix(ocelot, val, PTP_PIN_CFG, TOD_ACC_PIN);
672b49d128SYangbo Lu
682b49d128SYangbo Lu ocelot_write_rix(ocelot, lower_32_bits(ts->tv_sec), PTP_PIN_TOD_SEC_LSB,
692b49d128SYangbo Lu TOD_ACC_PIN);
702b49d128SYangbo Lu ocelot_write_rix(ocelot, upper_32_bits(ts->tv_sec), PTP_PIN_TOD_SEC_MSB,
712b49d128SYangbo Lu TOD_ACC_PIN);
722b49d128SYangbo Lu ocelot_write_rix(ocelot, ts->tv_nsec, PTP_PIN_TOD_NSEC, TOD_ACC_PIN);
732b49d128SYangbo Lu
742b49d128SYangbo Lu val = ocelot_read_rix(ocelot, PTP_PIN_CFG, TOD_ACC_PIN);
752b49d128SYangbo Lu val &= ~(PTP_PIN_CFG_SYNC | PTP_PIN_CFG_ACTION_MASK | PTP_PIN_CFG_DOM);
762b49d128SYangbo Lu val |= PTP_PIN_CFG_ACTION(PTP_PIN_ACTION_LOAD);
772b49d128SYangbo Lu
782b49d128SYangbo Lu ocelot_write_rix(ocelot, val, PTP_PIN_CFG, TOD_ACC_PIN);
792b49d128SYangbo Lu
802b49d128SYangbo Lu spin_unlock_irqrestore(&ocelot->ptp_clock_lock, flags);
818670dc33SXiaoliang Yang
828670dc33SXiaoliang Yang if (ocelot->ops->tas_clock_adjust)
838670dc33SXiaoliang Yang ocelot->ops->tas_clock_adjust(ocelot);
848670dc33SXiaoliang Yang
852b49d128SYangbo Lu return 0;
862b49d128SYangbo Lu }
872b49d128SYangbo Lu EXPORT_SYMBOL(ocelot_ptp_settime64);
882b49d128SYangbo Lu
ocelot_ptp_adjtime(struct ptp_clock_info * ptp,s64 delta)892b49d128SYangbo Lu int ocelot_ptp_adjtime(struct ptp_clock_info *ptp, s64 delta)
902b49d128SYangbo Lu {
912b49d128SYangbo Lu if (delta > -(NSEC_PER_SEC / 2) && delta < (NSEC_PER_SEC / 2)) {
922b49d128SYangbo Lu struct ocelot *ocelot = container_of(ptp, struct ocelot,
932b49d128SYangbo Lu ptp_info);
942b49d128SYangbo Lu unsigned long flags;
952b49d128SYangbo Lu u32 val;
962b49d128SYangbo Lu
972b49d128SYangbo Lu spin_lock_irqsave(&ocelot->ptp_clock_lock, flags);
982b49d128SYangbo Lu
992b49d128SYangbo Lu val = ocelot_read_rix(ocelot, PTP_PIN_CFG, TOD_ACC_PIN);
1002b49d128SYangbo Lu val &= ~(PTP_PIN_CFG_SYNC | PTP_PIN_CFG_ACTION_MASK |
1012b49d128SYangbo Lu PTP_PIN_CFG_DOM);
1022b49d128SYangbo Lu val |= PTP_PIN_CFG_ACTION(PTP_PIN_ACTION_IDLE);
1032b49d128SYangbo Lu
1042b49d128SYangbo Lu ocelot_write_rix(ocelot, val, PTP_PIN_CFG, TOD_ACC_PIN);
1052b49d128SYangbo Lu
1062b49d128SYangbo Lu ocelot_write_rix(ocelot, 0, PTP_PIN_TOD_SEC_LSB, TOD_ACC_PIN);
1072b49d128SYangbo Lu ocelot_write_rix(ocelot, 0, PTP_PIN_TOD_SEC_MSB, TOD_ACC_PIN);
1082b49d128SYangbo Lu ocelot_write_rix(ocelot, delta, PTP_PIN_TOD_NSEC, TOD_ACC_PIN);
1092b49d128SYangbo Lu
1102b49d128SYangbo Lu val = ocelot_read_rix(ocelot, PTP_PIN_CFG, TOD_ACC_PIN);
1112b49d128SYangbo Lu val &= ~(PTP_PIN_CFG_SYNC | PTP_PIN_CFG_ACTION_MASK |
1122b49d128SYangbo Lu PTP_PIN_CFG_DOM);
1132b49d128SYangbo Lu val |= PTP_PIN_CFG_ACTION(PTP_PIN_ACTION_DELTA);
1142b49d128SYangbo Lu
1152b49d128SYangbo Lu ocelot_write_rix(ocelot, val, PTP_PIN_CFG, TOD_ACC_PIN);
1162b49d128SYangbo Lu
1172b49d128SYangbo Lu spin_unlock_irqrestore(&ocelot->ptp_clock_lock, flags);
1188670dc33SXiaoliang Yang
1198670dc33SXiaoliang Yang if (ocelot->ops->tas_clock_adjust)
1208670dc33SXiaoliang Yang ocelot->ops->tas_clock_adjust(ocelot);
1212b49d128SYangbo Lu } else {
1222b49d128SYangbo Lu /* Fall back using ocelot_ptp_settime64 which is not exact. */
1232b49d128SYangbo Lu struct timespec64 ts;
1242b49d128SYangbo Lu u64 now;
1252b49d128SYangbo Lu
1262b49d128SYangbo Lu ocelot_ptp_gettime64(ptp, &ts);
1272b49d128SYangbo Lu
1282b49d128SYangbo Lu now = ktime_to_ns(timespec64_to_ktime(ts));
1292b49d128SYangbo Lu ts = ns_to_timespec64(now + delta);
1302b49d128SYangbo Lu
1312b49d128SYangbo Lu ocelot_ptp_settime64(ptp, &ts);
1322b49d128SYangbo Lu }
1338670dc33SXiaoliang Yang
1342b49d128SYangbo Lu return 0;
1352b49d128SYangbo Lu }
1362b49d128SYangbo Lu EXPORT_SYMBOL(ocelot_ptp_adjtime);
1372b49d128SYangbo Lu
ocelot_ptp_adjfine(struct ptp_clock_info * ptp,long scaled_ppm)1382b49d128SYangbo Lu int ocelot_ptp_adjfine(struct ptp_clock_info *ptp, long scaled_ppm)
1392b49d128SYangbo Lu {
1402b49d128SYangbo Lu struct ocelot *ocelot = container_of(ptp, struct ocelot, ptp_info);
1412b49d128SYangbo Lu u32 unit = 0, direction = 0;
1422b49d128SYangbo Lu unsigned long flags;
1432b49d128SYangbo Lu u64 adj = 0;
1442b49d128SYangbo Lu
1452b49d128SYangbo Lu spin_lock_irqsave(&ocelot->ptp_clock_lock, flags);
1462b49d128SYangbo Lu
1472b49d128SYangbo Lu if (!scaled_ppm)
1482b49d128SYangbo Lu goto disable_adj;
1492b49d128SYangbo Lu
1502b49d128SYangbo Lu if (scaled_ppm < 0) {
1512b49d128SYangbo Lu direction = PTP_CFG_CLK_ADJ_CFG_DIR;
1522b49d128SYangbo Lu scaled_ppm = -scaled_ppm;
1532b49d128SYangbo Lu }
1542b49d128SYangbo Lu
1552b49d128SYangbo Lu adj = PSEC_PER_SEC << 16;
1562b49d128SYangbo Lu do_div(adj, scaled_ppm);
1572b49d128SYangbo Lu do_div(adj, 1000);
1582b49d128SYangbo Lu
1592b49d128SYangbo Lu /* If the adjustment value is too large, use ns instead */
1602b49d128SYangbo Lu if (adj >= (1L << 30)) {
1612b49d128SYangbo Lu unit = PTP_CFG_CLK_ADJ_FREQ_NS;
1622b49d128SYangbo Lu do_div(adj, 1000);
1632b49d128SYangbo Lu }
1642b49d128SYangbo Lu
1652b49d128SYangbo Lu /* Still too big */
1662b49d128SYangbo Lu if (adj >= (1L << 30))
1672b49d128SYangbo Lu goto disable_adj;
1682b49d128SYangbo Lu
1692b49d128SYangbo Lu ocelot_write(ocelot, unit | adj, PTP_CLK_CFG_ADJ_FREQ);
1702b49d128SYangbo Lu ocelot_write(ocelot, PTP_CFG_CLK_ADJ_CFG_ENA | direction,
1712b49d128SYangbo Lu PTP_CLK_CFG_ADJ_CFG);
1722b49d128SYangbo Lu
1732b49d128SYangbo Lu spin_unlock_irqrestore(&ocelot->ptp_clock_lock, flags);
1742b49d128SYangbo Lu return 0;
1752b49d128SYangbo Lu
1762b49d128SYangbo Lu disable_adj:
1772b49d128SYangbo Lu ocelot_write(ocelot, 0, PTP_CLK_CFG_ADJ_CFG);
1782b49d128SYangbo Lu
1792b49d128SYangbo Lu spin_unlock_irqrestore(&ocelot->ptp_clock_lock, flags);
1802b49d128SYangbo Lu return 0;
1812b49d128SYangbo Lu }
1822b49d128SYangbo Lu EXPORT_SYMBOL(ocelot_ptp_adjfine);
1832b49d128SYangbo Lu
ocelot_ptp_verify(struct ptp_clock_info * ptp,unsigned int pin,enum ptp_pin_function func,unsigned int chan)184cc2d87bbSYangbo Lu int ocelot_ptp_verify(struct ptp_clock_info *ptp, unsigned int pin,
185cc2d87bbSYangbo Lu enum ptp_pin_function func, unsigned int chan)
186cc2d87bbSYangbo Lu {
187cc2d87bbSYangbo Lu switch (func) {
188cc2d87bbSYangbo Lu case PTP_PF_NONE:
189cc2d87bbSYangbo Lu case PTP_PF_PEROUT:
190cc2d87bbSYangbo Lu break;
191cc2d87bbSYangbo Lu case PTP_PF_EXTTS:
192cc2d87bbSYangbo Lu case PTP_PF_PHYSYNC:
193cc2d87bbSYangbo Lu return -1;
194cc2d87bbSYangbo Lu }
195cc2d87bbSYangbo Lu return 0;
196cc2d87bbSYangbo Lu }
197cc2d87bbSYangbo Lu EXPORT_SYMBOL(ocelot_ptp_verify);
198cc2d87bbSYangbo Lu
ocelot_ptp_enable(struct ptp_clock_info * ptp,struct ptp_clock_request * rq,int on)199cc2d87bbSYangbo Lu int ocelot_ptp_enable(struct ptp_clock_info *ptp,
200cc2d87bbSYangbo Lu struct ptp_clock_request *rq, int on)
201cc2d87bbSYangbo Lu {
202cc2d87bbSYangbo Lu struct ocelot *ocelot = container_of(ptp, struct ocelot, ptp_info);
203ecf9f9b7SVladimir Oltean struct timespec64 ts_phase, ts_period;
204cc2d87bbSYangbo Lu enum ocelot_ptp_pins ptp_pin;
205cc2d87bbSYangbo Lu unsigned long flags;
206cc2d87bbSYangbo Lu bool pps = false;
207cc2d87bbSYangbo Lu int pin = -1;
208ecf9f9b7SVladimir Oltean s64 wf_high;
209ecf9f9b7SVladimir Oltean s64 wf_low;
210cc2d87bbSYangbo Lu u32 val;
211cc2d87bbSYangbo Lu
212cc2d87bbSYangbo Lu switch (rq->type) {
213cc2d87bbSYangbo Lu case PTP_CLK_REQ_PEROUT:
214cc2d87bbSYangbo Lu /* Reject requests with unsupported flags */
215ecf9f9b7SVladimir Oltean if (rq->perout.flags & ~(PTP_PEROUT_DUTY_CYCLE |
216ecf9f9b7SVladimir Oltean PTP_PEROUT_PHASE))
217cc2d87bbSYangbo Lu return -EOPNOTSUPP;
218cc2d87bbSYangbo Lu
219cc2d87bbSYangbo Lu pin = ptp_find_pin(ocelot->ptp_clock, PTP_PF_PEROUT,
220cc2d87bbSYangbo Lu rq->perout.index);
221cc2d87bbSYangbo Lu if (pin == 0)
222cc2d87bbSYangbo Lu ptp_pin = PTP_PIN_0;
223cc2d87bbSYangbo Lu else if (pin == 1)
224cc2d87bbSYangbo Lu ptp_pin = PTP_PIN_1;
225cc2d87bbSYangbo Lu else if (pin == 2)
226cc2d87bbSYangbo Lu ptp_pin = PTP_PIN_2;
227cc2d87bbSYangbo Lu else if (pin == 3)
228cc2d87bbSYangbo Lu ptp_pin = PTP_PIN_3;
229cc2d87bbSYangbo Lu else
230cc2d87bbSYangbo Lu return -EBUSY;
231cc2d87bbSYangbo Lu
232cc2d87bbSYangbo Lu ts_period.tv_sec = rq->perout.period.sec;
233cc2d87bbSYangbo Lu ts_period.tv_nsec = rq->perout.period.nsec;
234cc2d87bbSYangbo Lu
235cc2d87bbSYangbo Lu if (ts_period.tv_sec == 1 && ts_period.tv_nsec == 0)
236cc2d87bbSYangbo Lu pps = true;
237cc2d87bbSYangbo Lu
238cc2d87bbSYangbo Lu /* Handle turning off */
239cc2d87bbSYangbo Lu if (!on) {
240cc2d87bbSYangbo Lu spin_lock_irqsave(&ocelot->ptp_clock_lock, flags);
241cc2d87bbSYangbo Lu val = PTP_PIN_CFG_ACTION(PTP_PIN_ACTION_IDLE);
242cc2d87bbSYangbo Lu ocelot_write_rix(ocelot, val, PTP_PIN_CFG, ptp_pin);
243cc2d87bbSYangbo Lu spin_unlock_irqrestore(&ocelot->ptp_clock_lock, flags);
244cc2d87bbSYangbo Lu break;
245cc2d87bbSYangbo Lu }
246cc2d87bbSYangbo Lu
247ecf9f9b7SVladimir Oltean if (rq->perout.flags & PTP_PEROUT_PHASE) {
248ecf9f9b7SVladimir Oltean ts_phase.tv_sec = rq->perout.phase.sec;
249ecf9f9b7SVladimir Oltean ts_phase.tv_nsec = rq->perout.phase.nsec;
250ecf9f9b7SVladimir Oltean } else {
251ecf9f9b7SVladimir Oltean /* Compatibility */
252ecf9f9b7SVladimir Oltean ts_phase.tv_sec = rq->perout.start.sec;
253ecf9f9b7SVladimir Oltean ts_phase.tv_nsec = rq->perout.start.nsec;
254ecf9f9b7SVladimir Oltean }
255ecf9f9b7SVladimir Oltean if (ts_phase.tv_sec || (ts_phase.tv_nsec && !pps)) {
256ecf9f9b7SVladimir Oltean dev_warn(ocelot->dev,
257ecf9f9b7SVladimir Oltean "Absolute start time not supported!\n");
258ecf9f9b7SVladimir Oltean dev_warn(ocelot->dev,
259ecf9f9b7SVladimir Oltean "Accept nsec for PPS phase adjustment, otherwise start time should be 0 0.\n");
260ecf9f9b7SVladimir Oltean return -EINVAL;
261ecf9f9b7SVladimir Oltean }
262ecf9f9b7SVladimir Oltean
263ecf9f9b7SVladimir Oltean /* Calculate waveform high and low times */
264ecf9f9b7SVladimir Oltean if (rq->perout.flags & PTP_PEROUT_DUTY_CYCLE) {
265ecf9f9b7SVladimir Oltean struct timespec64 ts_on;
266ecf9f9b7SVladimir Oltean
267ecf9f9b7SVladimir Oltean ts_on.tv_sec = rq->perout.on.sec;
268ecf9f9b7SVladimir Oltean ts_on.tv_nsec = rq->perout.on.nsec;
269ecf9f9b7SVladimir Oltean
270ecf9f9b7SVladimir Oltean wf_high = timespec64_to_ns(&ts_on);
271ecf9f9b7SVladimir Oltean } else {
272ecf9f9b7SVladimir Oltean if (pps) {
273ecf9f9b7SVladimir Oltean wf_high = 1000;
274ecf9f9b7SVladimir Oltean } else {
275ecf9f9b7SVladimir Oltean wf_high = timespec64_to_ns(&ts_period);
276ecf9f9b7SVladimir Oltean wf_high = div_s64(wf_high, 2);
277ecf9f9b7SVladimir Oltean }
278ecf9f9b7SVladimir Oltean }
279ecf9f9b7SVladimir Oltean
280ecf9f9b7SVladimir Oltean wf_low = timespec64_to_ns(&ts_period);
281ecf9f9b7SVladimir Oltean wf_low -= wf_high;
282ecf9f9b7SVladimir Oltean
283cc2d87bbSYangbo Lu /* Handle PPS request */
284cc2d87bbSYangbo Lu if (pps) {
285cc2d87bbSYangbo Lu spin_lock_irqsave(&ocelot->ptp_clock_lock, flags);
286ecf9f9b7SVladimir Oltean ocelot_write_rix(ocelot, ts_phase.tv_nsec,
287cc2d87bbSYangbo Lu PTP_PIN_WF_LOW_PERIOD, ptp_pin);
288ecf9f9b7SVladimir Oltean ocelot_write_rix(ocelot, wf_high,
289cc2d87bbSYangbo Lu PTP_PIN_WF_HIGH_PERIOD, ptp_pin);
290cc2d87bbSYangbo Lu val = PTP_PIN_CFG_ACTION(PTP_PIN_ACTION_CLOCK);
291cc2d87bbSYangbo Lu val |= PTP_PIN_CFG_SYNC;
292cc2d87bbSYangbo Lu ocelot_write_rix(ocelot, val, PTP_PIN_CFG, ptp_pin);
293cc2d87bbSYangbo Lu spin_unlock_irqrestore(&ocelot->ptp_clock_lock, flags);
294cc2d87bbSYangbo Lu break;
295cc2d87bbSYangbo Lu }
296cc2d87bbSYangbo Lu
297cc2d87bbSYangbo Lu /* Handle periodic clock */
298ecf9f9b7SVladimir Oltean if (wf_high > 0x3fffffff || wf_high <= 0x6)
299ecf9f9b7SVladimir Oltean return -EINVAL;
300ecf9f9b7SVladimir Oltean if (wf_low > 0x3fffffff || wf_low <= 0x6)
301cc2d87bbSYangbo Lu return -EINVAL;
302cc2d87bbSYangbo Lu
303cc2d87bbSYangbo Lu spin_lock_irqsave(&ocelot->ptp_clock_lock, flags);
304ecf9f9b7SVladimir Oltean ocelot_write_rix(ocelot, wf_low, PTP_PIN_WF_LOW_PERIOD,
305ecf9f9b7SVladimir Oltean ptp_pin);
306ecf9f9b7SVladimir Oltean ocelot_write_rix(ocelot, wf_high, PTP_PIN_WF_HIGH_PERIOD,
307ecf9f9b7SVladimir Oltean ptp_pin);
308cc2d87bbSYangbo Lu val = PTP_PIN_CFG_ACTION(PTP_PIN_ACTION_CLOCK);
309cc2d87bbSYangbo Lu ocelot_write_rix(ocelot, val, PTP_PIN_CFG, ptp_pin);
310cc2d87bbSYangbo Lu spin_unlock_irqrestore(&ocelot->ptp_clock_lock, flags);
311cc2d87bbSYangbo Lu break;
312cc2d87bbSYangbo Lu default:
313cc2d87bbSYangbo Lu return -EOPNOTSUPP;
314cc2d87bbSYangbo Lu }
315cc2d87bbSYangbo Lu return 0;
316cc2d87bbSYangbo Lu }
317cc2d87bbSYangbo Lu EXPORT_SYMBOL(ocelot_ptp_enable);
318cc2d87bbSYangbo Lu
ocelot_populate_l2_ptp_trap_key(struct ocelot_vcap_filter * trap)319d50e41bfSVladimir Oltean static void ocelot_populate_l2_ptp_trap_key(struct ocelot_vcap_filter *trap)
320d50e41bfSVladimir Oltean {
321d50e41bfSVladimir Oltean trap->key_type = OCELOT_VCAP_KEY_ETYPE;
322d50e41bfSVladimir Oltean *(__be16 *)trap->key.etype.etype.value = htons(ETH_P_1588);
323d50e41bfSVladimir Oltean *(__be16 *)trap->key.etype.etype.mask = htons(0xffff);
324d50e41bfSVladimir Oltean }
325d50e41bfSVladimir Oltean
326d50e41bfSVladimir Oltean static void
ocelot_populate_ipv4_ptp_event_trap_key(struct ocelot_vcap_filter * trap)327d50e41bfSVladimir Oltean ocelot_populate_ipv4_ptp_event_trap_key(struct ocelot_vcap_filter *trap)
328d50e41bfSVladimir Oltean {
329d50e41bfSVladimir Oltean trap->key_type = OCELOT_VCAP_KEY_IPV4;
330d50e41bfSVladimir Oltean trap->key.ipv4.proto.value[0] = IPPROTO_UDP;
331d50e41bfSVladimir Oltean trap->key.ipv4.proto.mask[0] = 0xff;
332d50e41bfSVladimir Oltean trap->key.ipv4.dport.value = PTP_EV_PORT;
333d50e41bfSVladimir Oltean trap->key.ipv4.dport.mask = 0xffff;
334d50e41bfSVladimir Oltean }
335d50e41bfSVladimir Oltean
336d50e41bfSVladimir Oltean static void
ocelot_populate_ipv6_ptp_event_trap_key(struct ocelot_vcap_filter * trap)337d50e41bfSVladimir Oltean ocelot_populate_ipv6_ptp_event_trap_key(struct ocelot_vcap_filter *trap)
338d50e41bfSVladimir Oltean {
339d50e41bfSVladimir Oltean trap->key_type = OCELOT_VCAP_KEY_IPV6;
3402fcde9feSVladimir Oltean trap->key.ipv6.proto.value[0] = IPPROTO_UDP;
3412fcde9feSVladimir Oltean trap->key.ipv6.proto.mask[0] = 0xff;
342d50e41bfSVladimir Oltean trap->key.ipv6.dport.value = PTP_EV_PORT;
343d50e41bfSVladimir Oltean trap->key.ipv6.dport.mask = 0xffff;
344d50e41bfSVladimir Oltean }
345d50e41bfSVladimir Oltean
346d50e41bfSVladimir Oltean static void
ocelot_populate_ipv4_ptp_general_trap_key(struct ocelot_vcap_filter * trap)347d50e41bfSVladimir Oltean ocelot_populate_ipv4_ptp_general_trap_key(struct ocelot_vcap_filter *trap)
348d50e41bfSVladimir Oltean {
349d50e41bfSVladimir Oltean trap->key_type = OCELOT_VCAP_KEY_IPV4;
350d50e41bfSVladimir Oltean trap->key.ipv4.proto.value[0] = IPPROTO_UDP;
351d50e41bfSVladimir Oltean trap->key.ipv4.proto.mask[0] = 0xff;
352d50e41bfSVladimir Oltean trap->key.ipv4.dport.value = PTP_GEN_PORT;
353d50e41bfSVladimir Oltean trap->key.ipv4.dport.mask = 0xffff;
354d50e41bfSVladimir Oltean }
355d50e41bfSVladimir Oltean
356d50e41bfSVladimir Oltean static void
ocelot_populate_ipv6_ptp_general_trap_key(struct ocelot_vcap_filter * trap)357d50e41bfSVladimir Oltean ocelot_populate_ipv6_ptp_general_trap_key(struct ocelot_vcap_filter *trap)
358d50e41bfSVladimir Oltean {
359d50e41bfSVladimir Oltean trap->key_type = OCELOT_VCAP_KEY_IPV6;
3602fcde9feSVladimir Oltean trap->key.ipv6.proto.value[0] = IPPROTO_UDP;
3612fcde9feSVladimir Oltean trap->key.ipv6.proto.mask[0] = 0xff;
362d50e41bfSVladimir Oltean trap->key.ipv6.dport.value = PTP_GEN_PORT;
363d50e41bfSVladimir Oltean trap->key.ipv6.dport.mask = 0xffff;
364d50e41bfSVladimir Oltean }
365d50e41bfSVladimir Oltean
ocelot_l2_ptp_trap_add(struct ocelot * ocelot,int port)366d50e41bfSVladimir Oltean static int ocelot_l2_ptp_trap_add(struct ocelot *ocelot, int port)
367d50e41bfSVladimir Oltean {
368d50e41bfSVladimir Oltean unsigned long l2_cookie = OCELOT_VCAP_IS2_L2_PTP_TRAP(ocelot);
369d50e41bfSVladimir Oltean
370d50e41bfSVladimir Oltean return ocelot_trap_add(ocelot, port, l2_cookie, true,
371d50e41bfSVladimir Oltean ocelot_populate_l2_ptp_trap_key);
372d50e41bfSVladimir Oltean }
373d50e41bfSVladimir Oltean
ocelot_l2_ptp_trap_del(struct ocelot * ocelot,int port)374d50e41bfSVladimir Oltean static int ocelot_l2_ptp_trap_del(struct ocelot *ocelot, int port)
375d50e41bfSVladimir Oltean {
376d50e41bfSVladimir Oltean unsigned long l2_cookie = OCELOT_VCAP_IS2_L2_PTP_TRAP(ocelot);
377d50e41bfSVladimir Oltean
378d50e41bfSVladimir Oltean return ocelot_trap_del(ocelot, port, l2_cookie);
379d50e41bfSVladimir Oltean }
380d50e41bfSVladimir Oltean
ocelot_ipv4_ptp_trap_add(struct ocelot * ocelot,int port)381d50e41bfSVladimir Oltean static int ocelot_ipv4_ptp_trap_add(struct ocelot *ocelot, int port)
382d50e41bfSVladimir Oltean {
383d50e41bfSVladimir Oltean unsigned long ipv4_gen_cookie = OCELOT_VCAP_IS2_IPV4_GEN_PTP_TRAP(ocelot);
384d50e41bfSVladimir Oltean unsigned long ipv4_ev_cookie = OCELOT_VCAP_IS2_IPV4_EV_PTP_TRAP(ocelot);
385d50e41bfSVladimir Oltean int err;
386d50e41bfSVladimir Oltean
387d50e41bfSVladimir Oltean err = ocelot_trap_add(ocelot, port, ipv4_ev_cookie, true,
388d50e41bfSVladimir Oltean ocelot_populate_ipv4_ptp_event_trap_key);
389d50e41bfSVladimir Oltean if (err)
390d50e41bfSVladimir Oltean return err;
391d50e41bfSVladimir Oltean
392d50e41bfSVladimir Oltean err = ocelot_trap_add(ocelot, port, ipv4_gen_cookie, false,
393d50e41bfSVladimir Oltean ocelot_populate_ipv4_ptp_general_trap_key);
394d50e41bfSVladimir Oltean if (err)
395d50e41bfSVladimir Oltean ocelot_trap_del(ocelot, port, ipv4_ev_cookie);
396d50e41bfSVladimir Oltean
397d50e41bfSVladimir Oltean return err;
398d50e41bfSVladimir Oltean }
399d50e41bfSVladimir Oltean
ocelot_ipv4_ptp_trap_del(struct ocelot * ocelot,int port)400d50e41bfSVladimir Oltean static int ocelot_ipv4_ptp_trap_del(struct ocelot *ocelot, int port)
401d50e41bfSVladimir Oltean {
402d50e41bfSVladimir Oltean unsigned long ipv4_gen_cookie = OCELOT_VCAP_IS2_IPV4_GEN_PTP_TRAP(ocelot);
403d50e41bfSVladimir Oltean unsigned long ipv4_ev_cookie = OCELOT_VCAP_IS2_IPV4_EV_PTP_TRAP(ocelot);
404d50e41bfSVladimir Oltean int err;
405d50e41bfSVladimir Oltean
406d50e41bfSVladimir Oltean err = ocelot_trap_del(ocelot, port, ipv4_ev_cookie);
407d50e41bfSVladimir Oltean err |= ocelot_trap_del(ocelot, port, ipv4_gen_cookie);
408d50e41bfSVladimir Oltean return err;
409d50e41bfSVladimir Oltean }
410d50e41bfSVladimir Oltean
ocelot_ipv6_ptp_trap_add(struct ocelot * ocelot,int port)411d50e41bfSVladimir Oltean static int ocelot_ipv6_ptp_trap_add(struct ocelot *ocelot, int port)
412d50e41bfSVladimir Oltean {
413d50e41bfSVladimir Oltean unsigned long ipv6_gen_cookie = OCELOT_VCAP_IS2_IPV6_GEN_PTP_TRAP(ocelot);
414d50e41bfSVladimir Oltean unsigned long ipv6_ev_cookie = OCELOT_VCAP_IS2_IPV6_EV_PTP_TRAP(ocelot);
415d50e41bfSVladimir Oltean int err;
416d50e41bfSVladimir Oltean
417d50e41bfSVladimir Oltean err = ocelot_trap_add(ocelot, port, ipv6_ev_cookie, true,
418d50e41bfSVladimir Oltean ocelot_populate_ipv6_ptp_event_trap_key);
419d50e41bfSVladimir Oltean if (err)
420d50e41bfSVladimir Oltean return err;
421d50e41bfSVladimir Oltean
422d50e41bfSVladimir Oltean err = ocelot_trap_add(ocelot, port, ipv6_gen_cookie, false,
423d50e41bfSVladimir Oltean ocelot_populate_ipv6_ptp_general_trap_key);
424d50e41bfSVladimir Oltean if (err)
425d50e41bfSVladimir Oltean ocelot_trap_del(ocelot, port, ipv6_ev_cookie);
426d50e41bfSVladimir Oltean
427d50e41bfSVladimir Oltean return err;
428d50e41bfSVladimir Oltean }
429d50e41bfSVladimir Oltean
ocelot_ipv6_ptp_trap_del(struct ocelot * ocelot,int port)430d50e41bfSVladimir Oltean static int ocelot_ipv6_ptp_trap_del(struct ocelot *ocelot, int port)
431d50e41bfSVladimir Oltean {
432d50e41bfSVladimir Oltean unsigned long ipv6_gen_cookie = OCELOT_VCAP_IS2_IPV6_GEN_PTP_TRAP(ocelot);
433d50e41bfSVladimir Oltean unsigned long ipv6_ev_cookie = OCELOT_VCAP_IS2_IPV6_EV_PTP_TRAP(ocelot);
434d50e41bfSVladimir Oltean int err;
435d50e41bfSVladimir Oltean
436d50e41bfSVladimir Oltean err = ocelot_trap_del(ocelot, port, ipv6_ev_cookie);
437d50e41bfSVladimir Oltean err |= ocelot_trap_del(ocelot, port, ipv6_gen_cookie);
438d50e41bfSVladimir Oltean return err;
439d50e41bfSVladimir Oltean }
440d50e41bfSVladimir Oltean
ocelot_setup_ptp_traps(struct ocelot * ocelot,int port,bool l2,bool l4)441d50e41bfSVladimir Oltean static int ocelot_setup_ptp_traps(struct ocelot *ocelot, int port,
442d50e41bfSVladimir Oltean bool l2, bool l4)
443d50e41bfSVladimir Oltean {
44445d0fcb5SVladimir Oltean struct ocelot_port *ocelot_port = ocelot->ports[port];
445d50e41bfSVladimir Oltean int err;
446d50e41bfSVladimir Oltean
44745d0fcb5SVladimir Oltean ocelot_port->trap_proto &= ~(OCELOT_PROTO_PTP_L2 |
44845d0fcb5SVladimir Oltean OCELOT_PROTO_PTP_L4);
44945d0fcb5SVladimir Oltean
450d50e41bfSVladimir Oltean if (l2)
451d50e41bfSVladimir Oltean err = ocelot_l2_ptp_trap_add(ocelot, port);
452d50e41bfSVladimir Oltean else
453d50e41bfSVladimir Oltean err = ocelot_l2_ptp_trap_del(ocelot, port);
454d50e41bfSVladimir Oltean if (err)
455d50e41bfSVladimir Oltean return err;
456d50e41bfSVladimir Oltean
457d50e41bfSVladimir Oltean if (l4) {
458d50e41bfSVladimir Oltean err = ocelot_ipv4_ptp_trap_add(ocelot, port);
459d50e41bfSVladimir Oltean if (err)
460d50e41bfSVladimir Oltean goto err_ipv4;
461d50e41bfSVladimir Oltean
462d50e41bfSVladimir Oltean err = ocelot_ipv6_ptp_trap_add(ocelot, port);
463d50e41bfSVladimir Oltean if (err)
464d50e41bfSVladimir Oltean goto err_ipv6;
465d50e41bfSVladimir Oltean } else {
466d50e41bfSVladimir Oltean err = ocelot_ipv4_ptp_trap_del(ocelot, port);
467d50e41bfSVladimir Oltean
468d50e41bfSVladimir Oltean err |= ocelot_ipv6_ptp_trap_del(ocelot, port);
469d50e41bfSVladimir Oltean }
470d50e41bfSVladimir Oltean if (err)
471d50e41bfSVladimir Oltean return err;
472d50e41bfSVladimir Oltean
47345d0fcb5SVladimir Oltean if (l2)
47445d0fcb5SVladimir Oltean ocelot_port->trap_proto |= OCELOT_PROTO_PTP_L2;
47545d0fcb5SVladimir Oltean if (l4)
47645d0fcb5SVladimir Oltean ocelot_port->trap_proto |= OCELOT_PROTO_PTP_L4;
47745d0fcb5SVladimir Oltean
478d50e41bfSVladimir Oltean return 0;
479d50e41bfSVladimir Oltean
480d50e41bfSVladimir Oltean err_ipv6:
481d50e41bfSVladimir Oltean ocelot_ipv4_ptp_trap_del(ocelot, port);
482d50e41bfSVladimir Oltean err_ipv4:
483d50e41bfSVladimir Oltean if (l2)
484d50e41bfSVladimir Oltean ocelot_l2_ptp_trap_del(ocelot, port);
485d50e41bfSVladimir Oltean return err;
486d50e41bfSVladimir Oltean }
487d50e41bfSVladimir Oltean
ocelot_traps_to_ptp_rx_filter(unsigned int proto)48845d0fcb5SVladimir Oltean static int ocelot_traps_to_ptp_rx_filter(unsigned int proto)
48945d0fcb5SVladimir Oltean {
49045d0fcb5SVladimir Oltean if ((proto & OCELOT_PROTO_PTP_L2) && (proto & OCELOT_PROTO_PTP_L4))
49145d0fcb5SVladimir Oltean return HWTSTAMP_FILTER_PTP_V2_EVENT;
49245d0fcb5SVladimir Oltean else if (proto & OCELOT_PROTO_PTP_L2)
49345d0fcb5SVladimir Oltean return HWTSTAMP_FILTER_PTP_V2_L2_EVENT;
49445d0fcb5SVladimir Oltean else if (proto & OCELOT_PROTO_PTP_L4)
49545d0fcb5SVladimir Oltean return HWTSTAMP_FILTER_PTP_V2_L4_EVENT;
49645d0fcb5SVladimir Oltean
49745d0fcb5SVladimir Oltean return HWTSTAMP_FILTER_NONE;
49845d0fcb5SVladimir Oltean }
49945d0fcb5SVladimir Oltean
ocelot_ptp_tx_type_to_cmd(int tx_type,int * ptp_cmd)500*8e9b5e14SVladimir Oltean static int ocelot_ptp_tx_type_to_cmd(int tx_type, int *ptp_cmd)
501*8e9b5e14SVladimir Oltean {
502*8e9b5e14SVladimir Oltean switch (tx_type) {
503*8e9b5e14SVladimir Oltean case HWTSTAMP_TX_ON:
504*8e9b5e14SVladimir Oltean *ptp_cmd = IFH_REW_OP_TWO_STEP_PTP;
505*8e9b5e14SVladimir Oltean break;
506*8e9b5e14SVladimir Oltean case HWTSTAMP_TX_ONESTEP_SYNC:
507*8e9b5e14SVladimir Oltean /* IFH_REW_OP_ONE_STEP_PTP updates the correctionField,
508*8e9b5e14SVladimir Oltean * what we need to update is the originTimestamp.
509*8e9b5e14SVladimir Oltean */
510*8e9b5e14SVladimir Oltean *ptp_cmd = IFH_REW_OP_ORIGIN_PTP;
511*8e9b5e14SVladimir Oltean break;
512*8e9b5e14SVladimir Oltean case HWTSTAMP_TX_OFF:
513*8e9b5e14SVladimir Oltean *ptp_cmd = 0;
514*8e9b5e14SVladimir Oltean break;
515*8e9b5e14SVladimir Oltean default:
516*8e9b5e14SVladimir Oltean return -ERANGE;
517*8e9b5e14SVladimir Oltean }
518*8e9b5e14SVladimir Oltean
519*8e9b5e14SVladimir Oltean return 0;
520*8e9b5e14SVladimir Oltean }
521*8e9b5e14SVladimir Oltean
ocelot_hwstamp_get(struct ocelot * ocelot,int port,struct ifreq * ifr)522d50e41bfSVladimir Oltean int ocelot_hwstamp_get(struct ocelot *ocelot, int port, struct ifreq *ifr)
523d50e41bfSVladimir Oltean {
52445d0fcb5SVladimir Oltean struct ocelot_port *ocelot_port = ocelot->ports[port];
52545d0fcb5SVladimir Oltean struct hwtstamp_config cfg = {};
52645d0fcb5SVladimir Oltean
52745d0fcb5SVladimir Oltean switch (ocelot_port->ptp_cmd) {
52845d0fcb5SVladimir Oltean case IFH_REW_OP_TWO_STEP_PTP:
52945d0fcb5SVladimir Oltean cfg.tx_type = HWTSTAMP_TX_ON;
53045d0fcb5SVladimir Oltean break;
53145d0fcb5SVladimir Oltean case IFH_REW_OP_ORIGIN_PTP:
53245d0fcb5SVladimir Oltean cfg.tx_type = HWTSTAMP_TX_ONESTEP_SYNC;
53345d0fcb5SVladimir Oltean break;
53445d0fcb5SVladimir Oltean default:
53545d0fcb5SVladimir Oltean cfg.tx_type = HWTSTAMP_TX_OFF;
53645d0fcb5SVladimir Oltean break;
53745d0fcb5SVladimir Oltean }
53845d0fcb5SVladimir Oltean
53945d0fcb5SVladimir Oltean cfg.rx_filter = ocelot_traps_to_ptp_rx_filter(ocelot_port->trap_proto);
54045d0fcb5SVladimir Oltean
54145d0fcb5SVladimir Oltean return copy_to_user(ifr->ifr_data, &cfg, sizeof(cfg)) ? -EFAULT : 0;
542d50e41bfSVladimir Oltean }
543d50e41bfSVladimir Oltean EXPORT_SYMBOL(ocelot_hwstamp_get);
544d50e41bfSVladimir Oltean
ocelot_hwstamp_set(struct ocelot * ocelot,int port,struct ifreq * ifr)545d50e41bfSVladimir Oltean int ocelot_hwstamp_set(struct ocelot *ocelot, int port, struct ifreq *ifr)
546d50e41bfSVladimir Oltean {
547d50e41bfSVladimir Oltean struct ocelot_port *ocelot_port = ocelot->ports[port];
548*8e9b5e14SVladimir Oltean int ptp_cmd, old_ptp_cmd = ocelot_port->ptp_cmd;
549d50e41bfSVladimir Oltean bool l2 = false, l4 = false;
550d50e41bfSVladimir Oltean struct hwtstamp_config cfg;
551*8e9b5e14SVladimir Oltean bool old_l2, old_l4;
552d50e41bfSVladimir Oltean int err;
553d50e41bfSVladimir Oltean
554d50e41bfSVladimir Oltean if (copy_from_user(&cfg, ifr->ifr_data, sizeof(cfg)))
555d50e41bfSVladimir Oltean return -EFAULT;
556d50e41bfSVladimir Oltean
557d50e41bfSVladimir Oltean /* Tx type sanity check */
558*8e9b5e14SVladimir Oltean err = ocelot_ptp_tx_type_to_cmd(cfg.tx_type, &ptp_cmd);
559*8e9b5e14SVladimir Oltean if (err)
560*8e9b5e14SVladimir Oltean return err;
561d50e41bfSVladimir Oltean
562d50e41bfSVladimir Oltean switch (cfg.rx_filter) {
563d50e41bfSVladimir Oltean case HWTSTAMP_FILTER_NONE:
564d50e41bfSVladimir Oltean break;
565d50e41bfSVladimir Oltean case HWTSTAMP_FILTER_PTP_V2_L4_EVENT:
566d50e41bfSVladimir Oltean case HWTSTAMP_FILTER_PTP_V2_L4_SYNC:
567d50e41bfSVladimir Oltean case HWTSTAMP_FILTER_PTP_V2_L4_DELAY_REQ:
568d50e41bfSVladimir Oltean l4 = true;
569d50e41bfSVladimir Oltean break;
570d50e41bfSVladimir Oltean case HWTSTAMP_FILTER_PTP_V2_L2_EVENT:
571d50e41bfSVladimir Oltean case HWTSTAMP_FILTER_PTP_V2_L2_SYNC:
572d50e41bfSVladimir Oltean case HWTSTAMP_FILTER_PTP_V2_L2_DELAY_REQ:
573d50e41bfSVladimir Oltean l2 = true;
574d50e41bfSVladimir Oltean break;
575d50e41bfSVladimir Oltean case HWTSTAMP_FILTER_PTP_V2_EVENT:
576d50e41bfSVladimir Oltean case HWTSTAMP_FILTER_PTP_V2_SYNC:
577d50e41bfSVladimir Oltean case HWTSTAMP_FILTER_PTP_V2_DELAY_REQ:
578d50e41bfSVladimir Oltean l2 = true;
579d50e41bfSVladimir Oltean l4 = true;
580d50e41bfSVladimir Oltean break;
581d50e41bfSVladimir Oltean default:
582d50e41bfSVladimir Oltean return -ERANGE;
583d50e41bfSVladimir Oltean }
584d50e41bfSVladimir Oltean
585*8e9b5e14SVladimir Oltean old_l2 = ocelot_port->trap_proto & OCELOT_PROTO_PTP_L2;
586*8e9b5e14SVladimir Oltean old_l4 = ocelot_port->trap_proto & OCELOT_PROTO_PTP_L4;
587*8e9b5e14SVladimir Oltean
588d50e41bfSVladimir Oltean err = ocelot_setup_ptp_traps(ocelot, port, l2, l4);
58945d0fcb5SVladimir Oltean if (err)
590d50e41bfSVladimir Oltean return err;
591d50e41bfSVladimir Oltean
592*8e9b5e14SVladimir Oltean ocelot_port->ptp_cmd = ptp_cmd;
593*8e9b5e14SVladimir Oltean
59445d0fcb5SVladimir Oltean cfg.rx_filter = ocelot_traps_to_ptp_rx_filter(ocelot_port->trap_proto);
595d50e41bfSVladimir Oltean
596*8e9b5e14SVladimir Oltean if (copy_to_user(ifr->ifr_data, &cfg, sizeof(cfg))) {
597*8e9b5e14SVladimir Oltean err = -EFAULT;
598*8e9b5e14SVladimir Oltean goto out_restore_ptp_traps;
599*8e9b5e14SVladimir Oltean }
600*8e9b5e14SVladimir Oltean
601*8e9b5e14SVladimir Oltean return 0;
602*8e9b5e14SVladimir Oltean out_restore_ptp_traps:
603*8e9b5e14SVladimir Oltean ocelot_setup_ptp_traps(ocelot, port, old_l2, old_l4);
604*8e9b5e14SVladimir Oltean ocelot_port->ptp_cmd = old_ptp_cmd;
605*8e9b5e14SVladimir Oltean return err;
606d50e41bfSVladimir Oltean }
607d50e41bfSVladimir Oltean EXPORT_SYMBOL(ocelot_hwstamp_set);
608d50e41bfSVladimir Oltean
ocelot_get_ts_info(struct ocelot * ocelot,int port,struct ethtool_ts_info * info)609d50e41bfSVladimir Oltean int ocelot_get_ts_info(struct ocelot *ocelot, int port,
610d50e41bfSVladimir Oltean struct ethtool_ts_info *info)
611d50e41bfSVladimir Oltean {
612d50e41bfSVladimir Oltean info->phc_index = ocelot->ptp_clock ?
613d50e41bfSVladimir Oltean ptp_clock_index(ocelot->ptp_clock) : -1;
614d50e41bfSVladimir Oltean if (info->phc_index == -1) {
615d50e41bfSVladimir Oltean info->so_timestamping |= SOF_TIMESTAMPING_TX_SOFTWARE |
616d50e41bfSVladimir Oltean SOF_TIMESTAMPING_RX_SOFTWARE |
617d50e41bfSVladimir Oltean SOF_TIMESTAMPING_SOFTWARE;
618d50e41bfSVladimir Oltean return 0;
619d50e41bfSVladimir Oltean }
620d50e41bfSVladimir Oltean info->so_timestamping |= SOF_TIMESTAMPING_TX_SOFTWARE |
621d50e41bfSVladimir Oltean SOF_TIMESTAMPING_RX_SOFTWARE |
622d50e41bfSVladimir Oltean SOF_TIMESTAMPING_SOFTWARE |
623d50e41bfSVladimir Oltean SOF_TIMESTAMPING_TX_HARDWARE |
624d50e41bfSVladimir Oltean SOF_TIMESTAMPING_RX_HARDWARE |
625d50e41bfSVladimir Oltean SOF_TIMESTAMPING_RAW_HARDWARE;
626d50e41bfSVladimir Oltean info->tx_types = BIT(HWTSTAMP_TX_OFF) | BIT(HWTSTAMP_TX_ON) |
627d50e41bfSVladimir Oltean BIT(HWTSTAMP_TX_ONESTEP_SYNC);
628d50e41bfSVladimir Oltean info->rx_filters = BIT(HWTSTAMP_FILTER_NONE) |
629d50e41bfSVladimir Oltean BIT(HWTSTAMP_FILTER_PTP_V2_EVENT) |
630d50e41bfSVladimir Oltean BIT(HWTSTAMP_FILTER_PTP_V2_L2_EVENT) |
631d50e41bfSVladimir Oltean BIT(HWTSTAMP_FILTER_PTP_V2_L4_EVENT);
632d50e41bfSVladimir Oltean
633d50e41bfSVladimir Oltean return 0;
634d50e41bfSVladimir Oltean }
635d50e41bfSVladimir Oltean EXPORT_SYMBOL(ocelot_get_ts_info);
636d50e41bfSVladimir Oltean
ocelot_port_dequeue_ptp_tx_skb(struct ocelot * ocelot,int port,u8 ts_id,u32 seqid)6373b327726SVladimir Oltean static struct sk_buff *ocelot_port_dequeue_ptp_tx_skb(struct ocelot *ocelot,
6383b327726SVladimir Oltean int port, u8 ts_id,
6393b327726SVladimir Oltean u32 seqid)
640d50e41bfSVladimir Oltean {
641d50e41bfSVladimir Oltean struct ocelot_port *ocelot_port = ocelot->ports[port];
6423b327726SVladimir Oltean struct sk_buff *skb, *skb_tmp, *skb_match = NULL;
6433b327726SVladimir Oltean struct ptp_header *hdr;
644d50e41bfSVladimir Oltean
6455f2c6930SVladimir Oltean spin_lock(&ocelot->ts_id_lock);
646d50e41bfSVladimir Oltean
6473b327726SVladimir Oltean skb_queue_walk_safe(&ocelot_port->tx_skbs, skb, skb_tmp) {
6483b327726SVladimir Oltean if (OCELOT_SKB_CB(skb)->ts_id != ts_id)
6493b327726SVladimir Oltean continue;
6503b327726SVladimir Oltean
6513b327726SVladimir Oltean /* Check that the timestamp ID is for the expected PTP
6523b327726SVladimir Oltean * sequenceId. We don't have to test ptp_parse_header() against
6533b327726SVladimir Oltean * NULL, because we've pre-validated the packet's ptp_class.
6543b327726SVladimir Oltean */
6553b327726SVladimir Oltean hdr = ptp_parse_header(skb, OCELOT_SKB_CB(skb)->ptp_class);
6563b327726SVladimir Oltean if (seqid != ntohs(hdr->sequence_id))
6573b327726SVladimir Oltean continue;
6583b327726SVladimir Oltean
6593b327726SVladimir Oltean __skb_unlink(skb, &ocelot_port->tx_skbs);
6603b327726SVladimir Oltean ocelot->ptp_skbs_in_flight--;
6613b327726SVladimir Oltean skb_match = skb;
6623b327726SVladimir Oltean break;
6633b327726SVladimir Oltean }
6643b327726SVladimir Oltean
6653b327726SVladimir Oltean spin_unlock(&ocelot->ts_id_lock);
6663b327726SVladimir Oltean
6673b327726SVladimir Oltean return skb_match;
6683b327726SVladimir Oltean }
6693b327726SVladimir Oltean
ocelot_port_queue_ptp_tx_skb(struct ocelot * ocelot,int port,struct sk_buff * clone)6703b327726SVladimir Oltean static int ocelot_port_queue_ptp_tx_skb(struct ocelot *ocelot, int port,
6713b327726SVladimir Oltean struct sk_buff *clone)
6723b327726SVladimir Oltean {
6733b327726SVladimir Oltean struct ocelot_port *ocelot_port = ocelot->ports[port];
6743b327726SVladimir Oltean DECLARE_BITMAP(ts_id_in_flight, OCELOT_MAX_PTP_ID);
6753b327726SVladimir Oltean struct sk_buff *skb, *skb_tmp;
6763b327726SVladimir Oltean unsigned long n;
6773b327726SVladimir Oltean
6783b327726SVladimir Oltean spin_lock(&ocelot->ts_id_lock);
6793b327726SVladimir Oltean
6803b327726SVladimir Oltean /* To get a better chance of acquiring a timestamp ID, first flush the
6813b327726SVladimir Oltean * stale packets still waiting in the TX timestamping queue. They are
6823b327726SVladimir Oltean * probably lost.
6833b327726SVladimir Oltean */
6843b327726SVladimir Oltean skb_queue_walk_safe(&ocelot_port->tx_skbs, skb, skb_tmp) {
6853b327726SVladimir Oltean if (time_before(OCELOT_SKB_CB(skb)->ptp_tx_time +
6863b327726SVladimir Oltean OCELOT_PTP_TX_TSTAMP_TIMEOUT, jiffies)) {
6873b327726SVladimir Oltean dev_warn_ratelimited(ocelot->dev,
6883b327726SVladimir Oltean "port %d invalidating stale timestamp ID %u which seems lost\n",
6893b327726SVladimir Oltean port, OCELOT_SKB_CB(skb)->ts_id);
6903b327726SVladimir Oltean __skb_unlink(skb, &ocelot_port->tx_skbs);
6913b327726SVladimir Oltean kfree_skb(skb);
6923b327726SVladimir Oltean ocelot->ptp_skbs_in_flight--;
6933b327726SVladimir Oltean } else {
6943b327726SVladimir Oltean __set_bit(OCELOT_SKB_CB(skb)->ts_id, ts_id_in_flight);
6953b327726SVladimir Oltean }
6963b327726SVladimir Oltean }
6973b327726SVladimir Oltean
6983b327726SVladimir Oltean if (ocelot->ptp_skbs_in_flight == OCELOT_PTP_FIFO_SIZE) {
6995f2c6930SVladimir Oltean spin_unlock(&ocelot->ts_id_lock);
700d50e41bfSVladimir Oltean return -EBUSY;
701d50e41bfSVladimir Oltean }
702d50e41bfSVladimir Oltean
7033b327726SVladimir Oltean n = find_first_zero_bit(ts_id_in_flight, OCELOT_MAX_PTP_ID);
7043b327726SVladimir Oltean if (n == OCELOT_MAX_PTP_ID) {
7053b327726SVladimir Oltean spin_unlock(&ocelot->ts_id_lock);
7063b327726SVladimir Oltean return -EBUSY;
7073b327726SVladimir Oltean }
708d50e41bfSVladimir Oltean
7093b327726SVladimir Oltean /* Found an available timestamp ID, use it */
7103b327726SVladimir Oltean OCELOT_SKB_CB(clone)->ts_id = n;
7113b327726SVladimir Oltean OCELOT_SKB_CB(clone)->ptp_tx_time = jiffies;
712d50e41bfSVladimir Oltean ocelot->ptp_skbs_in_flight++;
7133b327726SVladimir Oltean __skb_queue_tail(&ocelot_port->tx_skbs, clone);
714d50e41bfSVladimir Oltean
7155f2c6930SVladimir Oltean spin_unlock(&ocelot->ts_id_lock);
716d50e41bfSVladimir Oltean
7173b327726SVladimir Oltean dev_dbg_ratelimited(ocelot->dev, "port %d timestamp id %lu\n", port, n);
7183b327726SVladimir Oltean
719d50e41bfSVladimir Oltean return 0;
720d50e41bfSVladimir Oltean }
721d50e41bfSVladimir Oltean
ocelot_ptp_is_onestep_sync(struct sk_buff * skb,unsigned int ptp_class)722d50e41bfSVladimir Oltean static bool ocelot_ptp_is_onestep_sync(struct sk_buff *skb,
723d50e41bfSVladimir Oltean unsigned int ptp_class)
724d50e41bfSVladimir Oltean {
725d50e41bfSVladimir Oltean struct ptp_header *hdr;
726d50e41bfSVladimir Oltean u8 msgtype, twostep;
727d50e41bfSVladimir Oltean
728d50e41bfSVladimir Oltean hdr = ptp_parse_header(skb, ptp_class);
729d50e41bfSVladimir Oltean if (!hdr)
730d50e41bfSVladimir Oltean return false;
731d50e41bfSVladimir Oltean
732d50e41bfSVladimir Oltean msgtype = ptp_get_msgtype(hdr, ptp_class);
733d50e41bfSVladimir Oltean twostep = hdr->flag_field[0] & 0x2;
734d50e41bfSVladimir Oltean
735d50e41bfSVladimir Oltean if (msgtype == PTP_MSGTYPE_SYNC && twostep == 0)
736d50e41bfSVladimir Oltean return true;
737d50e41bfSVladimir Oltean
738d50e41bfSVladimir Oltean return false;
739d50e41bfSVladimir Oltean }
740d50e41bfSVladimir Oltean
ocelot_port_txtstamp_request(struct ocelot * ocelot,int port,struct sk_buff * skb,struct sk_buff ** clone)741d50e41bfSVladimir Oltean int ocelot_port_txtstamp_request(struct ocelot *ocelot, int port,
742d50e41bfSVladimir Oltean struct sk_buff *skb,
743d50e41bfSVladimir Oltean struct sk_buff **clone)
744d50e41bfSVladimir Oltean {
745d50e41bfSVladimir Oltean struct ocelot_port *ocelot_port = ocelot->ports[port];
746d50e41bfSVladimir Oltean u8 ptp_cmd = ocelot_port->ptp_cmd;
747d50e41bfSVladimir Oltean unsigned int ptp_class;
748d50e41bfSVladimir Oltean int err;
749d50e41bfSVladimir Oltean
750d50e41bfSVladimir Oltean /* Don't do anything if PTP timestamping not enabled */
751d50e41bfSVladimir Oltean if (!ptp_cmd)
752d50e41bfSVladimir Oltean return 0;
753d50e41bfSVladimir Oltean
754d50e41bfSVladimir Oltean ptp_class = ptp_classify_raw(skb);
755d50e41bfSVladimir Oltean if (ptp_class == PTP_CLASS_NONE)
756d50e41bfSVladimir Oltean return -EINVAL;
757d50e41bfSVladimir Oltean
758d50e41bfSVladimir Oltean /* Store ptp_cmd in OCELOT_SKB_CB(skb)->ptp_cmd */
759d50e41bfSVladimir Oltean if (ptp_cmd == IFH_REW_OP_ORIGIN_PTP) {
760d50e41bfSVladimir Oltean if (ocelot_ptp_is_onestep_sync(skb, ptp_class)) {
761d50e41bfSVladimir Oltean OCELOT_SKB_CB(skb)->ptp_cmd = ptp_cmd;
762d50e41bfSVladimir Oltean return 0;
763d50e41bfSVladimir Oltean }
764d50e41bfSVladimir Oltean
765d50e41bfSVladimir Oltean /* Fall back to two-step timestamping */
766d50e41bfSVladimir Oltean ptp_cmd = IFH_REW_OP_TWO_STEP_PTP;
767d50e41bfSVladimir Oltean }
768d50e41bfSVladimir Oltean
769d50e41bfSVladimir Oltean if (ptp_cmd == IFH_REW_OP_TWO_STEP_PTP) {
770d50e41bfSVladimir Oltean *clone = skb_clone_sk(skb);
771d50e41bfSVladimir Oltean if (!(*clone))
772d50e41bfSVladimir Oltean return -ENOMEM;
773d50e41bfSVladimir Oltean
7743b327726SVladimir Oltean /* Store timestamp ID in OCELOT_SKB_CB(clone)->ts_id */
7753b327726SVladimir Oltean err = ocelot_port_queue_ptp_tx_skb(ocelot, port, *clone);
776d275b713SVladimir Oltean if (err) {
777d275b713SVladimir Oltean kfree_skb(*clone);
778d50e41bfSVladimir Oltean return err;
779d275b713SVladimir Oltean }
780d50e41bfSVladimir Oltean
7813b327726SVladimir Oltean skb_shinfo(*clone)->tx_flags |= SKBTX_IN_PROGRESS;
782d50e41bfSVladimir Oltean OCELOT_SKB_CB(skb)->ptp_cmd = ptp_cmd;
783d50e41bfSVladimir Oltean OCELOT_SKB_CB(*clone)->ptp_class = ptp_class;
784d50e41bfSVladimir Oltean }
785d50e41bfSVladimir Oltean
786d50e41bfSVladimir Oltean return 0;
787d50e41bfSVladimir Oltean }
788d50e41bfSVladimir Oltean EXPORT_SYMBOL(ocelot_port_txtstamp_request);
789d50e41bfSVladimir Oltean
ocelot_get_hwtimestamp(struct ocelot * ocelot,struct timespec64 * ts)790d50e41bfSVladimir Oltean static void ocelot_get_hwtimestamp(struct ocelot *ocelot,
791d50e41bfSVladimir Oltean struct timespec64 *ts)
792d50e41bfSVladimir Oltean {
793d50e41bfSVladimir Oltean unsigned long flags;
794d50e41bfSVladimir Oltean u32 val;
795d50e41bfSVladimir Oltean
796d50e41bfSVladimir Oltean spin_lock_irqsave(&ocelot->ptp_clock_lock, flags);
797d50e41bfSVladimir Oltean
798d50e41bfSVladimir Oltean /* Read current PTP time to get seconds */
799d50e41bfSVladimir Oltean val = ocelot_read_rix(ocelot, PTP_PIN_CFG, TOD_ACC_PIN);
800d50e41bfSVladimir Oltean
801d50e41bfSVladimir Oltean val &= ~(PTP_PIN_CFG_SYNC | PTP_PIN_CFG_ACTION_MASK | PTP_PIN_CFG_DOM);
802d50e41bfSVladimir Oltean val |= PTP_PIN_CFG_ACTION(PTP_PIN_ACTION_SAVE);
803d50e41bfSVladimir Oltean ocelot_write_rix(ocelot, val, PTP_PIN_CFG, TOD_ACC_PIN);
804d50e41bfSVladimir Oltean ts->tv_sec = ocelot_read_rix(ocelot, PTP_PIN_TOD_SEC_LSB, TOD_ACC_PIN);
805d50e41bfSVladimir Oltean
806d50e41bfSVladimir Oltean /* Read packet HW timestamp from FIFO */
807d50e41bfSVladimir Oltean val = ocelot_read(ocelot, SYS_PTP_TXSTAMP);
808d50e41bfSVladimir Oltean ts->tv_nsec = SYS_PTP_TXSTAMP_PTP_TXSTAMP(val);
809d50e41bfSVladimir Oltean
810d50e41bfSVladimir Oltean /* Sec has incremented since the ts was registered */
811d50e41bfSVladimir Oltean if ((ts->tv_sec & 0x1) != !!(val & SYS_PTP_TXSTAMP_PTP_TXSTAMP_SEC))
812d50e41bfSVladimir Oltean ts->tv_sec--;
813d50e41bfSVladimir Oltean
814d50e41bfSVladimir Oltean spin_unlock_irqrestore(&ocelot->ptp_clock_lock, flags);
815d50e41bfSVladimir Oltean }
816d50e41bfSVladimir Oltean
ocelot_get_txtstamp(struct ocelot * ocelot)817d50e41bfSVladimir Oltean void ocelot_get_txtstamp(struct ocelot *ocelot)
818d50e41bfSVladimir Oltean {
819d50e41bfSVladimir Oltean int budget = OCELOT_PTP_QUEUE_SZ;
820d50e41bfSVladimir Oltean
821d50e41bfSVladimir Oltean while (budget--) {
822d50e41bfSVladimir Oltean struct skb_shared_hwtstamps shhwtstamps;
823d50e41bfSVladimir Oltean u32 val, id, seqid, txport;
8243b327726SVladimir Oltean struct sk_buff *skb_match;
825d50e41bfSVladimir Oltean struct timespec64 ts;
826d50e41bfSVladimir Oltean
827d50e41bfSVladimir Oltean val = ocelot_read(ocelot, SYS_PTP_STATUS);
828d50e41bfSVladimir Oltean
829d50e41bfSVladimir Oltean /* Check if a timestamp can be retrieved */
830d50e41bfSVladimir Oltean if (!(val & SYS_PTP_STATUS_PTP_MESS_VLD))
831d50e41bfSVladimir Oltean break;
832d50e41bfSVladimir Oltean
833d50e41bfSVladimir Oltean WARN_ON(val & SYS_PTP_STATUS_PTP_OVFL);
834d50e41bfSVladimir Oltean
835d50e41bfSVladimir Oltean /* Retrieve the ts ID and Tx port */
836d50e41bfSVladimir Oltean id = SYS_PTP_STATUS_PTP_MESS_ID_X(val);
837d50e41bfSVladimir Oltean txport = SYS_PTP_STATUS_PTP_MESS_TXPORT_X(val);
838d50e41bfSVladimir Oltean seqid = SYS_PTP_STATUS_PTP_MESS_SEQ_ID(val);
839d50e41bfSVladimir Oltean
840d50e41bfSVladimir Oltean /* Retrieve its associated skb */
8413b327726SVladimir Oltean skb_match = ocelot_port_dequeue_ptp_tx_skb(ocelot, txport, id,
8423b327726SVladimir Oltean seqid);
8433b327726SVladimir Oltean if (!skb_match) {
8443b327726SVladimir Oltean dev_warn_ratelimited(ocelot->dev,
8453b327726SVladimir Oltean "port %d received TX timestamp (seqid %d, ts id %u) for packet previously declared stale\n",
8463b327726SVladimir Oltean txport, seqid, id);
8477f42e62aSVladimir Oltean goto next_ts;
848d50e41bfSVladimir Oltean }
849d50e41bfSVladimir Oltean
850d50e41bfSVladimir Oltean /* Get the h/w timestamp */
851d50e41bfSVladimir Oltean ocelot_get_hwtimestamp(ocelot, &ts);
852d50e41bfSVladimir Oltean
853d50e41bfSVladimir Oltean /* Set the timestamp into the skb */
854d50e41bfSVladimir Oltean memset(&shhwtstamps, 0, sizeof(shhwtstamps));
855d50e41bfSVladimir Oltean shhwtstamps.hwtstamp = ktime_set(ts.tv_sec, ts.tv_nsec);
856d50e41bfSVladimir Oltean skb_complete_tx_timestamp(skb_match, &shhwtstamps);
857d50e41bfSVladimir Oltean
8587f42e62aSVladimir Oltean next_ts:
859d50e41bfSVladimir Oltean ocelot_write(ocelot, SYS_PTP_NXT_PTP_NXT, SYS_PTP_NXT);
860d50e41bfSVladimir Oltean }
861d50e41bfSVladimir Oltean }
862d50e41bfSVladimir Oltean EXPORT_SYMBOL(ocelot_get_txtstamp);
863d50e41bfSVladimir Oltean
ocelot_init_timestamp(struct ocelot * ocelot,const struct ptp_clock_info * info)864881321b6SVladimir Oltean int ocelot_init_timestamp(struct ocelot *ocelot,
865881321b6SVladimir Oltean const struct ptp_clock_info *info)
8662b49d128SYangbo Lu {
8672b49d128SYangbo Lu struct ptp_clock *ptp_clock;
868cc2d87bbSYangbo Lu int i;
8692b49d128SYangbo Lu
8702b49d128SYangbo Lu ocelot->ptp_info = *info;
871cc2d87bbSYangbo Lu
872cc2d87bbSYangbo Lu for (i = 0; i < OCELOT_PTP_PINS_NUM; i++) {
873cc2d87bbSYangbo Lu struct ptp_pin_desc *p = &ocelot->ptp_pins[i];
874cc2d87bbSYangbo Lu
875cc2d87bbSYangbo Lu snprintf(p->name, sizeof(p->name), "switch_1588_dat%d", i);
876cc2d87bbSYangbo Lu p->index = i;
877cc2d87bbSYangbo Lu p->func = PTP_PF_NONE;
878cc2d87bbSYangbo Lu }
879cc2d87bbSYangbo Lu
880cc2d87bbSYangbo Lu ocelot->ptp_info.pin_config = &ocelot->ptp_pins[0];
881cc2d87bbSYangbo Lu
8822b49d128SYangbo Lu ptp_clock = ptp_clock_register(&ocelot->ptp_info, ocelot->dev);
8832b49d128SYangbo Lu if (IS_ERR(ptp_clock))
8842b49d128SYangbo Lu return PTR_ERR(ptp_clock);
8852b49d128SYangbo Lu /* Check if PHC support is missing at the configuration level */
8862b49d128SYangbo Lu if (!ptp_clock)
8872b49d128SYangbo Lu return 0;
8882b49d128SYangbo Lu
8892b49d128SYangbo Lu ocelot->ptp_clock = ptp_clock;
8902b49d128SYangbo Lu
8912b49d128SYangbo Lu ocelot_write(ocelot, SYS_PTP_CFG_PTP_STAMP_WID(30), SYS_PTP_CFG);
8922b49d128SYangbo Lu ocelot_write(ocelot, 0xffffffff, ANA_TABLES_PTP_ID_LOW);
8932b49d128SYangbo Lu ocelot_write(ocelot, 0xffffffff, ANA_TABLES_PTP_ID_HIGH);
8942b49d128SYangbo Lu
8952b49d128SYangbo Lu ocelot_write(ocelot, PTP_CFG_MISC_PTP_EN, PTP_CFG_MISC);
8962b49d128SYangbo Lu
8972b49d128SYangbo Lu return 0;
8982b49d128SYangbo Lu }
8992b49d128SYangbo Lu EXPORT_SYMBOL(ocelot_init_timestamp);
9002b49d128SYangbo Lu
ocelot_deinit_timestamp(struct ocelot * ocelot)9012b49d128SYangbo Lu int ocelot_deinit_timestamp(struct ocelot *ocelot)
9022b49d128SYangbo Lu {
9032b49d128SYangbo Lu if (ocelot->ptp_clock)
9042b49d128SYangbo Lu ptp_clock_unregister(ocelot->ptp_clock);
9052b49d128SYangbo Lu return 0;
9062b49d128SYangbo Lu }
9072b49d128SYangbo Lu EXPORT_SYMBOL(ocelot_deinit_timestamp);
908