Coverage Report

Created: 2026-04-01 06:39

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/openssl36/ssl/quic/quic_stream_map.c
Line
Count
Source
1
/*
2
 * Copyright 2022-2025 The OpenSSL Project Authors. All Rights Reserved.
3
 *
4
 * Licensed under the Apache License 2.0 (the "License").  You may not use
5
 * this file except in compliance with the License.  You can obtain a copy
6
 * in the file LICENSE in the source distribution or at
7
 * https://www.openssl.org/source/license.html
8
 */
9
10
#include "internal/quic_stream_map.h"
11
#include "internal/nelem.h"
12
#include "internal/quic_channel.h"
13
14
/*
15
 * QUIC Stream Map
16
 * ===============
17
 */
18
DEFINE_LHASH_OF_EX(QUIC_STREAM);
19
20
static void shutdown_flush_done(QUIC_STREAM_MAP *qsm, QUIC_STREAM *qs);
21
22
/* Circular list management. */
23
static void list_insert_tail(QUIC_STREAM_LIST_NODE *l,
24
    QUIC_STREAM_LIST_NODE *n)
25
171k
{
26
    /* Must not be in list. */
27
171k
    assert(n->prev == NULL && n->next == NULL
28
171k
        && l->prev != NULL && l->next != NULL);
29
30
171k
    n->prev = l->prev;
31
171k
    n->prev->next = n;
32
171k
    l->prev = n;
33
171k
    n->next = l;
34
171k
}
35
36
static void list_remove(QUIC_STREAM_LIST_NODE *l,
37
    QUIC_STREAM_LIST_NODE *n)
38
171k
{
39
171k
    assert(n->prev != NULL && n->next != NULL
40
171k
        && n->prev != n && n->next != n);
41
42
171k
    n->prev->next = n->next;
43
171k
    n->next->prev = n->prev;
44
171k
    n->next = n->prev = NULL;
45
171k
}
46
47
static QUIC_STREAM *list_next(QUIC_STREAM_LIST_NODE *l, QUIC_STREAM_LIST_NODE *n,
48
    size_t off)
49
72.9M
{
50
72.9M
    assert(n->prev != NULL && n->next != NULL
51
72.9M
        && (n == l || (n->prev != n && n->next != n))
52
72.9M
        && l->prev != NULL && l->next != NULL);
53
54
72.9M
    n = n->next;
55
56
72.9M
    if (n == l)
57
72.9M
        n = n->next;
58
72.9M
    if (n == l)
59
72.9M
        return NULL;
60
61
72.9M
    assert(n != NULL);
62
63
43.9k
    return (QUIC_STREAM *)(((char *)n) - off);
64
43.9k
}
65
66
43.4k
#define active_next(l, s) list_next((l), &(s)->active_node, \
67
43.4k
    offsetof(QUIC_STREAM, active_node))
68
0
#define accept_next(l, s) list_next((l), &(s)->accept_node, \
69
0
    offsetof(QUIC_STREAM, accept_node))
70
510
#define accept_head(l) list_next((l), (l), \
71
510
    offsetof(QUIC_STREAM, accept_node))
72
72.9M
#define ready_for_gc_head(l) list_next((l), (l), \
73
72.9M
    offsetof(QUIC_STREAM, ready_for_gc_node))
74
75
static unsigned long hash_stream(const QUIC_STREAM *s)
76
18.0M
{
77
18.0M
    return (unsigned long)s->id;
78
18.0M
}
79
80
static int cmp_stream(const QUIC_STREAM *a, const QUIC_STREAM *b)
81
349k
{
82
349k
    if (a->id < b->id)
83
0
        return -1;
84
349k
    if (a->id > b->id)
85
0
        return 1;
86
349k
    return 0;
87
349k
}
88
89
int ossl_quic_stream_map_init(QUIC_STREAM_MAP *qsm,
90
    uint64_t (*get_stream_limit_cb)(int uni, void *arg),
91
    void *get_stream_limit_cb_arg,
92
    QUIC_RXFC *max_streams_bidi_rxfc,
93
    QUIC_RXFC *max_streams_uni_rxfc,
94
    QUIC_CHANNEL *ch)
95
51.2k
{
96
51.2k
    qsm->map = lh_QUIC_STREAM_new(hash_stream, cmp_stream);
97
51.2k
    qsm->active_list.prev = qsm->active_list.next = &qsm->active_list;
98
51.2k
    qsm->accept_list.prev = qsm->accept_list.next = &qsm->accept_list;
99
51.2k
    qsm->ready_for_gc_list.prev = qsm->ready_for_gc_list.next
100
51.2k
        = &qsm->ready_for_gc_list;
101
51.2k
    qsm->rr_stepping = 1;
102
51.2k
    qsm->rr_counter = 0;
103
51.2k
    qsm->rr_cur = NULL;
104
105
51.2k
    qsm->num_accept_bidi = 0;
106
51.2k
    qsm->num_accept_uni = 0;
107
51.2k
    qsm->num_shutdown_flush = 0;
108
109
51.2k
    qsm->get_stream_limit_cb = get_stream_limit_cb;
110
51.2k
    qsm->get_stream_limit_cb_arg = get_stream_limit_cb_arg;
111
51.2k
    qsm->max_streams_bidi_rxfc = max_streams_bidi_rxfc;
112
51.2k
    qsm->max_streams_uni_rxfc = max_streams_uni_rxfc;
113
51.2k
    qsm->ch = ch;
114
51.2k
    return 1;
115
51.2k
}
116
117
static void release_each(QUIC_STREAM *stream, void *arg)
118
151k
{
119
151k
    QUIC_STREAM_MAP *qsm = arg;
120
121
151k
    ossl_quic_stream_map_release(qsm, stream);
122
151k
}
123
124
void ossl_quic_stream_map_cleanup(QUIC_STREAM_MAP *qsm)
125
51.2k
{
126
51.2k
    lh_QUIC_STREAM_set_down_load(qsm->map, 0);
127
51.2k
    ossl_quic_stream_map_visit(qsm, release_each, qsm);
128
129
51.2k
    lh_QUIC_STREAM_free(qsm->map);
130
51.2k
    qsm->map = NULL;
131
51.2k
}
132
133
void ossl_quic_stream_map_visit(QUIC_STREAM_MAP *qsm,
134
    void (*visit_cb)(QUIC_STREAM *stream, void *arg),
135
    void *visit_cb_arg)
136
209k
{
137
209k
    lh_QUIC_STREAM_doall_arg(qsm->map, visit_cb, visit_cb_arg);
138
209k
}
139
140
QUIC_STREAM *ossl_quic_stream_map_alloc(QUIC_STREAM_MAP *qsm,
141
    uint64_t stream_id,
142
    int type)
143
151k
{
144
151k
    QUIC_STREAM *s;
145
151k
    QUIC_STREAM key;
146
147
151k
    key.id = stream_id;
148
149
151k
    s = lh_QUIC_STREAM_retrieve(qsm->map, &key);
150
151k
    if (s != NULL)
151
0
        return NULL;
152
153
151k
    s = OPENSSL_zalloc(sizeof(*s));
154
151k
    if (s == NULL)
155
0
        return NULL;
156
157
151k
    s->id = stream_id;
158
151k
    s->type = type;
159
151k
    s->as_server = ossl_quic_channel_is_server(qsm->ch);
160
151k
    s->send_state = (ossl_quic_stream_is_local_init(s)
161
146k
                        || ossl_quic_stream_is_bidi(s))
162
151k
        ? QUIC_SSTREAM_STATE_READY
163
151k
        : QUIC_SSTREAM_STATE_NONE;
164
151k
    s->recv_state = (!ossl_quic_stream_is_local_init(s)
165
5.03k
                        || ossl_quic_stream_is_bidi(s))
166
151k
        ? QUIC_RSTREAM_STATE_RECV
167
151k
        : QUIC_RSTREAM_STATE_NONE;
168
169
151k
    s->send_final_size = UINT64_MAX;
170
171
151k
    lh_QUIC_STREAM_insert(qsm->map, s);
172
151k
    return s;
173
151k
}
174
175
void ossl_quic_stream_map_release(QUIC_STREAM_MAP *qsm, QUIC_STREAM *stream)
176
158k
{
177
158k
    if (stream == NULL)
178
7.03k
        return;
179
180
151k
    if (stream->active_node.next != NULL)
181
12.7k
        list_remove(&qsm->active_list, &stream->active_node);
182
151k
    if (stream->accept_node.next != NULL)
183
139k
        list_remove(&qsm->accept_list, &stream->accept_node);
184
151k
    if (stream->ready_for_gc_node.next != NULL)
185
0
        list_remove(&qsm->ready_for_gc_list, &stream->ready_for_gc_node);
186
187
151k
    ossl_quic_sstream_free(stream->sstream);
188
151k
    stream->sstream = NULL;
189
190
151k
    ossl_quic_rstream_free(stream->rstream);
191
151k
    stream->rstream = NULL;
192
193
151k
    lh_QUIC_STREAM_delete(qsm->map, stream);
194
151k
    OPENSSL_free(stream);
195
151k
}
196
197
QUIC_STREAM *ossl_quic_stream_map_get_by_id(QUIC_STREAM_MAP *qsm,
198
    uint64_t stream_id)
199
17.6M
{
200
17.6M
    QUIC_STREAM key;
201
202
17.6M
    key.id = stream_id;
203
204
17.6M
    return lh_QUIC_STREAM_retrieve(qsm->map, &key);
205
17.6M
}
206
207
static void stream_map_mark_active(QUIC_STREAM_MAP *qsm, QUIC_STREAM *s)
208
98.5k
{
209
98.5k
    if (s->active)
210
74.0k
        return;
211
212
24.4k
    list_insert_tail(&qsm->active_list, &s->active_node);
213
214
24.4k
    if (qsm->rr_cur == NULL)
215
17.8k
        qsm->rr_cur = s;
216
217
24.4k
    s->active = 1;
218
24.4k
}
219
220
static void stream_map_mark_inactive(QUIC_STREAM_MAP *qsm, QUIC_STREAM *s)
221
1.08M
{
222
1.08M
    if (!s->active)
223
1.07M
        return;
224
225
11.7k
    if (qsm->rr_cur == s)
226
10.4k
        qsm->rr_cur = active_next(&qsm->active_list, s);
227
11.7k
    if (qsm->rr_cur == s)
228
9.02k
        qsm->rr_cur = NULL;
229
230
11.7k
    list_remove(&qsm->active_list, &s->active_node);
231
232
11.7k
    s->active = 0;
233
11.7k
}
234
235
void ossl_quic_stream_map_set_rr_stepping(QUIC_STREAM_MAP *qsm, size_t stepping)
236
0
{
237
0
    qsm->rr_stepping = stepping;
238
0
    qsm->rr_counter = 0;
239
0
}
240
241
static int stream_has_data_to_send(QUIC_STREAM *s)
242
1.08M
{
243
1.08M
    OSSL_QUIC_FRAME_STREAM shdr;
244
1.08M
    OSSL_QTX_IOVEC iov[2];
245
1.08M
    size_t num_iov;
246
1.08M
    uint64_t fc_credit, fc_swm, fc_limit;
247
248
1.08M
    switch (s->send_state) {
249
93.6k
    case QUIC_SSTREAM_STATE_READY:
250
138k
    case QUIC_SSTREAM_STATE_SEND:
251
138k
    case QUIC_SSTREAM_STATE_DATA_SENT:
252
        /*
253
         * We can still have data to send in DATA_SENT due to retransmissions,
254
         * etc.
255
         */
256
138k
        break;
257
950k
    default:
258
950k
        return 0; /* Nothing to send. */
259
1.08M
    }
260
261
    /*
262
     * We cannot determine if we have data to send simply by checking if
263
     * ossl_quic_txfc_get_credit() is zero, because we may also have older
264
     * stream data we need to retransmit. The SSTREAM returns older data first,
265
     * so we do a simple comparison of the next chunk the SSTREAM wants to send
266
     * against the TXFC CWM.
267
     */
268
138k
    num_iov = OSSL_NELEM(iov);
269
138k
    if (!ossl_quic_sstream_get_stream_frame(s->sstream, 0, &shdr, iov,
270
138k
            &num_iov))
271
104k
        return 0;
272
273
33.6k
    fc_credit = ossl_quic_txfc_get_credit(&s->txfc, 0);
274
33.6k
    fc_swm = ossl_quic_txfc_get_swm(&s->txfc);
275
33.6k
    fc_limit = fc_swm + fc_credit;
276
277
33.6k
    return (shdr.is_fin && shdr.len == 0) || shdr.offset < fc_limit;
278
138k
}
279
280
static ossl_unused int qsm_send_part_permits_gc(const QUIC_STREAM *qs)
281
0
{
282
0
    switch (qs->send_state) {
283
0
    case QUIC_SSTREAM_STATE_NONE:
284
0
    case QUIC_SSTREAM_STATE_DATA_RECVD:
285
0
    case QUIC_SSTREAM_STATE_RESET_RECVD:
286
0
        return 1;
287
0
    default:
288
0
        return 0;
289
0
    }
290
0
}
291
292
static int qsm_ready_for_gc(QUIC_STREAM_MAP *qsm, QUIC_STREAM *qs)
293
1.18M
{
294
1.18M
    int recv_stream_fully_drained = 0; /* TODO(QUIC FUTURE): Optimisation */
295
296
    /*
297
     * If sstream has no FIN, we auto-reset it at marked-for-deletion time, so
298
     * we don't need to worry about that here.
299
     */
300
1.18M
    assert(!qs->deleted
301
1.18M
        || !ossl_quic_stream_has_send(qs)
302
1.18M
        || ossl_quic_stream_send_is_reset(qs)
303
1.18M
        || ossl_quic_stream_send_get_final_size(qs, NULL));
304
305
1.18M
    return qs->deleted
306
12.5k
        && (!ossl_quic_stream_has_recv(qs)
307
12.5k
            || recv_stream_fully_drained
308
12.5k
            || qs->acked_stop_sending)
309
0
        && (!ossl_quic_stream_has_send(qs)
310
0
            || qs->send_state == QUIC_SSTREAM_STATE_DATA_RECVD
311
0
            || qs->send_state == QUIC_SSTREAM_STATE_RESET_RECVD);
312
1.18M
}
313
314
int ossl_quic_stream_map_is_local_allowed_by_stream_limit(QUIC_STREAM_MAP *qsm,
315
    uint64_t stream_ordinal,
316
    int is_uni)
317
70.5k
{
318
70.5k
    uint64_t stream_limit;
319
320
70.5k
    if (qsm->get_stream_limit_cb == NULL)
321
0
        return 1;
322
323
70.5k
    stream_limit = qsm->get_stream_limit_cb(is_uni, qsm->get_stream_limit_cb_arg);
324
70.5k
    return stream_ordinal < stream_limit;
325
70.5k
}
326
327
void ossl_quic_stream_map_update_state(QUIC_STREAM_MAP *qsm, QUIC_STREAM *s)
328
1.18M
{
329
1.18M
    int should_be_active, allowed_by_stream_limit = 1;
330
331
1.18M
    if (ossl_quic_stream_is_server_init(s) == ossl_quic_channel_is_server(qsm->ch)) {
332
62.0k
        int is_uni = !ossl_quic_stream_is_bidi(s);
333
62.0k
        uint64_t stream_ordinal = s->id >> 2;
334
335
62.0k
        allowed_by_stream_limit
336
62.0k
            = ossl_quic_stream_map_is_local_allowed_by_stream_limit(qsm,
337
62.0k
                stream_ordinal,
338
62.0k
                is_uni);
339
62.0k
    }
340
341
1.18M
    if (s->send_state == QUIC_SSTREAM_STATE_DATA_SENT
342
0
        && ossl_quic_sstream_is_totally_acked(s->sstream))
343
0
        ossl_quic_stream_map_notify_totally_acked(qsm, s);
344
1.18M
    else if (s->shutdown_flush
345
0
        && s->send_state == QUIC_SSTREAM_STATE_SEND
346
0
        && ossl_quic_sstream_is_totally_acked(s->sstream))
347
0
        shutdown_flush_done(qsm, s);
348
349
1.18M
    if (!s->ready_for_gc) {
350
1.18M
        s->ready_for_gc = qsm_ready_for_gc(qsm, s);
351
1.18M
        if (s->ready_for_gc)
352
0
            list_insert_tail(&qsm->ready_for_gc_list, &s->ready_for_gc_node);
353
1.18M
    }
354
355
1.18M
    should_be_active
356
1.18M
        = allowed_by_stream_limit
357
1.18M
        && !s->ready_for_gc
358
1.18M
        && ((ossl_quic_stream_has_recv(s)
359
1.18M
                && !ossl_quic_stream_recv_is_reset(s)
360
1.14M
                && (s->recv_state == QUIC_RSTREAM_STATE_RECV
361
1.08M
                    && (s->want_max_stream_data
362
1.06M
                        || ossl_quic_rxfc_has_cwm_changed(&s->rxfc, 0))))
363
1.17M
            || s->want_stop_sending
364
1.14M
            || s->want_reset_stream
365
1.11M
            || (!s->peer_stop_sending && stream_has_data_to_send(s)));
366
367
1.18M
    if (should_be_active)
368
98.5k
        stream_map_mark_active(qsm, s);
369
1.08M
    else
370
1.08M
        stream_map_mark_inactive(qsm, s);
371
1.18M
}
372
373
/*
374
 * Stream Send Part State Management
375
 * =================================
376
 */
377
378
int ossl_quic_stream_map_ensure_send_part_id(QUIC_STREAM_MAP *qsm,
379
    QUIC_STREAM *qs)
380
17.9k
{
381
17.9k
    switch (qs->send_state) {
382
0
    case QUIC_SSTREAM_STATE_NONE:
383
        /* Stream without send part - caller error. */
384
0
        return 0;
385
386
17.9k
    case QUIC_SSTREAM_STATE_READY:
387
        /*
388
         * We always allocate a stream ID upfront, so we don't need to do it
389
         * here.
390
         */
391
17.9k
        qs->send_state = QUIC_SSTREAM_STATE_SEND;
392
17.9k
        return 1;
393
394
0
    default:
395
        /* Nothing to do. */
396
0
        return 1;
397
17.9k
    }
398
17.9k
}
399
400
int ossl_quic_stream_map_notify_all_data_sent(QUIC_STREAM_MAP *qsm,
401
    QUIC_STREAM *qs)
402
0
{
403
0
    switch (qs->send_state) {
404
0
    default:
405
        /* Wrong state - caller error. */
406
0
    case QUIC_SSTREAM_STATE_NONE:
407
        /* Stream without send part - caller error. */
408
0
        return 0;
409
410
0
    case QUIC_SSTREAM_STATE_SEND:
411
0
        if (!ossl_quic_sstream_get_final_size(qs->sstream, &qs->send_final_size))
412
0
            return 0;
413
414
0
        qs->send_state = QUIC_SSTREAM_STATE_DATA_SENT;
415
0
        return 1;
416
0
    }
417
0
}
418
419
static void shutdown_flush_done(QUIC_STREAM_MAP *qsm, QUIC_STREAM *qs)
420
7.02k
{
421
7.02k
    if (!qs->shutdown_flush)
422
7.02k
        return;
423
424
7.02k
    assert(qsm->num_shutdown_flush > 0);
425
0
    qs->shutdown_flush = 0;
426
0
    --qsm->num_shutdown_flush;
427
428
    /*
429
     * when num_shutdown_flush becomes zero we need to poke
430
     * SSL_poll() it's time to poke to SSL_shutdown() to proceed
431
     * with shutdown process as all streams are gone (flushed).
432
     */
433
0
    if (qsm->num_shutdown_flush == 0)
434
0
        ossl_quic_channel_notify_flush_done(qsm->ch);
435
0
}
436
437
int ossl_quic_stream_map_notify_totally_acked(QUIC_STREAM_MAP *qsm,
438
    QUIC_STREAM *qs)
439
0
{
440
0
    switch (qs->send_state) {
441
0
    default:
442
        /* Wrong state - caller error. */
443
0
    case QUIC_SSTREAM_STATE_NONE:
444
        /* Stream without send part - caller error. */
445
0
        return 0;
446
447
0
    case QUIC_SSTREAM_STATE_DATA_SENT:
448
0
        qs->send_state = QUIC_SSTREAM_STATE_DATA_RECVD;
449
        /* We no longer need a QUIC_SSTREAM in this state. */
450
0
        ossl_quic_sstream_free(qs->sstream);
451
0
        qs->sstream = NULL;
452
453
0
        shutdown_flush_done(qsm, qs);
454
0
        return 1;
455
0
    }
456
0
}
457
458
int ossl_quic_stream_map_reset_stream_send_part(QUIC_STREAM_MAP *qsm,
459
    QUIC_STREAM *qs,
460
    uint64_t aec)
461
103k
{
462
103k
    switch (qs->send_state) {
463
0
    default:
464
0
    case QUIC_SSTREAM_STATE_NONE:
465
        /*
466
         * RESET_STREAM pertains to sending part only, so we cannot reset a
467
         * receive-only stream.
468
         */
469
0
    case QUIC_SSTREAM_STATE_DATA_RECVD:
470
        /*
471
         * RFC 9000 s. 3.3: A sender MUST NOT [...] send RESET_STREAM from a
472
         * terminal state. If the stream has already finished normally and the
473
         * peer has acknowledged this, we cannot reset it.
474
         */
475
0
        return 0;
476
477
14.5k
    case QUIC_SSTREAM_STATE_READY:
478
14.5k
        if (!ossl_quic_stream_map_ensure_send_part_id(qsm, qs))
479
0
            return 0;
480
481
        /* FALLTHROUGH */
482
17.9k
    case QUIC_SSTREAM_STATE_SEND:
483
        /*
484
         * If we already have a final size (e.g. because we are coming from
485
         * DATA_SENT), we have to be consistent with that, so don't change it.
486
         * If we don't already have a final size, determine a final size value.
487
         * This is the value which we will end up using for a RESET_STREAM frame
488
         * for flow control purposes. We could send the stream size (total
489
         * number of bytes appended to QUIC_SSTREAM by the application), but it
490
         * is in our interest to exclude any bytes we have not actually
491
         * transmitted yet, to avoid unnecessarily consuming flow control
492
         * credit. We can get this from the TXFC.
493
         */
494
17.9k
        qs->send_final_size = ossl_quic_txfc_get_swm(&qs->txfc);
495
496
        /* FALLTHROUGH */
497
17.9k
    case QUIC_SSTREAM_STATE_DATA_SENT:
498
17.9k
        qs->reset_stream_aec = aec;
499
17.9k
        qs->want_reset_stream = 1;
500
17.9k
        qs->send_state = QUIC_SSTREAM_STATE_RESET_SENT;
501
502
17.9k
        ossl_quic_sstream_free(qs->sstream);
503
17.9k
        qs->sstream = NULL;
504
505
17.9k
        shutdown_flush_done(qsm, qs);
506
17.9k
        ossl_quic_stream_map_update_state(qsm, qs);
507
17.9k
        return 1;
508
509
85.2k
    case QUIC_SSTREAM_STATE_RESET_SENT:
510
85.2k
    case QUIC_SSTREAM_STATE_RESET_RECVD:
511
        /*
512
         * Idempotent - no-op. In any case, do not send RESET_STREAM again - as
513
         * mentioned, we must not send it from a terminal state.
514
         */
515
85.2k
        return 1;
516
103k
    }
517
103k
}
518
519
int ossl_quic_stream_map_notify_reset_stream_acked(QUIC_STREAM_MAP *qsm,
520
    QUIC_STREAM *qs)
521
0
{
522
0
    switch (qs->send_state) {
523
0
    default:
524
        /* Wrong state - caller error. */
525
0
    case QUIC_SSTREAM_STATE_NONE:
526
        /* Stream without send part - caller error. */
527
0
        return 0;
528
529
0
    case QUIC_SSTREAM_STATE_RESET_SENT:
530
0
        qs->send_state = QUIC_SSTREAM_STATE_RESET_RECVD;
531
0
        return 1;
532
533
0
    case QUIC_SSTREAM_STATE_RESET_RECVD:
534
        /* Already in the correct state. */
535
0
        return 1;
536
0
    }
537
0
}
538
539
/*
540
 * Stream Receive Part State Management
541
 * ====================================
542
 */
543
544
int ossl_quic_stream_map_notify_size_known_recv_part(QUIC_STREAM_MAP *qsm,
545
    QUIC_STREAM *qs,
546
    uint64_t final_size)
547
8.30k
{
548
8.30k
    switch (qs->recv_state) {
549
0
    default:
550
        /* Wrong state - caller error. */
551
0
    case QUIC_RSTREAM_STATE_NONE:
552
        /* Stream without receive part - caller error. */
553
0
        return 0;
554
555
8.30k
    case QUIC_RSTREAM_STATE_RECV:
556
8.30k
        qs->recv_state = QUIC_RSTREAM_STATE_SIZE_KNOWN;
557
8.30k
        return 1;
558
8.30k
    }
559
8.30k
}
560
561
int ossl_quic_stream_map_notify_totally_received(QUIC_STREAM_MAP *qsm,
562
    QUIC_STREAM *qs)
563
3.20k
{
564
3.20k
    switch (qs->recv_state) {
565
0
    default:
566
        /* Wrong state - caller error. */
567
0
    case QUIC_RSTREAM_STATE_NONE:
568
        /* Stream without receive part - caller error. */
569
0
        return 0;
570
571
3.20k
    case QUIC_RSTREAM_STATE_SIZE_KNOWN:
572
3.20k
        qs->recv_state = QUIC_RSTREAM_STATE_DATA_RECVD;
573
3.20k
        qs->want_stop_sending = 0;
574
3.20k
        return 1;
575
3.20k
    }
576
3.20k
}
577
578
int ossl_quic_stream_map_notify_totally_read(QUIC_STREAM_MAP *qsm,
579
    QUIC_STREAM *qs)
580
678
{
581
678
    switch (qs->recv_state) {
582
0
    default:
583
        /* Wrong state - caller error. */
584
0
    case QUIC_RSTREAM_STATE_NONE:
585
        /* Stream without receive part - caller error. */
586
0
        return 0;
587
588
678
    case QUIC_RSTREAM_STATE_DATA_RECVD:
589
678
        qs->recv_state = QUIC_RSTREAM_STATE_DATA_READ;
590
591
        /* QUIC_RSTREAM is no longer needed */
592
678
        ossl_quic_rstream_free(qs->rstream);
593
678
        qs->rstream = NULL;
594
678
        return 1;
595
678
    }
596
678
}
597
598
int ossl_quic_stream_map_notify_reset_recv_part(QUIC_STREAM_MAP *qsm,
599
    QUIC_STREAM *qs,
600
    uint64_t app_error_code,
601
    uint64_t final_size)
602
8.06k
{
603
8.06k
    uint64_t prev_final_size;
604
605
8.06k
    switch (qs->recv_state) {
606
0
    default:
607
0
    case QUIC_RSTREAM_STATE_NONE:
608
        /* Stream without receive part - caller error. */
609
0
        return 0;
610
611
1.91k
    case QUIC_RSTREAM_STATE_RECV:
612
1.91k
    case QUIC_RSTREAM_STATE_SIZE_KNOWN:
613
1.93k
    case QUIC_RSTREAM_STATE_DATA_RECVD:
614
1.93k
        if (ossl_quic_stream_recv_get_final_size(qs, &prev_final_size)
615
25
            && prev_final_size != final_size)
616
            /* Cannot change previous final size. */
617
0
            return 0;
618
619
1.93k
        qs->recv_state = QUIC_RSTREAM_STATE_RESET_RECVD;
620
1.93k
        qs->peer_reset_stream_aec = app_error_code;
621
622
        /* RFC 9000 s. 3.3: No point sending STOP_SENDING if already reset. */
623
1.93k
        qs->want_stop_sending = 0;
624
625
        /* QUIC_RSTREAM is no longer needed */
626
1.93k
        ossl_quic_rstream_free(qs->rstream);
627
1.93k
        qs->rstream = NULL;
628
629
1.93k
        ossl_quic_stream_map_update_state(qsm, qs);
630
1.93k
        return 1;
631
632
0
    case QUIC_RSTREAM_STATE_DATA_READ:
633
        /*
634
         * If we already retired the FIN to the application this is moot
635
         * - just ignore.
636
         */
637
6.13k
    case QUIC_RSTREAM_STATE_RESET_RECVD:
638
6.13k
    case QUIC_RSTREAM_STATE_RESET_READ:
639
        /* Could be a reordered/retransmitted frame - just ignore. */
640
6.13k
        return 1;
641
8.06k
    }
642
8.06k
}
643
644
int ossl_quic_stream_map_notify_app_read_reset_recv_part(QUIC_STREAM_MAP *qsm,
645
    QUIC_STREAM *qs)
646
195
{
647
195
    switch (qs->recv_state) {
648
0
    default:
649
        /* Wrong state - caller error. */
650
0
    case QUIC_RSTREAM_STATE_NONE:
651
        /* Stream without receive part - caller error. */
652
0
        return 0;
653
654
195
    case QUIC_RSTREAM_STATE_RESET_RECVD:
655
195
        qs->recv_state = QUIC_RSTREAM_STATE_RESET_READ;
656
195
        return 1;
657
195
    }
658
195
}
659
660
int ossl_quic_stream_map_stop_sending_recv_part(QUIC_STREAM_MAP *qsm,
661
    QUIC_STREAM *qs,
662
    uint64_t aec)
663
11.3k
{
664
11.3k
    if (qs->stop_sending)
665
0
        return 0;
666
667
11.3k
    switch (qs->recv_state) {
668
0
    default:
669
0
    case QUIC_RSTREAM_STATE_NONE:
670
        /* Send-only stream, so this makes no sense. */
671
0
    case QUIC_RSTREAM_STATE_DATA_RECVD:
672
0
    case QUIC_RSTREAM_STATE_DATA_READ:
673
        /*
674
         * Not really any point in STOP_SENDING if we already received all data.
675
         */
676
0
    case QUIC_RSTREAM_STATE_RESET_RECVD:
677
0
    case QUIC_RSTREAM_STATE_RESET_READ:
678
        /*
679
         * RFC 9000 s. 3.5: "STOP_SENDING SHOULD only be sent for a stream that
680
         * has not been reset by the peer."
681
         *
682
         * No point in STOP_SENDING if the peer already reset their send part.
683
         */
684
0
        return 0;
685
686
9.48k
    case QUIC_RSTREAM_STATE_RECV:
687
11.3k
    case QUIC_RSTREAM_STATE_SIZE_KNOWN:
688
        /*
689
         * RFC 9000 s. 3.5: "If the stream is in the Recv or Size Known state,
690
         * the transport SHOULD signal this by sending a STOP_SENDING frame to
691
         * prompt closure of the stream in the opposite direction."
692
         *
693
         * Note that it does make sense to send STOP_SENDING for a receive part
694
         * of a stream which has a known size (because we have received a FIN)
695
         * but which still has other (previous) stream data yet to be received.
696
         */
697
11.3k
        break;
698
11.3k
    }
699
700
11.3k
    qs->stop_sending = 1;
701
11.3k
    qs->stop_sending_aec = aec;
702
11.3k
    return ossl_quic_stream_map_schedule_stop_sending(qsm, qs);
703
11.3k
}
704
705
/* Called to mark STOP_SENDING for generation, or regeneration after loss. */
706
int ossl_quic_stream_map_schedule_stop_sending(QUIC_STREAM_MAP *qsm, QUIC_STREAM *qs)
707
11.3k
{
708
11.3k
    if (!qs->stop_sending)
709
0
        return 0;
710
711
    /*
712
     * Ignore the call as a no-op if already scheduled, or in a state
713
     * where it makes no sense to send STOP_SENDING.
714
     */
715
11.3k
    if (qs->want_stop_sending)
716
0
        return 1;
717
718
11.3k
    switch (qs->recv_state) {
719
0
    default:
720
0
        return 1; /* ignore */
721
9.48k
    case QUIC_RSTREAM_STATE_RECV:
722
11.3k
    case QUIC_RSTREAM_STATE_SIZE_KNOWN:
723
        /*
724
         * RFC 9000 s. 3.5: "An endpoint is expected to send another
725
         * STOP_SENDING frame if a packet containing a previous STOP_SENDING is
726
         * lost. However, once either all stream data or a RESET_STREAM frame
727
         * has been received for the stream -- that is, the stream is in any
728
         * state other than "Recv" or "Size Known" -- sending a STOP_SENDING
729
         * frame is unnecessary."
730
         */
731
11.3k
        break;
732
11.3k
    }
733
734
11.3k
    qs->want_stop_sending = 1;
735
11.3k
    ossl_quic_stream_map_update_state(qsm, qs);
736
11.3k
    return 1;
737
11.3k
}
738
739
QUIC_STREAM *ossl_quic_stream_map_peek_accept_queue(QUIC_STREAM_MAP *qsm)
740
510
{
741
510
    return accept_head(&qsm->accept_list);
742
510
}
743
744
QUIC_STREAM *ossl_quic_stream_map_find_in_accept_queue(QUIC_STREAM_MAP *qsm,
745
    int is_uni)
746
0
{
747
0
    QUIC_STREAM *qs;
748
749
0
    if (ossl_quic_stream_map_get_accept_queue_len(qsm, is_uni) == 0)
750
0
        return NULL;
751
752
0
    qs = ossl_quic_stream_map_peek_accept_queue(qsm);
753
0
    while (qs != NULL) {
754
0
        if ((is_uni && !ossl_quic_stream_is_bidi(qs))
755
0
            || (!is_uni && ossl_quic_stream_is_bidi(qs)))
756
0
            break;
757
0
        qs = accept_next(&qsm->accept_list, qs);
758
0
    }
759
0
    return qs;
760
0
}
761
762
void ossl_quic_stream_map_push_accept_queue(QUIC_STREAM_MAP *qsm,
763
    QUIC_STREAM *s)
764
146k
{
765
146k
    list_insert_tail(&qsm->accept_list, &s->accept_node);
766
146k
    if (ossl_quic_stream_is_bidi(s))
767
39.3k
        ++qsm->num_accept_bidi;
768
107k
    else
769
107k
        ++qsm->num_accept_uni;
770
146k
}
771
772
static QUIC_RXFC *qsm_get_max_streams_rxfc(QUIC_STREAM_MAP *qsm, QUIC_STREAM *s)
773
7.48k
{
774
7.48k
    return ossl_quic_stream_is_bidi(s)
775
7.48k
        ? qsm->max_streams_bidi_rxfc
776
7.48k
        : qsm->max_streams_uni_rxfc;
777
7.48k
}
778
779
void ossl_quic_stream_map_remove_from_accept_queue(QUIC_STREAM_MAP *qsm,
780
    QUIC_STREAM *s,
781
    OSSL_TIME rtt)
782
7.48k
{
783
7.48k
    QUIC_RXFC *max_streams_rxfc;
784
785
7.48k
    list_remove(&qsm->accept_list, &s->accept_node);
786
7.48k
    if (ossl_quic_stream_is_bidi(s))
787
6.84k
        --qsm->num_accept_bidi;
788
640
    else
789
640
        --qsm->num_accept_uni;
790
791
7.48k
    if ((max_streams_rxfc = qsm_get_max_streams_rxfc(qsm, s)) != NULL)
792
7.48k
        (void)ossl_quic_rxfc_on_retire(max_streams_rxfc, 1, rtt);
793
7.48k
}
794
795
size_t ossl_quic_stream_map_get_accept_queue_len(QUIC_STREAM_MAP *qsm, int is_uni)
796
23.2k
{
797
23.2k
    return is_uni ? qsm->num_accept_uni : qsm->num_accept_bidi;
798
23.2k
}
799
800
size_t ossl_quic_stream_map_get_total_accept_queue_len(QUIC_STREAM_MAP *qsm)
801
11.6k
{
802
11.6k
    return ossl_quic_stream_map_get_accept_queue_len(qsm, /*is_uni=*/0)
803
11.6k
        + ossl_quic_stream_map_get_accept_queue_len(qsm, /*is_uni=*/1);
804
11.6k
}
805
806
void ossl_quic_stream_map_gc(QUIC_STREAM_MAP *qsm)
807
72.9M
{
808
72.9M
    QUIC_STREAM *qs;
809
810
72.9M
    while ((qs = ready_for_gc_head(&qsm->ready_for_gc_list)) != NULL) {
811
0
        ossl_quic_stream_map_release(qsm, qs);
812
0
    }
813
72.9M
}
814
815
static int eligible_for_shutdown_flush(QUIC_STREAM *qs)
816
0
{
817
    /*
818
     * We only care about servicing the send part of a stream (if any) during
819
     * shutdown flush. We make sure we flush a stream if it is either
820
     * non-terminated or was terminated normally such as via
821
     * SSL_stream_conclude. A stream which was terminated via a reset is not
822
     * flushed, and we will have thrown away the send buffer in that case
823
     * anyway.
824
     */
825
0
    switch (qs->send_state) {
826
0
    case QUIC_SSTREAM_STATE_SEND:
827
0
    case QUIC_SSTREAM_STATE_DATA_SENT:
828
0
        return !ossl_quic_sstream_is_totally_acked(qs->sstream);
829
0
    default:
830
0
        return 0;
831
0
    }
832
0
}
833
834
static void begin_shutdown_flush_each(QUIC_STREAM *qs, void *arg)
835
0
{
836
0
    QUIC_STREAM_MAP *qsm = arg;
837
838
0
    if (!eligible_for_shutdown_flush(qs) || qs->shutdown_flush)
839
0
        return;
840
841
0
    qs->shutdown_flush = 1;
842
0
    ++qsm->num_shutdown_flush;
843
0
}
844
845
void ossl_quic_stream_map_begin_shutdown_flush(QUIC_STREAM_MAP *qsm)
846
0
{
847
0
    qsm->num_shutdown_flush = 0;
848
849
0
    ossl_quic_stream_map_visit(qsm, begin_shutdown_flush_each, qsm);
850
0
}
851
852
int ossl_quic_stream_map_is_shutdown_flush_finished(QUIC_STREAM_MAP *qsm)
853
0
{
854
0
    return qsm->num_shutdown_flush == 0;
855
0
}
856
857
/*
858
 * QUIC Stream Iterator
859
 * ====================
860
 */
861
void ossl_quic_stream_iter_init(QUIC_STREAM_ITER *it, QUIC_STREAM_MAP *qsm,
862
    int advance_rr)
863
9.29M
{
864
9.29M
    it->qsm = qsm;
865
9.29M
    it->stream = it->first_stream = qsm->rr_cur;
866
9.29M
    if (advance_rr && it->stream != NULL
867
14.6k
        && ++qsm->rr_counter >= qsm->rr_stepping) {
868
14.6k
        qsm->rr_counter = 0;
869
14.6k
        qsm->rr_cur = active_next(&qsm->active_list, qsm->rr_cur);
870
14.6k
    }
871
9.29M
}
872
873
void ossl_quic_stream_iter_next(QUIC_STREAM_ITER *it)
874
18.2k
{
875
18.2k
    if (it->stream == NULL)
876
0
        return;
877
878
18.2k
    it->stream = active_next(&it->qsm->active_list, it->stream);
879
18.2k
    if (it->stream == it->first_stream)
880
14.4k
        it->stream = NULL;
881
18.2k
}