Skip to content

Commit 77b5877

Browse files
edumazetsmb49
authored andcommitted
af_unix: fix struct pid leaks in OOB support
BugLink: https://bugs.launchpad.net/bugs/2023224 [ Upstream commit 2aab4b9 ] syzbot reported struct pid leak [1]. Issue is that queue_oob() calls maybe_add_creds() which potentially holds a reference on a pid. But skb->destructor is not set (either directly or by calling unix_scm_to_skb()) This means that subsequent kfree_skb() or consume_skb() would leak this reference. In this fix, I chose to fully support scm even for the OOB message. [1] BUG: memory leak unreferenced object 0xffff8881053e7f80 (size 128): comm "syz-executor242", pid 5066, jiffies 4294946079 (age 13.220s) hex dump (first 32 bytes): 01 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 ................ 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 ................ backtrace: [<ffffffff812ae26a>] alloc_pid+0x6a/0x560 kernel/pid.c:180 [<ffffffff812718df>] copy_process+0x169f/0x26c0 kernel/fork.c:2285 [<ffffffff81272b37>] kernel_clone+0xf7/0x610 kernel/fork.c:2684 [<ffffffff812730cc>] __do_sys_clone+0x7c/0xb0 kernel/fork.c:2825 [<ffffffff849ad699>] do_syscall_x64 arch/x86/entry/common.c:50 [inline] [<ffffffff849ad699>] do_syscall_64+0x39/0xb0 arch/x86/entry/common.c:80 [<ffffffff84a0008b>] entry_SYSCALL_64_after_hwframe+0x63/0xcd Fixes: 314001f ("af_unix: Add OOB support") Reported-by: [email protected] Signed-off-by: Eric Dumazet <[email protected]> Cc: Rao Shoaib <[email protected]> Reviewed-by: Kuniyuki Iwashima <[email protected]> Link: https://lore.kernel.org/r/[email protected] Signed-off-by: Jakub Kicinski <[email protected]> Signed-off-by: Sasha Levin <[email protected]> Signed-off-by: Kamal Mostafa <[email protected]> Signed-off-by: Luke Nowakowski-Krijger <[email protected]>
1 parent 4ab51bd commit 77b5877

File tree

1 file changed

+8
-2
lines changed

1 file changed

+8
-2
lines changed

net/unix/af_unix.c

Lines changed: 8 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -1970,7 +1970,8 @@ static int unix_dgram_sendmsg(struct socket *sock, struct msghdr *msg,
19701970
#define UNIX_SKB_FRAGS_SZ (PAGE_SIZE << get_order(32768))
19711971

19721972
#if IS_ENABLED(CONFIG_AF_UNIX_OOB)
1973-
static int queue_oob(struct socket *sock, struct msghdr *msg, struct sock *other)
1973+
static int queue_oob(struct socket *sock, struct msghdr *msg, struct sock *other,
1974+
struct scm_cookie *scm, bool fds_sent)
19741975
{
19751976
struct unix_sock *ousk = unix_sk(other);
19761977
struct sk_buff *skb;
@@ -1981,6 +1982,11 @@ static int queue_oob(struct socket *sock, struct msghdr *msg, struct sock *other
19811982
if (!skb)
19821983
return err;
19831984

1985+
err = unix_scm_to_skb(scm, skb, !fds_sent);
1986+
if (err < 0) {
1987+
kfree_skb(skb);
1988+
return err;
1989+
}
19841990
skb_put(skb, 1);
19851991
err = skb_copy_datagram_from_iter(skb, 0, &msg->msg_iter, 1);
19861992

@@ -2108,7 +2114,7 @@ static int unix_stream_sendmsg(struct socket *sock, struct msghdr *msg,
21082114

21092115
#if IS_ENABLED(CONFIG_AF_UNIX_OOB)
21102116
if (msg->msg_flags & MSG_OOB) {
2111-
err = queue_oob(sock, msg, other);
2117+
err = queue_oob(sock, msg, other, &scm, fds_sent);
21122118
if (err)
21132119
goto out_err;
21142120
sent++;

0 commit comments

Comments
 (0)