1 // SPDX-License-Identifier: GPL-2.0
3 // Copyright (C) 2018 MOSER-BAER AG
6 #define pr_fmt(fmt) "InES_PTP: " fmt
8 #include <linux/ethtool.h>
9 #include <linux/export.h>
10 #include <linux/if_vlan.h>
11 #include <linux/mii_timestamper.h>
12 #include <linux/module.h>
13 #include <linux/net_tstamp.h>
15 #include <linux/of_address.h>
16 #include <linux/of_irq.h>
17 #include <linux/phy.h>
18 #include <linux/platform_device.h>
19 #include <linux/ptp_classify.h>
20 #include <linux/ptp_clock_kernel.h>
21 #include <linux/stddef.h>
23 MODULE_DESCRIPTION("Driver for the ZHAW InES PTP time stamping IP core");
24 MODULE_AUTHOR("Richard Cochran <richardcochran@gmail.com>");
25 MODULE_VERSION("1.0");
26 MODULE_LICENSE("GPL");
29 #define MCAST_MAC_SELECT_SHIFT 2
30 #define MCAST_MAC_SELECT_MASK 0x3
31 #define IO_RESET BIT(1)
32 #define PTP_RESET BIT(0)
34 /* VERSION register */
35 #define IF_MAJOR_VER_SHIFT 12
36 #define IF_MAJOR_VER_MASK 0xf
37 #define IF_MINOR_VER_SHIFT 8
38 #define IF_MINOR_VER_MASK 0xf
39 #define FPGA_MAJOR_VER_SHIFT 4
40 #define FPGA_MAJOR_VER_MASK 0xf
41 #define FPGA_MINOR_VER_SHIFT 0
42 #define FPGA_MINOR_VER_MASK 0xf
44 /* INT_STAT register */
45 #define RX_INTR_STATUS_3 BIT(5)
46 #define RX_INTR_STATUS_2 BIT(4)
47 #define RX_INTR_STATUS_1 BIT(3)
48 #define TX_INTR_STATUS_3 BIT(2)
49 #define TX_INTR_STATUS_2 BIT(1)
50 #define TX_INTR_STATUS_1 BIT(0)
52 /* INT_MSK register */
53 #define RX_INTR_MASK_3 BIT(5)
54 #define RX_INTR_MASK_2 BIT(4)
55 #define RX_INTR_MASK_1 BIT(3)
56 #define TX_INTR_MASK_3 BIT(2)
57 #define TX_INTR_MASK_2 BIT(1)
58 #define TX_INTR_MASK_1 BIT(0)
60 /* BUF_STAT register */
61 #define RX_FIFO_NE_3 BIT(5)
62 #define RX_FIFO_NE_2 BIT(4)
63 #define RX_FIFO_NE_1 BIT(3)
64 #define TX_FIFO_NE_3 BIT(2)
65 #define TX_FIFO_NE_2 BIT(1)
66 #define TX_FIFO_NE_1 BIT(0)
68 /* PORT_CONF register */
69 #define CM_ONE_STEP BIT(6)
70 #define PHY_SPEED_SHIFT 4
71 #define PHY_SPEED_MASK 0x3
72 #define P2P_DELAY_WR_POS_SHIFT 2
73 #define P2P_DELAY_WR_POS_MASK 0x3
74 #define PTP_MODE_SHIFT 0
75 #define PTP_MODE_MASK 0x3
77 /* TS_STAT_TX register */
78 #define TS_ENABLE BIT(15)
79 #define DATA_READ_POS_SHIFT 8
80 #define DATA_READ_POS_MASK 0x1f
81 #define DISCARDED_EVENTS_SHIFT 4
82 #define DISCARDED_EVENTS_MASK 0xf
84 #define INES_N_PORTS 3
85 #define INES_REGISTER_SIZE 0x80
86 #define INES_PORT_OFFSET 0x20
87 #define INES_PORT_SIZE 0x20
88 #define INES_FIFO_DEPTH 90
89 #define INES_MAX_EVENTS 100
93 #define TC_E2E_PTP_V2 2
94 #define TC_P2P_PTP_V2 3
96 #define PHY_SPEED_10 0
97 #define PHY_SPEED_100 1
98 #define PHY_SPEED_1000 2
101 ((PHY_SPEED_1000 << PHY_SPEED_SHIFT) | (BC_PTP_V2 << PTP_MODE_SHIFT))
103 #define ines_read32(s, r) __raw_readl((void __iomem *)&s->regs->r)
104 #define ines_write32(s, v, r) __raw_writel(v, (void __iomem *)&s->regs->r)
106 #define MESSAGE_TYPE_SYNC 1
107 #define MESSAGE_TYPE_P_DELAY_REQ 2
108 #define MESSAGE_TYPE_P_DELAY_RESP 3
109 #define MESSAGE_TYPE_DELAY_REQ 4
111 static LIST_HEAD(ines_clocks
);
112 static DEFINE_MUTEX(ines_clocks_lock
);
114 struct ines_global_regs
{
125 struct ines_port_registers
{
134 struct ines_timestamp
{
135 struct list_head list
;
146 struct ines_port_registers
*regs
;
147 struct mii_timestamper mii_ts
;
148 struct ines_clock
*clock
;
152 struct delayed_work ts_work
;
153 /* lock protects event list and tx_skb */
155 struct sk_buff
*tx_skb
;
156 struct list_head events
;
157 struct list_head pool
;
158 struct ines_timestamp pool_data
[INES_MAX_EVENTS
];
162 struct ines_port port
[INES_N_PORTS
];
163 struct ines_global_regs __iomem
*regs
;
165 struct device_node
*node
;
167 struct list_head list
;
170 static bool ines_match(struct sk_buff
*skb
, unsigned int ptp_class
,
171 struct ines_timestamp
*ts
, struct device
*dev
);
172 static int ines_rxfifo_read(struct ines_port
*port
);
173 static u64
ines_rxts64(struct ines_port
*port
, unsigned int words
);
174 static bool ines_timestamp_expired(struct ines_timestamp
*ts
);
175 static u64
ines_txts64(struct ines_port
*port
, unsigned int words
);
176 static void ines_txtstamp_work(struct work_struct
*work
);
177 static bool is_sync_pdelay_resp(struct sk_buff
*skb
, int type
);
178 static u8
tag_to_msgtype(u8 tag
);
180 static void ines_clock_cleanup(struct ines_clock
*clock
)
182 struct ines_port
*port
;
185 for (i
= 0; i
< INES_N_PORTS
; i
++) {
186 port
= &clock
->port
[i
];
187 cancel_delayed_work_sync(&port
->ts_work
);
191 static int ines_clock_init(struct ines_clock
*clock
, struct device
*device
,
194 struct device_node
*node
= device
->of_node
;
195 unsigned long port_addr
;
196 struct ines_port
*port
;
199 INIT_LIST_HEAD(&clock
->list
);
203 clock
->regs
= clock
->base
;
205 for (i
= 0; i
< INES_N_PORTS
; i
++) {
206 port
= &clock
->port
[i
];
207 port_addr
= (unsigned long) clock
->base
+
208 INES_PORT_OFFSET
+ i
* INES_PORT_SIZE
;
209 port
->regs
= (struct ines_port_registers
*) port_addr
;
212 INIT_DELAYED_WORK(&port
->ts_work
, ines_txtstamp_work
);
213 spin_lock_init(&port
->lock
);
214 INIT_LIST_HEAD(&port
->events
);
215 INIT_LIST_HEAD(&port
->pool
);
216 for (j
= 0; j
< INES_MAX_EVENTS
; j
++)
217 list_add(&port
->pool_data
[j
].list
, &port
->pool
);
220 ines_write32(clock
, 0xBEEF, test
);
221 ines_write32(clock
, 0xBEEF, test2
);
223 dev_dbg(device
, "ID 0x%x\n", ines_read32(clock
, id
));
224 dev_dbg(device
, "TEST 0x%x\n", ines_read32(clock
, test
));
225 dev_dbg(device
, "VERSION 0x%x\n", ines_read32(clock
, version
));
226 dev_dbg(device
, "TEST2 0x%x\n", ines_read32(clock
, test2
));
228 for (i
= 0; i
< INES_N_PORTS
; i
++) {
229 port
= &clock
->port
[i
];
230 ines_write32(port
, PORT_CONF
, port_conf
);
236 static struct ines_port
*ines_find_port(struct device_node
*node
, u32 index
)
238 struct ines_port
*port
= NULL
;
239 struct ines_clock
*clock
;
240 struct list_head
*this;
242 mutex_lock(&ines_clocks_lock
);
243 list_for_each(this, &ines_clocks
) {
244 clock
= list_entry(this, struct ines_clock
, list
);
245 if (clock
->node
== node
) {
246 port
= &clock
->port
[index
];
250 mutex_unlock(&ines_clocks_lock
);
254 static u64
ines_find_rxts(struct ines_port
*port
, struct sk_buff
*skb
, int type
)
256 struct list_head
*this, *next
;
257 struct ines_timestamp
*ts
;
261 if (type
== PTP_CLASS_NONE
)
264 spin_lock_irqsave(&port
->lock
, flags
);
265 ines_rxfifo_read(port
);
266 list_for_each_safe(this, next
, &port
->events
) {
267 ts
= list_entry(this, struct ines_timestamp
, list
);
268 if (ines_timestamp_expired(ts
)) {
269 list_del_init(&ts
->list
);
270 list_add(&ts
->list
, &port
->pool
);
273 if (ines_match(skb
, type
, ts
, port
->clock
->dev
)) {
274 ns
= ts
->sec
* 1000000000ULL + ts
->nsec
;
275 list_del_init(&ts
->list
);
276 list_add(&ts
->list
, &port
->pool
);
280 spin_unlock_irqrestore(&port
->lock
, flags
);
285 static u64
ines_find_txts(struct ines_port
*port
, struct sk_buff
*skb
)
287 unsigned int class = ptp_classify_raw(skb
), i
;
288 u32 data_rd_pos
, buf_stat
, mask
, ts_stat_tx
;
289 struct ines_timestamp ts
;
293 mask
= TX_FIFO_NE_1
<< port
->index
;
295 spin_lock_irqsave(&port
->lock
, flags
);
297 for (i
= 0; i
< INES_FIFO_DEPTH
; i
++) {
299 buf_stat
= ines_read32(port
->clock
, buf_stat
);
300 if (!(buf_stat
& mask
)) {
301 dev_dbg(port
->clock
->dev
,
302 "Tx timestamp FIFO unexpectedly empty\n");
305 ts_stat_tx
= ines_read32(port
, ts_stat_tx
);
306 data_rd_pos
= (ts_stat_tx
>> DATA_READ_POS_SHIFT
) &
309 dev_err(port
->clock
->dev
,
310 "unexpected Tx read pos %u\n", data_rd_pos
);
314 ts
.tag
= ines_read32(port
, ts_tx
);
315 ts
.sec
= ines_txts64(port
, 3);
316 ts
.nsec
= ines_txts64(port
, 2);
317 ts
.clkid
= ines_txts64(port
, 4);
318 ts
.portnum
= ines_read32(port
, ts_tx
);
319 ts
.seqid
= ines_read32(port
, ts_tx
);
321 if (ines_match(skb
, class, &ts
, port
->clock
->dev
)) {
322 ns
= ts
.sec
* 1000000000ULL + ts
.nsec
;
327 spin_unlock_irqrestore(&port
->lock
, flags
);
331 static int ines_hwtstamp(struct mii_timestamper
*mii_ts
,
332 struct kernel_hwtstamp_config
*cfg
,
333 struct netlink_ext_ack
*extack
)
335 struct ines_port
*port
= container_of(mii_ts
, struct ines_port
, mii_ts
);
336 u32 cm_one_step
= 0, port_conf
, ts_stat_rx
, ts_stat_tx
;
339 switch (cfg
->tx_type
) {
340 case HWTSTAMP_TX_OFF
:
344 ts_stat_tx
= TS_ENABLE
;
346 case HWTSTAMP_TX_ONESTEP_P2P
:
347 ts_stat_tx
= TS_ENABLE
;
348 cm_one_step
= CM_ONE_STEP
;
354 switch (cfg
->rx_filter
) {
355 case HWTSTAMP_FILTER_NONE
:
358 case HWTSTAMP_FILTER_ALL
:
359 case HWTSTAMP_FILTER_PTP_V1_L4_EVENT
:
360 case HWTSTAMP_FILTER_PTP_V1_L4_SYNC
:
361 case HWTSTAMP_FILTER_PTP_V1_L4_DELAY_REQ
:
363 case HWTSTAMP_FILTER_PTP_V2_L4_EVENT
:
364 case HWTSTAMP_FILTER_PTP_V2_L4_SYNC
:
365 case HWTSTAMP_FILTER_PTP_V2_L4_DELAY_REQ
:
366 case HWTSTAMP_FILTER_PTP_V2_L2_EVENT
:
367 case HWTSTAMP_FILTER_PTP_V2_L2_SYNC
:
368 case HWTSTAMP_FILTER_PTP_V2_L2_DELAY_REQ
:
369 case HWTSTAMP_FILTER_PTP_V2_EVENT
:
370 case HWTSTAMP_FILTER_PTP_V2_SYNC
:
371 case HWTSTAMP_FILTER_PTP_V2_DELAY_REQ
:
372 ts_stat_rx
= TS_ENABLE
;
373 cfg
->rx_filter
= HWTSTAMP_FILTER_PTP_V2_EVENT
;
379 spin_lock_irqsave(&port
->lock
, flags
);
381 port_conf
= ines_read32(port
, port_conf
);
382 port_conf
&= ~CM_ONE_STEP
;
383 port_conf
|= cm_one_step
;
385 ines_write32(port
, port_conf
, port_conf
);
386 ines_write32(port
, ts_stat_rx
, ts_stat_rx
);
387 ines_write32(port
, ts_stat_tx
, ts_stat_tx
);
389 port
->rxts_enabled
= ts_stat_rx
== TS_ENABLE
;
390 port
->txts_enabled
= ts_stat_tx
== TS_ENABLE
;
392 spin_unlock_irqrestore(&port
->lock
, flags
);
397 static void ines_link_state(struct mii_timestamper
*mii_ts
,
398 struct phy_device
*phydev
)
400 struct ines_port
*port
= container_of(mii_ts
, struct ines_port
, mii_ts
);
401 u32 port_conf
, speed_conf
;
404 switch (phydev
->speed
) {
406 speed_conf
= PHY_SPEED_10
<< PHY_SPEED_SHIFT
;
409 speed_conf
= PHY_SPEED_100
<< PHY_SPEED_SHIFT
;
412 speed_conf
= PHY_SPEED_1000
<< PHY_SPEED_SHIFT
;
415 dev_err(port
->clock
->dev
, "bad speed: %d\n", phydev
->speed
);
418 spin_lock_irqsave(&port
->lock
, flags
);
420 port_conf
= ines_read32(port
, port_conf
);
421 port_conf
&= ~(0x3 << PHY_SPEED_SHIFT
);
422 port_conf
|= speed_conf
;
424 ines_write32(port
, port_conf
, port_conf
);
426 spin_unlock_irqrestore(&port
->lock
, flags
);
429 static bool ines_match(struct sk_buff
*skb
, unsigned int ptp_class
,
430 struct ines_timestamp
*ts
, struct device
*dev
)
432 struct ptp_header
*hdr
;
437 if (unlikely(ptp_class
& PTP_CLASS_V1
))
440 hdr
= ptp_parse_header(skb
, ptp_class
);
444 msgtype
= ptp_get_msgtype(hdr
, ptp_class
);
445 clkid
= be64_to_cpup((__be64
*)&hdr
->source_port_identity
.clock_identity
.id
[0]);
446 portn
= be16_to_cpu(hdr
->source_port_identity
.port_number
);
447 seqid
= be16_to_cpu(hdr
->sequence_id
);
449 if (tag_to_msgtype(ts
->tag
& 0x7) != msgtype
) {
450 dev_dbg(dev
, "msgtype mismatch ts %hhu != skb %hhu\n",
451 tag_to_msgtype(ts
->tag
& 0x7), msgtype
);
454 if (ts
->clkid
!= clkid
) {
455 dev_dbg(dev
, "clkid mismatch ts %llx != skb %llx\n",
459 if (ts
->portnum
!= portn
) {
460 dev_dbg(dev
, "portn mismatch ts %hu != skb %hu\n",
464 if (ts
->seqid
!= seqid
) {
465 dev_dbg(dev
, "seqid mismatch ts %hu != skb %hu\n",
473 static bool ines_rxtstamp(struct mii_timestamper
*mii_ts
,
474 struct sk_buff
*skb
, int type
)
476 struct ines_port
*port
= container_of(mii_ts
, struct ines_port
, mii_ts
);
477 struct skb_shared_hwtstamps
*ssh
;
480 if (!port
->rxts_enabled
)
483 ns
= ines_find_rxts(port
, skb
, type
);
487 ssh
= skb_hwtstamps(skb
);
488 ssh
->hwtstamp
= ns_to_ktime(ns
);
494 static int ines_rxfifo_read(struct ines_port
*port
)
496 u32 data_rd_pos
, buf_stat
, mask
, ts_stat_rx
;
497 struct ines_timestamp
*ts
;
500 mask
= RX_FIFO_NE_1
<< port
->index
;
502 for (i
= 0; i
< INES_FIFO_DEPTH
; i
++) {
503 if (list_empty(&port
->pool
)) {
504 dev_err(port
->clock
->dev
, "event pool is empty\n");
507 buf_stat
= ines_read32(port
->clock
, buf_stat
);
508 if (!(buf_stat
& mask
))
511 ts_stat_rx
= ines_read32(port
, ts_stat_rx
);
512 data_rd_pos
= (ts_stat_rx
>> DATA_READ_POS_SHIFT
) &
515 dev_err(port
->clock
->dev
, "unexpected Rx read pos %u\n",
520 ts
= list_first_entry(&port
->pool
, struct ines_timestamp
, list
);
521 ts
->tmo
= jiffies
+ HZ
;
522 ts
->tag
= ines_read32(port
, ts_rx
);
523 ts
->sec
= ines_rxts64(port
, 3);
524 ts
->nsec
= ines_rxts64(port
, 2);
525 ts
->clkid
= ines_rxts64(port
, 4);
526 ts
->portnum
= ines_read32(port
, ts_rx
);
527 ts
->seqid
= ines_read32(port
, ts_rx
);
529 list_del_init(&ts
->list
);
530 list_add_tail(&ts
->list
, &port
->events
);
536 static u64
ines_rxts64(struct ines_port
*port
, unsigned int words
)
542 word
= ines_read32(port
, ts_rx
);
545 for (i
= 0; i
< words
; i
++) {
546 word
= ines_read32(port
, ts_rx
);
553 static bool ines_timestamp_expired(struct ines_timestamp
*ts
)
555 return time_after(jiffies
, ts
->tmo
);
558 static int ines_ts_info(struct mii_timestamper
*mii_ts
,
559 struct kernel_ethtool_ts_info
*info
)
561 info
->so_timestamping
=
562 SOF_TIMESTAMPING_TX_HARDWARE
|
563 SOF_TIMESTAMPING_TX_SOFTWARE
|
564 SOF_TIMESTAMPING_RX_HARDWARE
|
565 SOF_TIMESTAMPING_RAW_HARDWARE
;
568 (1 << HWTSTAMP_TX_OFF
) |
569 (1 << HWTSTAMP_TX_ON
) |
570 (1 << HWTSTAMP_TX_ONESTEP_P2P
);
573 (1 << HWTSTAMP_FILTER_NONE
) |
574 (1 << HWTSTAMP_FILTER_PTP_V2_EVENT
);
579 static u64
ines_txts64(struct ines_port
*port
, unsigned int words
)
585 word
= ines_read32(port
, ts_tx
);
588 for (i
= 0; i
< words
; i
++) {
589 word
= ines_read32(port
, ts_tx
);
596 static bool ines_txts_onestep(struct ines_port
*port
, struct sk_buff
*skb
, int type
)
601 spin_lock_irqsave(&port
->lock
, flags
);
602 port_conf
= ines_read32(port
, port_conf
);
603 spin_unlock_irqrestore(&port
->lock
, flags
);
605 if (port_conf
& CM_ONE_STEP
)
606 return is_sync_pdelay_resp(skb
, type
);
611 static void ines_txtstamp(struct mii_timestamper
*mii_ts
,
612 struct sk_buff
*skb
, int type
)
614 struct ines_port
*port
= container_of(mii_ts
, struct ines_port
, mii_ts
);
615 struct sk_buff
*old_skb
= NULL
;
618 if (!port
->txts_enabled
|| ines_txts_onestep(port
, skb
, type
)) {
623 spin_lock_irqsave(&port
->lock
, flags
);
626 old_skb
= port
->tx_skb
;
630 spin_unlock_irqrestore(&port
->lock
, flags
);
634 schedule_delayed_work(&port
->ts_work
, 1);
637 static void ines_txtstamp_work(struct work_struct
*work
)
639 struct ines_port
*port
=
640 container_of(work
, struct ines_port
, ts_work
.work
);
641 struct skb_shared_hwtstamps ssh
;
646 spin_lock_irqsave(&port
->lock
, flags
);
649 spin_unlock_irqrestore(&port
->lock
, flags
);
651 ns
= ines_find_txts(port
, skb
);
656 ssh
.hwtstamp
= ns_to_ktime(ns
);
657 skb_complete_tx_timestamp(skb
, &ssh
);
660 static bool is_sync_pdelay_resp(struct sk_buff
*skb
, int type
)
662 struct ptp_header
*hdr
;
665 hdr
= ptp_parse_header(skb
, type
);
669 msgtype
= ptp_get_msgtype(hdr
, type
);
672 case PTP_MSGTYPE_SYNC
:
673 case PTP_MSGTYPE_PDELAY_RESP
:
680 static u8
tag_to_msgtype(u8 tag
)
683 case MESSAGE_TYPE_SYNC
:
684 return PTP_MSGTYPE_SYNC
;
685 case MESSAGE_TYPE_P_DELAY_REQ
:
686 return PTP_MSGTYPE_PDELAY_REQ
;
687 case MESSAGE_TYPE_P_DELAY_RESP
:
688 return PTP_MSGTYPE_PDELAY_RESP
;
689 case MESSAGE_TYPE_DELAY_REQ
:
690 return PTP_MSGTYPE_DELAY_REQ
;
695 static struct mii_timestamper
*ines_ptp_probe_channel(struct device
*device
,
698 struct device_node
*node
= device
->of_node
;
699 struct ines_port
*port
;
701 if (index
> INES_N_PORTS
- 1) {
702 dev_err(device
, "bad port index %u\n", index
);
703 return ERR_PTR(-EINVAL
);
705 port
= ines_find_port(node
, index
);
707 dev_err(device
, "missing port index %u\n", index
);
708 return ERR_PTR(-ENODEV
);
710 port
->mii_ts
.rxtstamp
= ines_rxtstamp
;
711 port
->mii_ts
.txtstamp
= ines_txtstamp
;
712 port
->mii_ts
.hwtstamp
= ines_hwtstamp
;
713 port
->mii_ts
.link_state
= ines_link_state
;
714 port
->mii_ts
.ts_info
= ines_ts_info
;
716 return &port
->mii_ts
;
719 static void ines_ptp_release_channel(struct device
*device
,
720 struct mii_timestamper
*mii_ts
)
724 static struct mii_timestamping_ctrl ines_ctrl
= {
725 .probe_channel
= ines_ptp_probe_channel
,
726 .release_channel
= ines_ptp_release_channel
,
729 static int ines_ptp_ctrl_probe(struct platform_device
*pld
)
731 struct ines_clock
*clock
;
735 addr
= devm_platform_ioremap_resource(pld
, 0);
740 clock
= kzalloc(sizeof(*clock
), GFP_KERNEL
);
745 if (ines_clock_init(clock
, &pld
->dev
, addr
)) {
750 err
= register_mii_tstamp_controller(&pld
->dev
, &ines_ctrl
);
755 mutex_lock(&ines_clocks_lock
);
756 list_add_tail(&ines_clocks
, &clock
->list
);
757 mutex_unlock(&ines_clocks_lock
);
759 dev_set_drvdata(&pld
->dev
, clock
);
764 static void ines_ptp_ctrl_remove(struct platform_device
*pld
)
766 struct ines_clock
*clock
= dev_get_drvdata(&pld
->dev
);
768 unregister_mii_tstamp_controller(&pld
->dev
);
769 mutex_lock(&ines_clocks_lock
);
770 list_del(&clock
->list
);
771 mutex_unlock(&ines_clocks_lock
);
772 ines_clock_cleanup(clock
);
776 static const struct of_device_id ines_ptp_ctrl_of_match
[] = {
777 { .compatible
= "ines,ptp-ctrl" },
781 MODULE_DEVICE_TABLE(of
, ines_ptp_ctrl_of_match
);
783 static struct platform_driver ines_ptp_ctrl_driver
= {
784 .probe
= ines_ptp_ctrl_probe
,
785 .remove
= ines_ptp_ctrl_remove
,
787 .name
= "ines_ptp_ctrl",
788 .of_match_table
= ines_ptp_ctrl_of_match
,
791 module_platform_driver(ines_ptp_ctrl_driver
);