Line data Source code
1 : #ifndef HEADER_fd_quic_transport_params_h
2 : #define HEADER_fd_quic_transport_params_h
3 :
4 : #include "../../../util/fd_util_base.h"
5 : #include <stdio.h>
6 :
7 : // TODO set proper defaults, and delete DFT_UNKNOWN
8 : #define DFT_UNKNOWN 0
9 :
10 : //23456789.123456789.123456789.123456789.123456789.123456789.123456789.123456789.123456789.123456789.123456789
11 : //........1.........2.........3.........4.........5.........6.........7.........8.........9.........0.........
12 18348 : #define FD_QUIC_TRANSPORT_PARAMS(X, ...) \
13 18348 : X( original_destination_connection_id, \
14 6303 : 0x00, \
15 6303 : CONN_ID, \
16 6303 : DFT_UNKNOWN, \
17 6303 : "This parameter is the value of the Destination Connection ID field from the " \
18 6303 : "first Initial packet sent by the client; see Section 7.3. This transport " \
19 6303 : "parameter is only sent by a server.", \
20 24078 : __VA_ARGS__ ) \
21 24078 : X( max_idle_timeout, \
22 24072 : 0x01, \
23 24072 : VARINT, \
24 24072 : DFT_UNKNOWN, \
25 24072 : "The maximum idle timeout is a value in milliseconds that is encoded as an " \
26 24072 : "integer; see (Section 10.1). Idle timeout is disabled when both endpoints omit " \
27 24072 : "this transport parameter or specify a value of 0.", \
28 24072 : __VA_ARGS__ ) \
29 12045 : X( stateless_reset_token, \
30 12039 : 0x02, \
31 12039 : TOKEN, \
32 12039 : DFT_UNKNOWN, \
33 12039 : "A stateless reset token is used in verifying a stateless reset; see Section " \
34 12039 : "10.3. This parameter is a sequence of 16 bytes. This transport parameter MUST " \
35 12039 : "NOT be sent by a client but MAY be sent by a server. A server that does not send " \
36 12039 : "this transport parameter cannot use stateless reset (Section 10.3) for the " \
37 12039 : "connection ID negotiated during the handshake.", \
38 24075 : __VA_ARGS__ ) \
39 24075 : X( max_udp_payload_size, \
40 24069 : 0x03, \
41 24069 : VARINT, \
42 24069 : 65527, \
43 24069 : "The maximum UDP payload size parameter is an integer value that limits the size " \
44 24069 : "of UDP payloads that the endpoint is willing to receive. UDP datagrams with " \
45 24069 : "payloads larger than this limit are not likely to be processed by the receiver. " \
46 24069 : "The default for this parameter is the maximum permitted UDP payload of 65527. " \
47 24069 : "Values below 1200 are invalid.\n" \
48 24069 : "This limit does act as an additional constraint on datagram size in the same way " \
49 24069 : "as the path MTU, but it is a property of the endpoint and not the path; see " \
50 24069 : "Section 14. It is expected that this is the space an endpoint dedicates to " \
51 24069 : "holding incoming packets.", \
52 24078 : __VA_ARGS__ ) \
53 24078 : X( initial_max_data, \
54 24072 : 0x04, \
55 24072 : VARINT, \
56 24072 : DFT_UNKNOWN, \
57 24072 : "The initial maximum data parameter is an integer value that contains the initial " \
58 24072 : "value for the maximum amount of data that can be sent on the connection. This is " \
59 24072 : "equivalent to sending a MAX_DATA (Section 19.9) for the connection immediately " \
60 24072 : "after completing the handshake.", \
61 24072 : __VA_ARGS__ ) \
62 12048 : X( initial_max_stream_data_bidi_local, \
63 12042 : 0x05, \
64 12042 : VARINT, \
65 12042 : DFT_UNKNOWN, \
66 12042 : "This parameter is an integer value specifying the initial flow control limit for " \
67 12042 : "locally initiated bidirectional streams. This limit applies to newly created " \
68 12042 : "bidirectional streams opened by the endpoint that sends the transport parameter. " \
69 12042 : "In client transport parameters, this applies to streams with an identifier with " \
70 12042 : "the least significant two bits set to 0x00; in server transport parameters, this " \
71 12042 : "applies to streams with the least significant two bits set to 0x01.", \
72 12048 : __VA_ARGS__ ) \
73 12048 : X( initial_max_stream_data_bidi_remote, \
74 12042 : 0x06, \
75 12042 : VARINT, \
76 12042 : DFT_UNKNOWN, \
77 12042 : "This parameter is an integer value specifying the initial flow control limit for " \
78 12042 : "peer-initiated bidirectional streams. This limit applies to newly created " \
79 12042 : "bidirectional streams opened by the endpoint that receives the transport " \
80 12042 : "parameter. In client transport parameters, this applies to streams with an " \
81 12042 : "identifier with the least significant two bits set to 0x01; in server transport " \
82 12042 : "parameters, this applies to streams with the least significant two bits set to " \
83 12042 : "0x00.", \
84 24078 : __VA_ARGS__ ) \
85 24078 : X( initial_max_stream_data_uni, \
86 24072 : 0x07, \
87 24072 : VARINT, \
88 24072 : DFT_UNKNOWN, \
89 24072 : "This parameter is an integer value specifying the initial flow control limit for " \
90 24072 : "unidirectional streams. This limit applies to newly created unidirectional " \
91 24072 : "streams opened by the endpoint that receives the transport parameter. In client " \
92 24072 : "transport parameters, this applies to streams with an identifier with the least " \
93 24072 : "significant two bits set to 0x03; in server transport parameters, this applies " \
94 24072 : "to streams with the least significant two bits set to 0x02.", \
95 24078 : __VA_ARGS__ ) \
96 24078 : X( initial_max_streams_bidi, \
97 24072 : 0x08, \
98 24072 : VARINT, \
99 24072 : DFT_UNKNOWN, \
100 24072 : "The initial maximum bidirectional streams parameter is an integer value that " \
101 24072 : "contains the initial maximum number of bidirectional streams the endpoint that " \
102 24072 : "receives this transport parameter is permitted to initiate. If this parameter is " \
103 24072 : "absent or zero, the peer cannot open bidirectional streams until a MAX_STREAMS " \
104 24072 : "frame is sent. Setting this parameter is equivalent to sending a MAX_STREAMS " \
105 24072 : "(Section 19.11) of the corresponding type with the same value.", \
106 24078 : __VA_ARGS__ ) \
107 24078 : X( initial_max_streams_uni, \
108 24072 : 0x09, \
109 24072 : VARINT, \
110 24072 : DFT_UNKNOWN, \
111 24072 : "The initial maximum unidirectional streams parameter is an integer value that " \
112 24072 : "contains the initial maximum number of unidirectional streams the endpoint that " \
113 24072 : "receives this transport parameter is permitted to initiate. If this parameter is " \
114 24072 : "absent or zero, the peer cannot open unidirectional streams until a MAX_STREAMS " \
115 24072 : "frame is sent. Setting this parameter is equivalent to sending a MAX_STREAMS " \
116 24072 : "(Section 19.11) of the corresponding type with the same value.", \
117 24078 : __VA_ARGS__ ) \
118 24078 : X( ack_delay_exponent, \
119 24072 : 0x0a, \
120 24072 : VARINT, \
121 24072 : DFT_UNKNOWN, \
122 24072 : "The acknowledgment delay exponent is an integer value indicating an exponent " \
123 24072 : "used to decode the ACK Delay field in the ACK frame (Section 19.3). If this " \
124 24072 : "value is absent, a default value of 3 is assumed (indicating a multiplier of 8).\n" \
125 24072 : "Values above 20 are invalid.", \
126 24078 : __VA_ARGS__ ) \
127 24078 : X( max_ack_delay, \
128 24072 : 0x0b, \
129 24072 : VARINT, \
130 24072 : 25, \
131 24072 : "The maximum acknowledgment delay is an integer value indicating the maximum " \
132 24072 : "amount of time in milliseconds by which the endpoint will delay sending " \
133 24072 : "acknowledgments. This value SHOULD include the receiver's expected delays in " \
134 24072 : "alarms firing. For example, if a receiver sets a timer for 5ms and alarms " \
135 24072 : "commonly fire up to 1ms late, then it should send a max_ack_delay of 6ms. If " \
136 24072 : "this value is absent, a default of 25 milliseconds is assumed. Values of 2^14 or " \
137 24072 : "greater are invalid.", \
138 24078 : __VA_ARGS__ ) \
139 24078 : X( disable_active_migration, \
140 24072 : 0x0c, \
141 24072 : ZERO_LENGTH, \
142 24072 : DFT_UNKNOWN, \
143 24072 : "The disable active migration transport parameter is included if the endpoint " \
144 24072 : "does not support active connection migration (Section 9) on the address being " \
145 24072 : "used during the handshake. An endpoint that receives this transport parameter " \
146 24072 : "MUST NOT use a new local address when sending to the address that the peer used " \
147 24072 : "during the handshake. This transport parameter does not prohibit connection " \
148 24072 : "migration after a client has acted on a preferred_address transport parameter.\n" \
149 24072 : "This parameter is a zero-length value.", \
150 24072 : __VA_ARGS__ ) \
151 12045 : X( preferred_address, \
152 12039 : 0x0d, \
153 12039 : PREFERRED_ADDRESS, \
154 12039 : DFT_UNKNOWN, \
155 12039 : "The server's preferred address is used to effect a change in server address at " \
156 12039 : "the end of the handshake, as described in Section 9.6. This transport parameter " \
157 12039 : "is only sent by a server. Servers MAY choose to only send a preferred address of " \
158 12039 : "one address family by sending an all-zero address and port (0.0.0.0:0 or [::]:0) " \
159 12039 : "for the other family. IP addresses are encoded in network byte order.\n" \
160 12039 : "The preferred_address transport parameter contains an address and port for both " \
161 12039 : "IPv4 and IPv6. The four-byte IPv4 Address field is followed by the associated " \
162 12039 : "two-byte IPv4 Port field. This is followed by a 16-byte IPv6 Address field and " \
163 12039 : "two-byte IPv6 Port field. After address and port pairs, a Connection ID Length " \
164 12039 : "field describes the length of the following Connection ID field. Finally, a " \
165 12039 : "16-byte Stateless Reset Token field includes the stateless reset token " \
166 12039 : "associated with the connection ID. The format of this transport parameter is " \
167 12039 : "shown in Figure 22 below.", \
168 12048 : __VA_ARGS__ ) \
169 12048 : X( active_connection_id_limit, \
170 12042 : 0x0e, \
171 12042 : VARINT, \
172 12042 : 2, \
173 12042 : "This is an integer value specifying the maximum number of connection IDs from " \
174 12042 : "the peer that an endpoint is willing to store. This value includes the " \
175 12042 : "connection ID received during the handshake, that received in the " \
176 12042 : "preferred_address transport parameter, and those received in NEW_CONNECTION_ID " \
177 12042 : "frames. The value of the active_connection_id_limit parameter MUST be at least " \
178 12042 : "2. An endpoint that receives a value less than 2 MUST close the connection with " \
179 12042 : "an error of type TRANSPORT_PARAMETER_ERROR. If this transport parameter is " \
180 12042 : "absent, a default of 2 is assumed. If an endpoint issues a zero-length " \
181 12042 : "connection ID, it will never send a NEW_CONNECTION_ID frame and therefore " \
182 12042 : "ignores the active_connection_id_limit value received from its peer.", \
183 24078 : __VA_ARGS__ ) \
184 24078 : X( initial_source_connection_id, \
185 24072 : 0x0f, \
186 24072 : CONN_ID, \
187 24072 : DFT_UNKNOWN, \
188 24072 : "This is the value that the endpoint included in the Source Connection ID field " \
189 24072 : "of the first Initial packet it sends for the connection; see Section 7.3.", \
190 24072 : __VA_ARGS__ ) \
191 12048 : X( retry_source_connection_id, \
192 12042 : 0x10, \
193 12042 : CONN_ID, \
194 12042 : DFT_UNKNOWN, \
195 12042 : "This is the value that the server included in the Source Connection ID field of " \
196 12042 : "a Retry packet; see Section 7.3. This transport parameter is only sent by a " \
197 12042 : "server.", \
198 12042 : __VA_ARGS__ )
199 :
200 3 : #define FD_QUIC_PREFERRED_ADDRESS_SZ_MAX (61)
201 :
202 : void
203 : fd_quic_dump_transport_param_desc( FILE * out );
204 :
205 : // TODO verify max length on these - CONN_ID and TOKEN
206 : // PREFERRED_ADDRESS is incomplete
207 : #define FD_QUIC_MBR_TYPE_VARINT(NAME,TYPE) \
208 : ulong NAME; \
209 : uchar NAME##_present;
210 : #define FD_QUIC_MBR_TYPE_CONN_ID(NAME,TYPE) \
211 : uchar NAME##_len; \
212 : uchar NAME[20]; \
213 : uchar NAME##_present;
214 : #define FD_QUIC_MBR_TYPE_ZERO_LENGTH(NAME,TYPE) \
215 : uchar NAME##_present;
216 : #define FD_QUIC_MBR_TYPE_TOKEN(NAME,TYPE) \
217 : uint NAME##_len; \
218 : uchar NAME[16]; \
219 : uchar NAME##_present;
220 : #define FD_QUIC_MBR_TYPE_PREFERRED_ADDRESS(NAME,TYPE) \
221 : uint NAME##_len; \
222 : uchar NAME[FD_QUIC_PREFERRED_ADDRESS_SZ_MAX]; \
223 : uchar NAME##_present;
224 :
225 : struct fd_quic_transport_params {
226 : #define __( NAME, ID, TYPE, DFT, DESC, ... ) \
227 : FD_QUIC_MBR_TYPE_##TYPE(NAME,TYPE)
228 : FD_QUIC_TRANSPORT_PARAMS( __, _ )
229 : #undef __
230 : };
231 : typedef struct fd_quic_transport_params fd_quic_transport_params_t;
232 :
233 : #define FD_QUIC_TRANSPORT_PARAM_SET( TP, NAME, VALUE ) \
234 26616 : do { (TP)->NAME##_present = 1; (TP)->NAME = VALUE; } while(0);
235 : #define FD_QUIC_TRANSPORT_PARAM_UNSET( TP, NAME ) \
236 : do { (TP)->NAME##_present = 0; } while(0);
237 :
238 : /* parses the varint at *buf (capacity *buf_sz)
239 : advances the *buf and reduces *buf_sz by the number of bytes
240 : consumed */
241 : static inline ulong
242 : fd_quic_tp_parse_varint( uchar const ** buf,
243 253308 : ulong * buf_sz ) {
244 :
245 253308 : if( FD_UNLIKELY( *buf_sz == 0 ) ) return ~(ulong)0;
246 :
247 253305 : uint width = 1u << ( (uint)(*buf)[0] >> 6u );
248 253305 : if( FD_UNLIKELY( *buf_sz < width ) ) return ~(ulong)0;
249 :
250 253305 : ulong value = (ulong)( (*buf)[0] & 0x3f );
251 253377 : for( ulong j=1; j<width; ++j ) {
252 72 : value = ( value<<8UL ) + (ulong)(*buf)[j];
253 72 : }
254 :
255 253305 : *buf += width;
256 253305 : *buf_sz -= width;
257 :
258 253305 : return value;
259 253305 : }
260 :
261 : /* parse the entire buffer into the supplied transport parameters
262 :
263 : unknown transport parameters are ignored as per spec
264 :
265 : returns
266 : 0 success
267 : -1 failed to parse */
268 : int
269 : fd_quic_decode_transport_params( fd_quic_transport_params_t * params,
270 : uchar const * buf,
271 : ulong buf_sz );
272 :
273 : /* dump all transport parameters to stdout */
274 : void
275 : fd_quic_dump_transport_params( fd_quic_transport_params_t const * params,
276 : FILE * out );
277 :
278 :
279 : /* encode transport parameters into a buffer
280 : args
281 : buf the buffer to write encoded transport params into
282 : buf_sz the size of buffer buf
283 : params the parameters to be encoded
284 :
285 : returns the number of bytes written */
286 : ulong
287 : fd_quic_encode_transport_params( uchar * buf,
288 : ulong buf_sz,
289 : fd_quic_transport_params_t const * params );
290 :
291 : #endif
292 :
|