1#[cfg(feature = "sfv")]
28use std::convert::TryFrom;
29
30use std::ptr;
31use std::slice;
32
33use libc::c_int;
34use libc::c_void;
35use libc::size_t;
36use libc::ssize_t;
37
38use crate::*;
39
40use crate::h3::NameValue;
41use crate::h3::Priority;
42
43#[no_mangle]
44pub extern "C" fn quiche_h3_config_new() -> *mut h3::Config {
45 match h3::Config::new() {
46 Ok(c) => Box::into_raw(Box::new(c)),
47
48 Err(_) => ptr::null_mut(),
49 }
50}
51
52#[no_mangle]
53pub extern "C" fn quiche_h3_config_set_max_field_section_size(
54 config: &mut h3::Config, v: u64,
55) {
56 config.set_max_field_section_size(v);
57}
58
59#[no_mangle]
60pub extern "C" fn quiche_h3_config_set_qpack_max_table_capacity(
61 config: &mut h3::Config, v: u64,
62) {
63 config.set_qpack_max_table_capacity(v);
64}
65
66#[no_mangle]
67pub extern "C" fn quiche_h3_config_set_qpack_blocked_streams(
68 config: &mut h3::Config, v: u64,
69) {
70 config.set_qpack_blocked_streams(v);
71}
72
73#[no_mangle]
74pub extern "C" fn quiche_h3_config_enable_extended_connect(
75 config: &mut h3::Config, enabled: bool,
76) {
77 config.enable_extended_connect(enabled);
78}
79
80#[no_mangle]
81pub extern "C" fn quiche_h3_config_free(config: *mut h3::Config) {
82 drop(unsafe { Box::from_raw(config) });
83}
84
85#[no_mangle]
86pub extern "C" fn quiche_h3_conn_new_with_transport(
87 quic_conn: &mut Connection, config: &mut h3::Config,
88) -> *mut h3::Connection {
89 match h3::Connection::with_transport(quic_conn, config) {
90 Ok(c) => Box::into_raw(Box::new(c)),
91
92 Err(_) => ptr::null_mut(),
93 }
94}
95
96#[no_mangle]
97pub extern "C" fn quiche_h3_for_each_setting(
98 conn: &h3::Connection,
99 cb: extern "C" fn(identifier: u64, value: u64, argp: *mut c_void) -> c_int,
100 argp: *mut c_void,
101) -> c_int {
102 match conn.peer_settings_raw() {
103 Some(raw) => {
104 for setting in raw {
105 let rc = cb(setting.0, setting.1, argp);
106
107 if rc != 0 {
108 return rc;
109 }
110 }
111
112 0
113 },
114
115 None => -1,
116 }
117}
118
119#[no_mangle]
120pub extern "C" fn quiche_h3_conn_poll(
121 conn: &mut h3::Connection, quic_conn: &mut Connection,
122 ev: *mut *const h3::Event,
123) -> i64 {
124 match conn.poll(quic_conn) {
125 Ok((id, v)) => {
126 unsafe {
127 *ev = Box::into_raw(Box::new(v));
128 }
129
130 id as i64
131 },
132
133 Err(e) => e.to_c() as i64,
134 }
135}
136
137#[no_mangle]
138pub extern "C" fn quiche_h3_event_type(ev: &h3::Event) -> u32 {
139 match ev {
140 h3::Event::Headers { .. } => 0,
141
142 h3::Event::Data => 1,
143
144 h3::Event::Finished => 2,
145
146 h3::Event::GoAway => 3,
147
148 h3::Event::Reset { .. } => 4,
149
150 h3::Event::PriorityUpdate => 5,
151 }
152}
153
154#[no_mangle]
155pub extern "C" fn quiche_h3_event_for_each_header(
156 ev: &h3::Event,
157 cb: extern "C" fn(
158 name: *const u8,
159 name_len: size_t,
160
161 value: *const u8,
162 value_len: size_t,
163
164 argp: *mut c_void,
165 ) -> c_int,
166 argp: *mut c_void,
167) -> c_int {
168 match ev {
169 h3::Event::Headers { list, .. } =>
170 for h in list {
171 let rc = cb(
172 h.name().as_ptr(),
173 h.name().len(),
174 h.value().as_ptr(),
175 h.value().len(),
176 argp,
177 );
178
179 if rc != 0 {
180 return rc;
181 }
182 },
183
184 _ => unreachable!(),
185 }
186
187 0
188}
189
190#[no_mangle]
191pub extern "C" fn quiche_h3_event_headers_has_more_frames(
192 ev: &h3::Event,
193) -> bool {
194 match ev {
195 h3::Event::Headers { more_frames, .. } => *more_frames,
196
197 _ => unreachable!(),
198 }
199}
200
201#[no_mangle]
202pub extern "C" fn quiche_h3_extended_connect_enabled_by_peer(
203 conn: &h3::Connection,
204) -> bool {
205 conn.extended_connect_enabled_by_peer()
206}
207
208#[no_mangle]
209pub extern "C" fn quiche_h3_event_free(ev: *mut h3::Event) {
210 drop(unsafe { Box::from_raw(ev) });
211}
212
213#[repr(C)]
214pub struct Header {
215 name: *mut u8,
216 name_len: usize,
217
218 value: *mut u8,
219 value_len: usize,
220}
221
222#[no_mangle]
223pub extern "C" fn quiche_h3_send_request(
224 conn: &mut h3::Connection, quic_conn: &mut Connection,
225 headers: *const Header, headers_len: size_t, fin: bool,
226) -> i64 {
227 let req_headers = headers_from_ptr(headers, headers_len);
228
229 match conn.send_request(quic_conn, &req_headers, fin) {
230 Ok(v) => v as i64,
231
232 Err(e) => e.to_c() as i64,
233 }
234}
235
236#[no_mangle]
237pub extern "C" fn quiche_h3_send_response(
238 conn: &mut h3::Connection, quic_conn: &mut Connection, stream_id: u64,
239 headers: *const Header, headers_len: size_t, fin: bool,
240) -> c_int {
241 let resp_headers = headers_from_ptr(headers, headers_len);
242
243 match conn.send_response(quic_conn, stream_id, &resp_headers, fin) {
244 Ok(_) => 0,
245
246 Err(e) => e.to_c() as c_int,
247 }
248}
249
250#[no_mangle]
251pub extern "C" fn quiche_h3_send_response_with_priority(
252 conn: &mut h3::Connection, quic_conn: &mut Connection, stream_id: u64,
253 headers: *const Header, headers_len: size_t, priority: &Priority, fin: bool,
254) -> c_int {
255 let resp_headers = headers_from_ptr(headers, headers_len);
256
257 match conn.send_response_with_priority(
258 quic_conn,
259 stream_id,
260 &resp_headers,
261 priority,
262 fin,
263 ) {
264 Ok(_) => 0,
265
266 Err(e) => e.to_c() as c_int,
267 }
268}
269
270#[no_mangle]
271pub extern "C" fn quiche_h3_send_additional_headers(
272 conn: &mut h3::Connection, quic_conn: &mut Connection, stream_id: u64,
273 headers: *const Header, headers_len: size_t, is_trailer_section: bool,
274 fin: bool,
275) -> c_int {
276 let headers = headers_from_ptr(headers, headers_len);
277
278 match conn.send_additional_headers(
279 quic_conn,
280 stream_id,
281 &headers,
282 is_trailer_section,
283 fin,
284 ) {
285 Ok(_) => 0,
286
287 Err(e) => e.to_c() as c_int,
288 }
289}
290
291#[no_mangle]
292pub extern "C" fn quiche_h3_send_body(
293 conn: &mut h3::Connection, quic_conn: &mut Connection, stream_id: u64,
294 body: *const u8, body_len: size_t, fin: bool,
295) -> ssize_t {
296 if body_len > <ssize_t>::MAX as usize {
297 panic!("The provided buffer is too large");
298 }
299
300 let body = unsafe { slice::from_raw_parts(body, body_len) };
301
302 match conn.send_body(quic_conn, stream_id, body, fin) {
303 Ok(v) => v as ssize_t,
304
305 Err(e) => e.to_c(),
306 }
307}
308
309#[no_mangle]
310pub extern "C" fn quiche_h3_recv_body(
311 conn: &mut h3::Connection, quic_conn: &mut Connection, stream_id: u64,
312 out: *mut u8, out_len: size_t,
313) -> ssize_t {
314 if out_len > <ssize_t>::MAX as usize {
315 panic!("The provided buffer is too large");
316 }
317
318 let out = unsafe { slice::from_raw_parts_mut(out, out_len) };
319
320 match conn.recv_body(quic_conn, stream_id, out) {
321 Ok(v) => v as ssize_t,
322
323 Err(e) => e.to_c(),
324 }
325}
326
327#[no_mangle]
328pub extern "C" fn quiche_h3_send_goaway(
329 conn: &mut h3::Connection, quic_conn: &mut Connection, id: u64,
330) -> c_int {
331 match conn.send_goaway(quic_conn, id) {
332 Ok(()) => 0,
333
334 Err(e) => e.to_c() as c_int,
335 }
336}
337
338#[no_mangle]
339#[cfg(feature = "sfv")]
340pub extern "C" fn quiche_h3_parse_extensible_priority(
341 priority: *const u8, priority_len: size_t, parsed: &mut Priority,
342) -> c_int {
343 let priority = unsafe { slice::from_raw_parts(priority, priority_len) };
344
345 match h3::Priority::try_from(priority) {
346 Ok(v) => {
347 parsed.urgency = v.urgency;
348 parsed.incremental = v.incremental;
349 0
350 },
351
352 Err(e) => e.to_c() as c_int,
353 }
354}
355
356#[no_mangle]
357pub extern "C" fn quiche_h3_send_priority_update_for_request(
358 conn: &mut h3::Connection, quic_conn: &mut Connection, stream_id: u64,
359 priority: &Priority,
360) -> c_int {
361 match conn.send_priority_update_for_request(quic_conn, stream_id, priority) {
362 Ok(()) => 0,
363
364 Err(e) => e.to_c() as c_int,
365 }
366}
367
368#[no_mangle]
369pub extern "C" fn quiche_h3_take_last_priority_update(
370 conn: &mut h3::Connection, prioritized_element_id: u64,
371 cb: extern "C" fn(
372 priority_field_value: *const u8,
373 priority_field_value_len: size_t,
374 argp: *mut c_void,
375 ) -> c_int,
376 argp: *mut c_void,
377) -> c_int {
378 match conn.take_last_priority_update(prioritized_element_id) {
379 Ok(priority) => {
380 let rc = cb(priority.as_ptr(), priority.len(), argp);
381
382 if rc != 0 {
383 return rc;
384 }
385
386 0
387 },
388
389 Err(e) => e.to_c() as c_int,
390 }
391}
392
393#[no_mangle]
394pub extern "C" fn quiche_h3_dgram_enabled_by_peer(
395 conn: &h3::Connection, quic_conn: &Connection,
396) -> bool {
397 conn.dgram_enabled_by_peer(quic_conn)
398}
399
400#[no_mangle]
401pub extern "C" fn quiche_h3_conn_free(conn: *mut h3::Connection) {
402 drop(unsafe { Box::from_raw(conn) });
403}
404
405fn headers_from_ptr<'a>(
406 ptr: *const Header, len: size_t,
407) -> Vec<h3::HeaderRef<'a>> {
408 let headers = unsafe { slice::from_raw_parts(ptr, len) };
409
410 let mut out = Vec::new();
411
412 for h in headers {
413 out.push({
414 let name = unsafe { slice::from_raw_parts(h.name, h.name_len) };
415 let value = unsafe { slice::from_raw_parts(h.value, h.value_len) };
416
417 h3::HeaderRef::new(name, value)
418 });
419 }
420
421 out
422}
423
424#[repr(C)]
425pub struct Stats {
426 qpack_encoder_stream_recv_bytes: u64,
427 qpack_decoder_stream_recv_bytes: u64,
428}
429
430#[no_mangle]
431pub extern "C" fn quiche_h3_conn_stats(conn: &h3::Connection, out: &mut Stats) {
432 let stats = conn.stats();
433
434 out.qpack_encoder_stream_recv_bytes = stats.qpack_encoder_stream_recv_bytes;
435 out.qpack_decoder_stream_recv_bytes = stats.qpack_decoder_stream_recv_bytes;
436}