123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375 |
- /*
- * 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 dst, Value value)
- : Instruction(Type::Load)
- , m_dst(dst)
- , m_value(value)
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- Register m_dst;
- Value m_value;
- };
- class LoadRegister final : public Instruction {
- public:
- LoadRegister(Register dst, Register src)
- : Instruction(Type::LoadRegister)
- , m_dst(dst)
- , m_src(src)
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- Register m_dst;
- Register m_src;
- };
- #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 dst, Register src1, Register src2) \
- : Instruction(Type::OpTitleCase) \
- , m_dst(dst) \
- , m_src1(src1) \
- , m_src2(src2) \
- { \
- } \
- \
- void execute(Bytecode::Interpreter&) const; \
- String to_string() const; \
- \
- private: \
- Register m_dst; \
- Register m_src1; \
- Register m_src2; \
- };
- 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(Register dst, Register src) \
- : Instruction(Type::OpTitleCase) \
- , m_dst(dst) \
- , m_src(src) \
- { \
- } \
- \
- void execute(Bytecode::Interpreter&) const; \
- String to_string() const; \
- \
- private: \
- Register m_dst; \
- Register m_src; \
- };
- JS_ENUMERATE_COMMON_UNARY_OPS(JS_DECLARE_COMMON_UNARY_OP)
- #undef JS_DECLARE_COMMON_UNARY_OP
- class NewString final : public Instruction {
- public:
- NewString(Register dst, String string)
- : Instruction(Type::NewString)
- , m_dst(dst)
- , m_string(move(string))
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- Register m_dst;
- String m_string;
- };
- class NewObject final : public Instruction {
- public:
- explicit NewObject(Register dst)
- : Instruction(Type::NewObject)
- , m_dst(dst)
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- Register m_dst;
- };
- class NewBigInt final : public Instruction {
- public:
- explicit NewBigInt(Register dst, Crypto::SignedBigInteger bigint)
- : Instruction(Type::NewBigInt)
- , m_dst(dst)
- , m_bigint(move(bigint))
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- Register m_dst;
- Crypto::SignedBigInteger m_bigint;
- };
- class SetVariable final : public Instruction {
- public:
- SetVariable(FlyString identifier, Register src)
- : Instruction(Type::SetVariable)
- , m_identifier(move(identifier))
- , m_src(src)
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- FlyString m_identifier;
- Register m_src;
- };
- class GetVariable final : public Instruction {
- public:
- GetVariable(Register dst, FlyString identifier)
- : Instruction(Type::GetVariable)
- , m_dst(dst)
- , m_identifier(move(identifier))
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- Register m_dst;
- FlyString m_identifier;
- };
- class GetById final : public Instruction {
- public:
- GetById(Register dst, Register base, FlyString property)
- : Instruction(Type::GetById)
- , m_dst(dst)
- , m_base(base)
- , m_property(move(property))
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- Register m_dst;
- Register m_base;
- FlyString m_property;
- };
- class PutById final : public Instruction {
- public:
- PutById(Register base, FlyString property, Register src)
- : Instruction(Type::PutById)
- , m_base(base)
- , m_property(move(property))
- , m_src(src)
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- Register m_base;
- FlyString m_property;
- Register m_src;
- };
- class Jump : public Instruction {
- public:
- explicit Jump(Type type, Optional<Label> target = {})
- : Instruction(type)
- , m_target(move(target))
- {
- }
- explicit Jump(Optional<Label> target = {})
- : Instruction(Type::Jump)
- , m_target(move(target))
- {
- }
- void set_target(Optional<Label> target) { m_target = move(target); }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- protected:
- Optional<Label> m_target;
- };
- class JumpIfFalse final : public Jump {
- public:
- explicit JumpIfFalse(Register result, Optional<Label> target = {})
- : Jump(Type::JumpIfFalse, move(target))
- , m_result(result)
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- Register m_result;
- };
- class JumpIfTrue : public Jump {
- public:
- explicit JumpIfTrue(Register result, Optional<Label> target = {})
- : Jump(Type::JumpIfTrue, move(target))
- , m_result(result)
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- Register m_result;
- };
- class JumpIfNullish final : public Jump {
- public:
- explicit JumpIfNullish(Register result, Optional<Label> target = {})
- : Jump(Type::JumpIfNullish, move(target))
- , m_result(result)
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- Register m_result;
- };
- // NOTE: This instruction is variable-width depending on the number of arguments!
- class Call final : public Instruction {
- public:
- Call(Register dst, Register callee, Register this_value, Vector<Register> const& arguments)
- : Instruction(Type::Call)
- , m_dst(dst)
- , 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_dst;
- 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:
- explicit Return(Optional<Register> argument)
- : Instruction(Type::Return)
- , m_argument(move(argument))
- {
- }
- void execute(Bytecode::Interpreter&) const;
- String to_string() const;
- private:
- Optional<Register> m_argument;
- };
- }
|