Skip to content

Commit 9777500

Browse files
nhormandavem330
authored andcommitted
af_packet: add interframe drop cmsg (v6)
Add Ancilliary data to better represent loss information I've had a few requests recently to provide more detail regarding frame loss during an AF_PACKET packet capture session. Specifically the requestors want to see where in a packet sequence frames were lost, i.e. they want to see that 40 frames were lost between frames 302 and 303 in a packet capture file. In order to do this we need: 1) The kernel to export this data to user space 2) The applications to make use of it This patch addresses item (1). It does this by doing the following: A) Anytime we drop a frame for which we would increment po->stats.tp_drops, we also no increment a stats called po->stats.tp_gap. B) Every time we successfully enqueue a frame to sk_receive_queue, we record the value of po->stats.tp_gap in skb->mark. skb->cb would nominally be the place to record this, but since all the space there is used up, we're overloading skb->mark. Its safe to do since any enqueued packet is guaranteed to be unshared at this point, and skb->mark isn't used for anything else in the rx path to the application. After we record tp_gap in the skb, we zero po->stats.tp_gap. This allows us to keep a counter of the number of frames lost between any two enqueued packets C) When the application goes to dequeue a frame from the packet socket, we look at skb->mark for that frame. If it is non-zero, we add a cmsg chunk to the msghdr of level SOL_PACKET and type PACKET_GAPDATA. Its a 32 bit integer that represents the number of frames lost between this packet and the last previous frame received. Note there is a chance that if there is frame loss after a receive, and then the socket is closed, some gap data might be lost. This is covered by the use of the PACKET_AUXDATA socket option, which gives total loss data. With a bit of math, the final gap can be determined that way. I've tested this patch myself, and it works well. Signed-off-by: Neil Horman <[email protected]> Signed-off-by: Eric Dumazet <[email protected]> include/linux/if_packet.h | 2 ++ net/packet/af_packet.c | 33 +++++++++++++++++++++++++++++++++ 2 files changed, 35 insertions(+) Signed-off-by: David S. Miller <[email protected]>
1 parent 69ef969 commit 9777500

File tree

2 files changed

+35
-0
lines changed

2 files changed

+35
-0
lines changed

include/linux/if_packet.h

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -48,11 +48,13 @@ struct sockaddr_ll
4848
#define PACKET_RESERVE 12
4949
#define PACKET_TX_RING 13
5050
#define PACKET_LOSS 14
51+
#define PACKET_GAPDATA 15
5152

5253
struct tpacket_stats
5354
{
5455
unsigned int tp_packets;
5556
unsigned int tp_drops;
57+
unsigned int tp_gap;
5658
};
5759

5860
struct tpacket_auxdata

net/packet/af_packet.c

Lines changed: 33 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -523,6 +523,31 @@ static inline unsigned int run_filter(struct sk_buff *skb, struct sock *sk,
523523
return res;
524524
}
525525

526+
/*
527+
* If we've lost frames since the last time we queued one to the
528+
* sk_receive_queue, we need to record it here.
529+
* This must be called under the protection of the socket lock
530+
* to prevent racing with other softirqs and user space
531+
*/
532+
static inline void record_packet_gap(struct sk_buff *skb,
533+
struct packet_sock *po)
534+
{
535+
/*
536+
* We overload the mark field here, since we're about
537+
* to enqueue to a receive queue and no body else will
538+
* use this field at this point
539+
*/
540+
skb->mark = po->stats.tp_gap;
541+
po->stats.tp_gap = 0;
542+
return;
543+
544+
}
545+
546+
static inline __u32 check_packet_gap(struct sk_buff *skb)
547+
{
548+
return skb->mark;
549+
}
550+
526551
/*
527552
This function makes lazy skb cloning in hope that most of packets
528553
are discarded by BPF.
@@ -626,6 +651,7 @@ static int packet_rcv(struct sk_buff *skb, struct net_device *dev,
626651

627652
spin_lock(&sk->sk_receive_queue.lock);
628653
po->stats.tp_packets++;
654+
record_packet_gap(skb, po);
629655
__skb_queue_tail(&sk->sk_receive_queue, skb);
630656
spin_unlock(&sk->sk_receive_queue.lock);
631657
sk->sk_data_ready(sk, skb->len);
@@ -634,6 +660,7 @@ static int packet_rcv(struct sk_buff *skb, struct net_device *dev,
634660
drop_n_acct:
635661
spin_lock(&sk->sk_receive_queue.lock);
636662
po->stats.tp_drops++;
663+
po->stats.tp_gap++;
637664
spin_unlock(&sk->sk_receive_queue.lock);
638665

639666
drop_n_restore:
@@ -811,6 +838,7 @@ static int tpacket_rcv(struct sk_buff *skb, struct net_device *dev,
811838

812839
ring_is_full:
813840
po->stats.tp_drops++;
841+
po->stats.tp_gap++;
814842
spin_unlock(&sk->sk_receive_queue.lock);
815843

816844
sk->sk_data_ready(sk, 0);
@@ -1418,6 +1446,7 @@ static int packet_recvmsg(struct kiocb *iocb, struct socket *sock,
14181446
struct sk_buff *skb;
14191447
int copied, err;
14201448
struct sockaddr_ll *sll;
1449+
__u32 gap;
14211450

14221451
err = -EINVAL;
14231452
if (flags & ~(MSG_PEEK|MSG_DONTWAIT|MSG_TRUNC|MSG_CMSG_COMPAT))
@@ -1496,6 +1525,10 @@ static int packet_recvmsg(struct kiocb *iocb, struct socket *sock,
14961525
put_cmsg(msg, SOL_PACKET, PACKET_AUXDATA, sizeof(aux), &aux);
14971526
}
14981527

1528+
gap = check_packet_gap(skb);
1529+
if (gap)
1530+
put_cmsg(msg, SOL_PACKET, PACKET_GAPDATA, sizeof(__u32), &gap);
1531+
14991532
/*
15001533
* Free or return the buffer as appropriate. Again this
15011534
* hides all the races and re-entrancy issues from us.

0 commit comments

Comments
 (0)