2019-01-29 17:56:16 +03:00
|
|
|
use self::ops::{Operator, ScopeOp};
|
2019-02-27 16:40:19 +03:00
|
|
|
use crate::{
|
|
|
|
pass::Pass,
|
|
|
|
scope::{IdentType, ScopeKind},
|
|
|
|
};
|
2019-01-29 17:56:16 +03:00
|
|
|
use ast::*;
|
2019-02-05 06:50:19 +03:00
|
|
|
use fxhash::FxHashMap;
|
2019-01-29 17:56:16 +03:00
|
|
|
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;
|
|
|
|
|
2019-02-05 06:50:19 +03:00
|
|
|
struct Hygiene<'a> {
|
2019-02-27 16:40:19 +03:00
|
|
|
current: Scope<'a>,
|
|
|
|
ident_type: IdentType,
|
2019-02-05 06:50:19 +03:00
|
|
|
}
|
|
|
|
|
2019-01-29 17:56:16 +03:00
|
|
|
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,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-27 16:40:19 +03:00
|
|
|
pub fn hygiene() -> impl Pass + 'static {
|
2019-02-05 06:50:19 +03:00
|
|
|
#[derive(Clone, Copy)]
|
2019-01-29 17:56:16 +03:00
|
|
|
struct MarkClearer;
|
|
|
|
impl Fold<Span> for MarkClearer {
|
|
|
|
fn fold(&mut self, span: Span) -> Span {
|
|
|
|
span.with_ctxt(SyntaxContext::empty())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-05 06:50:19 +03:00
|
|
|
chain_at!(
|
|
|
|
Module,
|
|
|
|
Hygiene {
|
|
|
|
current: Default::default(),
|
2019-02-27 16:40:19 +03:00
|
|
|
ident_type: IdentType::Ref,
|
2019-02-05 06:50:19 +03:00
|
|
|
},
|
|
|
|
MarkClearer
|
|
|
|
)
|
2019-01-29 17:56:16 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
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 {
|
2019-02-05 06:50:19 +03:00
|
|
|
current: Scope::new(ScopeKind::Block, Some(&self.current)).into(),
|
2019-02-27 16:40:19 +03:00
|
|
|
ident_type: IdentType::Ref,
|
2019-01-29 17:56:16 +03:00
|
|
|
};
|
|
|
|
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 {
|
2019-02-05 06:50:19 +03:00
|
|
|
current: Scope::new(ScopeKind::Fn, Some(&self.current)).into(),
|
2019-02-27 16:40:19 +03:00
|
|
|
ident_type: IdentType::Ref,
|
2019-01-29 17:56:16 +03:00
|
|
|
};
|
|
|
|
|
2019-02-27 16:40:19 +03:00
|
|
|
folder.ident_type = IdentType::Binding;
|
2019-01-29 17:56:16 +03:00
|
|
|
node.params = node.params.fold_with(&mut folder);
|
|
|
|
|
2019-02-27 16:40:19 +03:00
|
|
|
folder.ident_type = IdentType::Ref;
|
2019-01-29 17:56:16 +03:00
|
|
|
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 {
|
2019-02-27 16:40:19 +03:00
|
|
|
let old = self.ident_type;
|
|
|
|
self.ident_type = IdentType::Binding;
|
2019-01-29 17:56:16 +03:00
|
|
|
let name = decl.name.fold_with(self);
|
2019-02-27 16:40:19 +03:00
|
|
|
self.ident_type = old;
|
2019-01-29 17:56:16 +03:00
|
|
|
|
|
|
|
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 {
|
2019-02-27 16:40:19 +03:00
|
|
|
match self.ident_type {
|
|
|
|
IdentType::Binding => self.add_declared_ref(i.clone()),
|
|
|
|
IdentType::Ref => {
|
|
|
|
self.add_used_ref(i.clone());
|
|
|
|
}
|
|
|
|
IdentType::Label => {
|
|
|
|
// We currently does not touch labels
|
|
|
|
return i;
|
|
|
|
}
|
2019-01-29 17:56:16 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
i
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Fold<Expr> for Hygiene<'a> {
|
|
|
|
fn fold(&mut self, node: Expr) -> Expr {
|
2019-02-27 16:40:19 +03:00
|
|
|
let old = self.ident_type;
|
|
|
|
self.ident_type = IdentType::Ref;
|
2019-01-29 17:56:16 +03:00
|
|
|
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),
|
|
|
|
};
|
|
|
|
|
2019-02-27 16:40:19 +03:00
|
|
|
self.ident_type = old;
|
2019-01-29 17:56:16 +03:00
|
|
|
|
|
|
|
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
|
2019-02-05 06:50:19 +03:00
|
|
|
pub declared_symbols: RefCell<FxHashMap<JsWord, Vec<SyntaxContext>>>,
|
2019-01-29 17:56:16 +03:00
|
|
|
|
|
|
|
pub(crate) ops: RefCell<Vec<ScopeOp>>,
|
|
|
|
}
|
|
|
|
|
2019-02-05 06:50:19 +03:00
|
|
|
impl<'a> Default for Scope<'a> {
|
|
|
|
fn default() -> Self {
|
|
|
|
Scope::new(ScopeKind::Fn, None)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-29 17:56:16 +03:00
|
|
|
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,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-05 06:50:19 +03:00
|
|
|
#[macro_export]
|
|
|
|
macro_rules! track_ident {
|
|
|
|
($T:tt) => {
|
|
|
|
impl<'a> Fold<ExportSpecifier> for $T<'a> {
|
|
|
|
fn fold(&mut self, s: ExportSpecifier) -> ExportSpecifier {
|
2019-02-27 16:40:19 +03:00
|
|
|
let old = self.ident_type;
|
|
|
|
self.ident_type = IdentType::Ref;
|
2019-01-29 17:56:16 +03:00
|
|
|
|
2019-02-05 06:50:19 +03:00
|
|
|
let s = s.fold_children(self);
|
2019-01-29 17:56:16 +03:00
|
|
|
|
2019-02-27 16:40:19 +03:00
|
|
|
self.ident_type = old;
|
2019-01-29 17:56:16 +03:00
|
|
|
|
2019-02-05 06:50:19 +03:00
|
|
|
s
|
|
|
|
}
|
|
|
|
}
|
2019-01-29 17:56:16 +03:00
|
|
|
|
2019-02-05 06:50:19 +03:00
|
|
|
impl<'a> Fold<ImportSpecifier> for $T<'a> {
|
|
|
|
fn fold(&mut self, s: ImportSpecifier) -> ImportSpecifier {
|
2019-02-27 16:40:19 +03:00
|
|
|
let old = self.ident_type;
|
|
|
|
self.ident_type = IdentType::Binding;
|
2019-01-29 17:56:16 +03:00
|
|
|
|
2019-02-05 06:50:19 +03:00
|
|
|
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
|
|
|
|
}),
|
|
|
|
};
|
2019-01-29 17:56:16 +03:00
|
|
|
|
2019-02-27 16:40:19 +03:00
|
|
|
self.ident_type = old;
|
2019-02-05 06:50:19 +03:00
|
|
|
|
|
|
|
s
|
|
|
|
}
|
2019-01-29 17:56:16 +03:00
|
|
|
}
|
2019-02-05 06:50:19 +03:00
|
|
|
|
|
|
|
impl<'a> Fold<Constructor> for $T<'a> {
|
|
|
|
fn fold(&mut self, c: Constructor) -> Constructor {
|
2019-02-27 16:40:19 +03:00
|
|
|
let old = self.ident_type;
|
|
|
|
self.ident_type = IdentType::Binding;
|
2019-02-05 06:50:19 +03:00
|
|
|
let params = c.params.fold_with(self);
|
2019-02-27 16:40:19 +03:00
|
|
|
self.ident_type = old;
|
2019-02-05 06:50:19 +03:00
|
|
|
|
|
|
|
let body = c.body.fold_with(self);
|
|
|
|
let key = c.key.fold_with(self);
|
|
|
|
|
|
|
|
Constructor {
|
|
|
|
params,
|
|
|
|
body,
|
|
|
|
key,
|
|
|
|
..c
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-20 05:35:41 +03:00
|
|
|
impl<'a> Fold<SetterProp> for $T<'a> {
|
|
|
|
fn fold(&mut self, f: SetterProp) -> SetterProp {
|
2019-02-27 16:40:19 +03:00
|
|
|
let old = self.ident_type;
|
|
|
|
self.ident_type = IdentType::Binding;
|
2019-02-20 05:35:41 +03:00
|
|
|
let param = f.param.fold_with(self);
|
2019-02-27 16:40:19 +03:00
|
|
|
self.ident_type = old;
|
2019-02-20 05:35:41 +03:00
|
|
|
|
|
|
|
let body = f.body.fold_with(self);
|
|
|
|
|
|
|
|
SetterProp { param, body, ..f }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// impl<'a> Fold<GetterProp> for $T<'a> {
|
|
|
|
// fn fold(&mut self, f: GetterProp) -> GetterProp {
|
|
|
|
// let body = f.body.fold_with(self);
|
|
|
|
|
|
|
|
// GetterProp { body, ..c }
|
|
|
|
// }
|
|
|
|
// }
|
|
|
|
|
2019-02-05 06:50:19 +03:00
|
|
|
impl<'a> Fold<CatchClause> for $T<'a> {
|
|
|
|
fn fold(&mut self, c: CatchClause) -> CatchClause {
|
2019-02-27 16:40:19 +03:00
|
|
|
let old = self.ident_type;
|
|
|
|
self.ident_type = IdentType::Binding;
|
2019-02-05 06:50:19 +03:00
|
|
|
let param = c.param.fold_with(self);
|
2019-02-27 16:40:19 +03:00
|
|
|
self.ident_type = old;
|
2019-02-05 06:50:19 +03:00
|
|
|
|
|
|
|
let body = c.body.fold_with(self);
|
|
|
|
|
|
|
|
CatchClause { param, body, ..c }
|
|
|
|
}
|
|
|
|
}
|
2019-02-27 16:40:19 +03:00
|
|
|
|
|
|
|
impl<'a> Fold<LabeledStmt> for $T<'a> {
|
|
|
|
fn fold(&mut self, s: LabeledStmt) -> LabeledStmt {
|
|
|
|
let old = self.ident_type;
|
|
|
|
self.ident_type = IdentType::Label;
|
|
|
|
let label = s.label.fold_with(self);
|
|
|
|
self.ident_type = old;
|
|
|
|
|
|
|
|
let body = s.body.fold_with(self);
|
|
|
|
|
|
|
|
LabeledStmt { label, body, ..s }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Fold<BreakStmt> for $T<'a> {
|
|
|
|
fn fold(&mut self, s: BreakStmt) -> BreakStmt {
|
|
|
|
let old = self.ident_type;
|
|
|
|
self.ident_type = IdentType::Label;
|
|
|
|
let label = s.label.fold_with(self);
|
|
|
|
self.ident_type = old;
|
|
|
|
|
|
|
|
BreakStmt { label, ..s }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Fold<ContinueStmt> for $T<'a> {
|
|
|
|
fn fold(&mut self, s: ContinueStmt) -> ContinueStmt {
|
|
|
|
let old = self.ident_type;
|
|
|
|
self.ident_type = IdentType::Label;
|
|
|
|
let label = s.label.fold_with(self);
|
|
|
|
self.ident_type = old;
|
|
|
|
|
|
|
|
ContinueStmt { label, ..s }
|
|
|
|
}
|
|
|
|
}
|
2019-03-07 16:42:16 +03:00
|
|
|
|
|
|
|
impl<'a> Fold<ClassDecl> for $T<'a> {
|
|
|
|
fn fold(&mut self, n: ClassDecl) -> ClassDecl {
|
|
|
|
let old = self.ident_type;
|
|
|
|
self.ident_type = IdentType::Binding;
|
|
|
|
let ident = n.ident.fold_with(self);
|
|
|
|
self.ident_type = old;
|
|
|
|
|
|
|
|
let class = n.class.fold_with(self);
|
|
|
|
|
|
|
|
ClassDecl { ident, class, ..n }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Fold<ClassExpr> for $T<'a> {
|
|
|
|
fn fold(&mut self, n: ClassExpr) -> ClassExpr {
|
|
|
|
let old = self.ident_type;
|
|
|
|
self.ident_type = IdentType::Binding;
|
|
|
|
let ident = n.ident.fold_with(self);
|
|
|
|
self.ident_type = old;
|
|
|
|
|
|
|
|
let class = n.class.fold_with(self);
|
|
|
|
|
|
|
|
ClassExpr { ident, class, ..n }
|
|
|
|
}
|
|
|
|
}
|
2019-02-05 06:50:19 +03:00
|
|
|
};
|
2019-01-29 17:56:16 +03:00
|
|
|
}
|
|
|
|
|
2019-02-05 06:50:19 +03:00
|
|
|
track_ident!(Hygiene);
|
2019-01-29 17:56:16 +03:00
|
|
|
|
2019-02-05 06:50:19 +03:00
|
|
|
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(),
|
2019-02-27 16:40:19 +03:00
|
|
|
ident_type: IdentType::Ref,
|
2019-02-05 06:50:19 +03:00
|
|
|
};
|
|
|
|
|
2019-02-27 16:40:19 +03:00
|
|
|
folder.ident_type = IdentType::Binding;
|
2019-02-05 06:50:19 +03:00
|
|
|
node.params = node.params.fold_with(&mut folder);
|
2019-01-29 17:56:16 +03:00
|
|
|
|
2019-02-27 16:40:19 +03:00
|
|
|
folder.ident_type = IdentType::Ref;
|
2019-02-05 06:50:19 +03:00
|
|
|
node.body = node.body.fold_with(&mut folder);
|
|
|
|
|
|
|
|
folder.apply_ops(node)
|
2019-01-29 17:56:16 +03:00
|
|
|
}
|
|
|
|
}
|