#![recursion_limit = "256"]
extern crate proc_macro;
extern crate syn;
#[macro_use]
extern crate quote;
use proc_macro::TokenStream;
#[proc_macro_attribute]
pub fn ctor(_attribute: TokenStream, function: TokenStream) -> TokenStream {
let item: syn::Item = syn::parse_macro_input!(function);
if let syn::Item::Fn(function) = item {
validate_item("ctor", &function);
let syn::ItemFn {
attrs,
block,
vis,
sig:
syn::Signature {
ident,
unsafety,
constness,
abi,
..
},
..
} = function;
let ctor_ident =
syn::parse_str::<syn::Ident>(format!("{}___rust_ctor___ctor", ident).as_ref())
.expect("Unable to create identifier");
let output = quote!(
#[cfg(not(any(target_os = "linux", target_os = "android", target_os = "freebsd", target_os = "netbsd", target_os = "openbsd", target_os = "illumos", target_os = "macos", target_os = "ios", windows)))]
compile_error!("#[ctor] is not supported on the current target");
#(#attrs)*
#vis #unsafety extern #abi #constness fn #ident() #block
#[used]
#[allow(non_upper_case_globals)]
#[cfg_attr(any(target_os = "linux", target_os = "android"), link_section = ".init_array")]
#[cfg_attr(target_os = "freebsd", link_section = ".init_array")]
#[cfg_attr(target_os = "netbsd", link_section = ".init_array")]
#[cfg_attr(target_os = "openbsd", link_section = ".init_array")]
#[cfg_attr(target_os = "illumos", link_section = ".init_array")]
#[cfg_attr(any(target_os = "macos", target_os = "ios"), link_section = "__DATA,__mod_init_func")]
#[cfg_attr(windows, link_section = ".CRT$XCU")]
static #ctor_ident
:
unsafe extern "C" fn() =
{
#[cfg_attr(any(target_os = "linux", target_os = "android"), link_section = ".text.startup")]
unsafe extern "C" fn #ctor_ident() { #ident() };
#ctor_ident
}
;
);
output.into()
} else if let syn::Item::Static(var) = item {
let syn::ItemStatic {
ident,
mutability,
expr,
attrs,
ty,
vis,
..
} = var;
if let Some(_) = mutability {
panic!("#[ctor]-annotated static objects must not be mutable");
}
if attrs.iter().any(|attr| {
attr.path
.segments
.iter()
.any(|segment| segment.ident == "no_mangle")
}) {
panic!("#[ctor]-annotated static objects do not support #[no_mangle]");
}
let ctor_ident =
syn::parse_str::<syn::Ident>(format!("{}___rust_ctor___ctor", ident).as_ref())
.expect("Unable to create identifier");
let storage_ident =
syn::parse_str::<syn::Ident>(format!("{}___rust_ctor___storage", ident).as_ref())
.expect("Unable to create identifier");
let output = quote!(
#[cfg(not(any(target_os = "linux", target_os = "android", target_os = "freebsd", target_os = "netbsd", target_os = "openbsd", target_os = "illumos", target_os = "macos", target_os = "ios", windows)))]
compile_error!("#[ctor] is not supported on the current target");
static mut #storage_ident: Option<#ty> = None;
#[doc(hidden)]
#[allow(non_camel_case_types)]
#vis struct #ident<T> {
_data: core::marker::PhantomData<T>
}
#(#attrs)*
#vis static #ident: #ident<#ty> = #ident {
_data: core::marker::PhantomData::<#ty>
};
impl core::ops::Deref for #ident<#ty> {
type Target = #ty;
fn deref(&self) -> &'static #ty {
unsafe {
#storage_ident.as_ref().unwrap()
}
}
}
#[used]
#[allow(non_upper_case_globals)]
#[cfg_attr(any(target_os = "linux", target_os = "android"), link_section = ".init_array")]
#[cfg_attr(target_os = "freebsd", link_section = ".init_array")]
#[cfg_attr(target_os = "netbsd", link_section = ".init_array")]
#[cfg_attr(target_os = "openbsd", link_section = ".init_array")]
#[cfg_attr(target_os = "illumos", link_section = ".init_array")]
#[cfg_attr(any(target_os = "macos", target_os = "ios"), link_section = "__DATA,__mod_init_func")]
#[cfg_attr(windows, link_section = ".CRT$XCU")]
static #ctor_ident
:
unsafe extern "C" fn() = {
#[cfg_attr(any(target_os = "linux", target_os = "android"), link_section = ".text.startup")]
unsafe extern "C" fn initer() {
#storage_ident = Some(#expr);
}; initer }
;
);
output.into()
} else {
panic!("#[ctor] items must be functions or static globals");
}
}
#[proc_macro_attribute]
pub fn dtor(_attribute: TokenStream, function: TokenStream) -> TokenStream {
let function: syn::ItemFn = syn::parse_macro_input!(function);
validate_item("dtor", &function);
let syn::ItemFn {
attrs,
block,
vis,
sig:
syn::Signature {
ident,
unsafety,
constness,
abi,
..
},
..
} = function;
let mod_ident =
syn::parse_str::<syn::Ident>(format!("{}___rust_dtor___mod", ident).as_ref())
.expect("Unable to create identifier");
let dtor_ident =
syn::parse_str::<syn::Ident>(format!("{}___rust_dtor___dtor", ident).as_ref())
.expect("Unable to create identifier");
let output = quote!(
#[cfg(not(any(target_os = "linux", target_os = "android", target_os = "freebsd", target_os = "netbsd", target_os = "openbsd", target_os = "illumos", target_os = "macos", target_os = "ios", windows)))]
compile_error!("#[dtor] is not supported on the current target");
#(#attrs)*
#vis #unsafety extern #abi #constness fn #ident() #block
#[cfg(not(any(
target_os = "macos",
target_os = "ios",
)))]
mod #mod_ident {
use super::#ident;
extern "C" {
fn atexit(cb: unsafe extern fn());
}
#[used]
#[allow(non_upper_case_globals)]
#[cfg_attr(any(target_os = "linux", target_os = "android"), link_section = ".init_array")]
#[cfg_attr(target_os = "freebsd", link_section = ".init_array")]
#[cfg_attr(target_os = "netbsd", link_section = ".init_array")]
#[cfg_attr(target_os = "openbsd", link_section = ".init_array")]
#[cfg_attr(target_os = "illumos", link_section = ".init_array")]
#[cfg_attr(windows, link_section = ".CRT$XCU")]
static __dtor_export
:
unsafe extern "C" fn() =
{
#[cfg_attr(any(target_os = "linux", target_os = "android"), link_section = ".text.exit")]
unsafe extern "C" fn #dtor_ident() { #ident() };
#[cfg_attr(any(target_os = "linux", target_os = "android"), link_section = ".text.startup")]
unsafe extern fn __dtor_atexit() {
atexit(#dtor_ident);
};
__dtor_atexit
};
}
#[cfg(any(
target_os = "macos",
target_os = "ios",
))]
mod #mod_ident {
use super::#ident;
#[used]
#[allow(non_upper_case_globals)]
#[cfg_attr(any(target_os = "macos", target_os = "ios"), link_section = "__DATA,__mod_term_func")]
static __dtor_export
:
unsafe extern "C" fn() =
{
unsafe extern fn __dtor() { #ident() };
__dtor
};
}
);
output.into()
}
fn validate_item(typ: &str, item: &syn::ItemFn) {
let syn::ItemFn { vis, sig, .. } = item;
match vis {
syn::Visibility::Inherited => {}
_ => panic!("#[{}] methods must not have visibility modifiers", typ),
}
if sig.inputs.len() > 0 {
panic!("#[{}] methods may not have parameters", typ);
}
match sig.output {
syn::ReturnType::Default => {}
_ => panic!("#[{}] methods must not have return types", typ),
}
}