]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
sctp: stream: fully roll back denied add-stream state
authorWyatt Feng <bronzed_45_vested@icloud.com>
Fri, 5 Jun 2026 05:53:42 +0000 (13:53 +0800)
committerJakub Kicinski <kuba@kernel.org>
Wed, 10 Jun 2026 00:26:25 +0000 (17:26 -0700)
When ADD_OUT_STREAMS is denied, SCTP only shrinks the queued chunks and
then lowers outcnt. That leaves removed stream metadata behind, so a
later re-add can reuse a stale ext and hit a null-pointer dereference in
the scheduler get path.

Fix the rollback by tearing down the removed stream state the same way
other stream resizes do. Unschedule the current scheduler state, drop
the removed stream ext state with sctp_stream_outq_migrate(), and then
reschedule the remaining streams.

This keeps scheduler-private RR/FC/PRIO lists consistent while fully
rolling back denied outgoing stream additions.

Fixes: 637784ade221 ("sctp: introduce priority based stream scheduler")
Cc: stable@kernel.org
Reported-by: Yuan Tan <yuantan098@gmail.com>
Reported-by: Yifan Wu <yifanwucs@gmail.com>
Reported-by: Juefei Pu <tomapufckgml@gmail.com>
Reported-by: Zhengchuan Liang <zcliangcn@gmail.com>
Reported-by: Xin Liu <bird@lzu.edu.cn>
Signed-off-by: Wyatt Feng <bronzed_45_vested@icloud.com>
Signed-off-by: Ren Wei <n05ec@lzu.edu.cn>
Acked-by: Xin Long <lucien.xin@gmail.com>
Link: https://patch.msgid.link/d78954ecd94954653ee299400e98d74a03a6f7d3.1780603399.git.bronzed_45_vested@icloud.com
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
net/sctp/stream.c

index c2247793c88b80f2a6476aae4f0d02bab6d94b35..5c2fdedea088ffc4f03fa6c04c744aa4a098b4f2 100644 (file)
@@ -1038,6 +1038,7 @@ struct sctp_chunk *sctp_process_strreset_resp(
                        stsn, rtsn, GFP_ATOMIC);
        } else if (req->type == SCTP_PARAM_RESET_ADD_OUT_STREAMS) {
                struct sctp_strreset_addstrm *addstrm;
+               const struct sctp_sched_ops *sched;
                __u16 number;
 
                addstrm = (struct sctp_strreset_addstrm *)req;
@@ -1048,7 +1049,10 @@ struct sctp_chunk *sctp_process_strreset_resp(
                        for (i = number; i < stream->outcnt; i++)
                                SCTP_SO(stream, i)->state = SCTP_STREAM_OPEN;
                } else {
-                       sctp_stream_shrink_out(stream, number);
+                       sched = sctp_sched_ops_from_stream(stream);
+                       sched->unsched_all(stream);
+                       sctp_stream_outq_migrate(stream, NULL, number);
+                       sched->sched_all(stream);
                        stream->outcnt = number;
                }