2019-10-30 15:41:50 +00:00
|
|
|
//! This modules handles hygiene information.
|
|
|
|
//!
|
|
|
|
//! Specifically, `ast` + `Hygiene` allows you to create a `Name`. Note that, at
|
|
|
|
//! this moment, this is horribly incomplete and handles only `$crate`.
|
2023-11-17 18:07:31 +00:00
|
|
|
use base_db::span::{MacroCallId, SyntaxContextId};
|
2019-10-30 15:41:50 +00:00
|
|
|
|
2023-11-17 18:07:31 +00:00
|
|
|
use crate::db::ExpandDatabase;
|
2019-10-30 15:41:50 +00:00
|
|
|
|
2023-10-06 12:47:11 +00:00
|
|
|
#[derive(Debug, Clone, Hash, PartialEq, Eq)]
|
|
|
|
pub struct SyntaxContextData {
|
2023-11-17 18:07:31 +00:00
|
|
|
pub outer_expn: Option<MacroCallId>,
|
|
|
|
pub outer_transparency: Transparency,
|
|
|
|
pub parent: SyntaxContextId,
|
2023-10-06 12:47:11 +00:00
|
|
|
/// This context, but with all transparent and semi-transparent expansions filtered away.
|
2023-11-17 18:07:31 +00:00
|
|
|
pub opaque: SyntaxContextId,
|
2023-10-06 12:47:11 +00:00
|
|
|
/// This context, but with all transparent expansions filtered away.
|
2023-11-17 18:07:31 +00:00
|
|
|
pub opaque_and_semitransparent: SyntaxContextId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl SyntaxContextData {
|
|
|
|
pub fn root() -> Self {
|
|
|
|
SyntaxContextData {
|
|
|
|
outer_expn: None,
|
|
|
|
outer_transparency: Transparency::Opaque,
|
|
|
|
parent: SyntaxContextId::ROOT,
|
|
|
|
opaque: SyntaxContextId::ROOT,
|
|
|
|
opaque_and_semitransparent: SyntaxContextId::ROOT,
|
|
|
|
}
|
|
|
|
}
|
2023-10-06 12:47:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// A property of a macro expansion that determines how identifiers
|
|
|
|
/// produced by that expansion are resolved.
|
|
|
|
#[derive(Copy, Clone, PartialEq, Eq, PartialOrd, Hash, Debug)]
|
|
|
|
pub enum Transparency {
|
|
|
|
/// Identifier produced by a transparent expansion is always resolved at call-site.
|
|
|
|
/// Call-site spans in procedural macros, hygiene opt-out in `macro` should use this.
|
|
|
|
Transparent,
|
|
|
|
/// Identifier produced by a semi-transparent expansion may be resolved
|
|
|
|
/// either at call-site or at definition-site.
|
|
|
|
/// If it's a local variable, label or `$crate` then it's resolved at def-site.
|
|
|
|
/// Otherwise it's resolved at call-site.
|
|
|
|
/// `macro_rules` macros behave like this, built-in macros currently behave like this too,
|
|
|
|
/// but that's an implementation detail.
|
|
|
|
SemiTransparent,
|
|
|
|
/// Identifier produced by an opaque expansion is always resolved at definition-site.
|
|
|
|
/// Def-site spans in procedural macros, identifiers from `macro` by default use this.
|
|
|
|
Opaque,
|
|
|
|
}
|
|
|
|
|
|
|
|
pub(super) fn apply_mark(
|
2023-11-17 18:07:31 +00:00
|
|
|
db: &dyn ExpandDatabase,
|
|
|
|
ctxt: SyntaxContextId,
|
|
|
|
call_id: MacroCallId,
|
|
|
|
transparency: Transparency,
|
2023-10-06 12:47:11 +00:00
|
|
|
) -> SyntaxContextId {
|
2023-11-17 18:07:31 +00:00
|
|
|
if transparency == Transparency::Opaque {
|
|
|
|
return apply_mark_internal(db, ctxt, Some(call_id), transparency);
|
2019-10-30 15:41:50 +00:00
|
|
|
}
|
|
|
|
|
2023-11-17 18:07:31 +00:00
|
|
|
let call_site_ctxt = db.lookup_intern_macro_call(call_id).call_site;
|
|
|
|
let mut call_site_ctxt = if transparency == Transparency::SemiTransparent {
|
|
|
|
call_site_ctxt.normalize_to_macros_2_0(db)
|
|
|
|
} else {
|
|
|
|
call_site_ctxt.normalize_to_macro_rules(db)
|
|
|
|
};
|
2020-05-01 03:23:03 +00:00
|
|
|
|
2023-11-17 18:07:31 +00:00
|
|
|
if call_site_ctxt.is_root(db) {
|
|
|
|
return apply_mark_internal(db, ctxt, Some(call_id), transparency);
|
2021-01-03 10:47:57 +00:00
|
|
|
}
|
2021-01-02 12:25:05 +00:00
|
|
|
|
2023-11-17 18:07:31 +00:00
|
|
|
// Otherwise, `expn_id` is a macros 1.0 definition and the call site is in a
|
|
|
|
// macros 2.0 expansion, i.e., a macros 1.0 invocation is in a macros 2.0 definition.
|
|
|
|
//
|
|
|
|
// In this case, the tokens from the macros 1.0 definition inherit the hygiene
|
|
|
|
// at their invocation. That is, we pretend that the macros 1.0 definition
|
|
|
|
// was defined at its invocation (i.e., inside the macros 2.0 definition)
|
|
|
|
// so that the macros 2.0 definition remains hygienic.
|
|
|
|
//
|
|
|
|
// See the example at `test/ui/hygiene/legacy_interaction.rs`.
|
|
|
|
for (call_id, transparency) in ctxt.marks(db) {
|
|
|
|
call_site_ctxt = apply_mark_internal(db, call_site_ctxt, call_id, transparency);
|
2021-01-04 02:53:31 +00:00
|
|
|
}
|
2023-11-17 18:07:31 +00:00
|
|
|
apply_mark_internal(db, call_site_ctxt, Some(call_id), transparency)
|
2021-01-04 02:53:31 +00:00
|
|
|
}
|
|
|
|
|
2023-11-17 18:07:31 +00:00
|
|
|
fn apply_mark_internal(
|
|
|
|
db: &dyn ExpandDatabase,
|
|
|
|
ctxt: SyntaxContextId,
|
|
|
|
call_id: Option<MacroCallId>,
|
|
|
|
transparency: Transparency,
|
|
|
|
) -> SyntaxContextId {
|
|
|
|
let syntax_context_data = db.lookup_intern_syntax_context(ctxt);
|
|
|
|
let mut opaque = syntax_context_data.opaque;
|
|
|
|
let mut opaque_and_semitransparent = syntax_context_data.opaque_and_semitransparent;
|
|
|
|
|
|
|
|
if transparency >= Transparency::Opaque {
|
|
|
|
let parent = opaque;
|
|
|
|
let new_opaque = SyntaxContextId::SELF_REF;
|
|
|
|
// But we can't just grab the to be allocated ID either as that would not deduplicate
|
|
|
|
// things!
|
|
|
|
// So we need a new salsa store type here ...
|
|
|
|
opaque = db.intern_syntax_context(SyntaxContextData {
|
|
|
|
outer_expn: call_id,
|
|
|
|
outer_transparency: transparency,
|
|
|
|
parent,
|
|
|
|
opaque: new_opaque,
|
|
|
|
opaque_and_semitransparent: new_opaque,
|
|
|
|
});
|
|
|
|
}
|
2021-01-04 02:53:31 +00:00
|
|
|
|
2023-11-17 18:07:31 +00:00
|
|
|
if transparency >= Transparency::SemiTransparent {
|
|
|
|
let parent = opaque_and_semitransparent;
|
|
|
|
let new_opaque_and_semitransparent = SyntaxContextId::SELF_REF;
|
|
|
|
opaque_and_semitransparent = db.intern_syntax_context(SyntaxContextData {
|
|
|
|
outer_expn: call_id,
|
|
|
|
outer_transparency: transparency,
|
|
|
|
parent,
|
|
|
|
opaque,
|
|
|
|
opaque_and_semitransparent: new_opaque_and_semitransparent,
|
|
|
|
});
|
|
|
|
}
|
2021-01-04 02:53:31 +00:00
|
|
|
|
2023-11-17 18:07:31 +00:00
|
|
|
let parent = ctxt;
|
|
|
|
db.intern_syntax_context(SyntaxContextData {
|
|
|
|
outer_expn: call_id,
|
|
|
|
outer_transparency: transparency,
|
|
|
|
parent,
|
|
|
|
opaque,
|
|
|
|
opaque_and_semitransparent,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
pub trait SyntaxContextExt {
|
|
|
|
fn is_root(self, db: &dyn ExpandDatabase) -> bool;
|
|
|
|
fn normalize_to_macro_rules(self, db: &dyn ExpandDatabase) -> Self;
|
|
|
|
fn normalize_to_macros_2_0(self, db: &dyn ExpandDatabase) -> Self;
|
|
|
|
fn parent_ctxt(self, db: &dyn ExpandDatabase) -> Self;
|
|
|
|
fn outer_mark(self, db: &dyn ExpandDatabase) -> (Option<MacroCallId>, Transparency);
|
|
|
|
fn marks(self, db: &dyn ExpandDatabase) -> Vec<(Option<MacroCallId>, Transparency)>;
|
|
|
|
}
|
2021-01-04 02:53:31 +00:00
|
|
|
|
2023-11-17 18:07:31 +00:00
|
|
|
#[inline(always)]
|
|
|
|
fn handle_self_ref(p: SyntaxContextId, n: SyntaxContextId) -> SyntaxContextId {
|
|
|
|
match n {
|
|
|
|
SyntaxContextId::SELF_REF => p,
|
|
|
|
_ => n,
|
2021-01-04 02:53:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-17 18:07:31 +00:00
|
|
|
impl SyntaxContextExt for SyntaxContextId {
|
|
|
|
fn is_root(self, db: &dyn ExpandDatabase) -> bool {
|
|
|
|
db.lookup_intern_syntax_context(self).outer_expn.is_none()
|
|
|
|
}
|
|
|
|
fn normalize_to_macro_rules(self, db: &dyn ExpandDatabase) -> Self {
|
|
|
|
handle_self_ref(self, db.lookup_intern_syntax_context(self).opaque_and_semitransparent)
|
|
|
|
}
|
|
|
|
fn normalize_to_macros_2_0(self, db: &dyn ExpandDatabase) -> Self {
|
|
|
|
handle_self_ref(self, db.lookup_intern_syntax_context(self).opaque)
|
|
|
|
}
|
|
|
|
fn parent_ctxt(self, db: &dyn ExpandDatabase) -> Self {
|
|
|
|
db.lookup_intern_syntax_context(self).parent
|
|
|
|
}
|
|
|
|
fn outer_mark(self, db: &dyn ExpandDatabase) -> (Option<MacroCallId>, Transparency) {
|
|
|
|
let data = db.lookup_intern_syntax_context(self);
|
|
|
|
(data.outer_expn, data.outer_transparency)
|
|
|
|
}
|
|
|
|
fn marks(mut self, db: &dyn ExpandDatabase) -> Vec<(Option<MacroCallId>, Transparency)> {
|
|
|
|
let mut marks = Vec::new();
|
|
|
|
while self != SyntaxContextId::ROOT {
|
|
|
|
marks.push(self.outer_mark(db));
|
|
|
|
self = self.parent_ctxt(db);
|
|
|
|
}
|
|
|
|
marks.reverse();
|
|
|
|
marks
|
2020-05-01 03:23:03 +00:00
|
|
|
}
|
2019-10-30 15:41:50 +00:00
|
|
|
}
|
2023-11-17 18:07:31 +00:00
|
|
|
|
|
|
|
// pub(super) fn with_ctxt_from_mark(db: &ExpandDatabase, file_id: HirFileId) {
|
|
|
|
// self.with_ctxt_from_mark(expn_id, Transparency::Transparent)
|
|
|
|
// }
|
|
|
|
// pub(super) fn with_call_site_ctxt(db: &ExpandDatabase, file_id: HirFileId) {
|
|
|
|
// self.with_ctxt_from_mark(expn_id, Transparency::Transparent)
|
|
|
|
// }
|