Skip to content

Commit 7d11e04

Browse files
wdebruijkuba-moo
authored andcommitted
net: do not write to msg_get_inq in callee
NULL pointer dereference fix. msg_get_inq is an input field from caller to callee. Don't set it in the callee, as the caller may not clear it on struct reuse. This is a kernel-internal variant of msghdr only, and the only user does reinitialize the field. So this is not critical for that reason. But it is more robust to avoid the write, and slightly simpler code. And it fixes a bug, see below. Callers set msg_get_inq to request the input queue length to be returned in msg_inq. This is equivalent to but independent from the SO_INQ request to return that same info as a cmsg (tp->recvmsg_inq). To reduce branching in the hot path the second also sets the msg_inq. That is WAI. This is a fix to commit 4d14429 ("af_unix: don't post cmsg for SO_INQ unless explicitly asked for"), which fixed the inverse. Also avoid NULL pointer dereference in unix_stream_read_generic if state->msg is NULL and msg->msg_get_inq is written. A NULL state->msg can happen when splicing as of commit 2b51457 ("net: af_unix: implement splice for stream af_unix sockets"). Also collapse two branches using a bitwise or. Cc: stable@vger.kernel.org Fixes: 4d14429 ("af_unix: don't post cmsg for SO_INQ unless explicitly asked for") Link: https://lore.kernel.org/netdev/willemdebruijn.kernel.24d8030f7a3de@gmail.com/ Signed-off-by: Willem de Bruijn <willemb@google.com> Reviewed-by: Jens Axboe <axboe@kernel.dk> Reviewed-by: Eric Dumazet <edumazet@google.com> Reviewed-by: Kuniyuki Iwashima <kuniyu@google.com> Link: https://patch.msgid.link/20260106150626.3944363-1-willemdebruijn.kernel@gmail.com Signed-off-by: Jakub Kicinski <kuba@kernel.org>
1 parent 3358995 commit 7d11e04

2 files changed

Lines changed: 6 additions & 10 deletions

File tree

net/ipv4/tcp.c

Lines changed: 3 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -2652,10 +2652,8 @@ static int tcp_recvmsg_locked(struct sock *sk, struct msghdr *msg, size_t len,
26522652
if (sk->sk_state == TCP_LISTEN)
26532653
goto out;
26542654

2655-
if (tp->recvmsg_inq) {
2655+
if (tp->recvmsg_inq)
26562656
*cmsg_flags = TCP_CMSG_INQ;
2657-
msg->msg_get_inq = 1;
2658-
}
26592657
timeo = sock_rcvtimeo(sk, flags & MSG_DONTWAIT);
26602658

26612659
/* Urgent data needs to be handled specially. */
@@ -2929,10 +2927,10 @@ int tcp_recvmsg(struct sock *sk, struct msghdr *msg, size_t len, int flags,
29292927
ret = tcp_recvmsg_locked(sk, msg, len, flags, &tss, &cmsg_flags);
29302928
release_sock(sk);
29312929

2932-
if ((cmsg_flags || msg->msg_get_inq) && ret >= 0) {
2930+
if ((cmsg_flags | msg->msg_get_inq) && ret >= 0) {
29332931
if (cmsg_flags & TCP_CMSG_TS)
29342932
tcp_recv_timestamp(msg, sk, &tss);
2935-
if (msg->msg_get_inq) {
2933+
if ((cmsg_flags & TCP_CMSG_INQ) | msg->msg_get_inq) {
29362934
msg->msg_inq = tcp_inq_hint(sk);
29372935
if (cmsg_flags & TCP_CMSG_INQ)
29382936
put_cmsg(msg, SOL_TCP, TCP_CM_INQ,

net/unix/af_unix.c

Lines changed: 3 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -2904,7 +2904,6 @@ static int unix_stream_read_generic(struct unix_stream_read_state *state,
29042904
unsigned int last_len;
29052905
struct unix_sock *u;
29062906
int copied = 0;
2907-
bool do_cmsg;
29082907
int err = 0;
29092908
long timeo;
29102909
int target;
@@ -2930,9 +2929,6 @@ static int unix_stream_read_generic(struct unix_stream_read_state *state,
29302929

29312930
u = unix_sk(sk);
29322931

2933-
do_cmsg = READ_ONCE(u->recvmsg_inq);
2934-
if (do_cmsg)
2935-
msg->msg_get_inq = 1;
29362932
redo:
29372933
/* Lock the socket to prevent queue disordering
29382934
* while sleeps in memcpy_tomsg
@@ -3090,9 +3086,11 @@ static int unix_stream_read_generic(struct unix_stream_read_state *state,
30903086

30913087
mutex_unlock(&u->iolock);
30923088
if (msg) {
3089+
bool do_cmsg = READ_ONCE(u->recvmsg_inq);
3090+
30933091
scm_recv_unix(sock, msg, &scm, flags);
30943092

3095-
if (msg->msg_get_inq && (copied ?: err) >= 0) {
3093+
if ((do_cmsg | msg->msg_get_inq) && (copied ?: err) >= 0) {
30963094
msg->msg_inq = READ_ONCE(u->inq_len);
30973095
if (do_cmsg)
30983096
put_cmsg(msg, SOL_SOCKET, SCM_INQ,

0 commit comments

Comments
 (0)