Commit aef37619 authored by Sebastian Andrzej Siewior's avatar Sebastian Andrzej Siewior Committed by Greg Kroah-Hartman

net: dp83640: expire old TX-skb

[ Upstream commit 53bc8d2a ]

During sendmsg() a cloned skb is saved via dp83640_txtstamp() in
->tx_queue. After the NIC sends this packet, the PHY will reply with a
timestamp for that TX packet. If the cable is pulled at the right time I
don't see that packet. It might gets flushed as part of queue shutdown
on NIC's side.
Once the link is up again then after the next sendmsg() we enqueue
another skb in dp83640_txtstamp() and have two on the list. Then the PHY
will send a reply and decode_txts() attaches it to the first skb on the
list.
No crash occurs since refcounting works but we are one packet behind.
linuxptp/ptp4l usually closes the socket and opens a new one (in such a
timeout case) so those "stale" replies never get there. However it does
not resume normal operation anymore.

Purge old skbs in decode_txts().

Fixes: cb646e2b ("ptp: Added a clock driver for the National Semiconductor PHYTER.")
Signed-off-by: default avatarSebastian Andrzej Siewior <bigeasy@linutronix.de>
Reviewed-by: Kurt Kanzenbach's avatarKurt Kanzenbach <kurt@linutronix.de>
Acked-by: default avatarRichard Cochran <richardcochran@gmail.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 25805f7e
...@@ -893,14 +893,14 @@ static void decode_txts(struct dp83640_private *dp83640, ...@@ -893,14 +893,14 @@ static void decode_txts(struct dp83640_private *dp83640,
struct phy_txts *phy_txts) struct phy_txts *phy_txts)
{ {
struct skb_shared_hwtstamps shhwtstamps; struct skb_shared_hwtstamps shhwtstamps;
struct dp83640_skb_info *skb_info;
struct sk_buff *skb; struct sk_buff *skb;
u64 ns;
u8 overflow; u8 overflow;
u64 ns;
/* We must already have the skb that triggered this. */ /* We must already have the skb that triggered this. */
again:
skb = skb_dequeue(&dp83640->tx_queue); skb = skb_dequeue(&dp83640->tx_queue);
if (!skb) { if (!skb) {
pr_debug("have timestamp but tx_queue empty\n"); pr_debug("have timestamp but tx_queue empty\n");
return; return;
...@@ -915,6 +915,11 @@ static void decode_txts(struct dp83640_private *dp83640, ...@@ -915,6 +915,11 @@ static void decode_txts(struct dp83640_private *dp83640,
} }
return; return;
} }
skb_info = (struct dp83640_skb_info *)skb->cb;
if (time_after(jiffies, skb_info->tmo)) {
kfree_skb(skb);
goto again;
}
ns = phy2txts(phy_txts); ns = phy2txts(phy_txts);
memset(&shhwtstamps, 0, sizeof(shhwtstamps)); memset(&shhwtstamps, 0, sizeof(shhwtstamps));
...@@ -1466,6 +1471,7 @@ static bool dp83640_rxtstamp(struct phy_device *phydev, ...@@ -1466,6 +1471,7 @@ static bool dp83640_rxtstamp(struct phy_device *phydev,
static void dp83640_txtstamp(struct phy_device *phydev, static void dp83640_txtstamp(struct phy_device *phydev,
struct sk_buff *skb, int type) struct sk_buff *skb, int type)
{ {
struct dp83640_skb_info *skb_info = (struct dp83640_skb_info *)skb->cb;
struct dp83640_private *dp83640 = phydev->priv; struct dp83640_private *dp83640 = phydev->priv;
switch (dp83640->hwts_tx_en) { switch (dp83640->hwts_tx_en) {
...@@ -1478,6 +1484,7 @@ static void dp83640_txtstamp(struct phy_device *phydev, ...@@ -1478,6 +1484,7 @@ static void dp83640_txtstamp(struct phy_device *phydev,
/* fall through */ /* fall through */
case HWTSTAMP_TX_ON: case HWTSTAMP_TX_ON:
skb_shinfo(skb)->tx_flags |= SKBTX_IN_PROGRESS; skb_shinfo(skb)->tx_flags |= SKBTX_IN_PROGRESS;
skb_info->tmo = jiffies + SKB_TIMESTAMP_TIMEOUT;
skb_queue_tail(&dp83640->tx_queue, skb); skb_queue_tail(&dp83640->tx_queue, skb);
break; break;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment