/* * 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. */ #ifndef NGHTTP3_STREAM_H #define NGHTTP3_STREAM_H #ifdef HAVE_CONFIG_H # include #endif /* HAVE_CONFIG_H */ #include #include "nghttp3_map.h" #include "nghttp3_tnode.h" #include "nghttp3_ringbuf.h" #include "nghttp3_buf.h" #include "nghttp3_frame.h" #include "nghttp3_qpack.h" #include "nghttp3_objalloc.h" #define NGHTTP3_STREAM_MIN_CHUNK_SIZE 256 /* NGHTTP3_MIN_UNSENT_BYTES is the minimum unsent bytes which is large enough to fill outgoing single QUIC packet. */ #define NGHTTP3_MIN_UNSENT_BYTES 4096 /* NGHTTP3_STREAM_MIN_WRITELEN is the minimum length of write to cause the stream to reschedule. */ #define NGHTTP3_STREAM_MIN_WRITELEN 800 /* nghttp3_stream_type is unidirectional stream type. */ typedef uint64_t nghttp3_stream_type; #define NGHTTP3_STREAM_TYPE_CONTROL 0x00 #define NGHTTP3_STREAM_TYPE_PUSH 0x01 #define NGHTTP3_STREAM_TYPE_QPACK_ENCODER 0x02 #define NGHTTP3_STREAM_TYPE_QPACK_DECODER 0x03 #define NGHTTP3_STREAM_TYPE_UNKNOWN UINT64_MAX typedef enum nghttp3_ctrl_stream_state { NGHTTP3_CTRL_STREAM_STATE_FRAME_TYPE, NGHTTP3_CTRL_STREAM_STATE_FRAME_LENGTH, NGHTTP3_CTRL_STREAM_STATE_SETTINGS, NGHTTP3_CTRL_STREAM_STATE_GOAWAY, NGHTTP3_CTRL_STREAM_STATE_MAX_PUSH_ID, NGHTTP3_CTRL_STREAM_STATE_IGN_FRAME, NGHTTP3_CTRL_STREAM_STATE_SETTINGS_ID, NGHTTP3_CTRL_STREAM_STATE_SETTINGS_VALUE, NGHTTP3_CTRL_STREAM_STATE_PRIORITY_UPDATE_PRI_ELEM_ID, NGHTTP3_CTRL_STREAM_STATE_PRIORITY_UPDATE, } nghttp3_ctrl_stream_state; typedef enum nghttp3_req_stream_state { NGHTTP3_REQ_STREAM_STATE_FRAME_TYPE, NGHTTP3_REQ_STREAM_STATE_FRAME_LENGTH, NGHTTP3_REQ_STREAM_STATE_DATA, NGHTTP3_REQ_STREAM_STATE_HEADERS, NGHTTP3_REQ_STREAM_STATE_IGN_FRAME, NGHTTP3_REQ_STREAM_STATE_IGN_REST, } nghttp3_req_stream_state; typedef struct nghttp3_varint_read_state { int64_t acc; size_t left; } nghttp3_varint_read_state; typedef struct nghttp3_stream_read_state { nghttp3_varint_read_state rvint; nghttp3_frame fr; int state; int64_t left; } nghttp3_stream_read_state; /* NGHTTP3_STREAM_FLAG_NONE indicates that no flag is set. */ #define NGHTTP3_STREAM_FLAG_NONE 0x0000u /* NGHTTP3_STREAM_FLAG_TYPE_IDENTIFIED is set when a unidirectional stream type is identified. */ #define NGHTTP3_STREAM_FLAG_TYPE_IDENTIFIED 0x0001u /* NGHTTP3_STREAM_FLAG_FC_BLOCKED indicates that stream is blocked by QUIC flow control. */ #define NGHTTP3_STREAM_FLAG_FC_BLOCKED 0x0002u /* NGHTTP3_STREAM_FLAG_READ_DATA_BLOCKED indicates that application is temporarily unable to provide data. */ #define NGHTTP3_STREAM_FLAG_READ_DATA_BLOCKED 0x0004u /* NGHTTP3_STREAM_FLAG_WRITE_END_STREAM indicates that application finished to feed outgoing data. */ #define NGHTTP3_STREAM_FLAG_WRITE_END_STREAM 0x0008u /* NGHTTP3_STREAM_FLAG_QPACK_DECODE_BLOCKED indicates that stream is blocked due to QPACK decoding. */ #define NGHTTP3_STREAM_FLAG_QPACK_DECODE_BLOCKED 0x0010u /* NGHTTP3_STREAM_FLAG_READ_EOF indicates that remote endpoint sent fin. */ #define NGHTTP3_STREAM_FLAG_READ_EOF 0x0020u /* NGHTTP3_STREAM_FLAG_CLOSED indicates that QUIC stream was closed. nghttp3_stream object can still alive because it might be blocked by QPACK decoder. */ #define NGHTTP3_STREAM_FLAG_CLOSED 0x0040u /* NGHTTP3_STREAM_FLAG_SHUT_WR indicates that any further write operation to a stream is prohibited. */ #define NGHTTP3_STREAM_FLAG_SHUT_WR 0x0100u /* NGHTTP3_STREAM_FLAG_SHUT_RD indicates that a read-side stream is closed abruptly and any incoming and pending stream data is just discarded for a stream. */ #define NGHTTP3_STREAM_FLAG_SHUT_RD 0x0200u /* NGHTTP3_STREAM_FLAG_SERVER_PRIORITY_SET indicates that server overrides stream priority. */ #define NGHTTP3_STREAM_FLAG_SERVER_PRIORITY_SET 0x0400u /* NGHTTP3_STREAM_FLAG_PRIORITY_UPDATE_RECVED indicates that server received PRIORITY_UPDATE frame for this stream. */ #define NGHTTP3_STREAM_FLAG_PRIORITY_UPDATE_RECVED 0x0800u /* NGHTTP3_STREAM_FLAG_HTTP_ERROR indicates that NGHTTP3_ERR_MALFORMED_HTTP_HEADER error is encountered while processing incoming HTTP fields. */ #define NGHTTP3_STREAM_FLAG_HTTP_ERROR 0x1000u typedef enum nghttp3_stream_http_state { NGHTTP3_HTTP_STATE_NONE, NGHTTP3_HTTP_STATE_REQ_INITIAL, NGHTTP3_HTTP_STATE_REQ_BEGIN, NGHTTP3_HTTP_STATE_REQ_HEADERS_BEGIN, NGHTTP3_HTTP_STATE_REQ_HEADERS_END, NGHTTP3_HTTP_STATE_REQ_DATA_BEGIN, NGHTTP3_HTTP_STATE_REQ_DATA_END, NGHTTP3_HTTP_STATE_REQ_TRAILERS_BEGIN, NGHTTP3_HTTP_STATE_REQ_TRAILERS_END, NGHTTP3_HTTP_STATE_REQ_END, NGHTTP3_HTTP_STATE_RESP_INITIAL, NGHTTP3_HTTP_STATE_RESP_BEGIN, NGHTTP3_HTTP_STATE_RESP_HEADERS_BEGIN, NGHTTP3_HTTP_STATE_RESP_HEADERS_END, NGHTTP3_HTTP_STATE_RESP_DATA_BEGIN, NGHTTP3_HTTP_STATE_RESP_DATA_END, NGHTTP3_HTTP_STATE_RESP_TRAILERS_BEGIN, NGHTTP3_HTTP_STATE_RESP_TRAILERS_END, NGHTTP3_HTTP_STATE_RESP_END, } nghttp3_stream_http_state; typedef enum nghttp3_stream_http_event { NGHTTP3_HTTP_EVENT_DATA_BEGIN, NGHTTP3_HTTP_EVENT_DATA_END, NGHTTP3_HTTP_EVENT_HEADERS_BEGIN, NGHTTP3_HTTP_EVENT_HEADERS_END, NGHTTP3_HTTP_EVENT_MSG_END, } nghttp3_stream_http_event; typedef struct nghttp3_stream nghttp3_stream; /* * nghttp3_stream_acked_data is a callback function which is invoked * when data sent on stream denoted by |stream_id| supplied from * application is acknowledged by remote endpoint. The number of * bytes acknowledged is given in |datalen|. * * The implementation of this callback must return 0 if it succeeds. * Returning NGHTTP3_ERR_CALLBACK_FAILURE will return to the caller * immediately. Any values other than 0 is treated as * NGHTTP3_ERR_CALLBACK_FAILURE. */ typedef int (*nghttp3_stream_acked_data)(nghttp3_stream *stream, int64_t stream_id, uint64_t datalen, void *user_data); typedef struct nghttp3_stream_callbacks { nghttp3_stream_acked_data acked_data; } nghttp3_stream_callbacks; typedef struct nghttp3_http_state { /* status_code is HTTP status code received. This field is used if connection is initialized as client. */ int32_t status_code; /* content_length is the value of received content-length header field. */ int64_t content_length; /* recv_content_length is the number of body bytes received so far. */ int64_t recv_content_length; nghttp3_pri pri; uint32_t flags; } nghttp3_http_state; struct nghttp3_stream { union { struct { const nghttp3_mem *mem; nghttp3_objalloc *out_chunk_objalloc; nghttp3_objalloc *stream_objalloc; nghttp3_tnode node; nghttp3_pq_entry qpack_blocked_pe; nghttp3_stream_callbacks callbacks; nghttp3_ringbuf frq; nghttp3_ringbuf chunks; nghttp3_ringbuf outq; /* inq stores the stream raw data which cannot be read because stream is blocked by QPACK decoder. */ nghttp3_ringbuf inq; nghttp3_qpack_stream_context qpack_sctx; /* conn is a reference to underlying connection. It could be NULL if stream is not a request stream. */ nghttp3_conn *conn; void *user_data; /* unsent_bytes is the number of bytes in outq not written yet */ uint64_t unsent_bytes; /* outq_idx is an index into outq where next write is made. */ size_t outq_idx; /* outq_offset is write offset relative to the element at outq_idx in outq. */ uint64_t outq_offset; /* ack_offset is offset acknowledged by peer relative to the first element in outq. */ uint64_t ack_offset; /* ack_done is the number of bytes notified to an application that they are acknowledged inside the first outq element if it is of type NGHTTP3_BUF_TYPE_ALIEN. */ uint64_t ack_done; uint64_t unscheduled_nwrite; nghttp3_stream_type type; nghttp3_stream_read_state rstate; /* error_code indicates the reason of closure of this stream. */ uint64_t error_code; struct { uint64_t offset; nghttp3_stream_http_state hstate; } tx; struct { nghttp3_stream_http_state hstate; nghttp3_http_state http; } rx; uint16_t flags; }; nghttp3_opl_entry oplent; }; }; nghttp3_objalloc_decl(stream, nghttp3_stream, oplent); typedef struct nghttp3_frame_entry { nghttp3_frame fr; union { struct { nghttp3_settings *local_settings; } settings; struct { nghttp3_data_reader dr; } data; } aux; } nghttp3_frame_entry; int nghttp3_stream_new(nghttp3_stream **pstream, int64_t stream_id, const nghttp3_stream_callbacks *callbacks, nghttp3_objalloc *out_chunk_objalloc, nghttp3_objalloc *stream_objalloc, const nghttp3_mem *mem); void nghttp3_stream_del(nghttp3_stream *stream); void nghttp3_varint_read_state_reset(nghttp3_varint_read_state *rvint); void nghttp3_stream_read_state_reset(nghttp3_stream_read_state *rstate); nghttp3_ssize nghttp3_read_varint(nghttp3_varint_read_state *rvint, const uint8_t *src, size_t srclen, int fin); int nghttp3_stream_frq_add(nghttp3_stream *stream, const nghttp3_frame_entry *frent); int nghttp3_stream_fill_outq(nghttp3_stream *stream); int nghttp3_stream_write_stream_type(nghttp3_stream *stream); size_t nghttp3_stream_writev(nghttp3_stream *stream, int *pfin, nghttp3_vec *vec, size_t veccnt); int nghttp3_stream_write_qpack_decoder_stream(nghttp3_stream *stream); int nghttp3_stream_outq_add(nghttp3_stream *stream, const nghttp3_typed_buf *tbuf); int nghttp3_stream_write_headers(nghttp3_stream *stream, nghttp3_frame_entry *frent); int nghttp3_stream_write_header_block(nghttp3_stream *stream, nghttp3_qpack_encoder *qenc, nghttp3_stream *qenc_stream, nghttp3_buf *rbuf, nghttp3_buf *ebuf, int64_t frame_type, const nghttp3_nv *nva, size_t nvlen); int nghttp3_stream_write_data(nghttp3_stream *stream, int *peof, nghttp3_frame_entry *frent); int nghttp3_stream_write_settings(nghttp3_stream *stream, nghttp3_frame_entry *frent); int nghttp3_stream_write_goaway(nghttp3_stream *stream, nghttp3_frame_entry *frent); int nghttp3_stream_write_priority_update(nghttp3_stream *stream, nghttp3_frame_entry *frent); int nghttp3_stream_ensure_chunk(nghttp3_stream *stream, size_t need); nghttp3_buf *nghttp3_stream_get_chunk(nghttp3_stream *stream); int nghttp3_stream_is_blocked(nghttp3_stream *stream); void nghttp3_stream_add_outq_offset(nghttp3_stream *stream, size_t n); /* * nghttp3_stream_outq_write_done returns nonzero if all contents in * outq have been written. */ int nghttp3_stream_outq_write_done(nghttp3_stream *stream); int nghttp3_stream_add_ack_offset(nghttp3_stream *stream, uint64_t n); /* * nghttp3_stream_is_active returns nonzero if |stream| is active. In * other words, it has something to send. This function does not take * into account its descendants. */ int nghttp3_stream_is_active(nghttp3_stream *stream); /* * nghttp3_stream_require_schedule returns nonzero if |stream| should * be scheduled. In other words, |stream| or its descendants have * something to send. */ int nghttp3_stream_require_schedule(nghttp3_stream *stream); int nghttp3_stream_buffer_data(nghttp3_stream *stream, const uint8_t *src, size_t srclen); size_t nghttp3_stream_get_buffered_datalen(nghttp3_stream *stream); int nghttp3_stream_ensure_qpack_stream_context(nghttp3_stream *stream); void nghttp3_stream_delete_qpack_stream_context(nghttp3_stream *stream); int nghttp3_stream_transit_rx_http_state(nghttp3_stream *stream, nghttp3_stream_http_event event); int nghttp3_stream_empty_headers_allowed(nghttp3_stream *stream); /* * nghttp3_stream_uni returns nonzero if stream identified by * |stream_id| is unidirectional. */ int nghttp3_stream_uni(int64_t stream_id); /* * nghttp3_client_stream_bidi returns nonzero if stream identified by * |stream_id| is client initiated bidirectional stream. */ int nghttp3_client_stream_bidi(int64_t stream_id); /* * nghttp3_client_stream_uni returns nonzero if stream identified by * |stream_id| is client initiated unidirectional stream. */ int nghttp3_client_stream_uni(int64_t stream_id); /* * nghttp3_server_stream_uni returns nonzero if stream identified by * |stream_id| is server initiated unidirectional stream. */ int nghttp3_server_stream_uni(int64_t stream_id); #endif /* NGHTTP3_STREAM_H */