123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259 |
- "use strict";
- exports.__esModule = true;
- exports.visitor = undefined;
- var _getIterator2 = require("babel-runtime/core-js/get-iterator");
- var _getIterator3 = _interopRequireDefault(_getIterator2);
- var _babelTemplate = require("babel-template");
- var _babelTemplate2 = _interopRequireDefault(_babelTemplate);
- var _babelTypes = require("babel-types");
- var t = _interopRequireWildcard(_babelTypes);
- function _interopRequireWildcard(obj) { if (obj && obj.__esModule) { return obj; } else { var newObj = {}; if (obj != null) { for (var key in obj) { if (Object.prototype.hasOwnProperty.call(obj, key)) newObj[key] = obj[key]; } } newObj.default = obj; return newObj; } }
- function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
- var buildRest = (0, _babelTemplate2.default)("\n for (var LEN = ARGUMENTS.length,\n ARRAY = Array(ARRAY_LEN),\n KEY = START;\n KEY < LEN;\n KEY++) {\n ARRAY[ARRAY_KEY] = ARGUMENTS[KEY];\n }\n");
- var restIndex = (0, _babelTemplate2.default)("\n ARGUMENTS.length <= INDEX ? undefined : ARGUMENTS[INDEX]\n");
- var restIndexImpure = (0, _babelTemplate2.default)("\n REF = INDEX, ARGUMENTS.length <= REF ? undefined : ARGUMENTS[REF]\n");
- var restLength = (0, _babelTemplate2.default)("\n ARGUMENTS.length <= OFFSET ? 0 : ARGUMENTS.length - OFFSET\n");
- var memberExpressionOptimisationVisitor = {
- Scope: function Scope(path, state) {
- if (!path.scope.bindingIdentifierEquals(state.name, state.outerBinding)) {
- path.skip();
- }
- },
- Flow: function Flow(path) {
- if (path.isTypeCastExpression()) return;
- path.skip();
- },
- "Function|ClassProperty": function FunctionClassProperty(path, state) {
- var oldNoOptimise = state.noOptimise;
- state.noOptimise = true;
- path.traverse(memberExpressionOptimisationVisitor, state);
- state.noOptimise = oldNoOptimise;
- path.skip();
- },
- ReferencedIdentifier: function ReferencedIdentifier(path, state) {
- var node = path.node;
- if (node.name === "arguments") {
- state.deopted = true;
- }
- if (node.name !== state.name) return;
- if (state.noOptimise) {
- state.deopted = true;
- } else {
- var parentPath = path.parentPath;
- if (parentPath.listKey === "params" && parentPath.key < state.offset) {
- return;
- }
- if (parentPath.isMemberExpression({ object: node })) {
- var grandparentPath = parentPath.parentPath;
- var argsOptEligible = !state.deopted && !(grandparentPath.isAssignmentExpression() && parentPath.node === grandparentPath.node.left || grandparentPath.isLVal() || grandparentPath.isForXStatement() || grandparentPath.isUpdateExpression() || grandparentPath.isUnaryExpression({ operator: "delete" }) || (grandparentPath.isCallExpression() || grandparentPath.isNewExpression()) && parentPath.node === grandparentPath.node.callee);
- if (argsOptEligible) {
- if (parentPath.node.computed) {
- if (parentPath.get("property").isBaseType("number")) {
- state.candidates.push({ cause: "indexGetter", path: path });
- return;
- }
- } else if (parentPath.node.property.name === "length") {
- state.candidates.push({ cause: "lengthGetter", path: path });
- return;
- }
- }
- }
- if (state.offset === 0 && parentPath.isSpreadElement()) {
- var call = parentPath.parentPath;
- if (call.isCallExpression() && call.node.arguments.length === 1) {
- state.candidates.push({ cause: "argSpread", path: path });
- return;
- }
- }
- state.references.push(path);
- }
- },
- BindingIdentifier: function BindingIdentifier(_ref, state) {
- var node = _ref.node;
- if (node.name === state.name) {
- state.deopted = true;
- }
- }
- };
- function hasRest(node) {
- return t.isRestElement(node.params[node.params.length - 1]);
- }
- function optimiseIndexGetter(path, argsId, offset) {
- var index = void 0;
- if (t.isNumericLiteral(path.parent.property)) {
- index = t.numericLiteral(path.parent.property.value + offset);
- } else if (offset === 0) {
- index = path.parent.property;
- } else {
- index = t.binaryExpression("+", path.parent.property, t.numericLiteral(offset));
- }
- var scope = path.scope;
- if (!scope.isPure(index)) {
- var temp = scope.generateUidIdentifierBasedOnNode(index);
- scope.push({ id: temp, kind: "var" });
- path.parentPath.replaceWith(restIndexImpure({
- ARGUMENTS: argsId,
- INDEX: index,
- REF: temp
- }));
- } else {
- path.parentPath.replaceWith(restIndex({
- ARGUMENTS: argsId,
- INDEX: index
- }));
- }
- }
- function optimiseLengthGetter(path, argsId, offset) {
- if (offset) {
- path.parentPath.replaceWith(restLength({
- ARGUMENTS: argsId,
- OFFSET: t.numericLiteral(offset)
- }));
- } else {
- path.replaceWith(argsId);
- }
- }
- var visitor = exports.visitor = {
- Function: function Function(path) {
- var node = path.node,
- scope = path.scope;
- if (!hasRest(node)) return;
- var rest = node.params.pop().argument;
- var argsId = t.identifier("arguments");
- argsId._shadowedFunctionLiteral = path;
- var state = {
- references: [],
- offset: node.params.length,
- argumentsNode: argsId,
- outerBinding: scope.getBindingIdentifier(rest.name),
- candidates: [],
- name: rest.name,
- deopted: false
- };
- path.traverse(memberExpressionOptimisationVisitor, state);
- if (!state.deopted && !state.references.length) {
- for (var _iterator = state.candidates, _isArray = Array.isArray(_iterator), _i = 0, _iterator = _isArray ? _iterator : (0, _getIterator3.default)(_iterator);;) {
- var _ref3;
- if (_isArray) {
- if (_i >= _iterator.length) break;
- _ref3 = _iterator[_i++];
- } else {
- _i = _iterator.next();
- if (_i.done) break;
- _ref3 = _i.value;
- }
- var _ref4 = _ref3;
- var _path = _ref4.path,
- cause = _ref4.cause;
- switch (cause) {
- case "indexGetter":
- optimiseIndexGetter(_path, argsId, state.offset);
- break;
- case "lengthGetter":
- optimiseLengthGetter(_path, argsId, state.offset);
- break;
- default:
- _path.replaceWith(argsId);
- }
- }
- return;
- }
- state.references = state.references.concat(state.candidates.map(function (_ref5) {
- var path = _ref5.path;
- return path;
- }));
- state.deopted = state.deopted || !!node.shadow;
- var start = t.numericLiteral(node.params.length);
- var key = scope.generateUidIdentifier("key");
- var len = scope.generateUidIdentifier("len");
- var arrKey = key;
- var arrLen = len;
- if (node.params.length) {
- arrKey = t.binaryExpression("-", key, start);
- arrLen = t.conditionalExpression(t.binaryExpression(">", len, start), t.binaryExpression("-", len, start), t.numericLiteral(0));
- }
- var loop = buildRest({
- ARGUMENTS: argsId,
- ARRAY_KEY: arrKey,
- ARRAY_LEN: arrLen,
- START: start,
- ARRAY: rest,
- KEY: key,
- LEN: len
- });
- if (state.deopted) {
- loop._blockHoist = node.params.length + 1;
- node.body.body.unshift(loop);
- } else {
- loop._blockHoist = 1;
- var target = path.getEarliestCommonAncestorFrom(state.references).getStatementParent();
- target.findParent(function (path) {
- if (path.isLoop()) {
- target = path;
- } else {
- return path.isFunction();
- }
- });
- target.insertBefore(loop);
- }
- }
- };
|