diff options
author | Eric Dumazet <dada1@cosmosbay.com> | 2007-03-26 01:14:49 -0400 |
---|---|---|
committer | David S. Miller <davem@sunset.davemloft.net> | 2007-04-26 01:24:21 -0400 |
commit | 92f37fd2ee805aa77925c1e64fd56088b46094fc (patch) | |
tree | 8251c38b83ab362116dac89d94412ce229b42831 /net/socket.c | |
parent | c7a3c5da35055e2fa97ed4f0da3eec4bd0ef4c38 (diff) |
[NET]: Adding SO_TIMESTAMPNS / SCM_TIMESTAMPNS support
Now that network timestamps use ktime_t infrastructure, we can add a new
SOL_SOCKET sockopt SO_TIMESTAMPNS.
This command is similar to SO_TIMESTAMP, but permits transmission of
a 'timespec struct' instead of a 'timeval struct' control message.
(nanosecond resolution instead of microsecond)
Control message is labelled SCM_TIMESTAMPNS instead of SCM_TIMESTAMP
A socket cannot mix SO_TIMESTAMP and SO_TIMESTAMPNS : the two modes are
mutually exclusive.
sock_recv_timestamp() became too big to be fully inlined so I added a
__sock_recv_timestamp() helper function.
Signed-off-by: Eric Dumazet <dada1@cosmosbay.com>
CC: linux-arch@vger.kernel.org
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/socket.c')
-rw-r--r-- | net/socket.c | 29 |
1 files changed, 29 insertions, 0 deletions
diff --git a/net/socket.c b/net/socket.c index cf18c5eb592e..a7bd0df115b2 100644 --- a/net/socket.c +++ b/net/socket.c | |||
@@ -585,6 +585,35 @@ int kernel_sendmsg(struct socket *sock, struct msghdr *msg, | |||
585 | return result; | 585 | return result; |
586 | } | 586 | } |
587 | 587 | ||
588 | /* | ||
589 | * called from sock_recv_timestamp() if sock_flag(sk, SOCK_RCVTSTAMP) | ||
590 | */ | ||
591 | void __sock_recv_timestamp(struct msghdr *msg, struct sock *sk, | ||
592 | struct sk_buff *skb) | ||
593 | { | ||
594 | ktime_t kt = skb->tstamp; | ||
595 | |||
596 | if (!sock_flag(sk, SOCK_RCVTSTAMPNS)) { | ||
597 | struct timeval tv; | ||
598 | /* Race occurred between timestamp enabling and packet | ||
599 | receiving. Fill in the current time for now. */ | ||
600 | if (kt.tv64 == 0) | ||
601 | kt = ktime_get_real(); | ||
602 | skb->tstamp = kt; | ||
603 | tv = ktime_to_timeval(kt); | ||
604 | put_cmsg(msg, SOL_SOCKET, SCM_TIMESTAMP, sizeof(tv), &tv); | ||
605 | } else { | ||
606 | struct timespec ts; | ||
607 | /* Race occurred between timestamp enabling and packet | ||
608 | receiving. Fill in the current time for now. */ | ||
609 | if (kt.tv64 == 0) | ||
610 | kt = ktime_get_real(); | ||
611 | skb->tstamp = kt; | ||
612 | ts = ktime_to_timespec(kt); | ||
613 | put_cmsg(msg, SOL_SOCKET, SCM_TIMESTAMPNS, sizeof(ts), &ts); | ||
614 | } | ||
615 | } | ||
616 | |||
588 | static inline int __sock_recvmsg(struct kiocb *iocb, struct socket *sock, | 617 | static inline int __sock_recvmsg(struct kiocb *iocb, struct socket *sock, |
589 | struct msghdr *msg, size_t size, int flags) | 618 | struct msghdr *msg, size_t size, int flags) |
590 | { | 619 | { |