// Copyright 2015 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_INTERPRETER_BYTECODES_H_ #define V8_INTERPRETER_BYTECODES_H_ #include <cstdint> #include <iosfwd> #include <string> #include <vector> #include "src/globals.h" #include "src/interpreter/bytecode-operands.h" // This interface and it's implementation are independent of the // libv8_base library as they are used by the interpreter and the // standalone mkpeephole table generator program. namespace v8 { namespace internal { namespace interpreter { // The list of bytecodes which are interpreted by the interpreter. // Format is V(<bytecode>, <accumulator_use>, <operands>). #define BYTECODE_LIST(V) \ /* Extended width operands */ \ V(Wide, AccumulatorUse::kNone) \ V(ExtraWide, AccumulatorUse::kNone) \ \ /* Debug Breakpoints - one for each possible size of unscaled bytecodes */ \ /* and one for each operand widening prefix bytecode */ \ V(DebugBreakWide, AccumulatorUse::kReadWrite) \ V(DebugBreakExtraWide, AccumulatorUse::kReadWrite) \ V(DebugBreak0, AccumulatorUse::kReadWrite) \ V(DebugBreak1, AccumulatorUse::kReadWrite, OperandType::kReg) \ V(DebugBreak2, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kReg) \ V(DebugBreak3, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kReg, OperandType::kReg) \ V(DebugBreak4, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kReg, OperandType::kReg, OperandType::kReg) \ V(DebugBreak5, AccumulatorUse::kReadWrite, OperandType::kRuntimeId, \ OperandType::kReg, OperandType::kReg) \ V(DebugBreak6, AccumulatorUse::kReadWrite, OperandType::kRuntimeId, \ OperandType::kReg, OperandType::kReg, OperandType::kReg) \ \ /* Loading the accumulator */ \ V(LdaZero, AccumulatorUse::kWrite) \ V(LdaSmi, AccumulatorUse::kWrite, OperandType::kImm) \ V(LdaUndefined, AccumulatorUse::kWrite) \ V(LdaNull, AccumulatorUse::kWrite) \ V(LdaTheHole, AccumulatorUse::kWrite) \ V(LdaTrue, AccumulatorUse::kWrite) \ V(LdaFalse, AccumulatorUse::kWrite) \ V(LdaConstant, AccumulatorUse::kWrite, OperandType::kIdx) \ \ /* Globals */ \ V(LdaGlobal, AccumulatorUse::kWrite, OperandType::kIdx, OperandType::kIdx) \ V(LdaGlobalInsideTypeof, AccumulatorUse::kWrite, OperandType::kIdx, \ OperandType::kIdx) \ V(StaGlobal, AccumulatorUse::kRead, OperandType::kIdx, OperandType::kIdx) \ \ /* Context operations */ \ V(PushContext, AccumulatorUse::kRead, OperandType::kRegOut) \ V(PopContext, AccumulatorUse::kNone, OperandType::kReg) \ V(LdaContextSlot, AccumulatorUse::kWrite, OperandType::kReg, \ OperandType::kIdx, OperandType::kUImm) \ V(LdaImmutableContextSlot, AccumulatorUse::kWrite, OperandType::kReg, \ OperandType::kIdx, OperandType::kUImm) \ V(LdaCurrentContextSlot, AccumulatorUse::kWrite, OperandType::kIdx) \ V(LdaImmutableCurrentContextSlot, AccumulatorUse::kWrite, OperandType::kIdx) \ V(StaContextSlot, AccumulatorUse::kRead, OperandType::kReg, \ OperandType::kIdx, OperandType::kUImm) \ V(StaCurrentContextSlot, AccumulatorUse::kRead, OperandType::kIdx) \ \ /* Load-Store lookup slots */ \ V(LdaLookupSlot, AccumulatorUse::kWrite, OperandType::kIdx) \ V(LdaLookupContextSlot, AccumulatorUse::kWrite, OperandType::kIdx, \ OperandType::kIdx, OperandType::kUImm) \ V(LdaLookupGlobalSlot, AccumulatorUse::kWrite, OperandType::kIdx, \ OperandType::kIdx, OperandType::kUImm) \ V(LdaLookupSlotInsideTypeof, AccumulatorUse::kWrite, OperandType::kIdx) \ V(LdaLookupContextSlotInsideTypeof, AccumulatorUse::kWrite, \ OperandType::kIdx, OperandType::kIdx, OperandType::kUImm) \ V(LdaLookupGlobalSlotInsideTypeof, AccumulatorUse::kWrite, \ OperandType::kIdx, OperandType::kIdx, OperandType::kUImm) \ V(StaLookupSlot, AccumulatorUse::kReadWrite, OperandType::kIdx, \ OperandType::kFlag8) \ \ /* Register-accumulator transfers */ \ V(Ldar, AccumulatorUse::kWrite, OperandType::kReg) \ V(Star, AccumulatorUse::kRead, OperandType::kRegOut) \ \ /* Register-register transfers */ \ V(Mov, AccumulatorUse::kNone, OperandType::kReg, OperandType::kRegOut) \ \ /* Property loads (LoadIC) operations */ \ V(LdaNamedProperty, AccumulatorUse::kWrite, OperandType::kReg, \ OperandType::kIdx, OperandType::kIdx) \ V(LdaKeyedProperty, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kIdx) \ \ /* Operations on module variables */ \ V(LdaModuleVariable, AccumulatorUse::kWrite, OperandType::kImm, \ OperandType::kUImm) \ V(StaModuleVariable, AccumulatorUse::kRead, OperandType::kImm, \ OperandType::kUImm) \ \ /* Propery stores (StoreIC) operations */ \ V(StaNamedProperty, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kIdx, OperandType::kIdx) \ V(StaNamedOwnProperty, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kIdx, OperandType::kIdx) \ V(StaKeyedProperty, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kReg, OperandType::kIdx) \ V(StaInArrayLiteral, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kReg, OperandType::kIdx) \ V(StaDataPropertyInLiteral, AccumulatorUse::kRead, OperandType::kReg, \ OperandType::kReg, OperandType::kFlag8, OperandType::kIdx) \ V(CollectTypeProfile, AccumulatorUse::kRead, OperandType::kImm) \ \ /* Binary Operators */ \ V(Add, AccumulatorUse::kReadWrite, OperandType::kReg, OperandType::kIdx) \ V(Sub, AccumulatorUse::kReadWrite, OperandType::kReg, OperandType::kIdx) \ V(Mul, AccumulatorUse::kReadWrite, OperandType::kReg, OperandType::kIdx) \ V(Div, AccumulatorUse::kReadWrite, OperandType::kReg, OperandType::kIdx) \ V(Mod, AccumulatorUse::kReadWrite, OperandType::kReg, OperandType::kIdx) \ V(Exp, AccumulatorUse::kReadWrite, OperandType::kReg, OperandType::kIdx) \ V(BitwiseOr, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kIdx) \ V(BitwiseXor, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kIdx) \ V(BitwiseAnd, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kIdx) \ V(ShiftLeft, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kIdx) \ V(ShiftRight, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kIdx) \ V(ShiftRightLogical, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kIdx) \ \ /* Binary operators with immediate operands */ \ V(AddSmi, AccumulatorUse::kReadWrite, OperandType::kImm, OperandType::kIdx) \ V(SubSmi, AccumulatorUse::kReadWrite, OperandType::kImm, OperandType::kIdx) \ V(MulSmi, AccumulatorUse::kReadWrite, OperandType::kImm, OperandType::kIdx) \ V(DivSmi, AccumulatorUse::kReadWrite, OperandType::kImm, OperandType::kIdx) \ V(ModSmi, AccumulatorUse::kReadWrite, OperandType::kImm, OperandType::kIdx) \ V(ExpSmi, AccumulatorUse::kReadWrite, OperandType::kImm, OperandType::kIdx) \ V(BitwiseOrSmi, AccumulatorUse::kReadWrite, OperandType::kImm, \ OperandType::kIdx) \ V(BitwiseXorSmi, AccumulatorUse::kReadWrite, OperandType::kImm, \ OperandType::kIdx) \ V(BitwiseAndSmi, AccumulatorUse::kReadWrite, OperandType::kImm, \ OperandType::kIdx) \ V(ShiftLeftSmi, AccumulatorUse::kReadWrite, OperandType::kImm, \ OperandType::kIdx) \ V(ShiftRightSmi, AccumulatorUse::kReadWrite, OperandType::kImm, \ OperandType::kIdx) \ V(ShiftRightLogicalSmi, AccumulatorUse::kReadWrite, OperandType::kImm, \ OperandType::kIdx) \ \ /* Unary Operators */ \ V(Inc, AccumulatorUse::kReadWrite, OperandType::kIdx) \ V(Dec, AccumulatorUse::kReadWrite, OperandType::kIdx) \ V(Negate, AccumulatorUse::kReadWrite, OperandType::kIdx) \ V(BitwiseNot, AccumulatorUse::kReadWrite, OperandType::kIdx) \ V(ToBooleanLogicalNot, AccumulatorUse::kReadWrite) \ V(LogicalNot, AccumulatorUse::kReadWrite) \ V(TypeOf, AccumulatorUse::kReadWrite) \ V(DeletePropertyStrict, AccumulatorUse::kReadWrite, OperandType::kReg) \ V(DeletePropertySloppy, AccumulatorUse::kReadWrite, OperandType::kReg) \ \ /* GetSuperConstructor operator */ \ V(GetSuperConstructor, AccumulatorUse::kRead, OperandType::kRegOut) \ \ /* Call operations */ \ V(CallAnyReceiver, AccumulatorUse::kWrite, OperandType::kReg, \ OperandType::kRegList, OperandType::kRegCount, OperandType::kIdx) \ V(CallProperty, AccumulatorUse::kWrite, OperandType::kReg, \ OperandType::kRegList, OperandType::kRegCount, OperandType::kIdx) \ V(CallProperty0, AccumulatorUse::kWrite, OperandType::kReg, \ OperandType::kReg, OperandType::kIdx) \ V(CallProperty1, AccumulatorUse::kWrite, OperandType::kReg, \ OperandType::kReg, OperandType::kReg, OperandType::kIdx) \ V(CallProperty2, AccumulatorUse::kWrite, OperandType::kReg, \ OperandType::kReg, OperandType::kReg, OperandType::kReg, \ OperandType::kIdx) \ V(CallUndefinedReceiver, AccumulatorUse::kWrite, OperandType::kReg, \ OperandType::kRegList, OperandType::kRegCount, OperandType::kIdx) \ V(CallUndefinedReceiver0, AccumulatorUse::kWrite, OperandType::kReg, \ OperandType::kIdx) \ V(CallUndefinedReceiver1, AccumulatorUse::kWrite, OperandType::kReg, \ OperandType::kReg, OperandType::kIdx) \ V(CallUndefinedReceiver2, AccumulatorUse::kWrite, OperandType::kReg, \ OperandType::kReg, OperandType::kReg, OperandType::kIdx) \ V(CallWithSpread, AccumulatorUse::kWrite, OperandType::kReg, \ OperandType::kRegList, OperandType::kRegCount, OperandType::kIdx) \ V(CallRuntime, AccumulatorUse::kWrite, OperandType::kRuntimeId, \ OperandType::kRegList, OperandType::kRegCount) \ V(CallRuntimeForPair, AccumulatorUse::kNone, OperandType::kRuntimeId, \ OperandType::kRegList, OperandType::kRegCount, OperandType::kRegOutPair) \ V(CallJSRuntime, AccumulatorUse::kWrite, OperandType::kNativeContextIndex, \ OperandType::kRegList, OperandType::kRegCount) \ \ /* Intrinsics */ \ V(InvokeIntrinsic, AccumulatorUse::kWrite, OperandType::kIntrinsicId, \ OperandType::kRegList, OperandType::kRegCount) \ \ /* Construct operators */ \ V(Construct, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kRegList, OperandType::kRegCount, OperandType::kIdx) \ V(ConstructWithSpread, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kRegList, OperandType::kRegCount, OperandType::kIdx) \ \ /* Test Operators */ \ V(TestEqual, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kIdx) \ V(TestEqualStrict, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kIdx) \ V(TestLessThan, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kIdx) \ V(TestGreaterThan, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kIdx) \ V(TestLessThanOrEqual, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kIdx) \ V(TestGreaterThanOrEqual, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kIdx) \ V(TestReferenceEqual, AccumulatorUse::kReadWrite, OperandType::kReg) \ V(TestInstanceOf, AccumulatorUse::kReadWrite, OperandType::kReg, \ OperandType::kIdx) \ V(TestIn, AccumulatorUse::kReadWrite, OperandType::kReg) \ V(TestUndetectable, AccumulatorUse::kReadWrite) \ V(TestNull, AccumulatorUse::kReadWrite) \ V(TestUndefined, AccumulatorUse::kReadWrite) \ V(TestTypeOf, AccumulatorUse::kReadWrite, OperandType::kFlag8) \ \ /* Cast operators */ \ V(ToName, AccumulatorUse::kRead, OperandType::kRegOut) \ V(ToNumber, AccumulatorUse::kReadWrite, OperandType::kIdx) \ V(ToNumeric, AccumulatorUse::kReadWrite, OperandType::kIdx) \ V(ToObject, AccumulatorUse::kRead, OperandType::kRegOut) \ V(ToString, AccumulatorUse::kReadWrite) \ \ /* Literals */ \ V(CreateRegExpLiteral, AccumulatorUse::kWrite, OperandType::kIdx, \ OperandType::kIdx, OperandType::kFlag8) \ V(CreateArrayLiteral, AccumulatorUse::kWrite, OperandType::kIdx, \ OperandType::kIdx, OperandType::kFlag8) \ V(CreateEmptyArrayLiteral, AccumulatorUse::kWrite, OperandType::kIdx) \ V(CreateObjectLiteral, AccumulatorUse::kNone, OperandType::kIdx, \ OperandType::kIdx, OperandType::kFlag8, OperandType::kRegOut) \ V(CreateEmptyObjectLiteral, AccumulatorUse::kWrite) \ V(CloneObject, AccumulatorUse::kWrite, OperandType::kReg, \ OperandType::kFlag8, OperandType::kIdx) \ \ /* Tagged templates */ \ V(GetTemplateObject, AccumulatorUse::kWrite, OperandType::kIdx, \ OperandType::kIdx) \ \ /* Closure allocation */ \ V(CreateClosure, AccumulatorUse::kWrite, OperandType::kIdx, \ OperandType::kIdx, OperandType::kFlag8) \ \ /* Context allocation */ \ V(CreateBlockContext, AccumulatorUse::kWrite, OperandType::kIdx) \ V(CreateCatchContext, AccumulatorUse::kWrite, OperandType::kReg, \ OperandType::kIdx) \ V(CreateFunctionContext, AccumulatorUse::kWrite, OperandType::kIdx, \ OperandType::kUImm) \ V(CreateEvalContext, AccumulatorUse::kWrite, OperandType::kIdx, \ OperandType::kUImm) \ V(CreateWithContext, AccumulatorUse::kWrite, OperandType::kReg, \ OperandType::kIdx) \ \ /* Arguments allocation */ \ V(CreateMappedArguments, AccumulatorUse::kWrite) \ V(CreateUnmappedArguments, AccumulatorUse::kWrite) \ V(CreateRestParameter, AccumulatorUse::kWrite) \ \ /* Control Flow -- carefully ordered for efficient checks */ \ /* - [Unconditional jumps] */ \ V(JumpLoop, AccumulatorUse::kNone, OperandType::kUImm, OperandType::kImm) \ /* - [Forward jumps] */ \ V(Jump, AccumulatorUse::kNone, OperandType::kUImm) \ /* - [Start constant jumps] */ \ V(JumpConstant, AccumulatorUse::kNone, OperandType::kIdx) \ /* - [Conditional jumps] */ \ /* - [Conditional constant jumps] */ \ V(JumpIfNullConstant, AccumulatorUse::kRead, OperandType::kIdx) \ V(JumpIfNotNullConstant, AccumulatorUse::kRead, OperandType::kIdx) \ V(JumpIfUndefinedConstant, AccumulatorUse::kRead, OperandType::kIdx) \ V(JumpIfNotUndefinedConstant, AccumulatorUse::kRead, OperandType::kIdx) \ V(JumpIfTrueConstant, AccumulatorUse::kRead, OperandType::kIdx) \ V(JumpIfFalseConstant, AccumulatorUse::kRead, OperandType::kIdx) \ V(JumpIfJSReceiverConstant, AccumulatorUse::kRead, OperandType::kIdx) \ /* - [Start ToBoolean jumps] */ \ V(JumpIfToBooleanTrueConstant, AccumulatorUse::kRead, OperandType::kIdx) \ V(JumpIfToBooleanFalseConstant, AccumulatorUse::kRead, OperandType::kIdx) \ /* - [End constant jumps] */ \ /* - [Conditional immediate jumps] */ \ V(JumpIfToBooleanTrue, AccumulatorUse::kRead, OperandType::kUImm) \ V(JumpIfToBooleanFalse, AccumulatorUse::kRead, OperandType::kUImm) \ /* - [End ToBoolean jumps] */ \ V(JumpIfTrue, AccumulatorUse::kRead, OperandType::kUImm) \ V(JumpIfFalse, AccumulatorUse::kRead, OperandType::kUImm) \ V(JumpIfNull, AccumulatorUse::kRead, OperandType::kUImm) \ V(JumpIfNotNull, AccumulatorUse::kRead, OperandType::kUImm) \ V(JumpIfUndefined, AccumulatorUse::kRead, OperandType::kUImm) \ V(JumpIfNotUndefined, AccumulatorUse::kRead, OperandType::kUImm) \ V(JumpIfJSReceiver, AccumulatorUse::kRead, OperandType::kUImm) \ \ /* Smi-table lookup for switch statements */ \ V(SwitchOnSmiNoFeedback, AccumulatorUse::kRead, OperandType::kIdx, \ OperandType::kUImm, OperandType::kImm) \ \ /* Complex flow control For..in */ \ V(ForInEnumerate, AccumulatorUse::kWrite, OperandType::kReg) \ V(ForInPrepare, AccumulatorUse::kRead, OperandType::kRegOutTriple, \ OperandType::kIdx) \ V(ForInContinue, AccumulatorUse::kWrite, OperandType::kReg, \ OperandType::kReg) \ V(ForInNext, AccumulatorUse::kWrite, OperandType::kReg, OperandType::kReg, \ OperandType::kRegPair, OperandType::kIdx) \ V(ForInStep, AccumulatorUse::kWrite, OperandType::kReg) \ \ /* Perform a stack guard check */ \ V(StackCheck, AccumulatorUse::kNone) \ \ /* Update the pending message */ \ V(SetPendingMessage, AccumulatorUse::kReadWrite) \ \ /* Non-local flow control */ \ V(Throw, AccumulatorUse::kRead) \ V(ReThrow, AccumulatorUse::kRead) \ V(Return, AccumulatorUse::kRead) \ V(ThrowReferenceErrorIfHole, AccumulatorUse::kRead, OperandType::kIdx) \ V(ThrowSuperNotCalledIfHole, AccumulatorUse::kRead) \ V(ThrowSuperAlreadyCalledIfNotHole, AccumulatorUse::kRead) \ \ /* Generators */ \ V(SwitchOnGeneratorState, AccumulatorUse::kNone, OperandType::kReg, \ OperandType::kIdx, OperandType::kUImm) \ V(SuspendGenerator, AccumulatorUse::kRead, OperandType::kReg, \ OperandType::kRegList, OperandType::kRegCount, OperandType::kUImm) \ V(ResumeGenerator, AccumulatorUse::kWrite, OperandType::kReg, \ OperandType::kRegOutList, OperandType::kRegCount) \ \ /* Debugger */ \ V(Debugger, AccumulatorUse::kNone) \ \ /* Block Coverage */ \ V(IncBlockCounter, AccumulatorUse::kNone, OperandType::kIdx) \ \ /* Execution Abort (internal error) */ \ V(Abort, AccumulatorUse::kNone, OperandType::kIdx) \ \ /* Illegal bytecode */ \ V(Illegal, AccumulatorUse::kNone) // List of debug break bytecodes. #define DEBUG_BREAK_PLAIN_BYTECODE_LIST(V) \ V(DebugBreak0) \ V(DebugBreak1) \ V(DebugBreak2) \ V(DebugBreak3) \ V(DebugBreak4) \ V(DebugBreak5) \ V(DebugBreak6) #define DEBUG_BREAK_PREFIX_BYTECODE_LIST(V) \ V(DebugBreakWide) \ V(DebugBreakExtraWide) #define DEBUG_BREAK_BYTECODE_LIST(V) \ DEBUG_BREAK_PLAIN_BYTECODE_LIST(V) \ DEBUG_BREAK_PREFIX_BYTECODE_LIST(V) // Lists of jump bytecodes. #define JUMP_UNCONDITIONAL_IMMEDIATE_BYTECODE_LIST(V) \ V(JumpLoop) \ V(Jump) #define JUMP_UNCONDITIONAL_CONSTANT_BYTECODE_LIST(V) V(JumpConstant) #define JUMP_TOBOOLEAN_CONDITIONAL_IMMEDIATE_BYTECODE_LIST(V) \ V(JumpIfToBooleanTrue) \ V(JumpIfToBooleanFalse) #define JUMP_TOBOOLEAN_CONDITIONAL_CONSTANT_BYTECODE_LIST(V) \ V(JumpIfToBooleanTrueConstant) \ V(JumpIfToBooleanFalseConstant) #define JUMP_CONDITIONAL_IMMEDIATE_BYTECODE_LIST(V) \ JUMP_TOBOOLEAN_CONDITIONAL_IMMEDIATE_BYTECODE_LIST(V) \ V(JumpIfTrue) \ V(JumpIfFalse) \ V(JumpIfNull) \ V(JumpIfNotNull) \ V(JumpIfUndefined) \ V(JumpIfNotUndefined) \ V(JumpIfJSReceiver) \ #define JUMP_CONDITIONAL_CONSTANT_BYTECODE_LIST(V) \ JUMP_TOBOOLEAN_CONDITIONAL_CONSTANT_BYTECODE_LIST(V) \ V(JumpIfNullConstant) \ V(JumpIfNotNullConstant) \ V(JumpIfUndefinedConstant) \ V(JumpIfNotUndefinedConstant) \ V(JumpIfTrueConstant) \ V(JumpIfFalseConstant) \ V(JumpIfJSReceiverConstant) \ #define JUMP_CONSTANT_BYTECODE_LIST(V) \ JUMP_UNCONDITIONAL_CONSTANT_BYTECODE_LIST(V) \ JUMP_CONDITIONAL_CONSTANT_BYTECODE_LIST(V) #define JUMP_IMMEDIATE_BYTECODE_LIST(V) \ JUMP_UNCONDITIONAL_IMMEDIATE_BYTECODE_LIST(V) \ JUMP_CONDITIONAL_IMMEDIATE_BYTECODE_LIST(V) #define JUMP_TO_BOOLEAN_BYTECODE_LIST(V) \ JUMP_TOBOOLEAN_CONDITIONAL_IMMEDIATE_BYTECODE_LIST(V) \ JUMP_TOBOOLEAN_CONDITIONAL_CONSTANT_BYTECODE_LIST(V) #define JUMP_UNCONDITIONAL_BYTECODE_LIST(V) \ JUMP_UNCONDITIONAL_IMMEDIATE_BYTECODE_LIST(V) \ JUMP_UNCONDITIONAL_CONSTANT_BYTECODE_LIST(V) #define JUMP_CONDITIONAL_BYTECODE_LIST(V) \ JUMP_CONDITIONAL_IMMEDIATE_BYTECODE_LIST(V) \ JUMP_CONDITIONAL_CONSTANT_BYTECODE_LIST(V) #define JUMP_FORWARD_BYTECODE_LIST(V) \ V(Jump) \ V(JumpConstant) \ JUMP_CONDITIONAL_BYTECODE_LIST(V) #define JUMP_BYTECODE_LIST(V) \ JUMP_FORWARD_BYTECODE_LIST(V) \ V(JumpLoop) #define RETURN_BYTECODE_LIST(V) \ V(Return) \ V(SuspendGenerator) // Enumeration of interpreter bytecodes. enum class Bytecode : uint8_t { #define DECLARE_BYTECODE(Name, ...) k##Name, BYTECODE_LIST(DECLARE_BYTECODE) #undef DECLARE_BYTECODE #define COUNT_BYTECODE(x, ...) +1 // The COUNT_BYTECODE macro will turn this into kLast = -1 +1 +1... which will // evaluate to the same value as the last real bytecode. kLast = -1 BYTECODE_LIST(COUNT_BYTECODE) #undef COUNT_BYTECODE }; class V8_EXPORT_PRIVATE Bytecodes final : public AllStatic { public: // The maximum number of operands a bytecode may have. static const int kMaxOperands = 5; // The total number of bytecodes used. static const int kBytecodeCount = static_cast<int>(Bytecode::kLast) + 1; // Returns string representation of |bytecode|. static const char* ToString(Bytecode bytecode); // Returns string representation of |bytecode|. static std::string ToString(Bytecode bytecode, OperandScale operand_scale); // Returns byte value of bytecode. static uint8_t ToByte(Bytecode bytecode) { DCHECK_LE(bytecode, Bytecode::kLast); return static_cast<uint8_t>(bytecode); } // Returns bytecode for |value|. static Bytecode FromByte(uint8_t value) { Bytecode bytecode = static_cast<Bytecode>(value); DCHECK_LE(bytecode, Bytecode::kLast); return bytecode; } // Returns the prefix bytecode representing an operand scale to be // applied to a a bytecode. static Bytecode OperandScaleToPrefixBytecode(OperandScale operand_scale) { switch (operand_scale) { case OperandScale::kQuadruple: return Bytecode::kExtraWide; case OperandScale::kDouble: return Bytecode::kWide; default: UNREACHABLE(); } } // Returns true if the operand scale requires a prefix bytecode. static bool OperandScaleRequiresPrefixBytecode(OperandScale operand_scale) { return operand_scale != OperandScale::kSingle; } // Returns the scaling applied to scalable operands if bytecode is // is a scaling prefix. static OperandScale PrefixBytecodeToOperandScale(Bytecode bytecode) { switch (bytecode) { case Bytecode::kExtraWide: case Bytecode::kDebugBreakExtraWide: return OperandScale::kQuadruple; case Bytecode::kWide: case Bytecode::kDebugBreakWide: return OperandScale::kDouble; default: UNREACHABLE(); } } // Returns how accumulator is used by |bytecode|. static AccumulatorUse GetAccumulatorUse(Bytecode bytecode) { DCHECK_LE(bytecode, Bytecode::kLast); return kAccumulatorUse[static_cast<size_t>(bytecode)]; } // Returns true if |bytecode| reads the accumulator. static bool ReadsAccumulator(Bytecode bytecode) { return BytecodeOperands::ReadsAccumulator(GetAccumulatorUse(bytecode)); } // Returns true if |bytecode| writes the accumulator. static bool WritesAccumulator(Bytecode bytecode) { return BytecodeOperands::WritesAccumulator(GetAccumulatorUse(bytecode)); } // Return true if |bytecode| is an accumulator load without effects, // e.g. LdaConstant, LdaTrue, Ldar. static constexpr bool IsAccumulatorLoadWithoutEffects(Bytecode bytecode) { return bytecode == Bytecode::kLdar || bytecode == Bytecode::kLdaZero || bytecode == Bytecode::kLdaSmi || bytecode == Bytecode::kLdaNull || bytecode == Bytecode::kLdaTrue || bytecode == Bytecode::kLdaFalse || bytecode == Bytecode::kLdaUndefined || bytecode == Bytecode::kLdaTheHole || bytecode == Bytecode::kLdaConstant || bytecode == Bytecode::kLdaContextSlot || bytecode == Bytecode::kLdaCurrentContextSlot || bytecode == Bytecode::kLdaImmutableContextSlot || bytecode == Bytecode::kLdaImmutableCurrentContextSlot; } // Returns true if |bytecode| is a compare operation without external effects // (e.g., Type cooersion). static constexpr bool IsCompareWithoutEffects(Bytecode bytecode) { return bytecode == Bytecode::kTestUndetectable || bytecode == Bytecode::kTestNull || bytecode == Bytecode::kTestUndefined || bytecode == Bytecode::kTestTypeOf; } // Return true if |bytecode| is a register load without effects, // e.g. Mov, Star. static constexpr bool IsRegisterLoadWithoutEffects(Bytecode bytecode) { return bytecode == Bytecode::kMov || bytecode == Bytecode::kPopContext || bytecode == Bytecode::kPushContext || bytecode == Bytecode::kStar; } // Returns true if the bytecode is a conditional jump taking // an immediate byte operand (OperandType::kImm). static constexpr bool IsConditionalJumpImmediate(Bytecode bytecode) { return bytecode >= Bytecode::kJumpIfToBooleanTrue && bytecode <= Bytecode::kJumpIfJSReceiver; } // Returns true if the bytecode is a conditional jump taking // a constant pool entry (OperandType::kIdx). static constexpr bool IsConditionalJumpConstant(Bytecode bytecode) { return bytecode >= Bytecode::kJumpIfNullConstant && bytecode <= Bytecode::kJumpIfToBooleanFalseConstant; } // Returns true if the bytecode is a conditional jump taking // any kind of operand. static constexpr bool IsConditionalJump(Bytecode bytecode) { return bytecode >= Bytecode::kJumpIfNullConstant && bytecode <= Bytecode::kJumpIfJSReceiver; } // Returns true if the bytecode is an unconditional jump. static constexpr bool IsUnconditionalJump(Bytecode bytecode) { return bytecode >= Bytecode::kJumpLoop && bytecode <= Bytecode::kJumpConstant; } // Returns true if the bytecode is a jump or a conditional jump taking // an immediate byte operand (OperandType::kImm). static constexpr bool IsJumpImmediate(Bytecode bytecode) { return bytecode == Bytecode::kJump || bytecode == Bytecode::kJumpLoop || IsConditionalJumpImmediate(bytecode); } // Returns true if the bytecode is a jump or conditional jump taking a // constant pool entry (OperandType::kIdx). static constexpr bool IsJumpConstant(Bytecode bytecode) { return bytecode >= Bytecode::kJumpConstant && bytecode <= Bytecode::kJumpIfToBooleanFalseConstant; } // Returns true if the bytecode is a jump that internally coerces the // accumulator to a boolean. static constexpr bool IsJumpIfToBoolean(Bytecode bytecode) { return bytecode >= Bytecode::kJumpIfToBooleanTrueConstant && bytecode <= Bytecode::kJumpIfToBooleanFalse; } // Returns true if the bytecode is a jump or conditional jump taking // any kind of operand. static constexpr bool IsJump(Bytecode bytecode) { return bytecode >= Bytecode::kJumpLoop && bytecode <= Bytecode::kJumpIfJSReceiver; } // Returns true if the bytecode is a forward jump or conditional jump taking // any kind of operand. static constexpr bool IsForwardJump(Bytecode bytecode) { return bytecode >= Bytecode::kJump && bytecode <= Bytecode::kJumpIfJSReceiver; } // Return true if |bytecode| is a jump without effects, // e.g. any jump excluding those that include type coercion like // JumpIfTrueToBoolean. static constexpr bool IsJumpWithoutEffects(Bytecode bytecode) { return IsJump(bytecode) && !IsJumpIfToBoolean(bytecode); } // Returns true if the bytecode is a switch. static constexpr bool IsSwitch(Bytecode bytecode) { return bytecode == Bytecode::kSwitchOnSmiNoFeedback || bytecode == Bytecode::kSwitchOnGeneratorState; } // Returns true if |bytecode| has no effects. These bytecodes only manipulate // interpreter frame state and will never throw. static constexpr bool IsWithoutExternalSideEffects(Bytecode bytecode) { return (IsAccumulatorLoadWithoutEffects(bytecode) || IsRegisterLoadWithoutEffects(bytecode) || IsCompareWithoutEffects(bytecode) || IsJumpWithoutEffects(bytecode) || IsSwitch(bytecode)); } // Returns true if the bytecode is Ldar or Star. static constexpr bool IsLdarOrStar(Bytecode bytecode) { return bytecode == Bytecode::kLdar || bytecode == Bytecode::kStar; } // Returns true if the bytecode is a call or a constructor call. static constexpr bool IsCallOrConstruct(Bytecode bytecode) { return bytecode == Bytecode::kCallAnyReceiver || bytecode == Bytecode::kCallProperty || bytecode == Bytecode::kCallProperty0 || bytecode == Bytecode::kCallProperty1 || bytecode == Bytecode::kCallProperty2 || bytecode == Bytecode::kCallUndefinedReceiver || bytecode == Bytecode::kCallUndefinedReceiver0 || bytecode == Bytecode::kCallUndefinedReceiver1 || bytecode == Bytecode::kCallUndefinedReceiver2 || bytecode == Bytecode::kConstruct || bytecode == Bytecode::kCallWithSpread || bytecode == Bytecode::kConstructWithSpread || bytecode == Bytecode::kCallJSRuntime; } // Returns true if the bytecode is a call to the runtime. static constexpr bool IsCallRuntime(Bytecode bytecode) { return bytecode == Bytecode::kCallRuntime || bytecode == Bytecode::kCallRuntimeForPair || bytecode == Bytecode::kInvokeIntrinsic; } // Returns true if the bytecode is a scaling prefix bytecode. static constexpr bool IsPrefixScalingBytecode(Bytecode bytecode) { return bytecode == Bytecode::kExtraWide || bytecode == Bytecode::kWide || bytecode == Bytecode::kDebugBreakExtraWide || bytecode == Bytecode::kDebugBreakWide; } // Returns true if the bytecode can be lazily deserialized. static constexpr bool IsLazy(Bytecode bytecode) { // Currently, all handlers are deserialized lazily. return true; } // Returns true if the bytecode returns. static constexpr bool Returns(Bytecode bytecode) { #define OR_BYTECODE(NAME) || bytecode == Bytecode::k##NAME return false RETURN_BYTECODE_LIST(OR_BYTECODE); #undef OR_BYTECODE } // Returns the number of operands expected by |bytecode|. static int NumberOfOperands(Bytecode bytecode) { DCHECK_LE(bytecode, Bytecode::kLast); return kOperandCount[static_cast<size_t>(bytecode)]; } // Returns the i-th operand of |bytecode|. static OperandType GetOperandType(Bytecode bytecode, int i) { DCHECK_LE(bytecode, Bytecode::kLast); DCHECK_LT(i, NumberOfOperands(bytecode)); DCHECK_GE(i, 0); return GetOperandTypes(bytecode)[i]; } // Returns a pointer to an array of operand types terminated in // OperandType::kNone. static const OperandType* GetOperandTypes(Bytecode bytecode) { DCHECK_LE(bytecode, Bytecode::kLast); return kOperandTypes[static_cast<size_t>(bytecode)]; } static bool OperandIsScalableSignedByte(Bytecode bytecode, int operand_index) { DCHECK_LE(bytecode, Bytecode::kLast); return kOperandTypeInfos[static_cast<size_t>(bytecode)][operand_index] == OperandTypeInfo::kScalableSignedByte; } static bool OperandIsScalableUnsignedByte(Bytecode bytecode, int operand_index) { DCHECK_LE(bytecode, Bytecode::kLast); return kOperandTypeInfos[static_cast<size_t>(bytecode)][operand_index] == OperandTypeInfo::kScalableUnsignedByte; } static bool OperandIsScalable(Bytecode bytecode, int operand_index) { return OperandIsScalableSignedByte(bytecode, operand_index) || OperandIsScalableUnsignedByte(bytecode, operand_index); } // Returns true if the bytecode has wider operand forms. static bool IsBytecodeWithScalableOperands(Bytecode bytecode); // Returns the size of the i-th operand of |bytecode|. static OperandSize GetOperandSize(Bytecode bytecode, int i, OperandScale operand_scale) { CHECK_LT(i, NumberOfOperands(bytecode)); return GetOperandSizes(bytecode, operand_scale)[i]; } // Returns the operand sizes of |bytecode| with scale |operand_scale|. static const OperandSize* GetOperandSizes(Bytecode bytecode, OperandScale operand_scale) { DCHECK_LE(bytecode, Bytecode::kLast); DCHECK_GE(operand_scale, OperandScale::kSingle); DCHECK_LE(operand_scale, OperandScale::kLast); STATIC_ASSERT(static_cast<int>(OperandScale::kQuadruple) == 4 && OperandScale::kLast == OperandScale::kQuadruple); int scale_index = static_cast<int>(operand_scale) >> 1; return kOperandSizes[scale_index][static_cast<size_t>(bytecode)]; } // Returns the offset of the i-th operand of |bytecode| relative to the start // of the bytecode. static int GetOperandOffset(Bytecode bytecode, int i, OperandScale operand_scale); // Returns the size of the bytecode including its operands for the // given |operand_scale|. static int Size(Bytecode bytecode, OperandScale operand_scale) { DCHECK_LE(bytecode, Bytecode::kLast); STATIC_ASSERT(static_cast<int>(OperandScale::kQuadruple) == 4 && OperandScale::kLast == OperandScale::kQuadruple); int scale_index = static_cast<int>(operand_scale) >> 1; return kBytecodeSizes[scale_index][static_cast<size_t>(bytecode)]; } // Returns a debug break bytecode to replace |bytecode|. static Bytecode GetDebugBreak(Bytecode bytecode); // Returns the equivalent jump bytecode without the accumulator coercion. static Bytecode GetJumpWithoutToBoolean(Bytecode bytecode); // Returns true if there is a call in the most-frequently executed path // through the bytecode's handler. static bool MakesCallAlongCriticalPath(Bytecode bytecode); // Returns the receiver mode of the given call bytecode. static ConvertReceiverMode GetReceiverMode(Bytecode bytecode) { DCHECK(IsCallOrConstruct(bytecode) || bytecode == Bytecode::kInvokeIntrinsic); switch (bytecode) { case Bytecode::kCallProperty: case Bytecode::kCallProperty0: case Bytecode::kCallProperty1: case Bytecode::kCallProperty2: return ConvertReceiverMode::kNotNullOrUndefined; case Bytecode::kCallUndefinedReceiver: case Bytecode::kCallUndefinedReceiver0: case Bytecode::kCallUndefinedReceiver1: case Bytecode::kCallUndefinedReceiver2: case Bytecode::kCallJSRuntime: return ConvertReceiverMode::kNullOrUndefined; case Bytecode::kCallAnyReceiver: case Bytecode::kConstruct: case Bytecode::kCallWithSpread: case Bytecode::kConstructWithSpread: case Bytecode::kInvokeIntrinsic: return ConvertReceiverMode::kAny; default: UNREACHABLE(); } } // Returns true if the bytecode is a debug break. static bool IsDebugBreak(Bytecode bytecode); // Returns true if |operand_type| is any type of register operand. static bool IsRegisterOperandType(OperandType operand_type); // Returns true if |operand_type| represents a register used as an input. static bool IsRegisterInputOperandType(OperandType operand_type); // Returns true if |operand_type| represents a register used as an output. static bool IsRegisterOutputOperandType(OperandType operand_type); // Returns true if |operand_type| represents a register list operand. static bool IsRegisterListOperandType(OperandType operand_type); // Returns true if the handler for |bytecode| should look ahead and inline a // dispatch to a Star bytecode. static bool IsStarLookahead(Bytecode bytecode, OperandScale operand_scale); // Returns the number of registers represented by a register operand. For // instance, a RegPair represents two registers. Should not be called for // kRegList which has a variable number of registers based on the following // kRegCount operand. static int GetNumberOfRegistersRepresentedBy(OperandType operand_type) { switch (operand_type) { case OperandType::kReg: case OperandType::kRegOut: return 1; case OperandType::kRegPair: case OperandType::kRegOutPair: return 2; case OperandType::kRegOutTriple: return 3; case OperandType::kRegList: case OperandType::kRegOutList: UNREACHABLE(); default: return 0; } UNREACHABLE(); } // Returns the size of |operand_type| for |operand_scale|. static OperandSize SizeOfOperand(OperandType operand_type, OperandScale operand_scale) { DCHECK_LE(operand_type, OperandType::kLast); DCHECK_GE(operand_scale, OperandScale::kSingle); DCHECK_LE(operand_scale, OperandScale::kLast); STATIC_ASSERT(static_cast<int>(OperandScale::kQuadruple) == 4 && OperandScale::kLast == OperandScale::kQuadruple); int scale_index = static_cast<int>(operand_scale) >> 1; return kOperandKindSizes[scale_index][static_cast<size_t>(operand_type)]; } // Returns true if |operand_type| is a runtime-id operand (kRuntimeId). static bool IsRuntimeIdOperandType(OperandType operand_type); // Returns true if |operand_type| is unsigned, false if signed. static bool IsUnsignedOperandType(OperandType operand_type); // Returns true if a handler is generated for a bytecode at a given // operand scale. All bytecodes have handlers at OperandScale::kSingle, // but only bytecodes with scalable operands have handlers with larger // OperandScale values. static bool BytecodeHasHandler(Bytecode bytecode, OperandScale operand_scale); // Return the operand scale required to hold a signed operand with |value|. static OperandScale ScaleForSignedOperand(int32_t value) { if (value >= kMinInt8 && value <= kMaxInt8) { return OperandScale::kSingle; } else if (value >= kMinInt16 && value <= kMaxInt16) { return OperandScale::kDouble; } else { return OperandScale::kQuadruple; } } // Return the operand scale required to hold an unsigned operand with |value|. static OperandScale ScaleForUnsignedOperand(uint32_t value) { if (value <= kMaxUInt8) { return OperandScale::kSingle; } else if (value <= kMaxUInt16) { return OperandScale::kDouble; } else { return OperandScale::kQuadruple; } } // Return the operand size required to hold an unsigned operand with |value|. static OperandSize SizeForUnsignedOperand(uint32_t value) { if (value <= kMaxUInt8) { return OperandSize::kByte; } else if (value <= kMaxUInt16) { return OperandSize::kShort; } else { return OperandSize::kQuad; } } static Address bytecode_size_table_address() { return reinterpret_cast<Address>(const_cast<int*>(&kBytecodeSizes[0][0])); } private: static const OperandType* const kOperandTypes[]; static const OperandTypeInfo* const kOperandTypeInfos[]; static const int kOperandCount[]; static const int kNumberOfRegisterOperands[]; static const AccumulatorUse kAccumulatorUse[]; static const bool kIsScalable[]; static const int kBytecodeSizes[3][kBytecodeCount]; static const OperandSize* const kOperandSizes[3][kBytecodeCount]; static OperandSize const kOperandKindSizes[3][BytecodeOperands::kOperandTypeCount]; }; V8_EXPORT_PRIVATE std::ostream& operator<<(std::ostream& os, const Bytecode& bytecode); } // namespace interpreter } // namespace internal } // namespace v8 #endif // V8_INTERPRETER_BYTECODES_H_