2014-02-05 04:31:33 +00:00
|
|
|
// Copyright 2012-2014 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.
|
|
|
|
|
2014-11-16 01:30:33 +00:00
|
|
|
use rustc::session::Session;
|
2014-02-05 04:31:33 +00:00
|
|
|
|
2016-03-22 16:40:24 +00:00
|
|
|
use generated_code;
|
2014-02-05 04:31:33 +00:00
|
|
|
|
|
|
|
use std::cell::Cell;
|
|
|
|
|
2017-01-12 23:32:00 +00:00
|
|
|
use syntax::parse::lexer::{self, StringReader};
|
2016-11-29 22:50:08 +00:00
|
|
|
use syntax::parse::token::{self, Token};
|
|
|
|
use syntax::symbol::keywords;
|
2016-06-21 22:08:13 +00:00
|
|
|
use syntax_pos::*;
|
2014-02-05 04:31:33 +00:00
|
|
|
|
2015-01-04 03:54:18 +00:00
|
|
|
#[derive(Clone)]
|
2014-02-05 04:31:33 +00:00
|
|
|
pub struct SpanUtils<'a> {
|
|
|
|
pub sess: &'a Session,
|
2016-04-17 22:30:55 +00:00
|
|
|
// FIXME given that we clone SpanUtils all over the place, this err_count is
|
|
|
|
// probably useless and any logic relying on it is bogus.
|
2015-03-26 00:06:52 +00:00
|
|
|
pub err_count: Cell<isize>,
|
2014-02-05 04:31:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> SpanUtils<'a> {
|
2015-07-14 02:21:54 +00:00
|
|
|
pub fn new(sess: &'a Session) -> SpanUtils<'a> {
|
2015-09-27 20:20:49 +00:00
|
|
|
SpanUtils {
|
2017-08-07 05:54:09 +00:00
|
|
|
sess,
|
2015-09-27 20:20:49 +00:00
|
|
|
err_count: Cell::new(0),
|
|
|
|
}
|
2015-07-14 02:21:54 +00:00
|
|
|
}
|
|
|
|
|
2018-09-11 14:39:07 +00:00
|
|
|
pub fn make_filename_string(&self, file: &SourceFile) -> String {
|
|
|
|
match &file.name {
|
2018-09-24 14:53:49 +00: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 14:39:07 +00: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 05:05:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-02-05 04:31:33 +00:00
|
|
|
pub fn snippet(&self, span: Span) -> String {
|
2018-08-18 10:14:09 +00:00
|
|
|
match self.sess.source_map().span_to_snippet(span) {
|
2015-02-05 15:02:22 +00:00
|
|
|
Ok(s) => s,
|
|
|
|
Err(_) => String::new(),
|
2014-02-05 04:31:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn retokenise_span(&self, span: Span) -> StringReader<'a> {
|
2016-06-10 10:00:21 +00:00
|
|
|
lexer::StringReader::retokenize(&self.sess.parse_sess, span)
|
2016-11-20 18:07:40 +00:00
|
|
|
}
|
|
|
|
|
2014-02-05 04:31:33 +00:00
|
|
|
// Re-parses a path and returns the span for the last identifier in the path
|
|
|
|
pub fn span_for_last_ident(&self, span: Span) -> Option<Span> {
|
|
|
|
let mut result = None;
|
|
|
|
|
|
|
|
let mut toks = self.retokenise_span(span);
|
2015-01-25 10:58:43 +00:00
|
|
|
let mut bracket_count = 0;
|
2014-02-05 04:31:33 +00:00
|
|
|
loop {
|
2014-11-03 04:52:00 +00:00
|
|
|
let ts = toks.real_token();
|
2014-10-27 08:22:52 +00:00
|
|
|
if ts.tok == token::Eof {
|
2017-11-07 21:43:05 +00:00
|
|
|
return result;
|
2014-02-05 04:31:33 +00:00
|
|
|
}
|
2015-09-02 03:37:07 +00:00
|
|
|
if bracket_count == 0 && (ts.tok.is_ident() || ts.tok.is_keyword(keywords::SelfValue)) {
|
2014-02-05 04:31:33 +00:00
|
|
|
result = Some(ts.sp);
|
|
|
|
}
|
|
|
|
|
|
|
|
bracket_count += match ts.tok {
|
2014-10-27 08:22:52 +00:00
|
|
|
token::Lt => 1,
|
|
|
|
token::Gt => -1,
|
|
|
|
token::BinOp(token::Shr) => -2,
|
2015-09-02 03:37:07 +00:00
|
|
|
_ => 0,
|
2014-02-05 04:31:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return the span for the first identifier in the path.
|
|
|
|
pub fn span_for_first_ident(&self, span: Span) -> Option<Span> {
|
|
|
|
let mut toks = self.retokenise_span(span);
|
2015-01-25 10:58:43 +00:00
|
|
|
let mut bracket_count = 0;
|
2014-02-05 04:31:33 +00:00
|
|
|
loop {
|
2014-11-03 04:52:00 +00:00
|
|
|
let ts = toks.real_token();
|
2014-10-27 08:22:52 +00:00
|
|
|
if ts.tok == token::Eof {
|
2014-02-05 04:31:33 +00:00
|
|
|
return None;
|
|
|
|
}
|
2015-09-02 03:37:07 +00:00
|
|
|
if bracket_count == 0 && (ts.tok.is_ident() || ts.tok.is_keyword(keywords::SelfValue)) {
|
2016-06-10 10:00:21 +00:00
|
|
|
return Some(ts.sp);
|
2014-02-05 04:31:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
bracket_count += match ts.tok {
|
2014-10-27 08:22:52 +00:00
|
|
|
token::Lt => 1,
|
|
|
|
token::Gt => -1,
|
|
|
|
token::BinOp(token::Shr) => -2,
|
2015-09-02 03:37:07 +00:00
|
|
|
_ => 0,
|
2014-02-05 04:31:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return the span for the last ident before a `<` and outside any
|
2016-11-15 04:03:27 +00:00
|
|
|
// angle brackets, or the last span.
|
2014-02-05 04:31:33 +00:00
|
|
|
pub fn sub_span_for_type_name(&self, span: Span) -> Option<Span> {
|
|
|
|
let mut toks = self.retokenise_span(span);
|
2014-11-03 04:52:00 +00:00
|
|
|
let mut prev = toks.real_token();
|
2014-02-05 04:31:33 +00:00
|
|
|
let mut result = None;
|
2016-11-15 04:03:27 +00:00
|
|
|
|
|
|
|
// We keep track of the following two counts - the depth of nesting of
|
|
|
|
// angle brackets, and the depth of nesting of square brackets. For the
|
|
|
|
// angle bracket count, we only count tokens which occur outside of any
|
2018-02-16 03:45:37 +00:00
|
|
|
// square brackets (i.e. bracket_count == 0). The intuition here is
|
2016-11-15 04:03:27 +00:00
|
|
|
// that we want to count angle brackets in the type, but not any which
|
|
|
|
// could be in expression context (because these could mean 'less than',
|
|
|
|
// etc.).
|
|
|
|
let mut angle_count = 0;
|
2015-01-25 10:58:43 +00:00
|
|
|
let mut bracket_count = 0;
|
2014-02-05 04:31:33 +00:00
|
|
|
loop {
|
2014-11-03 04:52:00 +00:00
|
|
|
let next = toks.real_token();
|
2014-02-05 04:31:33 +00:00
|
|
|
|
2017-11-07 21:43:05 +00:00
|
|
|
if (next.tok == token::Lt || next.tok == token::Colon) && angle_count == 0
|
|
|
|
&& bracket_count == 0 && prev.tok.is_ident()
|
|
|
|
{
|
2014-02-05 04:31:33 +00:00
|
|
|
result = Some(prev.sp);
|
|
|
|
}
|
|
|
|
|
2016-11-15 04:03:27 +00:00
|
|
|
if bracket_count == 0 {
|
|
|
|
angle_count += match prev.tok {
|
|
|
|
token::Lt => 1,
|
|
|
|
token::Gt => -1,
|
|
|
|
token::BinOp(token::Shl) => 2,
|
|
|
|
token::BinOp(token::Shr) => -2,
|
|
|
|
_ => 0,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2014-02-05 04:31:33 +00:00
|
|
|
bracket_count += match prev.tok {
|
2016-11-15 04:03:27 +00:00
|
|
|
token::OpenDelim(token::Bracket) => 1,
|
|
|
|
token::CloseDelim(token::Bracket) => -1,
|
2015-09-02 03:37:07 +00:00
|
|
|
_ => 0,
|
2014-02-05 04:31:33 +00:00
|
|
|
};
|
|
|
|
|
2014-10-27 08:22:52 +00:00
|
|
|
if next.tok == token::Eof {
|
2014-02-05 04:31:33 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
prev = next;
|
|
|
|
}
|
2018-02-16 03:45:37 +00:00
|
|
|
#[cfg(debug_assertions)] {
|
|
|
|
if angle_count != 0 || bracket_count != 0 {
|
2018-08-18 10:14:09 +00:00
|
|
|
let loc = self.sess.source_map().lookup_char_pos(span.lo());
|
2018-02-16 03:45:37 +00:00
|
|
|
span_bug!(
|
|
|
|
span,
|
2018-08-30 23:01:08 +00:00
|
|
|
"Mis-counted brackets when breaking path? Parsing '{}' in {}, line {}",
|
2018-02-16 03:45:37 +00:00
|
|
|
self.snippet(span),
|
|
|
|
loc.file.name,
|
|
|
|
loc.line
|
|
|
|
);
|
|
|
|
}
|
2014-02-05 04:31:33 +00:00
|
|
|
}
|
2018-02-16 03:45:37 +00:00
|
|
|
if result.is_none() && prev.tok.is_ident() {
|
2016-06-10 10:00:21 +00:00
|
|
|
return Some(prev.sp);
|
2014-02-05 04:31:33 +00:00
|
|
|
}
|
2016-06-10 10:00:21 +00:00
|
|
|
result
|
2014-02-05 04:31:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn sub_span_before_token(&self, span: Span, tok: Token) -> Option<Span> {
|
|
|
|
let mut toks = self.retokenise_span(span);
|
2014-11-03 04:52:00 +00:00
|
|
|
let mut prev = toks.real_token();
|
2014-02-05 04:31:33 +00:00
|
|
|
loop {
|
2014-10-27 08:22:52 +00:00
|
|
|
if prev.tok == token::Eof {
|
2014-02-05 04:31:33 +00:00
|
|
|
return None;
|
|
|
|
}
|
2014-11-03 04:52:00 +00:00
|
|
|
let next = toks.real_token();
|
2014-02-05 04:31:33 +00:00
|
|
|
if next.tok == tok {
|
2016-06-10 10:00:21 +00:00
|
|
|
return Some(prev.sp);
|
2014-02-05 04:31:33 +00:00
|
|
|
}
|
|
|
|
prev = next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-23 09:29:41 +00:00
|
|
|
pub fn sub_span_of_token(&self, span: Span, tok: Token) -> Option<Span> {
|
|
|
|
let mut toks = self.retokenise_span(span);
|
|
|
|
loop {
|
|
|
|
let next = toks.real_token();
|
|
|
|
if next.tok == token::Eof {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
if next.tok == tok {
|
2016-06-10 10:00:21 +00:00
|
|
|
return Some(next.sp);
|
2014-11-23 09:29:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-02 03:37:07 +00:00
|
|
|
pub fn sub_span_after_keyword(&self, span: Span, keyword: keywords::Keyword) -> Option<Span> {
|
2015-01-20 20:15:37 +00:00
|
|
|
self.sub_span_after(span, |t| t.is_keyword(keyword))
|
|
|
|
}
|
|
|
|
|
2015-09-02 03:37:07 +00:00
|
|
|
fn sub_span_after<F: Fn(Token) -> bool>(&self, span: Span, f: F) -> Option<Span> {
|
2014-02-05 04:31:33 +00:00
|
|
|
let mut toks = self.retokenise_span(span);
|
|
|
|
loop {
|
2014-11-03 04:52:00 +00:00
|
|
|
let ts = toks.real_token();
|
2014-10-27 08:22:52 +00:00
|
|
|
if ts.tok == token::Eof {
|
2014-02-05 04:31:33 +00:00
|
|
|
return None;
|
|
|
|
}
|
2015-01-20 20:15:37 +00:00
|
|
|
if f(ts.tok) {
|
2014-11-03 04:52:00 +00:00
|
|
|
let ts = toks.real_token();
|
2014-10-27 08:22:52 +00:00
|
|
|
if ts.tok == token::Eof {
|
2017-11-07 21:43:05 +00:00
|
|
|
return None;
|
2014-02-05 04:31:33 +00:00
|
|
|
} else {
|
2016-06-10 10:00:21 +00:00
|
|
|
return Some(ts.sp);
|
2014-02-05 04:31:33 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-08 02:45:15 +00: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();
|
|
|
|
// if ts.tok == token::Eof {
|
|
|
|
// return None;
|
|
|
|
// }
|
|
|
|
// if ts.tok == token::Not {
|
|
|
|
// let ts = toks.real_token();
|
|
|
|
// if ts.tok.is_ident() {
|
|
|
|
// 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 {
|
|
|
|
// if prev.tok == token::Eof {
|
|
|
|
// return None;
|
|
|
|
// }
|
|
|
|
// let ts = toks.real_token();
|
|
|
|
// if ts.tok == token::Not {
|
|
|
|
// if prev.tok.is_ident() {
|
|
|
|
// return Some(prev.sp);
|
|
|
|
// } else {
|
|
|
|
// return None;
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
// prev = ts;
|
|
|
|
// }
|
|
|
|
// }
|
2016-02-03 07:44:53 +00:00
|
|
|
|
2016-01-21 22:58:09 +00: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.
|
|
|
|
pub fn filter_generated(&self, sub_span: Option<Span>, parent: Span) -> bool {
|
|
|
|
if !generated_code(parent) {
|
2018-09-03 12:31:57 +00:00
|
|
|
// Edge case - this occurs on generated code with incorrect expansion info.
|
|
|
|
return sub_span.is_none()
|
2016-01-21 22:58:09 +00:00
|
|
|
}
|
|
|
|
// If sub_span is none, filter out generated code.
|
2017-08-01 02:43:11 +00:00
|
|
|
let sub_span = match sub_span {
|
|
|
|
Some(ss) => ss,
|
|
|
|
None => return true,
|
|
|
|
};
|
2016-01-29 07:22:55 +00:00
|
|
|
|
2018-08-18 10:13:56 +00:00
|
|
|
//If the span comes from a fake source_file, filter it.
|
2017-11-07 21:43:05 +00:00
|
|
|
if !self.sess
|
2018-08-18 10:14:09 +00:00
|
|
|
.source_map()
|
2017-11-07 21:43:05 +00:00
|
|
|
.lookup_char_pos(parent.lo())
|
|
|
|
.file
|
|
|
|
.is_real_file()
|
|
|
|
{
|
2016-01-29 07:22:55 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Otherwise, a generated span is deemed invalid if it is not a sub-span of the root
|
2016-01-21 22:58:09 +00:00
|
|
|
// callsite. This filters out macro internal variables and most malformed spans.
|
2017-08-01 02:43:11 +00:00
|
|
|
!parent.source_callsite().contains(sub_span)
|
2016-01-21 22:58:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! filter {
|
2017-11-06 02:52:42 +00:00
|
|
|
($util: expr, $span: expr, $parent: expr, None) => {
|
2016-01-21 22:58:09 +00:00
|
|
|
if $util.filter_generated($span, $parent) {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
($util: expr, $span: ident, $parent: expr) => {
|
|
|
|
if $util.filter_generated($span, $parent) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
};
|
2014-02-05 04:31:33 +00:00
|
|
|
}
|