Cleanup descend_into_macros_impl

This commit is contained in:
Lukas Wirth 2021-09-27 18:44:40 +02:00
parent c4251319fd
commit 75660ff94f

View file

@ -544,38 +544,54 @@ impl<'db> SemanticsImpl<'db> {
let sa = self.analyze(&parent); let sa = self.analyze(&parent);
let mut queue = vec![InFile::new(sa.file_id, token)]; let mut queue = vec![InFile::new(sa.file_id, token)];
let mut cache = self.expansion_info_cache.borrow_mut(); let mut cache = self.expansion_info_cache.borrow_mut();
let mut process_expansion_for_token =
|queue: &mut Vec<_>, file_id, item, token: InFile<&_>| {
let mapped_tokens = cache
.entry(file_id)
.or_insert_with(|| file_id.expansion_info(self.db.upcast()))
.as_ref()?
.map_token_down(self.db.upcast(), item, token)?;
let len = queue.len();
// requeue the tokens we got from mapping our current token down
queue.extend(mapped_tokens.inspect(|token| {
if let Some(parent) = token.value.parent() {
self.cache(find_root(&parent), token.file_id);
}
}));
// if the length changed we have found a mapping for the token
(queue.len() != len).then(|| ())
};
// Remap the next token in the queue into a macro call its in, if it is not being remapped // Remap the next token in the queue into a macro call its in, if it is not being remapped
// either due to not being in a macro-call or because its unused push it into the result vec, // either due to not being in a macro-call or because its unused push it into the result vec,
// otherwise push the remapped tokens back into the queue as they can potentially be remapped again. // otherwise push the remapped tokens back into the queue as they can potentially be remapped again.
while let Some(token) = queue.pop() { while let Some(token) = queue.pop() {
self.db.unwind_if_cancelled(); self.db.unwind_if_cancelled();
let was_not_remapped = (|| { let was_not_remapped = (|| {
if let Some((call_id, item)) = token // are we inside an attribute macro call
let containing_attribute_macro_call = self.with_ctx(|ctx| {
token
.value .value
.ancestors() .ancestors()
.filter_map(ast::Item::cast) .filter_map(ast::Item::cast)
.filter_map(|item| { .filter_map(|item| {
self.with_ctx(|ctx| ctx.item_to_macro_call(token.with_value(item.clone()))) Some((ctx.item_to_macro_call(token.with_value(item.clone()))?, item))
.zip(Some(item))
}) })
.last() .last()
{ });
if let Some((call_id, item)) = containing_attribute_macro_call {
let file_id = call_id.as_file(); let file_id = call_id.as_file();
let tokens = cache return process_expansion_for_token(
.entry(file_id) &mut queue,
.or_insert_with(|| file_id.expansion_info(self.db.upcast())) file_id,
.as_ref()? Some(item),
.map_token_down(self.db.upcast(), Some(item), token.as_ref())?; token.as_ref(),
);
let len = queue.len();
queue.extend(tokens.inspect(|token| {
if let Some(parent) = token.value.parent() {
self.cache(find_root(&parent), token.file_id);
}
}));
return (queue.len() != len).then(|| ());
} }
// or are we inside a function-like macro call
if let Some(macro_call) = token.value.ancestors().find_map(ast::MacroCall::cast) { if let Some(macro_call) = token.value.ancestors().find_map(ast::MacroCall::cast) {
let tt = macro_call.token_tree()?; let tt = macro_call.token_tree()?;
let l_delim = match tt.left_delimiter_token() { let l_delim = match tt.left_delimiter_token() {
@ -589,21 +605,12 @@ impl<'db> SemanticsImpl<'db> {
if !TextRange::new(l_delim, r_delim).contains_range(token.value.text_range()) { if !TextRange::new(l_delim, r_delim).contains_range(token.value.text_range()) {
return None; return None;
} }
let file_id = sa.expand(self.db, token.with_value(&macro_call))?;
let tokens = cache
.entry(file_id)
.or_insert_with(|| file_id.expansion_info(self.db.upcast()))
.as_ref()?
.map_token_down(self.db.upcast(), None, token.as_ref())?;
let len = queue.len(); let file_id = sa.expand(self.db, token.with_value(&macro_call))?;
queue.extend(tokens.inspect(|token| { return process_expansion_for_token(&mut queue, file_id, None, token.as_ref());
if let Some(parent) = token.value.parent() {
self.cache(find_root(&parent), token.file_id);
}
}));
return (queue.len() != len).then(|| ());
} }
// outside of a macro invocation so this is a "final" token
None None
})() })()
.is_none(); .is_none();