All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.google.javascript.jscomp.OptimizeCalls Maven / Gradle / Ivy

Go to download

Closure Compiler is a JavaScript optimizing compiler. It parses your JavaScript, analyzes it, removes dead code and rewrites and minimizes what's left. It also checks syntax, variable references, and types, and warns about common JavaScript pitfalls. It is used in many of Google's JavaScript apps, including Gmail, Google Web Search, Google Maps, and Google Docs.

There is a newer version: v20240317
Show newest version
/*
 * Copyright 2010 The Closure Compiler Authors.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package com.google.javascript.jscomp;

import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkNotNull;
import static com.google.common.base.Preconditions.checkState;

import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableListMultimap;
import com.google.common.collect.ImmutableSet;
import com.google.errorprone.annotations.CanIgnoreReturnValue;
import com.google.javascript.jscomp.NodeTraversal.ScopedCallback;
import com.google.javascript.rhino.Node;
import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import org.jspecify.nullness.Nullable;

/**
 * A root pass that is a container for other passes that should run on with a single call graph.
 *
 * 

Known passes include: * *

    *
  • {@link OptimizeParameters} (remove unused and inline constant parameters) *
  • {@link OptimizeReturns} (remove unused) *
  • {@link DevirtualizeMethods} *
*/ class OptimizeCalls implements CompilerPass { private final AbstractCompiler compiler; private final ImmutableList passes; private final boolean considerExterns; private OptimizeCalls( AbstractCompiler compiler, ImmutableList passes, boolean considerExterns) { this.compiler = compiler; this.passes = passes; this.considerExterns = considerExterns; } static Builder builder() { return new Builder(); } interface CallGraphCompilerPass { void process(Node externs, Node root, ReferenceMap references); } static final class Builder { private AbstractCompiler compiler; private final ImmutableList.Builder passes = ImmutableList.builder(); private @Nullable Boolean considerExterns; // Nullable to force users to specify a value. @CanIgnoreReturnValue public Builder setCompiler(AbstractCompiler compiler) { this.compiler = compiler; return this; } /** * Sets whether or not to include references to extern names and properties in the {@link * ReferenceMap} being generated. * *

If considered, references to externs in both extern code and executable code will * be collected. Otherwise, neither will be. * *

This setting allows extern references to be effectively invisible to passes that should * not mutate them. */ @CanIgnoreReturnValue public Builder setConsiderExterns(boolean b) { this.considerExterns = b; return this; } @CanIgnoreReturnValue public Builder addPass(CallGraphCompilerPass pass) { this.passes.add(pass); return this; } public OptimizeCalls build() { checkNotNull(compiler); checkNotNull(considerExterns); return new OptimizeCalls(compiler, passes.build(), considerExterns); } private Builder() {} } @Override public void process(Node externs, Node root) { // Only global names are collected, which is insufficient if names have not been normalized. checkState(compiler.getLifeCycleStage().isNormalized()); if (passes.isEmpty()) { return; } final ReferenceMap references = new ReferenceMap(); NodeTraversal.traverseRoots( compiler, new ReferenceMapBuildingCallback(references), externs, root); eliminateAccessorsFrom(references); for (CallGraphCompilerPass pass : passes) { pass.process(externs, root, references); } } /** * Delete getter and setter names from {@code references}. * *

Accessor names are disqualified from being in the {@code ReferenceMap}. We don't * intentionally collect them, but other properties may share the same names. One reason why we do * this is exemplified below: * *

{@code
   * class A {
   *   pure() { }
   * }
   *
   * class B {
   *   get pure() { return impure; }
   * }
   *
   * var x = (Math.random() > 0.5) ? new A() : new B();
   * x.pure(); // We can't safely optimize this call.
   * }
*/ private void eliminateAccessorsFrom(ReferenceMap references) { references.props.keySet().removeAll(compiler.getAccessorSummary().getAccessors().keySet()); } /** A reference map for global symbols and properties. */ static class ReferenceMap { private Scope globalScope; private final LinkedHashMap> names = new LinkedHashMap<>(); private final LinkedHashMap> props = new LinkedHashMap<>(); private void addReference(LinkedHashMap> data, String name, Node n) { ArrayList refs = data.computeIfAbsent(name, (String k) -> new ArrayList<>()); refs.add(n); } void addNameReference(String name, Node n) { addReference(names, name, n); } void addPropReference(String name, Node n) { addReference(props, name, n); } Scope getGlobalScope() { return globalScope; } Iterable>> getNameReferences() { return names.entrySet(); } Iterable>> getPropReferences() { return props.entrySet(); } /** * Given a set of references, returns the set of known definitions; specifically, those of the * form: `function x() { }` or `x = ...;` * *

As much as possible, functions are collected from conditional definitions. This is useful * for optimizations that can be performed when the callers are known but all definitions may * not be (unused call results, parameters that are never provided). Examples expressions: * *

    *
  • `(a(), function() {})` *
  • `a && function(){}` *
  • `b || function(){}` *
  • `a ? function() {} : function() {}` *
* * @param definitionSites The definition site nodes to search for associated functions. These * should taken from a {@link ReferenceMap} since only the node types collected by {@link * ReferenceMap} are supported. * @return A mapping from the input {@code definitionSites} to each of their associated * functions. */ static ImmutableListMultimap getFunctionNodes(List definitionSites) { ImmutableListMultimap.Builder result = ImmutableListMultimap.builder(); for (Node def : definitionSites) { result.putAll(def, definitionFunctionNodesFor(def)); } return result.build(); } /** * Collects potential definition FUNCTIONs associated with a method definition site. * * @see {@link #getFunctionNodes} */ private static ImmutableList definitionFunctionNodesFor(Node definitionSite) { if (definitionSite.isGetterDef() || definitionSite.isSetterDef()) { // TODO(nickreid): Support getters and setters. Ignore them for now since they aren't // "called". return ImmutableList.of(); } // Ignore detached nodes. Node parent = definitionSite.getParent(); if (parent == null) { return ImmutableList.of(); } ImmutableList.Builder fns = ImmutableList.builder(); switch (parent.getToken()) { case CLASS: if (definitionSite.isFirstChildOf(parent)) { Node constructorFnDef = NodeUtil.getEs6ClassConstructorMemberFunctionDef(parent); if (constructorFnDef != null) { fns.add(constructorFnDef.getOnlyChild()); } } break; case FUNCTION: fns.add(parent); break; case CLASS_MEMBERS: if (definitionSite.isMemberFunctionDef()) { fns.add(definitionSite.getLastChild()); } else { checkArgument(definitionSite.isMemberFieldDef(), definitionSite); Node value = definitionSite.getFirstChild(); if (value != null) { addValueFunctionNodes(fns, value); } } break; case OBJECTLIT: checkArgument( definitionSite.isStringKey() || definitionSite.isMemberFunctionDef(), // definitionSite); addValueFunctionNodes(fns, definitionSite.getLastChild()); break; case ASSIGN: // Only a candidate if the assign isn't consumed. Node target = parent.getFirstChild(); Node value = parent.getLastChild(); if (definitionSite == target) { addValueFunctionNodes(fns, value); } break; case CONST: case LET: case VAR: if (definitionSite.isName() && definitionSite.hasChildren()) { addValueFunctionNodes(fns, definitionSite.getFirstChild()); } break; default: break; } return fns.build(); } private static void addValueFunctionNodes(ImmutableList.Builder fns, Node n) { // TODO(johnlenz): add member definitions switch (n.getToken()) { case CLASS: { Node constructorFnDef = NodeUtil.getEs6ClassConstructorMemberFunctionDef(n); if (constructorFnDef != null) { fns.add(constructorFnDef.getOnlyChild()); } } break; case FUNCTION: fns.add(n); break; case HOOK: addValueFunctionNodes(fns, n.getSecondChild()); addValueFunctionNodes(fns, n.getLastChild()); break; case OR: case AND: case COALESCE: addValueFunctionNodes(fns, n.getFirstChild()); addValueFunctionNodes(fns, n.getLastChild()); break; case CAST: case COMMA: addValueFunctionNodes(fns, n.getLastChild()); break; default: // do nothing. break; } } /** * Whether the provided node acts as the target function in a new or call or optional chain call * expression including .call expressions. For example, returns true for 'x' in 'x?.call()'. */ static boolean isNormalOrOptChainCallOrNewTarget(Node n) { return isCallTarget(n) || isNewTarget(n) || isOptChainCallTarget(n); } /** * Whether the provided node acts as the target function in a call expression including .call * expressions. For example, returns true for 'x' in 'x.call()'. */ static boolean isCallTarget(Node n) { Node parent = n.getParent(); if (parent.isCall() && n.isFirstChildOf(parent)) { return true; } Node grandParent = parent.getParent(); return parent.isGetProp() && grandParent.isCall() && parent.isFirstChildOf(grandParent) && parent.getString().equals("call"); } /** * Whether the provided node acts as the target function in an optional chain call expression * including .call expressions. For example, returns true for 'x' in 'x?.call()'. */ static boolean isOptChainCallTarget(Node n) { Node parent = n.getParent(); if (parent.isOptChainCall() && n.isFirstChildOf(parent)) { return true; } Node grandParent = parent.getParent(); return parent.isOptChainGetProp() && grandParent.isOptChainCall() && parent.isFirstChildOf(grandParent) && parent.getString().equals("call"); } /** Whether the provided node acts as the target function in a new expression. */ static boolean isNewTarget(Node n) { Node parent = n.getParent(); return parent.isNew() && parent.getFirstChild() == n; } /** * Finds the associated call node for a node for which isNormalOrOptChainCallOrNewTarget returns * true. */ static Node getCallOrNewNodeForTarget(Node n) { Node maybeCall = n.getParent(); checkState(n.isFirstChildOf(maybeCall), "%s\n\n%s", maybeCall, n); if (NodeUtil.isCallOrNew(maybeCall)) { // e.g. `n` input param is the `a` in `a()` or `a?.()` return maybeCall; } else { // e.g. `n` input param is the `a` in `a.b()` or `a?.b()`. Node child = maybeCall; maybeCall = child.getParent(); checkState(NodeUtil.isNormalOrOptChainGetProp(child), child); checkState(NodeUtil.isNormalOrOptChainCall(maybeCall), maybeCall); checkState(child.isFirstChildOf(maybeCall), "%s\n\n%s", maybeCall, child); return maybeCall; } } /** * Finds the call argument node matching the first parameter of the called function for a node * for which isNormalOrOptChainCallOrNewTarget returns true. Specifically, corrects for the * additional argument provided to .call expressions. */ static Node getFirstArgumentForCallOrNewOrDotCall(Node n) { return getArgumentForCallOrNewOrDotCall(n, 0); } /** * Finds the call argument node matching the parameter at the specified index of the called * function for a node for which isNormalOrOptChainCallOrNewTarget returns true. Specifically, * corrects for the additional argument provided to .call expressions. */ static Node getArgumentForCallOrNewOrDotCall(Node n, int index) { int adjustedIndex = index; Node parent = n.getParent(); if (!(parent.isCall() || parent.isOptChainCall() || parent.isNew())) { parent = parent.getParent(); if (NodeUtil.isFunctionObjectCall(parent)) { adjustedIndex++; } } return NodeUtil.getArgumentForCallOrNew(parent, adjustedIndex); } static boolean isSimpleAssignmentTarget(Node n) { Node parent = n.getParent(); // `ref = value;` return parent.isAssign() && n.isFirstChildOf(parent) && parent.getParent().isExprResult(); } } private static Set safeSet(@Nullable Set set) { return (set != null) ? ImmutableSet.copyOf(set) : ImmutableSet.of(); } private final class ReferenceMapBuildingCallback implements ScopedCallback { final Set externProps; final ReferenceMap references; private Scope globalScope; public ReferenceMapBuildingCallback(ReferenceMap references) { this.externProps = safeSet(compiler.getExternProperties()); this.references = references; } @Override public void visit(NodeTraversal t, Node n, Node unused) { switch (n.getToken()) { case NAME: maybeAddNameReference(n.getString(), n); break; case OPTCHAIN_GETPROP: case GETPROP: maybeAddPropReference(n.getString(), n); break; case CALL: // If we are using goog.reflect.objectProperty on this symbol, we will assume that it // gets referenced. Node fnName = n.getFirstChild(); if (compiler.getCodingConvention().isPropertyRenameFunction(fnName)) { Node propName = NodeUtil.getArgumentForCallOrNew(n, 0); if (propName != null) { maybeAddPropReference(propName.getString(), n); } } break; case STRING_KEY: case GETTER_DEF: case SETTER_DEF: case MEMBER_FUNCTION_DEF: case MEMBER_FIELD_DEF: // ignore quoted keys. if (!n.isQuotedStringKey()) { maybeAddPropReference(n.getString(), n); } break; case SUPER: visitSuper(n); break; case COMPUTED_PROP: case COMPUTED_FIELD_DEF: case OPTCHAIN_GETELEM: case GETELEM: // Ignore quoted keys. // TODO(johnlenz): support symbols. case OBJECT_REST: case OBJECT_SPREAD: // Don't worry about invisible accesses using these. To be invoked there would need to be // downstream references that use the actual name. We'd see those. default: break; } } private void visitSuper(Node superNode) { // Determine whether this is a super() constructor call. // If it is, identify the super class and record this as a reference to that. Node parent = superNode.getParent(); if (parent.isCall() && superNode.isFirstChildOf(parent)) { Node enclosingClass = checkNotNull(NodeUtil.getEnclosingClass(parent)); Node extendsNode = enclosingClass.getSecondChild(); checkState(!extendsNode.isEmpty(), "super call appears in class without extends clause"); if (extendsNode.isName()) { maybeAddNameReference(extendsNode.getString(), superNode); } else if (extendsNode.isGetProp()) { // NOTE: Theoretically we could also include an optional chain getprop here, but // A) it's a runtime error if the value ends up being undefined, so that's bad code // B) the author is indicating uncertainty, so we should be cautious. maybeAddPropReference(extendsNode.getString(), superNode); } // else we cannot tell what super() is referencing (e.g. `class extends getMixin() {`) } } private void maybeAddNameReference(String name, Node n) { // TODO(b/129503101): Why are we limiting ourselves to global names? Var var = globalScope.getSlot(name); if (var != null && (considerExterns || !var.isExtern())) { // As every name declaration is unique due to normalizations, it is only necessary to build // the global scope and ask it if it knows about a name as it can never be shadowed. references.addNameReference(name, n); } } private void maybeAddPropReference(String name, Node n) { if (considerExterns || !externProps.contains(name)) { references.addPropReference(name, n); } } @Override public boolean shouldTraverse(NodeTraversal t, Node n, Node parent) { if (n.isScript()) { // Even when considering externs, we only care about top-level identifiers. Dummy function // parameters, for example, shouldn't be considered references. return (considerExterns && t.inGlobalScope()) || !n.isFromExterns(); } else { return true; } } @Override public void enterScope(NodeTraversal t) { if (t.inGlobalScope()) { this.globalScope = t.getScope(); references.globalScope = this.globalScope; } } @Override public void exitScope(NodeTraversal t) {} } /** @return Whether the provide name may be a candidate for call optimizations. */ static boolean mayBeOptimizableName(AbstractCompiler compiler, String name) { if (compiler.getCodingConvention().isExported(name)) { return false; } // Avoid modifying a few special case functions. Specifically, $jscomp.inherits to // recognize 'inherits' calls. (b/27244988) if (name.equals(NodeUtil.JSC_PROPERTY_NAME_FN) || name.equals("inherits") || name.equals("$jscomp$inherits") || name.equals("goog$inherits")) { return false; } return true; } /** @return Whether the reference is a known non-aliasing reference. */ static boolean isAllowedReference(Node n) { Node parent = n.getParent(); switch (parent.getToken()) { case FOR_IN: case FOR_OF: case FOR_AWAIT_OF: // inspecting the properties is allowed. return parent.getSecondChild() == n; case INSTANCEOF: case TYPEOF: case IN: return true; case GETELEM: case GETPROP: case OPTCHAIN_GETPROP: case OPTCHAIN_GETELEM: // Calls escape the "this" value. a.foo() aliases "a" as "this" but general // property references do not. Node grandparent = parent.getParent(); if (n == parent.getFirstChild() && grandparent != null && (grandparent.isCall() || grandparent.isOptChainCall())) { return false; // `a.foo()` or `a?.foo()` or `a?.[foo]()` } return true; case CLASS: if (n.isFirstChildOf(parent)) { // class Name { // this is a definition, not a read reference return false; } else { // class SubClass extends Name { checkState(n.isSecondChildOf(parent), parent); // find the constructor if (NodeUtil.getEs6ClassConstructorMemberFunctionDef(parent) == null) { // The subclass has no explicit constructor, so `new SubClass()` implicitly calls // `new Name(...arguments)`. This hidden call makes it harder to safely optimize the // `Name` constructor, so we won't do it. return false; } else { // We can still optimize the constructor of the class being extended as // long as all child classes have explicit constructors, so we can see the // `super()` calls in them and update them. return true; } } // default: if (NodeUtil.isNameDeclaration(parent) && !n.hasChildren()) { // allow "let x;" return true; } } return false; } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy