use core::fmt::{self, Display, Formatter};
use proc_macro2::Span;
use syn::{Attribute, DeriveInput, Error, Lit, Meta, NestedMeta};
pub struct Config<Repr: KindRepr> {
pub allowed_combinations_message: &'static str,
pub derive_unaligned: bool,
pub allowed_combinations: &'static [&'static [Repr]],
pub disallowed_but_legal_combinations: &'static [&'static [Repr]],
}
impl<R: KindRepr> Config<R> {
pub fn validate_reprs(&self, input: &DeriveInput) -> Result<Vec<R>, Vec<Error>> {
let mut reprs = reprs(&input.attrs)?;
reprs.sort();
if self.derive_unaligned && reprs.iter().any(KindRepr::is_align_gt_one) {
return Err(vec![Error::new_spanned(
input,
"cannot derive Unaligned with repr(align(N > 1))",
)]);
}
reprs.retain(|repr: &R| !repr.is_align());
if reprs.is_empty() {
Err(vec![Error::new(Span::call_site(), "must have a non-align #[repr(...)] attribute in order to guarantee this type's memory layout")])
} else if self.allowed_combinations.contains(&reprs.as_slice()) {
Ok(reprs)
} else if self.disallowed_but_legal_combinations.contains(&reprs.as_slice()) {
Err(vec![Error::new_spanned(input, self.allowed_combinations_message)])
} else {
Err(vec![Error::new_spanned(input, "conflicting representation hints")])
}
}
}
pub trait KindRepr: 'static + Sized + Ord {
fn is_align(&self) -> bool;
fn is_align_gt_one(&self) -> bool;
fn parse(meta: &NestedMeta) -> syn::Result<Self>;
}
macro_rules! define_kind_specific_repr {
($type_name:expr, $repr_name:ident, $($repr_variant:ident),*) => {
#[derive(Copy, Clone, Debug, Eq, PartialEq)]
pub enum $repr_name {
$($repr_variant,)*
Align(u64),
}
impl KindRepr for $repr_name {
fn is_align(&self) -> bool {
match self {
$repr_name::Align(_) => true,
_ => false,
}
}
fn is_align_gt_one(&self) -> bool {
match self {
$repr_name::Align(n) => n > &1,
_ => false,
}
}
fn parse(meta: &NestedMeta) -> syn::Result<$repr_name> {
match Repr::from_nested_meta(meta)? {
$(Repr::$repr_variant => Ok($repr_name::$repr_variant),)*
Repr::Align(u) => Ok($repr_name::Align(u)),
_ => Err(Error::new_spanned(meta, concat!("unsupported representation for deriving FromBytes, AsBytes, or Unaligned on ", $type_name)))
}
}
}
impl PartialOrd for $repr_name {
fn partial_cmp(&self, other: &Self) -> Option<core::cmp::Ordering> {
Some(self.cmp(other))
}
}
impl Ord for $repr_name {
fn cmp(&self, other: &Self) -> core::cmp::Ordering {
format!("{:?}", self).cmp(&format!("{:?}", other))
}
}
impl core::fmt::Display for $repr_name {
fn fmt(&self, f: &mut core::fmt::Formatter) -> core::fmt::Result {
match self {
$($repr_name::$repr_variant => Repr::$repr_variant,)*
$repr_name::Align(u) => Repr::Align(*u),
}.fmt(f)
}
}
}
}
define_kind_specific_repr!("a struct", StructRepr, C, Transparent, Packed);
define_kind_specific_repr!(
"an enum", EnumRepr, C, U8, U16, U32, U64, Usize, I8, I16, I32, I64, Isize
);
#[derive(Copy, Clone, Eq, PartialEq)]
pub enum Repr {
U8,
U16,
U32,
U64,
Usize,
I8,
I16,
I32,
I64,
Isize,
C,
Transparent,
Packed,
Align(u64),
}
impl Repr {
fn from_nested_meta(meta: &NestedMeta) -> Result<Repr, Error> {
match meta {
NestedMeta::Meta(Meta::Word(word)) => match format!("{}", word).as_str() {
"u8" => return Ok(Repr::U8),
"u16" => return Ok(Repr::U16),
"u32" => return Ok(Repr::U32),
"u64" => return Ok(Repr::U64),
"usize" => return Ok(Repr::Usize),
"i8" => return Ok(Repr::I8),
"i16" => return Ok(Repr::I16),
"i32" => return Ok(Repr::I32),
"i64" => return Ok(Repr::I64),
"isize" => return Ok(Repr::Isize),
"C" => return Ok(Repr::C),
"transparent" => return Ok(Repr::Transparent),
"packed" => return Ok(Repr::Packed),
_ => {}
},
NestedMeta::Meta(Meta::List(list)) => {
if let [&NestedMeta::Literal(Lit::Int(ref n))] =
list.nested.iter().collect::<Vec<_>>().as_slice()
{
return Ok(Repr::Align(n.value()));
}
}
_ => {}
}
Err(Error::new_spanned(meta, "unrecognized representation hint"))
}
}
impl Display for Repr {
fn fmt(&self, f: &mut Formatter) -> Result<(), fmt::Error> {
if let Repr::Align(n) = self {
return write!(f, "repr(align({}))", n);
}
write!(
f,
"repr({})",
match self {
Repr::U8 => "u8",
Repr::U16 => "u16",
Repr::U32 => "u32",
Repr::U64 => "u64",
Repr::Usize => "usize",
Repr::I8 => "i8",
Repr::I16 => "i16",
Repr::I32 => "i32",
Repr::I64 => "i64",
Repr::Isize => "isize",
Repr::C => "C",
Repr::Transparent => "transparent",
Repr::Packed => "packed",
_ => unreachable!(),
}
)
}
}
fn reprs<R: KindRepr>(attrs: &[Attribute]) -> Result<Vec<R>, Vec<Error>> {
let mut reprs = Vec::new();
let mut errors = Vec::new();
for attr in attrs {
if let Some(Meta::List(meta_list)) = attr.interpret_meta() {
if format!("{}", meta_list.ident) == "repr" {
for nested_meta in &meta_list.nested {
match R::parse(nested_meta) {
Ok(repr) => reprs.push(repr),
Err(err) => errors.push(err),
}
}
}
}
}
if !errors.is_empty() {
return Err(errors);
}
Ok(reprs)
}