aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorWillem de Bruijn <willemb@google.com>2014-08-04 22:11:45 -0400
committerDavid S. Miller <davem@davemloft.net>2014-08-05 19:35:53 -0400
commitf24b9be5957b38bb420b838115040dc2031b7d0c (patch)
tree9fdb50bbe86f08ff9c012137fa854d45e156b8f9
parenta2b81b35f9e5ade210e4df2001f7a30ac390114d (diff)
net-timestamp: extend SCM_TIMESTAMPING ancillary data struct
Applications that request kernel tx timestamps with SO_TIMESTAMPING read timestamps as recvmsg() ancillary data. The response is defined implicitly as timespec[3]. 1) define struct scm_timestamping explicitly and 2) add support for new tstamp types. On tx, scm_timestamping always accompanies a sock_extended_err. Define previously unused field ee_info to signal the type of ts[0]. Introduce SCM_TSTAMP_SND to define the existing behavior. The reception path is not modified. On rx, no struct similar to sock_extended_err is passed along with SCM_TIMESTAMPING. Signed-off-by: Willem de Bruijn <willemb@google.com> Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r--include/linux/skbuff.h3
-rw-r--r--include/net/sock.h4
-rw-r--r--include/uapi/linux/errqueue.h18
-rw-r--r--net/core/skbuff.c1
-rw-r--r--net/socket.c20
5 files changed, 36 insertions, 10 deletions
diff --git a/include/linux/skbuff.h b/include/linux/skbuff.h
index 281deced7469..477f0f60db45 100644
--- a/include/linux/skbuff.h
+++ b/include/linux/skbuff.h
@@ -249,6 +249,9 @@ enum {
249 SKBTX_SHARED_FRAG = 1 << 5, 249 SKBTX_SHARED_FRAG = 1 << 5,
250}; 250};
251 251
252#define SKBTX_ANY_SW_TSTAMP SKBTX_SW_TSTAMP
253#define SKBTX_ANY_TSTAMP (SKBTX_HW_TSTAMP | SKBTX_ANY_SW_TSTAMP)
254
252/* 255/*
253 * The callback notifies userspace to release buffers when skb DMA is done in 256 * The callback notifies userspace to release buffers when skb DMA is done in
254 * lower device, the skb last reference should be 0 when calling this. 257 * lower device, the skb last reference should be 0 when calling this.
diff --git a/include/net/sock.h b/include/net/sock.h
index b91c8868ab8d..02f5b35e65f1 100644
--- a/include/net/sock.h
+++ b/include/net/sock.h
@@ -2169,7 +2169,9 @@ sock_recv_timestamp(struct msghdr *msg, struct sock *sk, struct sk_buff *skb)
2169 */ 2169 */
2170 if (sock_flag(sk, SOCK_RCVTSTAMP) || 2170 if (sock_flag(sk, SOCK_RCVTSTAMP) ||
2171 sock_flag(sk, SOCK_TIMESTAMPING_RX_SOFTWARE) || 2171 sock_flag(sk, SOCK_TIMESTAMPING_RX_SOFTWARE) ||
2172 (kt.tv64 && sock_flag(sk, SOCK_TIMESTAMPING_SOFTWARE)) || 2172 (kt.tv64 &&
2173 (sock_flag(sk, SOCK_TIMESTAMPING_SOFTWARE) ||
2174 skb_shinfo(skb)->tx_flags & SKBTX_ANY_SW_TSTAMP)) ||
2173 (hwtstamps->hwtstamp.tv64 && 2175 (hwtstamps->hwtstamp.tv64 &&
2174 sock_flag(sk, SOCK_TIMESTAMPING_RAW_HARDWARE))) 2176 sock_flag(sk, SOCK_TIMESTAMPING_RAW_HARDWARE)))
2175 __sock_recv_timestamp(msg, sk, skb); 2177 __sock_recv_timestamp(msg, sk, skb);
diff --git a/include/uapi/linux/errqueue.h b/include/uapi/linux/errqueue.h
index aacd4fb7102a..accee72cae7c 100644
--- a/include/uapi/linux/errqueue.h
+++ b/include/uapi/linux/errqueue.h
@@ -22,5 +22,23 @@ struct sock_extended_err {
22 22
23#define SO_EE_OFFENDER(ee) ((struct sockaddr*)((ee)+1)) 23#define SO_EE_OFFENDER(ee) ((struct sockaddr*)((ee)+1))
24 24
25/**
26 * struct scm_timestamping - timestamps exposed through cmsg
27 *
28 * The timestamping interfaces SO_TIMESTAMPING, MSG_TSTAMP_*
29 * communicate network timestamps by passing this struct in a cmsg with
30 * recvmsg(). See Documentation/networking/timestamping.txt for details.
31 */
32struct scm_timestamping {
33 struct timespec ts[3];
34};
35
36/* The type of scm_timestamping, passed in sock_extended_err ee_info.
37 * This defines the type of ts[0]. For SCM_TSTAMP_SND only, if ts[0]
38 * is zero, then this is a hardware timestamp and recorded in ts[2].
39 */
40enum {
41 SCM_TSTAMP_SND, /* driver passed skb to NIC, or HW */
42};
25 43
26#endif /* _UAPI_LINUX_ERRQUEUE_H */ 44#endif /* _UAPI_LINUX_ERRQUEUE_H */
diff --git a/net/core/skbuff.c b/net/core/skbuff.c
index c1a33033cbe2..c9f68802e992 100644
--- a/net/core/skbuff.c
+++ b/net/core/skbuff.c
@@ -3521,6 +3521,7 @@ void skb_tstamp_tx(struct sk_buff *orig_skb,
3521 memset(serr, 0, sizeof(*serr)); 3521 memset(serr, 0, sizeof(*serr));
3522 serr->ee.ee_errno = ENOMSG; 3522 serr->ee.ee_errno = ENOMSG;
3523 serr->ee.ee_origin = SO_EE_ORIGIN_TIMESTAMPING; 3523 serr->ee.ee_origin = SO_EE_ORIGIN_TIMESTAMPING;
3524 serr->ee.ee_info = SCM_TSTAMP_SND;
3524 3525
3525 err = sock_queue_err_skb(sk, skb); 3526 err = sock_queue_err_skb(sk, skb);
3526 3527
diff --git a/net/socket.c b/net/socket.c
index d8222c025061..dc0cc5d95ee5 100644
--- a/net/socket.c
+++ b/net/socket.c
@@ -106,6 +106,7 @@
106#include <linux/sockios.h> 106#include <linux/sockios.h>
107#include <linux/atalk.h> 107#include <linux/atalk.h>
108#include <net/busy_poll.h> 108#include <net/busy_poll.h>
109#include <linux/errqueue.h>
109 110
110#ifdef CONFIG_NET_RX_BUSY_POLL 111#ifdef CONFIG_NET_RX_BUSY_POLL
111unsigned int sysctl_net_busy_read __read_mostly; 112unsigned int sysctl_net_busy_read __read_mostly;
@@ -697,7 +698,7 @@ void __sock_recv_timestamp(struct msghdr *msg, struct sock *sk,
697 struct sk_buff *skb) 698 struct sk_buff *skb)
698{ 699{
699 int need_software_tstamp = sock_flag(sk, SOCK_RCVTSTAMP); 700 int need_software_tstamp = sock_flag(sk, SOCK_RCVTSTAMP);
700 struct timespec ts[3]; 701 struct scm_timestamping tss;
701 int empty = 1; 702 int empty = 1;
702 struct skb_shared_hwtstamps *shhwtstamps = 703 struct skb_shared_hwtstamps *shhwtstamps =
703 skb_hwtstamps(skb); 704 skb_hwtstamps(skb);
@@ -714,24 +715,25 @@ void __sock_recv_timestamp(struct msghdr *msg, struct sock *sk,
714 put_cmsg(msg, SOL_SOCKET, SCM_TIMESTAMP, 715 put_cmsg(msg, SOL_SOCKET, SCM_TIMESTAMP,
715 sizeof(tv), &tv); 716 sizeof(tv), &tv);
716 } else { 717 } else {
717 skb_get_timestampns(skb, &ts[0]); 718 struct timespec ts;
719 skb_get_timestampns(skb, &ts);
718 put_cmsg(msg, SOL_SOCKET, SCM_TIMESTAMPNS, 720 put_cmsg(msg, SOL_SOCKET, SCM_TIMESTAMPNS,
719 sizeof(ts[0]), &ts[0]); 721 sizeof(ts), &ts);
720 } 722 }
721 } 723 }
722 724
723 725 memset(&tss, 0, sizeof(tss));
724 memset(ts, 0, sizeof(ts)); 726 if ((sock_flag(sk, SOCK_TIMESTAMPING_SOFTWARE) ||
725 if (sock_flag(sk, SOCK_TIMESTAMPING_SOFTWARE) && 727 skb_shinfo(skb)->tx_flags & SKBTX_ANY_SW_TSTAMP) &&
726 ktime_to_timespec_cond(skb->tstamp, ts + 0)) 728 ktime_to_timespec_cond(skb->tstamp, tss.ts + 0))
727 empty = 0; 729 empty = 0;
728 if (shhwtstamps && 730 if (shhwtstamps &&
729 sock_flag(sk, SOCK_TIMESTAMPING_RAW_HARDWARE) && 731 sock_flag(sk, SOCK_TIMESTAMPING_RAW_HARDWARE) &&
730 ktime_to_timespec_cond(shhwtstamps->hwtstamp, ts + 2)) 732 ktime_to_timespec_cond(shhwtstamps->hwtstamp, tss.ts + 2))
731 empty = 0; 733 empty = 0;
732 if (!empty) 734 if (!empty)
733 put_cmsg(msg, SOL_SOCKET, 735 put_cmsg(msg, SOL_SOCKET,
734 SCM_TIMESTAMPING, sizeof(ts), &ts); 736 SCM_TIMESTAMPING, sizeof(tss), &tss);
735} 737}
736EXPORT_SYMBOL_GPL(__sock_recv_timestamp); 738EXPORT_SYMBOL_GPL(__sock_recv_timestamp);
737 739