|
|
@@ -298,6 +298,7 @@ struct sock_common {
|
|
|
* @sk_filter: socket filtering instructions
|
|
|
* @sk_timer: sock cleanup timer
|
|
|
* @sk_stamp: time stamp of last packet received
|
|
|
+ * @sk_stamp_seq: lock for accessing sk_stamp on 32 bit architectures only
|
|
|
* @sk_tsflags: SO_TIMESTAMPING socket options
|
|
|
* @sk_tskey: counter to disambiguate concurrent tstamp requests
|
|
|
* @sk_zckey: counter to order MSG_ZEROCOPY notifications
|
|
|
@@ -474,6 +475,9 @@ struct sock {
|
|
|
const struct cred *sk_peer_cred;
|
|
|
long sk_rcvtimeo;
|
|
|
ktime_t sk_stamp;
|
|
|
+#if BITS_PER_LONG==32
|
|
|
+ seqlock_t sk_stamp_seq;
|
|
|
+#endif
|
|
|
u16 sk_tsflags;
|
|
|
u8 sk_shutdown;
|
|
|
u32 sk_tskey;
|
|
|
@@ -2290,6 +2294,34 @@ static inline void sk_drops_add(struct sock *sk, const struct sk_buff *skb)
|
|
|
atomic_add(segs, &sk->sk_drops);
|
|
|
}
|
|
|
|
|
|
+static inline ktime_t sock_read_timestamp(struct sock *sk)
|
|
|
+{
|
|
|
+#if BITS_PER_LONG==32
|
|
|
+ unsigned int seq;
|
|
|
+ ktime_t kt;
|
|
|
+
|
|
|
+ do {
|
|
|
+ seq = read_seqbegin(&sk->sk_stamp_seq);
|
|
|
+ kt = sk->sk_stamp;
|
|
|
+ } while (read_seqretry(&sk->sk_stamp_seq, seq));
|
|
|
+
|
|
|
+ return kt;
|
|
|
+#else
|
|
|
+ return sk->sk_stamp;
|
|
|
+#endif
|
|
|
+}
|
|
|
+
|
|
|
+static inline void sock_write_timestamp(struct sock *sk, ktime_t kt)
|
|
|
+{
|
|
|
+#if BITS_PER_LONG==32
|
|
|
+ write_seqlock(&sk->sk_stamp_seq);
|
|
|
+ sk->sk_stamp = kt;
|
|
|
+ write_sequnlock(&sk->sk_stamp_seq);
|
|
|
+#else
|
|
|
+ sk->sk_stamp = kt;
|
|
|
+#endif
|
|
|
+}
|
|
|
+
|
|
|
void __sock_recv_timestamp(struct msghdr *msg, struct sock *sk,
|
|
|
struct sk_buff *skb);
|
|
|
void __sock_recv_wifi_status(struct msghdr *msg, struct sock *sk,
|
|
|
@@ -2314,7 +2346,7 @@ sock_recv_timestamp(struct msghdr *msg, struct sock *sk, struct sk_buff *skb)
|
|
|
(sk->sk_tsflags & SOF_TIMESTAMPING_RAW_HARDWARE)))
|
|
|
__sock_recv_timestamp(msg, sk, skb);
|
|
|
else
|
|
|
- sk->sk_stamp = kt;
|
|
|
+ sock_write_timestamp(sk, kt);
|
|
|
|
|
|
if (sock_flag(sk, SOCK_WIFI_STATUS) && skb->wifi_acked_valid)
|
|
|
__sock_recv_wifi_status(msg, sk, skb);
|
|
|
@@ -2335,9 +2367,9 @@ static inline void sock_recv_ts_and_drops(struct msghdr *msg, struct sock *sk,
|
|
|
if (sk->sk_flags & FLAGS_TS_OR_DROPS || sk->sk_tsflags & TSFLAGS_ANY)
|
|
|
__sock_recv_ts_and_drops(msg, sk, skb);
|
|
|
else if (unlikely(sock_flag(sk, SOCK_TIMESTAMP)))
|
|
|
- sk->sk_stamp = skb->tstamp;
|
|
|
+ sock_write_timestamp(sk, skb->tstamp);
|
|
|
else if (unlikely(sk->sk_stamp == SK_DEFAULT_STAMP))
|
|
|
- sk->sk_stamp = 0;
|
|
|
+ sock_write_timestamp(sk, 0);
|
|
|
}
|
|
|
|
|
|
void __sock_tx_timestamp(__u16 tsflags, __u8 *tx_flags);
|