litespeed-quic/test/unittests/test_frame_reader.c
2017-09-22 17:00:03 -04:00

1131 lines
38 KiB
C

/* Copyright (c) 2017 LiteSpeed Technologies Inc. See LICENSE. */
#include <assert.h>
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/queue.h>
#include "lsquic.h"
#include "lsquic_frame_common.h"
#include "lsquic_arr.h"
#include "lsquic_hpack_dec.h"
#include "lsquic_mm.h"
#include "lsquic_logger.h"
#define FRAME_READER_TESTING 0x100
#include "lsquic_frame_reader.h"
struct callback_value /* What callback returns */
{
enum {
CV_HEADERS,
CV_SETTINGS,
CV_PUSH_PROMISE,
CV_PRIORITY,
CV_ERROR,
} type;
unsigned stream_off; /* Checked only if not zero */
union {
struct uncompressed_headers uh;
struct {
uint16_t id;
uint32_t value;
} setting;
void *push_promise;
struct cv_error {
enum frame_reader_error code;
uint32_t stream_id;
} error;
struct cv_priority {
uint32_t stream_id;
int exclusive;
uint32_t dep_stream_id;
unsigned weight;
} priority;
} u;
};
void
compare_headers (const struct uncompressed_headers *got_uh,
const struct uncompressed_headers *exp_uh)
{
assert(got_uh->uh_stream_id == exp_uh->uh_stream_id);
assert(got_uh->uh_oth_stream_id == exp_uh->uh_oth_stream_id);
assert(got_uh->uh_weight == exp_uh->uh_weight);
assert(got_uh->uh_exclusive == exp_uh->uh_exclusive);
assert(got_uh->uh_size == exp_uh->uh_size);
assert(strlen(got_uh->uh_headers) == got_uh->uh_size);
assert(got_uh->uh_off == exp_uh->uh_off);
assert(got_uh->uh_flags == exp_uh->uh_flags);
assert(0 == memcmp(got_uh->uh_headers, exp_uh->uh_headers, got_uh->uh_size));
}
void
compare_push_promises (const struct uncompressed_headers *got_uh,
const struct uncompressed_headers *exp_uh)
{
assert(got_uh->uh_stream_id == exp_uh->uh_stream_id);
assert(got_uh->uh_oth_stream_id == exp_uh->uh_oth_stream_id);
assert(got_uh->uh_size == exp_uh->uh_size);
assert(strlen(got_uh->uh_headers) == got_uh->uh_size);
assert(got_uh->uh_flags == exp_uh->uh_flags);
assert(0 == memcmp(got_uh->uh_headers, exp_uh->uh_headers, got_uh->uh_size));
}
void
compare_priorities (const struct cv_priority *got_prio,
const struct cv_priority *exp_prio)
{
assert(got_prio->stream_id == exp_prio->stream_id);
assert(got_prio->exclusive == exp_prio->exclusive);
assert(got_prio->dep_stream_id == exp_prio->dep_stream_id);
assert(got_prio->weight == exp_prio->weight);
}
void
compare_errors (const struct cv_error *got_err,
const struct cv_error *exp_err)
{
assert(got_err->code == exp_err->code);
assert(got_err->stream_id == exp_err->stream_id);
}
static void
compare_cb_vals (const struct callback_value *got,
const struct callback_value *exp)
{
assert(got->type == exp->type);
if (exp->stream_off)
assert(exp->stream_off == got->stream_off);
switch (got->type)
{
case CV_HEADERS:
compare_headers(&got->u.uh, &exp->u.uh);
break;
case CV_PUSH_PROMISE:
compare_push_promises(&got->u.uh, &exp->u.uh);
break;
case CV_ERROR:
compare_errors(&got->u.error, &exp->u.error);
break;
case CV_PRIORITY:
compare_priorities(&got->u.priority, &exp->u.priority);
break;
case CV_SETTINGS:
/* TODO */
break;
}
}
static struct {
size_t in_sz;
size_t in_off;
size_t in_max_req_sz;
size_t in_max_sz;
unsigned char in_buf[0x1000];
} input;
static struct cb_ctx {
unsigned n_cb_vals;
struct callback_value cb_vals[10];
} g_cb_ctx;
static void
reset_cb_ctx (struct cb_ctx *cb_ctx)
{
cb_ctx->n_cb_vals = 0;
memset(&cb_ctx->cb_vals, 0xA5, sizeof(cb_ctx->cb_vals));
}
static size_t
uh_size (const struct uncompressed_headers *uh)
{
return sizeof(*uh) - FRAME_READER_TESTING + uh->uh_size;
}
static void
on_incoming_headers (void *ctx, struct uncompressed_headers *uh)
{
struct cb_ctx *cb_ctx = ctx;
assert(cb_ctx == &g_cb_ctx);
unsigned i = cb_ctx->n_cb_vals++;
assert(i < sizeof(cb_ctx->cb_vals) / sizeof(cb_ctx->cb_vals[0]));
cb_ctx->cb_vals[i].type = CV_HEADERS;
cb_ctx->cb_vals[i].stream_off = input.in_off;
assert(uh_size(uh) <= sizeof(*uh));
memcpy(&cb_ctx->cb_vals[i].u.uh, uh, uh_size(uh) + 1 /* NUL byte */);
free(uh);
}
static void
on_push_promise (void *ctx, struct uncompressed_headers *uh)
{
struct cb_ctx *cb_ctx = ctx;
assert(cb_ctx == &g_cb_ctx);
unsigned i = cb_ctx->n_cb_vals++;
assert(i < sizeof(cb_ctx->cb_vals) / sizeof(cb_ctx->cb_vals[0]));
cb_ctx->cb_vals[i].type = CV_PUSH_PROMISE;
cb_ctx->cb_vals[i].stream_off = input.in_off;
assert(uh_size(uh) <= sizeof(*uh));
memcpy(&cb_ctx->cb_vals[i].u.uh, uh, uh_size(uh) + 1 /* NUL byte */);
free(uh);
}
static void
on_error (void *ctx, uint32_t stream_id, enum frame_reader_error error)
{
struct cb_ctx *cb_ctx = ctx;
assert(cb_ctx == &g_cb_ctx);
unsigned i = cb_ctx->n_cb_vals++;
assert(i < sizeof(cb_ctx->cb_vals) / sizeof(cb_ctx->cb_vals[0]));
cb_ctx->cb_vals[i].type = CV_ERROR;
cb_ctx->cb_vals[i].u.error.stream_id = stream_id;
cb_ctx->cb_vals[i].u.error.code = error;
cb_ctx->cb_vals[i].stream_off = input.in_off;
}
static void
on_settings (void *ctx, uint16_t id, uint32_t value)
{
struct cb_ctx *cb_ctx = ctx;
assert(cb_ctx == &g_cb_ctx);
unsigned i = cb_ctx->n_cb_vals++;
assert(i < sizeof(cb_ctx->cb_vals) / sizeof(cb_ctx->cb_vals[0]));
cb_ctx->cb_vals[i].type = CV_SETTINGS;
cb_ctx->cb_vals[i].u.setting.id = id;
cb_ctx->cb_vals[i].u.setting.value = value;
cb_ctx->cb_vals[i].stream_off = input.in_off;
}
static void
on_priority (void *ctx, uint32_t stream_id, int exclusive,
uint32_t dep_stream_id, unsigned weight)
{
struct cb_ctx *cb_ctx = ctx;
assert(cb_ctx == &g_cb_ctx);
unsigned i = cb_ctx->n_cb_vals++;
assert(i < sizeof(cb_ctx->cb_vals) / sizeof(cb_ctx->cb_vals[0]));
cb_ctx->cb_vals[i].type = CV_PRIORITY;
cb_ctx->cb_vals[i].u.priority.stream_id = stream_id;
cb_ctx->cb_vals[i].u.priority.exclusive = exclusive;
cb_ctx->cb_vals[i].u.priority.dep_stream_id = dep_stream_id;
cb_ctx->cb_vals[i].u.priority.weight = weight;
cb_ctx->cb_vals[i].stream_off = input.in_off;
}
static const struct frame_reader_callbacks frame_callbacks = {
.frc_on_headers = on_incoming_headers,
.frc_on_push_promise = on_push_promise,
.frc_on_settings = on_settings,
.frc_on_priority = on_priority,
.frc_on_error = on_error,
};
static ssize_t
read_from_stream (struct lsquic_stream *stream, void *buf, size_t sz)
{
if (sz > input.in_max_req_sz)
input.in_max_req_sz = sz;
if (input.in_sz - input.in_off < sz)
sz = input.in_sz - input.in_off;
if (sz > input.in_max_sz)
sz = input.in_max_sz;
memcpy(buf, input.in_buf + input.in_off, sz);
input.in_off += sz;
return sz;
}
struct frame_reader_test {
unsigned frt_lineno;
/* Input */
enum frame_reader_flags frt_fr_flags;
unsigned char frt_buf[0x100];
unsigned short frt_bufsz;
unsigned frt_max_headers_sz;
/* Output */
unsigned short frt_in_off;
int frt_err; /* True if expecting error */
unsigned frt_n_cb_vals;
struct callback_value frt_cb_vals[10];
};
#define UH_HEADERS(str) .uh_headers = (str), .uh_size = sizeof(str) - 1
static const struct frame_reader_test tests[] = {
{ .frt_lineno = __LINE__,
.frt_fr_flags = 0,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x04,
/* Type: */ 0x01,
/* Flags: */ HFHF_END_HEADERS,
0x80| /* <----- This bit must be ignored */
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Block fragment: */
0x48, 0x82, 0x64, 0x02,
},
.frt_bufsz = 13,
.frt_n_cb_vals = 1,
.frt_cb_vals = {
{
.type = CV_HEADERS,
.u.uh = {
.uh_stream_id = 12345,
.uh_oth_stream_id = 0,
.uh_weight = 0,
.uh_exclusive = -1,
.uh_off = 0,
UH_HEADERS("HTTP/1.1 302 Found\r\n\r\n"),
},
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = 0,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x16,
/* Type: */ 0x01,
/* Flags: */ HFHF_END_HEADERS|HFHF_PADDED,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Padding length */0x11,
/* Block fragment: */
0x48, 0x82, 0x64, 0x02,
/* Padding: */ 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
0xFF,
},
.frt_bufsz = 9 + 1 + 4 + 17,
.frt_n_cb_vals = 1,
.frt_cb_vals = {
{
.type = CV_HEADERS,
.u.uh = {
.uh_stream_id = 12345,
.uh_oth_stream_id = 0,
.uh_weight = 0,
.uh_exclusive = -1,
.uh_off = 0,
UH_HEADERS("HTTP/1.1 302 Found\r\n\r\n"),
},
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = 0,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x1B,
/* Type: */ 0x01,
/* Flags: */ HFHF_END_HEADERS|HFHF_PADDED|HFHF_PRIORITY|
HFHF_END_STREAM,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Padding length */0x11,
/* Exclusive: */ 0x80|
/* Dep Stream Id: */
0x00, 0x00, 0x12, 0x34,
/* Weight: */ 0xFF,
/* Block fragment: */
0x48, 0x82, 0x64, 0x02,
/* Padding: */ 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
0xFF,
/* Length: */ 0x00, 0x00, 0x05,
/* Type: */ HTTP_FRAME_PRIORITY,
/* Flags: */ 0x00,
/* Stream Id: */ 0x00, 0x00, 0x00, 0x39,
/* Dep Stream Id: */0x80, 0x00, 0x00, 0x19,
/* Weight: */ 0x77,
},
.frt_bufsz = 9 + 1 + 5 + 4 + 17
+ 9 + 5,
.frt_n_cb_vals = 2,
.frt_cb_vals = {
{
.type = CV_HEADERS,
.u.uh = {
.uh_stream_id = 12345,
.uh_oth_stream_id = 0x1234,
.uh_weight = 0xFF + 1,
.uh_exclusive = 1,
.uh_off = 0,
.uh_flags = UH_FIN,
UH_HEADERS("HTTP/1.1 302 Found\r\n\r\n"),
},
},
{
.type = CV_PRIORITY,
.u.priority = {
.stream_id = 0x39,
.exclusive = 1,
.dep_stream_id = 0x19,
.weight = 0x77 + 1,
},
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = 0,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x09,
/* Type: */ 0x01,
/* Flags: */ HFHF_END_HEADERS|HFHF_PRIORITY,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Exclusive: */ 0x00|
/* Dep Stream Id: */
0x00, 0x00, 0x12, 0x34,
/* Weight: */ 0x00,
/* Block fragment: */
0x48, 0x82, 0x64, 0x02,
},
.frt_bufsz = 9 + 5 + 4,
.frt_n_cb_vals = 1,
.frt_cb_vals = {
{
.type = CV_HEADERS,
.u.uh = {
.uh_stream_id = 12345,
.uh_oth_stream_id = 0x1234,
.uh_weight = 1,
.uh_exclusive = 0,
.uh_off = 0,
UH_HEADERS("HTTP/1.1 302 Found\r\n\r\n"),
},
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = 0,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x0E,
/* Type: */ 0x01,
/* Flags: */ HFHF_END_HEADERS|HFHF_PRIORITY,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Exclusive: */ 0x00|
/* Dep Stream Id: */
0x00, 0x00, 0x12, 0x34,
/* Weight: */ 0x00,
/* Block fragment: */
0x48, 0x82, 0x64, 0x02,
0x60, 0x03, 0x61, 0x3d, 0x62,
},
.frt_bufsz = 9 + 5 + 4 + 5,
.frt_n_cb_vals = 1,
.frt_cb_vals = {
{
.type = CV_HEADERS,
.u.uh = {
.uh_stream_id = 12345,
.uh_oth_stream_id = 0x1234,
.uh_weight = 1,
.uh_exclusive = 0,
.uh_off = 0,
UH_HEADERS("HTTP/1.1 302 Found\r\n"
"Cookie: a=b\r\n\r\n"),
},
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = 0,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x18,
/* Type: */ 0x01,
/* Flags: */ HFHF_END_HEADERS|HFHF_PRIORITY,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Exclusive: */ 0x00|
/* Dep Stream Id: */
0x00, 0x00, 0x12, 0x34,
/* Weight: */ 0x00,
/* Block fragment: */
0x48, 0x82, 0x64, 0x02,
0x60, 0x03, 0x61, 0x3d, 0x62,
0x60, 0x03, 0x63, 0x3d, 0x64,
0x60, 0x03, 0x65, 0x3d, 0x66,
},
.frt_bufsz = 9 + 5 + 4 + 15,
.frt_n_cb_vals = 1,
.frt_cb_vals = {
{
.type = CV_HEADERS,
.u.uh = {
.uh_stream_id = 12345,
.uh_oth_stream_id = 0x1234,
.uh_weight = 1,
.uh_exclusive = 0,
.uh_off = 0,
UH_HEADERS("HTTP/1.1 302 Found\r\n"
"Cookie: a=b; c=d; e=f\r\n\r\n"),
},
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = FRF_SERVER,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x16,
/* Type: */ 0x01,
/* Flags: */ HFHF_END_HEADERS|HFHF_PRIORITY,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Exclusive: */ 0x00|
/* Dep Stream Id: */
0x00, 0x00, 0x12, 0x34,
/* Weight: */ 0x00,
/* Block fragment: */
0x82, 0x84, 0x86, 0x41, 0x8c, 0xf1, 0xe3, 0xc2,
0xe5, 0xf2, 0x3a, 0x6b, 0xa0, 0xab, 0x90, 0xf4,
0xff,
/* Length: */ 0x00, 0x00, 0xEE,
/* Type: */ HTTP_FRAME_CONTINUATION,
/* Flags: */ HFHF_END_HEADERS,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Block fragment: */
'W', 'H', 'A', 'T', 'E', 'V', 'E', 'R',
},
.frt_bufsz = 9 + 5 + 17
+ 9 + 0 + 8,
.frt_err = 1,
.frt_in_off = 9 + 5 + 17 + 9,
.frt_n_cb_vals = 1,
.frt_cb_vals = {
{
.type = CV_HEADERS,
.u.uh = {
.uh_stream_id = 12345,
.uh_oth_stream_id = 0x1234,
.uh_weight = 1,
.uh_exclusive = 0,
.uh_off = 0,
UH_HEADERS("GET / HTTP/1.1\r\nHost: www.example.com\r\n\r\n"),
},
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = FRF_SERVER,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x16,
/* Type: */ 0x01,
/* Flags: */ HFHF_END_HEADERS|HFHF_PRIORITY,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Exclusive: */ 0x00|
/* Dep Stream Id: */
0x00, 0x00, 0x12, 0x34,
/* Weight: */ 0x00,
/* Block fragment: */
0x82, 0x84, 0x86, 0x41, 0x8c, 0xf1, 0xe3, 0xc2,
0xe5, 0xf2, 0x3a, 0x6b, 0xa0, 0xab, 0x90, 0xf4,
0xff,
/* Length: */ 0x00, 0x00, 0xEE,
/* Type: */ HTTP_FRAME_CONTINUATION,
/* Flags: */ HFHF_END_HEADERS,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Block fragment: */
'W', 'H', 'A', 'T', 'E', 'V', 'E', 'R',
},
.frt_bufsz = 9 + 5 + 17
+ 9 + 0 + 8,
.frt_err = 1,
.frt_in_off = 9 + 5 + 17 + 9,
.frt_n_cb_vals = 1,
.frt_cb_vals = {
{
.type = CV_HEADERS,
.u.uh = {
.uh_stream_id = 12345,
.uh_oth_stream_id = 0x1234,
.uh_weight = 1,
.uh_exclusive = 0,
.uh_off = 0,
UH_HEADERS("GET / HTTP/1.1\r\nHost: www.example.com\r\n\r\n"),
},
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = FRF_SERVER,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x16,
/* Type: */ 0x01,
/* Flags: */ HFHF_PRIORITY,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Exclusive: */ 0x00|
/* Dep Stream Id: */
0x00, 0x00, 0x12, 0x34,
/* Weight: */ 0x00,
/* Block fragment: */
0x82, 0x84, 0x86, 0x41, 0x8c, 0xf1, 0xe3, 0xc2,
0xe5, 0xf2, 0x3a, 0x6b, 0xa0, 0xab, 0x90, 0xf4,
0xff,
/* Length: */ 0x00, 0x00, 0xEE,
/* Type: */ HTTP_FRAME_CONTINUATION,
/* Flags: */ HFHF_END_HEADERS,
/* Stream Id: */ 0x00, 0xFF, 0x30, 0x39, /* Stream ID does not match */
/* Block fragment: */
'W', 'H', 'A', 'T', 'E', 'V', 'E', 'R',
},
.frt_bufsz = 9 + 5 + 17
+ 9 + 0 + 8,
.frt_err = 1,
.frt_in_off = 9 + 5 + 17 + 9,
.frt_n_cb_vals = 0,
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = FRF_SERVER,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0xEE,
/* Type: */ HTTP_FRAME_CONTINUATION,
/* Flags: */ HFHF_END_HEADERS,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Block fragment: */
'W', 'H', 'A', 'T', 'E', 'V', 'E', 'R',
},
.frt_bufsz = 9 + 0 + 8,
.frt_err = 1,
.frt_in_off = 9,
.frt_n_cb_vals = 0,
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = FRF_SERVER,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x10,
/* Type: */ 0x01,
/* Flags: */ 0x00, /* Note absence of HFHF_END_HEADERS */
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Block fragment:
* perl hpack.pl :method GET :path / host www.example.com
*/
0x82, 0x84, 0x66, 0x8c, 0xf1, 0xe3, 0xc2, 0xe5,
0xf2, 0x3a, 0x6b, 0xa0, 0xab, 0x90, 0xf4, 0xff,
/* Length: */ 0x00, 0x00, 0x08,
/* Type: */ 0x01,
/* Flags: */ HFHF_END_HEADERS,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Block fragment: */
'W', 'H', 'A', 'T', 'E', 'V', 'E', 'R',
},
.frt_bufsz = 9 + 0 + 16
+ 9 + 0 + 8,
.frt_in_off = 9 + 16 + 9,
.frt_err = 1,
.frt_n_cb_vals = 1,
.frt_cb_vals = {
{
.type = CV_ERROR,
.u.error = {
.stream_id = 0x3039,
.code = FR_ERR_EXPECTED_CONTIN,
},
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = FRF_SERVER,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x10,
/* Type: */ 0x01,
/* Flags: */ HFHF_END_HEADERS,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Block fragment:
* perl hpack.pl :method GET :path / host www.example.com
*/
0x82, 0x84, 0x66, 0x8c, 0xf1, 0xe3, 0xc2, 0xe5,
0xf2, 0x3a, 0x6b, 0xa0, 0xab, 0x90, 0xf4, 0xff,
/* Length: */ 0x00, 0x00, 0x1A,
/* Type: */ 0x01,
/* Flags: */ HFHF_END_HEADERS|HFHF_PRIORITY,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Exclusive: */ 0x00|
/* Dep Stream Id: */
0x00, 0x00, 0x12, 0x34,
/* Weight: */ 0x00,
/* Block fragment:
* perl hpack.pl :method GET :path / :scheme http Host www.example.com
*/
0x82, 0x84, 0x86, 0x40, 0x83, 0xc6, 0x74, 0x27,
0x8c, 0xf1, 0xe3, 0xc2, 0xe5, 0xf2, 0x3a, 0x6b,
0xa0, 0xab, 0x90, 0xf4, 0xff,
/* Length: */ 0x00, 0x00, 0x11,
/* Type: */ 0x01,
/* Flags: */ HFHF_END_HEADERS,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Block fragment: */
0x82, 0x84, 0x86, 0x41, 0x8c, 0xf1, 0xe3, 0xc2,
0xe5, 0xf2, 0x3a, 0x6b, 0xa0, 0xab, 0x90, 0xf4,
0xff,
},
.frt_bufsz = 9 + 0 + 16
+ 9 + 5 + 21
+ 9 + 0 + 17,
.frt_n_cb_vals = 3,
.frt_cb_vals = {
{
.type = CV_ERROR,
.u.error = {
.stream_id = 12345,
.code = FR_ERR_INCOMPL_REQ_PSEH,
},
},
{
.type = CV_ERROR,
.u.error = {
.stream_id = 12345,
.code = FR_ERR_UPPERCASE_HEADER,
},
},
{
.type = CV_HEADERS,
.u.uh = {
.uh_stream_id = 12345,
.uh_oth_stream_id = 0,
.uh_weight = 0,
.uh_exclusive = -1,
.uh_off = 0,
UH_HEADERS("GET / HTTP/1.1\r\nHost: www.example.com\r\n\r\n"),
},
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = 0,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x15,
/* Type: */ HTTP_FRAME_PUSH_PROMISE,
/* Flags: */ HFHF_END_HEADERS,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Dep stream Id: */0x00, 0x12, 0x34, 0x56,
/* Block fragment: */
0x82, 0x84, 0x86, 0x41, 0x8c, 0xf1, 0xe3, 0xc2,
0xe5, 0xf2, 0x3a, 0x6b, 0xa0, 0xab, 0x90, 0xf4,
0xff,
},
.frt_bufsz = 9 + 0 + 0x15,
.frt_n_cb_vals = 1,
.frt_cb_vals = {
{
.type = CV_PUSH_PROMISE,
.u.uh = {
.uh_stream_id = 12345,
.uh_oth_stream_id = 0x123456,
.uh_flags = UH_PP,
UH_HEADERS("GET / HTTP/1.1\r\nHost: www.example.com\r\n\r\n"),
},
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = 0,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x02,
/* Type: */ HTTP_FRAME_HEADERS,
/* Flags: */ 0x00,
0x80| /* <----- This bit must be ignored */
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Block fragment: */
0x48, 0x82,
/* Length: */ 0x00, 0x00, 0x02,
/* Type: */ HTTP_FRAME_CONTINUATION,
/* Flags: */ HFHF_END_HEADERS,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Block fragment: */
0x64, 0x02,
},
.frt_bufsz = 9 + 2 + 9 + 2,
.frt_n_cb_vals = 1,
.frt_cb_vals = {
{
.type = CV_HEADERS,
.u.uh = {
.uh_stream_id = 12345,
.uh_oth_stream_id = 0,
.uh_weight = 0,
.uh_exclusive = -1,
.uh_off = 0,
UH_HEADERS("HTTP/1.1 302 Found\r\n\r\n"),
},
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = 0,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x00,
/* Type: */ HTTP_FRAME_SETTINGS,
/* Flags: */ 0x00,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
},
.frt_bufsz = 9,
.frt_n_cb_vals = 1,
.frt_err = 1,
.frt_cb_vals = {
{
.type = CV_ERROR,
.u.error.code = FR_ERR_INVALID_FRAME_SIZE,
.u.error.stream_id = 12345,
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = 0,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x07,
/* Type: */ HTTP_FRAME_SETTINGS,
/* Flags: */ 0x00,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
},
.frt_bufsz = 9 + 7,
.frt_n_cb_vals = 1,
.frt_err = 1,
.frt_in_off = 9,
.frt_cb_vals = {
{
.type = CV_ERROR,
.u.error.code = FR_ERR_INVALID_FRAME_SIZE,
.u.error.stream_id = 12345,
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = 0,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x06,
/* Type: */ HTTP_FRAME_SETTINGS,
/* Flags: */ 0x00,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
},
.frt_bufsz = 9 + 6,
.frt_n_cb_vals = 1,
.frt_err = 1,
.frt_in_off = 9,
.frt_cb_vals = {
{
.type = CV_ERROR,
.u.error.code = FR_ERR_NONZERO_STREAM_ID,
.u.error.stream_id = 12345,
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = 0,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x0C,
/* Type: */ HTTP_FRAME_SETTINGS,
/* Flags: */ 0x00,
/* Stream Id: */ 0x00, 0x00, 0x00, 0x00,
0x00, SETTINGS_INITIAL_WINDOW_SIZE,
0x01, 0x02, 0x03, 0x04,
0x00, SETTINGS_HEADER_TABLE_SIZE,
0x02, 0x03, 0x04, 0x05,
},
.frt_bufsz = 9 + 12,
.frt_n_cb_vals = 2,
.frt_cb_vals = {
{
.type = CV_SETTINGS,
.u.setting.id = SETTINGS_INITIAL_WINDOW_SIZE,
.u.setting.value = 0x01020304,
},
{
.type = CV_SETTINGS,
.u.setting.id = SETTINGS_HEADER_TABLE_SIZE,
.u.setting.value = 0x02030405,
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = 0,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x09,
/* Type: */ 0x01,
/* Flags: */ HFHF_END_HEADERS|HFHF_PRIORITY,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Exclusive: */ 0x00|
/* Dep Stream Id: */
0x00, 0x00, 0x12, 0x34,
/* Weight: */ 0x00,
/* Block fragment: */
0x48, 0x82, 0x64, 0x02,
/* Length: */ 0x00, 0x00, 0x06,
/* Type: */ HTTP_FRAME_SETTINGS,
/* Flags: */ 0x00,
/* Stream Id: */ 0x00, 0x00, 0x00, 0x00,
0x00, SETTINGS_INITIAL_WINDOW_SIZE,
0x01, 0x02, 0x03, 0x04,
},
.frt_bufsz = 9 + 5 + 4 + 9 + 6,
.frt_max_headers_sz = 10,
.frt_n_cb_vals = 2,
.frt_cb_vals = {
{
.type = CV_ERROR,
.stream_off = 9 + 5 + 4,
.u.error.code = FR_ERR_HEADERS_TOO_LARGE,
.u.error.stream_id = 12345,
},
{
.type = CV_SETTINGS,
.u.setting.id = SETTINGS_INITIAL_WINDOW_SIZE,
.u.setting.value = 0x01020304,
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = 0,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x11,
/* Type: */ 0x01,
/* Flags: */ HFHF_END_HEADERS,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Block fragment: */
/* 0x11 bytes of no consequence: they are not
* parsed.
*/
000, 001, 002, 003, 004, 005, 006, 007,
010, 011, 012, 013, 014, 015, 016, 017,
020,
/* Length: */ 0x00, 0x00, 0x06,
/* Type: */ HTTP_FRAME_SETTINGS,
/* Flags: */ 0x00,
/* Stream Id: */ 0x00, 0x00, 0x00, 0x00,
0x00, SETTINGS_INITIAL_WINDOW_SIZE,
0x01, 0x02, 0x03, 0x04,
},
.frt_bufsz = 9 + 0 + 0x11 + 9 + 6,
.frt_max_headers_sz = 0x10,
.frt_n_cb_vals = 2,
.frt_cb_vals = {
{
.type = CV_ERROR,
.stream_off = 9,
.u.error.code = FR_ERR_HEADERS_TOO_LARGE,
.u.error.stream_id = 12345,
},
{
.type = CV_SETTINGS,
.u.setting.id = SETTINGS_INITIAL_WINDOW_SIZE,
.u.setting.value = 0x01020304,
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = 0,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x10,
/* Type: */ 0x01,
/* Flags: */ 0x00,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Block fragment: */
/* 0x10 bytes of no consequence: they are not
* parsed.
*/
000, 001, 002, 003, 004, 005, 006, 007,
010, 011, 012, 013, 014, 015, 016, 017,
/* Length: */ 0x00, 0x00, 0x10,
/* Type: */ HTTP_FRAME_CONTINUATION,
/* Flags: */ 0x00,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Block fragment: */
000, 001, 002, 003, 004, 005, 006, 007,
010, 011, 012, 013, 014, 015, 016, 017,
/* Length: */ 0x00, 0x00, 0x10,
/* Type: */ HTTP_FRAME_CONTINUATION,
/* Flags: */ HFHF_END_HEADERS,
/* Stream Id: */ 0x00, 0x00, 0x30, 0x39,
/* Block fragment: */
000, 001, 002, 003, 004, 005, 006, 007,
010, 011, 012, 013, 014, 015, 016, 017,
/* Length: */ 0x00, 0x00, 0x06,
/* Type: */ HTTP_FRAME_SETTINGS,
/* Flags: */ 0x00,
/* Stream Id: */ 0x00, 0x00, 0x00, 0x00,
0x00, SETTINGS_INITIAL_WINDOW_SIZE,
0x01, 0x02, 0x03, 0x04,
},
.frt_bufsz = 9 + 0 + 0x10 + 9 + 0 + 0x10 + 9 + 0 + 0x10 + 9 + 6,
.frt_max_headers_sz = 0x19,
.frt_n_cb_vals = 2,
.frt_cb_vals = {
{
.type = CV_ERROR,
.stream_off = 9 + 0 + 0x10 + 9,
.u.error.code = FR_ERR_HEADERS_TOO_LARGE,
.u.error.stream_id = 12345,
},
{
.type = CV_SETTINGS,
.u.setting.id = SETTINGS_INITIAL_WINDOW_SIZE,
.u.setting.value = 0x01020304,
},
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = 0,
.frt_buf = {
/* Length: */ 0x00, 0x00,
0x04, /* <-- wrong payload size */
/* Type: */ HTTP_FRAME_PRIORITY,
/* Flags: */ 0x00,
/* Stream Id: */ 0x00, 0x00, 0x00, 0x39,
/* Dep Stream Id: */0x80, 0x00, 0x00, 0x19,
/* Weight: */ 0x77,
},
.frt_bufsz = 9 + 5,
.frt_n_cb_vals = 1,
.frt_err = 1,
.frt_in_off = 9,
.frt_cb_vals = {
{
.type = CV_ERROR,
.stream_off = 9,
.u.error.code = FR_ERR_INVALID_FRAME_SIZE,
.u.error.stream_id = 0x39,
}
},
},
{ .frt_lineno = __LINE__,
.frt_fr_flags = 0,
.frt_buf = {
/* Length: */ 0x00, 0x00, 0x05,
/* Type: */ HTTP_FRAME_PRIORITY,
/* Flags: */ 0x00,
/* Stream Id: */ 0x00, 0x00, 0x00, 0x00, /* Invalid stream ID */
/* Dep Stream Id: */0x80, 0x00, 0x00, 0x19,
/* Weight: */ 0x77,
},
.frt_bufsz = 9 + 5,
.frt_n_cb_vals = 1,
.frt_err = 1,
.frt_in_off = 9,
.frt_cb_vals = {
{
.type = CV_ERROR,
.stream_off = 9,
.u.error.code = FR_ERR_ZERO_STREAM_ID,
.u.error.stream_id = 0x00,
}
},
},
{
.frt_bufsz = 0,
},
};
static void
test_one_frt (const struct frame_reader_test *frt)
{
struct lsquic_frame_reader *fr;
unsigned short exp_off;
struct lsquic_hdec hdec;
struct lsquic_mm mm;
int s;
lsquic_mm_init(&mm);
lsquic_hdec_init(&hdec);
memset(&input, 0, sizeof(input));
memcpy(input.in_buf, frt->frt_buf, frt->frt_bufsz);
input.in_sz = frt->frt_bufsz;
do
{
reset_cb_ctx(&g_cb_ctx);
input.in_off = 0;
++input.in_max_sz;
fr = lsquic_frame_reader_new(frt->frt_fr_flags, frt->frt_max_headers_sz,
&mm, NULL, read_from_stream, &hdec, &frame_callbacks, &g_cb_ctx);
do
{
s = lsquic_frame_reader_read(fr);
if (s != 0)
break;
}
while (input.in_off < input.in_sz);
assert(frt->frt_err || 0 == s);
assert(g_cb_ctx.n_cb_vals == frt->frt_n_cb_vals);
unsigned i;
for (i = 0; i < g_cb_ctx.n_cb_vals; ++i)
compare_cb_vals(&g_cb_ctx.cb_vals[i], &frt->frt_cb_vals[i]);
exp_off = frt->frt_in_off;
if (!exp_off)
exp_off = frt->frt_bufsz;
assert(input.in_off == exp_off);
lsquic_frame_reader_destroy(fr);
}
while (input.in_max_sz < input.in_max_req_sz);
lsquic_hdec_cleanup(&hdec);
lsquic_mm_cleanup(&mm);
}
int
main (int argc, char **argv)
{
int opt;
while (-1 != (opt = getopt(argc, argv, "l:")))
{
switch (opt)
{
case 'l':
lsquic_log_to_fstream(stderr, LLTS_NONE);
lsquic_logger_lopt(optarg);
break;
default:
exit(1);
}
}
const struct frame_reader_test *frt;
for (frt = tests; frt->frt_bufsz > 0; ++frt)
test_one_frt(frt);
return 0;
}