#![no_std]
#![warn(rust_2018_idioms)]
#![deny(clippy::std_instead_of_core)]
#![deny(clippy::std_instead_of_alloc)]
#![allow(dead_code)]
#![allow(clippy::bool_to_int_with_if)]
#![forbid(unsafe_code)]
#[macro_use]
extern crate alloc;
extern crate std;
use alloc::vec::Vec;
use bytes::Bytes;
use core::fmt;
use core::net::{IpAddr, SocketAddr};
use core::ops;
use std::time::Instant;
mod association;
pub use crate::association::Association;
pub use crate::association::AssociationError;
pub use crate::association::Event;
pub use crate::association::stats::AssociationStats;
pub use crate::association::stream::{ReliabilityType, Stream, StreamEvent, StreamId, StreamState};
pub(crate) mod chunk;
pub use crate::chunk::ErrorCauseCode;
pub use crate::chunk::chunk_payload_data::{ChunkPayloadData, PayloadProtocolIdentifier};
mod config;
pub use crate::config::{
ClientConfig, DEFAULT_SCTP_PORT, EndpointConfig, MAX_SNAP_INIT_BYTES, ServerConfig,
TransportConfig, generate_snap_token,
};
mod endpoint;
pub use crate::endpoint::{
AidCollisionKind, AssociationHandle, ConnectError, DatagramEvent, Endpoint, SnapError, SnapSide,
};
mod error;
pub use crate::error::Error;
mod packet;
mod shared;
pub use crate::shared::{AssociationEvent, AssociationId, EcnCodepoint, EndpointEvent};
pub(crate) mod param;
pub(crate) mod queue;
pub use crate::queue::reassembly_queue::{Chunk, Chunks};
pub(crate) mod util;
#[cfg(feature = "_fuzz")]
pub mod _fuzz {
use bytes::BytesMut;
use crate::packet::Packet;
use crate::util::generate_packet_checksum;
pub fn fuzz_packet_unmarshal(data: &[u8]) {
if data.len() < 12 {
return;
}
let mut buf = BytesMut::from(data);
buf[8] = 0;
buf[9] = 0;
buf[10] = 0;
buf[11] = 0;
let raw = buf.freeze();
let checksum = generate_packet_checksum(&raw);
let mut buf = BytesMut::from(raw.as_ref());
buf[8..12].copy_from_slice(&checksum.to_le_bytes());
let raw = buf.freeze();
let _ = Packet::unmarshal(&raw);
}
}
#[derive(Debug, Copy, Clone, Eq, PartialEq, Ord, PartialOrd, Hash, Default)]
pub enum Side {
#[default]
Client = 0,
Server = 1,
}
impl fmt::Display for Side {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
let s = match *self {
Side::Client => "Client",
Side::Server => "Server",
};
write!(f, "{}", s)
}
}
impl Side {
#[inline]
pub fn is_client(self) -> bool {
self == Side::Client
}
#[inline]
pub fn is_server(self) -> bool {
self == Side::Server
}
}
impl ops::Not for Side {
type Output = Side;
fn not(self) -> Side {
match self {
Side::Client => Side::Server,
Side::Server => Side::Client,
}
}
}
use crate::packet::PartialDecode;
#[derive(Debug)]
pub enum Payload {
PartialDecode(PartialDecode),
RawEncode(Vec<Bytes>),
}
#[derive(Debug)]
pub struct Transmit {
pub now: Instant,
pub remote: SocketAddr,
pub ecn: Option<EcnCodepoint>,
pub local_ip: Option<IpAddr>,
pub payload: Payload,
}
#[cfg(test)]
mod test {
use alloc::sync::Arc;
use super::*;
#[test]
fn ensure_send_sync() {
fn is_send_sync(_a: impl Send + Sync) {}
let c = EndpointConfig::new();
let e = Endpoint::new(Arc::new(c), None);
is_send_sync(e);
let a = Association::default();
is_send_sync(a);
}
}