Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
18 changes: 18 additions & 0 deletions zerompk/tests/derive.rs
Original file line number Diff line number Diff line change
Expand Up @@ -124,6 +124,14 @@ enum BasicLevel {
High,
}

#[derive(ToMessagePack, FromMessagePack, Debug, PartialEq)]
#[msgpack(c_enum)]
#[repr(i8)]
enum ReprI8Enum {
MinusOne = -1,
Zero = 0,
}

#[derive(ToMessagePack, FromMessagePack, Debug, PartialEq)]
struct RecursiveNode {
next: Option<Box<RecursiveNode>>,
Expand Down Expand Up @@ -595,6 +603,16 @@ fn derive_c_enum_with_implicit_discriminant() {
assert_eq!(decoded, value);
}

#[test]
fn derive_c_enum_with_signed_discriminant() {
let value = ReprI8Enum::MinusOne;
let data = zerompk::to_msgpack_vec(&value).unwrap();
assert_eq!(data, vec![0xff]);

let decoded: ReprI8Enum = zerompk::from_msgpack(&data).unwrap();
assert_eq!(decoded, value);
}

#[test]
fn derive_c_enum_unknown_value_is_error() {
let err = zerompk::from_msgpack::<HttpStatus>(&[0x03]).unwrap_err();
Expand Down
72 changes: 61 additions & 11 deletions zerompk_derive/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -43,12 +43,20 @@ struct TypeConfig {
repr: Option<Repr>,
c_enum: bool,
allow_unknown_fields: bool,
c_enum_repr: CEnumRepr,
}

#[derive(Clone, Copy, PartialEq, Eq)]
enum CEnumRepr {
Signed,
Unsigned,
}

fn parse_type_config_from_attrs(attrs: &[syn::Attribute]) -> Result<TypeConfig> {
let mut repr = None;
let mut c_enum = false;
let mut allow_unknown_fields = false;
let mut c_enum_repr = CEnumRepr::Unsigned;

for attr in attrs {
if !attr.path().is_ident("msgpack") {
Expand Down Expand Up @@ -91,10 +99,31 @@ fn parse_type_config_from_attrs(attrs: &[syn::Attribute]) -> Result<TypeConfig>
})?;
}

if c_enum {
for attr in attrs {
if !attr.path().is_ident("repr") {
continue;
}

attr.parse_nested_meta(|meta| {
if meta.path.is_ident("i8")
|| meta.path.is_ident("i16")
|| meta.path.is_ident("i32")
|| meta.path.is_ident("i64")
|| meta.path.is_ident("isize")
{
c_enum_repr = CEnumRepr::Signed;
}
Ok(())
})?;
}
}

Ok(TypeConfig {
repr,
c_enum,
allow_unknown_fields,
c_enum_repr,
})
}

Expand Down Expand Up @@ -851,7 +880,7 @@ fn expand(input: DeriveInput, kind: DeriveKind) -> Result<proc_macro2::TokenStre
}

if type_cfg.c_enum {
expand_c_enum(&data)?
expand_c_enum(&data, type_cfg.c_enum_repr)?
} else {
expand_enum(&data)?
}
Expand Down Expand Up @@ -1341,7 +1370,7 @@ fn expand_map_struct(data: &DataStruct, allow_unknown_fields: bool) -> Result<Im
Ok(ImplBody { write, read })
}

fn expand_c_enum(data: &DataEnum) -> Result<ImplBody> {
fn expand_c_enum(data: &DataEnum, repr: CEnumRepr) -> Result<ImplBody> {
let mut write_arms = Vec::new();
let mut read_arms = Vec::new();

Expand All @@ -1355,16 +1384,32 @@ fn expand_c_enum(data: &DataEnum) -> Result<ImplBody> {
));
}

write_arms.push(quote! {
Self::#v_ident => {
writer.write_u64(Self::#v_ident as u64)?;
Ok(())
match repr {
CEnumRepr::Signed => {
write_arms.push(quote! {
Self::#v_ident => {
writer.write_i64(Self::#v_ident as i64)?;
Ok(())
}
});

read_arms.push(quote! {
__value if __value == (Self::#v_ident as i64) => Ok(Self::#v_ident)
});
}
});
CEnumRepr::Unsigned => {
write_arms.push(quote! {
Self::#v_ident => {
writer.write_u64(Self::#v_ident as u64)?;
Ok(())
}
});

read_arms.push(quote! {
__value if __value == (Self::#v_ident as u64) => Ok(Self::#v_ident)
});
read_arms.push(quote! {
__value if __value == (Self::#v_ident as u64) => Ok(Self::#v_ident)
});
}
}
}

let write = quote! {
Expand All @@ -1373,8 +1418,13 @@ fn expand_c_enum(data: &DataEnum) -> Result<ImplBody> {
}
};

let read_value = match repr {
CEnumRepr::Signed => quote! { reader.read_i64()? },
CEnumRepr::Unsigned => quote! { reader.read_u64()? },
};

let read = quote! {
let __value = reader.read_u64()?;
let __value = #read_value;
match __value {
#( #read_arms, )*
_ => Err(::zerompk::Error::InvalidMarker(0)),
Expand Down