Coverage Report

Created: 2023-11-19 06:15

/src/h2o/lib/http3/qpack.c
Line
Count
Source (jump to first uncovered line)
1
/*
2
 * Copyright (c) 2018 Fastly, Kazuho Oku
3
 *
4
 * Permission is hereby granted, free of charge, to any person obtaining a copy
5
 * of this software and associated documentation files (the "Software"), to
6
 * deal in the Software without restriction, including without limitation the
7
 * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
8
 * sell copies of the Software, and to permit persons to whom the Software is
9
 * furnished to do so, subject to the following conditions:
10
 *
11
 * The above copyright notice and this permission notice shall be included in
12
 * all copies or substantial portions of the Software.
13
 *
14
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
17
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
18
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
19
 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
20
 * IN THE SOFTWARE.
21
 */
22
#include <stddef.h>
23
#include <stdint.h>
24
#include <stdio.h>
25
#include <stdlib.h>
26
#include "picotls.h"
27
#include "h2o.h"
28
#include "h2o/hpack.h"
29
#include "h2o/qpack.h"
30
#include "h2o/http3_common.h"
31
32
0
#define HEADER_ENTRY_SIZE_OFFSET 32
33
34
/**
35
 * a mem-shared object that contains the name and value of a header field
36
 */
37
struct st_h2o_qpack_header_t {
38
    h2o_iovec_t *name;
39
    size_t value_len;
40
    h2o_iovec_t _name_buf;
41
    unsigned soft_errors;
42
    char value[1];
43
};
44
45
struct st_h2o_qpack_header_table_t {
46
    /**
47
     * pointers to the buffer structure; [buf_start, buf_end) is the memory allocated, [first, last) are the active entries
48
     */
49
    struct st_h2o_qpack_header_t **buf_start, **first, **last, **buf_end;
50
    /**
51
     * absolute index of `first`
52
     */
53
    int64_t base_offset;
54
    /**
55
     * current and maximum size
56
     */
57
    size_t num_bytes, max_size;
58
};
59
60
struct st_h2o_qpack_blocked_streams_t {
61
    int64_t stream_id;
62
    int64_t largest_ref;
63
    union {
64
        struct {
65
            uint8_t is_blocking;
66
        } encoder_flags;
67
    };
68
};
69
70
struct st_h2o_qpack_decoder_t {
71
    /**
72
     *
73
     */
74
    struct st_h2o_qpack_header_table_t table;
75
    /**
76
     * maximum header table size declared by itself. Current max set by peer is available in `table.max_size`.
77
     */
78
    uint32_t header_table_size;
79
    /**
80
     *
81
     */
82
    uint32_t max_entries;
83
    /**
84
     * number of updates since last sync
85
     */
86
    uint32_t insert_count;
87
    /**
88
     *
89
     */
90
    uint64_t total_inserts;
91
    /**
92
     *
93
     */
94
    uint16_t max_blocked;
95
    struct {
96
        /**
97
         * contains list of blocked streams (sorted in the ascending order of largest_ref)
98
         */
99
        H2O_VECTOR(struct st_h2o_qpack_blocked_streams_t) list;
100
        /**
101
         * number of blocked streams that are unblocked. They are evicted parse_request / response is being called.
102
         */
103
        size_t num_unblocked;
104
    } blocked_streams;
105
};
106
107
struct st_h2o_qpack_encoder_t {
108
    /**
109
     * the header table
110
     */
111
    struct st_h2o_qpack_header_table_t table;
112
    /**
113
     * maximum id of the insertion being acked (inclusive)
114
     */
115
    int64_t largest_known_received;
116
    /**
117
     * SETTINGS_QPACK_BLOCKED_STREAMS
118
     */
119
    uint16_t max_blocked;
120
    /**
121
     * number of potentially blocked HEADERS (not streams, sorry!) We count header blocks rather than streams because it is easier.
122
     * Hopefully it would work well.
123
     */
124
    uint16_t num_blocked;
125
    /**
126
     * list of unacked streams
127
     */
128
    H2O_VECTOR(struct st_h2o_qpack_blocked_streams_t) inflight;
129
};
130
131
struct st_h2o_qpack_flatten_context_t {
132
    h2o_qpack_encoder_t *qpack;
133
    h2o_mem_pool_t *pool;
134
    int64_t stream_id;
135
    h2o_byte_vector_t *encoder_buf;
136
    h2o_byte_vector_t headers_buf;
137
    int64_t base_index;
138
    int64_t largest_ref;
139
};
140
141
57.6k
#define MAX_HEADER_NAME_LENGTH 128
142
66.1k
#define MAX_HEADER_VALUE_LENGTH 4096
143
144
const char *h2o_qpack_err_header_name_too_long = "header name too long";
145
const char *h2o_qpack_err_header_value_too_long = "header value too long";
146
const char *h2o_qpack_err_header_exceeds_table_size = "header exceeds table size";
147
const char *h2o_qpack_err_invalid_max_size = "invalid max size";
148
const char *h2o_qpack_err_invalid_static_reference = "invalid static reference";
149
const char *h2o_qpack_err_invalid_dynamic_reference = "invalid dynamic reference";
150
const char *h2o_qpack_err_invalid_duplicate = "invalid duplicate";
151
const char *h2o_qpack_err_invalid_pseudo_header = "invalid pseudo header";
152
153
static void header_table_init(struct st_h2o_qpack_header_table_t *table, size_t max_size)
154
6.41k
{
155
6.41k
    *table = (struct st_h2o_qpack_header_table_t){NULL, NULL, NULL, NULL, 1, 0, max_size};
156
6.41k
}
157
158
static void header_table_dispose(struct st_h2o_qpack_header_table_t *table)
159
6.41k
{
160
6.41k
    while (table->first != table->last)
161
0
        h2o_mem_release_shared(*table->first++);
162
6.41k
    free(table->buf_start);
163
6.41k
}
164
165
static void header_table_evict(struct st_h2o_qpack_header_table_t *table, size_t delta)
166
0
{
167
0
    while (table->first != table->last) {
168
0
        if (table->num_bytes + delta <= table->max_size)
169
0
            return;
170
0
        table->num_bytes -= (*table->first)->name->len + (*table->first)->value_len + HEADER_ENTRY_SIZE_OFFSET;
171
0
        h2o_mem_release_shared(*table->first);
172
0
        *table->first++ = NULL;
173
0
        ++table->base_offset;
174
0
    }
175
0
    assert(table->num_bytes == 0);
176
0
}
177
178
static void header_table_insert(struct st_h2o_qpack_header_table_t *table, struct st_h2o_qpack_header_t *added)
179
0
{
180
0
    header_table_evict(table, added->name->len + added->value_len + HEADER_ENTRY_SIZE_OFFSET);
181
182
0
    if (table->last == table->buf_end) {
183
0
        size_t count = table->last - table->first, new_capacity = count <= 2 ? 4 : count * 2;
184
0
        if (new_capacity > table->buf_end - table->buf_start) {
185
0
            struct st_h2o_qpack_header_t **newbuf = h2o_mem_alloc(sizeof(*newbuf) * new_capacity);
186
0
            memcpy(newbuf, table->first, sizeof(*newbuf) * count);
187
0
            free(table->buf_start);
188
0
            table->buf_start = newbuf;
189
0
            table->first = newbuf;
190
0
            table->last = newbuf + count;
191
0
            table->buf_end = newbuf + new_capacity;
192
0
        } else {
193
0
            assert(table->buf_start != table->first);
194
0
            memmove(table->buf_start, table->first, sizeof(*table->buf_start) * count);
195
0
            table->first = table->buf_start;
196
0
            table->last = table->buf_start + count;
197
0
        }
198
0
        memset(table->last, 0, sizeof(*table->last) * (table->buf_end - table->last));
199
0
    }
200
0
    *table->last++ = added;
201
0
    table->num_bytes += added->name->len + added->value_len + HEADER_ENTRY_SIZE_OFFSET;
202
0
}
203
204
static const h2o_qpack_static_table_entry_t *resolve_static_abs(int64_t index, const char **err_desc)
205
0
{
206
0
    if (index >= sizeof(h2o_qpack_static_table) / sizeof(h2o_qpack_static_table[0])) {
207
0
        *err_desc = h2o_qpack_err_invalid_static_reference;
208
0
        return NULL;
209
0
    }
210
0
    return h2o_qpack_static_table + index;
211
0
}
212
213
static struct st_h2o_qpack_header_t *resolve_dynamic_abs(struct st_h2o_qpack_header_table_t *table, int64_t index,
214
                                                         const char **err_desc)
215
693
{
216
693
    if (index < table->base_offset)
217
0
        goto Invalid;
218
693
    index -= table->base_offset;
219
693
    if (index >= table->last - table->first)
220
693
        goto Invalid;
221
0
    return table->first[index];
222
693
Invalid:
223
693
    *err_desc = h2o_qpack_err_invalid_dynamic_reference;
224
693
    return NULL;
225
693
}
226
227
static int decode_int(int64_t *value, const uint8_t **src, const uint8_t *src_end, unsigned prefix_bits)
228
717k
{
229
717k
    if ((*value = h2o_hpack_decode_int(src, src_end, prefix_bits)) < 0)
230
103
        return *value == H2O_HTTP2_ERROR_INCOMPLETE ? H2O_HTTP3_ERROR_INCOMPLETE : H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
231
716k
    return 0;
232
717k
}
233
234
static size_t decode_value(char *outbuf, unsigned *soft_errors, int is_huff, const uint8_t *src, size_t srclen,
235
                           const char **err_desc)
236
65.7k
{
237
65.7k
    size_t outlen;
238
239
65.7k
    if (is_huff) {
240
16.3k
        if ((outlen = h2o_hpack_decode_huffman(outbuf, soft_errors, src, srclen, 0, err_desc)) == SIZE_MAX)
241
36
            return SIZE_MAX;
242
49.3k
    } else {
243
49.3k
        h2o_hpack_validate_header_value(soft_errors, (void *)src, srclen);
244
49.3k
        memcpy(outbuf, src, srclen);
245
49.3k
        outlen = srclen;
246
49.3k
    }
247
65.6k
    outbuf[outlen] = '\0';
248
249
65.6k
    return outlen;
250
65.7k
}
251
252
h2o_qpack_decoder_t *h2o_qpack_create_decoder(uint32_t header_table_size, uint16_t max_blocked)
253
6.41k
{
254
6.41k
    h2o_qpack_decoder_t *qpack = h2o_mem_alloc(sizeof(*qpack));
255
256
6.41k
    qpack->insert_count = 0;
257
6.41k
    qpack->header_table_size = header_table_size;
258
6.41k
    qpack->max_entries = header_table_size / 32;
259
6.41k
    qpack->total_inserts = 0;
260
6.41k
    qpack->max_blocked = max_blocked;
261
6.41k
    header_table_init(&qpack->table, qpack->header_table_size);
262
6.41k
    memset(&qpack->blocked_streams, 0, sizeof(qpack->blocked_streams));
263
264
6.41k
    return qpack;
265
6.41k
}
266
267
void h2o_qpack_destroy_decoder(h2o_qpack_decoder_t *qpack)
268
6.41k
{
269
6.41k
    header_table_dispose(&qpack->table);
270
6.41k
    free(qpack->blocked_streams.list.entries);
271
6.41k
    free(qpack);
272
6.41k
}
273
274
static void decoder_link_blocked(h2o_qpack_decoder_t *qpack, int64_t stream_id, int64_t largest_ref)
275
0
{
276
0
    size_t i;
277
278
0
    h2o_vector_reserve(NULL, &qpack->blocked_streams.list, qpack->blocked_streams.list.size + 1);
279
0
    for (i = qpack->blocked_streams.list.size; i != 0; --i)
280
0
        if (qpack->blocked_streams.list.entries[i - 1].largest_ref <= largest_ref)
281
0
            break;
282
0
    if (i != qpack->blocked_streams.list.size)
283
0
        memmove(qpack->blocked_streams.list.entries + i + 1, qpack->blocked_streams.list.entries + i,
284
0
                sizeof(qpack->blocked_streams.list.entries[0]) * (qpack->blocked_streams.list.size - i));
285
0
    qpack->blocked_streams.list.entries[i] = (struct st_h2o_qpack_blocked_streams_t){stream_id, largest_ref};
286
0
    ++qpack->blocked_streams.list.size;
287
0
}
288
289
static void decoder_insert(h2o_qpack_decoder_t *qpack, struct st_h2o_qpack_header_t *added)
290
0
{
291
0
    ++qpack->insert_count;
292
0
    ++qpack->total_inserts;
293
0
    fprintf(stderr, "#%s:%" PRIu64 ":%.*s\t%.*s\n", __FUNCTION__, qpack->total_inserts, (int)added->name->len, added->name->base,
294
0
            (int)added->value_len, added->value);
295
0
    header_table_insert(&qpack->table, added);
296
0
}
297
298
static int decode_value_and_insert(h2o_qpack_decoder_t *qpack, struct st_h2o_qpack_header_t *header, int is_huff,
299
                                   const uint8_t *qstr, size_t qstrlen, const char **err_desc)
300
0
{
301
0
    if ((header->value_len = decode_value(header->value, &header->soft_errors, is_huff, qstr, qstrlen, err_desc)) == SIZE_MAX)
302
0
        goto Fail;
303
0
    if (header->name->len + header->value_len + HEADER_ENTRY_SIZE_OFFSET > qpack->table.max_size) {
304
0
        *err_desc = h2o_qpack_err_header_exceeds_table_size;
305
0
        goto Fail;
306
0
    }
307
0
    decoder_insert(qpack, header);
308
0
    return 0;
309
0
Fail:
310
0
    h2o_mem_release_shared(header);
311
0
    return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
312
0
}
313
314
static int insert_token_header(h2o_qpack_decoder_t *qpack, const h2o_token_t *name, int value_is_huff, const uint8_t *value,
315
                               size_t value_len, const char **err_desc)
316
0
{
317
0
    struct st_h2o_qpack_header_t *header =
318
0
        h2o_mem_alloc_shared(NULL, offsetof(struct st_h2o_qpack_header_t, value) + (value_len * 2) + 1, NULL);
319
320
0
    header->name = (h2o_iovec_t *)&name->buf;
321
0
    header->soft_errors = 0;
322
0
    return decode_value_and_insert(qpack, header, value_is_huff, value, value_len, err_desc);
323
0
}
324
325
static int insert_literal_header(h2o_qpack_decoder_t *qpack, const char *name, size_t name_len, int value_is_huff,
326
                                 const uint8_t *value, size_t value_len, unsigned soft_errors, const char **err_desc)
327
0
{
328
0
    size_t value_capacity = (value_is_huff ? value_len * 2 : value_len) + 1;
329
0
    struct st_h2o_qpack_header_t *header =
330
0
        h2o_mem_alloc_shared(NULL, offsetof(struct st_h2o_qpack_header_t, value) + value_capacity + name_len + 1, NULL);
331
332
0
    header->_name_buf = h2o_iovec_init(header->value + value_capacity, name_len);
333
0
    memcpy(header->_name_buf.base, name, name_len);
334
0
    header->_name_buf.base[name_len] = '\0';
335
0
    header->_name_buf.len = name_len;
336
0
    header->name = &header->_name_buf;
337
0
    header->soft_errors = soft_errors;
338
339
0
    return decode_value_and_insert(qpack, header, value_is_huff, value, value_len, err_desc);
340
0
}
341
342
static int64_t qpack_table_total_inserts(struct st_h2o_qpack_header_table_t *table)
343
5.64k
{
344
5.64k
    return table->base_offset + (table->last - table->first);
345
5.64k
}
346
347
static int insert_with_name_reference(h2o_qpack_decoder_t *qpack, int name_is_static, int64_t name_index, int value_is_huff,
348
                                      const uint8_t *value, int64_t value_len, const char **err_desc)
349
0
{
350
0
    if (value_len >= MAX_HEADER_VALUE_LENGTH) {
351
0
        *err_desc = h2o_qpack_err_header_value_too_long;
352
0
        return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
353
0
    }
354
355
0
    if (name_is_static) {
356
0
        const h2o_qpack_static_table_entry_t *ref;
357
0
        if ((ref = resolve_static_abs(name_index, err_desc)) == NULL)
358
0
            return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
359
0
        return insert_token_header(qpack, ref->name, value_is_huff, value, value_len, err_desc);
360
0
    } else {
361
0
        struct st_h2o_qpack_header_t *ref;
362
0
        int64_t base_index = qpack_table_total_inserts(&qpack->table) - 1;
363
0
        if (name_index > base_index) {
364
0
            *err_desc = h2o_qpack_err_invalid_dynamic_reference;
365
0
            return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
366
0
        }
367
0
        if ((ref = resolve_dynamic_abs(&qpack->table, base_index - name_index, err_desc)) == NULL)
368
0
            return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
369
0
        if (h2o_iovec_is_token(ref->name)) {
370
0
            return insert_token_header(qpack, (h2o_token_t *)ref->name, value_is_huff, value, value_len, err_desc);
371
0
        } else {
372
0
            return insert_literal_header(qpack, ref->name->base, ref->name->len, value_is_huff, value, value_len,
373
0
                                         ref->soft_errors & H2O_HPACK_SOFT_ERROR_BIT_INVALID_NAME, err_desc);
374
0
        }
375
0
    }
376
0
}
377
378
static int insert_without_name_reference(h2o_qpack_decoder_t *qpack, int qnhuff, const uint8_t *qn, int64_t qnlen, int qvhuff,
379
                                         const uint8_t *qv, int64_t qvlen, const char **err_desc)
380
0
{
381
0
    h2o_iovec_t name;
382
0
    unsigned soft_errors = 0;
383
384
0
    if (qnlen >= MAX_HEADER_NAME_LENGTH) {
385
0
        *err_desc = h2o_qpack_err_header_name_too_long;
386
0
        return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
387
0
    }
388
0
    if (qvlen >= MAX_HEADER_VALUE_LENGTH) {
389
0
        *err_desc = h2o_qpack_err_header_value_too_long;
390
0
        return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
391
0
    }
392
393
0
    if (qnhuff) {
394
0
        name.base = alloca(qnlen * 2);
395
0
        if ((name.len = h2o_hpack_decode_huffman(name.base, &soft_errors, qn, qnlen, 1, err_desc)) == SIZE_MAX)
396
0
            return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
397
0
    } else {
398
0
        if (!h2o_hpack_validate_header_name(&soft_errors, (void *)qn, qnlen, err_desc))
399
0
            return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
400
0
        name = h2o_iovec_init(qn, qnlen);
401
0
    }
402
403
0
    const h2o_token_t *token;
404
0
    if ((token = h2o_lookup_token(name.base, name.len)) != NULL) {
405
0
        return insert_token_header(qpack, token, qvhuff, qv, qvlen, err_desc);
406
0
    } else {
407
0
        return insert_literal_header(qpack, name.base, name.len, qvhuff, qv, qvlen, soft_errors, err_desc);
408
0
    }
409
0
}
410
411
static int duplicate(h2o_qpack_decoder_t *qpack, int64_t index, const char **err_desc)
412
0
{
413
0
    if (index >= qpack->table.last - qpack->table.first) {
414
0
        *err_desc = h2o_qpack_err_invalid_duplicate;
415
0
        return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
416
0
    }
417
418
0
    struct st_h2o_qpack_header_t *header = qpack->table.last[-index - 1];
419
0
    h2o_mem_addref_shared(header);
420
0
    decoder_insert(qpack, header);
421
0
    return 0;
422
0
}
423
424
static int dynamic_table_size_update(h2o_qpack_decoder_t *qpack, int64_t max_size, const char **err_desc)
425
0
{
426
0
    if (max_size > qpack->header_table_size) {
427
0
        *err_desc = h2o_qpack_err_invalid_max_size;
428
0
        return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
429
0
    }
430
431
0
    qpack->table.max_size = max_size;
432
0
    header_table_evict(&qpack->table, 0);
433
0
    return 0;
434
0
}
435
436
int h2o_qpack_decoder_handle_input(h2o_qpack_decoder_t *qpack, int64_t **unblocked_stream_ids, size_t *num_unblocked,
437
                                   const uint8_t **_src, const uint8_t *src_end, const char **err_desc)
438
0
{
439
0
    if (qpack->blocked_streams.num_unblocked != 0) {
440
0
        size_t remaining = qpack->blocked_streams.list.size - qpack->blocked_streams.num_unblocked;
441
0
        if (remaining != 0)
442
0
            memmove(qpack->blocked_streams.list.entries, qpack->blocked_streams.list.entries + remaining,
443
0
                    sizeof(qpack->blocked_streams.list.entries[0]) * remaining);
444
0
        qpack->blocked_streams.list.size = remaining;
445
0
        qpack->blocked_streams.num_unblocked = 0;
446
0
    }
447
448
0
    const uint8_t *src = *_src;
449
0
    int ret = 0;
450
451
0
    while (src != src_end && ret == 0) {
452
0
        switch (*src >> 5) {
453
0
        default: /* insert with name reference */ {
454
0
            int64_t name_index, value_len;
455
0
            int name_is_static = (*src & 0x40) != 0;
456
0
            if ((ret = decode_int(&name_index, &src, src_end, 6)) != 0)
457
0
                goto Exit;
458
0
            if (src == src_end)
459
0
                goto Exit;
460
0
            int value_is_huff = (*src & 0x80) != 0;
461
0
            if ((ret = decode_int(&value_len, &src, src_end, 7)) != 0)
462
0
                goto Exit;
463
0
            if (!(src + value_len <= src_end))
464
0
                goto Exit;
465
0
            ret = insert_with_name_reference(qpack, name_is_static, name_index, value_is_huff, src, value_len, err_desc);
466
0
            src += value_len;
467
0
        } break;
468
0
        case 2:
469
0
        case 3: /* insert without name reference */ {
470
0
            int64_t name_len, value_len;
471
0
            int name_is_huff = (*src & 0x20) != 0;
472
0
            if ((ret = decode_int(&name_len, &src, src_end, 5)) != 0)
473
0
                goto Exit;
474
0
            if (!(src + name_len < src_end))
475
0
                goto Exit;
476
0
            const uint8_t *name = src;
477
0
            src += name_len;
478
0
            int value_is_huff = (*src & 0x80) != 0;
479
0
            if ((ret = decode_int(&value_len, &src, src_end, 7)) != 0)
480
0
                goto Exit;
481
0
            if (!(src + value_len <= src_end))
482
0
                goto Exit;
483
0
            ret = insert_without_name_reference(qpack, name_is_huff, name, name_len, value_is_huff, src, value_len, err_desc);
484
0
            src += value_len;
485
0
        } break;
486
0
        case 0: /* duplicate */ {
487
0
            int64_t index;
488
0
            if ((ret = decode_int(&index, &src, src_end, 5)) != 0)
489
0
                goto Exit;
490
0
            ret = duplicate(qpack, index, err_desc);
491
0
        } break;
492
0
        case 1: /* dynamic table size update */ {
493
0
            int64_t max_size;
494
0
            if ((ret = decode_int(&max_size, &src, src_end, 5)) != 0)
495
0
                goto Exit;
496
0
            ret = dynamic_table_size_update(qpack, max_size, err_desc);
497
0
        } break;
498
0
        }
499
0
        *_src = src;
500
0
    }
501
502
0
Exit:
503
0
    if (ret == H2O_HTTP2_ERROR_INCOMPLETE)
504
0
        ret = 0;
505
0
    if (ret == 0) {
506
        /* build list of newly unblocked streams ids reusing the memory of the blocked streams list (nasty!) */
507
0
        *unblocked_stream_ids = &qpack->blocked_streams.list.entries[0].stream_id;
508
0
        for (qpack->blocked_streams.num_unblocked = 0; qpack->blocked_streams.num_unblocked < qpack->blocked_streams.list.size;
509
0
             ++qpack->blocked_streams.num_unblocked) {
510
0
            if (qpack->blocked_streams.list.entries[qpack->blocked_streams.num_unblocked].largest_ref > qpack->total_inserts)
511
0
                break;
512
0
            (*unblocked_stream_ids)[qpack->blocked_streams.num_unblocked] =
513
0
                qpack->blocked_streams.list.entries[qpack->blocked_streams.num_unblocked].stream_id;
514
0
        }
515
0
        *num_unblocked = qpack->blocked_streams.num_unblocked;
516
0
    }
517
0
    return (int)ret;
518
0
}
519
520
size_t h2o_qpack_decoder_send_state_sync(h2o_qpack_decoder_t *qpack, uint8_t *outbuf)
521
0
{
522
0
    if (qpack->insert_count == 0)
523
0
        return 0;
524
525
0
    uint8_t *dst = outbuf;
526
0
    *dst = 0;
527
0
    dst = h2o_hpack_encode_int(dst, qpack->insert_count, 6);
528
0
    qpack->insert_count = 0;
529
530
0
    return dst - outbuf;
531
0
}
532
533
size_t h2o_qpack_decoder_send_stream_cancel(h2o_qpack_decoder_t *qpack, uint8_t *outbuf, int64_t stream_id)
534
0
{
535
0
    outbuf[0] = 0x40;
536
0
    return h2o_hpack_encode_int(outbuf, stream_id, 6) - outbuf;
537
0
}
538
539
static const h2o_qpack_static_table_entry_t *resolve_static(const uint8_t **src, const uint8_t *src_end, unsigned prefix_bits,
540
                                                            const char **err_desc)
541
580k
{
542
580k
    int64_t index;
543
544
580k
    if (decode_int(&index, src, src_end, prefix_bits) != 0)
545
9
        goto Fail;
546
580k
    assert(index >= 0);
547
580k
    if (!(index < sizeof(h2o_qpack_static_table) / sizeof(h2o_qpack_static_table[0])))
548
152
        goto Fail;
549
580k
    return h2o_qpack_static_table + index;
550
551
161
Fail:
552
161
    *err_desc = h2o_qpack_err_invalid_static_reference;
553
161
    return NULL;
554
580k
}
555
556
static struct st_h2o_qpack_header_t *resolve_dynamic(struct st_h2o_qpack_header_table_t *table, int64_t base_index,
557
                                                     const uint8_t **src, const uint8_t *src_end, unsigned prefix_bits,
558
                                                     const char **err_desc)
559
455
{
560
455
    int64_t off;
561
562
455
    if (decode_int(&off, src, src_end, prefix_bits) != 0 || off >= base_index) {
563
154
        *err_desc = h2o_qpack_err_invalid_dynamic_reference;
564
154
        return NULL;
565
154
    }
566
301
    return resolve_dynamic_abs(table, base_index - off, err_desc);
567
455
}
568
569
static struct st_h2o_qpack_header_t *resolve_dynamic_postbase(struct st_h2o_qpack_header_table_t *table, int64_t base_index,
570
                                                              const uint8_t **src, const uint8_t *src_end, unsigned prefix_bits,
571
                                                              const char **err_desc)
572
459
{
573
459
    int64_t off;
574
575
459
    if (decode_int(&off, src, src_end, prefix_bits) != 0 || off > INT64_MAX - base_index - 1) {
576
67
        *err_desc = h2o_qpack_err_invalid_dynamic_reference;
577
67
        return NULL;
578
67
    }
579
392
    return resolve_dynamic_abs(table, base_index + off + 1, err_desc);
580
459
}
581
582
static h2o_iovec_t *decode_header_name_literal(h2o_mem_pool_t *pool, unsigned *soft_errors, const uint8_t **src,
583
                                               const uint8_t *src_end, unsigned prefix_bits, const char **err_desc)
584
57.6k
{
585
57.6k
    h2o_iovec_t buf = {NULL};
586
57.6k
    const h2o_token_t *token;
587
57.6k
    int is_huff;
588
57.6k
    int64_t len;
589
590
    /* obtain flags and length */
591
57.6k
    is_huff = (**src >> prefix_bits) & 1;
592
57.6k
    if (decode_int(&len, src, src_end, prefix_bits) != 0 || len > MAX_HEADER_NAME_LENGTH) {
593
70
        *err_desc = h2o_qpack_err_header_name_too_long;
594
70
        goto Fail;
595
70
    }
596
57.6k
    if (src_end - *src < len)
597
58
        goto Fail;
598
599
    /* decode and convert to token (if possible) */
600
57.5k
    if (is_huff) {
601
16.9k
        buf.base = h2o_mem_alloc_pool(pool, char, len * 2 + 1);
602
16.9k
        if ((buf.len = h2o_hpack_decode_huffman(buf.base, soft_errors, *src, len, 1, err_desc)) == SIZE_MAX)
603
52
            goto Fail;
604
16.8k
        buf.base[buf.len] = '\0';
605
16.8k
        token = h2o_lookup_token(buf.base, buf.len);
606
40.6k
    } else if ((token = h2o_lookup_token((const char *)*src, len)) != NULL) {
607
        /* was an uncompressed token */
608
28.0k
    } else {
609
28.0k
        if (!h2o_hpack_validate_header_name(soft_errors, (void *)*src, len, err_desc))
610
58
            goto Fail;
611
28.0k
        buf = h2o_strdup(pool, (void *)*src, len);
612
28.0k
    }
613
57.4k
    *src += len;
614
615
    /* return the result */
616
57.4k
    if (token != NULL)
617
12.9k
        return (h2o_iovec_t *)&token->buf;
618
44.5k
    h2o_iovec_t *ret = h2o_mem_alloc_pool(pool, h2o_iovec_t, 1);
619
44.5k
    *ret = buf;
620
44.5k
    return ret;
621
622
238
Fail:
623
238
    return NULL;
624
57.4k
}
625
626
static h2o_iovec_t decode_header_value_literal(h2o_mem_pool_t *pool, unsigned *soft_errors, const uint8_t **src,
627
                                               const uint8_t *src_end, const char **err_desc)
628
67.5k
{
629
67.5k
    h2o_iovec_t buf;
630
67.5k
    int64_t len;
631
632
    /* validate *src pointer before dereferencing it for the huffman bit check */
633
67.5k
    if (!(*src < src_end))
634
1.39k
        goto Fail;
635
66.2k
    int is_huff = (**src & 0x80) != 0;
636
637
66.2k
    if (decode_int(&len, src, src_end, 7) != 0 || len > MAX_HEADER_VALUE_LENGTH) {
638
75
        *err_desc = h2o_qpack_err_header_value_too_long;
639
75
        goto Fail;
640
75
    }
641
66.1k
    if (src_end - *src < len)
642
411
        goto Fail;
643
644
65.7k
    buf.base = h2o_mem_alloc_pool(pool, char, is_huff ? len * 2 + 1 : len + 1);
645
65.7k
    if ((buf.len = decode_value(buf.base, soft_errors, is_huff, *src, len, err_desc)) == SIZE_MAX)
646
36
        goto Fail;
647
65.6k
    *src += len;
648
649
65.6k
    return buf;
650
1.91k
Fail:
651
1.91k
    return h2o_iovec_init(NULL, 0);
652
65.7k
}
653
654
struct st_h2o_qpack_decode_header_ctx_t {
655
    h2o_qpack_decoder_t *qpack;
656
    /**
657
     * These values are non-negative.
658
     */
659
    int64_t req_insert_count, base_index;
660
};
661
662
static size_t send_header_ack(h2o_qpack_decoder_t *qpack, struct st_h2o_qpack_decode_header_ctx_t *ctx, uint8_t *outbuf,
663
                              int64_t stream_id)
664
2.28k
{
665
2.28k
    if (ctx->req_insert_count == 0)
666
2.28k
        return 0;
667
0
    outbuf[0] = 0x80;
668
0
    return h2o_hpack_encode_int(outbuf, stream_id, 7) - outbuf;
669
2.28k
}
670
671
static int decode_header(h2o_mem_pool_t *pool, void *_ctx, h2o_iovec_t **name, h2o_iovec_t *value, const uint8_t **src,
672
                         const uint8_t *src_end, const char **err_desc)
673
639k
{
674
639k
    struct st_h2o_qpack_decode_header_ctx_t *ctx = _ctx;
675
639k
    unsigned soft_errors;
676
677
639k
    switch (**src >> 4) {
678
218k
    case 12:
679
229k
    case 13:
680
267k
    case 14:
681
570k
    case 15: /* indexed static header field */ {
682
570k
        const h2o_qpack_static_table_entry_t *entry;
683
570k
        if ((entry = resolve_static(src, src_end, 6, err_desc)) == NULL)
684
94
            goto Fail;
685
570k
        *name = (h2o_iovec_t *)&entry->name->buf;
686
570k
        *value = entry->value;
687
570k
        soft_errors = 0;
688
570k
    } break;
689
113
    case 8:
690
130
    case 9:
691
154
    case 10:
692
226
    case 11: /* indexed dynamic header field */ {
693
226
        struct st_h2o_qpack_header_t *entry;
694
226
        if ((entry = resolve_dynamic(&ctx->qpack->table, ctx->base_index, src, src_end, 6, err_desc)) == NULL)
695
226
            goto Fail;
696
0
        h2o_mem_link_shared(pool, entry);
697
0
        *name = entry->name;
698
0
        *value = h2o_iovec_init(entry->value, entry->value_len);
699
0
        soft_errors = entry->soft_errors;
700
0
    } break;
701
5.08k
    case 5:
702
10.2k
    case 7: /* literal header field with static name reference */ {
703
10.2k
        const h2o_qpack_static_table_entry_t *entry;
704
10.2k
        if ((entry = resolve_static(src, src_end, 4, err_desc)) == NULL)
705
67
            goto Fail;
706
10.1k
        *name = (h2o_iovec_t *)&entry->name->buf;
707
10.1k
        soft_errors = 0;
708
10.1k
        if ((*value = decode_header_value_literal(pool, &soft_errors, src, src_end, err_desc)).base == NULL)
709
130
            goto Fail;
710
10.1k
    } break;
711
10.0k
    case 4:
712
229
    case 6: /* literal header field with dynamic name reference */ {
713
229
        struct st_h2o_qpack_header_t *entry;
714
229
        if ((entry = resolve_dynamic(&ctx->qpack->table, ctx->base_index, src, src_end, 4, err_desc)) == NULL)
715
229
            goto Fail;
716
0
        h2o_mem_link_shared(pool, entry);
717
0
        *name = entry->name;
718
0
        soft_errors = (entry->soft_errors) & H2O_HPACK_SOFT_ERROR_BIT_INVALID_NAME;
719
0
        if ((*value = decode_header_value_literal(pool, &soft_errors, src, src_end, err_desc)).base == NULL)
720
0
            goto Fail;
721
0
    } break;
722
46.6k
    case 2:
723
57.6k
    case 3: /* literal header field without name reference */ {
724
57.6k
        soft_errors = 0;
725
57.6k
        if ((*name = decode_header_name_literal(pool, &soft_errors, src, src_end, 3, err_desc)) == NULL)
726
238
            goto Fail;
727
57.4k
        if ((*value = decode_header_value_literal(pool, &soft_errors, src, src_end, err_desc)).base == NULL)
728
1.78k
            goto Fail;
729
57.4k
    } break;
730
55.6k
    case 1: /* indexed header field with post-base index */ {
731
154
        struct st_h2o_qpack_header_t *entry;
732
154
        if ((entry = resolve_dynamic_postbase(&ctx->qpack->table, ctx->base_index, src, src_end, 4, err_desc)) == NULL)
733
154
            goto Fail;
734
0
        h2o_mem_link_shared(pool, entry);
735
0
        *name = entry->name;
736
0
        *value = h2o_iovec_init(entry->value, entry->value_len);
737
0
        soft_errors = entry->soft_errors;
738
0
    } break;
739
305
    case 0: /* literal header field with post-base name reference */ {
740
305
        struct st_h2o_qpack_header_t *entry;
741
305
        if ((entry = resolve_dynamic_postbase(&ctx->qpack->table, ctx->base_index, src, src_end, 3, err_desc)) == NULL)
742
305
            goto Fail;
743
0
        h2o_mem_link_shared(pool, entry);
744
0
        *name = entry->name;
745
0
        soft_errors = (entry->soft_errors) & H2O_HPACK_SOFT_ERROR_BIT_INVALID_NAME;
746
0
        if ((*value = decode_header_value_literal(pool, &soft_errors, src, src_end, err_desc)).base == NULL)
747
0
            goto Fail;
748
0
    } break;
749
0
    default:
750
0
        h2o_fatal("unreachable");
751
0
        soft_errors = 0;
752
0
        break;
753
639k
    }
754
755
635k
    if (soft_errors != 0) {
756
38.6k
        *err_desc = (soft_errors & H2O_HPACK_SOFT_ERROR_BIT_INVALID_NAME) != 0
757
38.6k
                        ? h2o_hpack_soft_err_found_invalid_char_in_header_name
758
38.6k
                        : h2o_hpack_soft_err_found_invalid_char_in_header_value;
759
38.6k
        return H2O_HTTP2_ERROR_INVALID_HEADER_CHAR;
760
38.6k
    }
761
597k
    return 0;
762
3.23k
Fail:
763
3.23k
    return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
764
635k
}
765
766
static int parse_decode_context(h2o_qpack_decoder_t *qpack, struct st_h2o_qpack_decode_header_ctx_t *ctx, int64_t stream_id,
767
                                const uint8_t **src, const uint8_t *src_end)
768
5.99k
{
769
5.99k
    ctx->qpack = qpack;
770
771
    /* largest reference */
772
5.99k
    if (decode_int(&ctx->req_insert_count, src, src_end, 8) != 0)
773
68
        return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
774
5.92k
    if (ctx->req_insert_count > 0) {
775
131
        if (qpack->max_entries == 0)
776
131
            return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
777
0
        const uint32_t full_range = 2 * qpack->max_entries;
778
0
        uint64_t max_value = qpack->total_inserts + qpack->max_entries;
779
0
        uint64_t rounded = max_value / full_range * full_range;
780
0
        ctx->req_insert_count += rounded - 1;
781
0
        if (ctx->req_insert_count > max_value) {
782
0
            if (ctx->req_insert_count <= full_range)
783
0
                return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
784
0
            ctx->req_insert_count -= full_range;
785
0
        }
786
0
        if (ctx->req_insert_count == 0)
787
0
            return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
788
        /* Peer cannot send no more than PTLS_QUICINT_MAX instructions. That is because one QPACK instruction is no smaller than one
789
         * byte, and the maximum length of a QUIC stream (that conveys QPACK instructions) is 2^62 bytes in QUIC v1. */
790
0
        if (ctx->req_insert_count > PTLS_QUICINT_MAX)
791
0
            return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
792
0
    }
793
794
    /* sign and base index */
795
5.79k
    if (*src >= src_end)
796
1
        return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
797
5.79k
    int sign = (**src & 0x80) != 0;
798
5.79k
    int64_t delta_base;
799
5.79k
    if (decode_int(&delta_base, src, src_end, 7) != 0)
800
2
        return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
801
5.79k
    if (delta_base > PTLS_QUICINT_MAX)
802
2
        return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
803
5.78k
    ctx->base_index = sign == 0 ? ctx->req_insert_count + delta_base : ctx->req_insert_count - delta_base - 1;
804
5.78k
    if (ctx->base_index < 0) {
805
        /* Reject negative base index though current QPACK specification doesn't mention such case; let's keep our eyes on
806
         * https://github.com/quicwg/base-drafts/issues/4938 */
807
145
        return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
808
145
    }
809
810
    /* is the stream blocked? */
811
5.64k
    if (ctx->req_insert_count >= qpack_table_total_inserts(&qpack->table)) {
812
0
        if (qpack->blocked_streams.list.size + 1 >= qpack->max_blocked)
813
0
            return H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
814
0
        decoder_link_blocked(qpack, stream_id, ctx->req_insert_count);
815
0
        return H2O_HTTP3_ERROR_INCOMPLETE;
816
0
    }
817
818
5.64k
    return 0;
819
5.64k
}
820
821
static int normalize_error_code(int err)
822
3.35k
{
823
    /* convert H2 errors (except invaild_header_char) to QPACK error code */
824
3.35k
    if (err < 0 && err != H2O_HTTP2_ERROR_INVALID_HEADER_CHAR)
825
126
        err = H2O_HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
826
3.35k
    return err;
827
3.35k
}
828
829
int h2o_qpack_parse_request(h2o_mem_pool_t *pool, h2o_qpack_decoder_t *qpack, int64_t stream_id, h2o_iovec_t *method,
830
                            const h2o_url_scheme_t **scheme, h2o_iovec_t *authority, h2o_iovec_t *path, h2o_headers_t *headers,
831
                            int *pseudo_header_exists_map, size_t *content_length, h2o_cache_digests_t **digests,
832
                            h2o_iovec_t *datagram_flow_id, uint8_t *outbuf, size_t *outbufsize, const uint8_t *_src, size_t len,
833
                            const char **err_desc)
834
5.99k
{
835
5.99k
    struct st_h2o_qpack_decode_header_ctx_t ctx;
836
5.99k
    const uint8_t *src = _src, *src_end = src + len;
837
5.99k
    int ret;
838
839
5.99k
    if ((ret = parse_decode_context(qpack, &ctx, stream_id, &src, src_end)) != 0)
840
349
        return ret;
841
5.64k
    if ((ret =
842
5.64k
             h2o_hpack_parse_request(pool, decode_header, &ctx, method, scheme, authority, path, headers, pseudo_header_exists_map,
843
5.64k
                                     content_length, digests, datagram_flow_id, src, src_end - src, err_desc)) != 0) {
844
        /* bail out if the error is a hard error, otherwise build header ack then return */
845
3.58k
        if (ret != H2O_HTTP2_ERROR_INVALID_HEADER_CHAR)
846
3.35k
            return normalize_error_code(ret);
847
3.58k
    }
848
849
2.28k
    *outbufsize = send_header_ack(qpack, &ctx, outbuf, stream_id);
850
2.28k
    return ret;
851
5.64k
}
852
853
int h2o_qpack_parse_response(h2o_mem_pool_t *pool, h2o_qpack_decoder_t *qpack, int64_t stream_id, int *status,
854
                             h2o_headers_t *headers, h2o_iovec_t *datagram_flow_id, uint8_t *outbuf, size_t *outbufsize,
855
                             const uint8_t *_src, size_t len, const char **err_desc)
856
0
{
857
0
    struct st_h2o_qpack_decode_header_ctx_t ctx;
858
0
    const uint8_t *src = _src, *src_end = src + len;
859
0
    int ret;
860
861
0
    if ((ret = parse_decode_context(qpack, &ctx, stream_id, &src, src_end)) != 0)
862
0
        return ret;
863
0
    if ((ret = h2o_hpack_parse_response(pool, decode_header, &ctx, status, headers, datagram_flow_id, src, src_end - src,
864
0
                                        err_desc)) != 0)
865
0
        return normalize_error_code(ret);
866
867
0
    *outbufsize = send_header_ack(qpack, &ctx, outbuf, stream_id);
868
0
    return 0;
869
0
}
870
871
h2o_qpack_encoder_t *h2o_qpack_create_encoder(uint32_t header_table_size, uint16_t max_blocked)
872
0
{
873
0
    h2o_qpack_encoder_t *qpack = h2o_mem_alloc(sizeof(*qpack));
874
0
    header_table_init(&qpack->table, header_table_size);
875
0
    qpack->largest_known_received = 0;
876
0
    qpack->max_blocked = max_blocked;
877
0
    qpack->num_blocked = 0;
878
0
    memset(&qpack->inflight, 0, sizeof(qpack->inflight));
879
0
    return qpack;
880
0
}
881
882
void h2o_qpack_destroy_encoder(h2o_qpack_encoder_t *qpack)
883
0
{
884
0
    header_table_dispose(&qpack->table);
885
0
    free(qpack->inflight.entries);
886
0
    free(qpack);
887
0
}
888
889
static int handle_table_state_synchronize(h2o_qpack_encoder_t *qpack, int64_t insert_count, const char **err_desc)
890
0
{
891
0
    if (qpack == NULL || insert_count == 0)
892
0
        goto Error;
893
894
0
    int64_t new_value = qpack->largest_known_received + insert_count;
895
0
    if (new_value >= qpack_table_total_inserts(&qpack->table))
896
0
        goto Error;
897
0
    qpack->largest_known_received = new_value;
898
899
0
    return 0;
900
0
Error:
901
0
    *err_desc = "Table State Synchronize: invalid argument";
902
0
    return H2O_HTTP3_ERROR_QPACK_DECODER_STREAM;
903
0
}
904
905
static void evict_inflight_by_index(h2o_qpack_encoder_t *qpack, size_t index)
906
0
{
907
0
    if (qpack->inflight.entries[index].encoder_flags.is_blocking)
908
0
        --qpack->num_blocked;
909
0
    --qpack->inflight.size;
910
911
0
    if (qpack->inflight.size == 0) {
912
0
        free(qpack->inflight.entries);
913
0
        memset(&qpack->inflight, 0, sizeof(qpack->inflight));
914
0
    } else if (index < qpack->inflight.size) {
915
0
        memmove(qpack->inflight.entries + index, qpack->inflight.entries + index + 1, qpack->inflight.size - index);
916
0
    }
917
0
}
918
919
static int handle_header_ack(h2o_qpack_encoder_t *qpack, int64_t stream_id, const char **err_desc)
920
0
{
921
0
    size_t i;
922
923
0
    if (qpack != NULL) {
924
0
        for (i = 0; i < qpack->inflight.size; ++i)
925
0
            if (qpack->inflight.entries[i].stream_id == stream_id)
926
0
                goto Found;
927
0
    }
928
    /* not found */
929
0
    *err_desc = "Header Acknowledgement: invalid stream id";
930
0
    return H2O_HTTP3_ERROR_QPACK_DECODER_STREAM;
931
932
0
Found:
933
    /* update largest reference */
934
0
    if (qpack->largest_known_received < qpack->inflight.entries[i].largest_ref)
935
0
        qpack->largest_known_received = qpack->inflight.entries[i].largest_ref;
936
    /* evict the found entry */
937
0
    evict_inflight_by_index(qpack, i);
938
939
0
    return 0;
940
0
}
941
942
static int handle_stream_cancellation(h2o_qpack_encoder_t *qpack, int64_t stream_id, const char **err_desc)
943
0
{
944
0
    size_t index = 0;
945
946
0
    if (qpack != NULL) {
947
0
        while (index < qpack->inflight.size) {
948
0
            if (qpack->inflight.entries[index].stream_id == stream_id) {
949
0
                evict_inflight_by_index(qpack, index);
950
0
            } else {
951
0
                ++index;
952
0
            }
953
0
        }
954
0
    }
955
956
0
    return 0;
957
0
}
958
959
int h2o_qpack_encoder_handle_input(h2o_qpack_encoder_t *qpack, const uint8_t **_src, const uint8_t *src_end, const char **err_desc)
960
0
{
961
0
    const uint8_t *src = *_src;
962
0
    int ret = 0;
963
964
0
    while (src != src_end && ret == 0) {
965
0
        switch (*src >> 6) {
966
0
        case 0: /* table state synchronize */ {
967
0
            int64_t insert_count;
968
0
            if ((ret = decode_int(&insert_count, &src, src_end, 6)) != 0)
969
0
                goto Exit;
970
0
            ret = handle_table_state_synchronize(qpack, insert_count, err_desc);
971
0
        } break;
972
0
        default: /* header ack */ {
973
0
            int64_t stream_id;
974
0
            if ((ret = decode_int(&stream_id, &src, src_end, 7)) != 0)
975
0
                goto Exit;
976
0
            ret = handle_header_ack(qpack, stream_id, err_desc);
977
0
        } break;
978
0
        case 1: /* stream cancellation */ {
979
0
            int64_t stream_id;
980
0
            if ((ret = decode_int(&stream_id, &src, src_end, 6)) != 0)
981
0
                goto Exit;
982
0
            ret = handle_stream_cancellation(qpack, stream_id, err_desc);
983
0
        } break;
984
0
        }
985
0
        *_src = src;
986
0
    }
987
988
0
Exit:
989
0
    if (ret == H2O_HTTP3_ERROR_INCOMPLETE)
990
0
        ret = 0;
991
0
    return (int)ret;
992
0
}
993
994
static int64_t lookup_dynamic(h2o_qpack_encoder_t *qpack, const h2o_iovec_t *name, h2o_iovec_t value, int acked_only, int *is_exact)
995
0
{
996
0
    size_t i;
997
0
    int64_t name_found = -1;
998
999
0
    for (i = acked_only ? qpack->largest_known_received : qpack_table_total_inserts(&qpack->table) - 1;
1000
0
         i >= qpack->table.base_offset; --i) {
1001
0
        struct st_h2o_qpack_header_t *entry = qpack->table.first[i - qpack->table.base_offset];
1002
        /* compare names (and continue unless they match) */
1003
0
        if (h2o_iovec_is_token(name)) {
1004
0
            if (name != entry->name)
1005
0
                continue;
1006
0
        } else {
1007
0
            if (!h2o_memis(name->base, name->len, entry->name->base, entry->name->len))
1008
0
                continue;
1009
0
        }
1010
        /* compare values */
1011
0
        if (h2o_memis(value.base, value.len, entry->value, entry->value_len)) {
1012
0
            *is_exact = 1;
1013
0
            return i;
1014
0
        }
1015
0
        if (name_found == -1)
1016
0
            name_found = i;
1017
0
    }
1018
1019
0
    *is_exact = 0;
1020
0
    return name_found;
1021
0
}
1022
1023
static void flatten_int(h2o_byte_vector_t *buf, int64_t value, unsigned prefix_bits)
1024
8.68k
{
1025
8.68k
    uint8_t *p = h2o_hpack_encode_int(buf->entries + buf->size, value, prefix_bits);
1026
8.68k
    buf->size = p - buf->entries;
1027
8.68k
}
1028
1029
static void flatten_string(h2o_byte_vector_t *buf, const char *src, size_t len, unsigned prefix_bits, int dont_compress)
1030
5.39k
{
1031
5.39k
    size_t hufflen;
1032
1033
5.39k
    if (dont_compress || (hufflen = h2o_hpack_encode_huffman(buf->entries + buf->size + 1, (void *)src, len)) == SIZE_MAX) {
1034
        /* uncompressed */
1035
1.52k
        buf->entries[buf->size] &= ~((2 << prefix_bits) - 1); /* clear huffman mark */
1036
1.52k
        flatten_int(buf, len, prefix_bits);
1037
1.52k
        memcpy(buf->entries + buf->size, src, len);
1038
1.52k
        buf->size += len;
1039
3.86k
    } else {
1040
        /* build huffman header and adjust the location (if necessary) */
1041
3.86k
        uint8_t tmpbuf[H2O_HPACK_ENCODE_INT_MAX_LENGTH], *p = tmpbuf;
1042
3.86k
        *p = buf->entries[buf->size] & ~((1 << prefix_bits) - 1);
1043
3.86k
        *p |= (1 << prefix_bits); /* huffman mark */
1044
3.86k
        p = h2o_hpack_encode_int(p, hufflen, prefix_bits);
1045
3.86k
        if (p - tmpbuf == 1) {
1046
3.86k
            buf->entries[buf->size] = tmpbuf[0];
1047
3.86k
        } else {
1048
0
            memmove(buf->entries + buf->size + (p - tmpbuf), buf->entries + buf->size + 1, hufflen);
1049
0
            memcpy(buf->entries + buf->size, tmpbuf, p - tmpbuf);
1050
0
        }
1051
3.86k
        buf->size += p - tmpbuf + hufflen;
1052
3.86k
    }
1053
5.39k
}
1054
1055
static void emit_insert_with_nameref(h2o_qpack_encoder_t *qpack, h2o_mem_pool_t *pool, h2o_byte_vector_t *buf, int is_static,
1056
                                     int64_t index, h2o_iovec_t value)
1057
0
{
1058
0
    h2o_vector_reserve(pool, buf, buf->size + (H2O_HPACK_ENCODE_INT_MAX_LENGTH * 2) + value.len);
1059
0
    buf->entries[buf->size] = 0x80 | (is_static ? 0x40 : 0);
1060
0
    flatten_int(buf, index, 6);
1061
0
    flatten_string(buf, value.base, value.len, 7, 0);
1062
0
}
1063
1064
static void emit_insert_without_nameref(h2o_qpack_encoder_t *qpack, h2o_mem_pool_t *pool, h2o_byte_vector_t *buf,
1065
                                        const h2o_iovec_t *name, h2o_iovec_t value)
1066
0
{
1067
0
    h2o_vector_reserve(pool, buf, buf->size + (H2O_HPACK_ENCODE_INT_MAX_LENGTH * 2) + name->len + value.len);
1068
0
    buf->entries[buf->size] = 0x40;
1069
0
    flatten_string(buf, name->base, name->len, 5, 0);
1070
0
    flatten_string(buf, value.base, value.len, 7, 0);
1071
0
}
1072
1073
static void flatten_static_indexed(struct st_h2o_qpack_flatten_context_t *ctx, int32_t index)
1074
1.76k
{
1075
1.76k
    h2o_vector_reserve(ctx->pool, &ctx->headers_buf, ctx->headers_buf.size + H2O_HPACK_ENCODE_INT_MAX_LENGTH);
1076
1.76k
    ctx->headers_buf.entries[ctx->headers_buf.size] = 0xc0;
1077
1.76k
    flatten_int(&ctx->headers_buf, index, 6);
1078
1.76k
}
1079
1080
static void flatten_dynamic_indexed(struct st_h2o_qpack_flatten_context_t *ctx, int64_t index)
1081
0
{
1082
0
    h2o_vector_reserve(ctx->pool, &ctx->headers_buf, ctx->headers_buf.size + H2O_HPACK_ENCODE_INT_MAX_LENGTH);
1083
1084
0
    if (index > ctx->largest_ref)
1085
0
        ctx->largest_ref = index;
1086
1087
0
    if (index <= ctx->base_index) {
1088
        /* indexed */
1089
0
        ctx->headers_buf.entries[ctx->headers_buf.size] = 0x80;
1090
0
        flatten_int(&ctx->headers_buf, ctx->base_index - index, 6);
1091
0
    } else {
1092
        /* indexed (post-base) */
1093
0
        ctx->headers_buf.entries[ctx->headers_buf.size] = 0x10;
1094
0
        flatten_int(&ctx->headers_buf, index - ctx->base_index - 1, 4);
1095
0
    }
1096
0
}
1097
1098
static void flatten_static_nameref(struct st_h2o_qpack_flatten_context_t *ctx, int32_t index, h2o_iovec_t value, int dont_compress)
1099
5.39k
{
1100
5.39k
    h2o_vector_reserve(ctx->pool, &ctx->headers_buf, ctx->headers_buf.size + H2O_HPACK_ENCODE_INT_MAX_LENGTH * 2 + value.len);
1101
5.39k
    ctx->headers_buf.entries[ctx->headers_buf.size] = 0x50 | (dont_compress ? 0x20 : 0);
1102
5.39k
    flatten_int(&ctx->headers_buf, index, 4);
1103
5.39k
    flatten_string(&ctx->headers_buf, value.base, value.len, 7, dont_compress);
1104
5.39k
}
1105
1106
static void flatten_dynamic_nameref(struct st_h2o_qpack_flatten_context_t *ctx, int64_t index, h2o_iovec_t value, int dont_compress)
1107
0
{
1108
0
    if (index > ctx->largest_ref)
1109
0
        ctx->largest_ref = index;
1110
1111
0
    h2o_vector_reserve(ctx->pool, &ctx->headers_buf, ctx->headers_buf.size + H2O_HPACK_ENCODE_INT_MAX_LENGTH * 2 + value.len);
1112
0
    if (index <= ctx->base_index) {
1113
0
        ctx->headers_buf.entries[ctx->headers_buf.size] = 0x40 | (dont_compress ? 0x20 : 0);
1114
0
        flatten_int(&ctx->headers_buf, ctx->base_index - index, 4);
1115
0
    } else {
1116
0
        ctx->headers_buf.entries[ctx->headers_buf.size] = dont_compress ? 0x8 : 0;
1117
0
        flatten_int(&ctx->headers_buf, index - ctx->base_index - 1, 3);
1118
0
    }
1119
0
    flatten_string(&ctx->headers_buf, value.base, value.len, 7, dont_compress);
1120
0
}
1121
1122
static void flatten_without_nameref(struct st_h2o_qpack_flatten_context_t *ctx, const h2o_iovec_t *name, h2o_iovec_t value,
1123
                                    int dont_compress)
1124
0
{
1125
0
    h2o_vector_reserve(ctx->pool, &ctx->headers_buf,
1126
0
                       ctx->headers_buf.size + H2O_HPACK_ENCODE_INT_MAX_LENGTH * 2 + name->len + value.len);
1127
0
    ctx->headers_buf.entries[ctx->headers_buf.size] = 0x20 | (dont_compress ? 0x10 : 0);
1128
0
    flatten_string(&ctx->headers_buf, name->base, name->len, 3, 0);
1129
0
    flatten_string(&ctx->headers_buf, value.base, value.len, 7, dont_compress);
1130
0
}
1131
1132
static void do_flatten_header(struct st_h2o_qpack_flatten_context_t *ctx, int32_t static_index, int is_exact, int likely_to_repeat,
1133
                              const h2o_iovec_t *name, h2o_iovec_t value, h2o_header_flags_t flags)
1134
5.39k
{
1135
5.39k
    int64_t dynamic_index;
1136
1137
5.39k
    if (static_index >= 0 && is_exact) {
1138
0
        flatten_static_indexed(ctx, static_index);
1139
0
        return;
1140
0
    }
1141
1142
5.39k
    if (ctx->qpack != NULL) {
1143
        /* try dynamic indexed */
1144
0
        if ((dynamic_index = lookup_dynamic(ctx->qpack, name, value, ctx->encoder_buf == NULL, &is_exact)) >= 0 && is_exact) {
1145
0
            flatten_dynamic_indexed(ctx, dynamic_index);
1146
0
            return;
1147
0
        }
1148
        /* emit to encoder buf and dynamic index?
1149
         * At the moment the strategy is dumb; we emit encoder stream data until the table becomes full. Never triggers eviction.
1150
         */
1151
0
        if (likely_to_repeat && ctx->encoder_buf != NULL && ((static_index < 0 && dynamic_index < 0) || value.len >= 8) &&
1152
0
            name->len + value.len + HEADER_ENTRY_SIZE_OFFSET <= ctx->qpack->table.max_size - ctx->qpack->table.num_bytes) {
1153
            /* emit instruction to decoder stream */
1154
0
            if (static_index >= 0) {
1155
0
                emit_insert_with_nameref(ctx->qpack, ctx->pool, ctx->encoder_buf, 1, static_index, value);
1156
0
            } else if (dynamic_index >= 0) {
1157
0
                emit_insert_with_nameref(ctx->qpack, ctx->pool, ctx->encoder_buf, 0, dynamic_index, value);
1158
0
            } else {
1159
0
                emit_insert_without_nameref(ctx->qpack, ctx->pool, ctx->encoder_buf, name, value);
1160
0
            }
1161
            /* register the entry to table */
1162
0
            struct st_h2o_qpack_header_t *added;
1163
0
            if (h2o_iovec_is_token(name)) {
1164
0
                added = h2o_mem_alloc_shared(NULL, offsetof(struct st_h2o_qpack_header_t, value) + value.len + 1, NULL);
1165
0
                added->name = (h2o_iovec_t *)name;
1166
0
            } else {
1167
0
                added =
1168
0
                    h2o_mem_alloc_shared(NULL, offsetof(struct st_h2o_qpack_header_t, value) + name->len + 1 + value.len + 1, NULL);
1169
0
                added->name = &added->_name_buf;
1170
0
                added->_name_buf = h2o_iovec_init(added->value + added->value_len + 1, name->len);
1171
0
                memcpy(added->_name_buf.base, name->base, name->len);
1172
0
                added->_name_buf.base[name->len] = '\0';
1173
0
            }
1174
0
            added->value_len = value.len;
1175
0
            memcpy(added->value, value.base, value.len);
1176
0
            added->value[value.len] = '\0';
1177
0
            header_table_insert(&ctx->qpack->table, added);
1178
            /* emit header field to headers block */
1179
0
            flatten_dynamic_indexed(ctx, qpack_table_total_inserts(&ctx->qpack->table) - 1);
1180
0
            return;
1181
0
        }
1182
5.39k
    } else {
1183
5.39k
        dynamic_index = -1;
1184
5.39k
    }
1185
1186
5.39k
    if (static_index >= 0) {
1187
5.39k
        flatten_static_nameref(ctx, static_index, value, flags.dont_compress);
1188
5.39k
    } else if (dynamic_index >= 0) {
1189
0
        flatten_dynamic_nameref(ctx, dynamic_index, value, flags.dont_compress);
1190
0
    } else {
1191
0
        flatten_without_nameref(ctx, name, value, flags.dont_compress);
1192
0
    }
1193
5.39k
}
1194
1195
static void flatten_header(struct st_h2o_qpack_flatten_context_t *ctx, const h2o_header_t *header)
1196
1.93k
{
1197
1.93k
    int32_t static_index = -1;
1198
1.93k
    int is_exact = 0, likely_to_repeat = 0;
1199
1200
    /* obtain static index if possible */
1201
1.93k
    if (h2o_iovec_is_token(header->name)) {
1202
1.93k
        const h2o_token_t *token = H2O_STRUCT_FROM_MEMBER(h2o_token_t, buf, header->name);
1203
1.93k
        static_index = h2o_qpack_lookup_static[token - h2o__tokens](header->value, &is_exact);
1204
1.93k
        likely_to_repeat = token->flags.likely_to_repeat;
1205
1.93k
    }
1206
1207
1.93k
    return do_flatten_header(ctx, static_index, is_exact, likely_to_repeat, header->name, header->value, header->flags);
1208
1.93k
}
1209
1210
static void flatten_known_header_with_static_lookup(struct st_h2o_qpack_flatten_context_t *ctx,
1211
                                                    h2o_qpack_lookup_static_cb lookup_cb, const h2o_token_t *name,
1212
                                                    h2o_iovec_t value)
1213
0
{
1214
0
    int is_exact;
1215
0
    int32_t static_index = lookup_cb(value, &is_exact);
1216
0
    assert(index >= 0);
1217
1218
0
    do_flatten_header(ctx, static_index, is_exact, name->flags.likely_to_repeat, &name->buf, value, (h2o_header_flags_t){0});
1219
0
}
1220
1221
/* header of the qpack message that are written afterwards */
1222
static const size_t PREFIX_CAPACITY =
1223
    1 /* frame header */ + 8 /* frame payload len */ + H2O_HPACK_ENCODE_INT_MAX_LENGTH + H2O_HPACK_ENCODE_INT_MAX_LENGTH;
1224
1225
static void prepare_flatten(struct st_h2o_qpack_flatten_context_t *ctx, h2o_qpack_encoder_t *qpack, h2o_mem_pool_t *pool,
1226
                            int64_t stream_id, h2o_byte_vector_t *encoder_buf)
1227
1.93k
{
1228
1.93k
    ctx->qpack = qpack;
1229
1.93k
    ctx->pool = pool;
1230
1.93k
    ctx->stream_id = stream_id;
1231
1.93k
    ctx->encoder_buf = qpack != NULL && qpack->num_blocked < qpack->max_blocked ? encoder_buf : NULL;
1232
1.93k
    ctx->headers_buf = (h2o_byte_vector_t){NULL};
1233
1.93k
    ctx->base_index = qpack != NULL ? qpack_table_total_inserts(&qpack->table) - 1 : 0;
1234
1.93k
    ctx->largest_ref = 0;
1235
1236
    /* allocate some space, hoping to avoid realloc, but not wasting too much */
1237
1.93k
    h2o_vector_reserve(ctx->pool, &ctx->headers_buf, PREFIX_CAPACITY + 100);
1238
1.93k
    ctx->headers_buf.size = PREFIX_CAPACITY;
1239
1.93k
}
1240
1241
static h2o_iovec_t finalize_flatten(struct st_h2o_qpack_flatten_context_t *ctx, size_t *serialized_header_len)
1242
1.93k
{
1243
1.93k
    if (ctx->largest_ref == 0) {
1244
1.93k
        ctx->base_index = 0;
1245
1.93k
    } else {
1246
0
        int is_blocking = 0;
1247
        /* adjust largest reference to achieve more compact representation on wire without risking blocking */
1248
0
        if (ctx->largest_ref < ctx->qpack->largest_known_received) {
1249
0
            ctx->largest_ref = ctx->qpack->largest_known_received;
1250
0
        } else if (ctx->largest_ref > ctx->qpack->largest_known_received) {
1251
0
            assert(ctx->qpack->num_blocked < ctx->qpack->max_blocked);
1252
0
            ++ctx->qpack->num_blocked;
1253
0
            is_blocking = 1;
1254
0
        }
1255
        /* mark as inflight */
1256
0
        h2o_vector_reserve(NULL, &ctx->qpack->inflight, ctx->qpack->inflight.size + 1);
1257
0
        ctx->qpack->inflight.entries[ctx->qpack->inflight.size++] =
1258
0
            (struct st_h2o_qpack_blocked_streams_t){ctx->stream_id, ctx->largest_ref, {{is_blocking}}};
1259
0
    }
1260
1261
0
    size_t start_off = PREFIX_CAPACITY;
1262
1263
1.93k
    { /* prepend largest ref and delta base index */
1264
1.93k
        uint8_t buf[H2O_HPACK_ENCODE_INT_MAX_LENGTH * 2], *p = buf;
1265
        /* largest_ref */
1266
1.93k
        *p = 0;
1267
1.93k
        p = h2o_hpack_encode_int(p, ctx->largest_ref != 0 ? ctx->largest_ref + 1 : 0, 8);
1268
        /* delta base index */
1269
1.93k
        if (ctx->largest_ref <= ctx->base_index) {
1270
1.93k
            *p = 0;
1271
1.93k
            p = h2o_hpack_encode_int(p, ctx->base_index - ctx->largest_ref, 7);
1272
1.93k
        } else {
1273
0
            *p = 0x80;
1274
0
            p = h2o_hpack_encode_int(p, ctx->largest_ref - ctx->base_index - 1, 7);
1275
0
        }
1276
1.93k
        memcpy(ctx->headers_buf.entries + start_off - (p - buf), buf, p - buf);
1277
1.93k
        start_off -= p - buf;
1278
1.93k
    }
1279
1280
1.93k
    if (serialized_header_len != NULL)
1281
1.93k
        *serialized_header_len = ctx->headers_buf.size - start_off;
1282
1283
    /* prepend frame header */
1284
1.93k
    size_t len_len = quicly_encodev_capacity(ctx->headers_buf.size - start_off);
1285
1.93k
    quicly_encodev(ctx->headers_buf.entries + start_off - len_len, ctx->headers_buf.size - start_off);
1286
1.93k
    start_off -= len_len;
1287
1.93k
    ctx->headers_buf.entries[--start_off] = H2O_HTTP3_FRAME_TYPE_HEADERS;
1288
1289
1.93k
    return h2o_iovec_init(ctx->headers_buf.entries + start_off, ctx->headers_buf.size - start_off);
1290
1.93k
}
1291
1292
h2o_iovec_t h2o_qpack_flatten_request(h2o_qpack_encoder_t *_qpack, h2o_mem_pool_t *_pool, int64_t _stream_id,
1293
                                      h2o_byte_vector_t *_encoder_buf, h2o_iovec_t method, const h2o_url_scheme_t *scheme,
1294
                                      h2o_iovec_t authority, h2o_iovec_t path, const h2o_header_t *headers, size_t num_headers,
1295
                                      h2o_iovec_t datagram_flow_id)
1296
0
{
1297
0
    struct st_h2o_qpack_flatten_context_t ctx;
1298
1299
0
    prepare_flatten(&ctx, _qpack, _pool, _stream_id, _encoder_buf);
1300
1301
    /* pseudo headers */
1302
0
    flatten_known_header_with_static_lookup(&ctx, h2o_qpack_lookup_method, H2O_TOKEN_METHOD, method);
1303
0
    int is_connect = h2o_memis(method.base, method.len, H2O_STRLIT("CONNECT"));
1304
0
    if (!is_connect) {
1305
0
        if (scheme == &H2O_URL_SCHEME_HTTP) {
1306
0
            flatten_static_indexed(&ctx, 22);
1307
0
        } else if (scheme == &H2O_URL_SCHEME_HTTPS) {
1308
0
            flatten_static_indexed(&ctx, 23);
1309
0
        } else {
1310
0
            flatten_known_header_with_static_lookup(&ctx, h2o_qpack_lookup_scheme, H2O_TOKEN_SCHEME, scheme->name);
1311
0
        }
1312
0
    }
1313
0
    flatten_known_header_with_static_lookup(&ctx, h2o_qpack_lookup_authority, H2O_TOKEN_AUTHORITY, authority);
1314
0
    if (!is_connect)
1315
0
        flatten_known_header_with_static_lookup(&ctx, h2o_qpack_lookup_path, H2O_TOKEN_PATH, path);
1316
1317
    /* flatten headers */
1318
0
    size_t i;
1319
0
    for (i = 0; i != num_headers; ++i)
1320
0
        flatten_header(&ctx, headers + i);
1321
1322
0
    if (datagram_flow_id.base != NULL)
1323
0
        flatten_known_header_with_static_lookup(&ctx, h2o_qpack_lookup_datagram_flow_id, H2O_TOKEN_DATAGRAM_FLOW_ID,
1324
0
                                                datagram_flow_id);
1325
1326
0
    return finalize_flatten(&ctx, NULL);
1327
0
}
1328
1329
h2o_iovec_t h2o_qpack_flatten_response(h2o_qpack_encoder_t *_qpack, h2o_mem_pool_t *_pool, int64_t _stream_id,
1330
                                       h2o_byte_vector_t *_encoder_buf, int status, const h2o_header_t *headers, size_t num_headers,
1331
                                       const h2o_iovec_t *server_name, size_t content_length, h2o_iovec_t datagram_flow_id,
1332
                                       size_t *serialized_header_len)
1333
1.93k
{
1334
1.93k
    struct st_h2o_qpack_flatten_context_t ctx;
1335
1336
1.93k
    prepare_flatten(&ctx, _qpack, _pool, _stream_id, _encoder_buf);
1337
1338
    /* pseudo headers */
1339
1.93k
    switch (status) {
1340
0
#define INDEXED_STATUS(cp, st)                                                                                                     \
1341
1.76k
    case st:                                                                                                                       \
1342
1.76k
        flatten_static_indexed(&ctx, cp);                                                                                          \
1343
1.76k
        break
1344
0
        INDEXED_STATUS(24, 103);
1345
566
        INDEXED_STATUS(25, 200);
1346
0
        INDEXED_STATUS(26, 304);
1347
819
        INDEXED_STATUS(27, 404);
1348
0
        INDEXED_STATUS(28, 503);
1349
0
        INDEXED_STATUS(63, 100);
1350
0
        INDEXED_STATUS(64, 204);
1351
0
        INDEXED_STATUS(65, 206);
1352
0
        INDEXED_STATUS(66, 302);
1353
376
        INDEXED_STATUS(67, 400);
1354
7
        INDEXED_STATUS(68, 403);
1355
0
        INDEXED_STATUS(69, 421);
1356
0
        INDEXED_STATUS(70, 425);
1357
0
        INDEXED_STATUS(71, 500);
1358
0
#undef INDEXED_STATUS
1359
163
    default: {
1360
163
        char status_str[sizeof(H2O_UINT16_LONGEST_STR)];
1361
163
        sprintf(status_str, "%" PRIu16, (uint16_t)status);
1362
163
        do_flatten_header(&ctx, 24, 0, H2O_TOKEN_STATUS->flags.likely_to_repeat, &H2O_TOKEN_STATUS->buf,
1363
163
                          h2o_iovec_init(status_str, strlen(status_str)), (h2o_header_flags_t){0});
1364
163
    } break;
1365
1.93k
    }
1366
1367
    /* TODO keep some kind of reference to the indexed Server header, and reuse it */
1368
1.93k
    if (server_name != NULL && server_name->len != 0)
1369
1.93k
        do_flatten_header(&ctx, 92, 0, H2O_TOKEN_SERVER->flags.likely_to_repeat, &H2O_TOKEN_SERVER->buf, *server_name,
1370
1.93k
                          (h2o_header_flags_t){0});
1371
1372
    /* content-length */
1373
1.93k
    if (content_length != SIZE_MAX) {
1374
1.36k
        if (content_length == 0) {
1375
0
            flatten_static_indexed(&ctx, 4);
1376
1.36k
        } else {
1377
1.36k
            char cl_str[sizeof(H2O_SIZE_T_LONGEST_STR)];
1378
1.36k
            size_t cl_len = (size_t)sprintf(cl_str, "%zu", content_length);
1379
1.36k
            do_flatten_header(&ctx, 4, 0, H2O_TOKEN_CONTENT_LENGTH->flags.likely_to_repeat, &H2O_TOKEN_CONTENT_LENGTH->buf,
1380
1.36k
                              h2o_iovec_init(cl_str, cl_len), (h2o_header_flags_t){0});
1381
1.36k
        }
1382
1.36k
    }
1383
1384
    /* flatten headers */
1385
1.93k
    size_t i;
1386
3.86k
    for (i = 0; i != num_headers; ++i)
1387
1.93k
        flatten_header(&ctx, headers + i);
1388
1389
1.93k
    if (datagram_flow_id.base != NULL)
1390
0
        flatten_known_header_with_static_lookup(&ctx, h2o_qpack_lookup_datagram_flow_id, H2O_TOKEN_DATAGRAM_FLOW_ID,
1391
0
                                                datagram_flow_id);
1392
1393
1.93k
    return finalize_flatten(&ctx, serialized_header_len);
1394
1.93k
}