| //@ aux-build:expand-expr.rs |
| // no-remap-src-base: check_expand_expr_file!() fails when enabled. |
| |
| #![feature(concat_bytes)] |
| extern crate expand_expr; |
| |
| use expand_expr::{ |
| check_expand_expr_file, echo_pm, expand_expr_fail, expand_expr_is, recursive_expand, |
| }; |
| |
| // Check builtin macros can be expanded. |
| |
| expand_expr_is!(13u32, line!()); |
| expand_expr_is!(24u32, column!()); |
| |
| expand_expr_is!("Hello, World!", concat!("Hello, ", "World", "!")); |
| expand_expr_is!("int10floats5.3booltrue", concat!("int", 10, "floats", 5.3, "bool", true)); |
| expand_expr_is!("Hello", concat!(r##"Hello"##)); |
| |
| expand_expr_is!("Included file contents\n", include_str!("auxiliary/included-file.txt")); |
| expand_expr_is!(b"Included file contents\n", include_bytes!("auxiliary/included-file.txt")); |
| |
| expand_expr_is!( |
| "contents: Included file contents\n", |
| concat!("contents: ", include_str!("auxiliary/included-file.txt")) |
| ); |
| |
| expand_expr_is!( |
| b"contents: Included file contents\n", |
| concat_bytes!(b"contents: ", include_bytes!("auxiliary/included-file.txt")) |
| ); |
| |
| // Correct value is checked for multiple sources. |
| check_expand_expr_file!(file!()); |
| |
| expand_expr_is!("hello", stringify!(hello)); |
| expand_expr_is!("10 + 20", stringify!(10 + 20)); |
| |
| macro_rules! echo_tts { |
| ($($t:tt)*) => { $($t)* }; |
| } |
| |
| macro_rules! echo_lit { |
| ($l:literal) => { |
| $l |
| }; |
| } |
| |
| macro_rules! echo_expr { |
| ($e:expr) => { |
| $e |
| }; |
| } |
| |
| macro_rules! simple_lit { |
| ($l:literal) => { |
| expand_expr_is!($l, $l); |
| expand_expr_is!($l, echo_lit!($l)); |
| expand_expr_is!($l, echo_expr!($l)); |
| expand_expr_is!($l, echo_tts!($l)); |
| expand_expr_is!($l, echo_pm!($l)); |
| const _: () = { |
| macro_rules! mac { |
| () => { |
| $l |
| }; |
| } |
| expand_expr_is!($l, mac!()); |
| expand_expr_is!($l, echo_expr!(mac!())); |
| expand_expr_is!($l, echo_tts!(mac!())); |
| expand_expr_is!($l, echo_pm!(mac!())); |
| }; |
| }; |
| } |
| |
| simple_lit!("Hello, World"); |
| simple_lit!('c'); |
| simple_lit!(b'c'); |
| simple_lit!(10); |
| simple_lit!(10.0); |
| simple_lit!(10.0f64); |
| simple_lit!(-3.14159); |
| simple_lit!(-3.5e10); |
| simple_lit!(0xFEED); |
| simple_lit!(-0xFEED); |
| simple_lit!(0b0100); |
| simple_lit!(-0b0100); |
| simple_lit!("string"); |
| simple_lit!(r##"raw string"##); |
| simple_lit!(b"byte string"); |
| simple_lit!(br##"raw byte string"##); |
| simple_lit!(true); |
| simple_lit!(false); |
| |
| // Ensure char escapes aren't normalized by expansion |
| simple_lit!("\u{0}"); |
| simple_lit!("\0"); |
| simple_lit!("\x00"); |
| simple_lit!('\u{0}'); |
| simple_lit!('\0'); |
| simple_lit!('\x00'); |
| simple_lit!(b"\x00"); |
| simple_lit!(b"\0"); |
| simple_lit!(b'\x00'); |
| simple_lit!(b'\0'); |
| |
| // Extra tokens after the string literal aren't ignored |
| expand_expr_fail!("string"; hello); //~ ERROR: expected one of `.`, `?`, or an operator, found `;` |
| |
| // Invalid expressions produce errors in addition to returning `Err(())`. |
| expand_expr_fail!($); //~ ERROR: expected expression, found `$` |
| expand_expr_fail!(echo_tts!($)); //~ ERROR: expected expression, found `$` |
| expand_expr_fail!(echo_pm!($)); //~ ERROR: expected expression, found `$` |
| |
| // We get errors reported and recover during macro expansion if the macro |
| // doesn't produce a valid expression. |
| expand_expr_is!("string", echo_tts!("string"; hello)); //~ ERROR: macro expansion ignores `hello` and any tokens following |
| expand_expr_is!("string", echo_pm!("string"; hello)); //~ ERROR: macro expansion ignores `;` and any tokens following |
| |
| // For now, fail if a non-literal expression is expanded. |
| expand_expr_fail!(arbitrary_expression() + "etc"); |
| expand_expr_fail!(echo_tts!(arbitrary_expression() + "etc")); |
| expand_expr_fail!(echo_expr!(arbitrary_expression() + "etc")); |
| expand_expr_fail!(echo_pm!(arbitrary_expression() + "etc")); |
| |
| const _: u32 = recursive_expand!(); //~ ERROR: recursion limit reached while expanding `recursive_expand!` |
| |
| fn main() { |
| // https://github.com/rust-lang/rust/issues/104414 |
| match b"Included file contents\n" { |
| include_bytes!("auxiliary/included-file.txt") => (), |
| _ => panic!("include_bytes! in pattern"), |
| } |
| } |