net: datagram: drop 'destructor' argument from several helpers
authorPaolo Abeni <pabeni@redhat.com>
Fri, 28 Feb 2020 13:45:22 +0000 (14:45 +0100)
committerDavid S. Miller <davem@davemloft.net>
Fri, 28 Feb 2020 20:12:53 +0000 (12:12 -0800)
The only users for such argument are the UDP protocol and the UNIX
socket family. We can safely reclaim the accounted memory directly
from the UDP code and, after the previous patch, we can do scm
stats accounting outside the datagram helpers.

Overall this cleans up a bit some datagram-related helpers, and
avoids an indirect call per packet in the UDP receive path.

v1 -> v2:
 - call scm_stat_del() only when not peeking - Kirill
 - fix build issue with CONFIG_INET_ESPINTCP

Signed-off-by: Paolo Abeni <pabeni@redhat.com>
Reviewed-by: Kirill Tkhai <ktkhai@virtuozzo.com>
Reviewed-by: Willem de Bruijn <willemb@google.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
include/linux/skbuff.h
net/core/datagram.c
net/ipv4/udp.c
net/unix/af_unix.c
net/xfrm/espintcp.c

index 5b50278c4bc852b7659f9af8e392a45630111d0b..21749b2cdc9b5ee061a4a878da7583eb33251246 100644 (file)
@@ -3514,23 +3514,15 @@ int __skb_wait_for_more_packets(struct sock *sk, struct sk_buff_head *queue,
 struct sk_buff *__skb_try_recv_from_queue(struct sock *sk,
                                          struct sk_buff_head *queue,
                                          unsigned int flags,
-                                         void (*destructor)(struct sock *sk,
-                                                          struct sk_buff *skb),
                                          int *off, int *err,
                                          struct sk_buff **last);
 struct sk_buff *__skb_try_recv_datagram(struct sock *sk,
                                        struct sk_buff_head *queue,
-                                       unsigned int flags,
-                                       void (*destructor)(struct sock *sk,
-                                                          struct sk_buff *skb),
-                                       int *off, int *err,
+                                       unsigned int flags, int *off, int *err,
                                        struct sk_buff **last);
 struct sk_buff *__skb_recv_datagram(struct sock *sk,
                                    struct sk_buff_head *sk_queue,
-                                   unsigned int flags,
-                                   void (*destructor)(struct sock *sk,
-                                                      struct sk_buff *skb),
-                                   int *off, int *err);
+                                   unsigned int flags, int *off, int *err);
 struct sk_buff *skb_recv_datagram(struct sock *sk, unsigned flags, int noblock,
                                  int *err);
 __poll_t datagram_poll(struct file *file, struct socket *sock,
index a78e7f864c1e1d51cddd21ecfd7fc736b410d778..4213081c6ed3d4fda69501641a8c76e041f26b42 100644 (file)
@@ -166,8 +166,6 @@ done:
 struct sk_buff *__skb_try_recv_from_queue(struct sock *sk,
                                          struct sk_buff_head *queue,
                                          unsigned int flags,
-                                         void (*destructor)(struct sock *sk,
-                                                          struct sk_buff *skb),
                                          int *off, int *err,
                                          struct sk_buff **last)
 {
@@ -198,8 +196,6 @@ struct sk_buff *__skb_try_recv_from_queue(struct sock *sk,
                        refcount_inc(&skb->users);
                } else {
                        __skb_unlink(skb, queue);
-                       if (destructor)
-                               destructor(sk, skb);
                }
                *off = _off;
                return skb;
@@ -212,7 +208,6 @@ struct sk_buff *__skb_try_recv_from_queue(struct sock *sk,
  *     @sk: socket
  *     @queue: socket queue from which to receive
  *     @flags: MSG\_ flags
- *     @destructor: invoked under the receive lock on successful dequeue
  *     @off: an offset in bytes to peek skb from. Returns an offset
  *           within an skb where data actually starts
  *     @err: error code returned
@@ -245,10 +240,7 @@ struct sk_buff *__skb_try_recv_from_queue(struct sock *sk,
  */
 struct sk_buff *__skb_try_recv_datagram(struct sock *sk,
                                        struct sk_buff_head *queue,
-                                       unsigned int flags,
-                                       void (*destructor)(struct sock *sk,
-                                                          struct sk_buff *skb),
-                                       int *off, int *err,
+                                       unsigned int flags, int *off, int *err,
                                        struct sk_buff **last)
 {
        struct sk_buff *skb;
@@ -269,8 +261,8 @@ struct sk_buff *__skb_try_recv_datagram(struct sock *sk,
                 * However, this function was correct in any case. 8)
                 */
                spin_lock_irqsave(&queue->lock, cpu_flags);
-               skb = __skb_try_recv_from_queue(sk, queue, flags, destructor,
-                                               off, &error, last);
+               skb = __skb_try_recv_from_queue(sk, queue, flags, off, &error,
+                                               last);
                spin_unlock_irqrestore(&queue->lock, cpu_flags);
                if (error)
                        goto no_packet;
@@ -293,10 +285,7 @@ EXPORT_SYMBOL(__skb_try_recv_datagram);
 
 struct sk_buff *__skb_recv_datagram(struct sock *sk,
                                    struct sk_buff_head *sk_queue,
-                                   unsigned int flags,
-                                   void (*destructor)(struct sock *sk,
-                                                      struct sk_buff *skb),
-                                   int *off, int *err)
+                                   unsigned int flags, int *off, int *err)
 {
        struct sk_buff *skb, *last;
        long timeo;
@@ -304,8 +293,8 @@ struct sk_buff *__skb_recv_datagram(struct sock *sk,
        timeo = sock_rcvtimeo(sk, flags & MSG_DONTWAIT);
 
        do {
-               skb = __skb_try_recv_datagram(sk, sk_queue, flags, destructor,
-                                             off, err, &last);
+               skb = __skb_try_recv_datagram(sk, sk_queue, flags, off, err,
+                                             &last);
                if (skb)
                        return skb;
 
@@ -326,7 +315,7 @@ struct sk_buff *skb_recv_datagram(struct sock *sk, unsigned int flags,
 
        return __skb_recv_datagram(sk, &sk->sk_receive_queue,
                                   flags | (noblock ? MSG_DONTWAIT : 0),
-                                  NULL, &off, err);
+                                  &off, err);
 }
 EXPORT_SYMBOL(skb_recv_datagram);
 
index 08a41f1e1cd22478b9ea7740fa34e3979373ac6f..a68e2ac37f26c991688d904edbb5f48b4af99b88 100644 (file)
@@ -1671,10 +1671,11 @@ struct sk_buff *__skb_recv_udp(struct sock *sk, unsigned int flags,
                error = -EAGAIN;
                do {
                        spin_lock_bh(&queue->lock);
-                       skb = __skb_try_recv_from_queue(sk, queue, flags,
-                                                       udp_skb_destructor,
-                                                       off, err, &last);
+                       skb = __skb_try_recv_from_queue(sk, queue, flags, off,
+                                                       err, &last);
                        if (skb) {
+                               if (!(flags & MSG_PEEK))
+                                       udp_skb_destructor(sk, skb);
                                spin_unlock_bh(&queue->lock);
                                return skb;
                        }
@@ -1692,9 +1693,10 @@ struct sk_buff *__skb_recv_udp(struct sock *sk, unsigned int flags,
                        spin_lock(&sk_queue->lock);
                        skb_queue_splice_tail_init(sk_queue, queue);
 
-                       skb = __skb_try_recv_from_queue(sk, queue, flags,
-                                                       udp_skb_dtor_locked,
-                                                       off, err, &last);
+                       skb = __skb_try_recv_from_queue(sk, queue, flags, off,
+                                                       err, &last);
+                       if (skb && !(flags & MSG_PEEK))
+                               udp_skb_dtor_locked(sk, skb);
                        spin_unlock(&sk_queue->lock);
                        spin_unlock_bh(&queue->lock);
                        if (skb)
index c46fa271fc4af3edc66ee684baf868c6db774cca..3385a7a0b231339f758cddd7cb1f4cb5ed2c3117 100644 (file)
@@ -2106,9 +2106,12 @@ static int unix_dgram_recvmsg(struct socket *sock, struct msghdr *msg,
 
                skip = sk_peek_offset(sk, flags);
                skb = __skb_try_recv_datagram(sk, &sk->sk_receive_queue, flags,
-                                             scm_stat_del, &skip, &err, &last);
-               if (skb)
+                                             &skip, &err, &last);
+               if (skb) {
+                       if (!(flags & MSG_PEEK))
+                               scm_stat_del(sk, skb);
                        break;
+               }
 
                mutex_unlock(&u->iolock);
 
index f15d6a564b0e63cc8d729d524043ab5531a84a62..037ea156d2f93b5e0f5611f884f766cacfaefff9 100644 (file)
@@ -100,7 +100,7 @@ static int espintcp_recvmsg(struct sock *sk, struct msghdr *msg, size_t len,
 
        flags |= nonblock ? MSG_DONTWAIT : 0;
 
-       skb = __skb_recv_datagram(sk, &ctx->ike_queue, flags, NULL, &off, &err);
+       skb = __skb_recv_datagram(sk, &ctx->ike_queue, flags, &off, &err);
        if (!skb)
                return err;