mirror of
https://git.savannah.gnu.org/git/guile.git
synced 2025-06-11 22:31:12 +02:00
Adapt VM stack to grow downward. This will make native compilation look more like the VM code, as we will be able to use native CALL instructions, taking proper advantage of the return address buffer. * libguile/continuations.c (scm_i_continuation_to_frame): Record offsets from stack top. * libguile/control.c (scm_i_prompt_pop_abort_args_x): Adapt for reversed order of arguments, and instead of relying on the abort to push on the number of arguments, make the caller save the stack depth, which allows us to compute the number of arguments ourselves. (reify_partial_continuation, scm_c_abort): Adapt to reversed stack order. * libguile/dynstack.c (scm_dynstack_wind_prompt): Since we wind the stack in a downward direction, subtract the reloc instead of adding it. * libguile/dynstack.h (SCM_F_DYNSTACK_PROMPT_ESCAPE_ONLY): Remove flag; instead rely on prompt-establishing code to save the stack depth. * libguile/eval.c (eval): Remove extraneous "volatile" declarations for variables that are not re-set between the setjmp and any longjmp. Adapt to save stack depth before instating the prompt. * libguile/foreign.c (scm_i_foreign_call): Adapt to receive arguments in reverse order. * libguile/frames.c (frame_stack_top, scm_i_frame_stack_top): Adapt to compute stack top instead of stack bottom. (scm_c_frame_closure): Adapt to stack growth change. (scm_frame_num_locals, scm_frame_local_ref, scm_frame_set_x): Use union data type to access stack. (RELOC): Reformat. (scm_c_frame_previous): Adapt to stack growth change. * libguile/frames.h: Adapt stack diagram to indicate that the stack grows up. (union scm_vm_stack_element): New data type used to access items on the stack. (SCM_FRAME_PREVIOUS_SP) (SCM_FRAME_RETURN_ADDRESS, SCM_FRAME_SET_RETURN_ADDRESS) (SCM_FRAME_DYNAMIC_LINK, SCM_FRAME_SET_DYNAMIC_LINK) (SCM_FRAME_LOCAL, SCM_FRAME_NUM_LOCALS): Adapt to stack representation change. (SCM_FRAME_SLOT): New helper. (SCM_VM_FRAME_FP, SCM_VM_FRAME_SP): Adapt to stack growth change. * libguile/stacks.c (scm_make_stack): Record offsets from top of stack. * libguile/throw.c (catch): Adapt to scm_i_prompt_pop_abort_args_x change. * libguile/vm-engine.c (ALLOC_FRAME, RESET_FRAME): (FRAME_LOCALS_COUNT_FROM): Adapt to stack growth change. (LOCAL_ADDRESS): Use SCM_FRAME_SLOT to get the address as the proper data type. (RETURN_ONE_VALUE, RETURN_VALUE_LIST): Adapt to stack growth change. (apply): Shuffling up the SMOB apply args can cause the stack to expand, so use ALLOC_FRAME instead of RESET_FRAME. (vm_engine): Adapt for stack growth change. * libguile/vm.c (vm_increase_sp, vm_push_sp, vm_restore_sp): Adapt to stack representation change. (scm_i_vm_cont_to_frame): Adapt to take offsets from the top. (scm_i_vm_capture_stack): Adapt to capture from the top. (vm_return_to_continuation_inner): Adapt for data type changes. (vm_return_to_continuation): Likewise, and instead of looping, just splat the saved arguments on with memcpy. (vm_dispatch_hook): Adapt to receive arguments in the reverse order. Adapt callers. (vm_abort): There is never a tail argument. Adapt to stack representation change. (vm_reinstate_partial_continuation) (vm_reinstate_partial_continuation_inner): Adapt to stack growth change. (allocate_stack, free_stack): Adapt to data type change. (expand_stack): Don't try to mremap(), as you can't grow a mapping from the bottom. Without knowing that there's a free mapping space right below the old stack, which there usually isn't on Linux, we have to copy. We can't use MAP_GROWSDOWN because Linux is buggy. (make_vm): Adapt to stack representation changes. (return_unused_stack_to_os): Round down instead of up, as the stack grows down. (scm_i_vm_mark_stack): Adapt to walk up the stack. (scm_i_vm_free_stack): Adapt to scm_vm changes. (vm_expand_stack_inner, reset_stack_limit, vm_expand_stack): Adapt to the stack growing down. (scm_call_n): Adapt to the stack growing down. Don't allow argv to point into the stack. * libguile/vm.h (struct scm_vm, struct scm_vm_cont): Adapt to hold the stack top and bottom.
121 lines
4.7 KiB
C
121 lines
4.7 KiB
C
/* Copyright (C) 2001, 2009, 2010, 2011, 2012, 2013, 2014, 2015 Free Software Foundation, Inc.
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Lesser General Public License
|
|
* as published by the Free Software Foundation; either version 3 of
|
|
* the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful, but
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with this library; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
|
|
* 02110-1301 USA
|
|
*/
|
|
|
|
#ifndef _SCM_VM_H_
|
|
#define _SCM_VM_H_
|
|
|
|
#include <libguile.h>
|
|
#include <libguile/programs.h>
|
|
|
|
enum {
|
|
SCM_VM_APPLY_HOOK,
|
|
SCM_VM_PUSH_CONTINUATION_HOOK,
|
|
SCM_VM_POP_CONTINUATION_HOOK,
|
|
SCM_VM_NEXT_HOOK,
|
|
SCM_VM_ABORT_CONTINUATION_HOOK,
|
|
SCM_VM_NUM_HOOKS,
|
|
};
|
|
|
|
#define SCM_VM_REGULAR_ENGINE 0
|
|
#define SCM_VM_DEBUG_ENGINE 1
|
|
#define SCM_VM_NUM_ENGINES 2
|
|
|
|
struct scm_vm {
|
|
scm_t_uint32 *ip; /* instruction pointer */
|
|
union scm_vm_stack_element *sp; /* stack pointer */
|
|
union scm_vm_stack_element *fp; /* frame pointer */
|
|
union scm_vm_stack_element *stack_limit; /* stack limit address */
|
|
int trace_level; /* traces enabled if trace_level > 0 */
|
|
union scm_vm_stack_element *sp_min_since_gc; /* deepest sp since last gc */
|
|
size_t stack_size; /* stack size */
|
|
union scm_vm_stack_element *stack_bottom; /* lowest address in allocated stack */
|
|
union scm_vm_stack_element *stack_top; /* highest address in allocated stack */
|
|
SCM overflow_handler_stack; /* alist of max-stack-size -> thunk */
|
|
SCM hooks[SCM_VM_NUM_HOOKS]; /* hooks */
|
|
int engine; /* which vm engine we're using */
|
|
};
|
|
|
|
SCM_INTERNAL struct scm_vm *scm_the_vm (void);
|
|
SCM_API SCM scm_call_with_vm (SCM proc, SCM args);
|
|
|
|
SCM_API SCM scm_call_with_stack_overflow_handler (SCM limit, SCM thunk,
|
|
SCM handler);
|
|
|
|
SCM_API SCM scm_vm_apply_hook (void);
|
|
SCM_API SCM scm_vm_push_continuation_hook (void);
|
|
SCM_API SCM scm_vm_pop_continuation_hook (void);
|
|
SCM_API SCM scm_vm_abort_continuation_hook (void);
|
|
SCM_API SCM scm_vm_next_hook (void);
|
|
SCM_API SCM scm_vm_trace_level (void);
|
|
SCM_API SCM scm_set_vm_trace_level_x (SCM level);
|
|
SCM_API SCM scm_vm_engine (void);
|
|
SCM_API SCM scm_set_vm_engine_x (SCM engine);
|
|
SCM_API SCM scm_set_default_vm_engine_x (SCM engine);
|
|
SCM_API void scm_c_set_vm_engine_x (int engine);
|
|
SCM_API void scm_c_set_default_vm_engine_x (int engine);
|
|
|
|
struct GC_ms_entry;
|
|
SCM_INTERNAL struct GC_ms_entry * scm_i_vm_mark_stack (struct scm_vm *,
|
|
struct GC_ms_entry *,
|
|
struct GC_ms_entry *);
|
|
SCM_INTERNAL void scm_i_vm_free_stack (struct scm_vm *vp);
|
|
|
|
#define SCM_F_VM_CONT_PARTIAL 0x1
|
|
#define SCM_F_VM_CONT_REWINDABLE 0x2
|
|
|
|
struct scm_vm_cont {
|
|
/* FIXME: sp isn't needed, it's effectively the same as
|
|
stack_bottom */
|
|
union scm_vm_stack_element *sp;
|
|
union scm_vm_stack_element *fp;
|
|
scm_t_uint32 *ra;
|
|
scm_t_ptrdiff stack_size;
|
|
union scm_vm_stack_element *stack_bottom;
|
|
scm_t_ptrdiff reloc;
|
|
scm_t_dynstack *dynstack;
|
|
scm_t_uint32 flags;
|
|
};
|
|
|
|
#define SCM_VM_CONT_P(OBJ) (SCM_HAS_TYP7 (OBJ, scm_tc7_vm_cont))
|
|
#define SCM_VM_CONT_DATA(CONT) ((struct scm_vm_cont *) SCM_CELL_WORD_1 (CONT))
|
|
#define SCM_VM_CONT_PARTIAL_P(CONT) (SCM_VM_CONT_DATA (CONT)->flags & SCM_F_VM_CONT_PARTIAL)
|
|
#define SCM_VM_CONT_REWINDABLE_P(CONT) (SCM_VM_CONT_DATA (CONT)->flags & SCM_F_VM_CONT_REWINDABLE)
|
|
|
|
SCM_API SCM scm_load_compiled_with_vm (SCM file);
|
|
|
|
SCM_INTERNAL SCM scm_i_call_with_current_continuation (SCM proc);
|
|
SCM_INTERNAL SCM scm_i_capture_current_stack (void);
|
|
SCM_INTERNAL SCM scm_i_vm_capture_stack (union scm_vm_stack_element *stack_top,
|
|
union scm_vm_stack_element *fp,
|
|
union scm_vm_stack_element *sp,
|
|
scm_t_uint32 *ra,
|
|
scm_t_dynstack *dynstack,
|
|
scm_t_uint32 flags);
|
|
SCM_INTERNAL int scm_i_vm_cont_to_frame (SCM cont, struct scm_frame *frame);
|
|
SCM_INTERNAL void scm_i_vm_cont_print (SCM x, SCM port,
|
|
scm_print_state *pstate);
|
|
SCM_INTERNAL void scm_bootstrap_vm (void);
|
|
SCM_INTERNAL void scm_init_vm (void);
|
|
|
|
#endif /* _SCM_VM_H_ */
|
|
|
|
/*
|
|
Local Variables:
|
|
c-file-style: "gnu"
|
|
End:
|
|
*/
|