52 lines
1.3 KiB
C++
Executable File
52 lines
1.3 KiB
C++
Executable File
namespace factor
|
|
{
|
|
|
|
inline static cell callstack_size(cell size)
|
|
{
|
|
return sizeof(callstack) + size;
|
|
}
|
|
|
|
stack_frame *fix_callstack_top(stack_frame *top, stack_frame *bottom);
|
|
stack_frame *frame_successor(stack_frame *frame);
|
|
code_block *frame_code(stack_frame *frame);
|
|
cell frame_executing(stack_frame *frame);
|
|
cell frame_scan(stack_frame *frame);
|
|
cell frame_type(stack_frame *frame);
|
|
|
|
PRIMITIVE(callstack);
|
|
PRIMITIVE(set_callstack);
|
|
PRIMITIVE(callstack_to_array);
|
|
PRIMITIVE(innermost_stack_frame_executing);
|
|
PRIMITIVE(innermost_stack_frame_scan);
|
|
PRIMITIVE(set_innermost_stack_frame_quot);
|
|
|
|
VM_ASM_API void save_callstack_bottom(stack_frame *callstack_bottom);
|
|
|
|
template<typename T> void iterate_callstack(cell top, cell bottom, T &iterator)
|
|
{
|
|
stack_frame *frame = (stack_frame *)bottom - 1;
|
|
|
|
while((cell)frame >= top)
|
|
{
|
|
iterator(frame);
|
|
frame = frame_successor(frame);
|
|
}
|
|
}
|
|
|
|
/* This is a little tricky. The iterator may allocate memory, so we
|
|
keep the callstack in a GC root and use relative offsets */
|
|
template<typename T> void iterate_callstack_object(callstack *stack_, T &iterator)
|
|
{
|
|
gc_root<callstack> stack(stack_);
|
|
fixnum frame_offset = untag_fixnum(stack->length) - sizeof(stack_frame);
|
|
|
|
while(frame_offset >= 0)
|
|
{
|
|
stack_frame *frame = stack->frame_at(frame_offset);
|
|
frame_offset -= frame->size;
|
|
iterator(frame);
|
|
}
|
|
}
|
|
|
|
}
|