123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183 |
- /* 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_FRAMES_H_
- #define _SCM_FRAMES_H_
- #include <libguile.h>
- #include "programs.h"
- /* Stack frames
- ------------
- It's a little confusing, but there are two representations of frames
- in this file: frame pointers, and Scheme objects wrapping those frame
- pointers. The former uses the SCM_FRAME macro prefix, the latter
- SCM_VM_FRAME prefix.
- The confusing thing is that only Scheme frame objects have functions
- that use them, and they use the lower-case scm_frame prefix.
- Stack frame layout
- ------------------
- | ... |
- +==================+ <- fp + 2 = SCM_FRAME_PREVIOUS_SP (fp)
- | Dynamic link |
- +------------------+
- | Return address |
- +==================+ <- fp
- | Local 0 |
- +------------------+
- | Local 1 |
- +------------------+
- | ... |
- +------------------+
- | Local N-1 |
- \------------------/ <- sp
- The stack grows down.
- The calling convention is that a caller prepares a stack frame
- consisting of the saved FP and the return address, followed by the
- procedure and then the arguments to the call, in order. Thus in the
- beginning of a call, the procedure being called is in slot 0, the
- first argument is in slot 1, and the SP points to the last argument.
- The number of arguments, including the procedure, is thus FP - SP.
- After ensuring that the correct number of arguments have been passed,
- a function will set the stack pointer to point to the last local
- slot. This lets a function allocate the temporary space that it
- needs once in the beginning of the call, instead of pushing and
- popping the stack pointer during the call's extent.
- When a program returns, it returns its values in the slots starting
- from local 1, as if the values were arguments to a tail call. We
- start from 1 instead of 0 for the convenience of the "values" builtin
- function, which can just leave its arguments in place.
- The callee resets the stack pointer to point to the last value. In
- this way the caller knows how many values there are: it's the number
- of words between the stack pointer and the slot at which the caller
- placed the procedure.
- After checking that the number of values returned is appropriate, the
- caller shuffles the values around (if needed), and resets the stack
- pointer back to its original value from before the call. */
- /* Each element on the stack occupies the same amount of space. */
- union scm_vm_stack_element
- {
- scm_t_uintptr as_uint;
- scm_t_uint32 *as_ip;
- SCM as_scm;
- double as_f64;
- scm_t_uint64 as_u64;
- scm_t_int64 as_s64;
- /* For GC purposes. */
- void *as_ptr;
- scm_t_bits as_bits;
- };
- #define SCM_FRAME_PREVIOUS_SP(fp) ((fp) + 2)
- #define SCM_FRAME_RETURN_ADDRESS(fp) ((fp)[0].as_ip)
- #define SCM_FRAME_SET_RETURN_ADDRESS(fp, ra) ((fp)[0].as_ip = (ra))
- #define SCM_FRAME_DYNAMIC_LINK(fp) ((fp) + (fp)[1].as_uint)
- #define SCM_FRAME_SET_DYNAMIC_LINK(fp, dl) ((fp)[1].as_uint = ((dl) - (fp)))
- #define SCM_FRAME_SLOT(fp,i) ((fp) - (i) - 1)
- #define SCM_FRAME_LOCAL(fp,i) (SCM_FRAME_SLOT (fp, i)->as_scm)
- #define SCM_FRAME_NUM_LOCALS(fp, sp) ((fp) - (sp))
- /*
- * Heap frames
- */
- #ifdef BUILDING_LIBGUILE
- struct scm_frame
- {
- void *stack_holder;
- scm_t_ptrdiff fp_offset;
- scm_t_ptrdiff sp_offset;
- scm_t_uint32 *ip;
- };
- enum scm_vm_frame_kind
- {
- SCM_VM_FRAME_KIND_VM,
- SCM_VM_FRAME_KIND_CONT
- };
- #define SCM_VM_FRAME_P(x) (SCM_HAS_TYP7 (x, scm_tc7_frame))
- #define SCM_VM_FRAME_KIND(x) ((enum scm_vm_frame_kind) (SCM_CELL_WORD_0 (x) >> 8))
- #define SCM_VM_FRAME_DATA(x) ((struct scm_frame *)SCM_CELL_WORD_1 (x))
- #define SCM_VM_FRAME_STACK_HOLDER(f) SCM_VM_FRAME_DATA (f)->stack_holder
- #define SCM_VM_FRAME_FP_OFFSET(f) SCM_VM_FRAME_DATA (f)->fp_offset
- #define SCM_VM_FRAME_SP_OFFSET(f) SCM_VM_FRAME_DATA (f)->sp_offset
- #define SCM_VM_FRAME_FP(f) (scm_i_frame_stack_top (f) - SCM_VM_FRAME_FP_OFFSET (f))
- #define SCM_VM_FRAME_SP(f) (scm_i_frame_stack_top (f) - SCM_VM_FRAME_SP_OFFSET (f))
- #define SCM_VM_FRAME_IP(f) SCM_VM_FRAME_DATA (f)->ip
- #define SCM_VM_FRAME_OFFSET(f) scm_i_frame_offset (f)
- #define SCM_VALIDATE_VM_FRAME(p,x) SCM_MAKE_VALIDATE (p, x, VM_FRAME_P)
- SCM_INTERNAL union scm_vm_stack_element* scm_i_frame_stack_top (SCM frame);
- SCM_INTERNAL scm_t_ptrdiff scm_i_frame_offset (SCM frame);
- /* See notes in frames.c before using this. */
- SCM_INTERNAL SCM scm_c_frame_closure (enum scm_vm_frame_kind kind,
- const struct scm_frame *frame);
- SCM_INTERNAL SCM scm_c_make_frame (enum scm_vm_frame_kind kind,
- const struct scm_frame *frame);
- SCM_INTERNAL int scm_c_frame_previous (enum scm_vm_frame_kind kind,
- struct scm_frame *frame);
- #endif
- SCM_API SCM scm_frame_p (SCM obj);
- SCM_API SCM scm_frame_procedure_name (SCM frame);
- SCM_API SCM scm_frame_call_representation (SCM frame);
- SCM_API SCM scm_frame_arguments (SCM frame);
- SCM_API SCM scm_frame_source (SCM frame);
- SCM_API SCM scm_frame_address (SCM frame);
- SCM_API SCM scm_frame_stack_pointer (SCM frame);
- SCM_API SCM scm_frame_instruction_pointer (SCM frame);
- SCM_API SCM scm_frame_return_address (SCM frame);
- SCM_API SCM scm_frame_dynamic_link (SCM frame);
- SCM_API SCM scm_frame_previous (SCM frame);
- SCM_INTERNAL void scm_i_frame_print (SCM frame, SCM port,
- scm_print_state *pstate);
- SCM_INTERNAL void scm_init_frames (void);
- #endif /* _SCM_FRAMES_H_ */
- /*
- Local Variables:
- c-file-style: "gnu"
- End:
- */
|