/* * Copyright (c) 2021, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #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 }; }; class Block { public: static NonnullOwnPtr create(); ~Block(); void seal(); void dump() const; ReadonlyBytes instruction_stream() const { return ReadonlyBytes { m_buffer, m_buffer_size }; } size_t register_count() const { return m_register_count; } void set_register_count(Badge, size_t count) { m_register_count = count; } void* next_slot() { return m_buffer + m_buffer_size; } void grow(size_t additional_size); private: Block(); size_t m_register_count { 0 }; u8* m_buffer { nullptr }; size_t m_buffer_capacity { 0 }; size_t m_buffer_size { 0 }; }; }