Commit 61d5d406 authored by Marcelo Ricardo Leitner's avatar Marcelo Ricardo Leitner Committed by David S. Miller

sctp: fix err handling of stream initialization

The fix on 951c6db9 fixed the issued reported there but introduced
another. When the allocation fails within sctp_stream_init() it is
okay/necessary to free the genradix. But it is also called when adding
new streams, from sctp_send_add_streams() and
sctp_process_strreset_addstrm_in() and in those situations it cannot
just free the genradix because by then it is a fully operational
association.

The fix here then is to only free the genradix in sctp_stream_init()
and on those other call sites  move on with what it already had and let
the subsequent error handling to handle it.

Tested with the reproducers from this report and the previous one,
with lksctp-tools and sctp-tests.

Reported-by: syzbot+9a1bc632e78a1a98488b@syzkaller.appspotmail.com
Fixes: 951c6db9 ("sctp: fix memleak on err handling of stream initialization")
Signed-off-by: default avatarMarcelo Ricardo Leitner <marcelo.leitner@gmail.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent feed8a4f
...@@ -84,10 +84,8 @@ static int sctp_stream_alloc_out(struct sctp_stream *stream, __u16 outcnt, ...@@ -84,10 +84,8 @@ static int sctp_stream_alloc_out(struct sctp_stream *stream, __u16 outcnt,
return 0; return 0;
ret = genradix_prealloc(&stream->out, outcnt, gfp); ret = genradix_prealloc(&stream->out, outcnt, gfp);
if (ret) { if (ret)
genradix_free(&stream->out);
return ret; return ret;
}
stream->outcnt = outcnt; stream->outcnt = outcnt;
return 0; return 0;
...@@ -102,10 +100,8 @@ static int sctp_stream_alloc_in(struct sctp_stream *stream, __u16 incnt, ...@@ -102,10 +100,8 @@ static int sctp_stream_alloc_in(struct sctp_stream *stream, __u16 incnt,
return 0; return 0;
ret = genradix_prealloc(&stream->in, incnt, gfp); ret = genradix_prealloc(&stream->in, incnt, gfp);
if (ret) { if (ret)
genradix_free(&stream->in);
return ret; return ret;
}
stream->incnt = incnt; stream->incnt = incnt;
return 0; return 0;
...@@ -123,7 +119,7 @@ int sctp_stream_init(struct sctp_stream *stream, __u16 outcnt, __u16 incnt, ...@@ -123,7 +119,7 @@ int sctp_stream_init(struct sctp_stream *stream, __u16 outcnt, __u16 incnt,
* a new one with new outcnt to save memory if needed. * a new one with new outcnt to save memory if needed.
*/ */
if (outcnt == stream->outcnt) if (outcnt == stream->outcnt)
goto in; goto handle_in;
/* Filter out chunks queued on streams that won't exist anymore */ /* Filter out chunks queued on streams that won't exist anymore */
sched->unsched_all(stream); sched->unsched_all(stream);
...@@ -132,24 +128,28 @@ int sctp_stream_init(struct sctp_stream *stream, __u16 outcnt, __u16 incnt, ...@@ -132,24 +128,28 @@ int sctp_stream_init(struct sctp_stream *stream, __u16 outcnt, __u16 incnt,
ret = sctp_stream_alloc_out(stream, outcnt, gfp); ret = sctp_stream_alloc_out(stream, outcnt, gfp);
if (ret) if (ret)
goto out; goto out_err;
for (i = 0; i < stream->outcnt; i++) for (i = 0; i < stream->outcnt; i++)
SCTP_SO(stream, i)->state = SCTP_STREAM_OPEN; SCTP_SO(stream, i)->state = SCTP_STREAM_OPEN;
in: handle_in:
sctp_stream_interleave_init(stream); sctp_stream_interleave_init(stream);
if (!incnt) if (!incnt)
goto out; goto out;
ret = sctp_stream_alloc_in(stream, incnt, gfp); ret = sctp_stream_alloc_in(stream, incnt, gfp);
if (ret) { if (ret)
goto in_err;
goto out;
in_err:
sched->free(stream); sched->free(stream);
genradix_free(&stream->in);
out_err:
genradix_free(&stream->out); genradix_free(&stream->out);
stream->outcnt = 0; stream->outcnt = 0;
goto out;
}
out: out:
return ret; return ret;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment