use crate::{Signature, Type, impl_type_with_repr, serialized::Format};
pub trait Basic: Type {
const SIGNATURE_CHAR: char;
const SIGNATURE_STR: &'static str;
fn alignment(format: Format) -> usize {
Self::SIGNATURE.alignment(format)
}
}
macro_rules! impl_type {
($for:ty) => {
impl Type for $for {
const SIGNATURE: &'static Signature = {
match Self::SIGNATURE_CHAR {
'y' => &Signature::U8,
'b' => &Signature::Bool,
'n' => &Signature::I16,
'q' => &Signature::U16,
'i' => &Signature::I32,
'u' => &Signature::U32,
'x' => &Signature::I64,
't' => &Signature::U64,
'd' => &Signature::F64,
's' => &Signature::Str,
'g' => &Signature::Signature,
'o' => &Signature::ObjectPath,
'v' => &Signature::Variant,
#[cfg(unix)]
'h' => &Signature::Fd,
_ => unreachable!(),
}
};
}
};
}
impl Basic for u8 {
const SIGNATURE_CHAR: char = 'y';
const SIGNATURE_STR: &'static str = "y";
}
impl_type!(u8);
impl Basic for std::num::NonZeroU8 {
const SIGNATURE_CHAR: char = u8::SIGNATURE_CHAR;
const SIGNATURE_STR: &'static str = u8::SIGNATURE_STR;
}
impl_type!(std::num::NonZeroU8);
impl Basic for i8 {
const SIGNATURE_CHAR: char = i16::SIGNATURE_CHAR;
const SIGNATURE_STR: &'static str = i16::SIGNATURE_STR;
}
impl_type!(i8);
impl Basic for std::num::NonZeroI8 {
const SIGNATURE_CHAR: char = i8::SIGNATURE_CHAR;
const SIGNATURE_STR: &'static str = i8::SIGNATURE_STR;
}
impl_type!(std::num::NonZeroI8);
impl Basic for bool {
const SIGNATURE_CHAR: char = 'b';
const SIGNATURE_STR: &'static str = "b";
}
impl_type!(bool);
impl Basic for i16 {
const SIGNATURE_CHAR: char = 'n';
const SIGNATURE_STR: &'static str = "n";
}
impl_type!(i16);
impl Basic for std::num::NonZeroI16 {
const SIGNATURE_CHAR: char = i16::SIGNATURE_CHAR;
const SIGNATURE_STR: &'static str = i16::SIGNATURE_STR;
}
impl_type!(std::num::NonZeroI16);
impl Basic for u16 {
const SIGNATURE_CHAR: char = 'q';
const SIGNATURE_STR: &'static str = "q";
}
impl_type!(u16);
impl Basic for std::num::NonZeroU16 {
const SIGNATURE_CHAR: char = u16::SIGNATURE_CHAR;
const SIGNATURE_STR: &'static str = u16::SIGNATURE_STR;
}
impl_type!(std::num::NonZeroU16);
impl Basic for i32 {
const SIGNATURE_CHAR: char = 'i';
const SIGNATURE_STR: &'static str = "i";
}
impl_type!(i32);
impl Basic for std::num::NonZeroI32 {
const SIGNATURE_CHAR: char = i32::SIGNATURE_CHAR;
const SIGNATURE_STR: &'static str = i32::SIGNATURE_STR;
}
impl_type!(std::num::NonZeroI32);
impl Basic for u32 {
const SIGNATURE_CHAR: char = 'u';
const SIGNATURE_STR: &'static str = "u";
}
impl_type!(u32);
impl Basic for std::num::NonZeroU32 {
const SIGNATURE_CHAR: char = u32::SIGNATURE_CHAR;
const SIGNATURE_STR: &'static str = u32::SIGNATURE_STR;
}
impl_type!(std::num::NonZeroU32);
impl Basic for i64 {
const SIGNATURE_CHAR: char = 'x';
const SIGNATURE_STR: &'static str = "x";
}
impl_type!(i64);
impl Basic for std::num::NonZeroI64 {
const SIGNATURE_CHAR: char = i64::SIGNATURE_CHAR;
const SIGNATURE_STR: &'static str = i64::SIGNATURE_STR;
}
impl_type!(std::num::NonZeroI64);
impl Basic for u64 {
const SIGNATURE_CHAR: char = 't';
const SIGNATURE_STR: &'static str = "t";
}
impl_type!(u64);
impl Basic for std::num::NonZeroU64 {
const SIGNATURE_CHAR: char = u64::SIGNATURE_CHAR;
const SIGNATURE_STR: &'static str = u64::SIGNATURE_STR;
}
impl_type!(std::num::NonZeroU64);
impl Basic for f32 {
const SIGNATURE_CHAR: char = f64::SIGNATURE_CHAR;
const SIGNATURE_STR: &'static str = f64::SIGNATURE_STR;
}
impl_type!(f32);
impl Basic for f64 {
const SIGNATURE_CHAR: char = 'd';
const SIGNATURE_STR: &'static str = "d";
}
impl_type!(f64);
impl Basic for str {
const SIGNATURE_CHAR: char = 's';
const SIGNATURE_STR: &'static str = "s";
}
impl_type!(str);
impl Basic for String {
const SIGNATURE_CHAR: char = 's';
const SIGNATURE_STR: &'static str = "s";
}
impl_type!(String);
impl Basic for char {
const SIGNATURE_CHAR: char = <&str>::SIGNATURE_CHAR;
const SIGNATURE_STR: &'static str = <&str>::SIGNATURE_STR;
}
impl_type!(char);
impl Basic for usize {
const SIGNATURE_CHAR: char = <u64 as Basic>::SIGNATURE_CHAR;
const SIGNATURE_STR: &'static str = <u64 as Basic>::SIGNATURE_STR;
}
impl_type_with_repr! {
usize => u64 {
usize {
samples = [usize::MAX, usize::MIN],
repr(n) = n as u64,
}
}
}
impl_type_with_repr! {
isize => i64 {
isize {
samples = [isize::MAX, isize::MIN],
repr(n) = n as i64,
}
}
}
impl Basic for isize {
const SIGNATURE_CHAR: char = <i64 as Basic>::SIGNATURE_CHAR;
const SIGNATURE_STR: &'static str = <i64 as Basic>::SIGNATURE_STR;
}
macro_rules! impl_basic_for_deref {
(
$type:ty,
<$($desc:tt)+
) => {
impl <$($desc)+ {
const SIGNATURE_CHAR: char = <$type>::SIGNATURE_CHAR;
const SIGNATURE_STR: &'static str = <$type>::SIGNATURE_STR;
}
};
}
impl_basic_for_deref!(T, <T: ?Sized + Basic> Basic for &T);
impl_basic_for_deref!(T, <T: ?Sized + Basic> Basic for &mut T);
impl_basic_for_deref!(T, <T: ?Sized + Basic + ToOwned> Basic for std::borrow::Cow<'_,T>);
impl_basic_for_deref!(T, <T: ?Sized + Basic> Basic for std::sync::Arc<T>);
impl_basic_for_deref!(T, <T: ?Sized + Basic> Basic for std::sync::Weak<T>);
impl_basic_for_deref!(T, <T: ?Sized + Basic> Basic for std::sync::Mutex<T>);
impl_basic_for_deref!(T, <T: ?Sized + Basic> Basic for std::sync::RwLock<T>);
impl_basic_for_deref!(T, <T: ?Sized + Basic> Basic for std::boxed::Box<T>);
impl_basic_for_deref!(T, <T: ?Sized + Basic> Basic for std::rc::Rc<T>);
impl_basic_for_deref!(T, <T: ?Sized + Basic> Basic for std::rc::Weak<T>);
impl_basic_for_deref!(T, <T: ?Sized + Basic> Basic for std::cell::Cell<T>);
impl_basic_for_deref!(T, <T: ?Sized + Basic> Basic for std::cell::RefCell<T>);
use std::{
cmp::Reverse,
num::{Saturating, Wrapping},
};
macro_rules! impl_basic_for_wrapper {
($($wrapper:ident<$T:ident>),+) => {
$(
impl<$T: Basic> Basic for $wrapper<$T> {
const SIGNATURE_CHAR: char = <$T>::SIGNATURE_CHAR;
const SIGNATURE_STR: &'static str = <$T>::SIGNATURE_STR;
}
)+
};
}
impl_basic_for_wrapper!(Wrapping<T>, Saturating<T>, Reverse<T>);
macro_rules! atomic_impl {
($($ty:ident $size:expr => $primitive:ident)*) => {
$(
#[cfg(target_has_atomic = $size)]
impl Basic for $ty {
const SIGNATURE_CHAR: char = <$primitive as Basic>::SIGNATURE_CHAR;
const SIGNATURE_STR: &'static str = <$primitive as Basic>::SIGNATURE_STR;
}
impl_type!($ty);
)*
}
}
use std::sync::atomic::{
AtomicBool, AtomicI8, AtomicI16, AtomicI32, AtomicIsize, AtomicU8, AtomicU16, AtomicU32,
AtomicUsize,
};
#[cfg(target_has_atomic = "64")]
use std::sync::atomic::{AtomicI64, AtomicU64};
atomic_impl! {
AtomicBool "8" => bool
AtomicI8 "8" => i8
AtomicI16 "16" => i16
AtomicI32 "32" => i32
AtomicIsize "ptr" => isize
AtomicU8 "8" => u8
AtomicU16 "16" => u16
AtomicU32 "32" => u32
AtomicUsize "ptr" => usize
}
#[cfg(target_has_atomic = "64")]
atomic_impl! {
AtomicI64 "64" => i64
AtomicU64 "64" => u64
}
#[cfg(feature = "heapless")]
impl<const CAP: usize> Basic for heapless::String<CAP> {
const SIGNATURE_CHAR: char = <&str as Basic>::SIGNATURE_CHAR;
const SIGNATURE_STR: &'static str = <&str as Basic>::SIGNATURE_STR;
}