// SPDX-License-Identifier: (BSD-2-Clause OR Apache-2.0) OR MIT use proc_macro2::{Span, TokenStream}; use syn::{Data, DataEnum, DataStruct, DataUnion, Error}; use crate::{ repr::{EnumRepr, StructUnionRepr}, util::{Ctx, FieldBounds, ImplBlockBuilder, Trait}, }; pub(crate) fn derive_unaligned(ctx: &Ctx, _top_level: Trait) -> Result { match &ctx.ast.data { Data::Struct(strct) => derive_unaligned_struct(ctx, strct), Data::Enum(enm) => derive_unaligned_enum(ctx, enm), Data::Union(unn) => derive_unaligned_union(ctx, unn), } } /// A struct is `Unaligned` if: /// - `repr(align)` is no more than 1 and either /// - `repr(C)` or `repr(transparent)` and /// - all fields `Unaligned` /// - `repr(packed)` fn derive_unaligned_struct(ctx: &Ctx, strct: &DataStruct) -> Result { let repr = StructUnionRepr::from_attrs(&ctx.ast.attrs)?; repr.unaligned_validate_no_align_gt_1()?; let field_bounds = if repr.is_packed_1() { FieldBounds::None } else if repr.is_c() || repr.is_transparent() { FieldBounds::ALL_SELF } else { return ctx.error_or_skip(Error::new( Span::call_site(), "must have #[repr(C)], #[repr(transparent)], or #[repr(packed)] attribute in order to guarantee this type's alignment", )); }; Ok(ImplBlockBuilder::new(ctx, strct, Trait::Unaligned, field_bounds).build()) } /// An enum is `Unaligned` if: /// - No `repr(align(N > 1))` /// - `repr(u8)` or `repr(i8)` fn derive_unaligned_enum(ctx: &Ctx, enm: &DataEnum) -> Result { let repr = EnumRepr::from_attrs(&ctx.ast.attrs)?; repr.unaligned_validate_no_align_gt_1()?; if !repr.is_u8() && !repr.is_i8() { return ctx.error_or_skip(Error::new( Span::call_site(), "must have #[repr(u8)] or #[repr(i8)] attribute in order to guarantee this type's alignment", )); } Ok(ImplBlockBuilder::new(ctx, enm, Trait::Unaligned, FieldBounds::ALL_SELF).build()) } /// Like structs, a union is `Unaligned` if: /// - `repr(align)` is no more than 1 and either /// - `repr(C)` or `repr(transparent)` and /// - all fields `Unaligned` /// - `repr(packed)` fn derive_unaligned_union(ctx: &Ctx, unn: &DataUnion) -> Result { let repr = StructUnionRepr::from_attrs(&ctx.ast.attrs)?; repr.unaligned_validate_no_align_gt_1()?; let field_type_trait_bounds = if repr.is_packed_1() { FieldBounds::None } else if repr.is_c() || repr.is_transparent() { FieldBounds::ALL_SELF } else { return ctx.error_or_skip(Error::new( Span::call_site(), "must have #[repr(C)], #[repr(transparent)], or #[repr(packed)] attribute in order to guarantee this type's alignment", )); }; Ok(ImplBlockBuilder::new(ctx, unn, Trait::Unaligned, field_type_trait_bounds).build()) }