sfc: support unicast PTP
authorÍñigo Huguet <ihuguet@redhat.com>
Fri, 31 Mar 2023 11:14:03 +0000 (13:14 +0200)
committerJakub Kicinski <kuba@kernel.org>
Tue, 4 Apr 2023 02:02:51 +0000 (19:02 -0700)
When sending a PTP event packet, add the correct filters that will make
that future incoming unicast PTP event packets will be timestamped.
The unicast address for the filter is gotten from the outgoing skb
before sending it.

Until now they were not timestamped because only filters that match with
the PTP multicast addressed were being configured into the NIC for the
PTP special channel. Packets received through different channels are not
timestamped, getting "received SYNC without timestamp" error in ptp4l.

Note that the inserted filters are never removed unless the NIC is stopped
or reconfigured, so efx_ptp_stop is called. Removal of old filters will
be handled by the next patch.

Additionally, cleanup a bit efx_ptp_xmit_skb_mc to use the reverse xmas
tree convention and remove an unnecessary assignment to rc variable in
void function.

Reported-by: Yalin Li <yalli@redhat.com>
Signed-off-by: Íñigo Huguet <ihuguet@redhat.com>
Reviewed-by: Edward Cree <ecree.xilinx@gmail.com>
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
drivers/net/ethernet/sfc/ptp.c

index d5631e02e626f851c4d0f1a1b53b1ed469fb6768..b495f3d0616b5b46337e07a3edb28d0cd983aafc 100644 (file)
@@ -215,10 +215,16 @@ struct efx_ptp_timeset {
 /**
  * struct efx_ptp_rxfilter - Filter for PTP packets
  * @list: Node of the list where the filter is added
+ * @ether_type: Network protocol of the filter (ETHER_P_IP / ETHER_P_IPV6)
+ * @loc_port: UDP port of the filter (PTP_EVENT_PORT / PTP_GENERAL_PORT)
+ * @loc_host: IPv4/v6 address of the filter
  * @handle: Handle ID for the MCDI filters table
  */
 struct efx_ptp_rxfilter {
        struct list_head list;
+       __be16 ether_type;
+       __be16 loc_port;
+       __be32 loc_host[4];
        int handle;
 };
 
@@ -367,6 +373,8 @@ static int efx_phc_settime(struct ptp_clock_info *ptp,
                           const struct timespec64 *e_ts);
 static int efx_phc_enable(struct ptp_clock_info *ptp,
                          struct ptp_clock_request *request, int on);
+static int efx_ptp_insert_unicast_filter(struct efx_nic *efx,
+                                        struct sk_buff *skb);
 
 bool efx_ptp_use_mac_tx_timestamps(struct efx_nic *efx)
 {
@@ -1112,6 +1120,8 @@ static void efx_ptp_xmit_skb_queue(struct efx_nic *efx, struct sk_buff *skb)
 
        tx_queue = efx_channel_get_tx_queue(ptp_data->channel, type);
        if (tx_queue && tx_queue->timestamping) {
+               skb_get(skb);
+
                /* This code invokes normal driver TX code which is always
                 * protected from softirqs when called from generic TX code,
                 * which in turn disables preemption. Look at __dev_queue_xmit
@@ -1135,6 +1145,13 @@ static void efx_ptp_xmit_skb_queue(struct efx_nic *efx, struct sk_buff *skb)
                local_bh_disable();
                efx_enqueue_skb(tx_queue, skb);
                local_bh_enable();
+
+               /* We need to add the filters after enqueuing the packet.
+                * Otherwise, there's high latency in sending back the
+                * timestamp, causing ptp4l timeouts
+                */
+               efx_ptp_insert_unicast_filter(efx, skb);
+               dev_consume_skb_any(skb);
        } else {
                WARN_ONCE(1, "PTP channel has no timestamped tx queue\n");
                dev_kfree_skb_any(skb);
@@ -1144,11 +1161,11 @@ static void efx_ptp_xmit_skb_queue(struct efx_nic *efx, struct sk_buff *skb)
 /* Transmit a PTP packet, via the MCDI interface, to the wire. */
 static void efx_ptp_xmit_skb_mc(struct efx_nic *efx, struct sk_buff *skb)
 {
+       MCDI_DECLARE_BUF(txtime, MC_CMD_PTP_OUT_TRANSMIT_LEN);
        struct efx_ptp_data *ptp_data = efx->ptp_data;
        struct skb_shared_hwtstamps timestamps;
-       int rc = -EIO;
-       MCDI_DECLARE_BUF(txtime, MC_CMD_PTP_OUT_TRANSMIT_LEN);
        size_t len;
+       int rc;
 
        MCDI_SET_DWORD(ptp_data->txbuf, PTP_IN_OP, MC_CMD_PTP_OP_TRANSMIT);
        MCDI_SET_DWORD(ptp_data->txbuf, PTP_IN_PERIPH_ID, 0);
@@ -1182,7 +1199,10 @@ static void efx_ptp_xmit_skb_mc(struct efx_nic *efx, struct sk_buff *skb)
 
        skb_tstamp_tx(skb, &timestamps);
 
-       rc = 0;
+       /* Add the filters after sending back the timestamp to avoid delaying it
+        * or ptp4l may timeout.
+        */
+       efx_ptp_insert_unicast_filter(efx, skb);
 
 fail:
        dev_kfree_skb_any(skb);
@@ -1298,6 +1318,21 @@ static inline void efx_ptp_process_rx(struct efx_nic *efx, struct sk_buff *skb)
        local_bh_enable();
 }
 
+static bool efx_ptp_filter_exists(struct list_head *filter_list,
+                                 struct efx_filter_spec *spec)
+{
+       struct efx_ptp_rxfilter *rxfilter;
+
+       list_for_each_entry(rxfilter, filter_list, list) {
+               if (rxfilter->ether_type == spec->ether_type &&
+                   rxfilter->loc_port == spec->loc_port &&
+                   !memcmp(rxfilter->loc_host, spec->loc_host, sizeof(spec->loc_host)))
+                       return true;
+       }
+
+       return false;
+}
+
 static void efx_ptp_remove_filters(struct efx_nic *efx,
                                   struct list_head *filter_list)
 {
@@ -1328,6 +1363,9 @@ static int efx_ptp_insert_filter(struct efx_nic *efx,
        struct efx_ptp_rxfilter *rxfilter;
        int rc;
 
+       if (efx_ptp_filter_exists(filter_list, spec))
+               return 0;
+
        rxfilter = kzalloc(sizeof(*rxfilter), GFP_KERNEL);
        if (!rxfilter)
                return -ENOMEM;
@@ -1337,6 +1375,9 @@ static int efx_ptp_insert_filter(struct efx_nic *efx,
                goto fail;
 
        rxfilter->handle = rc;
+       rxfilter->ether_type = spec->ether_type;
+       rxfilter->loc_port = spec->loc_port;
+       memcpy(rxfilter->loc_host, spec->loc_host, sizeof(spec->loc_host));
        list_add(&rxfilter->list, filter_list);
 
        return 0;
@@ -1430,6 +1471,60 @@ fail:
        return rc;
 }
 
+static bool efx_ptp_valid_unicast_event_pkt(struct sk_buff *skb)
+{
+       if (skb->protocol == htons(ETH_P_IP)) {
+               return ip_hdr(skb)->daddr != htonl(PTP_ADDR_IPV4) &&
+                       ip_hdr(skb)->protocol == IPPROTO_UDP &&
+                       udp_hdr(skb)->source == htons(PTP_EVENT_PORT);
+       } else if (skb->protocol == htons(ETH_P_IPV6)) {
+               struct in6_addr mcast_addr = {{PTP_ADDR_IPV6}};
+
+               return !ipv6_addr_equal(&ipv6_hdr(skb)->daddr, &mcast_addr) &&
+                       ipv6_hdr(skb)->nexthdr == IPPROTO_UDP &&
+                       udp_hdr(skb)->source == htons(PTP_EVENT_PORT);
+       }
+       return false;
+}
+
+static int efx_ptp_insert_unicast_filter(struct efx_nic *efx,
+                                        struct sk_buff *skb)
+{
+       struct efx_ptp_data *ptp = efx->ptp_data;
+       int rc;
+
+       if (!efx_ptp_valid_unicast_event_pkt(skb))
+               return -EINVAL;
+
+       if (skb->protocol == htons(ETH_P_IP)) {
+               __be32 addr = ip_hdr(skb)->saddr;
+
+               rc = efx_ptp_insert_ipv4_filter(efx, &ptp->rxfilters_ucast,
+                                               addr, PTP_EVENT_PORT);
+               if (rc < 0)
+                       goto out;
+
+               rc = efx_ptp_insert_ipv4_filter(efx, &ptp->rxfilters_ucast,
+                                               addr, PTP_GENERAL_PORT);
+       } else if (efx_ptp_use_mac_tx_timestamps(efx)) {
+               /* IPv6 PTP only supported by devices with MAC hw timestamp */
+               struct in6_addr *addr = &ipv6_hdr(skb)->saddr;
+
+               rc = efx_ptp_insert_ipv6_filter(efx, &ptp->rxfilters_ucast,
+                                               addr, PTP_EVENT_PORT);
+               if (rc < 0)
+                       goto out;
+
+               rc = efx_ptp_insert_ipv6_filter(efx, &ptp->rxfilters_ucast,
+                                               addr, PTP_GENERAL_PORT);
+       } else {
+               return -EOPNOTSUPP;
+       }
+
+out:
+       return rc;
+}
+
 static int efx_ptp_start(struct efx_nic *efx)
 {
        struct efx_ptp_data *ptp = efx->ptp_data;