// Copyright 2017 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. #ifndef V8_IC_BINARY_OP_ASSEMBLER_H_ #define V8_IC_BINARY_OP_ASSEMBLER_H_ #include #include "src/codegen/code-stub-assembler.h" namespace v8 { namespace internal { namespace compiler { class CodeAssemblerState; } // namespace compiler class BinaryOpAssembler : public CodeStubAssembler { public: explicit BinaryOpAssembler(compiler::CodeAssemblerState* state) : CodeStubAssembler(state) {} TNode Generate_AddWithFeedback( const LazyNode& context, TNode left, TNode right, TNode slot, const LazyNode& maybe_feedback_vector, UpdateFeedbackMode update_feedback_mode, bool rhs_known_smi); TNode Generate_SubtractWithFeedback( const LazyNode& context, TNode left, TNode right, TNode slot, const LazyNode& maybe_feedback_vector, UpdateFeedbackMode update_feedback_mode, bool rhs_known_smi); TNode Generate_MultiplyWithFeedback( const LazyNode& context, TNode left, TNode right, TNode slot, const LazyNode& maybe_feedback_vector, UpdateFeedbackMode update_feedback_mode, bool rhs_known_smi); TNode Generate_DivideWithFeedback( const LazyNode& context, TNode dividend, TNode divisor, TNode slot, const LazyNode& maybe_feedback_vector, UpdateFeedbackMode update_feedback_mode, bool rhs_known_smi); TNode Generate_ModulusWithFeedback( const LazyNode& context, TNode dividend, TNode divisor, TNode slot, const LazyNode& maybe_feedback_vector, UpdateFeedbackMode update_feedback_mode, bool rhs_known_smi); TNode Generate_ExponentiateWithFeedback( const LazyNode& context, TNode base, TNode exponent, TNode slot, const LazyNode& maybe_feedback_vector, UpdateFeedbackMode update_feedback_mode, bool rhs_known_smi); TNode Generate_BitwiseOrWithFeedback( const LazyNode& context, TNode left, TNode right, TNode slot, const LazyNode& maybe_feedback_vector, UpdateFeedbackMode update_feedback_mode, bool rhs_known_smi) { TNode result = Generate_BitwiseBinaryOpWithFeedback( Operation::kBitwiseOr, left, right, context, slot, maybe_feedback_vector, update_feedback_mode, rhs_known_smi); return result; } TNode Generate_BitwiseXorWithFeedback( const LazyNode& context, TNode left, TNode right, TNode slot, const LazyNode& maybe_feedback_vector, UpdateFeedbackMode update_feedback_mode, bool rhs_known_smi) { TNode result = Generate_BitwiseBinaryOpWithFeedback( Operation::kBitwiseXor, left, right, context, slot, maybe_feedback_vector, update_feedback_mode, rhs_known_smi); return result; } TNode Generate_BitwiseAndWithFeedback( const LazyNode& context, TNode left, TNode right, TNode slot, const LazyNode& maybe_feedback_vector, UpdateFeedbackMode update_feedback_mode, bool rhs_known_smi) { TNode result = Generate_BitwiseBinaryOpWithFeedback( Operation::kBitwiseAnd, left, right, context, slot, maybe_feedback_vector, update_feedback_mode, rhs_known_smi); return result; } TNode Generate_ShiftLeftWithFeedback( const LazyNode& context, TNode left, TNode right, TNode slot, const LazyNode& maybe_feedback_vector, UpdateFeedbackMode update_feedback_mode, bool rhs_known_smi) { TNode result = Generate_BitwiseBinaryOpWithFeedback( Operation::kShiftLeft, left, right, context, slot, maybe_feedback_vector, update_feedback_mode, rhs_known_smi); return result; } TNode Generate_ShiftRightWithFeedback( const LazyNode& context, TNode left, TNode right, TNode slot, const LazyNode& maybe_feedback_vector, UpdateFeedbackMode update_feedback_mode, bool rhs_known_smi) { TNode result = Generate_BitwiseBinaryOpWithFeedback( Operation::kShiftRight, left, right, context, slot, maybe_feedback_vector, update_feedback_mode, rhs_known_smi); return result; } TNode Generate_ShiftRightLogicalWithFeedback( const LazyNode& context, TNode left, TNode right, TNode slot, const LazyNode& maybe_feedback_vector, UpdateFeedbackMode update_feedback_mode, bool rhs_known_smi) { TNode result = Generate_BitwiseBinaryOpWithFeedback( Operation::kShiftRightLogical, left, right, context, slot, maybe_feedback_vector, update_feedback_mode, rhs_known_smi); return result; } TNode Generate_BitwiseBinaryOpWithFeedback( Operation bitwise_op, TNode left, TNode right, const LazyNode& context, TNode slot, const LazyNode& maybe_feedback_vector, UpdateFeedbackMode update_feedback_mode, bool rhs_known_smi) { return rhs_known_smi ? Generate_BitwiseBinaryOpWithSmiOperandAndOptionalFeedback( bitwise_op, left, right, context, &slot, &maybe_feedback_vector, update_feedback_mode) : Generate_BitwiseBinaryOpWithOptionalFeedback( bitwise_op, left, right, context, &slot, &maybe_feedback_vector, update_feedback_mode); } TNode Generate_BitwiseBinaryOp(Operation bitwise_op, TNode left, TNode right, TNode context) { return Generate_BitwiseBinaryOpWithOptionalFeedback( bitwise_op, left, right, [&] { return context; }, nullptr, nullptr, UpdateFeedbackMode::kOptionalFeedback); } private: using SmiOperation = std::function(TNode, TNode, TVariable*)>; using FloatOperation = std::function(TNode, TNode)>; TNode Generate_BinaryOperationWithFeedback( const LazyNode& context, TNode left, TNode right, TNode slot, const LazyNode& maybe_feedback_vector, const SmiOperation& smiOperation, const FloatOperation& floatOperation, Operation op, UpdateFeedbackMode update_feedback_mode, bool rhs_known_smi); TNode Generate_BitwiseBinaryOpWithOptionalFeedback( Operation bitwise_op, TNode left, TNode right, const LazyNode& context, TNode* slot, const LazyNode* maybe_feedback_vector, UpdateFeedbackMode update_feedback_mode); TNode Generate_BitwiseBinaryOpWithSmiOperandAndOptionalFeedback( Operation bitwise_op, TNode left, TNode right, const LazyNode& context, TNode* slot, const LazyNode* maybe_feedback_vector, UpdateFeedbackMode update_feedback_mode); // Check if output is known to be Smi when both operands of bitwise operation // are Smi. bool IsBitwiseOutputKnownSmi(Operation bitwise_op) { switch (bitwise_op) { case Operation::kBitwiseAnd: case Operation::kBitwiseOr: case Operation::kBitwiseXor: case Operation::kShiftRight: return true; default: return false; } } }; } // namespace internal } // namespace v8 #endif // V8_IC_BINARY_OP_ASSEMBLER_H_