123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414 |
- /*
- * Copyright (c) 2021, Andreas Kling <kling@serenityos.org>
- * Copyright (c) 2021, Linus Groh <linusg@serenityos.org>
- *
- * SPDX-License-Identifier: BSD-2-Clause
- */
- #pragma once
- #include <AK/FlyString.h>
- #include <LibCrypto/BigInt/SignedBigInteger.h>
- #include <LibJS/Bytecode/Instruction.h>
- #include <LibJS/Bytecode/Label.h>
- #include <LibJS/Bytecode/Register.h>
- #include <LibJS/Heap/Cell.h>
- #include <LibJS/Runtime/Value.h>
- namespace JS::Bytecode::Op {
- class Load final : public Instruction {
- public:
- Load(Register src)
- : Instruction(Type::Load)
- , m_src(src)
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- Register m_src;
- };
- class LoadImmediate final : public Instruction {
- public:
- LoadImmediate(Value value)
- : Instruction(Type::LoadImmediate)
- , m_value(value)
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- Value m_value;
- };
- class Store final : public Instruction {
- public:
- Store(Register dst)
- : Instruction(Type::Store)
- , m_dst(dst)
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() 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: \
- OpTitleCase(Register lhs_reg) \
- : Instruction(Type::OpTitleCase) \
- , m_lhs_reg(lhs_reg) \
- { \
- } \
- \
- void execute(Bytecode::Interpreter&) const; \
- String to_string() 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() const; \
- };
- JS_ENUMERATE_COMMON_UNARY_OPS(JS_DECLARE_COMMON_UNARY_OP)
- #undef JS_DECLARE_COMMON_UNARY_OP
- class NewString final : public Instruction {
- public:
- NewString(String string)
- : Instruction(Type::NewString)
- , m_string(move(string))
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- String m_string;
- };
- class NewObject final : public Instruction {
- public:
- NewObject()
- : Instruction(Type::NewObject)
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() 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() const;
- private:
- Crypto::SignedBigInteger m_bigint;
- };
- // NOTE: This instruction is variable-width depending on the number of elements!
- class NewArray final : public Instruction {
- public:
- NewArray(Vector<Register> 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() 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:
- ConcatString(Register lhs)
- : Instruction(Type::ConcatString)
- , m_lhs(lhs)
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- Register m_lhs;
- };
- class SetVariable final : public Instruction {
- public:
- SetVariable(FlyString identifier)
- : Instruction(Type::SetVariable)
- , m_identifier(move(identifier))
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- FlyString m_identifier;
- };
- class GetVariable final : public Instruction {
- public:
- GetVariable(FlyString identifier)
- : Instruction(Type::GetVariable)
- , m_identifier(move(identifier))
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- FlyString m_identifier;
- };
- class GetById final : public Instruction {
- public:
- GetById(FlyString property)
- : Instruction(Type::GetById)
- , m_property(move(property))
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- FlyString m_property;
- };
- class PutById final : public Instruction {
- public:
- PutById(Register base, FlyString property)
- : Instruction(Type::PutById)
- , m_base(base)
- , m_property(move(property))
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- Register m_base;
- FlyString m_property;
- };
- class Jump : public Instruction {
- public:
- constexpr static bool IsTerminator = true;
- explicit Jump(Type type, Optional<Label> taken_target = {}, Optional<Label> nontaken_target = {})
- : Instruction(type)
- , m_true_target(move(taken_target))
- , m_false_target(move(nontaken_target))
- {
- }
- explicit Jump(Optional<Label> taken_target = {}, Optional<Label> nontaken_target = {})
- : Instruction(Type::Jump)
- , m_true_target(move(taken_target))
- , m_false_target(move(nontaken_target))
- {
- }
- void set_targets(Optional<Label> true_target, Optional<Label> false_target)
- {
- m_true_target = move(true_target);
- m_false_target = move(false_target);
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- protected:
- Optional<Label> m_true_target;
- Optional<Label> m_false_target;
- };
- class JumpConditional final : public Jump {
- public:
- explicit JumpConditional(Optional<Label> true_target = {}, Optional<Label> false_target = {})
- : Jump(Type::JumpConditional, move(true_target), move(false_target))
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- };
- class JumpNullish final : public Jump {
- public:
- explicit JumpNullish(Optional<Label> true_target = {}, Optional<Label> false_target = {})
- : Jump(Type::JumpNullish, move(true_target), move(false_target))
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- };
- // NOTE: This instruction is variable-width depending on the number of arguments!
- class Call final : public Instruction {
- public:
- Call(Register callee, Register this_value, Vector<Register> const& arguments)
- : Instruction(Type::Call)
- , m_callee(callee)
- , m_this_value(this_value)
- , m_argument_count(arguments.size())
- {
- for (size_t i = 0; i < m_argument_count; ++i)
- m_arguments[i] = arguments[i];
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- size_t length() const { return sizeof(*this) + sizeof(Register) * m_argument_count; }
- private:
- Register m_callee;
- Register m_this_value;
- size_t m_argument_count { 0 };
- Register m_arguments[];
- };
- class EnterScope final : public Instruction {
- public:
- explicit EnterScope(ScopeNode const& scope_node)
- : Instruction(Type::EnterScope)
- , m_scope_node(scope_node)
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- ScopeNode const& m_scope_node;
- };
- class Return final : public Instruction {
- public:
- constexpr static bool IsTerminator = true;
- Return()
- : Instruction(Type::Return)
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- };
- }
- namespace JS::Bytecode {
- ALWAYS_INLINE void Instruction::execute(Bytecode::Interpreter& interpreter) const
- {
- #define __BYTECODE_OP(op) \
- case Instruction::Type::op: \
- return static_cast<Bytecode::Op::op const&>(*this).execute(interpreter);
- switch (type()) {
- ENUMERATE_BYTECODE_OPS(__BYTECODE_OP)
- default:
- VERIFY_NOT_REACHED();
- }
- #undef __BYTECODE_OP
- }
- ALWAYS_INLINE size_t Instruction::length() const
- {
- if (type() == Type::Call)
- return static_cast<Op::Call const&>(*this).length();
- else if (type() == Type::NewArray)
- return static_cast<Op::NewArray const&>(*this).length();
- #define __BYTECODE_OP(op) \
- case Type::op: \
- return sizeof(Op::op);
- switch (type()) {
- ENUMERATE_BYTECODE_OPS(__BYTECODE_OP)
- default:
- VERIFY_NOT_REACHED();
- }
- #undef __BYTECODE_OP
- }
- }
|