/* Copyright (c) 2017 LiteSpeed Technologies Inc. See LICENSE. */ #include #include #include #include #include #include #include "lsquic.h" #include "lsquic_types.h" #include "lsquic_alarmset.h" #include "lsquic_packet_common.h" #include "lsquic_parse.h" #include "lsquic_mm.h" #include "lsquic_packet_in.h" #include "lsquic_engine_public.h" #include "lsquic_version.h" static const struct parse_funcs *const pf = select_pf_by_ver(LSQVER_037); /* The struct is used to test both generation and parsing of version * negotiation packet. */ struct gen_ver_nego_test { int gvnt_lineno; /* Generate: inputs; parse: outputs */ lsquic_cid_t gvnt_cid; unsigned gvnt_versions; size_t gvnt_bufsz; /* Generate: outputs; parse: inputs */ int gvnt_len; /* Retval */ char gvnt_buf[0x40]; /* Contents */ }; static const struct gen_ver_nego_test tests[] = { { .gvnt_lineno = __LINE__, .gvnt_cid = 0x0102030405060708UL, .gvnt_versions = (1 << LSQVER_037), .gvnt_bufsz = 13, .gvnt_len = 13, .gvnt_buf = { PACKET_PUBLIC_FLAGS_VERSION| PACKET_PUBLIC_FLAGS_8BYTE_CONNECTION_ID, 0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01, /* Connection ID */ 'Q', '0', '3', '7', }, }, { .gvnt_lineno = __LINE__, .gvnt_cid = 0x0102030405060708UL, .gvnt_versions = (1 << LSQVER_037), .gvnt_bufsz = 12, .gvnt_len = -1, /* Buffer size is too small */ }, { .gvnt_lineno = __LINE__, .gvnt_cid = 0x0102030405060708UL, .gvnt_versions = (1 << LSQVER_037) | (1 << N_LSQVER), .gvnt_bufsz = 20, .gvnt_len = -1, /* Invalid version specified in the bitmask */ }, { .gvnt_lineno = __LINE__, .gvnt_cid = 0x0102030405060708UL, .gvnt_versions = (1 << LSQVER_037) | (1 << LSQVER_035) | (1 << LSQVER_038), .gvnt_bufsz = 21, .gvnt_len = 21, .gvnt_buf = { PACKET_PUBLIC_FLAGS_VERSION| PACKET_PUBLIC_FLAGS_8BYTE_CONNECTION_ID, 0x08, 0x07, 0x06, 0x05, 0x04, 0x03, 0x02, 0x01, /* Connection ID */ 'Q', '0', '3', '5', 'Q', '0', '3', '7', 'Q', '0', '3', '8', }, }, }; static void test_parsing_ver_nego (const struct gen_ver_nego_test *gvnt) { int s; lsquic_packet_in_t *packet_in; struct lsquic_mm mm; struct ver_iter vi; lsquic_ver_tag_t ver_tag; enum lsquic_version version; struct packin_parse_state ppstate; unsigned version_bitmask = gvnt->gvnt_versions; lsquic_mm_init(&mm); packet_in = lsquic_mm_get_packet_in(&mm); packet_in->pi_data = lsquic_mm_get_1370(&mm); packet_in->pi_flags |= PI_OWN_DATA; memcpy(packet_in->pi_data, gvnt->gvnt_buf, gvnt->gvnt_len); s = parse_packet_in_begin(packet_in, gvnt->gvnt_len, 0, &ppstate); assert(s == 0); for (s = packet_in_ver_first(packet_in, &vi, &ver_tag); s; s = packet_in_ver_next(&vi, &ver_tag)) { version = lsquic_tag2ver(ver_tag); assert(version < N_LSQVER); assert(version_bitmask & (1 << version)); version_bitmask &= ~(1 << version); } assert(0 == version_bitmask); lsquic_mm_put_packet_in(&mm, packet_in); lsquic_mm_cleanup(&mm); } static void run_gvnt (int i) { const struct gen_ver_nego_test *const gvnt = &tests[i]; unsigned char out[0x40]; assert(sizeof(out) <= sizeof(gvnt->gvnt_buf)); /* Internal sanity check */ int len = pf->pf_gen_ver_nego_pkt(out, gvnt->gvnt_bufsz, gvnt->gvnt_cid, gvnt->gvnt_versions); assert(("Packet length is correct", len == gvnt->gvnt_len)); if (gvnt->gvnt_len > 0) { assert(("Packet contents are correct", 0 == memcmp(out, gvnt->gvnt_buf, gvnt->gvnt_len))); test_parsing_ver_nego(gvnt); } } int main (void) { unsigned i; for (i = 0; i < sizeof(tests) / sizeof(tests[0]); ++i) run_gvnt(i); return 0; }