/* * Copyright (c) 2021, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include namespace JS::Bytecode { void Instruction::destroy(Instruction& instruction) { #define __BYTECODE_OP(op) \ case Type::op: \ static_cast(instruction).~op(); \ return; switch (instruction.type()) { ENUMERATE_BYTECODE_OPS(__BYTECODE_OP) default: VERIFY_NOT_REACHED(); } #undef __BYTECODE_OP } void Instruction::visit_labels(Function visitor) { #define __BYTECODE_OP(op) \ case Type::op: \ static_cast(*this).visit_labels_impl(move(visitor)); \ return; switch (type()) { ENUMERATE_BYTECODE_OPS(__BYTECODE_OP) default: VERIFY_NOT_REACHED(); } #undef __BYTECODE_OP } void Instruction::visit_operands(Function visitor) { #define __BYTECODE_OP(op) \ case Type::op: \ static_cast(*this).visit_operands_impl(move(visitor)); \ return; switch (type()) { ENUMERATE_BYTECODE_OPS(__BYTECODE_OP) default: VERIFY_NOT_REACHED(); } #undef __BYTECODE_OP } template concept HasVariableLength = Op::IsVariableLength; template concept HasFixedLength = !Op::IsVariableLength; template size_t get_length_impl(Op const& op) { return op.length_impl(); } // Function template for types without a length_impl method template size_t get_length_impl(Op const&) { return sizeof(Op); } size_t Instruction::length() const { #define __BYTECODE_OP(op) \ case Type::op: { \ auto& typed_op = static_cast(*this); \ return get_length_impl(typed_op); \ } switch (type()) { ENUMERATE_BYTECODE_OPS(__BYTECODE_OP) default: VERIFY_NOT_REACHED(); } #undef __BYTECODE_OP } UnrealizedSourceRange InstructionStreamIterator::source_range() const { VERIFY(m_executable); auto record = m_executable->source_map.get(offset()).value(); return { .source_code = m_executable->source_code, .start_offset = record.source_start_offset, .end_offset = record.source_end_offset, }; } Operand::Operand(Register reg) : m_type(Type::Register) , m_index(reg.index()) { } }