litespeed-quic/tests/test_h3_framing.c

1762 lines
54 KiB
C

/* Copyright (c) 2017 - 2022 LiteSpeed Technologies Inc. See LICENSE. */
/*
* test_h3_framing.c -- test generation of H3 frames
*/
#include <assert.h>
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/queue.h>
#include <sys/types.h>
#include <fcntl.h>
#include <limits.h>
#ifndef WIN32
#include <unistd.h>
#else
#include <getopt.h>
#endif
#include "lsquic.h"
#include "lsquic_packet_common.h"
#include "lsquic_packet_ietf.h"
#include "lsquic_alarmset.h"
#include "lsquic_packet_in.h"
#include "lsquic_conn_flow.h"
#include "lsquic_rtt.h"
#include "lsquic_sfcw.h"
#include "lsquic_varint.h"
#include "lsquic_hq.h"
#include "lsquic_hash.h"
#include "lsquic_stream.h"
#include "lsquic_types.h"
#include "lsquic_malo.h"
#include "lsquic_mm.h"
#include "lsquic_conn_public.h"
#include "lsquic_logger.h"
#include "lsquic_parse.h"
#include "lsquic_conn.h"
#include "lsquic_engine_public.h"
#include "lsquic_cubic.h"
#include "lsquic_pacer.h"
#include "lsquic_senhist.h"
#include "lsquic_bw_sampler.h"
#include "lsquic_minmax.h"
#include "lsquic_bbr.h"
#include "lsquic_adaptive_cc.h"
#include "lsquic_send_ctl.h"
#include "lsquic_ver_neg.h"
#include "lsquic_packet_out.h"
#include "lsquic_enc_sess.h"
#include "lsqpack.h"
#include "lsxpack_header.h"
#include "lsquic_frab_list.h"
#include "lsquic_qenc_hdl.h"
#include "lsquic_http1x_if.h"
#include "lsquic_qdec_hdl.h"
#include "lsquic_varint.h"
#include "lsquic_hq.h"
#include "lsquic_data_in_if.h"
#define MIN(a, b) ((a) < (b) ? (a) : (b))
#define MAX(a, b) ((a) > (b) ? (a) : (b))
static const struct parse_funcs *g_pf = select_pf_by_ver(LSQVER_I001);
struct test_ctl_settings
{
int tcs_schedule_stream_packets_immediately;
int tcs_have_delayed_packets;
unsigned tcs_can_send;
enum buf_packet_type
tcs_bp_type;
enum packno_bits
tcs_guess_packno_bits,
tcs_calc_packno_bits;
};
static struct test_ctl_settings g_ctl_settings;
static void
init_buf (void *buf, size_t sz);
/* Set values to default */
static void
init_test_ctl_settings (struct test_ctl_settings *settings)
{
settings->tcs_schedule_stream_packets_immediately = 1;
settings->tcs_have_delayed_packets = 0;
settings->tcs_can_send = UINT_MAX;
settings->tcs_bp_type = BPT_HIGHEST_PRIO;
settings->tcs_guess_packno_bits = PACKNO_BITS_1;
settings->tcs_calc_packno_bits = PACKNO_BITS_1;
}
enum packno_bits
lsquic_send_ctl_calc_packno_bits (struct lsquic_send_ctl *ctl)
{
return g_ctl_settings.tcs_calc_packno_bits;
}
int
lsquic_send_ctl_schedule_stream_packets_immediately (struct lsquic_send_ctl *ctl)
{
return g_ctl_settings.tcs_schedule_stream_packets_immediately;
}
int
lsquic_send_ctl_have_delayed_packets (const struct lsquic_send_ctl *ctl)
{
return g_ctl_settings.tcs_have_delayed_packets;
}
int
lsquic_send_ctl_can_send (struct lsquic_send_ctl *ctl)
{
return ctl->sc_n_scheduled < g_ctl_settings.tcs_can_send;
}
enum packno_bits
lsquic_send_ctl_guess_packno_bits (struct lsquic_send_ctl *ctl)
{
return g_ctl_settings.tcs_guess_packno_bits;
}
enum buf_packet_type
lsquic_send_ctl_determine_bpt (struct lsquic_send_ctl *ctl,
const struct lsquic_stream *stream)
{
return g_ctl_settings.tcs_bp_type;
}
/* This function is only here to avoid crash in the test: */
void
lsquic_engine_add_conn_to_tickable (struct lsquic_engine_public *enpub,
lsquic_conn_t *conn)
{
}
static unsigned n_closed;
static enum stream_ctor_flags stream_ctor_flags =
SCF_CALL_ON_NEW|SCF_DI_AUTOSWITCH;
struct test_ctx {
lsquic_stream_t *stream;
};
static lsquic_stream_ctx_t *
on_new_stream (void *stream_if_ctx, lsquic_stream_t *stream)
{
struct test_ctx *test_ctx = stream_if_ctx;
test_ctx->stream = stream;
return NULL;
}
static void
on_close (lsquic_stream_t *stream, lsquic_stream_ctx_t *st_h)
{
++n_closed;
}
const struct lsquic_stream_if stream_if = {
.on_new_stream = on_new_stream,
.on_close = on_close,
};
static size_t
read_from_scheduled_packets (lsquic_send_ctl_t *send_ctl, lsquic_stream_id_t stream_id,
unsigned char *const begin, size_t bufsz, uint64_t first_offset, int *p_fin,
int fullcheck)
{
const struct parse_funcs *const pf_local = send_ctl->sc_conn_pub->lconn->cn_pf;
unsigned char *p = begin;
unsigned char *const end = p + bufsz;
const struct frame_rec *frec;
struct packet_out_frec_iter pofi;
struct lsquic_packet_out *packet_out;
struct stream_frame frame;
enum quic_frame_type expected_type;
int len, fin = 0;
expected_type = QUIC_FRAME_STREAM;
TAILQ_FOREACH(packet_out, &send_ctl->sc_scheduled_packets, po_next)
for (frec = lsquic_pofi_first(&pofi, packet_out); frec;
frec = lsquic_pofi_next(&pofi))
{
if (fullcheck)
{
assert(frec->fe_frame_type == expected_type);
if (0 && packet_out->po_packno != 1)
{
/* First packet may contain two stream frames, do not
* check it.
*/
assert(!lsquic_pofi_next(&pofi));
if (TAILQ_NEXT(packet_out, po_next))
{
assert(packet_out->po_data_sz == packet_out->po_n_alloc);
assert(frec->fe_len == packet_out->po_data_sz);
}
}
}
if (frec->fe_frame_type == expected_type &&
frec->fe_stream->id == stream_id)
{
assert(!fin);
if (QUIC_FRAME_STREAM == expected_type)
len = pf_local->pf_parse_stream_frame(packet_out->po_data + frec->fe_off,
packet_out->po_data_sz - frec->fe_off, &frame);
else
len = pf_local->pf_parse_crypto_frame(packet_out->po_data + frec->fe_off,
packet_out->po_data_sz - frec->fe_off, &frame);
assert(len > 0);
if (QUIC_FRAME_STREAM == expected_type)
assert(frame.stream_id == frec->fe_stream->id);
else
assert(frame.stream_id == ~0ULL);
/* Otherwise not enough to copy to: */
assert(end - p >= frame.data_frame.df_size);
/* Checks offset ordering: */
assert(frame.data_frame.df_offset ==
first_offset + (uintptr_t) (p - begin));
if (frame.data_frame.df_fin)
{
assert(!fin);
fin = 1;
}
memcpy(p, packet_out->po_data + frec->fe_off + len -
frame.data_frame.df_size, frame.data_frame.df_size);
p += frame.data_frame.df_size;
}
}
if (p_fin)
*p_fin = fin;
return p + bufsz - end;
}
static struct test_ctx test_ctx;
struct test_objs {
struct lsquic_engine_public eng_pub;
struct lsquic_conn lconn;
struct lsquic_conn_public conn_pub;
struct lsquic_send_ctl send_ctl;
struct lsquic_alarmset alset;
void *stream_if_ctx;
struct ver_neg ver_neg;
const struct lsquic_stream_if *
stream_if;
unsigned initial_stream_window;
enum stream_ctor_flags ctor_flags;
struct qpack_enc_hdl qeh;
struct qpack_dec_hdl qdh;
struct lsquic_hset_if hsi_if;
};
static int s_ack_written;
static void
write_ack (struct lsquic_conn *conn, struct lsquic_packet_out *packet_out)
{
/* We don't need to generate full-blown ACK, as logic in
* lsquic_send_ctl_rollback() only looks at po_frame_types.
*/
packet_out->po_frame_types |= QUIC_FRAME_ACK;
s_ack_written = 1;
}
static int s_can_write_ack;
static int
can_write_ack (struct lsquic_conn *lconn)
{
return s_can_write_ack;
}
static struct network_path network_path;
static struct network_path *
get_network_path (struct lsquic_conn *lconn, const struct sockaddr *sa)
{
return &network_path;
}
static enum {
SNAPSHOT_STATE_NONE = 0,
SNAPSHOT_STATE_TAKEN = 1 << 0,
SNAPSHOT_STATE_ROLLED_BACK = 1 << 1,
} s_snapshot_state;
static void
ack_snapshot (struct lsquic_conn *lconn, struct ack_state *ack_state)
{
s_snapshot_state |= SNAPSHOT_STATE_TAKEN;
}
static void
ack_rollback (struct lsquic_conn *lconn, struct ack_state *ack_state)
{
s_snapshot_state |= SNAPSHOT_STATE_ROLLED_BACK;
}
static const struct conn_iface our_conn_if =
{
.ci_can_write_ack = can_write_ack,
.ci_write_ack = write_ack,
.ci_get_path = get_network_path,
.ci_ack_snapshot = ack_snapshot,
.ci_ack_rollback = ack_rollback,
};
#if LSQUIC_CONN_STATS
static struct conn_stats s_conn_stats;
#endif
static void
init_test_objs (struct test_objs *tobjs, unsigned initial_conn_window,
unsigned initial_stream_window, unsigned short packet_sz)
{
int s;
memset(tobjs, 0, sizeof(*tobjs));
LSCONN_INITIALIZE(&tobjs->lconn);
tobjs->lconn.cn_pf = g_pf;
tobjs->lconn.cn_version = LSQVER_I001;
tobjs->lconn.cn_esf_c = &lsquic_enc_session_common_ietf_v1;
network_path.np_pack_size = packet_sz;
tobjs->lconn.cn_if = &our_conn_if;
lsquic_mm_init(&tobjs->eng_pub.enp_mm);
TAILQ_INIT(&tobjs->conn_pub.sending_streams);
TAILQ_INIT(&tobjs->conn_pub.read_streams);
TAILQ_INIT(&tobjs->conn_pub.write_streams);
TAILQ_INIT(&tobjs->conn_pub.service_streams);
lsquic_cfcw_init(&tobjs->conn_pub.cfcw, &tobjs->conn_pub,
initial_conn_window);
lsquic_conn_cap_init(&tobjs->conn_pub.conn_cap, initial_conn_window);
lsquic_alarmset_init(&tobjs->alset, 0);
tobjs->conn_pub.mm = &tobjs->eng_pub.enp_mm;
tobjs->conn_pub.lconn = &tobjs->lconn;
tobjs->conn_pub.enpub = &tobjs->eng_pub;
tobjs->conn_pub.send_ctl = &tobjs->send_ctl;
tobjs->conn_pub.packet_out_malo =
lsquic_malo_create(sizeof(struct lsquic_packet_out));
tobjs->conn_pub.path = &network_path;
#if LSQUIC_CONN_STATS
tobjs->conn_pub.conn_stats = &s_conn_stats;
#endif
tobjs->initial_stream_window = initial_stream_window;
tobjs->eng_pub.enp_settings.es_cc_algo = 1; /* Cubic */
tobjs->eng_pub.enp_hsi_if = &tobjs->hsi_if;
lsquic_send_ctl_init(&tobjs->send_ctl, &tobjs->alset, &tobjs->eng_pub,
&tobjs->ver_neg, &tobjs->conn_pub, 0);
tobjs->send_ctl.sc_adaptive_cc.acc_cubic.cu_cwnd = ~0ull;
tobjs->send_ctl.sc_cong_ctl = &tobjs->send_ctl.sc_adaptive_cc.acc_cubic;
tobjs->stream_if = &stream_if;
tobjs->stream_if_ctx = &test_ctx;
tobjs->ctor_flags = stream_ctor_flags;
//if ((1 << tobjs->lconn.cn_version) & LSQUIC_IETF_VERSIONS)
{
lsquic_qeh_init(&tobjs->qeh, &tobjs->lconn);
s = lsquic_qeh_settings(&tobjs->qeh, 0, 0, 0, 0);
assert(0 == s);
tobjs->conn_pub.u.ietf.qeh = &tobjs->qeh;
lsquic_qdh_init(&tobjs->qdh, &tobjs->lconn, 0, &tobjs->eng_pub, 0, 0);
tobjs->conn_pub.u.ietf.qdh = &tobjs->qdh;
}
}
static void
deinit_test_objs (struct test_objs *tobjs)
{
assert(!lsquic_malo_first(tobjs->eng_pub.enp_mm.malo.stream_frame));
lsquic_send_ctl_cleanup(&tobjs->send_ctl);
lsquic_malo_destroy(tobjs->conn_pub.packet_out_malo);
lsquic_mm_cleanup(&tobjs->eng_pub.enp_mm);
//if ((1 << tobjs->lconn.cn_version) & LSQUIC_IETF_VERSIONS)
lsquic_qeh_cleanup(&tobjs->qeh);
}
static struct lsquic_stream *
new_stream (struct test_objs *tobjs, unsigned stream_id, uint64_t send_off)
{
return lsquic_stream_new(stream_id, &tobjs->conn_pub, tobjs->stream_if,
tobjs->stream_if_ctx, tobjs->initial_stream_window, send_off,
tobjs->ctor_flags);
}
struct packetization_test_stream_ctx
{
const unsigned char *buf;
unsigned len, off, write_size;
int flush_after_each_write;
};
static lsquic_stream_ctx_t *
packetization_on_new_stream (void *stream_if_ctx, lsquic_stream_t *stream)
{
lsquic_stream_wantwrite(stream, 1);
return stream_if_ctx;
}
static void
packetization_on_close (lsquic_stream_t *stream, lsquic_stream_ctx_t *st_h)
{
}
#define RANDOM_WRITE_SIZE ~0U
static unsigned
calc_n_to_write (unsigned write_size)
{
if (write_size == RANDOM_WRITE_SIZE)
return rand() % 1000 + 1;
else
return write_size;
}
static void
packetization_write_as_much_as_you_can (lsquic_stream_t *stream,
lsquic_stream_ctx_t *ctx)
{
struct packetization_test_stream_ctx *const pack_ctx = (void *) ctx;
unsigned n_to_write, n_sched;
ssize_t n_written;
size_t avail;
int s;
while (pack_ctx->off < pack_ctx->len)
{
n_to_write = calc_n_to_write(pack_ctx->write_size);
n_sched = lsquic_send_ctl_n_scheduled(stream->conn_pub->send_ctl);
if (n_to_write > pack_ctx->len - pack_ctx->off)
n_to_write = pack_ctx->len - pack_ctx->off;
n_written = lsquic_stream_write(stream, pack_ctx->buf + pack_ctx->off,
n_to_write);
if (n_written == 0)
{
if (n_to_write && SSHS_BEGIN == stream->sm_send_headers_state
&& lsquic_send_ctl_can_send(stream->conn_pub->send_ctl))
{
avail = lsquic_stream_write_avail(stream);
assert(avail == 0
|| lsquic_send_ctl_n_scheduled(
stream->conn_pub->send_ctl) > n_sched);
}
break;
}
pack_ctx->off += n_written;
if (pack_ctx->flush_after_each_write)
{
s = lsquic_stream_flush(stream);
assert(s == 0);
}
}
s = lsquic_stream_flush(stream);
assert(s == 0);
lsquic_stream_wantwrite(stream, 0);
}
static void
packetization_perform_one_write (lsquic_stream_t *stream,
lsquic_stream_ctx_t *ctx)
{
struct packetization_test_stream_ctx *const pack_ctx = (void *) ctx;
unsigned n_to_write, n_sched;
ssize_t n_written;
size_t avail;
int s;
n_to_write = calc_n_to_write(pack_ctx->write_size);
if (n_to_write > pack_ctx->len - pack_ctx->off)
n_to_write = pack_ctx->len - pack_ctx->off;
n_sched = lsquic_send_ctl_n_scheduled(stream->conn_pub->send_ctl);
n_written = lsquic_stream_write(stream, pack_ctx->buf + pack_ctx->off,
n_to_write);
assert(n_written >= 0);
if (n_written == 0 && SSHS_BEGIN == stream->sm_send_headers_state
&& n_to_write
&& lsquic_send_ctl_can_send(stream->conn_pub->send_ctl))
{
avail = lsquic_stream_write_avail(stream);
assert(avail == 0
|| lsquic_send_ctl_n_scheduled(
stream->conn_pub->send_ctl) > n_sched);
}
pack_ctx->off += n_written;
if (pack_ctx->flush_after_each_write)
{
s = lsquic_stream_flush(stream);
assert(s == 0);
}
if (n_written == 0)
lsquic_stream_wantwrite(stream, 0);
}
static const struct lsquic_stream_if packetization_inside_once_stream_if = {
.on_new_stream = packetization_on_new_stream,
.on_close = packetization_on_close,
.on_write = packetization_write_as_much_as_you_can,
};
static const struct lsquic_stream_if packetization_inside_many_stream_if = {
.on_new_stream = packetization_on_new_stream,
.on_close = packetization_on_close,
.on_write = packetization_perform_one_write,
};
#define XHDR(name_, value_) .buf = name_ value_, .name_offset = 0, .name_len = sizeof(name_) - 1, .val_offset = sizeof(name_) - 1, .val_len = sizeof(value_) - 1,
static void
test_hq_framing (int sched_immed, int dispatch_once, unsigned wsize,
int flush_after_each_write, size_t conn_limit,
unsigned n_packets, unsigned short packet_sz)
{
struct test_objs tobjs;
struct lsquic_stream *stream;
size_t nw;
int fin, s;
unsigned char *buf_in, *buf_out;
const size_t buf_in_sz = 0x40000, buf_out_sz = 0x500000;
/* We'll write headers first after which stream will switch to using
* data-framing writer. This is simply so that we don't have to
* expose more stream things only for testing.
*/
struct lsxpack_header header = { XHDR(":method", "GET") };
struct lsquic_http_headers headers = { 1, &header, };
buf_in = malloc(buf_in_sz);
buf_out = malloc(buf_out_sz);
assert(buf_in && buf_out);
struct packetization_test_stream_ctx packet_stream_ctx =
{
.buf = buf_in,
.off = 0,
.len = buf_in_sz,
.write_size = wsize,
.flush_after_each_write = flush_after_each_write,
};
init_buf(buf_in, buf_in_sz);
init_test_ctl_settings(&g_ctl_settings);
g_ctl_settings.tcs_schedule_stream_packets_immediately = sched_immed;
stream_ctor_flags |= SCF_IETF;
init_test_objs(&tobjs, conn_limit ? conn_limit : buf_out_sz, buf_out_sz, packet_sz);
tobjs.stream_if_ctx = &packet_stream_ctx;
tobjs.ctor_flags |= SCF_HTTP|SCF_IETF;
if (sched_immed)
{
g_ctl_settings.tcs_can_send = n_packets;
if (dispatch_once)
{
tobjs.stream_if = &packetization_inside_once_stream_if;
tobjs.ctor_flags |= SCF_DISP_RW_ONCE;
}
else
tobjs.stream_if = &packetization_inside_many_stream_if;
}
else
{
lsquic_send_ctl_set_max_bpq_count(n_packets);
g_ctl_settings.tcs_can_send = INT_MAX;
/* Need this for on_new_stream() callback not to mess with
* the context, otherwise this is not used.
*/
tobjs.stream_if = &packetization_inside_many_stream_if;
}
stream = new_stream(&tobjs, 0, buf_out_sz);
s = lsquic_stream_send_headers(stream, &headers, 0);
assert(0 == s);
if (sched_immed)
{
lsquic_stream_dispatch_write_events(stream);
lsquic_stream_flush(stream);
}
else
{
packetization_write_as_much_as_you_can(stream,
(void *) &packet_stream_ctx);
g_ctl_settings.tcs_schedule_stream_packets_immediately = 1;
lsquic_send_ctl_schedule_buffered(&tobjs.send_ctl, BPT_HIGHEST_PRIO);
g_ctl_settings.tcs_schedule_stream_packets_immediately = 0;
}
lsquic_send_ctl_set_max_bpq_count(10);
/* Verify written data: */
nw = read_from_scheduled_packets(&tobjs.send_ctl, 0, buf_out, buf_out_sz,
0, &fin, 1);
if (!conn_limit)
assert(nw > buf_in_sz);
{ /* Remove framing and verify contents */
const unsigned char *src;
unsigned char *dst;
uint64_t sz;
unsigned frame_type;
int s;
src = buf_out;
dst = buf_out;
while (src < buf_out + nw)
{
frame_type = *src++;
s = vint_read(src, buf_out + buf_out_sz, &sz);
assert(s > 0);
/* In some rare circumstances it is possible to produce zero-length
* DATA frames:
*
* assert(sz > 0);
*/
assert(sz < (1 << 14));
src += s;
if (src == buf_out + s + 1)
{
/* Ignore headers */
assert(frame_type == HQFT_HEADERS);
src += sz;
}
else
{
assert(frame_type == HQFT_DATA);
if (src + sz > buf_out + nw) /* Chopped DATA frame (last) */
sz = buf_out + nw - src;
memmove(dst, src, sz);
dst += sz;
src += sz;
}
}
if (!conn_limit)
assert(buf_in_sz == (uintptr_t) dst - (uintptr_t) buf_out);
assert(0 == memcmp(buf_in, buf_out, (uintptr_t) dst - (uintptr_t) buf_out));
}
lsquic_stream_destroy(stream);
deinit_test_objs(&tobjs);
free(buf_in);
free(buf_out);
stream_ctor_flags &= ~SCF_IETF;
}
static void
main_test_hq_framing (void)
{
const unsigned wsizes[] = { 1, 2, 3, 7, 10, 50, 100, 201, 211, 1000, 2003, 20000, };
const size_t conn_limits[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13,
14, 15, 16, 17, 18, 19, 20, 21, 30, 31, 32, 33, 63, 64, 128, 200, 255,
256, 512, 1024, 2045, 2048, 2049, 3000, 4091, 4096, 4097, 5000, 8192,
16 * 1024 - 1, 16 * 1024, 32 * 1024, 33 * 1024, };
const unsigned n_packets[] = { 0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 100, UINT_MAX, };
const unsigned short packet_sz[] = { 1252, 1370, 0x1000, 0xFF00, };
unsigned i, j, k, l;
int sched_immed, dispatch_once, flush_after_each_write;
for (sched_immed = 0; sched_immed <= 1; ++sched_immed)
for (dispatch_once = 0; dispatch_once <= 1; ++dispatch_once)
for (i = 0; i < sizeof(wsizes) / sizeof(wsizes[i]); ++i)
for (j = 0; j < sizeof(conn_limits) / sizeof(conn_limits[j]); ++j)
for (flush_after_each_write = 0; flush_after_each_write < 2; ++flush_after_each_write)
for (k = 0; k < sizeof(n_packets) / sizeof(n_packets[0]); ++k)
for (l = 0; l < sizeof(packet_sz) / sizeof(packet_sz[0]); ++l)
test_hq_framing(sched_immed, dispatch_once, wsizes[i], flush_after_each_write, conn_limits[j], n_packets[k], packet_sz[l]);
}
/* Instead of the not-very-random testing done in main_test_hq_framing(),
* the fuzz-guided testing initializes parameters based on the fuzz input
* file. This allows afl-fuzz explore the code paths.
*/
void
fuzz_guided_hq_framing_testing (const char *input)
{
/* Range */ /* Bytes from file */
unsigned short packet_sz; /* [200, 0x3FFF] */ /* 2 */
unsigned wsize; /* [1, 20000] */ /* 2 */
unsigned n_packets; /* [1, 255] and UINT_MAX */ /* 1 */
size_t conn_limit; /* [1, 33K] */ /* 2 */
int sched_immed; /* 0 or 1 */ /* 1 */
int dispatch_once; /* 0 or 1 */ /* 0 (same as above) */
int flush_after_each_write; /* 0 or 1 */ /* 0 (same as above) */
/* TOTAL: 8 bytes */
FILE *f;
size_t nread;
uint16_t tmp;
unsigned char buf[9];
f = fopen(input, "rb");
if (!f)
{
assert(0);
return;
}
nread = fread(buf, 1, sizeof(buf), f);
if (nread != 8)
goto cleanup;
memcpy(&tmp, &buf[0], 2);
if (tmp < 200)
tmp = 200;
else if (tmp > IQUIC_MAX_OUT_PACKET_SZ)
tmp = IQUIC_MAX_OUT_PACKET_SZ;
packet_sz = tmp;
memcpy(&tmp, &buf[2], 2);
if (tmp < 1)
tmp = 1;
else if (tmp > 20000)
tmp = 20000;
wsize = tmp;
if (buf[4])
n_packets = buf[4];
else
n_packets = UINT_MAX;
memcpy(&tmp, &buf[5], 2);
if (tmp < 1)
tmp = 1;
else if (tmp > 33 * 1024)
tmp = 33 * 1024;
conn_limit = tmp;
sched_immed = !!(buf[7] & 1);
dispatch_once = !!(buf[7] & 2);
flush_after_each_write = !!(buf[7] & 4);
test_hq_framing(sched_immed, dispatch_once, wsize,
flush_after_each_write, conn_limit, n_packets, packet_sz);
cleanup:
(void) fclose(f);
}
struct pwritev_stream_ctx
{
int limit; /* Test limit */
size_t avail;
const unsigned char *input;
size_t input_sz;
ssize_t nw; /* Number of bytes written */
};
static ssize_t
my_preadv (void *user_data, const struct iovec *iov, int iovcnt)
{
struct pwritev_stream_ctx *const pw_ctx = user_data;
const unsigned char *p;
size_t ntoread, tocopy;
int i;
ntoread = 0;
for (i = 0; i < iovcnt; ++i)
ntoread += iov[i].iov_len;
if (pw_ctx->limit < 0)
{
if ((size_t) -pw_ctx->limit < ntoread)
ntoread -= (size_t) -pw_ctx->limit;
}
else if ((size_t) pw_ctx->limit < ntoread)
ntoread = (size_t) pw_ctx->limit;
assert(ntoread <= pw_ctx->input_sz); /* Self-check */
p = pw_ctx->input;
for (i = 0; i < iovcnt; ++i)
{
tocopy = MIN(iov[i].iov_len, ntoread - (p - pw_ctx->input));
memcpy(iov[i].iov_base, p, tocopy);
p += tocopy;
if (ntoread == (size_t) (p - pw_ctx->input))
break;
}
assert(ntoread == (size_t) (p - pw_ctx->input));
return (ssize_t) (p - pw_ctx->input);
}
static void
pwritev_on_write (lsquic_stream_t *stream, lsquic_stream_ctx_t *ctx)
{
struct pwritev_stream_ctx *const pw_ctx = (void *) ctx;
ssize_t nw;
nw = lsquic_stream_pwritev(stream, my_preadv, pw_ctx, pw_ctx->input_sz);
pw_ctx->nw = nw;
lsquic_stream_wantwrite(stream, 0);
}
static const struct lsquic_stream_if pwritev_stream_if = {
.on_new_stream = packetization_on_new_stream,
.on_close = packetization_on_close,
.on_write = pwritev_on_write,
};
static void
test_pwritev (enum lsquic_version version, int http, int sched_immed,
int limit, unsigned short packet_sz, size_t prologue_sz,
unsigned n_packets)
{
struct test_objs tobjs;
struct lsquic_stream *stream;
size_t nw;
int fin, s;
unsigned char *buf_in, *buf_out;
/* Some values that are large enough: */
const size_t buf_in_sz = MAX(n_packets * packet_sz, 0x1000),
buf_out_sz = (float) buf_in_sz * 1.1;
const int ietf = (1 << version) & LSQUIC_IETF_VERSIONS;
const enum stream_ctor_flags ietf_flags = ietf ? SCF_IETF : 0;
s_snapshot_state = 0;
s_ack_written = 0;
/* We'll write headers first after which stream will switch to using
* data-framing writer. This is simply so that we don't have to
* expose more stream things only for testing.
*/
struct lsxpack_header header = { XHDR(":method", "GET") };
struct lsquic_http_headers headers = { 1, &header, };
buf_in = malloc(buf_in_sz);
buf_out = malloc(buf_out_sz);
assert(buf_in && buf_out);
struct pwritev_stream_ctx pwritev_stream_ctx =
{
.input = buf_in + prologue_sz,
.input_sz = buf_in_sz - prologue_sz,
.limit = limit,
};
init_buf(buf_in, buf_in_sz);
init_test_ctl_settings(&g_ctl_settings);
g_ctl_settings.tcs_schedule_stream_packets_immediately = sched_immed;
stream_ctor_flags |= ietf_flags;
init_test_objs(&tobjs, buf_out_sz, buf_out_sz, packet_sz);
tobjs.lconn.cn_version = version;
tobjs.lconn.cn_esf_c = select_esf_common_by_ver(version);
tobjs.stream_if_ctx = &pwritev_stream_ctx;
tobjs.ctor_flags |= (http ? SCF_HTTP : 0)|ietf_flags;
if (sched_immed)
{
g_ctl_settings.tcs_can_send = n_packets;
tobjs.stream_if = &pwritev_stream_if;
}
else
{
lsquic_send_ctl_set_max_bpq_count(n_packets);
g_ctl_settings.tcs_can_send = INT_MAX;
g_ctl_settings.tcs_bp_type = BPT_OTHER_PRIO;
/* Need this for on_new_stream() callback not to mess with
* the context, otherwise this is not used.
*/
tobjs.stream_if = &pwritev_stream_if;
}
stream = new_stream(&tobjs, 0, buf_out_sz);
if (http)
{
if (ietf)
{
s = lsquic_stream_send_headers(stream, &headers, 0);
assert(0 == s);
}
else
/* Here we fake it in order not to have to set up frame writer. */
stream->stream_flags |= STREAM_HEADERS_SENT;
}
if (prologue_sz)
{
ssize_t written = lsquic_stream_write(stream, buf_in, prologue_sz);
assert(written > 0 && (size_t) written == prologue_sz);
}
if (sched_immed)
{
lsquic_stream_dispatch_write_events(stream);
assert(!(s_snapshot_state & SNAPSHOT_STATE_TAKEN));
// lsquic_stream_flush(stream);
}
else
{
pwritev_on_write(stream, (void *) &pwritev_stream_ctx);
assert(s_snapshot_state & SNAPSHOT_STATE_TAKEN);
if (n_packets > 0
&& s_ack_written
&& tobjs.send_ctl.sc_buffered_packets[BPT_OTHER_PRIO].bpq_count == 0)
assert(s_snapshot_state & SNAPSHOT_STATE_ROLLED_BACK);
g_ctl_settings.tcs_schedule_stream_packets_immediately = 1;
lsquic_send_ctl_schedule_buffered(&tobjs.send_ctl, BPT_OTHER_PRIO);
g_ctl_settings.tcs_schedule_stream_packets_immediately = 0;
lsquic_send_ctl_set_max_bpq_count(10);
}
assert(pwritev_stream_ctx.nw >= 0);
/* Verify written data: */
nw = read_from_scheduled_packets(&tobjs.send_ctl, 0, buf_out, buf_out_sz,
0, &fin, 1);
assert(nw <= buf_in_sz);
if (ietf && http)
{ /* Remove framing and verify contents */
const unsigned char *src;
unsigned char *dst;
uint64_t sz;
unsigned frame_type;
int s;
src = buf_out;
dst = buf_out;
while (src < buf_out + nw)
{
frame_type = *src++;
s = vint_read(src, buf_out + buf_out_sz, &sz);
assert(s > 0);
/* In some rare circumstances it is possible to produce zero-length
* DATA frames:
*
* assert(sz > 0);
*/
assert(sz < (1 << 14));
src += s;
if (src == buf_out + s + 1)
{
/* Ignore headers */
assert(frame_type == HQFT_HEADERS);
src += sz;
}
else
{
assert(frame_type == HQFT_DATA);
if (src + sz > buf_out + nw) /* Chopped DATA frame (last) */
sz = buf_out + nw - src;
memmove(dst, src, sz);
dst += sz;
src += sz;
}
}
assert(nw <= buf_in_sz);
if (n_packets && pwritev_stream_ctx.nw)
{
assert((size_t) pwritev_stream_ctx.nw + prologue_sz == (uintptr_t) dst - (uintptr_t) buf_out);
assert(0 == memcmp(buf_in, buf_out, (uintptr_t) dst - (uintptr_t) buf_out));
}
else
assert((uintptr_t) dst - (uintptr_t) buf_out == 0
|| (uintptr_t) dst - (uintptr_t) buf_out == prologue_sz);
}
else
{
assert(nw <= buf_in_sz);
assert(nw <= buf_out_sz);
if (n_packets && pwritev_stream_ctx.nw)
{
assert((size_t) pwritev_stream_ctx.nw + prologue_sz == nw);
assert(0 == memcmp(buf_in, buf_out, (size_t) nw));
}
else
assert(nw == 0 || nw == prologue_sz);
}
lsquic_stream_destroy(stream);
deinit_test_objs(&tobjs);
free(buf_in);
free(buf_out);
stream_ctor_flags &= ~ietf_flags;
}
static void
main_test_pwritev (void)
{
const int limits[] = { INT_MAX, -1, -2, -3, -7, -10, -50, -100, -201, -211,
-1000, -2003, -3000, -4000, -17803, -20000, 16 * 1024, 16 * 1024 - 1,
16 * 1024 - 2, 8000, 273, 65, 63, 10, 5, 1, 0, };
unsigned n_packets;
const unsigned short packet_sz[] = { 1252, 1370, 0x1000, 0xFF00, };
const size_t prologues[] = { 0, 17, 238, };
unsigned i, j, k;
enum lsquic_version version;
int http, sched_immed;
const struct { unsigned iovecs, frames; } combos[] =
{
{ 32, 16, },
{ 16, 16, },
{ 16, 8, },
{ 3, 7, },
{ 7, 3, },
{ 100, 100, },
}, *combo = combos;
s_can_write_ack = 1;
run_test:
for (version = 0; version < N_LSQVER; ++version)
if ((1 << version) & LSQUIC_SUPPORTED_VERSIONS)
for (http = 0; http < 2; ++http)
for (sched_immed = 0; sched_immed <= 1; ++sched_immed)
for (i = 0; i < sizeof(limits) / sizeof(limits[i]); ++i)
for (j = 0; j < sizeof(packet_sz) / sizeof(packet_sz[0]);
++j)
for (k = 0; k < sizeof(prologues) / sizeof(prologues[0]); ++k)
for (n_packets = 1; n_packets < 21; ++n_packets)
test_pwritev(version, http, sched_immed,
limits[i], packet_sz[j], prologues[k], n_packets);
if (combo < combos + sizeof(combos) / sizeof(combos[0]))
{
lsquic_stream_set_pwritev_params(combo->iovecs, combo->frames);
++combo;
goto run_test;
}
s_can_write_ack = 0;
}
/* Instead of the not-very-random testing done in main_test_pwritev(),
* the fuzz-guided testing initializes parameters based on the fuzz input
* file. This allows afl-fuzz explore the code paths.
*/
void
fuzz_guided_pwritev_testing (const char *input)
{
/* Range */ /* Bytes from file */
unsigned short packet_sz; /* [1200, 0xFF00] */ /* 2 */
int limit; /* [INT_MIN, INT_MAX] */ /* 2 */
unsigned n_packets; /* [0, 255] */ /* 1 */
unsigned n_iovecs; /* [0, 255] */ /* 1 */
unsigned n_frames; /* [0, 255] */ /* 1 */
size_t prologue_sz; /* [0, 170] */ /* 1 */
enum lsquic_version version;/* [0,7] */ /* 1 */
int sched_immed; /* 0 or 1 */ /* 1 (same byte) */
int http; /* 0 or 1 */ /* 1 (same byte) */
/* TOTAL: 9 bytes */
FILE *f;
size_t nread;
union {
uint16_t tmp;
int16_t itmp;
} u;
unsigned char buf[10];
f = fopen(input, "rb");
if (!f)
{
assert(0);
return;
}
nread = fread(buf, 1, sizeof(buf), f);
if (nread != 9)
goto cleanup;
memcpy(&u.tmp, &buf[0], 2);
if (u.tmp < 1200)
u.tmp = 1200;
else if (u.tmp > 0xFF00)
u.tmp = 0xFF00;
packet_sz = u.tmp;
memcpy(&u.itmp, &buf[2], 2);
if (u.itmp < SHRT_MIN / 2)
limit = INT_MIN;
else if (u.itmp < SHRT_MIN / 4)
limit = 0;
else if (u.itmp > SHRT_MAX / 2)
limit = INT_MAX;
else if (u.itmp > SHRT_MAX / 2)
limit = 0;
else
limit = u.itmp;
n_packets = buf[4];
n_iovecs = buf[5];
n_frames = buf[6];
prologue_sz = buf[7];
if (prologue_sz > 170)
prologue_sz = 170;
switch (buf[8] & 7)
{
case 0: version = LSQVER_043; break;
case 1: version = LSQVER_046; break;
case 2: version = LSQVER_050; break;
case 3: version = LSQVER_ID27; break;
case 4: version = LSQVER_ID29; break;
default:
case 5: version = LSQVER_I001; break;
}
sched_immed = !!(buf[8] & 0x08);
http = !!(buf[8] & 0x10);
lsquic_stream_set_pwritev_params(n_iovecs, n_frames);
test_pwritev(version, http, sched_immed, limit, packet_sz, prologue_sz,
n_packets);
cleanup:
(void) fclose(f);
}
static unsigned
count_hq_frames (const struct lsquic_stream *stream)
{
const struct stream_hq_frame *shf;
unsigned n_frames = 0;
STAILQ_FOREACH(shf, &stream->sm_hq_frames, shf_next)
++n_frames;
return n_frames;
}
static void
test_frame_header_split (unsigned n_packets, unsigned extra_sz,
int add_one_more)
{
struct test_objs tobjs;
struct lsquic_stream *stream;
size_t nw;
int fin, s;
unsigned char *buf_in, *buf_out;
const unsigned wsize = 70;
const size_t buf_in_sz = wsize, buf_out_sz = 0x500000;
unsigned n_frames;
struct lsxpack_header header = { XHDR(":method", "GET") };
struct lsquic_http_headers headers = { 1, &header, };
buf_in = malloc(buf_in_sz);
buf_out = malloc(buf_out_sz);
assert(buf_in && buf_out);
struct packetization_test_stream_ctx packet_stream_ctx =
{
.buf = buf_in,
.off = 0,
.len = buf_in_sz,
.write_size = wsize,
.flush_after_each_write = 0,
};
init_buf(buf_in, buf_in_sz);
init_test_ctl_settings(&g_ctl_settings);
g_ctl_settings.tcs_schedule_stream_packets_immediately = 1;
stream_ctor_flags |= SCF_IETF;
init_test_objs(&tobjs, 0x1000, buf_out_sz, 1252);
tobjs.stream_if_ctx = &packet_stream_ctx;
tobjs.ctor_flags |= SCF_HTTP|SCF_IETF;
g_ctl_settings.tcs_can_send = n_packets;
tobjs.stream_if = &packetization_inside_once_stream_if;
tobjs.ctor_flags |= SCF_DISP_RW_ONCE;
struct lsquic_packet_out *const packet_out
= lsquic_send_ctl_new_packet_out(&tobjs.send_ctl, 0, PNS_APP, &network_path);
assert(packet_out);
const size_t pad_size = packet_out->po_n_alloc
- 2 /* STREAM header */
- 5 /* 3-byte HEADERS frame */
- extra_sz;
packet_out->po_data_sz = pad_size;
lsquic_send_ctl_scheduled_one(&tobjs.send_ctl, packet_out);
stream = new_stream(&tobjs, 0, buf_out_sz);
s = lsquic_stream_send_headers(stream, &headers, 0);
assert(0 == s);
const ssize_t w = lsquic_stream_write(stream, buf_in, buf_in_sz);
assert(w >= 0 && (size_t) w == buf_in_sz);
n_frames = count_hq_frames(stream);
assert((int) n_frames == 1 + (w > 0));
lsquic_stream_flush(stream);
n_frames = count_hq_frames(stream);
assert((int) n_frames == !!stream->sm_n_buffered);
if (add_one_more)
{
++g_ctl_settings.tcs_can_send;
lsquic_stream_flush(stream);
}
/* Verify written data: */
nw = read_from_scheduled_packets(&tobjs.send_ctl, 0, buf_out, buf_out_sz,
0, &fin, 1);
{ /* Remove framing and verify contents */
const unsigned char *src;
unsigned char *dst;
uint64_t sz;
unsigned frame_type;
int s;
src = buf_out;
dst = buf_out;
while (src < buf_out + nw)
{
frame_type = *src++;
s = vint_read(src, buf_out + buf_out_sz, &sz);
assert(s > 0);
assert(sz > 0);
assert(sz < (1 << 14));
src += s;
if (src == buf_out + s + 1)
{
/* Ignore headers */
assert(frame_type == HQFT_HEADERS);
src += sz;
}
else
{
assert(frame_type == HQFT_DATA);
if (src + sz > buf_out + nw) /* Chopped DATA frame (last) */
sz = buf_out + nw - src;
memmove(dst, src, sz);
dst += sz;
src += sz;
}
}
assert(0 == memcmp(buf_in, buf_out, (uintptr_t) dst - (uintptr_t) buf_out));
}
lsquic_stream_destroy(stream);
deinit_test_objs(&tobjs);
free(buf_in);
free(buf_out);
stream_ctor_flags &= ~SCF_IETF;
}
static void
test_zero_size_frame (void)
{
struct test_objs tobjs;
struct lsquic_stream *stream;
ssize_t w;
size_t nw;
int fin, s;
unsigned char *buf_in, *buf_out;
const unsigned wsize = 7000;
const size_t buf_in_sz = wsize, buf_out_sz = 0x500000;
struct lsxpack_header header = { XHDR(":method", "GET") };
struct lsquic_http_headers headers = { 1, &header, };
buf_in = malloc(buf_in_sz);
buf_out = malloc(buf_out_sz);
assert(buf_in && buf_out);
struct packetization_test_stream_ctx packet_stream_ctx =
{
.buf = buf_in,
.off = 0,
.len = buf_in_sz,
.write_size = wsize,
.flush_after_each_write = 0,
};
init_buf(buf_in, buf_in_sz);
init_test_ctl_settings(&g_ctl_settings);
g_ctl_settings.tcs_schedule_stream_packets_immediately = 1;
stream_ctor_flags |= SCF_IETF;
init_test_objs(&tobjs, 0x1000, buf_out_sz, 1252);
tobjs.stream_if_ctx = &packet_stream_ctx;
tobjs.ctor_flags |= SCF_HTTP|SCF_IETF;
g_ctl_settings.tcs_can_send = 1;
tobjs.stream_if = &packetization_inside_once_stream_if;
tobjs.ctor_flags |= SCF_DISP_RW_ONCE;
struct lsquic_packet_out *const packet_out
= lsquic_send_ctl_new_packet_out(&tobjs.send_ctl, 0, PNS_APP, &network_path);
assert(packet_out);
const size_t pad_size = packet_out->po_n_alloc
- 2 /* STREAM header */
- 5 /* 3-byte HEADERS frame */
- 3;
packet_out->po_data_sz = pad_size;
lsquic_send_ctl_scheduled_one(&tobjs.send_ctl, packet_out);
stream = new_stream(&tobjs, 0, buf_out_sz);
s = lsquic_stream_send_headers(stream, &headers, 0);
assert(0 == s);
w = lsquic_stream_write(stream, buf_in, buf_in_sz);
assert(w >= 0);
lsquic_stream_flush(stream);
g_ctl_settings.tcs_can_send++;
w = lsquic_stream_write(stream, buf_in, buf_in_sz);
assert(w >= 0);
lsquic_stream_flush(stream);
/* Verify written data: */
nw = read_from_scheduled_packets(&tobjs.send_ctl, 0, buf_out, buf_out_sz,
0, &fin, 1);
{ /* Remove framing and verify contents */
const unsigned char *src;
unsigned char *dst;
uint64_t sz;
unsigned frame_type;
int s;
src = buf_out;
dst = buf_out;
while (src < buf_out + nw)
{
frame_type = *src++;
s = vint_read(src, buf_out + buf_out_sz, &sz);
assert(s > 0);
assert(sz < (1 << 14));
src += s;
if (src == buf_out + s + 1)
{
/* Ignore headers */
assert(frame_type == HQFT_HEADERS);
src += sz;
}
else
{
assert(frame_type == HQFT_DATA);
if (src + sz > buf_out + nw) /* Chopped DATA frame (last) */
sz = buf_out + nw - src;
memmove(dst, src, sz);
dst += sz;
src += sz;
}
}
assert(0 == memcmp(buf_in, buf_out, (uintptr_t) dst - (uintptr_t) buf_out));
}
lsquic_stream_destroy(stream);
deinit_test_objs(&tobjs);
free(buf_in);
free(buf_out);
stream_ctor_flags &= ~SCF_IETF;
}
/* Create a new stream frame. Each stream frame has a real packet_in to
* back it up, just like in real code. The contents of the packet do
* not matter.
*/
static stream_frame_t *
new_frame_in_ext (struct test_objs *tobjs, size_t off, size_t sz, int fin,
const void *data)
{
lsquic_packet_in_t *packet_in;
stream_frame_t *frame;
assert(sz <= 1370);
packet_in = lsquic_mm_get_packet_in(&tobjs->eng_pub.enp_mm);
if (data)
packet_in->pi_data = (void *) data;
else
{
packet_in->pi_data = lsquic_mm_get_packet_in_buf(&tobjs->eng_pub.enp_mm, 1370);
packet_in->pi_flags |= PI_OWN_DATA;
memset(packet_in->pi_data, 'A', sz);
}
/* This is not how stream frame looks in the packet: we have no
* header. In our test case it does not matter, as we only care
* about stream frame.
*/
packet_in->pi_data_sz = sz;
packet_in->pi_refcnt = 1;
frame = lsquic_malo_get(tobjs->eng_pub.enp_mm.malo.stream_frame);
memset(frame, 0, sizeof(*frame));
frame->packet_in = packet_in;
frame->data_frame.df_offset = off;
frame->data_frame.df_size = sz;
frame->data_frame.df_data = &packet_in->pi_data[0];
frame->data_frame.df_fin = fin;
return frame;
}
/* Receiving DATA frame with zero payload should result in lsquic_stream_read()
* returning -1.
*/
static void
test_reading_zero_size_data_frame (void)
{
struct test_objs tobjs;
struct lsquic_stream *stream;
struct stream_frame *frame;
ssize_t nr;
int s;
unsigned char buf[2];
init_test_ctl_settings(&g_ctl_settings);
stream_ctor_flags |= SCF_IETF;
init_test_objs(&tobjs, 0x1000, 0x2000, 1252);
tobjs.ctor_flags |= SCF_HTTP|SCF_IETF;
stream = new_stream(&tobjs, 0, 0x1000);
/* Fake out reading of HEADERS frame: */
stream->stream_flags |= STREAM_HAVE_UH;
stream->sm_hq_filter.hqfi_flags |= HQFI_FLAG_HEADER;
/* One-byte DATA frame */
frame = new_frame_in_ext(&tobjs, 0, 3, 0, (uint8_t[3]){ 0, 1, 'a', });
s = lsquic_stream_frame_in(stream, frame);
assert(s == 0); /* Self-check */
/* Zero-length DATA frame */
frame = new_frame_in_ext(&tobjs, 3, 2, 0, (uint8_t[2]){ 0, 0, });
s = lsquic_stream_frame_in(stream, frame);
assert(s == 0); /* Self-check */
assert(stream->read_offset == 2); /* Self-check */
/* Read 'a' */
nr = lsquic_stream_read(stream, buf, 1);
assert(nr == 1);
assert(buf[0] == 'a');
/* Check that read returns -1 */
nr = lsquic_stream_read(stream, buf, sizeof(buf));
assert(nr == -1);
/* DATA frame was consumed: */
assert(stream->read_offset == 5);
lsquic_stream_destroy(stream);
deinit_test_objs(&tobjs);
stream_ctor_flags &= ~SCF_IETF;
}
/* Receiving DATA frame with zero payload should result in lsquic_stream_read()
* returning -1.
*/
static void
test_reading_zero_size_data_frame_scenario2 (void)
{
struct test_objs tobjs;
struct lsquic_stream *stream;
struct stream_frame *frame;
ssize_t nr;
int s;
unsigned char buf[2];
init_test_ctl_settings(&g_ctl_settings);
stream_ctor_flags |= SCF_IETF;
init_test_objs(&tobjs, 0x1000, 0x2000, 1252);
tobjs.ctor_flags |= SCF_HTTP|SCF_IETF;
stream = new_stream(&tobjs, 0, 0x1000);
/* Fake out reading of HEADERS frame: */
stream->stream_flags |= STREAM_HAVE_UH;
stream->sm_hq_filter.hqfi_flags |= HQFI_FLAG_HEADER;
/* Zero-length DATA frame */
frame = new_frame_in_ext(&tobjs, 0, 5, 0, (uint8_t[5]){ 0, 1, 'a', 0, 0, });
s = lsquic_stream_frame_in(stream, frame);
assert(s == 0); /* Self-check */
assert(stream->read_offset == 2); /* Self-check */
/* Read 'a' */
nr = lsquic_stream_read(stream, buf, 1);
assert(nr == 1);
assert(buf[0] == 'a');
/* Check that read returns -1 */
nr = lsquic_stream_read(stream, buf, sizeof(buf));
assert(nr == -1);
/* DATA frame was consumed: */
assert(stream->read_offset == 5);
lsquic_stream_destroy(stream);
deinit_test_objs(&tobjs);
stream_ctor_flags &= ~SCF_IETF;
}
/* Receiving DATA frame with zero payload should result in lsquic_stream_read()
* returning -1.
*/
static void
test_reading_zero_size_data_frame_scenario3 (void)
{
struct test_objs tobjs;
struct lsquic_stream *stream;
struct stream_frame *frame;
ssize_t nr;
int s;
unsigned char buf[2];
init_test_ctl_settings(&g_ctl_settings);
stream_ctor_flags |= SCF_IETF;
init_test_objs(&tobjs, 0x1000, 0x2000, 1252);
tobjs.ctor_flags |= SCF_HTTP|SCF_IETF;
stream = new_stream(&tobjs, 0, 0x1000);
/* Fake out reading of HEADERS frame: */
stream->stream_flags |= STREAM_HAVE_UH;
stream->sm_hq_filter.hqfi_flags |= HQFI_FLAG_HEADER;
/* Zero-length DATA frame */
frame = new_frame_in_ext(&tobjs, 0, 4, 0, (uint8_t[4]){ 0, 1, 'a', 0, });
s = lsquic_stream_frame_in(stream, frame);
assert(s == 0); /* Self-check */
assert(stream->read_offset == 2); /* Self-check */
/* Read 'a' */
nr = lsquic_stream_read(stream, buf, 1);
assert(nr == 1);
assert(buf[0] == 'a');
/* Check that read returns -1 */
nr = lsquic_stream_read(stream, buf, sizeof(buf));
assert(nr == -1);
/* Zero-length DATA frame */
frame = new_frame_in_ext(&tobjs, 4, 1, 0, (uint8_t[1]){ 0, });
s = lsquic_stream_frame_in(stream, frame);
assert(s == 0); /* Self-check */
/* Check that read returns -1 */
nr = lsquic_stream_read(stream, buf, sizeof(buf));
assert(nr == -1);
/* DATA frame was consumed: */
assert(stream->read_offset == 5);
lsquic_stream_destroy(stream);
deinit_test_objs(&tobjs);
stream_ctor_flags &= ~SCF_IETF;
}
int
main (int argc, char **argv)
{
const char *fuzz_hq_framing_input = NULL;
const char *fuzz_pwritev_input = NULL;
int opt, add_one_more;
unsigned n_packets, extra_sz;
lsquic_global_init(LSQUIC_GLOBAL_SERVER);
while (-1 != (opt = getopt(argc, argv, "f:p:l:")))
{
switch (opt)
{
case 'f':
fuzz_hq_framing_input = optarg;
break;
case 'p':
fuzz_pwritev_input = optarg;
break;
case 'l':
lsquic_log_to_fstream(stderr, 0);
lsquic_logger_lopt(optarg);
break;
default:
exit(1);
}
}
init_test_ctl_settings(&g_ctl_settings);
if (fuzz_hq_framing_input)
fuzz_guided_hq_framing_testing(fuzz_hq_framing_input);
else if (fuzz_pwritev_input)
fuzz_guided_pwritev_testing(fuzz_pwritev_input);
else
{
main_test_pwritev();
main_test_hq_framing();
for (n_packets = 1; n_packets <= 2; ++n_packets)
for (extra_sz = 0; extra_sz <= 2; ++extra_sz)
for (add_one_more = 0; add_one_more <= 1; ++add_one_more)
test_frame_header_split(n_packets, extra_sz, add_one_more);
test_zero_size_frame();
test_reading_zero_size_data_frame();
test_reading_zero_size_data_frame_scenario2();
test_reading_zero_size_data_frame_scenario3();
}
return 0;
}
static const char on_being_idle[] =
"ON BEING IDLE."
""
"Now, this is a subject on which I flatter myself I really am _au fait_."
"The gentleman who, when I was young, bathed me at wisdom's font for nine"
"guineas a term--no extras--used to say he never knew a boy who could"
"do less work in more time; and I remember my poor grandmother once"
"incidentally observing, in the course of an instruction upon the use"
"of the Prayer-book, that it was highly improbable that I should ever do"
"much that I ought not to do, but that she felt convinced beyond a doubt"
"that I should leave undone pretty well everything that I ought to do."
""
"I am afraid I have somewhat belied half the dear old lady's prophecy."
"Heaven help me! I have done a good many things that I ought not to have"
"done, in spite of my laziness. But I have fully confirmed the accuracy"
"of her judgment so far as neglecting much that I ought not to have"
"neglected is concerned. Idling always has been my strong point. I take"
"no credit to myself in the matter--it is a gift. Few possess it. There"
"are plenty of lazy people and plenty of slow-coaches, but a genuine"
"idler is a rarity. He is not a man who slouches about with his hands in"
"his pockets. On the contrary, his most startling characteristic is that"
"he is always intensely busy."
""
"It is impossible to enjoy idling thoroughly unless one has plenty of"
"work to do. There is no fun in doing nothing when you have nothing to"
"do. Wasting time is merely an occupation then, and a most exhausting"
"one. Idleness, like kisses, to be sweet must be stolen."
""
"Many years ago, when I was a young man, I was taken very ill--I never"
"could see myself that much was the matter with me, except that I had"
"a beastly cold. But I suppose it was something very serious, for the"
"doctor said that I ought to have come to him a month before, and that"
"if it (whatever it was) had gone on for another week he would not have"
"answered for the consequences. It is an extraordinary thing, but I"
"never knew a doctor called into any case yet but what it transpired"
"that another day's delay would have rendered cure hopeless. Our medical"
"guide, philosopher, and friend is like the hero in a melodrama--he"
"always comes upon the scene just, and only just, in the nick of time. It"
"is Providence, that is what it is."
""
"Well, as I was saying, I was very ill and was ordered to Buxton for a"
"month, with strict injunctions to do nothing whatever all the while"
"that I was there. \"Rest is what you require,\" said the doctor, \"perfect"
"rest.\""
""
"It seemed a delightful prospect. \"This man evidently understands my"
"complaint,\" said I, and I pictured to myself a glorious time--a four"
"weeks' _dolce far niente_ with a dash of illness in it. Not too much"
"illness, but just illness enough--just sufficient to give it the flavor"
"of suffering and make it poetical. I should get up late, sip chocolate,"
"and have my breakfast in slippers and a dressing-gown. I should lie out"
"in the garden in a hammock and read sentimental novels with a melancholy"
"ending, until the books should fall from my listless hand, and I should"
"recline there, dreamily gazing into the deep blue of the firmament,"
"watching the fleecy clouds floating like white-sailed ships across"
"its depths, and listening to the joyous song of the birds and the low"
"rustling of the trees. Or, on becoming too weak to go out of doors,"
"I should sit propped up with pillows at the open window of the"
"ground-floor front, and look wasted and interesting, so that all the"
"pretty girls would sigh as they passed by."
""
"And twice a day I should go down in a Bath chair to the Colonnade to"
"drink the waters. Oh, those waters! I knew nothing about them then,"
"and was rather taken with the idea. \"Drinking the waters\" sounded"
"fashionable and Queen Anne-fied, and I thought I should like them. But,"
"ugh! after the first three or four mornings! Sam Weller's description of"
"them as \"having a taste of warm flat-irons\" conveys only a faint idea of"
"their hideous nauseousness. If anything could make a sick man get well"
"quickly, it would be the knowledge that he must drink a glassful of them"
"every day until he was recovered. I drank them neat for six consecutive"
"days, and they nearly killed me; but after then I adopted the plan of"
"taking a stiff glass of brandy-and-water immediately on the top of them,"
"and found much relief thereby. I have been informed since, by various"
"eminent medical gentlemen, that the alcohol must have entirely"
"counteracted the effects of the chalybeate properties contained in the"
"water. I am glad I was lucky enough to hit upon the right thing."
;
static void
init_buf (void *buf, size_t sz)
{
unsigned char *p = buf;
unsigned char *const end = (unsigned char*)buf + sz;
size_t n;
while (p < end)
{
n = end - p;
if (sizeof(on_being_idle) - 1 < n)
n = sizeof(on_being_idle) - 1;
memcpy(p, on_being_idle, n);
p +=n;
}
assert(p == end);
}