/* * Copyright (c) 2021, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include namespace JS::Bytecode { Instruction::Instruction(Type type, size_t length) : m_type(type) , m_length(length) { VERIFY(length <= NumericLimits::max()); } 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 } UnrealizedSourceRange InstructionStreamIterator::source_range() const { VERIFY(m_executable); auto record = dereference().source_record(); return { .source_code = m_executable->source_code, .start_offset = record.source_start_offset, .end_offset = record.source_end_offset, }; } RefPtr InstructionStreamIterator::source_code() const { return m_executable ? m_executable->source_code.ptr() : nullptr; } Operand::Operand(Register reg) : m_type(Type::Register) , m_index(reg.index()) { } }