use proc_macro::TokenStream;
use proc_macro2::TokenStream as TokenStream2;
use quote::{quote, ToTokens};
use syn::parse_macro_input;
use crate::{cargo, construct};
use self::input::Input;
mod input;
pub(crate) fn expand(input: TokenStream) -> TokenStream {
let bitflags_input = TokenStream2::from(input.clone());
let input = parse_macro_input!(input as Input);
let format_string = format!(
"{{={}:__internal_bitflags_{}@{}@{}@{}}}",
input.ty().to_token_stream(),
input.ident(),
cargo::package_name(),
construct::crate_local_disambiguator(),
cargo::crate_name(),
);
let format_tag = construct::interned_string(&format_string, "bitflags", false);
let ident = input.ident();
let ty = input.ty();
let flag_statics = codegen_flag_statics(&input);
quote!(
const _: () = {
fn assert<T: defmt::export::UnsignedInt>() {}
assert::<#ty>;
#(#flag_statics)*
};
defmt::export::bitflags! {
#bitflags_input
}
impl defmt::Format for #ident {
fn format(&self, f: defmt::Formatter) {
defmt::unreachable!()
}
fn _format_tag() -> defmt::Str {
#format_tag
}
fn _format_data(&self) {
defmt::export::#ty(&self.bits());
}
}
)
.into()
}
fn codegen_flag_statics(input: &Input) -> Vec<TokenStream2> {
input
.flags()
.enumerate()
.map(|(i, flag)| {
let cfg_attrs = flag.cfg_attrs();
let var_name = flag.ident();
let struct_name = input.ident();
let repr_ty = input.ty();
let sym_name = construct::mangled_symbol_name(
"bitflags_value",
&format!("{}::{i}::{}", input.ident(), flag.ident()),
);
quote! {
#(#cfg_attrs)*
#[cfg_attr(target_os = "macos", link_section = ".defmt,end")]
#[cfg_attr(not(target_os = "macos"), link_section = ".defmt.end")]
#[export_name = #sym_name]
static #var_name: u128 = {
let coerced_value: #repr_ty = #struct_name::#var_name.bits;
coerced_value as u128
};
}
})
.collect::<Vec<_>>()
}