//@ check-pass //@ compile-flags: -Z span-debug //@ aux-build:test-macros.rs // Regression test for issue #75930 // Tests that we cfg-strip all targets before invoking // a derive macro // FIXME: We currently lose spans here (see issue #43081) #![no_std] // Don't load unnecessary hygiene information from std extern crate std; #[macro_use] extern crate test_macros; // Note: the expected output contains this sequence: // ``` // Punct { // ch: '<', // spacing: Joint, // span: $DIR/issue-75930-derive-cfg.rs:25:11: 25:12 (#0), // }, // Ident { // ident: "B", // span: $DIR/issue-75930-derive-cfg.rs:25:29: 25:30 (#0), // }, // ``` // It's surprising to see a `Joint` token tree followed by an `Ident` token // tree, because `Joint` is supposed to only be used if the following token is // `Punct`. // // It is because of this code from below: // ``` // struct Foo<#[cfg(FALSE)] A, B> // ``` // When the token stream is formed during parsing, `<` is followed immediately // by `#`, which is punctuation, so it is marked `Joint`. But before being // passed to the proc macro it is rewritten to this: // ``` // struct Foo // ``` // But the `Joint` marker on the `<` is not updated. Perhaps it should be // corrected before being passed to the proc macro? But a prior attempt to do // that kind of correction caused the problem seen in #76399, so maybe not. #[print_helper(a)] //~ WARN derive helper attribute is used before it is introduced //~| WARN derive helper attribute is used before it is introduced //~| WARN this was previously accepted //~| WARN this was previously accepted #[cfg_attr(not(FALSE), allow(dead_code))] #[print_attr] #[derive(Print)] #[print_helper(b)] struct Foo<#[cfg(FALSE)] A, B> { #[cfg(FALSE)] first: String, #[cfg_attr(FALSE, deny(warnings))] second: bool, third: [u8; { #[cfg(FALSE)] struct Bar; #[cfg(not(FALSE))] struct Inner; #[cfg(FALSE)] let a = 25; match true { #[cfg(FALSE)] true => {}, #[cfg_attr(not(FALSE), allow(warnings))] false => {}, _ => {} }; #[print_helper(should_be_removed)] fn removed_fn() { #![cfg(FALSE)] } #[print_helper(c)] #[cfg(not(FALSE))] fn kept_fn() { #![cfg(not(FALSE))] let my_val = true; } enum TupleEnum { Foo( #[cfg(FALSE)] u8, #[cfg(FALSE)] bool, #[cfg(not(FALSE))] i32, #[cfg(FALSE)] String, u8 ) } struct TupleStruct( #[cfg(FALSE)] String, #[cfg(not(FALSE))] i32, #[cfg(FALSE)] bool, u8 ); fn plain_removed_fn() { #![cfg_attr(not(FALSE), cfg(FALSE))] } 0 }], #[print_helper(d)] fourth: B } fn main() {}