/* Copyright (c) 2017 - 2019 LiteSpeed Technologies Inc. See LICENSE. */ /* * test_send_headers.c -- Test what happens when lsquic_stream_send_headers() * is called. */ #include #include #include #include #include #include #include #include #include #ifndef WIN32 #include #else #include #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_send_ctl.h" #include "lsquic_ver_neg.h" #include "lsquic_packet_out.h" #include "lsquic_enc_sess.h" #include "lsqpack.h" #include "lsquic_frab_list.h" #include "lsquic_qenc_hdl.h" #include "lsquic_varint.h" #include "lsquic_hq.h" #include "lsquic_data_in_if.h" #include "lsquic_headers.h" #include "lsquic_push_promise.h" static int s_call_wantwrite_in_ctor; static int s_wantwrite_arg; static int s_onwrite_called; static lsquic_stream_ctx_t * on_new_stream (void *stream_if_ctx, lsquic_stream_t *stream) { if (s_call_wantwrite_in_ctor) lsquic_stream_wantwrite(stream, s_wantwrite_arg); return NULL; } static void on_close (lsquic_stream_t *stream, lsquic_stream_ctx_t *st_h) { } static void on_write (lsquic_stream_t *stream, lsquic_stream_ctx_t *h) { s_onwrite_called = 1; lsquic_stream_wantwrite(stream, 0); } const struct lsquic_stream_if stream_if = { .on_new_stream = on_new_stream, .on_write = on_write, .on_close = on_close, }; enum buf_packet_type lsquic_send_ctl_determine_bpt (struct lsquic_send_ctl *ctl, const struct lsquic_stream *stream) { return BPT_HIGHEST_PRIO; } /* 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) { } 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; }; static int unit_test_doesnt_write_ack (struct lsquic_conn *lconn) { return 0; } 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 const struct conn_iface our_conn_if = { .ci_can_write_ack = unit_test_doesnt_write_ack, .ci_get_path = get_network_path, }; static void init_test_objs (struct test_objs *tobjs, unsigned initial_conn_window, unsigned initial_stream_window, enum stream_ctor_flags addl_ctor_flags) { int s; memset(tobjs, 0, sizeof(*tobjs)); LSCONN_INITIALIZE(&tobjs->lconn); tobjs->lconn.cn_pf = select_pf_by_ver(LSQVER_ID23); tobjs->lconn.cn_version = LSQVER_ID23; tobjs->lconn.cn_esf_c = &lsquic_enc_session_common_ietf_v1; network_path.np_pack_size = IQUIC_MAX_IPv4_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; tobjs->initial_stream_window = initial_stream_window; lsquic_send_ctl_init(&tobjs->send_ctl, &tobjs->alset, &tobjs->eng_pub, &tobjs->ver_neg, &tobjs->conn_pub, 0); tobjs->stream_if = &stream_if; tobjs->stream_if_ctx = NULL; tobjs->ctor_flags = SCF_CALL_ON_NEW|SCF_DI_AUTOSWITCH|SCF_HTTP |addl_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; } } 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); } static struct test_vals { /* What lsquic_qeh_write_headers() returns or sets */ enum qwh_status status; size_t prefix_sz; size_t headers_sz; uint64_t completion_offset; } test_vals; enum qwh_status lsquic_qeh_write_headers (struct qpack_enc_hdl *qeh, lsquic_stream_id_t stream_id, unsigned seqno, const struct lsquic_http_headers *headers, unsigned char *buf, size_t *prefix_sz, size_t *headers_sz, uint64_t *completion_offset, enum lsqpack_enc_header_flags *hflags) { memset(buf - *prefix_sz, 0xC5, *prefix_sz + *headers_sz); *prefix_sz = test_vals.prefix_sz; *headers_sz = test_vals.headers_sz; *completion_offset = test_vals.completion_offset; if (hflags) *hflags = 0; return test_vals.status; } static uint64_t s_enc_off; uint64_t lsquic_qeh_enc_off (struct qpack_enc_hdl *qeh) { return s_enc_off; } static void test_flushes_and_closes (void) { struct test_objs tobjs; struct lsquic_stream *stream; ssize_t nw; int s; struct uncompressed_headers *uh; void *hset; /* For our tests purposes, we treat headers as an opaque object */ struct lsquic_http_headers *headers = (void *) 1; init_test_objs(&tobjs, 0x1000, 0x1000, SCF_IETF); stream = new_stream(&tobjs, 0, 0x1000); test_vals.status = QWH_FULL; test_vals.prefix_sz = 2; test_vals.headers_sz = 40; test_vals.completion_offset = 0; s = lsquic_stream_send_headers(stream, headers, 0); assert(0 == s); assert(stream->sm_n_buffered == test_vals.prefix_sz + test_vals.headers_sz); assert(0 == stream->sm_hblock_sz); lsquic_stream_destroy(stream); stream = new_stream(&tobjs, 4, 0x1000); test_vals.status = QWH_PARTIAL; test_vals.prefix_sz = 2; test_vals.headers_sz = 40; test_vals.completion_offset = 10; s = lsquic_stream_send_headers(stream, headers, 0); assert(0 == s); assert(stream->sm_hblock_sz == test_vals.prefix_sz + test_vals.headers_sz); assert(0 == stream->sm_n_buffered); nw = lsquic_stream_write(stream, "hello", 5); assert(0 == nw); s = lsquic_stream_flush(stream); assert(s == 0); lsquic_stream_destroy(stream); /* Mock server side stream cycle */ stream = new_stream(&tobjs, 8, 0x1000); uh = calloc(1, sizeof(*uh)); *uh = (struct uncompressed_headers) { .uh_stream_id = stream->id, .uh_weight = 127, .uh_hset = (void *) 12345, }; s = lsquic_stream_uh_in(stream, uh); assert(s == 0); hset = lsquic_stream_get_hset(stream); assert(hset == (void *) 12345); s = lsquic_stream_shutdown(stream, 0); assert(0 == s); test_vals.status = QWH_PARTIAL; test_vals.prefix_sz = 2; test_vals.headers_sz = 40; test_vals.completion_offset = 10; assert(!(stream->sm_qflags & SMQF_WANT_WRITE)); /* Begin with them off */ s = lsquic_stream_send_headers(stream, headers, 0); assert(0 == s); assert(stream->sm_hblock_sz == test_vals.prefix_sz + test_vals.headers_sz); assert(0 == stream->sm_n_buffered); assert(stream->sm_qflags & SMQF_WANT_WRITE); /* Want write is now set */ nw = lsquic_stream_write(stream, "hello", 5); assert(0 == nw); s = lsquic_stream_flush(stream); assert(s == 0); s = lsquic_stream_close(stream); assert(s == 0); assert(stream->sm_hblock_sz == test_vals.prefix_sz + test_vals.headers_sz); assert(0 == stream->sm_n_buffered); assert(stream->sm_qflags & SMQF_WANT_WRITE); /* Still set */ s_enc_off = 10; /* Encoder is done writing */ lsquic_stream_dispatch_write_events(stream); assert(stream->sm_qflags & SMQF_CALL_ONCLOSE); lsquic_stream_acked(stream, QUIC_FRAME_STREAM); lsquic_stream_call_on_close(stream); assert(stream->sm_qflags & SMQF_FREE_STREAM); lsquic_stream_destroy(stream); deinit_test_objs(&tobjs); } static void test_headers_wantwrite_restoration (const int want_write) { struct test_objs tobjs; struct lsquic_stream *stream; ssize_t nw; int s; struct uncompressed_headers *uh; void *hset; s_call_wantwrite_in_ctor = 1; s_wantwrite_arg = want_write; /* For our tests purposes, we treat headers as an opaque object */ struct lsquic_http_headers *headers = (void *) 1; init_test_objs(&tobjs, 0x1000, 0x1000, SCF_IETF); /* Mock server side stream cycle */ stream = new_stream(&tobjs, 4 * __LINE__, 0x1000); uh = calloc(1, sizeof(*uh)); *uh = (struct uncompressed_headers) { .uh_stream_id = stream->id, .uh_weight = 127, .uh_hset = (void *) 12345, }; s = lsquic_stream_uh_in(stream, uh); assert(s == 0); hset = lsquic_stream_get_hset(stream); assert(hset == (void *) 12345); s = lsquic_stream_shutdown(stream, 0); assert(0 == s); test_vals.status = QWH_PARTIAL; test_vals.prefix_sz = 2; test_vals.headers_sz = 40; test_vals.completion_offset = 10; assert(want_write == !!(stream->sm_qflags & SMQF_WANT_WRITE)); s = lsquic_stream_send_headers(stream, headers, 0); assert(0 == s); assert(stream->sm_hblock_sz == test_vals.prefix_sz + test_vals.headers_sz); assert(0 == stream->sm_n_buffered); assert(stream->sm_qflags & SMQF_WANT_WRITE); /* Want write is now set */ nw = lsquic_stream_write(stream, "hello", 5); assert(0 == nw); s = lsquic_stream_flush(stream); assert(s == 0); s = lsquic_stream_close(stream); assert(s == 0); assert(stream->sm_hblock_sz == test_vals.prefix_sz + test_vals.headers_sz); assert(0 == stream->sm_n_buffered); assert(stream->sm_qflags & SMQF_WANT_WRITE); /* Still set */ s_enc_off = 10; /* Encoder is done writing */ lsquic_stream_dispatch_write_events(stream); assert(stream->sm_qflags & SMQF_CALL_ONCLOSE); lsquic_stream_acked(stream, QUIC_FRAME_STREAM); lsquic_stream_call_on_close(stream); assert(stream->sm_qflags & SMQF_FREE_STREAM); lsquic_stream_destroy(stream); stream = new_stream(&tobjs, 4 * __LINE__, 0x1000); uh = calloc(1, sizeof(*uh)); *uh = (struct uncompressed_headers) { .uh_stream_id = stream->id, .uh_weight = 127, .uh_hset = (void *) 12345, }; s = lsquic_stream_uh_in(stream, uh); assert(s == 0); hset = lsquic_stream_get_hset(stream); assert(hset == (void *) 12345); s = lsquic_stream_shutdown(stream, 0); assert(0 == s); test_vals.status = QWH_PARTIAL; test_vals.prefix_sz = 2; test_vals.headers_sz = 40; test_vals.completion_offset = 10; assert(want_write == !!(stream->sm_qflags & SMQF_WANT_WRITE)); s = lsquic_stream_send_headers(stream, headers, 0); assert(0 == s); assert(stream->sm_hblock_sz == test_vals.prefix_sz + test_vals.headers_sz); assert(0 == stream->sm_n_buffered); assert(stream->sm_qflags & SMQF_WANT_WRITE); /* Want write is now set */ s_enc_off = 10; /* Encoder is done writing */ lsquic_stream_dispatch_write_events(stream); assert(0 == stream->sm_hblock_sz); /* Wrote header */ assert(want_write == s_onwrite_called); lsquic_stream_destroy(stream); deinit_test_objs(&tobjs); s_call_wantwrite_in_ctor = 0; s_wantwrite_arg = 0; s_onwrite_called = 0; } static void test_pp_wantwrite_restoration (const int want_write) { struct test_objs tobjs; struct lsquic_stream *stream; int s; struct uncompressed_headers *uh; struct push_promise *promise; void *hset; s_call_wantwrite_in_ctor = 1; s_wantwrite_arg = want_write; init_test_objs(&tobjs, 0x1000, 0x1000, SCF_IETF); /* Mock server side stream cycle */ stream = new_stream(&tobjs, 4 * __LINE__, 10); uh = calloc(1, sizeof(*uh)); *uh = (struct uncompressed_headers) { .uh_stream_id = stream->id, .uh_weight = 127, .uh_hset = (void *) 12345, }; s = lsquic_stream_uh_in(stream, uh); assert(s == 0); hset = lsquic_stream_get_hset(stream); assert(hset == (void *) 12345); s = lsquic_stream_shutdown(stream, 0); assert(0 == s); promise = calloc(1, sizeof(*promise) + 20); promise->pp_id = 0; promise->pp_content_len = 20; assert(want_write == !!(stream->sm_qflags & SMQF_WANT_WRITE)); s = lsquic_stream_push_promise(stream, promise); assert(s == 0); assert((stream->stream_flags & (STREAM_NOPUSH|STREAM_PUSHING)) == (STREAM_NOPUSH|STREAM_PUSHING)); assert(stream->sm_qflags & SMQF_WANT_WRITE); /* Want write is now set */ /* Dispatch: there should be no progress made */ lsquic_stream_dispatch_write_events(stream); assert((stream->stream_flags & (STREAM_NOPUSH|STREAM_PUSHING)) == (STREAM_NOPUSH|STREAM_PUSHING)); assert(stream->sm_qflags & SMQF_WANT_WRITE); assert(SLIST_FIRST(&stream->sm_promises)->pp_write_state != PPWS_DONE); /* Now update window and dispatch again */ lsquic_stream_window_update(stream, 100); lsquic_stream_dispatch_write_events(stream); assert((stream->stream_flags & (STREAM_NOPUSH|STREAM_PUSHING)) == (STREAM_NOPUSH|STREAM_PUSHING)); assert(SLIST_FIRST(&stream->sm_promises)->pp_write_state == PPWS_DONE); /* Done! */ assert(want_write == s_onwrite_called); /* Restored: and on_write called */ lsquic_stream_destroy(stream); deinit_test_objs(&tobjs); s_call_wantwrite_in_ctor = 0; s_wantwrite_arg = 0; s_onwrite_called = 0; } /* 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; } static stream_frame_t * new_frame_in (struct test_objs *tobjs, size_t off, size_t sz, int fin) { return new_frame_in_ext(tobjs, off, sz, fin, NULL); } /* Test that reading from stream returns -1/EWOULDBLOCK if no headers are * available. */ static void test_read_headers (int ietf, int use_hset) { struct test_objs tobjs; struct lsquic_stream *stream; struct stream_frame *frame; ssize_t nr; int s; void *hset; unsigned char buf[1]; init_test_objs(&tobjs, 0x1000, 0x1000, ietf ? SCF_IETF : 0); stream = new_stream(&tobjs, 0, 0x1000); frame = new_frame_in(&tobjs, 0, 35, 1); s = lsquic_stream_frame_in(stream, frame); assert(s == 0); if (use_hset) { hset = lsquic_stream_get_hset(stream); assert(NULL == hset); } else { nr = lsquic_stream_read(stream, buf, sizeof(buf)); assert(-1 == nr); /* In GQUIC mode, the error is that the headers are no available yet. * In IETF mode, the error is that we hit EOF unexpectedly -- as headers * are sent on the same stream in HEADERS frame. */ if (!ietf) assert(EWOULDBLOCK == errno); } lsquic_stream_destroy(stream); deinit_test_objs(&tobjs); } int main (int argc, char **argv) { int opt; lsquic_global_init(LSQUIC_GLOBAL_SERVER); while (-1 != (opt = getopt(argc, argv, "l:"))) { switch (opt) { case 'l': lsquic_log_to_fstream(stderr, 0); lsquic_logger_lopt(optarg); break; default: exit(1); } } test_flushes_and_closes(); test_headers_wantwrite_restoration(0); test_headers_wantwrite_restoration(1); test_pp_wantwrite_restoration(0); test_pp_wantwrite_restoration(1); test_read_headers(0, 0); test_read_headers(0, 1); test_read_headers(1, 0); test_read_headers(1, 1); return 0; }