2014-11-15 20:30:33 -05:00
|
|
|
use rustc::session::Session;
|
2014-02-05 17:31:33 +13:00
|
|
|
|
2019-02-07 01:02:00 +09:00
|
|
|
use crate::generated_code;
|
2014-02-05 17:31:33 +13:00
|
|
|
|
2017-01-12 23:32:00 +00:00
|
|
|
use syntax::parse::lexer::{self, StringReader};
|
2019-06-04 17:55:23 +03:00
|
|
|
use syntax::parse::token::{self, TokenKind};
|
2016-06-21 18:08:13 -04:00
|
|
|
use syntax_pos::*;
|
2014-02-05 17:31:33 +13:00
|
|
|
|
2015-01-03 22:54:18 -05:00
|
|
|
#[derive(Clone)]
|
2014-02-05 17:31:33 +13:00
|
|
|
pub struct SpanUtils<'a> {
|
|
|
|
pub sess: &'a Session,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> SpanUtils<'a> {
|
2015-07-14 14:21:54 +12:00
|
|
|
pub fn new(sess: &'a Session) -> SpanUtils<'a> {
|
2015-09-28 09:20:49 +13:00
|
|
|
SpanUtils {
|
2017-08-06 22:54:09 -07:00
|
|
|
sess,
|
2015-09-28 09:20:49 +13:00
|
|
|
}
|
2015-07-14 14:21:54 +12:00
|
|
|
}
|
|
|
|
|
2018-09-11 16:39:07 +02:00
|
|
|
pub fn make_filename_string(&self, file: &SourceFile) -> String {
|
|
|
|
match &file.name {
|
2018-09-24 16:53:49 +02:00
|
|
|
FileName::Real(path) if !file.name_was_remapped => {
|
|
|
|
if path.is_absolute() {
|
|
|
|
self.sess.source_map().path_mapping()
|
|
|
|
.map_prefix(path.clone()).0
|
|
|
|
.display()
|
|
|
|
.to_string()
|
|
|
|
} else {
|
|
|
|
self.sess.working_dir.0
|
|
|
|
.join(&path)
|
|
|
|
.display()
|
|
|
|
.to_string()
|
|
|
|
}
|
2018-09-11 16:39:07 +02:00
|
|
|
},
|
|
|
|
// If the file name is already remapped, we assume the user
|
|
|
|
// configured it the way they wanted to, so use that directly
|
|
|
|
filename => filename.to_string()
|
2015-11-26 18:05:32 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-05 17:31:33 +13:00
|
|
|
pub fn snippet(&self, span: Span) -> String {
|
2018-08-18 12:14:09 +02:00
|
|
|
match self.sess.source_map().span_to_snippet(span) {
|
2015-02-05 16:02:22 +01:00
|
|
|
Ok(s) => s,
|
|
|
|
Err(_) => String::new(),
|
2014-02-05 17:31:33 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn retokenise_span(&self, span: Span) -> StringReader<'a> {
|
2016-06-10 13:00:21 +03:00
|
|
|
lexer::StringReader::retokenize(&self.sess.parse_sess, span)
|
2016-11-21 07:07:40 +13:00
|
|
|
}
|
|
|
|
|
2019-06-04 17:55:23 +03:00
|
|
|
pub fn sub_span_of_token(&self, span: Span, tok: TokenKind) -> Option<Span> {
|
2014-11-23 22:29:41 +13:00
|
|
|
let mut toks = self.retokenise_span(span);
|
|
|
|
loop {
|
|
|
|
let next = toks.real_token();
|
2019-06-04 18:48:40 +03:00
|
|
|
if next == token::Eof {
|
2014-11-23 22:29:41 +13:00
|
|
|
return None;
|
|
|
|
}
|
2019-06-04 18:48:40 +03:00
|
|
|
if next == tok {
|
|
|
|
return Some(next.span);
|
2014-11-23 22:29:41 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-08 14:45:15 +12:00
|
|
|
// // Return the name for a macro definition (identifier after first `!`)
|
|
|
|
// pub fn span_for_macro_def_name(&self, span: Span) -> Option<Span> {
|
|
|
|
// let mut toks = self.retokenise_span(span);
|
|
|
|
// loop {
|
|
|
|
// let ts = toks.real_token();
|
2019-06-04 18:48:40 +03:00
|
|
|
// if ts == token::Eof {
|
2017-06-08 14:45:15 +12:00
|
|
|
// return None;
|
|
|
|
// }
|
2019-06-04 18:48:40 +03:00
|
|
|
// if ts == token::Not {
|
2017-06-08 14:45:15 +12:00
|
|
|
// let ts = toks.real_token();
|
2019-06-04 18:48:40 +03:00
|
|
|
// if ts.kind.is_ident() {
|
2017-06-08 14:45:15 +12:00
|
|
|
// return Some(ts.sp);
|
|
|
|
// } else {
|
|
|
|
// return None;
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
|
|
|
// // Return the name for a macro use (identifier before first `!`).
|
|
|
|
// pub fn span_for_macro_use_name(&self, span:Span) -> Option<Span> {
|
|
|
|
// let mut toks = self.retokenise_span(span);
|
|
|
|
// let mut prev = toks.real_token();
|
|
|
|
// loop {
|
2019-06-04 18:48:40 +03:00
|
|
|
// if prev == token::Eof {
|
2017-06-08 14:45:15 +12:00
|
|
|
// return None;
|
|
|
|
// }
|
|
|
|
// let ts = toks.real_token();
|
2019-06-04 18:48:40 +03:00
|
|
|
// if ts == token::Not {
|
|
|
|
// if prev.kind.is_ident() {
|
2017-06-08 14:45:15 +12:00
|
|
|
// return Some(prev.sp);
|
|
|
|
// } else {
|
|
|
|
// return None;
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// prev = ts;
|
|
|
|
// }
|
|
|
|
// }
|
2016-02-03 20:44:53 +13:00
|
|
|
|
2016-01-22 11:58:09 +13:00
|
|
|
/// Return true if the span is generated code, and
|
|
|
|
/// it is not a subspan of the root callsite.
|
|
|
|
///
|
|
|
|
/// Used to filter out spans of minimal value,
|
|
|
|
/// such as references to macro internal variables.
|
2018-09-10 12:54:36 +12:00
|
|
|
pub fn filter_generated(&self, span: Span) -> bool {
|
2018-11-14 10:56:45 +13:00
|
|
|
if generated_code(span) {
|
2018-09-10 12:54:36 +12:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-08-18 12:13:56 +02:00
|
|
|
//If the span comes from a fake source_file, filter it.
|
2018-09-10 12:54:36 +12:00
|
|
|
!self.sess
|
2018-08-18 12:14:09 +02:00
|
|
|
.source_map()
|
2018-09-10 12:54:36 +12:00
|
|
|
.lookup_char_pos(span.lo())
|
2017-11-08 10:43:05 +13:00
|
|
|
.file
|
|
|
|
.is_real_file()
|
2016-01-22 11:58:09 +13:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! filter {
|
2018-09-10 12:54:36 +12:00
|
|
|
($util: expr, $parent: expr) => {
|
|
|
|
if $util.filter_generated($parent) {
|
2016-01-22 11:58:09 +13:00
|
|
|
return None;
|
|
|
|
}
|
|
|
|
};
|
2014-02-05 17:31:33 +13:00
|
|
|
}
|