quiche/h3/
ffi.rs

1// Copyright (C) 2019, Cloudflare, Inc.
2// All rights reserved.
3//
4// Redistribution and use in source and binary forms, with or without
5// modification, are permitted provided that the following conditions are
6// met:
7//
8//     * Redistributions of source code must retain the above copyright notice,
9//       this list of conditions and the following disclaimer.
10//
11//     * Redistributions in binary form must reproduce the above copyright
12//       notice, this list of conditions and the following disclaimer in the
13//       documentation and/or other materials provided with the distribution.
14//
15// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
16// IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
17// THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
18// PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
19// CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
20// EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
22// PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
23// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
24// NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
25// SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26
27#[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}