aboutsummaryrefslogtreecommitdiffstats
path: root/net/socket.c
diff options
context:
space:
mode:
Diffstat (limited to 'net/socket.c')
-rw-r--r--net/socket.c55
1 files changed, 32 insertions, 23 deletions
diff --git a/net/socket.c b/net/socket.c
index abf56b2a14f9..95ee7d8682e7 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;
@@ -609,15 +610,26 @@ void sock_release(struct socket *sock)
609} 610}
610EXPORT_SYMBOL(sock_release); 611EXPORT_SYMBOL(sock_release);
611 612
612void sock_tx_timestamp(struct sock *sk, __u8 *tx_flags) 613void sock_tx_timestamp(const struct sock *sk, __u8 *tx_flags)
613{ 614{
614 *tx_flags = 0; 615 u8 flags = *tx_flags;
615 if (sock_flag(sk, SOCK_TIMESTAMPING_TX_HARDWARE)) 616
616 *tx_flags |= SKBTX_HW_TSTAMP; 617 if (sk->sk_tsflags & SOF_TIMESTAMPING_TX_HARDWARE)
617 if (sock_flag(sk, SOCK_TIMESTAMPING_TX_SOFTWARE)) 618 flags |= SKBTX_HW_TSTAMP;
618 *tx_flags |= SKBTX_SW_TSTAMP; 619
620 if (sk->sk_tsflags & SOF_TIMESTAMPING_TX_SOFTWARE)
621 flags |= SKBTX_SW_TSTAMP;
622
623 if (sk->sk_tsflags & SOF_TIMESTAMPING_TX_SCHED)
624 flags |= SKBTX_SCHED_TSTAMP;
625
626 if (sk->sk_tsflags & SOF_TIMESTAMPING_TX_ACK)
627 flags |= SKBTX_ACK_TSTAMP;
628
619 if (sock_flag(sk, SOCK_WIFI_STATUS)) 629 if (sock_flag(sk, SOCK_WIFI_STATUS))
620 *tx_flags |= SKBTX_WIFI_STATUS; 630 flags |= SKBTX_WIFI_STATUS;
631
632 *tx_flags = flags;
621} 633}
622EXPORT_SYMBOL(sock_tx_timestamp); 634EXPORT_SYMBOL(sock_tx_timestamp);
623 635
@@ -697,7 +709,7 @@ void __sock_recv_timestamp(struct msghdr *msg, struct sock *sk,
697 struct sk_buff *skb) 709 struct sk_buff *skb)
698{ 710{
699 int need_software_tstamp = sock_flag(sk, SOCK_RCVTSTAMP); 711 int need_software_tstamp = sock_flag(sk, SOCK_RCVTSTAMP);
700 struct timespec ts[3]; 712 struct scm_timestamping tss;
701 int empty = 1; 713 int empty = 1;
702 struct skb_shared_hwtstamps *shhwtstamps = 714 struct skb_shared_hwtstamps *shhwtstamps =
703 skb_hwtstamps(skb); 715 skb_hwtstamps(skb);
@@ -714,28 +726,25 @@ void __sock_recv_timestamp(struct msghdr *msg, struct sock *sk,
714 put_cmsg(msg, SOL_SOCKET, SCM_TIMESTAMP, 726 put_cmsg(msg, SOL_SOCKET, SCM_TIMESTAMP,
715 sizeof(tv), &tv); 727 sizeof(tv), &tv);
716 } else { 728 } else {
717 skb_get_timestampns(skb, &ts[0]); 729 struct timespec ts;
730 skb_get_timestampns(skb, &ts);
718 put_cmsg(msg, SOL_SOCKET, SCM_TIMESTAMPNS, 731 put_cmsg(msg, SOL_SOCKET, SCM_TIMESTAMPNS,
719 sizeof(ts[0]), &ts[0]); 732 sizeof(ts), &ts);
720 } 733 }
721 } 734 }
722 735
723 736 memset(&tss, 0, sizeof(tss));
724 memset(ts, 0, sizeof(ts)); 737 if ((sk->sk_tsflags & SOF_TIMESTAMPING_SOFTWARE ||
725 if (sock_flag(sk, SOCK_TIMESTAMPING_SOFTWARE) && 738 skb_shinfo(skb)->tx_flags & SKBTX_ANY_SW_TSTAMP) &&
726 ktime_to_timespec_cond(skb->tstamp, ts + 0)) 739 ktime_to_timespec_cond(skb->tstamp, tss.ts + 0))
740 empty = 0;
741 if (shhwtstamps &&
742 (sk->sk_tsflags & SOF_TIMESTAMPING_RAW_HARDWARE) &&
743 ktime_to_timespec_cond(shhwtstamps->hwtstamp, tss.ts + 2))
727 empty = 0; 744 empty = 0;
728 if (shhwtstamps) {
729 if (sock_flag(sk, SOCK_TIMESTAMPING_SYS_HARDWARE) &&
730 ktime_to_timespec_cond(shhwtstamps->syststamp, ts + 1))
731 empty = 0;
732 if (sock_flag(sk, SOCK_TIMESTAMPING_RAW_HARDWARE) &&
733 ktime_to_timespec_cond(shhwtstamps->hwtstamp, ts + 2))
734 empty = 0;
735 }
736 if (!empty) 745 if (!empty)
737 put_cmsg(msg, SOL_SOCKET, 746 put_cmsg(msg, SOL_SOCKET,
738 SCM_TIMESTAMPING, sizeof(ts), &ts); 747 SCM_TIMESTAMPING, sizeof(tss), &tss);
739} 748}
740EXPORT_SYMBOL_GPL(__sock_recv_timestamp); 749EXPORT_SYMBOL_GPL(__sock_recv_timestamp);
741 750