mirror of
https://gitea.invidious.io/iv-org/litespeed-quic.git
synced 2024-08-15 00:53:43 +00:00
456 lines
14 KiB
C
456 lines
14 KiB
C
|
/* Copyright (c) 2017 - 2019 LiteSpeed Technologies Inc. See LICENSE. */
|
||
|
#include <assert.h>
|
||
|
#include <stdio.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
#ifndef WIN32
|
||
|
#include <sys/time.h>
|
||
|
#endif
|
||
|
|
||
|
#include "lsquic_types.h"
|
||
|
#include "lsquic_parse.h"
|
||
|
#include "lsquic_rechist.h"
|
||
|
#include "lsquic_util.h"
|
||
|
#include "lsquic.h"
|
||
|
|
||
|
static const struct parse_funcs *const pf = select_pf_by_ver(LSQVER_041);
|
||
|
|
||
|
|
||
|
static lsquic_packno_t
|
||
|
n_acked (const ack_info_t *acki)
|
||
|
{
|
||
|
lsquic_packno_t n = 0;
|
||
|
unsigned i;
|
||
|
for (i = 0; i < acki->n_ranges; ++i)
|
||
|
n += acki->ranges[i].high - acki->ranges[i].low + 1;
|
||
|
return n;
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
test1 (void)
|
||
|
{
|
||
|
/* Test taken from quic_framer_test.cc -- NewAckFrameOneAckBlock */
|
||
|
unsigned char ack_buf[] = {
|
||
|
/* TYPE N LL MM */
|
||
|
0xA0 | 0 << 4 | 1 << 2 | 1,
|
||
|
0x00, /* Number of timestamps */
|
||
|
0x12, 0x34, /* Largest acked */
|
||
|
0x00, 0x00, /* Zero delta time */
|
||
|
0x12, 0x34, /* first ack block length */
|
||
|
};
|
||
|
|
||
|
ack_info_t acki;
|
||
|
memset(&acki, 0xF1, sizeof(acki));
|
||
|
|
||
|
int len = pf->pf_parse_ack_frame(ack_buf, sizeof(ack_buf), &acki);
|
||
|
assert(("Parsed length is correct (8)", len == sizeof(ack_buf)));
|
||
|
assert(("Number of ranges is 1", acki.n_ranges == 1));
|
||
|
assert(("Largest acked is 0x1234", acki.ranges[0].high == 0x1234));
|
||
|
assert(("Lowest acked is 1", acki.ranges[0].low == 1));
|
||
|
assert(("Number of timestamps is 0", acki.n_timestamps == 0));
|
||
|
unsigned n = n_acked(&acki);
|
||
|
assert(("Number of acked packets is 0x1234", n == 0x1234));
|
||
|
|
||
|
{
|
||
|
size_t sz;
|
||
|
for (sz = 1; sz < sizeof(ack_buf); ++sz)
|
||
|
{
|
||
|
len = pf->pf_parse_ack_frame(ack_buf, sz, &acki);
|
||
|
assert(("Parsing truncated frame failed", len < 0));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
test2 (void)
|
||
|
{
|
||
|
/* Test taken from quic_framer_test.cc -- NewAckFrameOneAckBlock */
|
||
|
unsigned char ack_buf[] = {
|
||
|
/* TYPE N LL MM */
|
||
|
0xA0 | 1 << 4 | 1 << 2 | 1,
|
||
|
0x04, /* Num Blocks */
|
||
|
0x02, /* Num TS */
|
||
|
0x12, 0x34, /* Largest acked */
|
||
|
0x00, 0x00, /* Zero delta time. */
|
||
|
0x00, 0x01, /* First ack block length. */
|
||
|
0x01, /* Gap to next block. */
|
||
|
0x0e, 0xaf, /* Ack block length. */
|
||
|
0xff, /* Gap to next block. */
|
||
|
0x00, 0x00, /* Ack block length. */
|
||
|
0x91, /* Gap to next block. */
|
||
|
0x01, 0xea, /* Ack block length. */
|
||
|
0x05, /* Gap to next block. */
|
||
|
0x00, 0x04, /* Ack block length. */
|
||
|
0x01, /* Delta from largest observed. */
|
||
|
0x10, 0x32, 0x54, 0x76, /* Delta time. */ /* XXX do we use this at all? */
|
||
|
0x02, /* Delta from largest observed. */
|
||
|
0x10, 0x32, /* Delta time. */
|
||
|
};
|
||
|
|
||
|
/* We should get the following array of ranges:
|
||
|
* high low
|
||
|
* 0x1234 0x1234
|
||
|
* 0x1232 0x384
|
||
|
* 0x1F3 0xA
|
||
|
* 0x4 0x1
|
||
|
*/
|
||
|
static const struct { unsigned high, low; } ranges[] = {
|
||
|
{ 0x1234, 0x1234 },
|
||
|
{ 0x1232, 0x384 },
|
||
|
{ 0x1F3, 0xA },
|
||
|
{ 0x4, 0x1 },
|
||
|
};
|
||
|
|
||
|
ack_info_t acki;
|
||
|
memset(&acki, 0xF1, sizeof(acki));
|
||
|
|
||
|
int len = pf->pf_parse_ack_frame(ack_buf, sizeof(ack_buf), &acki);
|
||
|
assert(("Parsed length is correct (29)", len == sizeof(ack_buf)));
|
||
|
assert(("Number of ranges is 4", acki.n_ranges == 4));
|
||
|
assert(("Largest acked is 0x1234", acki.ranges[0].high == 0x1234));
|
||
|
assert(("Number of timestamps is 2", acki.n_timestamps == 2));
|
||
|
unsigned n = n_acked(&acki);
|
||
|
assert(("Number of acked packets is 4254", n == 4254));
|
||
|
|
||
|
for (n = 0; n < 4; ++n)
|
||
|
assert(("Range checks out", ranges[n].high == acki.ranges[n].high
|
||
|
&& ranges[n].low == acki.ranges[n].low));
|
||
|
|
||
|
{
|
||
|
size_t sz;
|
||
|
for (sz = 1; sz < sizeof(ack_buf); ++sz)
|
||
|
{
|
||
|
len = pf->pf_parse_ack_frame(ack_buf, sz, &acki);
|
||
|
assert(("Parsing truncated frame failed", len < 0));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
test3 (void)
|
||
|
{
|
||
|
/* Generated by our own code, but failed to parse... */
|
||
|
unsigned char ack_buf[] = {
|
||
|
/* TYPE N LL MM */
|
||
|
0xA0 | 1 << 4 | 0 << 2 | 0,
|
||
|
0x01, /* Num ACK block ranges */
|
||
|
0x00, /* Number of timestamps */
|
||
|
0x06, /* Largest ACKed */
|
||
|
0x00, 0x00, /* Delta time */
|
||
|
0x01, /* First ACK block length */
|
||
|
0x02, /* Gap to next block */
|
||
|
0x03, /* Ack block length */
|
||
|
};
|
||
|
|
||
|
/* We should get the following array of ranges:
|
||
|
* high low
|
||
|
* 6 6
|
||
|
* 3 1
|
||
|
*/
|
||
|
static const struct { unsigned high, low; } ranges[] = {
|
||
|
{ 6, 6, },
|
||
|
{ 3, 1, },
|
||
|
};
|
||
|
|
||
|
ack_info_t acki;
|
||
|
memset(&acki, 0xF1, sizeof(acki));
|
||
|
|
||
|
int len = pf->pf_parse_ack_frame(ack_buf, sizeof(ack_buf), &acki);
|
||
|
assert(("Parsed length is correct (9)", len == sizeof(ack_buf)));
|
||
|
assert(("Number of ranges is 2", acki.n_ranges == 2));
|
||
|
assert(("Largest acked is 6", acki.ranges[0].high == 6));
|
||
|
assert(("Number of timestamps is 0", acki.n_timestamps == 0));
|
||
|
unsigned n = n_acked(&acki);
|
||
|
assert(("Number of acked packets is 4", n == 4));
|
||
|
|
||
|
for (n = 0; n < 2; ++n)
|
||
|
assert(("Range checks out", ranges[n].high == acki.ranges[n].high
|
||
|
&& ranges[n].low == acki.ranges[n].low));
|
||
|
|
||
|
{
|
||
|
size_t sz;
|
||
|
for (sz = 1; sz < sizeof(ack_buf); ++sz)
|
||
|
{
|
||
|
len = pf->pf_parse_ack_frame(ack_buf, sz, &acki);
|
||
|
assert(("Parsing truncated frame failed", len < 0));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
test4 (void)
|
||
|
{
|
||
|
unsigned char ack_buf[] = {
|
||
|
/* TYPE N LL MM */
|
||
|
0xA0 | 1 << 4 | 0 << 2 | 0,
|
||
|
0x01, /* Num ACK block ranges */
|
||
|
0x00, /* Number of timestamps */
|
||
|
0x03, /* Largest ACKed */
|
||
|
0x23, 0x00, /* Delta time */
|
||
|
0x01, /* First ack block length */
|
||
|
0x01, /* Gap */
|
||
|
0x01, /* Ack block length */
|
||
|
};
|
||
|
|
||
|
/* We should get the following array of ranges:
|
||
|
* high low
|
||
|
* 6 6
|
||
|
* 3 1
|
||
|
*/
|
||
|
static const struct { unsigned high, low; } ranges[] = {
|
||
|
{ 3, 3, },
|
||
|
{ 1, 1, },
|
||
|
};
|
||
|
|
||
|
ack_info_t acki;
|
||
|
memset(&acki, 0xF1, sizeof(acki));
|
||
|
|
||
|
int len = pf->pf_parse_ack_frame(ack_buf, sizeof(ack_buf), &acki);
|
||
|
assert(("Parsed length is correct (9)", len == sizeof(ack_buf)));
|
||
|
assert(("Number of ranges is 2", acki.n_ranges == 2));
|
||
|
assert(("Largest acked is 3", acki.ranges[0].high == 3));
|
||
|
assert(("Number of timestamps is 0", acki.n_timestamps == 0));
|
||
|
unsigned n = n_acked(&acki);
|
||
|
assert(("Number of acked packets is 2", n == 2));
|
||
|
|
||
|
for (n = 0; n < 2; ++n)
|
||
|
assert(("Range checks out", ranges[n].high == acki.ranges[n].high
|
||
|
&& ranges[n].low == acki.ranges[n].low));
|
||
|
|
||
|
{
|
||
|
size_t sz;
|
||
|
for (sz = 1; sz < sizeof(ack_buf); ++sz)
|
||
|
{
|
||
|
len = pf->pf_parse_ack_frame(ack_buf, sz, &acki);
|
||
|
assert(("Parsing truncated frame failed", len < 0));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/* Four-byte packet numbers */
|
||
|
static void
|
||
|
test5 (void)
|
||
|
{
|
||
|
unsigned char ack_buf[] = {
|
||
|
/* TYPE N LL MM */
|
||
|
0xA0 | 1 << 4 | 2 << 2 | 2,
|
||
|
0x01, /* Num ack blocks ranges. */
|
||
|
0x00, /* Number of timestamps. */
|
||
|
0x23, 0x45, 0x67, 0x89,
|
||
|
0x00, 0x00, /* Zero delta time. */
|
||
|
0x00, 0x00, 0x00, 0x01, /* First ack block length. */
|
||
|
33 - 1, /* Gap to next block. */
|
||
|
0x23, 0x45, 0x67, 0x68, /* Ack block length. */
|
||
|
};
|
||
|
|
||
|
/* We should get the following array of ranges:
|
||
|
* high low
|
||
|
* 6 6
|
||
|
* 3 1
|
||
|
*/
|
||
|
static const struct { unsigned high, low; } ranges[] = {
|
||
|
{ 0x23456789, 0x23456789, },
|
||
|
{ 0x23456768, 1, },
|
||
|
};
|
||
|
|
||
|
ack_info_t acki;
|
||
|
memset(&acki, 0xF1, sizeof(acki));
|
||
|
|
||
|
int len = pf->pf_parse_ack_frame(ack_buf, sizeof(ack_buf), &acki);
|
||
|
assert(("Parsed length is correct (9)", len == sizeof(ack_buf)));
|
||
|
assert(("Number of ranges is 2", acki.n_ranges == 2));
|
||
|
assert(("Largest acked is 0x23456789", acki.ranges[0].high == 0x23456789));
|
||
|
assert(("Number of timestamps is 0", acki.n_timestamps == 0));
|
||
|
lsquic_packno_t n = n_acked(&acki);
|
||
|
assert(("Number of acked packets is correct", n == 0x23456768 + 1));
|
||
|
|
||
|
for (n = 0; n < 2; ++n)
|
||
|
assert(("Range checks out", ranges[n].high == acki.ranges[n].high
|
||
|
&& ranges[n].low == acki.ranges[n].low));
|
||
|
|
||
|
{
|
||
|
size_t sz;
|
||
|
for (sz = 1; sz < sizeof(ack_buf); ++sz)
|
||
|
{
|
||
|
len = pf->pf_parse_ack_frame(ack_buf, sz, &acki);
|
||
|
assert(("Parsing truncated frame failed", len < 0));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/* Six-byte packet numbers */
|
||
|
static void
|
||
|
test6 (void)
|
||
|
{
|
||
|
unsigned char ack_buf[] = {
|
||
|
/* TYPE N LL MM */
|
||
|
0xA0 | 1 << 4 | 3 << 2 | 3,
|
||
|
0x01, /* Num ack blocks ranges. */
|
||
|
0x00, /* Number of timestamps. */
|
||
|
0x00, 0x00, 0xAB, 0xCD, 0x23, 0x45, 0x67, 0x89,
|
||
|
0x00, 0x00, /* Zero delta time. */
|
||
|
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, /* First ack block length. */
|
||
|
33 - 1, /* Gap to next block. */
|
||
|
0x00, 0x00, 0xAB, 0xCD, 0x23, 0x45, 0x67, 0x68, /* Ack block length. */
|
||
|
};
|
||
|
|
||
|
static const struct { lsquic_packno_t high, low; } ranges[] = {
|
||
|
{ 0xABCD23456789, 0xABCD23456789, },
|
||
|
{ 0xABCD23456768, 1, },
|
||
|
};
|
||
|
|
||
|
ack_info_t acki;
|
||
|
memset(&acki, 0xF1, sizeof(acki));
|
||
|
|
||
|
int len = pf->pf_parse_ack_frame(ack_buf, sizeof(ack_buf), &acki);
|
||
|
assert(("Parsed length is correct", len == sizeof(ack_buf)));
|
||
|
assert(("Number of ranges is 2", acki.n_ranges == 2));
|
||
|
assert(("Largest acked is 0xABCD23456789", acki.ranges[0].high == 0xABCD23456789));
|
||
|
assert(("Number of timestamps is 0", acki.n_timestamps == 0));
|
||
|
lsquic_packno_t n = n_acked(&acki);
|
||
|
assert(("Number of acked packets is correct", n == 0xABCD23456768 + 1));
|
||
|
|
||
|
for (n = 0; n < 2; ++n)
|
||
|
assert(("Range checks out", ranges[n].high == acki.ranges[n].high
|
||
|
&& ranges[n].low == acki.ranges[n].low));
|
||
|
|
||
|
{
|
||
|
size_t sz;
|
||
|
for (sz = 1; sz < sizeof(ack_buf); ++sz)
|
||
|
{
|
||
|
len = pf->pf_parse_ack_frame(ack_buf, sz, &acki);
|
||
|
assert(("Parsing truncated frame failed", len < 0));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
test_max_ack (void)
|
||
|
{
|
||
|
lsquic_rechist_t rechist;
|
||
|
lsquic_time_t now;
|
||
|
unsigned i;
|
||
|
int has_missing, sz[2];
|
||
|
const struct lsquic_packno_range *range;
|
||
|
unsigned char buf[1500];
|
||
|
struct ack_info acki;
|
||
|
lsquic_cid_t cid = { .len = 0, };
|
||
|
|
||
|
lsquic_rechist_init(&rechist, &cid);
|
||
|
now = lsquic_time_now();
|
||
|
|
||
|
for (i = 1; i <= 300; ++i)
|
||
|
{
|
||
|
lsquic_rechist_received(&rechist, i * 10, now);
|
||
|
now += i * 1000;
|
||
|
}
|
||
|
|
||
|
memset(buf, 0xAA, sizeof(buf));
|
||
|
|
||
|
lsquic_packno_t largest = 0;
|
||
|
sz[0] = pf->pf_gen_ack_frame(buf, sizeof(buf),
|
||
|
(gaf_rechist_first_f) lsquic_rechist_first,
|
||
|
(gaf_rechist_next_f) lsquic_rechist_next,
|
||
|
(gaf_rechist_largest_recv_f) lsquic_rechist_largest_recv,
|
||
|
&rechist, now, &has_missing, &largest);
|
||
|
assert(sz[0] > 0);
|
||
|
assert(sz[0] <= (int) sizeof(buf));
|
||
|
assert(has_missing);
|
||
|
|
||
|
assert(0 == buf[ 2 ]); /* Number of timestamps */
|
||
|
assert(0xAA == buf[ sz[0] ]);
|
||
|
|
||
|
sz[1] = pf->pf_parse_ack_frame(buf, sizeof(buf), &acki);
|
||
|
assert(sz[1] == sz[0]);
|
||
|
assert(256 == acki.n_ranges);
|
||
|
|
||
|
for (range = lsquic_rechist_first(&rechist), i = 0;
|
||
|
range && i < acki.n_ranges;
|
||
|
range = lsquic_rechist_next(&rechist), ++i)
|
||
|
{
|
||
|
assert(range->high == acki.ranges[i].high);
|
||
|
assert(range->low == acki.ranges[i].low);
|
||
|
}
|
||
|
assert(i == 256);
|
||
|
|
||
|
lsquic_rechist_cleanup(&rechist);
|
||
|
}
|
||
|
|
||
|
|
||
|
static void
|
||
|
test_ack_truncation (void)
|
||
|
{
|
||
|
lsquic_rechist_t rechist;
|
||
|
lsquic_time_t now;
|
||
|
unsigned i;
|
||
|
int has_missing, sz[2];
|
||
|
const struct lsquic_packno_range *range;
|
||
|
unsigned char buf[1500];
|
||
|
struct ack_info acki;
|
||
|
size_t bufsz;
|
||
|
lsquic_cid_t cid = { .len = 0, };
|
||
|
|
||
|
lsquic_rechist_init(&rechist, &cid);
|
||
|
now = lsquic_time_now();
|
||
|
|
||
|
for (i = 1; i <= 300; ++i)
|
||
|
{
|
||
|
lsquic_rechist_received(&rechist, i * 10, now);
|
||
|
now += i * 1000;
|
||
|
}
|
||
|
|
||
|
for (bufsz = 200; bufsz < 210; ++bufsz)
|
||
|
{
|
||
|
memset(buf, 0xAA, sizeof(buf));
|
||
|
lsquic_packno_t largest = 0;
|
||
|
sz[0] = pf->pf_gen_ack_frame(buf, bufsz,
|
||
|
(gaf_rechist_first_f) lsquic_rechist_first,
|
||
|
(gaf_rechist_next_f) lsquic_rechist_next,
|
||
|
(gaf_rechist_largest_recv_f) lsquic_rechist_largest_recv,
|
||
|
&rechist, now, &has_missing, &largest);
|
||
|
assert(sz[0] > 0);
|
||
|
assert(sz[0] <= (int) bufsz);
|
||
|
assert(has_missing);
|
||
|
|
||
|
assert(0 == buf[ 2 ]); /* Number of timestamps */
|
||
|
assert(0xAA == buf[ sz[0] ]);
|
||
|
|
||
|
sz[1] = pf->pf_parse_ack_frame(buf, sizeof(buf), &acki);
|
||
|
assert(sz[1] == sz[0]);
|
||
|
assert(acki.n_ranges < 256);
|
||
|
|
||
|
for (range = lsquic_rechist_first(&rechist), i = 0;
|
||
|
range && i < acki.n_ranges;
|
||
|
range = lsquic_rechist_next(&rechist), ++i)
|
||
|
{
|
||
|
assert(range->high == acki.ranges[i].high);
|
||
|
assert(range->low == acki.ranges[i].low);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
lsquic_rechist_cleanup(&rechist);
|
||
|
}
|
||
|
|
||
|
|
||
|
int
|
||
|
main (void)
|
||
|
{
|
||
|
lsquic_global_init(LSQUIC_GLOBAL_SERVER);
|
||
|
test1();
|
||
|
test2();
|
||
|
test3();
|
||
|
test4();
|
||
|
test5();
|
||
|
test6();
|
||
|
test_max_ack();
|
||
|
test_ack_truncation();
|
||
|
return 0;
|
||
|
}
|