OSDN Git Service

unix_bind(): allocate addr earlier
authorAl Viro <viro@zeniv.linux.org.uk>
Sat, 19 Jun 2021 03:50:27 +0000 (03:50 +0000)
committerDavid S. Miller <davem@davemloft.net>
Mon, 21 Jun 2021 19:28:49 +0000 (12:28 -0700)
makes it easier to massage; we do pay for that by extra work
(kmalloc+memcpy+kfree) in some error cases, but those are not
on the hot paths anyway.

Signed-off-by: Al Viro <viro@zeniv.linux.org.uk>
Signed-off-by: David S. Miller <davem@davemloft.net>
net/unix/af_unix.c

index 9144762..a984cf3 100644 (file)
@@ -1040,6 +1040,15 @@ static int unix_bind(struct socket *sock, struct sockaddr *uaddr, int addr_len)
        if (err < 0)
                goto out;
        addr_len = err;
+       err = -ENOMEM;
+       addr = kmalloc(sizeof(*addr)+addr_len, GFP_KERNEL);
+       if (!addr)
+               goto out;
+
+       memcpy(addr->name, sunaddr, addr_len);
+       addr->len = addr_len;
+       addr->hash = hash ^ sk->sk_type;
+       refcount_set(&addr->refcnt, 1);
 
        if (sun_path[0]) {
                umode_t mode = S_IFSOCK |
@@ -1048,7 +1057,7 @@ static int unix_bind(struct socket *sock, struct sockaddr *uaddr, int addr_len)
                if (err) {
                        if (err == -EEXIST)
                                err = -EADDRINUSE;
-                       goto out;
+                       goto out_addr;
                }
        }
 
@@ -1060,16 +1069,6 @@ static int unix_bind(struct socket *sock, struct sockaddr *uaddr, int addr_len)
        if (u->addr)
                goto out_up;
 
-       err = -ENOMEM;
-       addr = kmalloc(sizeof(*addr)+addr_len, GFP_KERNEL);
-       if (!addr)
-               goto out_up;
-
-       memcpy(addr->name, sunaddr, addr_len);
-       addr->len = addr_len;
-       addr->hash = hash ^ sk->sk_type;
-       refcount_set(&addr->refcnt, 1);
-
        if (sun_path[0]) {
                addr->hash = UNIX_HASH_SIZE;
                hash = d_backing_inode(path.dentry)->i_ino & (UNIX_HASH_SIZE - 1);
@@ -1081,20 +1080,23 @@ static int unix_bind(struct socket *sock, struct sockaddr *uaddr, int addr_len)
                if (__unix_find_socket_byname(net, sunaddr, addr_len,
                                              sk->sk_type, hash)) {
                        spin_unlock(&unix_table_lock);
-                       unix_release_addr(addr);
                        goto out_up;
                }
                hash = addr->hash;
        }
 
-       err = 0;
        __unix_set_addr(sk, addr, hash);
        spin_unlock(&unix_table_lock);
+       addr = NULL;
+       err = 0;
 out_up:
        mutex_unlock(&u->bindlock);
 out_put:
        if (err)
                path_put(&path);
+out_addr:
+       if (addr)
+               unix_release_addr(addr);
 out:
        return err;
 }