/* Copyright (c) 2017 LiteSpeed Technologies Inc. See LICENSE. */ #include #include #include #include #include #include "lsquic_types.h" #include "lsquic_alarmset.h" #include "lsquic_packet_common.h" #include "lsquic_parse.h" #include "lsquic.h" struct test { /* Inputs. */ const struct parse_funcs *pf; size_t bufsz; lsquic_cid_t cid; /* Zero means connection ID is not specified */ const char *nonce; lsquic_packno_t packno; enum lsquic_packno_bits bits; /* The test has been retrofitted by adding bits parameter. The test can * be made more complicated by calculating packet number length based on * some other inputs. However, this is tested elsewhere. */ union { unsigned char buf[4]; lsquic_ver_tag_t val; } ver; /* Outputs */ int len; /* Retval */ char out[0x100]; /* Contents */ }; static const struct test tests[] = { { .pf = select_pf_by_ver(LSQVER_037), .bufsz = QUIC_MAX_PUBHDR_SZ, .cid = 0x0102030405060708UL, .nonce = NULL, .packno = 0x01020304, .bits = PACKNO_LEN_4, .len = 1 + 8 + 0 + 4, .out = { (0 << 2) /* Nonce present */ | 0x08 /* Connection ID present */ | 0x20 /* Packet number length */ , 0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01, /* Connection ID */ 0x04, 0x03, 0x02, 0x01, /* Packet number */ }, }, { .pf = select_pf_by_ver(LSQVER_039), .bufsz = QUIC_MAX_PUBHDR_SZ, .cid = 0x0102030405060708UL, .nonce = NULL, .packno = 0x01020304, .bits = PACKNO_LEN_4, .len = 1 + 8 + 0 + 4, .out = { (0 << 2) /* Nonce present */ | 0x08 /* Connection ID present */ | 0x20 /* Packet number length */ , 0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01, /* Connection ID */ 0x01, 0x02, 0x03, 0x04, /* Packet number */ }, }, { .pf = select_pf_by_ver(LSQVER_037), .bufsz = QUIC_MAX_PUBHDR_SZ, .cid = 0x0102030405060708UL, .nonce = NULL, .packno = 0x00, .bits = PACKNO_LEN_1, .len = 1 + 8 + 0 + 1, .out = { (0 << 2) /* Nonce present */ | 0x08 /* Connection ID present */ | 0x00 /* Packet number length */ , 0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01, /* Connection ID */ 0x00, /* Packet number */ }, }, { .pf = select_pf_by_ver(LSQVER_037), .bufsz = QUIC_MAX_PUBHDR_SZ, .cid = 0x0102030405060708UL, .nonce = NULL, .packno = 0x00, .bits = PACKNO_LEN_1, .ver.buf= { 'Q', '0', '3', '5', }, .len = 1 + 8 + 4 + 0 + 1, .out = { (0 << 2) /* Nonce present */ | 0x01 /* Version present */ | 0x08 /* Connection ID present */ | 0x00 /* Packet number length */ , 0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01, /* Connection ID */ 'Q', '0', '3', '5', 0x00, /* Packet number */ }, }, { .pf = select_pf_by_ver(LSQVER_039), .bufsz = QUIC_MAX_PUBHDR_SZ, .cid = 0x0102030405060708UL, .nonce = NULL, .packno = 0x09, .bits = PACKNO_LEN_1, .ver.buf= { 'Q', '0', '3', '9', }, .len = 1 + 8 + 4 + 0 + 1, .out = { (0 << 2) /* Nonce present */ | 0x01 /* Version present */ | 0x08 /* Connection ID present */ | 0x00 /* Packet number length */ , 0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01, /* Connection ID */ 'Q', '0', '3', '9', 0x09, /* Packet number */ }, }, #define NONCENSE "0123456789abcdefghijklmnopqrstuv" #define NONCENSE_BYTES '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v' { .pf = select_pf_by_ver(LSQVER_037), .bufsz = QUIC_MAX_PUBHDR_SZ, .cid = 0x0102030405060708UL, .nonce = NONCENSE, .packno = 0x00, .bits = PACKNO_LEN_1, .len = 1 + 8 + 32 + 1, .out = { (1 << 2) /* Nonce present */ | 0x08 /* Connection ID present */ | 0x00 /* Packet number length */ , 0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01, /* Connection ID */ NONCENSE_BYTES, 0x00, /* Packet number */ }, }, { .pf = select_pf_by_ver(LSQVER_037), .bufsz = QUIC_MAX_PUBHDR_SZ, .cid = 0, /* Do not set connection ID */ .nonce = NONCENSE, .packno = 0x00, .bits = PACKNO_LEN_1, .len = 1 + 0 + 32 + 1, .out = { (1 << 2) /* Nonce present */ | 0x00 /* Packet number length */ , NONCENSE_BYTES, 0x00, /* Packet number */ }, }, { .pf = select_pf_by_ver(LSQVER_037), .bufsz = QUIC_MAX_PUBHDR_SZ, .cid = 0x0102030405060708UL, .nonce = NONCENSE, .packno = 0x00, .bits = PACKNO_LEN_1, .ver.buf= { 'Q', '0', '3', '5', }, .len = 1 + 8 + 4 + 32 + 1, .out = { (1 << 2) /* Nonce present */ | 0x01 /* Version present */ | 0x08 /* Connection ID present */ | 0x00 /* Packet number length */ , 0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01, /* Connection ID */ 'Q', '0', '3', '5', NONCENSE_BYTES, 0x00, /* Packet number */ }, }, { .pf = select_pf_by_ver(LSQVER_037), .bufsz = QUIC_MAX_PUBHDR_SZ, .cid = 0x0102030405060708UL, .nonce = NONCENSE, .packno = 0xA0A1A2A3A4A5A6A7UL, .bits = PACKNO_LEN_6, .len = 1 + 8 + 32 + 6, .out = { (1 << 2) /* Nonce present */ | 0x08 /* Connection ID present */ | 0x30 /* Packet number length */ , 0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01, /* Connection ID */ NONCENSE_BYTES, 0xA7, 0xA6, 0xA5, 0xA4, 0xA3, 0xA2, }, }, { .pf = select_pf_by_ver(LSQVER_039), .bufsz = QUIC_MAX_PUBHDR_SZ, .cid = 0x0102030405060708UL, .nonce = NONCENSE, .packno = 0xA0A1A2A3A4A5A6A7UL, .bits = PACKNO_LEN_6, .len = 1 + 8 + 32 + 6, .out = { (1 << 2) /* Nonce present */ | 0x08 /* Connection ID present */ | 0x30 /* Packet number length */ , 0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01, /* Connection ID */ NONCENSE_BYTES, 0xA2, 0xA3, 0xA4, 0xA5, 0xA6, 0xA7, }, }, }; static void run_test (int i) { const struct test *const test = &tests[i]; unsigned char out[QUIC_MAX_PUBHDR_SZ]; int len = test->pf->pf_gen_reg_pkt_header(out, sizeof(out), (test->cid ? &test->cid : NULL), (test->ver.val ? &test->ver.val : NULL), (unsigned char *) test->nonce, test->packno, test->bits); assert(("Packet length is correct", len == test->len)); if (test->len > 0) assert(("Packet contents are correct", 0 == memcmp(out, test->out, len))); } int main (void) { unsigned i; for (i = 0; i < sizeof(tests) / sizeof(tests[0]); ++i) run_test(i); return 0; }