/* * Copyright (c) 2021, Andreas Kling * Copyright (c) 2021, Linus Groh * Copyright (c) 2021, Gunnar Beutner * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include #include #include namespace JS::Bytecode::Op { class Load final : public Instruction { public: explicit Load(Register src) : Instruction(Type::Load) , m_src(src) { } void execute(Bytecode::Interpreter&) const; String to_string(Bytecode::Executable const&) const; private: Register m_src; }; class LoadImmediate final : public Instruction { public: explicit LoadImmediate(Value value) : Instruction(Type::LoadImmediate) , m_value(value) { } void execute(Bytecode::Interpreter&) const; String to_string(Bytecode::Executable const&) const; private: Value m_value; }; class Store final : public Instruction { public: explicit Store(Register dst) : Instruction(Type::Store) , m_dst(dst) { } void execute(Bytecode::Interpreter&) const; String to_string(Bytecode::Executable const&) const; private: Register m_dst; }; #define JS_ENUMERATE_COMMON_BINARY_OPS(O) \ O(Add, add) \ O(Sub, sub) \ O(Mul, mul) \ O(Div, div) \ O(Exp, exp) \ O(Mod, mod) \ O(In, in) \ O(InstanceOf, instance_of) \ O(GreaterThan, greater_than) \ O(GreaterThanEquals, greater_than_equals) \ O(LessThan, less_than) \ O(LessThanEquals, less_than_equals) \ O(AbstractInequals, abstract_inequals) \ O(AbstractEquals, abstract_equals) \ O(TypedInequals, typed_inequals) \ O(TypedEquals, typed_equals) \ O(BitwiseAnd, bitwise_and) \ O(BitwiseOr, bitwise_or) \ O(BitwiseXor, bitwise_xor) \ O(LeftShift, left_shift) \ O(RightShift, right_shift) \ O(UnsignedRightShift, unsigned_right_shift) #define JS_DECLARE_COMMON_BINARY_OP(OpTitleCase, op_snake_case) \ class OpTitleCase final : public Instruction { \ public: \ explicit OpTitleCase(Register lhs_reg) \ : Instruction(Type::OpTitleCase) \ , m_lhs_reg(lhs_reg) \ { \ } \ \ void execute(Bytecode::Interpreter&) const; \ String to_string(Bytecode::Executable const&) const; \ \ private: \ Register m_lhs_reg; \ }; JS_ENUMERATE_COMMON_BINARY_OPS(JS_DECLARE_COMMON_BINARY_OP) #undef JS_DECLARE_COMMON_BINARY_OP #define JS_ENUMERATE_COMMON_UNARY_OPS(O) \ O(BitwiseNot, bitwise_not) \ O(Not, not_) \ O(UnaryPlus, unary_plus) \ O(UnaryMinus, unary_minus) \ O(Typeof, typeof_) #define JS_DECLARE_COMMON_UNARY_OP(OpTitleCase, op_snake_case) \ class OpTitleCase final : public Instruction { \ public: \ OpTitleCase() \ : Instruction(Type::OpTitleCase) \ { \ } \ \ void execute(Bytecode::Interpreter&) const; \ String to_string(Bytecode::Executable const&) const; \ }; JS_ENUMERATE_COMMON_UNARY_OPS(JS_DECLARE_COMMON_UNARY_OP) #undef JS_DECLARE_COMMON_UNARY_OP class NewString final : public Instruction { public: explicit NewString(StringTableIndex string) : Instruction(Type::NewString) , m_string(string) { } void execute(Bytecode::Interpreter&) const; String to_string(Bytecode::Executable const&) const; private: StringTableIndex m_string; }; class NewObject final : public Instruction { public: NewObject() : Instruction(Type::NewObject) { } void execute(Bytecode::Interpreter&) const; String to_string(Bytecode::Executable const&) const; }; class NewBigInt final : public Instruction { public: explicit NewBigInt(Crypto::SignedBigInteger bigint) : Instruction(Type::NewBigInt) , m_bigint(move(bigint)) { } void execute(Bytecode::Interpreter&) const; String to_string(Bytecode::Executable const&) const; private: Crypto::SignedBigInteger m_bigint; }; // NOTE: This instruction is variable-width depending on the number of elements! class NewArray final : public Instruction { public: explicit NewArray(Vector const& elements) : Instruction(Type::NewArray) , m_element_count(elements.size()) { for (size_t i = 0; i < m_element_count; ++i) m_elements[i] = elements[i]; } void execute(Bytecode::Interpreter&) const; String to_string(Bytecode::Executable const&) const; size_t length() const { return sizeof(*this) + sizeof(Register) * m_element_count; } private: size_t m_element_count { 0 }; Register m_elements[]; }; class ConcatString final : public Instruction { public: explicit ConcatString(Register lhs) : Instruction(Type::ConcatString) , m_lhs(lhs) { } void execute(Bytecode::Interpreter&) const; String to_string(Bytecode::Executable const&) const; private: Register m_lhs; }; class SetVariable final : public Instruction { public: explicit SetVariable(StringTableIndex identifier) : Instruction(Type::SetVariable) , m_identifier(identifier) { } void execute(Bytecode::Interpreter&) const; String to_string(Bytecode::Executable const&) const; private: StringTableIndex m_identifier; }; class GetVariable final : public Instruction { public: explicit GetVariable(StringTableIndex identifier) : Instruction(Type::GetVariable) , m_identifier(identifier) { } void execute(Bytecode::Interpreter&) const; String to_string(Bytecode::Executable const&) const; private: StringTableIndex m_identifier; }; class GetById final : public Instruction { public: explicit GetById(StringTableIndex property) : Instruction(Type::GetById) , m_property(property) { } void execute(Bytecode::Interpreter&) const; String to_string(Bytecode::Executable const&) const; private: StringTableIndex m_property; }; class PutById final : public Instruction { public: explicit PutById(Register base, StringTableIndex property) : Instruction(Type::PutById) , m_base(base) , m_property(property) { } void execute(Bytecode::Interpreter&) const; String to_string(Bytecode::Executable const&) const; private: Register m_base; StringTableIndex m_property; }; class GetByValue final : public Instruction { public: explicit GetByValue(Register base) : Instruction(Type::GetByValue) , m_base(base) { } void execute(Bytecode::Interpreter&) const; String to_string(Bytecode::Executable const&) const; private: Register m_base; }; class PutByValue final : public Instruction { public: PutByValue(Register base, Register property) : Instruction(Type::PutByValue) , m_base(base) , m_property(property) { } void execute(Bytecode::Interpreter&) const; String to_string(Bytecode::Executable const&) const; private: Register m_base; Register m_property; }; class Jump : public Instruction { public: constexpr static bool IsTerminator = true; explicit Jump(Type type, Optional