/* Copyright (C) 2001, 2009, 2010, 2011, 2012, 2013, 2014 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 */ #if HAVE_CONFIG_H # include #endif #include #include #include "_scm.h" #include "frames.h" #include "vm.h" #include /* Make sure assumptions on the layout of `struct scm_vm_frame' hold. */ verify (sizeof (SCM) == sizeof (SCM *)); verify (sizeof (struct scm_vm_frame) == 3 * sizeof (SCM)); verify (offsetof (struct scm_vm_frame, dynamic_link) == 0); SCM scm_c_make_frame (enum scm_vm_frame_kind kind, const struct scm_frame *frame) { struct scm_frame *p = scm_gc_malloc (sizeof (struct scm_frame), "vmframe"); p->stack_holder = frame->stack_holder; p->fp_offset = frame->fp_offset; p->sp_offset = frame->sp_offset; p->ip = frame->ip; return scm_cell (scm_tc7_frame | (kind << 8), (scm_t_bits)p); } void scm_i_frame_print (SCM frame, SCM port, scm_print_state *pstate) { scm_puts_unlocked ("#", port); } static SCM* frame_stack_base (enum scm_vm_frame_kind kind, const struct scm_frame *frame) { switch (kind) { case SCM_VM_FRAME_KIND_CONT: return ((struct scm_vm_cont *) frame->stack_holder)->stack_base; case SCM_VM_FRAME_KIND_VM: return ((struct scm_vm *) frame->stack_holder)->stack_base; default: abort (); } } static scm_t_ptrdiff frame_offset (enum scm_vm_frame_kind kind, const struct scm_frame *frame) { switch (kind) { case SCM_VM_FRAME_KIND_CONT: return ((struct scm_vm_cont *) frame->stack_holder)->reloc; case SCM_VM_FRAME_KIND_VM: return 0; default: abort (); } } SCM* scm_i_frame_stack_base (SCM frame) #define FUNC_NAME "frame-stack-base" { SCM_VALIDATE_VM_FRAME (1, frame); return frame_stack_base (SCM_VM_FRAME_KIND (frame), SCM_VM_FRAME_DATA (frame)); } #undef FUNC_NAME scm_t_ptrdiff scm_i_frame_offset (SCM frame) #define FUNC_NAME "frame-offset" { SCM_VALIDATE_VM_FRAME (1, frame); return frame_offset (SCM_VM_FRAME_KIND (frame), SCM_VM_FRAME_DATA (frame)); } #undef FUNC_NAME /* Scheme interface */ SCM_DEFINE (scm_frame_p, "frame?", 1, 0, 0, (SCM obj), "") #define FUNC_NAME s_scm_frame_p { return scm_from_bool (SCM_VM_FRAME_P (obj)); } #undef FUNC_NAME /* Retrieve the local in slot 0, which may or may not actually be a procedure, and may or may not actually be the procedure being applied. If you want the procedure, look it up from the IP. */ SCM scm_c_frame_closure (enum scm_vm_frame_kind kind, const struct scm_frame *frame) { SCM *fp = frame_stack_base (kind, frame) + frame->fp_offset; return SCM_FRAME_PROGRAM (fp); } SCM_DEFINE (scm_frame_procedure, "frame-procedure", 1, 0, 0, (SCM frame), "") #define FUNC_NAME s_scm_frame_procedure { SCM_VALIDATE_VM_FRAME (1, frame); /* FIXME: Retrieve procedure from address? */ return scm_c_frame_closure (SCM_VM_FRAME_KIND (frame), SCM_VM_FRAME_DATA (frame)); } #undef FUNC_NAME static SCM frame_arguments_var; static void init_frame_arguments_var (void) { frame_arguments_var = scm_c_private_lookup ("system vm frame", "frame-arguments"); } SCM_DEFINE (scm_frame_arguments, "frame-arguments", 1, 0, 0, (SCM frame), "") #define FUNC_NAME s_scm_frame_arguments { static scm_i_pthread_once_t once = SCM_I_PTHREAD_ONCE_INIT; scm_i_pthread_once (&once, init_frame_arguments_var); SCM_VALIDATE_VM_FRAME (1, frame); return scm_call_1 (scm_variable_ref (frame_arguments_var), frame); } #undef FUNC_NAME SCM_DEFINE (scm_frame_source, "frame-source", 1, 0, 0, (SCM frame), "") #define FUNC_NAME s_scm_frame_source { SCM_VALIDATE_VM_FRAME (1, frame); return scm_find_source_for_addr (scm_frame_instruction_pointer (frame)); } #undef FUNC_NAME SCM_DEFINE (scm_frame_num_locals, "frame-num-locals", 1, 0, 0, (SCM frame), "") #define FUNC_NAME s_scm_frame_num_locals { SCM *fp, *sp; SCM_VALIDATE_VM_FRAME (1, frame); fp = SCM_VM_FRAME_FP (frame); sp = SCM_VM_FRAME_SP (frame); return scm_from_ptrdiff_t (SCM_FRAME_NUM_LOCALS (fp, sp)); } #undef FUNC_NAME SCM_DEFINE (scm_frame_local_ref, "frame-local-ref", 2, 0, 0, (SCM frame, SCM index), "") #define FUNC_NAME s_scm_frame_local_ref { SCM *fp, *sp; unsigned int i; SCM_VALIDATE_VM_FRAME (1, frame); SCM_VALIDATE_UINT_COPY (2, index, i); fp = SCM_VM_FRAME_FP (frame); sp = SCM_VM_FRAME_SP (frame); if (i < SCM_FRAME_NUM_LOCALS (fp, sp)) return SCM_FRAME_LOCAL (fp, i); SCM_OUT_OF_RANGE (SCM_ARG2, index); } #undef FUNC_NAME /* Need same not-yet-active frame logic here as in frame-num-locals */ SCM_DEFINE (scm_frame_local_set_x, "frame-local-set!", 3, 0, 0, (SCM frame, SCM index, SCM val), "") #define FUNC_NAME s_scm_frame_local_set_x { SCM *fp, *sp; unsigned int i; SCM_VALIDATE_VM_FRAME (1, frame); SCM_VALIDATE_UINT_COPY (2, index, i); fp = SCM_VM_FRAME_FP (frame); sp = SCM_VM_FRAME_SP (frame); if (i < SCM_FRAME_NUM_LOCALS (fp, sp)) { SCM_FRAME_LOCAL (fp, i) = val; return SCM_UNSPECIFIED; } SCM_OUT_OF_RANGE (SCM_ARG2, index); } #undef FUNC_NAME SCM_DEFINE (scm_frame_address, "frame-address", 1, 0, 0, (SCM frame), "Return the frame pointer for @var{frame}.") #define FUNC_NAME s_scm_frame_address { SCM_VALIDATE_VM_FRAME (1, frame); return scm_from_uintptr_t ((scm_t_uintptr) SCM_VM_FRAME_FP (frame)); } #undef FUNC_NAME SCM_DEFINE (scm_frame_stack_pointer, "frame-stack-pointer", 1, 0, 0, (SCM frame), "") #define FUNC_NAME s_scm_frame_stack_pointer { SCM_VALIDATE_VM_FRAME (1, frame); return scm_from_uintptr_t ((scm_t_uintptr) SCM_VM_FRAME_SP (frame)); } #undef FUNC_NAME SCM_DEFINE (scm_frame_instruction_pointer, "frame-instruction-pointer", 1, 0, 0, (SCM frame), "") #define FUNC_NAME s_scm_frame_instruction_pointer { SCM_VALIDATE_VM_FRAME (1, frame); return scm_from_uintptr_t ((scm_t_uintptr) SCM_VM_FRAME_IP (frame)); } #undef FUNC_NAME SCM_DEFINE (scm_frame_return_address, "frame-return-address", 1, 0, 0, (SCM frame), "") #define FUNC_NAME s_scm_frame_return_address { SCM_VALIDATE_VM_FRAME (1, frame); return scm_from_uintptr_t ((scm_t_uintptr) (SCM_FRAME_RETURN_ADDRESS (SCM_VM_FRAME_FP (frame)))); } #undef FUNC_NAME #define RELOC(kind, frame, val) \ (((SCM *) (val)) + frame_offset (kind, frame)) SCM_DEFINE (scm_frame_dynamic_link, "frame-dynamic-link", 1, 0, 0, (SCM frame), "") #define FUNC_NAME s_scm_frame_dynamic_link { SCM_VALIDATE_VM_FRAME (1, frame); /* fixme: munge fp if holder is a continuation */ return scm_from_uintptr_t ((scm_t_uintptr) RELOC (SCM_VM_FRAME_KIND (frame), SCM_VM_FRAME_DATA (frame), SCM_FRAME_DYNAMIC_LINK (SCM_VM_FRAME_FP (frame)))); } #undef FUNC_NAME int scm_c_frame_previous (enum scm_vm_frame_kind kind, struct scm_frame *frame) { SCM *this_fp, *new_fp, *new_sp; SCM proc; again: this_fp = frame->fp_offset + frame_stack_base (kind, frame); new_fp = SCM_FRAME_DYNAMIC_LINK (this_fp); if (new_fp) { SCM *stack_base = frame_stack_base (kind, frame); new_fp = RELOC (kind, frame, new_fp); new_sp = SCM_FRAME_PREVIOUS_SP (this_fp); frame->fp_offset = new_fp - stack_base; frame->sp_offset = new_sp - stack_base; frame->ip = SCM_FRAME_RETURN_ADDRESS (this_fp); proc = SCM_FRAME_PROGRAM (new_fp); if (SCM_PROGRAM_P (proc) && SCM_PROGRAM_IS_BOOT (proc)) goto again; else return 1; } else return 0; } SCM_DEFINE (scm_frame_previous, "frame-previous", 1, 0, 0, (SCM frame), "") #define FUNC_NAME s_scm_frame_previous { enum scm_vm_frame_kind kind; struct scm_frame tmp; SCM_VALIDATE_VM_FRAME (1, frame); kind = SCM_VM_FRAME_KIND (frame); memcpy (&tmp, SCM_VM_FRAME_DATA (frame), sizeof tmp); if (!scm_c_frame_previous (SCM_VM_FRAME_KIND (frame), &tmp)) return SCM_BOOL_F; return scm_c_make_frame (kind, &tmp); } #undef FUNC_NAME void scm_init_frames (void) { #ifndef SCM_MAGIC_SNARFER #include "libguile/frames.x" #endif } /* Local Variables: c-file-style: "gnu" End: */