/* * Copyright (c) 2021, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include namespace JS::Bytecode { class InstructionStreamIterator { public: explicit InstructionStreamIterator(ReadonlyBytes bytes) : m_bytes(bytes) { } size_t offset() const { return m_offset; } bool at_end() const { return m_offset >= m_bytes.size(); } void jump(size_t offset) { VERIFY(offset <= m_bytes.size()); m_offset = offset; } Instruction const& operator*() const { return dereference(); } void operator++(); private: Instruction const& dereference() const { return *reinterpret_cast(m_bytes.data() + offset()); } ReadonlyBytes m_bytes; size_t m_offset { 0 }; }; struct UnwindInfo { BasicBlock const* handler; BasicBlock const* finalizer; }; class BasicBlock { AK_MAKE_NONCOPYABLE(BasicBlock); public: static NonnullOwnPtr create(String name, size_t size = 4 * KiB); ~BasicBlock(); void seal(); void dump(Executable const&) const; ReadonlyBytes instruction_stream() const { return ReadonlyBytes { m_buffer, m_buffer_size }; } size_t size() const { return m_buffer_size; } void* next_slot() { return m_buffer + m_buffer_size; } bool can_grow(size_t additional_size) const { return m_buffer_size + additional_size <= m_buffer_capacity; } void grow(size_t additional_size); void terminate(Badge) { m_is_terminated = true; } bool is_terminated() const { return m_is_terminated; } String const& name() const { return m_name; } private: BasicBlock(String name, size_t size); u8* m_buffer { nullptr }; size_t m_buffer_capacity { 0 }; size_t m_buffer_size { 0 }; bool m_is_terminated { false }; String m_name; }; }