/* * nghttp3 * * Copyright (c) 2019 nghttp3 contributors * * Permission is hereby granted, free of charge, to any person obtaining * a copy of this software and associated documentation files (the * "Software"), to deal in the Software without restriction, including * without limitation the rights to use, copy, modify, merge, publish, * distribute, sublicense, and/or sell copies of the Software, and to * permit persons to whom the Software is furnished to do so, subject to * the following conditions: * * The above copyright notice and this permission notice shall be * included in all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE * LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION * OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION * WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */ #include "nghttp3_conn.h" #include #include #include #include "nghttp3_mem.h" #include "nghttp3_macro.h" #include "nghttp3_err.h" #include "nghttp3_conv.h" #include "nghttp3_http.h" #include "nghttp3_unreachable.h" /* NGHTTP3_QPACK_ENCODER_MAX_DTABLE_CAPACITY is the upper bound of the dynamic table capacity that QPACK encoder is willing to use. */ #define NGHTTP3_QPACK_ENCODER_MAX_DTABLE_CAPACITY 4096 /* * conn_remote_stream_uni returns nonzero if |stream_id| is remote * unidirectional stream ID. */ static int conn_remote_stream_uni(nghttp3_conn *conn, int64_t stream_id) { if (conn->server) { return (stream_id & 0x03) == 0x02; } return (stream_id & 0x03) == 0x03; } static int conn_call_begin_headers(nghttp3_conn *conn, nghttp3_stream *stream) { int rv; if (!conn->callbacks.begin_headers) { return 0; } rv = conn->callbacks.begin_headers(conn, stream->node.id, conn->user_data, stream->user_data); if (rv != 0) { /* TODO Allow ignore headers */ return NGHTTP3_ERR_CALLBACK_FAILURE; } return 0; } static int conn_call_end_headers(nghttp3_conn *conn, nghttp3_stream *stream, int fin) { int rv; if (!conn->callbacks.end_headers) { return 0; } rv = conn->callbacks.end_headers(conn, stream->node.id, fin, conn->user_data, stream->user_data); if (rv != 0) { /* TODO Allow ignore headers */ return NGHTTP3_ERR_CALLBACK_FAILURE; } return 0; } static int conn_call_begin_trailers(nghttp3_conn *conn, nghttp3_stream *stream) { int rv; if (!conn->callbacks.begin_trailers) { return 0; } rv = conn->callbacks.begin_trailers(conn, stream->node.id, conn->user_data, stream->user_data); if (rv != 0) { /* TODO Allow ignore headers */ return NGHTTP3_ERR_CALLBACK_FAILURE; } return 0; } static int conn_call_end_trailers(nghttp3_conn *conn, nghttp3_stream *stream, int fin) { int rv; if (!conn->callbacks.end_trailers) { return 0; } rv = conn->callbacks.end_trailers(conn, stream->node.id, fin, conn->user_data, stream->user_data); if (rv != 0) { /* TODO Allow ignore headers */ return NGHTTP3_ERR_CALLBACK_FAILURE; } return 0; } static int conn_call_end_stream(nghttp3_conn *conn, nghttp3_stream *stream) { int rv; if (!conn->callbacks.end_stream) { return 0; } rv = conn->callbacks.end_stream(conn, stream->node.id, conn->user_data, stream->user_data); if (rv != 0) { return NGHTTP3_ERR_CALLBACK_FAILURE; } return 0; } static int conn_call_stop_sending(nghttp3_conn *conn, nghttp3_stream *stream, uint64_t app_error_code) { int rv; if (!conn->callbacks.stop_sending) { return 0; } rv = conn->callbacks.stop_sending(conn, stream->node.id, app_error_code, conn->user_data, stream->user_data); if (rv != 0) { return NGHTTP3_ERR_CALLBACK_FAILURE; } return 0; } static int conn_call_reset_stream(nghttp3_conn *conn, nghttp3_stream *stream, uint64_t app_error_code) { int rv; if (!conn->callbacks.reset_stream) { return 0; } rv = conn->callbacks.reset_stream(conn, stream->node.id, app_error_code, conn->user_data, stream->user_data); if (rv != 0) { return NGHTTP3_ERR_CALLBACK_FAILURE; } return 0; } static int conn_call_deferred_consume(nghttp3_conn *conn, nghttp3_stream *stream, size_t nconsumed) { int rv; if (nconsumed == 0 || !conn->callbacks.deferred_consume) { return 0; } rv = conn->callbacks.deferred_consume(conn, stream->node.id, nconsumed, conn->user_data, stream->user_data); if (rv != 0) { return NGHTTP3_ERR_CALLBACK_FAILURE; } return 0; } static int ricnt_less(const nghttp3_pq_entry *lhsx, const nghttp3_pq_entry *rhsx) { nghttp3_stream *lhs = nghttp3_struct_of(lhsx, nghttp3_stream, qpack_blocked_pe); nghttp3_stream *rhs = nghttp3_struct_of(rhsx, nghttp3_stream, qpack_blocked_pe); return lhs->qpack_sctx.ricnt < rhs->qpack_sctx.ricnt; } static int cycle_less(const nghttp3_pq_entry *lhsx, const nghttp3_pq_entry *rhsx) { const nghttp3_tnode *lhs = nghttp3_struct_of(lhsx, nghttp3_tnode, pe); const nghttp3_tnode *rhs = nghttp3_struct_of(rhsx, nghttp3_tnode, pe); if (lhs->cycle == rhs->cycle) { return lhs->id < rhs->id; } return rhs->cycle - lhs->cycle <= NGHTTP3_TNODE_MAX_CYCLE_GAP; } static int conn_new(nghttp3_conn **pconn, int server, int callbacks_version, const nghttp3_callbacks *callbacks, int settings_version, const nghttp3_settings *settings, const nghttp3_mem *mem, void *user_data) { int rv; nghttp3_conn *conn; size_t i; (void)callbacks_version; (void)settings_version; if (mem == NULL) { mem = nghttp3_mem_default(); } conn = nghttp3_mem_calloc(mem, 1, sizeof(nghttp3_conn)); if (conn == NULL) { return NGHTTP3_ERR_NOMEM; } nghttp3_objalloc_init(&conn->out_chunk_objalloc, NGHTTP3_STREAM_MIN_CHUNK_SIZE * 16, mem); nghttp3_objalloc_stream_init(&conn->stream_objalloc, 64, mem); nghttp3_map_init(&conn->streams, mem); rv = nghttp3_qpack_decoder_init(&conn->qdec, settings->qpack_max_dtable_capacity, settings->qpack_blocked_streams, mem); if (rv != 0) { goto qdec_init_fail; } rv = nghttp3_qpack_encoder_init( &conn->qenc, settings->qpack_encoder_max_dtable_capacity, mem); if (rv != 0) { goto qenc_init_fail; } nghttp3_pq_init(&conn->qpack_blocked_streams, ricnt_less, mem); for (i = 0; i < NGHTTP3_URGENCY_LEVELS; ++i) { nghttp3_pq_init(&conn->sched[i].spq, cycle_less, mem); } nghttp3_idtr_init(&conn->remote.bidi.idtr, server, mem); conn->callbacks = *callbacks; conn->local.settings = *settings; if (!server) { conn->local.settings.enable_connect_protocol = 0; } nghttp3_settings_default(&conn->remote.settings); conn->mem = mem; conn->user_data = user_data; conn->server = server; conn->rx.goaway_id = NGHTTP3_VARINT_MAX + 1; conn->tx.goaway_id = NGHTTP3_VARINT_MAX + 1; conn->rx.max_stream_id_bidi = -4; *pconn = conn; return 0; qenc_init_fail: nghttp3_qpack_decoder_free(&conn->qdec); qdec_init_fail: nghttp3_map_free(&conn->streams); nghttp3_objalloc_free(&conn->stream_objalloc); nghttp3_objalloc_free(&conn->out_chunk_objalloc); nghttp3_mem_free(mem, conn); return rv; } int nghttp3_conn_client_new_versioned(nghttp3_conn **pconn, int callbacks_version, const nghttp3_callbacks *callbacks, int settings_version, const nghttp3_settings *settings, const nghttp3_mem *mem, void *user_data) { int rv; rv = conn_new(pconn, /* server = */ 0, callbacks_version, callbacks, settings_version, settings, mem, user_data); if (rv != 0) { return rv; } return 0; } int nghttp3_conn_server_new_versioned(nghttp3_conn **pconn, int callbacks_version, const nghttp3_callbacks *callbacks, int settings_version, const nghttp3_settings *settings, const nghttp3_mem *mem, void *user_data) { int rv; rv = conn_new(pconn, /* server = */ 1, callbacks_version, callbacks, settings_version, settings, mem, user_data); if (rv != 0) { return rv; } return 0; } static int free_stream(void *data, void *ptr) { nghttp3_stream *stream = data; (void)ptr; nghttp3_stream_del(stream); return 0; } void nghttp3_conn_del(nghttp3_conn *conn) { size_t i; if (conn == NULL) { return; } nghttp3_buf_free(&conn->tx.qpack.ebuf, conn->mem); nghttp3_buf_free(&conn->tx.qpack.rbuf, conn->mem); nghttp3_idtr_free(&conn->remote.bidi.idtr); for (i = 0; i < NGHTTP3_URGENCY_LEVELS; ++i) { nghttp3_pq_free(&conn->sched[i].spq); } nghttp3_pq_free(&conn->qpack_blocked_streams); nghttp3_qpack_encoder_free(&conn->qenc); nghttp3_qpack_decoder_free(&conn->qdec); nghttp3_map_each_free(&conn->streams, free_stream, NULL); nghttp3_map_free(&conn->streams); nghttp3_objalloc_free(&conn->stream_objalloc); nghttp3_objalloc_free(&conn->out_chunk_objalloc); nghttp3_mem_free(conn->mem, conn); } static int conn_bidi_idtr_open(nghttp3_conn *conn, int64_t stream_id) { int rv; rv = nghttp3_idtr_open(&conn->remote.bidi.idtr, stream_id); if (rv != 0) { return rv; } if (nghttp3_ksl_len(&conn->remote.bidi.idtr.gap.gap) > 32) { nghttp3_gaptr_drop_first_gap(&conn->remote.bidi.idtr.gap); } return 0; } nghttp3_ssize nghttp3_conn_read_stream(nghttp3_conn *conn, int64_t stream_id, const uint8_t *src, size_t srclen, int fin) { nghttp3_stream *stream; size_t bidi_nproc; int rv; stream = nghttp3_conn_find_stream(conn, stream_id); if (stream == NULL) { /* TODO Assert idtr */ /* QUIC transport ensures that this is new stream. */ if (conn->server) { if (nghttp3_client_stream_bidi(stream_id)) { rv = conn_bidi_idtr_open(conn, stream_id); if (rv != 0) { if (nghttp3_err_is_fatal(rv)) { return rv; } /* Ignore return value. We might drop the first gap if there are many gaps if QUIC stack allows too many holes in stream ID space. idtr is used to decide whether PRIORITY_UPDATE frame should be ignored or not and the frame is optional. Ignoring them causes no harm. */ } conn->rx.max_stream_id_bidi = nghttp3_max(conn->rx.max_stream_id_bidi, stream_id); rv = nghttp3_conn_create_stream(conn, &stream, stream_id); if (rv != 0) { return rv; } if ((conn->flags & NGHTTP3_CONN_FLAG_GOAWAY_QUEUED) && conn->tx.goaway_id <= stream_id) { stream->rstate.state = NGHTTP3_REQ_STREAM_STATE_IGN_REST; rv = nghttp3_conn_reject_stream(conn, stream); if (rv != 0) { return rv; } } } else { /* unidirectional stream */ if (srclen == 0 && fin) { return 0; } rv = nghttp3_conn_create_stream(conn, &stream, stream_id); if (rv != 0) { return rv; } } stream->rx.hstate = NGHTTP3_HTTP_STATE_REQ_INITIAL; stream->tx.hstate = NGHTTP3_HTTP_STATE_REQ_INITIAL; } else if (nghttp3_stream_uni(stream_id)) { if (srclen == 0 && fin) { return 0; } rv = nghttp3_conn_create_stream(conn, &stream, stream_id); if (rv != 0) { return rv; } stream->rx.hstate = NGHTTP3_HTTP_STATE_RESP_INITIAL; stream->tx.hstate = NGHTTP3_HTTP_STATE_RESP_INITIAL; } else { /* client doesn't expect to receive new bidirectional stream from server. */ return NGHTTP3_ERR_H3_STREAM_CREATION_ERROR; } } else if (conn->server) { if (nghttp3_client_stream_bidi(stream_id)) { if (stream->rx.hstate == NGHTTP3_HTTP_STATE_NONE) { stream->rx.hstate = NGHTTP3_HTTP_STATE_REQ_INITIAL; stream->tx.hstate = NGHTTP3_HTTP_STATE_REQ_INITIAL; } } } if (srclen == 0 && !fin) { return 0; } if (nghttp3_stream_uni(stream_id)) { return nghttp3_conn_read_uni(conn, stream, src, srclen, fin); } if (fin) { stream->flags |= NGHTTP3_STREAM_FLAG_READ_EOF; } return nghttp3_conn_read_bidi(conn, &bidi_nproc, stream, src, srclen, fin); } static nghttp3_ssize conn_read_type(nghttp3_conn *conn, nghttp3_stream *stream, const uint8_t *src, size_t srclen, int fin) { nghttp3_stream_read_state *rstate = &stream->rstate; nghttp3_varint_read_state *rvint = &rstate->rvint; nghttp3_ssize nread; int64_t stream_type; assert(srclen); nread = nghttp3_read_varint(rvint, src, srclen, fin); if (nread < 0) { return NGHTTP3_ERR_H3_GENERAL_PROTOCOL_ERROR; } if (rvint->left) { return nread; } stream_type = rvint->acc; nghttp3_varint_read_state_reset(rvint); switch (stream_type) { case NGHTTP3_STREAM_TYPE_CONTROL: if (conn->flags & NGHTTP3_CONN_FLAG_CONTROL_OPENED) { return NGHTTP3_ERR_H3_STREAM_CREATION_ERROR; } conn->flags |= NGHTTP3_CONN_FLAG_CONTROL_OPENED; stream->type = NGHTTP3_STREAM_TYPE_CONTROL; rstate->state = NGHTTP3_CTRL_STREAM_STATE_FRAME_TYPE; break; case NGHTTP3_STREAM_TYPE_PUSH: return NGHTTP3_ERR_H3_STREAM_CREATION_ERROR; case NGHTTP3_STREAM_TYPE_QPACK_ENCODER: if (conn->flags & NGHTTP3_CONN_FLAG_QPACK_ENCODER_OPENED) { return NGHTTP3_ERR_H3_STREAM_CREATION_ERROR; } conn->flags |= NGHTTP3_CONN_FLAG_QPACK_ENCODER_OPENED; stream->type = NGHTTP3_STREAM_TYPE_QPACK_ENCODER; break; case NGHTTP3_STREAM_TYPE_QPACK_DECODER: if (conn->flags & NGHTTP3_CONN_FLAG_QPACK_DECODER_OPENED) { return NGHTTP3_ERR_H3_STREAM_CREATION_ERROR; } conn->flags |= NGHTTP3_CONN_FLAG_QPACK_DECODER_OPENED; stream->type = NGHTTP3_STREAM_TYPE_QPACK_DECODER; break; default: stream->type = NGHTTP3_STREAM_TYPE_UNKNOWN; break; } stream->flags |= NGHTTP3_STREAM_FLAG_TYPE_IDENTIFIED; return nread; } static int conn_delete_stream(nghttp3_conn *conn, nghttp3_stream *stream); nghttp3_ssize nghttp3_conn_read_uni(nghttp3_conn *conn, nghttp3_stream *stream, const uint8_t *src, size_t srclen, int fin) { nghttp3_ssize nread = 0; nghttp3_ssize nconsumed = 0; int rv; assert(srclen || fin); if (!(stream->flags & NGHTTP3_STREAM_FLAG_TYPE_IDENTIFIED)) { if (srclen == 0 && fin) { /* Ignore stream if it is closed before reading stream header. If it is closed while reading it, return error, making it consistent in our code base. */ if (stream->rstate.rvint.left) { return NGHTTP3_ERR_H3_GENERAL_PROTOCOL_ERROR; } rv = conn_delete_stream(conn, stream); assert(0 == rv); return 0; } nread = conn_read_type(conn, stream, src, srclen, fin); if (nread < 0) { return (int)nread; } if (!(stream->flags & NGHTTP3_STREAM_FLAG_TYPE_IDENTIFIED)) { assert((size_t)nread == srclen); return (nghttp3_ssize)srclen; } src += nread; srclen -= (size_t)nread; if (srclen == 0) { return nread; } } switch (stream->type) { case NGHTTP3_STREAM_TYPE_CONTROL: if (fin) { return NGHTTP3_ERR_H3_CLOSED_CRITICAL_STREAM; } nconsumed = nghttp3_conn_read_control(conn, stream, src, srclen); break; case NGHTTP3_STREAM_TYPE_QPACK_ENCODER: if (fin) { return NGHTTP3_ERR_H3_CLOSED_CRITICAL_STREAM; } nconsumed = nghttp3_conn_read_qpack_encoder(conn, src, srclen); break; case NGHTTP3_STREAM_TYPE_QPACK_DECODER: if (fin) { return NGHTTP3_ERR_H3_CLOSED_CRITICAL_STREAM; } nconsumed = nghttp3_conn_read_qpack_decoder(conn, src, srclen); break; case NGHTTP3_STREAM_TYPE_UNKNOWN: nconsumed = (nghttp3_ssize)srclen; rv = conn_call_stop_sending(conn, stream, NGHTTP3_H3_STREAM_CREATION_ERROR); if (rv != 0) { return rv; } break; default: nghttp3_unreachable(); } if (nconsumed < 0) { return nconsumed; } return nread + nconsumed; } static int frame_fin(nghttp3_stream_read_state *rstate, size_t len) { return (int64_t)len >= rstate->left; } nghttp3_ssize nghttp3_conn_read_control(nghttp3_conn *conn, nghttp3_stream *stream, const uint8_t *src, size_t srclen) { const uint8_t *p = src, *end = src + srclen; int rv; nghttp3_stream_read_state *rstate = &stream->rstate; nghttp3_varint_read_state *rvint = &rstate->rvint; nghttp3_ssize nread; size_t nconsumed = 0; int busy = 0; size_t len; const uint8_t *pri_field_value = NULL; size_t pri_field_valuelen = 0; assert(srclen); for (; p != end || busy;) { busy = 0; switch (rstate->state) { case NGHTTP3_CTRL_STREAM_STATE_FRAME_TYPE: assert(end - p > 0); nread = nghttp3_read_varint(rvint, p, (size_t)(end - p), /* fin = */ 0); if (nread < 0) { return NGHTTP3_ERR_H3_GENERAL_PROTOCOL_ERROR; } p += nread; nconsumed += (size_t)nread; if (rvint->left) { return (nghttp3_ssize)nconsumed; } rstate->fr.hd.type = rvint->acc; nghttp3_varint_read_state_reset(rvint); rstate->state = NGHTTP3_CTRL_STREAM_STATE_FRAME_LENGTH; if (p == end) { break; } /* Fall through */ case NGHTTP3_CTRL_STREAM_STATE_FRAME_LENGTH: assert(end - p > 0); nread = nghttp3_read_varint(rvint, p, (size_t)(end - p), /* fin = */ 0); if (nread < 0) { return NGHTTP3_ERR_H3_FRAME_ERROR; } p += nread; nconsumed += (size_t)nread; if (rvint->left) { return (nghttp3_ssize)nconsumed; } rstate->left = rstate->fr.hd.length = rvint->acc; nghttp3_varint_read_state_reset(rvint); if (!(conn->flags & NGHTTP3_CONN_FLAG_SETTINGS_RECVED)) { if (rstate->fr.hd.type != NGHTTP3_FRAME_SETTINGS) { return NGHTTP3_ERR_H3_MISSING_SETTINGS; } conn->flags |= NGHTTP3_CONN_FLAG_SETTINGS_RECVED; } else if (rstate->fr.hd.type == NGHTTP3_FRAME_SETTINGS) { return NGHTTP3_ERR_H3_FRAME_UNEXPECTED; } switch (rstate->fr.hd.type) { case NGHTTP3_FRAME_SETTINGS: /* SETTINGS frame might be empty. */ if (rstate->left == 0) { nghttp3_stream_read_state_reset(rstate); break; } rstate->state = NGHTTP3_CTRL_STREAM_STATE_SETTINGS; break; case NGHTTP3_FRAME_GOAWAY: if (rstate->left == 0) { return NGHTTP3_ERR_H3_FRAME_ERROR; } rstate->state = NGHTTP3_CTRL_STREAM_STATE_GOAWAY; break; case NGHTTP3_FRAME_MAX_PUSH_ID: if (!conn->server) { return NGHTTP3_ERR_H3_FRAME_UNEXPECTED; } if (rstate->left == 0) { return NGHTTP3_ERR_H3_FRAME_ERROR; } rstate->state = NGHTTP3_CTRL_STREAM_STATE_MAX_PUSH_ID; break; case NGHTTP3_FRAME_PRIORITY_UPDATE: if (!conn->server) { return NGHTTP3_ERR_H3_FRAME_UNEXPECTED; } if (rstate->left == 0) { return NGHTTP3_ERR_H3_FRAME_ERROR; } rstate->state = NGHTTP3_CTRL_STREAM_STATE_PRIORITY_UPDATE_PRI_ELEM_ID; break; case NGHTTP3_FRAME_PRIORITY_UPDATE_PUSH_ID: /* We do not support push */ return NGHTTP3_ERR_H3_ID_ERROR; case NGHTTP3_FRAME_CANCEL_PUSH: /* We do not support push */ case NGHTTP3_FRAME_DATA: case NGHTTP3_FRAME_HEADERS: case NGHTTP3_FRAME_PUSH_PROMISE: case NGHTTP3_H2_FRAME_PRIORITY: case NGHTTP3_H2_FRAME_PING: case NGHTTP3_H2_FRAME_WINDOW_UPDATE: case NGHTTP3_H2_FRAME_CONTINUATION: return NGHTTP3_ERR_H3_FRAME_UNEXPECTED; default: /* TODO Handle reserved frame type */ busy = 1; rstate->state = NGHTTP3_CTRL_STREAM_STATE_IGN_FRAME; break; } break; case NGHTTP3_CTRL_STREAM_STATE_SETTINGS: for (; p != end;) { if (rstate->left == 0) { nghttp3_stream_read_state_reset(rstate); break; } /* Read Identifier */ len = (size_t)nghttp3_min(rstate->left, (int64_t)(end - p)); assert(len > 0); nread = nghttp3_read_varint(rvint, p, len, frame_fin(rstate, len)); if (nread < 0) { return NGHTTP3_ERR_H3_FRAME_ERROR; } p += nread; nconsumed += (size_t)nread; rstate->left -= nread; if (rvint->left) { rstate->state = NGHTTP3_CTRL_STREAM_STATE_SETTINGS_ID; return (nghttp3_ssize)nconsumed; } rstate->fr.settings.iv[0].id = (uint64_t)rvint->acc; nghttp3_varint_read_state_reset(rvint); /* Read Value */ if (rstate->left == 0) { return NGHTTP3_ERR_H3_FRAME_ERROR; } len -= (size_t)nread; if (len == 0) { rstate->state = NGHTTP3_CTRL_STREAM_STATE_SETTINGS_VALUE; break; } nread = nghttp3_read_varint(rvint, p, len, frame_fin(rstate, len)); if (nread < 0) { return NGHTTP3_ERR_H3_FRAME_ERROR; } p += nread; nconsumed += (size_t)nread; rstate->left -= nread; if (rvint->left) { rstate->state = NGHTTP3_CTRL_STREAM_STATE_SETTINGS_VALUE; return (nghttp3_ssize)nconsumed; } rstate->fr.settings.iv[0].value = (uint64_t)rvint->acc; nghttp3_varint_read_state_reset(rvint); rv = nghttp3_conn_on_settings_entry_received(conn, &rstate->fr.settings); if (rv != 0) { return rv; } } break; case NGHTTP3_CTRL_STREAM_STATE_SETTINGS_ID: len = (size_t)nghttp3_min(rstate->left, (int64_t)(end - p)); assert(len > 0); nread = nghttp3_read_varint(rvint, p, len, frame_fin(rstate, len)); if (nread < 0) { return NGHTTP3_ERR_H3_FRAME_ERROR; } p += nread; nconsumed += (size_t)nread; rstate->left -= nread; if (rvint->left) { return (nghttp3_ssize)nconsumed; } rstate->fr.settings.iv[0].id = (uint64_t)rvint->acc; nghttp3_varint_read_state_reset(rvint); if (rstate->left == 0) { return NGHTTP3_ERR_H3_FRAME_ERROR; } rstate->state = NGHTTP3_CTRL_STREAM_STATE_SETTINGS_VALUE; if (p == end) { return (nghttp3_ssize)nconsumed; } /* Fall through */ case NGHTTP3_CTRL_STREAM_STATE_SETTINGS_VALUE: len = (size_t)nghttp3_min(rstate->left, (int64_t)(end - p)); assert(len > 0); nread = nghttp3_read_varint(rvint, p, len, frame_fin(rstate, len)); if (nread < 0) { return NGHTTP3_ERR_H3_FRAME_ERROR; } p += nread; nconsumed += (size_t)nread; rstate->left -= nread; if (rvint->left) { return (nghttp3_ssize)nconsumed; } rstate->fr.settings.iv[0].value = (uint64_t)rvint->acc; nghttp3_varint_read_state_reset(rvint); rv = nghttp3_conn_on_settings_entry_received(conn, &rstate->fr.settings); if (rv != 0) { return rv; } if (rstate->left) { rstate->state = NGHTTP3_CTRL_STREAM_STATE_SETTINGS; break; } nghttp3_stream_read_state_reset(rstate); break; case NGHTTP3_CTRL_STREAM_STATE_GOAWAY: len = (size_t)nghttp3_min(rstate->left, (int64_t)(end - p)); assert(len > 0); nread = nghttp3_read_varint(rvint, p, len, frame_fin(rstate, len)); if (nread < 0) { return NGHTTP3_ERR_H3_FRAME_ERROR; } p += nread; nconsumed += (size_t)nread; rstate->left -= nread; if (rvint->left) { return (nghttp3_ssize)nconsumed; } if (!conn->server && !nghttp3_client_stream_bidi(rvint->acc)) { return NGHTTP3_ERR_H3_ID_ERROR; } if (conn->rx.goaway_id < rvint->acc) { return NGHTTP3_ERR_H3_ID_ERROR; } conn->flags |= NGHTTP3_CONN_FLAG_GOAWAY_RECVED; conn->rx.goaway_id = rvint->acc; nghttp3_varint_read_state_reset(rvint); if (conn->callbacks.shutdown) { rv = conn->callbacks.shutdown(conn, conn->rx.goaway_id, conn->user_data); if (rv != 0) { return NGHTTP3_ERR_CALLBACK_FAILURE; } } nghttp3_stream_read_state_reset(rstate); break; case NGHTTP3_CTRL_STREAM_STATE_MAX_PUSH_ID: /* server side only */ len = (size_t)nghttp3_min(rstate->left, (int64_t)(end - p)); assert(len > 0); nread = nghttp3_read_varint(rvint, p, len, frame_fin(rstate, len)); if (nread < 0) { return NGHTTP3_ERR_H3_FRAME_ERROR; } p += nread; nconsumed += (size_t)nread; rstate->left -= nread; if (rvint->left) { return (nghttp3_ssize)nconsumed; } if (conn->local.uni.max_pushes > (uint64_t)rvint->acc + 1) { return NGHTTP3_ERR_H3_FRAME_ERROR; } conn->local.uni.max_pushes = (uint64_t)rvint->acc + 1; nghttp3_varint_read_state_reset(rvint); nghttp3_stream_read_state_reset(rstate); break; case NGHTTP3_CTRL_STREAM_STATE_PRIORITY_UPDATE_PRI_ELEM_ID: /* server side only */ len = (size_t)nghttp3_min(rstate->left, (int64_t)(end - p)); assert(len > 0); nread = nghttp3_read_varint(rvint, p, len, frame_fin(rstate, len)); if (nread < 0) { return NGHTTP3_ERR_H3_FRAME_ERROR; } p += nread; nconsumed += (size_t)nread; rstate->left -= nread; if (rvint->left) { return (nghttp3_ssize)nconsumed; } rstate->fr.priority_update.pri_elem_id = rvint->acc; nghttp3_varint_read_state_reset(rvint); if (rstate->left == 0) { rstate->fr.priority_update.pri.urgency = NGHTTP3_DEFAULT_URGENCY; rstate->fr.priority_update.pri.inc = 0; rv = nghttp3_conn_on_priority_update(conn, &rstate->fr.priority_update); if (rv != 0) { return rv; } nghttp3_stream_read_state_reset(rstate); break; } rstate->state = NGHTTP3_CTRL_STREAM_STATE_PRIORITY_UPDATE; /* Fall through */ case NGHTTP3_CTRL_STREAM_STATE_PRIORITY_UPDATE: /* We need to buffer Priority Field Value because it might be fragmented. */ len = (size_t)nghttp3_min(rstate->left, (int64_t)(end - p)); assert(len > 0); if (conn->rx.pri_fieldbuflen == 0 && rstate->left == (int64_t)len) { /* Everything is in the input buffer. Apply same length limit we impose when buffering the field. */ if (len > sizeof(conn->rx.pri_fieldbuf)) { busy = 1; rstate->state = NGHTTP3_CTRL_STREAM_STATE_IGN_FRAME; break; } pri_field_value = p; pri_field_valuelen = len; } else if (len + conn->rx.pri_fieldbuflen > sizeof(conn->rx.pri_fieldbuf)) { busy = 1; rstate->state = NGHTTP3_CTRL_STREAM_STATE_IGN_FRAME; break; } else { memcpy(conn->rx.pri_fieldbuf + conn->rx.pri_fieldbuflen, p, len); conn->rx.pri_fieldbuflen += len; if (rstate->left == (int64_t)len) { pri_field_value = conn->rx.pri_fieldbuf; pri_field_valuelen = conn->rx.pri_fieldbuflen; } } p += len; nconsumed += len; rstate->left -= (int64_t)len; if (rstate->left) { return (nghttp3_ssize)nconsumed; } rstate->fr.priority_update.pri.urgency = NGHTTP3_DEFAULT_URGENCY; rstate->fr.priority_update.pri.inc = 0; if (nghttp3_http_parse_priority(&rstate->fr.priority_update.pri, pri_field_value, pri_field_valuelen) != 0) { return NGHTTP3_ERR_H3_GENERAL_PROTOCOL_ERROR; } rv = nghttp3_conn_on_priority_update(conn, &rstate->fr.priority_update); if (rv != 0) { return rv; } conn->rx.pri_fieldbuflen = 0; nghttp3_stream_read_state_reset(rstate); break; case NGHTTP3_CTRL_STREAM_STATE_IGN_FRAME: len = (size_t)nghttp3_min(rstate->left, (int64_t)(end - p)); p += len; nconsumed += len; rstate->left -= (int64_t)len; if (rstate->left) { return (nghttp3_ssize)nconsumed; } nghttp3_stream_read_state_reset(rstate); break; default: nghttp3_unreachable(); } } return (nghttp3_ssize)nconsumed; } static int conn_delete_stream(nghttp3_conn *conn, nghttp3_stream *stream) { int bidi = nghttp3_client_stream_bidi(stream->node.id); int rv; rv = conn_call_deferred_consume(conn, stream, nghttp3_stream_get_buffered_datalen(stream)); if (rv != 0) { return rv; } if (bidi && conn->callbacks.stream_close) { rv = conn->callbacks.stream_close(conn, stream->node.id, stream->error_code, conn->user_data, stream->user_data); if (rv != 0) { return NGHTTP3_ERR_CALLBACK_FAILURE; } } rv = nghttp3_map_remove(&conn->streams, (nghttp3_map_key_type)stream->node.id); assert(0 == rv); nghttp3_stream_del(stream); return 0; } static int conn_process_blocked_stream_data(nghttp3_conn *conn, nghttp3_stream *stream) { nghttp3_buf *buf; size_t nproc; nghttp3_ssize nconsumed; int rv; size_t len; assert(nghttp3_client_stream_bidi(stream->node.id)); for (;;) { len = nghttp3_ringbuf_len(&stream->inq); if (len == 0) { break; } buf = nghttp3_ringbuf_get(&stream->inq, 0); nconsumed = nghttp3_conn_read_bidi( conn, &nproc, stream, buf->pos, nghttp3_buf_len(buf), len == 1 && (stream->flags & NGHTTP3_STREAM_FLAG_READ_EOF)); if (nconsumed < 0) { return (int)nconsumed; } buf->pos += nproc; rv = conn_call_deferred_consume(conn, stream, (size_t)nconsumed); if (rv != 0) { return 0; } if (nghttp3_buf_len(buf) == 0) { nghttp3_buf_free(buf, stream->mem); nghttp3_ringbuf_pop_front(&stream->inq); } if (stream->flags & NGHTTP3_STREAM_FLAG_QPACK_DECODE_BLOCKED) { break; } } if (!(stream->flags & NGHTTP3_STREAM_FLAG_QPACK_DECODE_BLOCKED) && (stream->flags & NGHTTP3_STREAM_FLAG_CLOSED)) { assert(stream->qpack_blocked_pe.index == NGHTTP3_PQ_BAD_INDEX); rv = conn_delete_stream(conn, stream); if (rv != 0) { return rv; } } return 0; } nghttp3_ssize nghttp3_conn_read_qpack_encoder(nghttp3_conn *conn, const uint8_t *src, size_t srclen) { nghttp3_ssize nconsumed = nghttp3_qpack_decoder_read_encoder(&conn->qdec, src, srclen); nghttp3_stream *stream; int rv; if (nconsumed < 0) { return nconsumed; } for (; !nghttp3_pq_empty(&conn->qpack_blocked_streams);) { stream = nghttp3_struct_of(nghttp3_pq_top(&conn->qpack_blocked_streams), nghttp3_stream, qpack_blocked_pe); if (nghttp3_qpack_stream_context_get_ricnt(&stream->qpack_sctx) > nghttp3_qpack_decoder_get_icnt(&conn->qdec)) { break; } nghttp3_conn_qpack_blocked_streams_pop(conn); stream->qpack_blocked_pe.index = NGHTTP3_PQ_BAD_INDEX; stream->flags &= (uint16_t)~NGHTTP3_STREAM_FLAG_QPACK_DECODE_BLOCKED; rv = conn_process_blocked_stream_data(conn, stream); if (rv != 0) { return rv; } } return nconsumed; } nghttp3_ssize nghttp3_conn_read_qpack_decoder(nghttp3_conn *conn, const uint8_t *src, size_t srclen) { return nghttp3_qpack_encoder_read_decoder(&conn->qenc, src, srclen); } static nghttp3_tnode *stream_get_sched_node(nghttp3_stream *stream) { return &stream->node; } static int conn_update_stream_priority(nghttp3_conn *conn, nghttp3_stream *stream, uint8_t pri) { assert(nghttp3_client_stream_bidi(stream->node.id)); if (stream->node.pri == pri) { return 0; } nghttp3_conn_unschedule_stream(conn, stream); stream->node.pri = pri; if (nghttp3_stream_require_schedule(stream)) { return nghttp3_conn_schedule_stream(conn, stream); } return 0; } nghttp3_ssize nghttp3_conn_read_bidi(nghttp3_conn *conn, size_t *pnproc, nghttp3_stream *stream, const uint8_t *src, size_t srclen, int fin) { const uint8_t *p = src, *end = src ? src + srclen : src; int rv; nghttp3_stream_read_state *rstate = &stream->rstate; nghttp3_varint_read_state *rvint = &rstate->rvint; nghttp3_ssize nread; size_t nconsumed = 0; int busy = 0; size_t len; if (stream->flags & NGHTTP3_STREAM_FLAG_SHUT_RD) { *pnproc = srclen; return (nghttp3_ssize)srclen; } if (stream->flags & NGHTTP3_STREAM_FLAG_QPACK_DECODE_BLOCKED) { *pnproc = 0; if (srclen == 0) { return 0; } rv = nghttp3_stream_buffer_data(stream, p, (size_t)(end - p)); if (rv != 0) { return rv; } return 0; } for (; p != end || busy;) { busy = 0; switch (rstate->state) { case NGHTTP3_REQ_STREAM_STATE_FRAME_TYPE: assert(end - p > 0); nread = nghttp3_read_varint(rvint, p, (size_t)(end - p), fin); if (nread < 0) { return NGHTTP3_ERR_H3_GENERAL_PROTOCOL_ERROR; } p += nread; nconsumed += (size_t)nread; if (rvint->left) { goto almost_done; } rstate->fr.hd.type = rvint->acc; nghttp3_varint_read_state_reset(rvint); rstate->state = NGHTTP3_REQ_STREAM_STATE_FRAME_LENGTH; if (p == end) { goto almost_done; } /* Fall through */ case NGHTTP3_REQ_STREAM_STATE_FRAME_LENGTH: assert(end - p > 0); nread = nghttp3_read_varint(rvint, p, (size_t)(end - p), fin); if (nread < 0) { return NGHTTP3_ERR_H3_FRAME_ERROR; } p += nread; nconsumed += (size_t)nread; if (rvint->left) { goto almost_done; } rstate->left = rstate->fr.hd.length = rvint->acc; nghttp3_varint_read_state_reset(rvint); switch (rstate->fr.hd.type) { case NGHTTP3_FRAME_DATA: rv = nghttp3_stream_transit_rx_http_state( stream, NGHTTP3_HTTP_EVENT_DATA_BEGIN); if (rv != 0) { return rv; } /* DATA frame might be empty. */ if (rstate->left == 0) { rv = nghttp3_stream_transit_rx_http_state( stream, NGHTTP3_HTTP_EVENT_DATA_END); assert(0 == rv); nghttp3_stream_read_state_reset(rstate); break; } rstate->state = NGHTTP3_REQ_STREAM_STATE_DATA; break; case NGHTTP3_FRAME_HEADERS: rv = nghttp3_stream_transit_rx_http_state( stream, NGHTTP3_HTTP_EVENT_HEADERS_BEGIN); if (rv != 0) { return rv; } if (rstate->left == 0) { rv = nghttp3_stream_empty_headers_allowed(stream); if (rv != 0) { return rv; } rv = nghttp3_stream_transit_rx_http_state( stream, NGHTTP3_HTTP_EVENT_HEADERS_END); assert(0 == rv); nghttp3_stream_read_state_reset(rstate); break; } switch (stream->rx.hstate) { case NGHTTP3_HTTP_STATE_REQ_HEADERS_BEGIN: case NGHTTP3_HTTP_STATE_RESP_HEADERS_BEGIN: rv = conn_call_begin_headers(conn, stream); break; case NGHTTP3_HTTP_STATE_REQ_TRAILERS_BEGIN: case NGHTTP3_HTTP_STATE_RESP_TRAILERS_BEGIN: rv = conn_call_begin_trailers(conn, stream); break; default: nghttp3_unreachable(); } if (rv != 0) { return rv; } rstate->state = NGHTTP3_REQ_STREAM_STATE_HEADERS; break; case NGHTTP3_FRAME_PUSH_PROMISE: /* We do not support push */ case NGHTTP3_FRAME_CANCEL_PUSH: case NGHTTP3_FRAME_SETTINGS: case NGHTTP3_FRAME_GOAWAY: case NGHTTP3_FRAME_MAX_PUSH_ID: case NGHTTP3_FRAME_PRIORITY_UPDATE: case NGHTTP3_FRAME_PRIORITY_UPDATE_PUSH_ID: case NGHTTP3_H2_FRAME_PRIORITY: case NGHTTP3_H2_FRAME_PING: case NGHTTP3_H2_FRAME_WINDOW_UPDATE: case NGHTTP3_H2_FRAME_CONTINUATION: return NGHTTP3_ERR_H3_FRAME_UNEXPECTED; default: /* TODO Handle reserved frame type */ busy = 1; rstate->state = NGHTTP3_REQ_STREAM_STATE_IGN_FRAME; break; } break; case NGHTTP3_REQ_STREAM_STATE_DATA: len = (size_t)nghttp3_min(rstate->left, (int64_t)(end - p)); rv = nghttp3_conn_on_data(conn, stream, p, len); if (rv != 0) { return rv; } p += len; rstate->left -= (int64_t)len; if (rstate->left) { goto almost_done; } rv = nghttp3_stream_transit_rx_http_state(stream, NGHTTP3_HTTP_EVENT_DATA_END); assert(0 == rv); nghttp3_stream_read_state_reset(rstate); break; case NGHTTP3_REQ_STREAM_STATE_HEADERS: len = (size_t)nghttp3_min(rstate->left, (int64_t)(end - p)); nread = nghttp3_conn_on_headers(conn, stream, p, len, (int64_t)len == rstate->left); if (nread < 0) { if (nread == NGHTTP3_ERR_MALFORMED_HTTP_HEADER) { goto http_header_error; } return nread; } p += nread; nconsumed += (size_t)nread; rstate->left -= nread; if (stream->flags & NGHTTP3_STREAM_FLAG_QPACK_DECODE_BLOCKED) { if (p != end && nghttp3_stream_get_buffered_datalen(stream) == 0) { rv = nghttp3_stream_buffer_data(stream, p, (size_t)(end - p)); if (rv != 0) { return rv; } } *pnproc = (size_t)(p - src); return (nghttp3_ssize)nconsumed; } if (rstate->left) { goto almost_done; } switch (stream->rx.hstate) { case NGHTTP3_HTTP_STATE_REQ_HEADERS_BEGIN: rv = nghttp3_http_on_request_headers(&stream->rx.http); break; case NGHTTP3_HTTP_STATE_RESP_HEADERS_BEGIN: rv = nghttp3_http_on_response_headers(&stream->rx.http); break; case NGHTTP3_HTTP_STATE_REQ_TRAILERS_BEGIN: case NGHTTP3_HTTP_STATE_RESP_TRAILERS_BEGIN: rv = 0; break; default: nghttp3_unreachable(); } if (rv != 0) { if (rv == NGHTTP3_ERR_MALFORMED_HTTP_HEADER) { goto http_header_error; } return rv; } switch (stream->rx.hstate) { case NGHTTP3_HTTP_STATE_REQ_HEADERS_BEGIN: /* Only server utilizes priority information to schedule streams. */ if (conn->server && (stream->rx.http.flags & NGHTTP3_HTTP_FLAG_PRIORITY) && !(stream->flags & NGHTTP3_STREAM_FLAG_PRIORITY_UPDATE_RECVED) && !(stream->flags & NGHTTP3_STREAM_FLAG_SERVER_PRIORITY_SET)) { rv = conn_update_stream_priority(conn, stream, stream->rx.http.pri); if (rv != 0) { return rv; } } /* fall through */ case NGHTTP3_HTTP_STATE_RESP_HEADERS_BEGIN: rv = conn_call_end_headers(conn, stream, p == end && fin); break; case NGHTTP3_HTTP_STATE_REQ_TRAILERS_BEGIN: case NGHTTP3_HTTP_STATE_RESP_TRAILERS_BEGIN: rv = conn_call_end_trailers(conn, stream, p == end && fin); break; default: nghttp3_unreachable(); } if (rv != 0) { return rv; } rv = nghttp3_stream_transit_rx_http_state(stream, NGHTTP3_HTTP_EVENT_HEADERS_END); assert(0 == rv); nghttp3_stream_read_state_reset(rstate); break; http_header_error: stream->flags |= NGHTTP3_STREAM_FLAG_HTTP_ERROR; busy = 1; rstate->state = NGHTTP3_REQ_STREAM_STATE_IGN_REST; rv = conn_call_stop_sending(conn, stream, NGHTTP3_H3_MESSAGE_ERROR); if (rv != 0) { return rv; } rv = conn_call_reset_stream(conn, stream, NGHTTP3_H3_MESSAGE_ERROR); if (rv != 0) { return rv; } break; case NGHTTP3_REQ_STREAM_STATE_IGN_FRAME: len = (size_t)nghttp3_min(rstate->left, (int64_t)(end - p)); p += len; nconsumed += len; rstate->left -= (int64_t)len; if (rstate->left) { goto almost_done; } nghttp3_stream_read_state_reset(rstate); break; case NGHTTP3_REQ_STREAM_STATE_IGN_REST: nconsumed += (size_t)(end - p); *pnproc = (size_t)(end - src); return (nghttp3_ssize)nconsumed; } } almost_done: if (fin) { switch (rstate->state) { case NGHTTP3_REQ_STREAM_STATE_FRAME_TYPE: if (rvint->left) { return NGHTTP3_ERR_H3_GENERAL_PROTOCOL_ERROR; } rv = nghttp3_stream_transit_rx_http_state(stream, NGHTTP3_HTTP_EVENT_MSG_END); if (rv != 0) { return rv; } rv = conn_call_end_stream(conn, stream); if (rv != 0) { return rv; } break; case NGHTTP3_REQ_STREAM_STATE_IGN_REST: break; default: return NGHTTP3_ERR_H3_FRAME_ERROR; } } *pnproc = (size_t)(p - src); return (nghttp3_ssize)nconsumed; } int nghttp3_conn_on_data(nghttp3_conn *conn, nghttp3_stream *stream, const uint8_t *data, size_t datalen) { int rv; rv = nghttp3_http_on_data_chunk(stream, datalen); if (rv != 0) { return rv; } if (!conn->callbacks.recv_data) { return 0; } rv = conn->callbacks.recv_data(conn, stream->node.id, data, datalen, conn->user_data, stream->user_data); if (rv != 0) { return NGHTTP3_ERR_CALLBACK_FAILURE; } return 0; } static nghttp3_pq *conn_get_sched_pq(nghttp3_conn *conn, nghttp3_tnode *tnode) { uint32_t urgency = nghttp3_pri_uint8_urgency(tnode->pri); assert(urgency < NGHTTP3_URGENCY_LEVELS); return &conn->sched[urgency].spq; } static nghttp3_ssize conn_decode_headers(nghttp3_conn *conn, nghttp3_stream *stream, const uint8_t *src, size_t srclen, int fin) { nghttp3_ssize nread; int rv; nghttp3_qpack_decoder *qdec = &conn->qdec; nghttp3_qpack_nv nv; uint8_t flags; nghttp3_buf buf; nghttp3_recv_header recv_header = NULL; nghttp3_http_state *http; int request = 0; int trailers = 0; switch (stream->rx.hstate) { case NGHTTP3_HTTP_STATE_REQ_HEADERS_BEGIN: request = 1; /* Fall through */ case NGHTTP3_HTTP_STATE_RESP_HEADERS_BEGIN: recv_header = conn->callbacks.recv_header; break; case NGHTTP3_HTTP_STATE_REQ_TRAILERS_BEGIN: request = 1; /* Fall through */ case NGHTTP3_HTTP_STATE_RESP_TRAILERS_BEGIN: trailers = 1; recv_header = conn->callbacks.recv_trailer; break; default: nghttp3_unreachable(); } http = &stream->rx.http; nghttp3_buf_wrap_init(&buf, (uint8_t *)src, srclen); buf.last = buf.end; for (;;) { nread = nghttp3_qpack_decoder_read_request(qdec, &stream->qpack_sctx, &nv, &flags, buf.pos, nghttp3_buf_len(&buf), fin); if (nread < 0) { return (int)nread; } buf.pos += nread; if (flags & NGHTTP3_QPACK_DECODE_FLAG_BLOCKED) { if (conn->local.settings.qpack_blocked_streams <= nghttp3_pq_size(&conn->qpack_blocked_streams)) { return NGHTTP3_ERR_QPACK_DECOMPRESSION_FAILED; } stream->flags |= NGHTTP3_STREAM_FLAG_QPACK_DECODE_BLOCKED; rv = nghttp3_conn_qpack_blocked_streams_push(conn, stream); if (rv != 0) { return rv; } break; } if (flags & NGHTTP3_QPACK_DECODE_FLAG_FINAL) { nghttp3_qpack_stream_context_reset(&stream->qpack_sctx); break; } if (nread == 0) { break; } if (flags & NGHTTP3_QPACK_DECODE_FLAG_EMIT) { rv = nghttp3_http_on_header( http, &nv, request, trailers, conn->server && conn->local.settings.enable_connect_protocol); switch (rv) { case NGHTTP3_ERR_MALFORMED_HTTP_HEADER: break; case NGHTTP3_ERR_REMOVE_HTTP_HEADER: rv = 0; break; case 0: if (recv_header) { rv = recv_header(conn, stream->node.id, nv.token, nv.name, nv.value, nv.flags, conn->user_data, stream->user_data); if (rv != 0) { rv = NGHTTP3_ERR_CALLBACK_FAILURE; } } break; default: nghttp3_unreachable(); } nghttp3_rcbuf_decref(nv.name); nghttp3_rcbuf_decref(nv.value); if (rv != 0) { return rv; } } } return buf.pos - src; } nghttp3_ssize nghttp3_conn_on_headers(nghttp3_conn *conn, nghttp3_stream *stream, const uint8_t *src, size_t srclen, int fin) { if (srclen == 0 && !fin) { return 0; } return conn_decode_headers(conn, stream, src, srclen, fin); } int nghttp3_conn_on_settings_entry_received(nghttp3_conn *conn, const nghttp3_frame_settings *fr) { const nghttp3_settings_entry *ent = &fr->iv[0]; nghttp3_settings *dest = &conn->remote.settings; /* TODO Check for duplicates */ switch (ent->id) { case NGHTTP3_SETTINGS_ID_MAX_FIELD_SECTION_SIZE: dest->max_field_section_size = ent->value; break; case NGHTTP3_SETTINGS_ID_QPACK_MAX_TABLE_CAPACITY: if (dest->qpack_max_dtable_capacity != 0) { return NGHTTP3_ERR_H3_SETTINGS_ERROR; } if (ent->value == 0) { break; } dest->qpack_max_dtable_capacity = (size_t)ent->value; nghttp3_qpack_encoder_set_max_dtable_capacity(&conn->qenc, (size_t)ent->value); break; case NGHTTP3_SETTINGS_ID_QPACK_BLOCKED_STREAMS: if (dest->qpack_blocked_streams != 0) { return NGHTTP3_ERR_H3_SETTINGS_ERROR; } if (ent->value == 0) { break; } dest->qpack_blocked_streams = (size_t)ent->value; nghttp3_qpack_encoder_set_max_blocked_streams( &conn->qenc, (size_t)nghttp3_min(100, ent->value)); break; case NGHTTP3_SETTINGS_ID_ENABLE_CONNECT_PROTOCOL: if (!conn->server) { break; } switch (ent->value) { case 0: if (dest->enable_connect_protocol) { return NGHTTP3_ERR_H3_SETTINGS_ERROR; } break; case 1: break; default: return NGHTTP3_ERR_H3_SETTINGS_ERROR; } dest->enable_connect_protocol = (int)ent->value; break; case NGHTTP3_H2_SETTINGS_ID_ENABLE_PUSH: case NGHTTP3_H2_SETTINGS_ID_MAX_CONCURRENT_STREAMS: case NGHTTP3_H2_SETTINGS_ID_INITIAL_WINDOW_SIZE: case NGHTTP3_H2_SETTINGS_ID_MAX_FRAME_SIZE: return NGHTTP3_ERR_H3_SETTINGS_ERROR; default: /* Ignore unknown settings ID */ break; } return 0; } static int conn_on_priority_update_stream(nghttp3_conn *conn, const nghttp3_frame_priority_update *fr) { int64_t stream_id = fr->pri_elem_id; nghttp3_stream *stream; int rv; if (!nghttp3_client_stream_bidi(stream_id) || nghttp3_ord_stream_id(stream_id) > conn->remote.bidi.max_client_streams) { return NGHTTP3_ERR_H3_ID_ERROR; } stream = nghttp3_conn_find_stream(conn, stream_id); if (stream == NULL) { if ((conn->flags & NGHTTP3_CONN_FLAG_GOAWAY_QUEUED) && conn->tx.goaway_id <= stream_id) { /* Connection is going down. Ignore priority signal. */ return 0; } rv = conn_bidi_idtr_open(conn, stream_id); if (rv != 0) { if (nghttp3_err_is_fatal(rv)) { return rv; } assert(rv == NGHTTP3_ERR_STREAM_IN_USE); /* The stream is gone. Just ignore. */ return 0; } conn->rx.max_stream_id_bidi = nghttp3_max(conn->rx.max_stream_id_bidi, stream_id); rv = nghttp3_conn_create_stream(conn, &stream, stream_id); if (rv != 0) { return rv; } stream->node.pri = nghttp3_pri_to_uint8(&fr->pri); stream->flags |= NGHTTP3_STREAM_FLAG_PRIORITY_UPDATE_RECVED; return 0; } if (stream->flags & NGHTTP3_STREAM_FLAG_SERVER_PRIORITY_SET) { return 0; } stream->flags |= NGHTTP3_STREAM_FLAG_PRIORITY_UPDATE_RECVED; return conn_update_stream_priority(conn, stream, nghttp3_pri_to_uint8(&fr->pri)); } int nghttp3_conn_on_priority_update(nghttp3_conn *conn, const nghttp3_frame_priority_update *fr) { assert(conn->server); assert(fr->hd.type == NGHTTP3_FRAME_PRIORITY_UPDATE); return conn_on_priority_update_stream(conn, fr); } static int conn_stream_acked_data(nghttp3_stream *stream, int64_t stream_id, uint64_t datalen, void *user_data) { nghttp3_conn *conn = stream->conn; int rv; if (!conn->callbacks.acked_stream_data) { return 0; } rv = conn->callbacks.acked_stream_data(conn, stream_id, datalen, conn->user_data, user_data); if (rv != 0) { return NGHTTP3_ERR_CALLBACK_FAILURE; } return 0; } int nghttp3_conn_create_stream(nghttp3_conn *conn, nghttp3_stream **pstream, int64_t stream_id) { nghttp3_stream *stream; int rv; nghttp3_stream_callbacks callbacks = { conn_stream_acked_data, }; rv = nghttp3_stream_new(&stream, stream_id, &callbacks, &conn->out_chunk_objalloc, &conn->stream_objalloc, conn->mem); if (rv != 0) { return rv; } stream->conn = conn; rv = nghttp3_map_insert(&conn->streams, (nghttp3_map_key_type)stream->node.id, stream); if (rv != 0) { nghttp3_stream_del(stream); return rv; } *pstream = stream; return 0; } nghttp3_stream *nghttp3_conn_find_stream(nghttp3_conn *conn, int64_t stream_id) { return nghttp3_map_find(&conn->streams, (nghttp3_map_key_type)stream_id); } int nghttp3_conn_bind_control_stream(nghttp3_conn *conn, int64_t stream_id) { nghttp3_stream *stream; nghttp3_frame_entry frent; int rv; assert(!conn->server || nghttp3_server_stream_uni(stream_id)); assert(conn->server || nghttp3_client_stream_uni(stream_id)); if (conn->tx.ctrl) { return NGHTTP3_ERR_INVALID_STATE; } rv = nghttp3_conn_create_stream(conn, &stream, stream_id); if (rv != 0) { return rv; } stream->type = NGHTTP3_STREAM_TYPE_CONTROL; conn->tx.ctrl = stream; rv = nghttp3_stream_write_stream_type(stream); if (rv != 0) { return rv; } frent.fr.hd.type = NGHTTP3_FRAME_SETTINGS; frent.aux.settings.local_settings = &conn->local.settings; return nghttp3_stream_frq_add(stream, &frent); } int nghttp3_conn_bind_qpack_streams(nghttp3_conn *conn, int64_t qenc_stream_id, int64_t qdec_stream_id) { nghttp3_stream *stream; int rv; assert(!conn->server || nghttp3_server_stream_uni(qenc_stream_id)); assert(!conn->server || nghttp3_server_stream_uni(qdec_stream_id)); assert(conn->server || nghttp3_client_stream_uni(qenc_stream_id)); assert(conn->server || nghttp3_client_stream_uni(qdec_stream_id)); if (conn->tx.qenc || conn->tx.qdec) { return NGHTTP3_ERR_INVALID_STATE; } rv = nghttp3_conn_create_stream(conn, &stream, qenc_stream_id); if (rv != 0) { return rv; } stream->type = NGHTTP3_STREAM_TYPE_QPACK_ENCODER; conn->tx.qenc = stream; rv = nghttp3_stream_write_stream_type(stream); if (rv != 0) { return rv; } rv = nghttp3_conn_create_stream(conn, &stream, qdec_stream_id); if (rv != 0) { return rv; } stream->type = NGHTTP3_STREAM_TYPE_QPACK_DECODER; conn->tx.qdec = stream; return nghttp3_stream_write_stream_type(stream); } static nghttp3_ssize conn_writev_stream(nghttp3_conn *conn, int64_t *pstream_id, int *pfin, nghttp3_vec *vec, size_t veccnt, nghttp3_stream *stream) { int rv; nghttp3_ssize n; assert(veccnt > 0); /* If stream is blocked by read callback, don't attempt to fill more. */ if (!(stream->flags & NGHTTP3_STREAM_FLAG_READ_DATA_BLOCKED)) { rv = nghttp3_stream_fill_outq(stream); if (rv != 0) { return rv; } } if (!nghttp3_stream_uni(stream->node.id) && conn->tx.qenc && !nghttp3_stream_is_blocked(conn->tx.qenc)) { n = nghttp3_stream_writev(conn->tx.qenc, pfin, vec, veccnt); if (n < 0) { return n; } if (n) { *pstream_id = conn->tx.qenc->node.id; return n; } } n = nghttp3_stream_writev(stream, pfin, vec, veccnt); if (n < 0) { return n; } /* We might just want to write stream fin without sending any stream data. */ if (n == 0 && *pfin == 0) { return 0; } *pstream_id = stream->node.id; return n; } nghttp3_ssize nghttp3_conn_writev_stream(nghttp3_conn *conn, int64_t *pstream_id, int *pfin, nghttp3_vec *vec, size_t veccnt) { nghttp3_ssize ncnt; nghttp3_stream *stream; int rv; *pstream_id = -1; *pfin = 0; if (veccnt == 0) { return 0; } if (conn->tx.ctrl && !nghttp3_stream_is_blocked(conn->tx.ctrl)) { ncnt = conn_writev_stream(conn, pstream_id, pfin, vec, veccnt, conn->tx.ctrl); if (ncnt) { return ncnt; } } if (conn->tx.qdec && !nghttp3_stream_is_blocked(conn->tx.qdec)) { rv = nghttp3_stream_write_qpack_decoder_stream(conn->tx.qdec); if (rv != 0) { return rv; } ncnt = conn_writev_stream(conn, pstream_id, pfin, vec, veccnt, conn->tx.qdec); if (ncnt) { return ncnt; } } if (conn->tx.qenc && !nghttp3_stream_is_blocked(conn->tx.qenc)) { ncnt = conn_writev_stream(conn, pstream_id, pfin, vec, veccnt, conn->tx.qenc); if (ncnt) { return ncnt; } } stream = nghttp3_conn_get_next_tx_stream(conn); if (stream == NULL) { return 0; } ncnt = conn_writev_stream(conn, pstream_id, pfin, vec, veccnt, stream); if (ncnt < 0) { return ncnt; } if (nghttp3_client_stream_bidi(stream->node.id) && !nghttp3_stream_require_schedule(stream)) { nghttp3_conn_unschedule_stream(conn, stream); } return ncnt; } nghttp3_stream *nghttp3_conn_get_next_tx_stream(nghttp3_conn *conn) { size_t i; nghttp3_tnode *tnode; nghttp3_pq *pq; for (i = 0; i < NGHTTP3_URGENCY_LEVELS; ++i) { pq = &conn->sched[i].spq; if (nghttp3_pq_empty(pq)) { continue; } tnode = nghttp3_struct_of(nghttp3_pq_top(pq), nghttp3_tnode, pe); return nghttp3_struct_of(tnode, nghttp3_stream, node); } return NULL; } int nghttp3_conn_add_write_offset(nghttp3_conn *conn, int64_t stream_id, size_t n) { nghttp3_stream *stream = nghttp3_conn_find_stream(conn, stream_id); int rv; if (stream == NULL) { return 0; } rv = nghttp3_stream_add_outq_offset(stream, n); if (rv != 0) { return rv; } stream->unscheduled_nwrite += n; if (!nghttp3_client_stream_bidi(stream->node.id)) { return 0; } if (!nghttp3_stream_require_schedule(stream)) { nghttp3_conn_unschedule_stream(conn, stream); return 0; } if (stream->unscheduled_nwrite < NGHTTP3_STREAM_MIN_WRITELEN) { return 0; } return nghttp3_conn_schedule_stream(conn, stream); } int nghttp3_conn_add_ack_offset(nghttp3_conn *conn, int64_t stream_id, uint64_t n) { nghttp3_stream *stream = nghttp3_conn_find_stream(conn, stream_id); if (stream == NULL) { return 0; } return nghttp3_stream_add_ack_offset(stream, n); } static int conn_submit_headers_data(nghttp3_conn *conn, nghttp3_stream *stream, const nghttp3_nv *nva, size_t nvlen, const nghttp3_data_reader *dr) { int rv; nghttp3_nv *nnva; nghttp3_frame_entry frent; rv = nghttp3_nva_copy(&nnva, nva, nvlen, conn->mem); if (rv != 0) { return rv; } frent.fr.hd.type = NGHTTP3_FRAME_HEADERS; frent.fr.headers.nva = nnva; frent.fr.headers.nvlen = nvlen; rv = nghttp3_stream_frq_add(stream, &frent); if (rv != 0) { nghttp3_nva_del(nnva, conn->mem); return rv; } if (dr) { frent.fr.hd.type = NGHTTP3_FRAME_DATA; frent.aux.data.dr = *dr; rv = nghttp3_stream_frq_add(stream, &frent); if (rv != 0) { return rv; } } if (nghttp3_stream_require_schedule(stream)) { return nghttp3_conn_schedule_stream(conn, stream); } return 0; } int nghttp3_conn_schedule_stream(nghttp3_conn *conn, nghttp3_stream *stream) { /* Assume that stream stays on the same urgency level */ nghttp3_tnode *node = stream_get_sched_node(stream); int rv; rv = nghttp3_tnode_schedule(node, conn_get_sched_pq(conn, node), stream->unscheduled_nwrite); if (rv != 0) { return rv; } stream->unscheduled_nwrite = 0; return 0; } int nghttp3_conn_ensure_stream_scheduled(nghttp3_conn *conn, nghttp3_stream *stream) { if (nghttp3_tnode_is_scheduled(stream_get_sched_node(stream))) { return 0; } return nghttp3_conn_schedule_stream(conn, stream); } void nghttp3_conn_unschedule_stream(nghttp3_conn *conn, nghttp3_stream *stream) { nghttp3_tnode *node = stream_get_sched_node(stream); nghttp3_tnode_unschedule(node, conn_get_sched_pq(conn, node)); } int nghttp3_conn_submit_request(nghttp3_conn *conn, int64_t stream_id, const nghttp3_nv *nva, size_t nvlen, const nghttp3_data_reader *dr, void *stream_user_data) { nghttp3_stream *stream; int rv; assert(!conn->server); assert(conn->tx.qenc); assert(nghttp3_client_stream_bidi(stream_id)); /* TODO Should we check that stream_id is client stream_id? */ /* TODO Check GOAWAY last stream ID */ if (nghttp3_stream_uni(stream_id)) { return NGHTTP3_ERR_INVALID_ARGUMENT; } if (conn->flags & NGHTTP3_CONN_FLAG_GOAWAY_RECVED) { return NGHTTP3_ERR_CONN_CLOSING; } stream = nghttp3_conn_find_stream(conn, stream_id); if (stream != NULL) { return NGHTTP3_ERR_STREAM_IN_USE; } rv = nghttp3_conn_create_stream(conn, &stream, stream_id); if (rv != 0) { return rv; } stream->rx.hstate = NGHTTP3_HTTP_STATE_RESP_INITIAL; stream->tx.hstate = NGHTTP3_HTTP_STATE_REQ_END; stream->user_data = stream_user_data; nghttp3_http_record_request_method(stream, nva, nvlen); if (dr == NULL) { stream->flags |= NGHTTP3_STREAM_FLAG_WRITE_END_STREAM; } return conn_submit_headers_data(conn, stream, nva, nvlen, dr); } int nghttp3_conn_submit_info(nghttp3_conn *conn, int64_t stream_id, const nghttp3_nv *nva, size_t nvlen) { nghttp3_stream *stream; /* TODO Verify that it is allowed to send info (non-final response) now. */ assert(conn->server); assert(conn->tx.qenc); stream = nghttp3_conn_find_stream(conn, stream_id); if (stream == NULL) { return NGHTTP3_ERR_STREAM_NOT_FOUND; } return conn_submit_headers_data(conn, stream, nva, nvlen, NULL); } int nghttp3_conn_submit_response(nghttp3_conn *conn, int64_t stream_id, const nghttp3_nv *nva, size_t nvlen, const nghttp3_data_reader *dr) { nghttp3_stream *stream; /* TODO Verify that it is allowed to send response now. */ assert(conn->server); assert(conn->tx.qenc); stream = nghttp3_conn_find_stream(conn, stream_id); if (stream == NULL) { return NGHTTP3_ERR_STREAM_NOT_FOUND; } if (dr == NULL) { stream->flags |= NGHTTP3_STREAM_FLAG_WRITE_END_STREAM; } return conn_submit_headers_data(conn, stream, nva, nvlen, dr); } int nghttp3_conn_submit_trailers(nghttp3_conn *conn, int64_t stream_id, const nghttp3_nv *nva, size_t nvlen) { nghttp3_stream *stream; /* TODO Verify that it is allowed to send trailer now. */ assert(conn->tx.qenc); stream = nghttp3_conn_find_stream(conn, stream_id); if (stream == NULL) { return NGHTTP3_ERR_STREAM_NOT_FOUND; } if (stream->flags & NGHTTP3_STREAM_FLAG_WRITE_END_STREAM) { return NGHTTP3_ERR_INVALID_STATE; } stream->flags |= NGHTTP3_STREAM_FLAG_WRITE_END_STREAM; return conn_submit_headers_data(conn, stream, nva, nvlen, NULL); } int nghttp3_conn_submit_shutdown_notice(nghttp3_conn *conn) { nghttp3_frame_entry frent; int rv; assert(conn->tx.ctrl); frent.fr.hd.type = NGHTTP3_FRAME_GOAWAY; frent.fr.goaway.id = conn->server ? NGHTTP3_SHUTDOWN_NOTICE_STREAM_ID : NGHTTP3_SHUTDOWN_NOTICE_PUSH_ID; assert(frent.fr.goaway.id <= conn->tx.goaway_id); rv = nghttp3_stream_frq_add(conn->tx.ctrl, &frent); if (rv != 0) { return rv; } conn->tx.goaway_id = frent.fr.goaway.id; conn->flags |= NGHTTP3_CONN_FLAG_GOAWAY_QUEUED; return 0; } int nghttp3_conn_shutdown(nghttp3_conn *conn) { nghttp3_frame_entry frent; int rv; assert(conn->tx.ctrl); frent.fr.hd.type = NGHTTP3_FRAME_GOAWAY; if (conn->server) { frent.fr.goaway.id = nghttp3_min((1ll << 62) - 4, conn->rx.max_stream_id_bidi + 4); } else { frent.fr.goaway.id = 0; } assert(frent.fr.goaway.id <= conn->tx.goaway_id); rv = nghttp3_stream_frq_add(conn->tx.ctrl, &frent); if (rv != 0) { return rv; } conn->tx.goaway_id = frent.fr.goaway.id; conn->flags |= NGHTTP3_CONN_FLAG_GOAWAY_QUEUED; return 0; } int nghttp3_conn_reject_stream(nghttp3_conn *conn, nghttp3_stream *stream) { int rv; rv = conn_call_stop_sending(conn, stream, NGHTTP3_H3_REQUEST_REJECTED); if (rv != 0) { return rv; } return conn_call_reset_stream(conn, stream, NGHTTP3_H3_REQUEST_REJECTED); } void nghttp3_conn_block_stream(nghttp3_conn *conn, int64_t stream_id) { nghttp3_stream *stream = nghttp3_conn_find_stream(conn, stream_id); if (stream == NULL) { return; } stream->flags |= NGHTTP3_STREAM_FLAG_FC_BLOCKED; stream->unscheduled_nwrite = 0; if (nghttp3_client_stream_bidi(stream->node.id)) { nghttp3_conn_unschedule_stream(conn, stream); } } void nghttp3_conn_shutdown_stream_write(nghttp3_conn *conn, int64_t stream_id) { nghttp3_stream *stream = nghttp3_conn_find_stream(conn, stream_id); if (stream == NULL) { return; } stream->flags |= NGHTTP3_STREAM_FLAG_SHUT_WR; stream->unscheduled_nwrite = 0; if (nghttp3_client_stream_bidi(stream->node.id)) { nghttp3_conn_unschedule_stream(conn, stream); } } int nghttp3_conn_unblock_stream(nghttp3_conn *conn, int64_t stream_id) { nghttp3_stream *stream = nghttp3_conn_find_stream(conn, stream_id); if (stream == NULL) { return 0; } stream->flags &= (uint16_t)~NGHTTP3_STREAM_FLAG_FC_BLOCKED; if (nghttp3_client_stream_bidi(stream->node.id) && nghttp3_stream_require_schedule(stream)) { return nghttp3_conn_ensure_stream_scheduled(conn, stream); } return 0; } int nghttp3_conn_is_stream_writable(nghttp3_conn *conn, int64_t stream_id) { nghttp3_stream *stream = nghttp3_conn_find_stream(conn, stream_id); if (stream == NULL) { return 0; } return (stream->flags & (NGHTTP3_STREAM_FLAG_FC_BLOCKED | NGHTTP3_STREAM_FLAG_READ_DATA_BLOCKED | NGHTTP3_STREAM_FLAG_SHUT_WR | NGHTTP3_STREAM_FLAG_CLOSED)) == 0; } int nghttp3_conn_resume_stream(nghttp3_conn *conn, int64_t stream_id) { nghttp3_stream *stream = nghttp3_conn_find_stream(conn, stream_id); if (stream == NULL) { return 0; } stream->flags &= (uint16_t)~NGHTTP3_STREAM_FLAG_READ_DATA_BLOCKED; if (nghttp3_client_stream_bidi(stream->node.id) && nghttp3_stream_require_schedule(stream)) { return nghttp3_conn_ensure_stream_scheduled(conn, stream); } return 0; } int nghttp3_conn_close_stream(nghttp3_conn *conn, int64_t stream_id, uint64_t app_error_code) { nghttp3_stream *stream = nghttp3_conn_find_stream(conn, stream_id); if (stream == NULL) { return NGHTTP3_ERR_STREAM_NOT_FOUND; } if (nghttp3_stream_uni(stream_id) && stream->type != NGHTTP3_STREAM_TYPE_UNKNOWN) { return NGHTTP3_ERR_H3_CLOSED_CRITICAL_STREAM; } stream->error_code = app_error_code; nghttp3_conn_unschedule_stream(conn, stream); if (stream->qpack_blocked_pe.index == NGHTTP3_PQ_BAD_INDEX) { return conn_delete_stream(conn, stream); } stream->flags |= NGHTTP3_STREAM_FLAG_CLOSED; return 0; } int nghttp3_conn_shutdown_stream_read(nghttp3_conn *conn, int64_t stream_id) { nghttp3_stream *stream; if (!nghttp3_client_stream_bidi(stream_id)) { return 0; } stream = nghttp3_conn_find_stream(conn, stream_id); if (stream) { if (stream->flags & NGHTTP3_STREAM_FLAG_SHUT_RD) { return 0; } stream->flags |= NGHTTP3_STREAM_FLAG_SHUT_RD; } return nghttp3_qpack_decoder_cancel_stream(&conn->qdec, stream_id); } int nghttp3_conn_qpack_blocked_streams_push(nghttp3_conn *conn, nghttp3_stream *stream) { assert(stream->qpack_blocked_pe.index == NGHTTP3_PQ_BAD_INDEX); return nghttp3_pq_push(&conn->qpack_blocked_streams, &stream->qpack_blocked_pe); } void nghttp3_conn_qpack_blocked_streams_pop(nghttp3_conn *conn) { assert(!nghttp3_pq_empty(&conn->qpack_blocked_streams)); nghttp3_pq_pop(&conn->qpack_blocked_streams); } void nghttp3_conn_set_max_client_streams_bidi(nghttp3_conn *conn, uint64_t max_streams) { assert(conn->server); assert(conn->remote.bidi.max_client_streams <= max_streams); conn->remote.bidi.max_client_streams = max_streams; } void nghttp3_conn_set_max_concurrent_streams(nghttp3_conn *conn, size_t max_concurrent_streams) { nghttp3_qpack_decoder_set_max_concurrent_streams(&conn->qdec, max_concurrent_streams); } int nghttp3_conn_set_stream_user_data(nghttp3_conn *conn, int64_t stream_id, void *stream_user_data) { nghttp3_stream *stream = nghttp3_conn_find_stream(conn, stream_id); if (stream == NULL) { return NGHTTP3_ERR_STREAM_NOT_FOUND; } stream->user_data = stream_user_data; return 0; } uint64_t nghttp3_conn_get_frame_payload_left(nghttp3_conn *conn, int64_t stream_id) { nghttp3_stream *stream = nghttp3_conn_find_stream(conn, stream_id); if (stream == NULL) { return 0; } return (uint64_t)stream->rstate.left; } int nghttp3_conn_get_stream_priority(nghttp3_conn *conn, nghttp3_pri *dest, int64_t stream_id) { nghttp3_stream *stream; assert(conn->server); if (!nghttp3_client_stream_bidi(stream_id)) { return NGHTTP3_ERR_INVALID_ARGUMENT; } stream = nghttp3_conn_find_stream(conn, stream_id); if (stream == NULL) { return NGHTTP3_ERR_STREAM_NOT_FOUND; } dest->urgency = nghttp3_pri_uint8_urgency(stream->node.pri); dest->inc = nghttp3_pri_uint8_inc(stream->node.pri); return 0; } int nghttp3_conn_set_stream_priority(nghttp3_conn *conn, int64_t stream_id, const nghttp3_pri *pri) { nghttp3_stream *stream; nghttp3_frame_entry frent; assert(pri->urgency < NGHTTP3_URGENCY_LEVELS); assert(pri->inc == 0 || pri->inc == 1); if (!nghttp3_client_stream_bidi(stream_id)) { return NGHTTP3_ERR_INVALID_ARGUMENT; } stream = nghttp3_conn_find_stream(conn, stream_id); if (stream == NULL) { return NGHTTP3_ERR_STREAM_NOT_FOUND; } if (conn->server) { stream->flags |= NGHTTP3_STREAM_FLAG_SERVER_PRIORITY_SET; return conn_update_stream_priority(conn, stream, nghttp3_pri_to_uint8(pri)); } frent.fr.hd.type = NGHTTP3_FRAME_PRIORITY_UPDATE; frent.fr.priority_update.pri_elem_id = stream_id; frent.fr.priority_update.pri = *pri; return nghttp3_stream_frq_add(conn->tx.ctrl, &frent); } int nghttp3_conn_is_remote_qpack_encoder_stream(nghttp3_conn *conn, int64_t stream_id) { nghttp3_stream *stream; if (!conn_remote_stream_uni(conn, stream_id)) { return 0; } stream = nghttp3_conn_find_stream(conn, stream_id); return stream && stream->type == NGHTTP3_STREAM_TYPE_QPACK_ENCODER; } void nghttp3_settings_default_versioned(int settings_version, nghttp3_settings *settings) { (void)settings_version; memset(settings, 0, sizeof(nghttp3_settings)); settings->max_field_section_size = NGHTTP3_VARINT_MAX; settings->qpack_encoder_max_dtable_capacity = NGHTTP3_QPACK_ENCODER_MAX_DTABLE_CAPACITY; }