// Copyright 2014 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "src/compiler/diamond.h" #include "src/compiler/graph-inl.h" #include "src/compiler/js-builtin-reducer.h" #include "src/compiler/node-matchers.h" #include "src/compiler/node-properties-inl.h" #include "src/types.h" namespace v8 { namespace internal { namespace compiler { // Helper method that assumes replacement nodes are pure values that don't // produce an effect. Replaces {node} with {reduction} and relaxes effects. static Reduction ReplaceWithPureReduction(Node* node, Reduction reduction) { if (reduction.Changed()) { NodeProperties::ReplaceWithValue(node, reduction.replacement()); return reduction; } return Reducer::NoChange(); } // Helper class to access JSCallFunction nodes that are potential candidates // for reduction when they have a BuiltinFunctionId associated with them. class JSCallReduction { public: explicit JSCallReduction(Node* node) : node_(node) {} // Determines whether the node is a JSCallFunction operation that targets a // constant callee being a well-known builtin with a BuiltinFunctionId. bool HasBuiltinFunctionId() { if (node_->opcode() != IrOpcode::kJSCallFunction) return false; HeapObjectMatcher m(NodeProperties::GetValueInput(node_, 0)); if (!m.HasValue() || !m.Value().handle()->IsJSFunction()) return false; Handle function = Handle::cast(m.Value().handle()); return function->shared()->HasBuiltinFunctionId(); } // Retrieves the BuiltinFunctionId as described above. BuiltinFunctionId GetBuiltinFunctionId() { DCHECK_EQ(IrOpcode::kJSCallFunction, node_->opcode()); HeapObjectMatcher m(NodeProperties::GetValueInput(node_, 0)); Handle function = Handle::cast(m.Value().handle()); return function->shared()->builtin_function_id(); } // Determines whether the call takes zero inputs. bool InputsMatchZero() { return GetJSCallArity() == 0; } // Determines whether the call takes one input of the given type. bool InputsMatchOne(Type* t1) { return GetJSCallArity() == 1 && NodeProperties::GetBounds(GetJSCallInput(0)).upper->Is(t1); } // Determines whether the call takes two inputs of the given types. bool InputsMatchTwo(Type* t1, Type* t2) { return GetJSCallArity() == 2 && NodeProperties::GetBounds(GetJSCallInput(0)).upper->Is(t1) && NodeProperties::GetBounds(GetJSCallInput(1)).upper->Is(t2); } // Determines whether the call takes inputs all of the given type. bool InputsMatchAll(Type* t) { for (int i = 0; i < GetJSCallArity(); i++) { if (!NodeProperties::GetBounds(GetJSCallInput(i)).upper->Is(t)) { return false; } } return true; } Node* left() { return GetJSCallInput(0); } Node* right() { return GetJSCallInput(1); } int GetJSCallArity() { DCHECK_EQ(IrOpcode::kJSCallFunction, node_->opcode()); // Skip first (i.e. callee) and second (i.e. receiver) operand. return node_->op()->ValueInputCount() - 2; } Node* GetJSCallInput(int index) { DCHECK_EQ(IrOpcode::kJSCallFunction, node_->opcode()); DCHECK_LT(index, GetJSCallArity()); // Skip first (i.e. callee) and second (i.e. receiver) operand. return NodeProperties::GetValueInput(node_, index + 2); } private: Node* node_; }; // ECMA-262, section 15.8.2.1. Reduction JSBuiltinReducer::ReduceMathAbs(Node* node) { JSCallReduction r(node); if (r.InputsMatchOne(Type::Unsigned32())) { // Math.abs(a:uint32) -> a return Replace(r.left()); } if (r.InputsMatchOne(Type::Number())) { // Math.abs(a:number) -> (a > 0 ? a : 0 - a) Node* value = r.left(); Node* zero = jsgraph()->ZeroConstant(); Node* cmp = graph()->NewNode(simplified()->NumberLessThan(), zero, value); Diamond d(graph(), common(), cmp); Node* neg = graph()->NewNode(simplified()->NumberSubtract(), zero, value); return Replace(d.Phi(kMachNone, value, neg)); } return NoChange(); } // ECMA-262, section 15.8.2.17. Reduction JSBuiltinReducer::ReduceMathSqrt(Node* node) { JSCallReduction r(node); if (r.InputsMatchOne(Type::Number())) { // Math.sqrt(a:number) -> Float64Sqrt(a) Node* value = graph()->NewNode(machine()->Float64Sqrt(), r.left()); return Replace(value); } return NoChange(); } // ECMA-262, section 15.8.2.11. Reduction JSBuiltinReducer::ReduceMathMax(Node* node) { JSCallReduction r(node); if (r.InputsMatchZero()) { // Math.max() -> -Infinity return Replace(jsgraph()->Constant(-V8_INFINITY)); } if (r.InputsMatchOne(Type::Number())) { // Math.max(a:number) -> a return Replace(r.left()); } if (r.InputsMatchAll(Type::Integral32())) { // Math.max(a:int32, b:int32, ...) Node* value = r.GetJSCallInput(0); for (int i = 1; i < r.GetJSCallArity(); i++) { Node* p = r.GetJSCallInput(i); Node* cmp = graph()->NewNode(simplified()->NumberLessThan(), value, p); Diamond d(graph(), common(), cmp); value = d.Phi(kMachNone, p, value); } return Replace(value); } return NoChange(); } // ES6 draft 08-24-14, section 20.2.2.19. Reduction JSBuiltinReducer::ReduceMathImul(Node* node) { JSCallReduction r(node); if (r.InputsMatchTwo(Type::Integral32(), Type::Integral32())) { // Math.imul(a:int32, b:int32) -> Int32Mul(a, b) Node* value = graph()->NewNode(machine()->Int32Mul(), r.left(), r.right()); return Replace(value); } return NoChange(); } // ES6 draft 08-24-14, section 20.2.2.17. Reduction JSBuiltinReducer::ReduceMathFround(Node* node) { JSCallReduction r(node); if (r.InputsMatchOne(Type::Number())) { // Math.fround(a:number) -> TruncateFloat64ToFloat32(a) Node* value = graph()->NewNode(machine()->TruncateFloat64ToFloat32(), r.left()); return Replace(value); } return NoChange(); } // ES6 draft 10-14-14, section 20.2.2.16. Reduction JSBuiltinReducer::ReduceMathFloor(Node* node) { if (!machine()->HasFloat64Floor()) return NoChange(); JSCallReduction r(node); if (r.InputsMatchOne(Type::Number())) { // Math.floor(a:number) -> Float64Floor(a) Node* value = graph()->NewNode(machine()->Float64Floor(), r.left()); return Replace(value); } return NoChange(); } // ES6 draft 10-14-14, section 20.2.2.10. Reduction JSBuiltinReducer::ReduceMathCeil(Node* node) { if (!machine()->HasFloat64Ceil()) return NoChange(); JSCallReduction r(node); if (r.InputsMatchOne(Type::Number())) { // Math.ceil(a:number) -> Float64Ceil(a) Node* value = graph()->NewNode(machine()->Float64Ceil(), r.left()); return Replace(value); } return NoChange(); } Reduction JSBuiltinReducer::Reduce(Node* node) { JSCallReduction r(node); // Dispatch according to the BuiltinFunctionId if present. if (!r.HasBuiltinFunctionId()) return NoChange(); switch (r.GetBuiltinFunctionId()) { case kMathAbs: return ReplaceWithPureReduction(node, ReduceMathAbs(node)); case kMathSqrt: return ReplaceWithPureReduction(node, ReduceMathSqrt(node)); case kMathMax: return ReplaceWithPureReduction(node, ReduceMathMax(node)); case kMathImul: return ReplaceWithPureReduction(node, ReduceMathImul(node)); case kMathFround: return ReplaceWithPureReduction(node, ReduceMathFround(node)); case kMathFloor: return ReplaceWithPureReduction(node, ReduceMathFloor(node)); case kMathCeil: return ReplaceWithPureReduction(node, ReduceMathCeil(node)); default: break; } return NoChange(); } } // namespace compiler } // namespace internal } // namespace v8