Skip to content

Commit

Permalink
perf(transformer/arrow-functions): reduce size of inlined visitor (ox…
Browse files Browse the repository at this point in the history
…c-project#8322)

Follow-on after oxc-project#8024.

Move code out of `visit_expression`.

`visit_expression` is marked `#[inline]`, which is a good idea because it's a hot path and most of the time the `Expression` is not `super()` and the function does nothing.

But this means we want `visit_expression` to be as small as possible to reduce the cost of inlining it. So only retain the fast path for "Is it `super()`? No it's not, so just continue traversal." and move everything else out into a separate function.
  • Loading branch information
overlookmotel committed Jan 8, 2025
1 parent 3dd08e9 commit 62e3f7e
Showing 1 changed file with 23 additions and 20 deletions.
43 changes: 23 additions & 20 deletions crates/oxc_transformer/src/common/arrow_function_converter.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1131,7 +1131,7 @@ impl<'a, 'v> ConstructorBodyThisAfterSuperInserter<'a, 'v> {
}

impl<'a> VisitMut<'a> for ConstructorBodyThisAfterSuperInserter<'a, '_> {
#[inline]
#[inline] // `#[inline]` because is a no-op
fn visit_class(&mut self, _class: &mut Class<'a>) {
// Do not need to insert in nested classes
}
Expand All @@ -1145,7 +1145,9 @@ impl<'a> VisitMut<'a> for ConstructorBodyThisAfterSuperInserter<'a, '_> {
self.visit_statement(stmt);
continue;
};
if let Some(assignment) = self.transform_super_call_expression(&expr_stmt.expression) {

if expr_stmt.expression.is_super_call_expression() {
let assignment = self.create_assignment_to_this_temp_var();
let new_stmt = self.ctx.ast.statement_expression(SPAN, assignment);
new_stmts.push((index, new_stmt));
} else {
Expand All @@ -1160,32 +1162,33 @@ impl<'a> VisitMut<'a> for ConstructorBodyThisAfterSuperInserter<'a, '_> {
}

/// `const A = super()` -> `const A = (super(), _this = this);`
// `#[inline]` to avoid a function call for all `Expressions` which are not `super()` (vast majority)
#[inline]
fn visit_expression(&mut self, expr: &mut Expression<'a>) {
if let Some(assignment) = self.transform_super_call_expression(expr) {
let span = expr.span();
let exprs =
self.ctx.ast.vec_from_array([self.ctx.ast.move_expression(expr), assignment]);
*expr = self.ctx.ast.expression_sequence(span, exprs);
if expr.is_super_call_expression() {
self.transform_super_call_expression(expr);
} else {
walk_expression(self, expr);
}
}
}

impl<'a> ConstructorBodyThisAfterSuperInserter<'a, '_> {
#[inline]
fn transform_super_call_expression(&mut self, expr: &Expression<'a>) -> Option<Expression<'a>> {
if expr.is_super_call_expression() {
let assignment = self.ctx.ast.expression_assignment(
SPAN,
AssignmentOperator::Assign,
self.this_var_binding.create_write_target(self.ctx),
self.ctx.ast.expression_this(SPAN),
);
Some(assignment)
} else {
None
}
/// `super()` -> `(super(), _this = this)`
fn transform_super_call_expression(&mut self, expr: &mut Expression<'a>) {
let assignment = self.create_assignment_to_this_temp_var();
let span = expr.span();
let exprs = self.ctx.ast.vec_from_array([self.ctx.ast.move_expression(expr), assignment]);
*expr = self.ctx.ast.expression_sequence(span, exprs);
}

/// `_this = this`
fn create_assignment_to_this_temp_var(&mut self) -> Expression<'a> {
self.ctx.ast.expression_assignment(
SPAN,
AssignmentOperator::Assign,
self.this_var_binding.create_write_target(self.ctx),
self.ctx.ast.expression_this(SPAN),
)
}
}

0 comments on commit 62e3f7e

Please sign in to comment.