Coverage Report

Created: 2025-08-28 07:07

/src/openssl33/ssl/quic/quic_fifd.c
Line
Count
Source (jump to first uncovered line)
1
/*
2
 * Copyright 2022-2024 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_fifd.h"
11
#include "internal/quic_wire.h"
12
#include "internal/qlog_event_helpers.h"
13
14
DEFINE_LIST_OF(tx_history, OSSL_ACKM_TX_PKT);
15
16
int ossl_quic_fifd_init(QUIC_FIFD *fifd,
17
                        QUIC_CFQ *cfq,
18
                        OSSL_ACKM *ackm,
19
                        QUIC_TXPIM *txpim,
20
                        /* stream_id is UINT64_MAX for the crypto stream */
21
                        QUIC_SSTREAM *(*get_sstream_by_id)(uint64_t stream_id,
22
                                                           uint32_t pn_space,
23
                                                           void *arg),
24
                        void *get_sstream_by_id_arg,
25
                        /* stream_id is UINT64_MAX if not applicable */
26
                        void (*regen_frame)(uint64_t frame_type,
27
                                            uint64_t stream_id,
28
                                            QUIC_TXPIM_PKT *pkt,
29
                                            void *arg),
30
                        void *regen_frame_arg,
31
                        void (*confirm_frame)(uint64_t frame_type,
32
                                              uint64_t stream_id,
33
                                              QUIC_TXPIM_PKT *pkt,
34
                                              void *arg),
35
                        void *confirm_frame_arg,
36
                        void (*sstream_updated)(uint64_t stream_id,
37
                                                void *arg),
38
                        void *sstream_updated_arg,
39
                        QLOG *(*get_qlog_cb)(void *arg),
40
                        void *get_qlog_cb_arg)
41
48.1k
{
42
48.1k
    if (cfq == NULL || ackm == NULL || txpim == NULL
43
48.1k
        || get_sstream_by_id == NULL || regen_frame == NULL)
44
0
        return 0;
45
46
48.1k
    fifd->cfq                   = cfq;
47
48.1k
    fifd->ackm                  = ackm;
48
48.1k
    fifd->txpim                 = txpim;
49
48.1k
    fifd->get_sstream_by_id     = get_sstream_by_id;
50
48.1k
    fifd->get_sstream_by_id_arg = get_sstream_by_id_arg;
51
48.1k
    fifd->regen_frame           = regen_frame;
52
48.1k
    fifd->regen_frame_arg       = regen_frame_arg;
53
48.1k
    fifd->confirm_frame         = confirm_frame;
54
48.1k
    fifd->confirm_frame_arg     = confirm_frame_arg;
55
48.1k
    fifd->sstream_updated       = sstream_updated;
56
48.1k
    fifd->sstream_updated_arg   = sstream_updated_arg;
57
48.1k
    fifd->get_qlog_cb           = get_qlog_cb;
58
48.1k
    fifd->get_qlog_cb_arg       = get_qlog_cb_arg;
59
48.1k
    return 1;
60
48.1k
}
61
62
void ossl_quic_fifd_cleanup(QUIC_FIFD *fifd)
63
48.1k
{
64
    /* No-op. */
65
48.1k
}
66
67
static void on_acked(void *arg)
68
510k
{
69
510k
    QUIC_TXPIM_PKT *pkt = arg;
70
510k
    QUIC_FIFD *fifd = pkt->fifd;
71
510k
    const QUIC_TXPIM_CHUNK *chunks = ossl_quic_txpim_pkt_get_chunks(pkt);
72
510k
    size_t i, num_chunks = ossl_quic_txpim_pkt_get_num_chunks(pkt);
73
510k
    QUIC_SSTREAM *sstream;
74
510k
    QUIC_CFQ_ITEM *cfq_item, *cfq_item_next;
75
76
    /* STREAM and CRYPTO stream chunks, FINs and stream FC frames */
77
543k
    for (i = 0; i < num_chunks; ++i) {
78
32.4k
        sstream = fifd->get_sstream_by_id(chunks[i].stream_id,
79
32.4k
                                          pkt->ackm_pkt.pkt_space,
80
32.4k
                                          fifd->get_sstream_by_id_arg);
81
32.4k
        if (sstream == NULL)
82
790
            continue;
83
84
31.6k
        if (chunks[i].end >= chunks[i].start)
85
            /* coverity[check_return]: Best effort - we cannot fail here. */
86
31.6k
            ossl_quic_sstream_mark_acked(sstream,
87
31.6k
                                         chunks[i].start, chunks[i].end);
88
89
31.6k
        if (chunks[i].has_fin && chunks[i].stream_id != UINT64_MAX)
90
0
            ossl_quic_sstream_mark_acked_fin(sstream);
91
92
31.6k
        if (chunks[i].has_stop_sending && chunks[i].stream_id != UINT64_MAX)
93
0
            fifd->confirm_frame(OSSL_QUIC_FRAME_TYPE_STOP_SENDING,
94
0
                                chunks[i].stream_id, pkt,
95
0
                                fifd->confirm_frame_arg);
96
97
31.6k
        if (chunks[i].has_reset_stream && chunks[i].stream_id != UINT64_MAX)
98
0
            fifd->confirm_frame(OSSL_QUIC_FRAME_TYPE_RESET_STREAM,
99
0
                                chunks[i].stream_id, pkt,
100
0
                                fifd->confirm_frame_arg);
101
102
31.6k
        if (ossl_quic_sstream_is_totally_acked(sstream))
103
19.2k
            fifd->sstream_updated(chunks[i].stream_id, fifd->sstream_updated_arg);
104
31.6k
    }
105
106
    /* GCR */
107
528k
    for (cfq_item = pkt->retx_head; cfq_item != NULL; cfq_item = cfq_item_next) {
108
17.7k
        cfq_item_next = cfq_item->pkt_next;
109
17.7k
        ossl_quic_cfq_release(fifd->cfq, cfq_item);
110
17.7k
    }
111
112
510k
    ossl_quic_txpim_pkt_release(fifd->txpim, pkt);
113
510k
}
114
115
static QLOG *fifd_get_qlog(QUIC_FIFD *fifd)
116
112k
{
117
112k
    if (fifd->get_qlog_cb == NULL)
118
0
        return NULL;
119
120
112k
    return fifd->get_qlog_cb(fifd->get_qlog_cb_arg);
121
112k
}
122
123
static void on_lost(void *arg)
124
134k
{
125
134k
    QUIC_TXPIM_PKT *pkt = arg;
126
134k
    QUIC_FIFD *fifd = pkt->fifd;
127
134k
    const QUIC_TXPIM_CHUNK *chunks = ossl_quic_txpim_pkt_get_chunks(pkt);
128
134k
    size_t i, num_chunks = ossl_quic_txpim_pkt_get_num_chunks(pkt);
129
134k
    QUIC_SSTREAM *sstream;
130
134k
    QUIC_CFQ_ITEM *cfq_item, *cfq_item_next;
131
134k
    int sstream_updated;
132
133
134k
    ossl_qlog_event_recovery_packet_lost(fifd_get_qlog(fifd), pkt);
134
135
    /* STREAM and CRYPTO stream chunks, FIN and stream FC frames */
136
144k
    for (i = 0; i < num_chunks; ++i) {
137
10.2k
        sstream = fifd->get_sstream_by_id(chunks[i].stream_id,
138
10.2k
                                          pkt->ackm_pkt.pkt_space,
139
10.2k
                                          fifd->get_sstream_by_id_arg);
140
10.2k
        if (sstream == NULL)
141
1.55k
            continue;
142
143
8.70k
        sstream_updated = 0;
144
145
8.70k
        if (chunks[i].end >= chunks[i].start) {
146
            /*
147
             * Note: If the stream is being reset, we do not need to retransmit
148
             * old data as this is pointless. In this case this will be handled
149
             * by (sstream == NULL) above as the QSM will free the QUIC_SSTREAM
150
             * and our call to get_sstream_by_id above will return NULL.
151
             */
152
8.70k
            ossl_quic_sstream_mark_lost(sstream,
153
8.70k
                                        chunks[i].start, chunks[i].end);
154
8.70k
            sstream_updated = 1;
155
8.70k
        }
156
157
8.70k
        if (chunks[i].has_fin && chunks[i].stream_id != UINT64_MAX) {
158
0
            ossl_quic_sstream_mark_lost_fin(sstream);
159
0
            sstream_updated = 1;
160
0
        }
161
162
8.70k
        if (chunks[i].has_stop_sending && chunks[i].stream_id != UINT64_MAX)
163
0
            fifd->regen_frame(OSSL_QUIC_FRAME_TYPE_STOP_SENDING,
164
0
                              chunks[i].stream_id, pkt,
165
0
                              fifd->regen_frame_arg);
166
167
8.70k
        if (chunks[i].has_reset_stream && chunks[i].stream_id != UINT64_MAX)
168
0
            fifd->regen_frame(OSSL_QUIC_FRAME_TYPE_RESET_STREAM,
169
0
                              chunks[i].stream_id, pkt,
170
0
                              fifd->regen_frame_arg);
171
172
        /*
173
         * Inform caller that stream needs an FC frame.
174
         *
175
         * Note: We could track whether an FC frame was sent originally for the
176
         * stream to determine if it really needs to be regenerated or not.
177
         * However, if loss has occurred, it's probably better to ensure the
178
         * peer has up-to-date flow control data for the stream. Given that
179
         * these frames are extremely small, we may as well always send it when
180
         * handling loss.
181
         */
182
8.70k
        fifd->regen_frame(OSSL_QUIC_FRAME_TYPE_MAX_STREAM_DATA,
183
8.70k
                          chunks[i].stream_id,
184
8.70k
                          pkt,
185
8.70k
                          fifd->regen_frame_arg);
186
187
8.70k
        if (sstream_updated && chunks[i].stream_id != UINT64_MAX)
188
1.84k
            fifd->sstream_updated(chunks[i].stream_id,
189
1.84k
                                  fifd->sstream_updated_arg);
190
8.70k
    }
191
192
    /* GCR */
193
175k
    for (cfq_item = pkt->retx_head; cfq_item != NULL; cfq_item = cfq_item_next) {
194
41.2k
        cfq_item_next = cfq_item->pkt_next;
195
41.2k
        ossl_quic_cfq_mark_lost(fifd->cfq, cfq_item, UINT32_MAX);
196
41.2k
    }
197
198
    /* Regenerate flag frames */
199
134k
    if (pkt->had_handshake_done_frame)
200
0
        fifd->regen_frame(OSSL_QUIC_FRAME_TYPE_HANDSHAKE_DONE,
201
0
                          UINT64_MAX, pkt,
202
0
                          fifd->regen_frame_arg);
203
204
134k
    if (pkt->had_max_data_frame)
205
0
        fifd->regen_frame(OSSL_QUIC_FRAME_TYPE_MAX_DATA,
206
0
                          UINT64_MAX, pkt,
207
0
                          fifd->regen_frame_arg);
208
209
134k
    if (pkt->had_max_streams_bidi_frame)
210
0
        fifd->regen_frame(OSSL_QUIC_FRAME_TYPE_MAX_STREAMS_BIDI,
211
0
                          UINT64_MAX, pkt,
212
0
                          fifd->regen_frame_arg);
213
214
134k
    if (pkt->had_max_streams_uni_frame)
215
0
        fifd->regen_frame(OSSL_QUIC_FRAME_TYPE_MAX_STREAMS_UNI,
216
0
                          UINT64_MAX, pkt,
217
0
                          fifd->regen_frame_arg);
218
219
134k
    if (pkt->had_ack_frame)
220
        /*
221
         * We always use the ACK_WITH_ECN frame type to represent the ACK frame
222
         * type in our callback; we assume it is the caller's job to decide
223
         * whether it wants to send ECN data or not.
224
         */
225
13.9k
        fifd->regen_frame(OSSL_QUIC_FRAME_TYPE_ACK_WITH_ECN,
226
13.9k
                          UINT64_MAX, pkt,
227
13.9k
                          fifd->regen_frame_arg);
228
229
134k
    ossl_quic_txpim_pkt_release(fifd->txpim, pkt);
230
134k
}
231
232
static void on_discarded(void *arg)
233
797k
{
234
797k
    QUIC_TXPIM_PKT *pkt = arg;
235
797k
    QUIC_FIFD *fifd = pkt->fifd;
236
797k
    QUIC_CFQ_ITEM *cfq_item, *cfq_item_next;
237
238
    /*
239
     * Don't need to do anything to SSTREAMs for STREAM and CRYPTO streams, as
240
     * we assume caller will clean them up.
241
     */
242
243
    /* GCR */
244
936k
    for (cfq_item = pkt->retx_head; cfq_item != NULL; cfq_item = cfq_item_next) {
245
139k
        cfq_item_next = cfq_item->pkt_next;
246
139k
        ossl_quic_cfq_release(fifd->cfq, cfq_item);
247
139k
    }
248
249
797k
    ossl_quic_txpim_pkt_release(fifd->txpim, pkt);
250
797k
}
251
252
int ossl_quic_fifd_pkt_commit(QUIC_FIFD *fifd, QUIC_TXPIM_PKT *pkt)
253
1.44M
{
254
1.44M
    QUIC_CFQ_ITEM *cfq_item;
255
1.44M
    const QUIC_TXPIM_CHUNK *chunks;
256
1.44M
    size_t i, num_chunks;
257
1.44M
    QUIC_SSTREAM *sstream;
258
259
1.44M
    pkt->fifd                   = fifd;
260
261
1.44M
    pkt->ackm_pkt.on_lost       = on_lost;
262
1.44M
    pkt->ackm_pkt.on_acked      = on_acked;
263
1.44M
    pkt->ackm_pkt.on_discarded  = on_discarded;
264
1.44M
    pkt->ackm_pkt.cb_arg        = pkt;
265
266
1.44M
    ossl_list_tx_history_init_elem(&pkt->ackm_pkt);
267
1.44M
    pkt->ackm_pkt.anext = pkt->ackm_pkt.lnext = NULL;
268
269
    /*
270
     * Mark the CFQ items which have been added to this packet as having been
271
     * transmitted.
272
     */
273
1.44M
    for (cfq_item = pkt->retx_head;
274
1.64M
         cfq_item != NULL;
275
1.44M
         cfq_item = cfq_item->pkt_next)
276
198k
        ossl_quic_cfq_mark_tx(fifd->cfq, cfq_item);
277
278
    /*
279
     * Mark the send stream chunks which have been added to the packet as having
280
     * been transmitted.
281
     */
282
1.44M
    chunks = ossl_quic_txpim_pkt_get_chunks(pkt);
283
1.44M
    num_chunks = ossl_quic_txpim_pkt_get_num_chunks(pkt);
284
1.53M
    for (i = 0; i < num_chunks; ++i) {
285
97.4k
        sstream = fifd->get_sstream_by_id(chunks[i].stream_id,
286
97.4k
                                          pkt->ackm_pkt.pkt_space,
287
97.4k
                                          fifd->get_sstream_by_id_arg);
288
97.4k
        if (sstream == NULL)
289
5.60k
            continue;
290
291
91.8k
        if (chunks[i].end >= chunks[i].start
292
91.8k
            && !ossl_quic_sstream_mark_transmitted(sstream,
293
91.8k
                                                   chunks[i].start,
294
91.8k
                                                   chunks[i].end))
295
0
            return 0;
296
297
91.8k
        if (chunks[i].has_fin
298
91.8k
            && !ossl_quic_sstream_mark_transmitted_fin(sstream,
299
0
                                                       chunks[i].end + 1))
300
0
                return 0;
301
91.8k
    }
302
303
    /* Inform the ACKM. */
304
1.44M
    return ossl_ackm_on_tx_packet(fifd->ackm, &pkt->ackm_pkt);
305
1.44M
}
306
307
void ossl_quic_fifd_set_qlog_cb(QUIC_FIFD *fifd, QLOG *(*get_qlog_cb)(void *arg),
308
                                void *get_qlog_cb_arg)
309
0
{
310
0
    fifd->get_qlog_cb       = get_qlog_cb;
311
0
    fifd->get_qlog_cb_arg   = get_qlog_cb_arg;
312
0
}