blob: 39564dcf6a740031c920fbb1d171661a2ff2b31b [file] [log] [blame]
// Copyright (C) 2018-2019, Cloudflare, Inc.
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are
// met:
//
// * Redistributions of source code must retain the above copyright notice,
// this list of conditions and the following disclaimer.
//
// * Redistributions in binary form must reproduce the above copyright
// notice, this list of conditions and the following disclaimer in the
// documentation and/or other materials provided with the distribution.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS
// IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
// THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
// PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
// CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
// EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
// PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
// LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
// NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
// SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
use std::ffi;
use std::ptr;
use std::slice;
use std::sync::atomic;
#[cfg(unix)]
use std::os::unix::io::FromRawFd;
use libc::c_char;
use libc::c_int;
use libc::c_void;
use libc::size_t;
use libc::ssize_t;
use crate::*;
#[no_mangle]
pub extern fn quiche_version() -> *const u8 {
//static VERSION: &str = concat!(env!("CARGO_PKG_VERSION"), "\0");
// ANDROID's build system doesn't support environment variables
// so we hardcode the package version here.
static VERSION: &str = concat!("0.6.0", "\0");
VERSION.as_ptr()
}
struct Logger {
cb: extern fn(line: *const u8, argp: *mut c_void),
argp: std::sync::atomic::AtomicPtr<c_void>,
}
impl log::Log for Logger {
fn enabled(&self, _metadata: &log::Metadata) -> bool {
true
}
fn log(&self, record: &log::Record) {
let line = format!("{}: {}\0", record.target(), record.args());
(self.cb)(line.as_ptr(), self.argp.load(atomic::Ordering::Relaxed));
}
fn flush(&self) {}
}
#[no_mangle]
pub extern fn quiche_enable_debug_logging(
cb: extern fn(line: *const u8, argp: *mut c_void), argp: *mut c_void,
) -> c_int {
let argp = atomic::AtomicPtr::new(argp);
let logger = Box::new(Logger { cb, argp });
if log::set_boxed_logger(logger).is_err() {
return -1;
}
log::set_max_level(log::LevelFilter::Trace);
0
}
#[no_mangle]
pub extern fn quiche_config_new(version: u32) -> *mut Config {
match Config::new(version) {
Ok(c) => Box::into_raw(Box::new(c)),
Err(_) => ptr::null_mut(),
}
}
#[no_mangle]
pub extern fn quiche_config_load_cert_chain_from_pem_file(
config: &mut Config, path: *const c_char,
) -> c_int {
let path = unsafe { ffi::CStr::from_ptr(path).to_str().unwrap() };
match config.load_cert_chain_from_pem_file(path) {
Ok(_) => 0,
Err(e) => e.to_c() as c_int,
}
}
#[no_mangle]
pub extern fn quiche_config_load_priv_key_from_pem_file(
config: &mut Config, path: *const c_char,
) -> c_int {
let path = unsafe { ffi::CStr::from_ptr(path).to_str().unwrap() };
match config.load_priv_key_from_pem_file(path) {
Ok(_) => 0,
Err(e) => e.to_c() as c_int,
}
}
#[no_mangle]
pub extern fn quiche_config_verify_peer(config: &mut Config, v: bool) {
config.verify_peer(v);
}
#[no_mangle]
pub extern fn quiche_config_grease(config: &mut Config, v: bool) {
config.grease(v);
}
#[no_mangle]
pub extern fn quiche_config_log_keys(config: &mut Config) {
config.log_keys();
}
#[no_mangle]
pub extern fn quiche_config_enable_early_data(config: &mut Config) {
config.enable_early_data();
}
#[no_mangle]
pub extern fn quiche_config_set_application_protos(
config: &mut Config, protos: *const u8, protos_len: size_t,
) -> c_int {
let protos = unsafe { slice::from_raw_parts(protos, protos_len) };
match config.set_application_protos(protos) {
Ok(_) => 0,
Err(e) => e.to_c() as c_int,
}
}
#[no_mangle]
pub extern fn quiche_config_set_max_idle_timeout(config: &mut Config, v: u64) {
config.set_max_idle_timeout(v);
}
#[no_mangle]
pub extern fn quiche_config_set_max_udp_payload_size(
config: &mut Config, v: u64,
) {
config.set_max_udp_payload_size(v);
}
#[no_mangle]
pub extern fn quiche_config_set_initial_max_data(config: &mut Config, v: u64) {
config.set_initial_max_data(v);
}
#[no_mangle]
pub extern fn quiche_config_set_initial_max_stream_data_bidi_local(
config: &mut Config, v: u64,
) {
config.set_initial_max_stream_data_bidi_local(v);
}
#[no_mangle]
pub extern fn quiche_config_set_initial_max_stream_data_bidi_remote(
config: &mut Config, v: u64,
) {
config.set_initial_max_stream_data_bidi_remote(v);
}
#[no_mangle]
pub extern fn quiche_config_set_initial_max_stream_data_uni(
config: &mut Config, v: u64,
) {
config.set_initial_max_stream_data_uni(v);
}
#[no_mangle]
pub extern fn quiche_config_set_initial_max_streams_bidi(
config: &mut Config, v: u64,
) {
config.set_initial_max_streams_bidi(v);
}
#[no_mangle]
pub extern fn quiche_config_set_initial_max_streams_uni(
config: &mut Config, v: u64,
) {
config.set_initial_max_streams_uni(v);
}
#[no_mangle]
pub extern fn quiche_config_set_ack_delay_exponent(config: &mut Config, v: u64) {
config.set_ack_delay_exponent(v);
}
#[no_mangle]
pub extern fn quiche_config_set_max_ack_delay(config: &mut Config, v: u64) {
config.set_max_ack_delay(v);
}
#[no_mangle]
pub extern fn quiche_config_set_disable_active_migration(
config: &mut Config, v: bool,
) {
config.set_disable_active_migration(v);
}
#[no_mangle]
pub extern fn quiche_config_set_cc_algorithm_name(
config: &mut Config, name: *const c_char,
) -> c_int {
let name = unsafe { ffi::CStr::from_ptr(name).to_str().unwrap() };
match config.set_cc_algorithm_name(name) {
Ok(_) => 0,
Err(e) => e.to_c() as c_int,
}
}
#[no_mangle]
pub extern fn quiche_config_set_cc_algorithm(
config: &mut Config, algo: CongestionControlAlgorithm,
) {
config.set_cc_algorithm(algo);
}
#[no_mangle]
pub extern fn quiche_config_enable_hystart(config: &mut Config, v: bool) {
config.enable_hystart(v);
}
#[no_mangle]
pub extern fn quiche_config_enable_dgram(
config: &mut Config, enabled: bool, recv_queue_len: size_t,
send_queue_len: size_t,
) {
config.enable_dgram(enabled, recv_queue_len, send_queue_len);
}
#[no_mangle]
pub extern fn quiche_config_free(config: *mut Config) {
unsafe { Box::from_raw(config) };
}
#[no_mangle]
pub extern fn quiche_header_info(
buf: *mut u8, buf_len: size_t, dcil: size_t, version: *mut u32, ty: *mut u8,
scid: *mut u8, scid_len: *mut size_t, dcid: *mut u8, dcid_len: *mut size_t,
token: *mut u8, token_len: *mut size_t,
) -> c_int {
let buf = unsafe { slice::from_raw_parts_mut(buf, buf_len) };
let hdr = match Header::from_slice(buf, dcil) {
Ok(v) => v,
Err(e) => return e.to_c() as c_int,
};
unsafe {
*version = hdr.version;
*ty = match hdr.ty {
Type::Initial => 1,
Type::Retry => 2,
Type::Handshake => 3,
Type::ZeroRTT => 4,
Type::Short => 5,
Type::VersionNegotiation => 6,
};
if *scid_len < hdr.scid.len() {
return -1;
}
let scid = slice::from_raw_parts_mut(scid, *scid_len);
let scid = &mut scid[..hdr.scid.len()];
scid.copy_from_slice(&hdr.scid);
*scid_len = hdr.scid.len();
if *dcid_len < hdr.dcid.len() {
return -1;
}
let dcid = slice::from_raw_parts_mut(dcid, *dcid_len);
let dcid = &mut dcid[..hdr.dcid.len()];
dcid.copy_from_slice(&hdr.dcid);
*dcid_len = hdr.dcid.len();
match hdr.token {
Some(tok) => {
if *token_len < tok.len() {
return -1;
}
let token = slice::from_raw_parts_mut(token, *token_len);
let token = &mut token[..tok.len()];
token.copy_from_slice(&tok);
*token_len = tok.len();
},
None => *token_len = 0,
}
}
0
}
#[no_mangle]
pub extern fn quiche_accept(
scid: *const u8, scid_len: size_t, odcid: *const u8, odcid_len: size_t,
config: &mut Config,
) -> *mut Connection {
let scid = unsafe { slice::from_raw_parts(scid, scid_len) };
let odcid = if !odcid.is_null() || odcid_len == 0 {
Some(unsafe { slice::from_raw_parts(odcid, odcid_len) })
} else {
None
};
match accept(scid, odcid, config) {
Ok(c) => Box::into_raw(Pin::into_inner(c)),
Err(_) => ptr::null_mut(),
}
}
#[no_mangle]
pub extern fn quiche_connect(
server_name: *const c_char, scid: *const u8, scid_len: size_t,
config: &mut Config,
) -> *mut Connection {
let server_name = if server_name.is_null() {
None
} else {
Some(unsafe { ffi::CStr::from_ptr(server_name).to_str().unwrap() })
};
let scid = unsafe { slice::from_raw_parts(scid, scid_len) };
match connect(server_name, scid, config) {
Ok(c) => Box::into_raw(Pin::into_inner(c)),
Err(_) => ptr::null_mut(),
}
}
#[no_mangle]
pub extern fn quiche_negotiate_version(
scid: *const u8, scid_len: size_t, dcid: *const u8, dcid_len: size_t,
out: *mut u8, out_len: size_t,
) -> ssize_t {
let scid = unsafe { slice::from_raw_parts(scid, scid_len) };
let dcid = unsafe { slice::from_raw_parts(dcid, dcid_len) };
let out = unsafe { slice::from_raw_parts_mut(out, out_len) };
match negotiate_version(scid, dcid, out) {
Ok(v) => v as ssize_t,
Err(e) => e.to_c(),
}
}
#[no_mangle]
pub extern fn quiche_version_is_supported(version: u32) -> bool {
version_is_supported(version)
}
#[no_mangle]
pub extern fn quiche_retry(
scid: *const u8, scid_len: size_t, dcid: *const u8, dcid_len: size_t,
new_scid: *const u8, new_scid_len: size_t, token: *const u8,
token_len: size_t, version: u32, out: *mut u8, out_len: size_t,
) -> ssize_t {
let scid = unsafe { slice::from_raw_parts(scid, scid_len) };
let dcid = unsafe { slice::from_raw_parts(dcid, dcid_len) };
let new_scid = unsafe { slice::from_raw_parts(new_scid, new_scid_len) };
let token = unsafe { slice::from_raw_parts(token, token_len) };
let out = unsafe { slice::from_raw_parts_mut(out, out_len) };
match retry(scid, dcid, new_scid, token, version, out) {
Ok(v) => v as ssize_t,
Err(e) => e.to_c(),
}
}
#[no_mangle]
pub extern fn quiche_conn_new_with_tls(
scid: *const u8, scid_len: size_t, odcid: *const u8, odcid_len: size_t,
config: &mut Config, ssl: *mut c_void, is_server: bool,
) -> *mut Connection {
let scid = unsafe { slice::from_raw_parts(scid, scid_len) };
let odcid = if !odcid.is_null() || odcid_len == 0 {
Some(unsafe { slice::from_raw_parts(odcid, odcid_len) })
} else {
None
};
let tls = unsafe { tls::Handshake::from_ptr(ssl) };
match Connection::with_tls(scid, odcid, config, tls, is_server) {
Ok(c) => Box::into_raw(Pin::into_inner(c)),
Err(_) => ptr::null_mut(),
}
}
#[no_mangle]
pub extern fn quiche_conn_set_keylog_path(
conn: &mut Connection, path: *const c_char,
) -> bool {
let filename = unsafe { ffi::CStr::from_ptr(path).to_str().unwrap() };
let file = std::fs::OpenOptions::new()
.create(true)
.append(true)
.open(filename);
let writer = match file {
Ok(f) => std::io::BufWriter::new(f),
Err(_) => return false,
};
conn.set_keylog(Box::new(writer));
true
}
#[no_mangle]
#[cfg(unix)]
pub extern fn quiche_conn_set_keylog_fd(conn: &mut Connection, fd: c_int) {
let f = unsafe { std::fs::File::from_raw_fd(fd) };
let writer = std::io::BufWriter::new(f);
conn.set_keylog(Box::new(writer));
}
#[no_mangle]
#[cfg(feature = "qlog")]
pub extern fn quiche_conn_set_qlog_path(
conn: &mut Connection, path: *const c_char, log_title: *const c_char,
log_desc: *const c_char,
) -> bool {
let filename = unsafe { ffi::CStr::from_ptr(path).to_str().unwrap() };
let file = std::fs::OpenOptions::new()
.write(true)
.create_new(true)
.open(filename);
let writer = match file {
Ok(f) => std::io::BufWriter::new(f),
Err(_) => return false,
};
let title = unsafe { ffi::CStr::from_ptr(log_title).to_str().unwrap() };
let description = unsafe { ffi::CStr::from_ptr(log_desc).to_str().unwrap() };
conn.set_qlog(
Box::new(writer),
title.to_string(),
format!("{} id={}", description, conn.trace_id),
);
true
}
#[no_mangle]
#[cfg(all(unix, feature = "qlog"))]
pub extern fn quiche_conn_set_qlog_fd(
conn: &mut Connection, fd: c_int, log_title: *const c_char,
log_desc: *const c_char,
) {
let f = unsafe { std::fs::File::from_raw_fd(fd) };
let writer = std::io::BufWriter::new(f);
let title = unsafe { ffi::CStr::from_ptr(log_title).to_str().unwrap() };
let description = unsafe { ffi::CStr::from_ptr(log_desc).to_str().unwrap() };
conn.set_qlog(
Box::new(writer),
title.to_string(),
format!("{} id={}", description, conn.trace_id),
);
}
#[no_mangle]
pub extern fn quiche_conn_recv(
conn: &mut Connection, buf: *mut u8, buf_len: size_t,
) -> ssize_t {
if buf_len > <ssize_t>::max_value() as usize {
panic!("The provided buffer is too large");
}
let buf = unsafe { slice::from_raw_parts_mut(buf, buf_len) };
match conn.recv(buf) {
Ok(v) => v as ssize_t,
Err(e) => e.to_c(),
}
}
#[no_mangle]
pub extern fn quiche_conn_send(
conn: &mut Connection, out: *mut u8, out_len: size_t,
) -> ssize_t {
if out_len > <ssize_t>::max_value() as usize {
panic!("The provided buffer is too large");
}
let out = unsafe { slice::from_raw_parts_mut(out, out_len) };
match conn.send(out) {
Ok(v) => v as ssize_t,
Err(e) => e.to_c(),
}
}
#[no_mangle]
pub extern fn quiche_conn_stream_recv(
conn: &mut Connection, stream_id: u64, out: *mut u8, out_len: size_t,
fin: &mut bool,
) -> ssize_t {
if out_len > <ssize_t>::max_value() as usize {
panic!("The provided buffer is too large");
}
let out = unsafe { slice::from_raw_parts_mut(out, out_len) };
let (out_len, out_fin) = match conn.stream_recv(stream_id, out) {
Ok(v) => v,
Err(e) => return e.to_c(),
};
*fin = out_fin;
out_len as ssize_t
}
#[no_mangle]
pub extern fn quiche_conn_stream_send(
conn: &mut Connection, stream_id: u64, buf: *const u8, buf_len: size_t,
fin: bool,
) -> ssize_t {
if buf_len > <ssize_t>::max_value() as usize {
panic!("The provided buffer is too large");
}
let buf = unsafe { slice::from_raw_parts(buf, buf_len) };
match conn.stream_send(stream_id, buf, fin) {
Ok(v) => v as ssize_t,
Err(e) => e.to_c(),
}
}
#[no_mangle]
pub extern fn quiche_conn_stream_shutdown(
conn: &mut Connection, stream_id: u64, direction: Shutdown, err: u64,
) -> c_int {
match conn.stream_shutdown(stream_id, direction, err) {
Ok(_) => 0,
Err(e) => e.to_c() as c_int,
}
}
#[no_mangle]
pub extern fn quiche_conn_stream_capacity(
conn: &mut Connection, stream_id: u64,
) -> ssize_t {
match conn.stream_capacity(stream_id) {
Ok(v) => v as ssize_t,
Err(e) => e.to_c(),
}
}
#[no_mangle]
pub extern fn quiche_conn_stream_finished(
conn: &mut Connection, stream_id: u64,
) -> bool {
conn.stream_finished(stream_id)
}
#[no_mangle]
pub extern fn quiche_conn_readable(conn: &Connection) -> *mut StreamIter {
Box::into_raw(Box::new(conn.readable()))
}
#[no_mangle]
pub extern fn quiche_conn_writable(conn: &Connection) -> *mut StreamIter {
Box::into_raw(Box::new(conn.writable()))
}
struct AppData(*mut c_void);
unsafe impl Send for AppData {}
#[no_mangle]
pub extern fn quiche_conn_stream_init_application_data(
conn: &mut Connection, stream_id: u64, data: *mut c_void,
) -> c_int {
match conn.stream_init_application_data(stream_id, AppData(data)) {
Ok(_) => 0,
Err(e) => e.to_c() as c_int,
}
}
#[no_mangle]
pub extern fn quiche_conn_stream_application_data(
conn: &mut Connection, stream_id: u64,
) -> *mut c_void {
match conn.stream_application_data(stream_id) {
Some(v) => v.downcast_mut::<AppData>().unwrap().0,
None => ptr::null_mut(),
}
}
#[no_mangle]
pub extern fn quiche_conn_close(
conn: &mut Connection, app: bool, err: u64, reason: *const u8,
reason_len: size_t,
) -> c_int {
let reason = unsafe { slice::from_raw_parts(reason, reason_len) };
match conn.close(app, err, reason) {
Ok(_) => 0,
Err(e) => e.to_c() as c_int,
}
}
#[no_mangle]
pub extern fn quiche_conn_timeout_as_nanos(conn: &mut Connection) -> u64 {
match conn.timeout() {
Some(timeout) => timeout.as_nanos() as u64,
None => std::u64::MAX,
}
}
#[no_mangle]
pub extern fn quiche_conn_timeout_as_millis(conn: &mut Connection) -> u64 {
match conn.timeout() {
Some(timeout) => timeout.as_millis() as u64,
None => std::u64::MAX,
}
}
#[no_mangle]
pub extern fn quiche_conn_on_timeout(conn: &mut Connection) {
conn.on_timeout()
}
#[no_mangle]
pub extern fn quiche_conn_application_proto(
conn: &mut Connection, out: &mut *const u8, out_len: &mut size_t,
) {
let proto = conn.application_proto();
*out = proto.as_ptr();
*out_len = proto.len();
}
#[no_mangle]
pub extern fn quiche_conn_is_established(conn: &mut Connection) -> bool {
conn.is_established()
}
#[no_mangle]
pub extern fn quiche_conn_is_in_early_data(conn: &mut Connection) -> bool {
conn.is_in_early_data()
}
#[no_mangle]
pub extern fn quiche_conn_is_closed(conn: &mut Connection) -> bool {
conn.is_closed()
}
#[no_mangle]
pub extern fn quiche_stream_iter_next(
iter: &mut StreamIter, stream_id: *mut u64,
) -> bool {
if let Some(v) = iter.next() {
unsafe { *stream_id = v };
return true;
}
false
}
#[no_mangle]
pub extern fn quiche_stream_iter_free(iter: *mut StreamIter) {
unsafe { Box::from_raw(iter) };
}
#[repr(C)]
pub struct Stats {
pub recv: usize,
pub sent: usize,
pub lost: usize,
pub rtt: u64,
pub cwnd: usize,
pub delivery_rate: u64,
}
#[no_mangle]
pub extern fn quiche_conn_stats(conn: &Connection, out: &mut Stats) {
let stats = conn.stats();
out.recv = stats.recv;
out.sent = stats.sent;
out.lost = stats.lost;
out.rtt = stats.rtt.as_nanos() as u64;
out.cwnd = stats.cwnd;
out.delivery_rate = stats.delivery_rate;
}
#[no_mangle]
pub extern fn quiche_conn_dgram_max_writable_len(conn: &Connection) -> ssize_t {
match conn.dgram_max_writable_len() {
None => Error::Done.to_c(),
Some(v) => v as ssize_t,
}
}
#[no_mangle]
pub extern fn quiche_conn_dgram_send(
conn: &mut Connection, buf: *const u8, buf_len: size_t,
) -> ssize_t {
if buf_len > <ssize_t>::max_value() as usize {
panic!("The provided buffer is too large");
}
let buf = unsafe { slice::from_raw_parts(buf, buf_len) };
match conn.dgram_send(buf) {
Ok(_) => buf_len as ssize_t,
Err(e) => e.to_c(),
}
}
#[no_mangle]
pub extern fn quiche_conn_dgram_recv(
conn: &mut Connection, out: *mut u8, out_len: size_t,
) -> ssize_t {
if out_len > <ssize_t>::max_value() as usize {
panic!("The provided buffer is too large");
}
let out = unsafe { slice::from_raw_parts_mut(out, out_len) };
let out_len = match conn.dgram_recv(out) {
Ok(v) => v,
Err(e) => return e.to_c(),
};
out_len as ssize_t
}
#[no_mangle]
pub extern fn quiche_conn_dgram_purge_outgoing(
conn: &mut Connection, f: extern fn(*const u8, size_t) -> bool,
) {
conn.dgram_purge_outgoing(|d: &[u8]| -> bool {
let ptr: *const u8 = d.as_ptr();
let len: size_t = d.len();
f(ptr, len)
});
}
#[no_mangle]
pub extern fn quiche_conn_free(conn: *mut Connection) {
unsafe { Box::from_raw(conn) };
}