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

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

There is a newer version: 9.0.8
Show newest version
/*
 * Copyright 2011 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.javascript.rhino.dtoa.DToA.numberToString;

import com.google.javascript.jscomp.NodeTraversal.AbstractPostOrderCallback;
import com.google.javascript.rhino.IR;
import com.google.javascript.rhino.Node;
import com.google.javascript.rhino.Token;
import java.util.HashSet;
import java.util.Set;

/**
 * Compiler pass that converts primitive calls:
 *
 * 

Converts goog.object.create(key1, val1, key2, val2, ...) where all of the keys are literals * into object literals. * *

Converts goog.object.createSet(key1, key2, ...) into an object literal with the given keys, * where all the values are {@code true}. * *

Converts goog.reflect.objectProperty(propName, object) to JSCompiler_renameProperty * * @author [email protected] (Andrew Grieve) */ final class ClosureOptimizePrimitives implements CompilerPass { static final DiagnosticType DUPLICATE_SET_MEMBER = DiagnosticType.warning("JSC_DUPLICATE_SET_MEMBER", "Found duplicate value ''{0}'' in set"); /** Reference to the JS compiler */ private final AbstractCompiler compiler; /** Whether property renaming is enabled */ private final boolean propertyRenamingEnabled; /** Whether we can use Es6 syntax */ private final boolean canUseEs6Syntax; /** * Identifies all calls to closure primitive functions */ private class FindPrimitives extends AbstractPostOrderCallback { @Override public void visit(NodeTraversal t, Node n, Node parent) { if (n.isCall()) { Node fn = n.getFirstChild(); if (compiler .getCodingConvention() .isPropertyRenameFunction(fn.getOriginalQualifiedName())) { processRenamePropertyCall(n); } else if (fn.matchesQualifiedName("goog$object$create") || fn.matchesQualifiedName("goog.object.create")) { processObjectCreateCall(n); } else if (fn.matchesQualifiedName("goog$object$createSet") || fn.matchesQualifiedName("goog.object.createSet")) { processObjectCreateSetCall(n); } } maybeProcessDomTagName(n); } } /** @param compiler The AbstractCompiler */ ClosureOptimizePrimitives( AbstractCompiler compiler, boolean propertyRenamingEnabled, boolean canUseEs6Syntax) { this.compiler = compiler; this.propertyRenamingEnabled = propertyRenamingEnabled; this.canUseEs6Syntax = canUseEs6Syntax; } @Override public void process(Node externs, Node root) { FindPrimitives pass = new FindPrimitives(); NodeTraversal.traverseEs6(compiler, root, pass); } /** * Converts all of the given call nodes to object literals that are safe to * do so. */ private void processObjectCreateCall(Node callNode) { Node curParam = callNode.getSecondChild(); if (canOptimizeObjectCreate(curParam)) { Node objNode = IR.objectlit().srcref(callNode); while (curParam != null) { Node keyNode = curParam; Node valueNode = curParam.getNext(); curParam = valueNode.getNext(); callNode.removeChild(keyNode); callNode.removeChild(valueNode); addKeyValueToObjLit(objNode, keyNode, valueNode); } callNode.replaceWith(objNode); compiler.reportChangeToEnclosingScope(objNode); } } /** * Converts all of the given call nodes to object literals that are safe to * do so. */ private void processRenamePropertyCall(Node callNode) { // Property reflection calls are only needed if // property renaming is enabled. Replace the call with the // first argument in all other cases. if (!propertyRenamingEnabled) { Node propName = NodeUtil.getArgumentForCallOrNew(callNode, 0); if (propName != null) { callNode.replaceWith(propName.detach()); compiler.reportChangeToEnclosingScope(propName); return; } } Node nameNode = callNode.getFirstChild(); if (nameNode.matchesQualifiedName(NodeUtil.JSC_PROPERTY_NAME_FN)) { return; } Node newTarget = IR.name(NodeUtil.JSC_PROPERTY_NAME_FN).useSourceInfoFrom(nameNode); newTarget.setOriginalName(nameNode.getOriginalQualifiedName()); callNode.replaceChild(nameNode, newTarget); callNode.putBooleanProp(Node.FREE_CALL, true); compiler.reportChangeToEnclosingScope(callNode); } /** * Returns whether the given call to goog.object.create can be converted to an * object literal. */ private boolean canOptimizeObjectCreate(Node firstParam) { Node curParam = firstParam; while (curParam != null) { if (!isOptimizableKey(curParam)) { return false; } curParam = curParam.getNext(); // Check for an odd number of parameters. if (curParam == null) { return false; } curParam = curParam.getNext(); } return true; } /** * Converts all of the given call nodes to object literals that are safe to * do so. */ private void processObjectCreateSetCall(Node callNode) { Node curParam = callNode.getSecondChild(); if (canOptimizeObjectCreateSet(curParam)) { Node objNode = IR.objectlit().srcref(callNode); while (curParam != null) { Node keyNode = curParam; Node valueNode = IR.trueNode().srcref(keyNode); curParam = curParam.getNext(); callNode.removeChild(keyNode); addKeyValueToObjLit(objNode, keyNode, valueNode); } callNode.replaceWith(objNode); compiler.reportChangeToEnclosingScope(objNode); } } /** * Returns whether the given call to goog.object.createSet can be converted to an object literal. */ private boolean canOptimizeObjectCreateSet(Node firstParam) { Node curParam = firstParam; Set keys = new HashSet<>(); while (curParam != null) { // All keys must be strings or numbers, otherwise we can't optimize the call. if (!isOptimizableKey(curParam)) { return false; } if (curParam.isString() || curParam.isNumber()) { String key = curParam.isString() ? curParam.getString() : numberToString(curParam.getDouble()); if (!keys.add(key)) { compiler.report(JSError.make(firstParam.getPrevious(), DUPLICATE_SET_MEMBER, key)); return false; } } curParam = curParam.getNext(); } return true; } private void addKeyValueToObjLit(Node objNode, Node keyNode, Node valueNode) { if (keyNode.isNumber() || keyNode.isString()) { if (keyNode.isNumber()) { keyNode = IR.string(numberToString(keyNode.getDouble())) .srcref(keyNode); } keyNode.setToken(Token.STRING_KEY); keyNode.setQuotedString(); objNode.addChildToBack(IR.propdef(keyNode, valueNode)); } else { objNode.addChildToBack(IR.computedProp(keyNode, valueNode).srcref(keyNode)); } } private boolean isOptimizableKey(Node curParam) { if (this.canUseEs6Syntax) { return !NodeUtil.isStatement(curParam); } else { // Not ES6, all keys must be strings or numbers. return curParam.isString() || curParam.isNumber(); } } /** * Converts the given node to string if it is safe to do so. */ private void maybeProcessDomTagName(Node n) { if (NodeUtil.isLValue(n)) { return; } String prefix = "goog$dom$TagName$"; String tagName; if (n.isName() && n.getString().startsWith(prefix)) { tagName = n.getString().substring(prefix.length()); } else if (n.isGetProp() && !n.getParent().isGetProp() && n.getFirstChild().matchesQualifiedName("goog.dom.TagName")) { tagName = n.getSecondChild().getString() .replaceFirst(".*\\$", ""); // Added by DisambiguateProperties. } else { return; } Node stringNode = IR.string(tagName).srcref(n); n.replaceWith(stringNode); compiler.reportChangeToEnclosingScope(stringNode); } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy