48 lines
1.5 KiB
Rust
48 lines
1.5 KiB
Rust
|
// Copyright 2018 The Rust Project Developers. See the COPYRIGHT
|
||
|
// file at the top-level directory of this distribution and at
|
||
|
// http://rust-lang.org/COPYRIGHT.
|
||
|
//
|
||
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
||
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
||
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
||
|
// option. This file may not be copied, modified, or distributed
|
||
|
// except according to those terms.
|
||
|
|
||
|
// no-prefer-dynamic
|
||
|
|
||
|
#![crate_type = "proc-macro"]
|
||
|
#![feature(proc_macro_diagnostic, proc_macro_span)]
|
||
|
|
||
|
extern crate proc_macro;
|
||
|
|
||
|
use proc_macro::{TokenStream, TokenTree, Span, Diagnostic};
|
||
|
|
||
|
fn parse(input: TokenStream) -> Result<(), Diagnostic> {
|
||
|
if let Some(TokenTree::Literal(lit)) = input.into_iter().next() {
|
||
|
let mut spans = vec![];
|
||
|
let string = lit.to_string();
|
||
|
for hi in string.matches("hi") {
|
||
|
let index = hi.as_ptr() as usize - string.as_ptr() as usize;
|
||
|
let subspan = lit.subspan(index..(index + hi.len())).unwrap();
|
||
|
spans.push(subspan);
|
||
|
}
|
||
|
|
||
|
if !spans.is_empty() {
|
||
|
Err(Span::call_site().error("found 'hi's").span_note(spans, "here"))
|
||
|
} else {
|
||
|
Ok(())
|
||
|
}
|
||
|
} else {
|
||
|
Err(Span::call_site().error("invalid input: expected string literal"))
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#[proc_macro]
|
||
|
pub fn subspan(input: TokenStream) -> TokenStream {
|
||
|
if let Err(diag) = parse(input) {
|
||
|
diag.emit();
|
||
|
}
|
||
|
|
||
|
TokenStream::new()
|
||
|
}
|