]> git.ipfire.org Git - thirdparty/kernel/stable.git/commitdiff
net: rose: lock the socket in rose_bind()
authorEric Dumazet <edumazet@google.com>
Mon, 3 Feb 2025 17:08:38 +0000 (17:08 +0000)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Thu, 13 Mar 2025 11:47:05 +0000 (12:47 +0100)
[ Upstream commit a1300691aed9ee852b0a9192e29e2bdc2411a7e6 ]

syzbot reported a soft lockup in rose_loopback_timer(),
with a repro calling bind() from multiple threads.

rose_bind() must lock the socket to avoid this issue.

Fixes: 1da177e4c3f4 ("Linux-2.6.12-rc2")
Reported-by: syzbot+7ff41b5215f0c534534e@syzkaller.appspotmail.com
Closes: https://lore.kernel.org/netdev/67a0f78d.050a0220.d7c5a.00a0.GAE@google.com/T/#u
Signed-off-by: Eric Dumazet <edumazet@google.com>
Acked-by: Paolo Abeni <pabeni@redhat.com>
Link: https://patch.msgid.link/20250203170838.3521361-1-edumazet@google.com
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
Signed-off-by: Sasha Levin <sashal@kernel.org>
net/rose/af_rose.c

index 65fd5b99f9deaf8fb865bab49ae14fc81c66c5c2..f8cd085c42345711d5ccb4b8bfeec4708975b4e6 100644 (file)
@@ -700,11 +700,9 @@ static int rose_bind(struct socket *sock, struct sockaddr *uaddr, int addr_len)
        struct net_device *dev;
        ax25_address *source;
        ax25_uid_assoc *user;
+       int err = -EINVAL;
        int n;
 
-       if (!sock_flag(sk, SOCK_ZAPPED))
-               return -EINVAL;
-
        if (addr_len != sizeof(struct sockaddr_rose) && addr_len != sizeof(struct full_sockaddr_rose))
                return -EINVAL;
 
@@ -717,8 +715,15 @@ static int rose_bind(struct socket *sock, struct sockaddr *uaddr, int addr_len)
        if ((unsigned int) addr->srose_ndigis > ROSE_MAX_DIGIS)
                return -EINVAL;
 
-       if ((dev = rose_dev_get(&addr->srose_addr)) == NULL)
-               return -EADDRNOTAVAIL;
+       lock_sock(sk);
+
+       if (!sock_flag(sk, SOCK_ZAPPED))
+               goto out_release;
+
+       err = -EADDRNOTAVAIL;
+       dev = rose_dev_get(&addr->srose_addr);
+       if (!dev)
+               goto out_release;
 
        source = &addr->srose_call;
 
@@ -729,7 +734,8 @@ static int rose_bind(struct socket *sock, struct sockaddr *uaddr, int addr_len)
        } else {
                if (ax25_uid_policy && !capable(CAP_NET_BIND_SERVICE)) {
                        dev_put(dev);
-                       return -EACCES;
+                       err = -EACCES;
+                       goto out_release;
                }
                rose->source_call   = *source;
        }
@@ -751,8 +757,10 @@ static int rose_bind(struct socket *sock, struct sockaddr *uaddr, int addr_len)
        rose_insert_socket(sk);
 
        sock_reset_flag(sk, SOCK_ZAPPED);
-
-       return 0;
+       err = 0;
+out_release:
+       release_sock(sk);
+       return err;
 }
 
 static int rose_connect(struct socket *sock, struct sockaddr *uaddr, int addr_len, int flags)