|
| 1 | +// force-host |
| 2 | +// no-prefer-dynamic |
| 3 | + |
| 4 | +#![crate_type = "proc-macro"] |
| 5 | + |
| 6 | +extern crate proc_macro; |
| 7 | + |
| 8 | +use proc_macro::TokenStream; |
| 9 | + |
| 10 | +macro_rules! checker { |
| 11 | + ($attr_name:ident, $expected:literal) => { |
| 12 | + #[proc_macro_attribute] |
| 13 | + pub fn $attr_name(attr: TokenStream, input: TokenStream) -> TokenStream { |
| 14 | + assert!(attr.to_string().is_empty()); |
| 15 | + assert_eq!(input.to_string(), $expected); |
| 16 | + TokenStream::new() |
| 17 | + } |
| 18 | + } |
| 19 | +} |
| 20 | + |
| 21 | +checker!(attr_extern, r#"extern "C" { |
| 22 | + fn ffi(#[a1] arg1: i32, #[a2] ...); |
| 23 | +}"#); |
| 24 | +checker!(attr_extern_cvar, r#"unsafe extern "C" fn cvar(arg1: i32, #[a1] mut args: ...) { }"#); |
| 25 | +checker!(attr_alias, "type Alias = fn(#[a1] u8, #[a2] ...);"); |
| 26 | +checker!(attr_free, "fn free(#[a1] arg1: u8) { let lam = |#[a2] W(x), #[a3] y| (); }"); |
| 27 | +checker!(attr_inherent_1, "fn inherent1(#[a1] self, #[a2] arg1: u8) { }"); |
| 28 | +checker!(attr_inherent_2, "fn inherent2(#[a1] &self, #[a2] arg1: u8) { }"); |
| 29 | +checker!(attr_inherent_3, "fn inherent3<'a>(#[a1] &'a mut self, #[a2] arg1: u8) { }"); |
| 30 | +checker!(attr_inherent_4, "fn inherent4<'a>(#[a1] self: Box<Self>, #[a2] arg1: u8) { }"); |
| 31 | +checker!(attr_trait_1, "fn trait1(#[a1] self, #[a2] arg1: u8);"); |
| 32 | +checker!(attr_trait_2, "fn trait2(#[a1] &self, #[a2] arg1: u8);"); |
| 33 | +checker!(attr_trait_3, "fn trait3<'a>(#[a1] &'a mut self, #[a2] arg1: u8);"); |
| 34 | +checker!(attr_trait_4, "fn trait4<'a>(#[a1] self: Box<Self>, #[a2] arg1: u8, #[a3] Vec<u8>);"); |
0 commit comments