#pragma once #include #include "class_file.h" #include "class_resolver.h" struct frame { struct Code_attribute * code; uint32_t * local_variable; uint32_t * operand_stack; uint32_t pc; uint32_t next_pc; uint16_t operand_stack_ix; uint8_t return_type; }; struct thread { struct class_file * current_class; struct method_info * current_method; }; struct stack { union { struct frame * frame; uint32_t * data; }; uint32_t ix; uint32_t capacity; }; struct vm { struct stack frame_stack; struct stack data_stack; struct thread current_thread; struct frame * current_frame; struct { int length; struct hash_table_entry * entry; } class_hash_table; }; static inline struct frame * stack_push_frame(struct stack * stack, int num_frames) { struct frame * frame = &stack->frame[stack->ix]; stack->ix += num_frames; assert(stack->ix <= stack->capacity); return frame; } static inline struct frame * stack_pop_frame(struct stack * stack, int num_frames) { stack->ix -= num_frames; assert(stack->ix >= 0); struct frame * frame = &stack->frame[stack->ix - 1]; return frame; } static inline uint32_t * stack_push_data(struct stack * stack, int num_data) { uint32_t * data = &stack->data[stack->ix]; stack->ix += num_data; assert(stack->ix <= stack->capacity); return data; } static inline uint32_t * stack_pop_data(struct stack * stack, int num_data) { stack->ix -= num_data; assert(stack->ix >= 0); uint32_t * data = &stack->data[stack->ix]; return data; } static inline void operand_stack_push_u32(struct frame * frame, uint32_t value) { frame->operand_stack[frame->operand_stack_ix] = value; frame->operand_stack_ix++; } static inline uint32_t operand_stack_pop_u32(struct frame * frame) { frame->operand_stack_ix--; uint32_t value = frame->operand_stack[frame->operand_stack_ix]; frame->operand_stack[frame->operand_stack_ix] = -1; return value; } static inline void operand_stack_dup_u32(struct frame * frame) { uint32_t value = frame->operand_stack[frame->operand_stack_ix - 1]; frame->operand_stack[frame->operand_stack_ix] = value; frame->operand_stack_ix++; } static inline void operand_stack_push_f32(struct frame * frame, float f) { uint32_t value = *((uint32_t *)&f); frame->operand_stack[frame->operand_stack_ix] = value; frame->operand_stack_ix++; } static inline float operand_stack_pop_f32(struct frame * frame) { frame->operand_stack_ix--; uint32_t value = frame->operand_stack[frame->operand_stack_ix]; frame->operand_stack[frame->operand_stack_ix] = -1; float f = *((float *)&value); return f; } static inline void operand_stack_push_u64(struct frame * frame, uint64_t value) { frame->operand_stack[frame->operand_stack_ix] = (uint32_t)(value >> 0); frame->operand_stack_ix++; frame->operand_stack[frame->operand_stack_ix] = (uint32_t)(value >> 32); frame->operand_stack_ix++; } static inline uint64_t operand_stack_pop_u64(struct frame * frame) { frame->operand_stack_ix--; uint64_t high = frame->operand_stack[frame->operand_stack_ix]; frame->operand_stack_ix--; uint64_t low = frame->operand_stack[frame->operand_stack_ix]; uint64_t value = (high << 32) | (low << 0); return value; } bool vm_initialize_class(struct vm * vm, struct class_entry * class_entry); void vm_static_method_call(struct vm * vm, struct class_file * class_file, struct method_info * method_info); void vm_method_return(struct vm * vm); void vm_execute(struct vm * vm);