43 lines
1.4 KiB
Rust
43 lines
1.4 KiB
Rust
//@ force-host
|
|
//@ no-prefer-dynamic
|
|
|
|
#![feature(proc_macro_diagnostic, proc_macro_span)]
|
|
#![crate_type = "proc-macro"]
|
|
|
|
extern crate proc_macro;
|
|
|
|
use proc_macro::{TokenStream, TokenTree, Span};
|
|
|
|
fn lit_span(tt: TokenTree) -> (Span, String) {
|
|
match tt {
|
|
TokenTree::Literal(..) |
|
|
TokenTree::Group(..) => (tt.span(), tt.to_string().trim().into()),
|
|
_ => panic!("expected a literal in token tree, got: {:?}", tt)
|
|
}
|
|
}
|
|
|
|
#[proc_macro]
|
|
pub fn parent_source_spans(input: TokenStream) -> TokenStream {
|
|
let mut tokens = input.into_iter();
|
|
let (sp1, str1) = lit_span(tokens.next().expect("first string"));
|
|
let _ = tokens.next();
|
|
let (sp2, str2) = lit_span(tokens.next().expect("second string"));
|
|
|
|
sp1.error(format!("first final: {}", str1)).emit();
|
|
sp2.error(format!("second final: {}", str2)).emit();
|
|
|
|
if let (Some(p1), Some(p2)) = (sp1.parent(), sp2.parent()) {
|
|
p1.error(format!("first parent: {}", str1)).emit();
|
|
p2.error(format!("second parent: {}", str2)).emit();
|
|
|
|
if let (Some(gp1), Some(gp2)) = (p1.parent(), p2.parent()) {
|
|
gp1.error(format!("first grandparent: {}", str1)).emit();
|
|
gp2.error(format!("second grandparent: {}", str2)).emit();
|
|
}
|
|
}
|
|
|
|
sp1.source().error(format!("first source: {}", str1)).emit();
|
|
sp2.source().error(format!("second source: {}", str2)).emit();
|
|
|
|
"ok".parse().unwrap()
|
|
}
|