swc/ecmascript/transforms/src/hygiene/mod.rs

498 lines
14 KiB
Rust
Raw Normal View History

use self::ops::{Operator, ScopeOp};
use crate::{pass::Pass, scope::ScopeKind, util::State};
use ast::*;
use fxhash::FxHashMap;
use std::cell::RefCell;
use swc_atoms::JsWord;
use swc_common::{Fold, FoldWith, Span, SyntaxContext};
mod ops;
#[cfg(test)]
mod tests;
const LOG: bool = false;
#[derive(Clone)]
struct Hygiene<'a> {
current: State<Scope<'a>>,
in_var_decl: State<bool>,
}
impl<'a> Hygiene<'a> {
fn add_declared_ref(&mut self, ident: Ident) {
let ctxt = ident.span.ctxt();
if cfg!(debug_assertions) && LOG {
eprintln!("Declaring {}{:?} ", ident.sym, ctxt);
}
let can_declare_without_renaming = self.current.can_declare(ident.sym.clone(), ctxt);
let sym = self.current.change_symbol(ident.sym, ctxt);
self.current
.declared_symbols
.borrow_mut()
.entry(sym.clone())
.or_insert_with(Vec::new)
.push(ctxt);
if can_declare_without_renaming {
// skip if previous symbol is declared on the same level.
return;
}
if cfg!(debug_assertions) && LOG {
eprintln!("Renaming from decl");
}
self.rename(sym, ctxt);
}
fn add_used_ref(&mut self, ident: Ident) {
let ctxt = ident.span.ctxt();
// We rename declaration instead of usage
let conflicts = self.current.conflicts(ident.sym.clone(), ctxt);
if cfg!(debug_assertions) && LOG && !conflicts.is_empty() {
eprintln!("Renaming from usage");
}
for cx in conflicts {
self.rename(ident.sym.clone(), cx)
}
}
fn rename(&mut self, sym: JsWord, ctxt: SyntaxContext) {
// symbol conflicts
let renamed = {
let mut i = 0;
loop {
i += 1;
let sym: JsWord = format!("{}{}", sym, i).into();
if !self.current.is_declared(&sym) {
break sym;
}
}
};
if cfg!(debug_assertions) && LOG {
eprintln!("\t{}{:?} -> {}", sym, ctxt, renamed);
}
let scope = self.current.scope_of(&sym, ctxt);
// Update symbol list
let mut declared_symbols = scope.declared_symbols.borrow_mut();
debug_assert!(
{
scope.ops.borrow().iter().all(|op| match *op {
ScopeOp::Rename { ref from, .. } => from.0 != sym || from.1 != ctxt,
})
},
"failed to rename {}{:?}: should not rename an ident multiple time\n{:?}",
sym,
ctxt,
scope.ops.borrow(),
);
let old = declared_symbols.entry(sym.clone()).or_default();
debug_assert!(old.contains(&ctxt));
old.retain(|c| *c != ctxt);
debug_assert!(old.len() == 0 || old.len() == 1);
let new = declared_symbols.entry(renamed.clone()).or_default();
new.push(ctxt);
debug_assert!(new.len() == 1);
scope.ops.borrow_mut().push(ScopeOp::Rename {
from: (sym, ctxt),
to: renamed,
});
}
}
pub fn hygiene() -> impl Pass + Clone + 'static {
#[derive(Clone, Copy)]
struct MarkClearer;
impl Fold<Span> for MarkClearer {
fn fold(&mut self, span: Span) -> Span {
span.with_ctxt(SyntaxContext::empty())
}
}
chain_at!(
Module,
Hygiene {
current: Default::default(),
in_var_decl: Default::default(),
},
MarkClearer
)
}
impl<'a> Hygiene<'a> {
fn apply_ops<N>(&mut self, node: N) -> N
where
for<'o> N: FoldWith<Operator<'o>>,
{
let ops = self.current.ops.borrow();
if ops.is_empty() {
return node;
}
node.fold_with(&mut Operator(&ops))
}
}
impl<'a> Fold<Module> for Hygiene<'a> {
fn fold(&mut self, module: Module) -> Module {
let module = module.fold_children(self);
self.apply_ops(module)
}
}
impl<'a> Fold<TryStmt> for Hygiene<'a> {
fn fold(&mut self, node: TryStmt) -> TryStmt {
TryStmt {
span: node.span,
block: node.block.fold_children(self),
handler: node.handler.fold_with(self),
finalizer: node.finalizer.fold_children(self),
}
}
}
impl<'a> Fold<BlockStmt> for Hygiene<'a> {
fn fold(&mut self, node: BlockStmt) -> BlockStmt {
let mut folder = Hygiene {
current: Scope::new(ScopeKind::Block, Some(&self.current)).into(),
in_var_decl: false.into(),
};
let node = node.fold_children(&mut folder);
folder.apply_ops(node)
}
}
impl<'a> Hygiene<'a> {
fn fold_fn(&mut self, ident: Option<Ident>, mut node: Function) -> Function {
match ident {
Some(ident) => {
self.add_declared_ref(ident);
}
_ => {}
}
let mut folder = Hygiene {
current: Scope::new(ScopeKind::Fn, Some(&self.current)).into(),
in_var_decl: false.into(),
};
folder.in_var_decl = true.into();
node.params = node.params.fold_with(&mut folder);
folder.in_var_decl = false.into();
node.body = node.body.map(|stmt| stmt.fold_children(&mut folder));
folder.apply_ops(node)
}
}
impl<'a> Fold<VarDeclarator> for Hygiene<'a> {
fn fold(&mut self, decl: VarDeclarator) -> VarDeclarator {
let old_in_var_decl = self.in_var_decl;
self.in_var_decl = true.into();
let name = decl.name.fold_with(self);
self.in_var_decl = old_in_var_decl;
let init = decl.init.fold_with(self);
VarDeclarator { name, init, ..decl }
}
}
impl<'a> Fold<FnExpr> for Hygiene<'a> {
fn fold(&mut self, mut node: FnExpr) -> FnExpr {
node.function = self.fold_fn(node.ident.clone(), node.function);
node
}
}
impl<'a> Fold<FnDecl> for Hygiene<'a> {
fn fold(&mut self, mut node: FnDecl) -> FnDecl {
node.function = self.fold_fn(Some(node.ident.clone()), node.function);
node
}
}
impl<'a> Fold<Ident> for Hygiene<'a> {
/// Invoked for `IdetifierRefrence` / `BindingIdentifier`
fn fold(&mut self, i: Ident) -> Ident {
if self.in_var_decl.value {
self.add_declared_ref(i.clone())
} else {
self.add_used_ref(i.clone());
}
i
}
}
impl<'a> Fold<Expr> for Hygiene<'a> {
fn fold(&mut self, node: Expr) -> Expr {
let old_in_var_decl = self.in_var_decl;
self.in_var_decl = false.into();
let node = match node {
Expr::Ident(..) => node.fold_children(self),
Expr::Member(e) => {
if e.computed {
Expr::Member(MemberExpr {
obj: e.obj.fold_with(self),
prop: e.prop.fold_with(self),
..e
})
} else {
Expr::Member(MemberExpr {
obj: e.obj.fold_with(self),
..e
})
}
}
Expr::This(..) => node,
_ => node.fold_children(self),
};
self.in_var_decl = old_in_var_decl;
node
}
}
#[derive(Debug)]
struct Scope<'a> {
/// Parent scope of this scope
pub parent: Option<&'a Scope<'a>>,
/// Kind of the scope.
pub kind: ScopeKind,
/// All references declared in this scope
pub declared_symbols: RefCell<FxHashMap<JsWord, Vec<SyntaxContext>>>,
pub(crate) ops: RefCell<Vec<ScopeOp>>,
}
impl<'a> Default for Scope<'a> {
fn default() -> Self {
Scope::new(ScopeKind::Fn, None)
}
}
impl<'a> Scope<'a> {
pub fn new(kind: ScopeKind, parent: Option<&'a Scope<'a>>) -> Self {
Scope {
parent,
kind,
declared_symbols: Default::default(),
// children: Default::default(),
ops: Default::default(),
}
}
fn scope_of(&self, sym: &JsWord, ctxt: SyntaxContext) -> &'a Scope {
if let Some(prev) = self.declared_symbols.borrow().get(sym) {
if prev.contains(&ctxt) {
return self;
}
}
match self.parent {
Some(ref parent) => parent.scope_of(sym, ctxt),
_ => self,
}
}
fn can_declare(&self, sym: JsWord, ctxt: SyntaxContext) -> bool {
match self.parent {
None => {}
Some(parent) => {
if !parent.can_declare(sym.clone(), ctxt) {
return false;
}
}
}
if let Some(ctxts) = self.declared_symbols.borrow().get(&sym) {
if ctxts.contains(&ctxt) {
// Already declared with same context.
true
} else {
false
}
} else {
// No variable named `sym` is declared
true
}
}
/// Returns all **conflicting** contexts.
///
/// It other words, all `SyntaxContext`s with same `sym` will be returned,
/// even when defined on parent scope.
fn conflicts(&self, sym: JsWord, ctxt: SyntaxContext) -> Vec<SyntaxContext> {
if cfg!(debug_assertions) && LOG {
eprintln!("Finding conflicts for {}{:?} ", sym, ctxt);
}
let sym = self.change_symbol(sym, ctxt);
// let scope = self.scope_of(&sym, ctxt);
let mut cur = Some(self);
let mut ctxts = vec![];
while let Some(scope) = cur {
if let Some(cxs) = scope.declared_symbols.borrow().get(&sym) {
ctxts.extend_from_slice(&cxs);
}
cur = scope.parent;
}
ctxts.retain(|c| *c != ctxt);
ctxts
}
fn change_symbol(&self, mut sym: JsWord, ctxt: SyntaxContext) -> JsWord {
let mut cur = Some(self);
while let Some(scope) = cur {
for op in scope.ops.borrow().iter() {
match *op {
ScopeOp::Rename { ref from, ref to } if from.0 == *sym && from.1 == ctxt => {
if cfg!(debug_assertions) && LOG {
eprintln!("Changing symbol: {}{:?} -> {}", sym, ctxt, to);
}
sym = to.clone()
}
_ => {}
}
}
cur = scope.parent;
}
sym
}
fn is_declared(&self, sym: &JsWord) -> bool {
if self.declared_symbols.borrow().contains_key(sym) {
return true;
}
for op in self.ops.borrow().iter() {
match *op {
ScopeOp::Rename { ref to, .. } if sym == to => return true,
_ => {}
}
}
match self.parent {
Some(parent) => parent.is_declared(sym),
_ => false,
}
}
}
#[macro_export]
macro_rules! track_ident {
($T:tt) => {
impl<'a> Fold<ExportSpecifier> for $T<'a> {
fn fold(&mut self, s: ExportSpecifier) -> ExportSpecifier {
let old_in_var_decl = self.in_var_decl;
self.in_var_decl = false.into();
let s = s.fold_children(self);
self.in_var_decl = old_in_var_decl;
s
}
}
impl<'a> Fold<ImportSpecifier> for $T<'a> {
fn fold(&mut self, s: ImportSpecifier) -> ImportSpecifier {
let old_in_var_decl = self.in_var_decl;
self.in_var_decl = true.into();
let s = match s {
ImportSpecifier::Specific(ImportSpecific { imported: None, .. })
| ImportSpecifier::Namespace(..)
| ImportSpecifier::Default(..) => s.fold_children(self),
ImportSpecifier::Specific(s) => ImportSpecifier::Specific(ImportSpecific {
local: s.local.fold_with(self),
..s
}),
};
self.in_var_decl = old_in_var_decl;
s
}
}
impl<'a> Fold<Constructor> for $T<'a> {
fn fold(&mut self, c: Constructor) -> Constructor {
let old_in_var_decl = self.in_var_decl;
self.in_var_decl = true.into();
let params = c.params.fold_with(self);
self.in_var_decl = old_in_var_decl;
let body = c.body.fold_with(self);
let key = c.key.fold_with(self);
Constructor {
params,
body,
key,
..c
}
}
}
impl<'a> Fold<CatchClause> for $T<'a> {
fn fold(&mut self, c: CatchClause) -> CatchClause {
let old_in_var_decl = self.in_var_decl;
self.in_var_decl = true.into();
let param = c.param.fold_with(self);
self.in_var_decl = old_in_var_decl;
let body = c.body.fold_with(self);
CatchClause { param, body, ..c }
}
}
};
}
track_ident!(Hygiene);
impl<'a> Fold<ArrowExpr> for Hygiene<'a> {
fn fold(&mut self, mut node: ArrowExpr) -> ArrowExpr {
let mut folder = Hygiene {
current: Scope::new(ScopeKind::Fn, Some(&self.current)).into(),
in_var_decl: false.into(),
};
folder.in_var_decl = true.into();
node.params = node.params.fold_with(&mut folder);
folder.in_var_decl = false.into();
node.body = node.body.fold_with(&mut folder);
folder.apply_ops(node)
}
}