123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178 |
- /* Copyright 2001,2009-2015,2018
- Free Software Foundation, Inc.
- This file is part of Guile.
- Guile 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.
- Guile 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 Guile. If not, see
- <https://www.gnu.org/licenses/>. */
- #ifndef _SCM_FRAMES_H_
- #define _SCM_FRAMES_H_
- #include <libguile/gc.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 + 3 = SCM_FRAME_PREVIOUS_SP (fp)
- | Dynamic link |
- +------------------------------+
- | Virtual return address (vRA) |
- +------------------------------+
- | Machine return address (mRA) |
- +==============================+ <- 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, the saved virtual return address, and the
- saved machine return address of the calling function, 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 0. 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
- {
- uintptr_t as_uint;
- uint32_t *as_vcode;
- uint8_t *as_mcode;
- SCM as_scm;
- double as_f64;
- uint64_t as_u64;
- int64_t as_s64;
- /* For GC purposes. */
- void *as_ptr;
- scm_t_bits as_bits;
- };
- #define SCM_FRAME_PREVIOUS_SP(fp) ((fp) + 3)
- #define SCM_FRAME_MACHINE_RETURN_ADDRESS(fp) ((fp)[0].as_mcode)
- #define SCM_FRAME_SET_MACHINE_RETURN_ADDRESS(fp, ra) ((fp)[0].as_mcode = (ra))
- #define SCM_FRAME_VIRTUAL_RETURN_ADDRESS(fp) ((fp)[1].as_vcode)
- #define SCM_FRAME_SET_VIRTUAL_RETURN_ADDRESS(fp, ra) ((fp)[1].as_vcode = (ra))
- #define SCM_FRAME_DYNAMIC_LINK(fp) ((fp) + (fp)[2].as_uint)
- #define SCM_FRAME_SET_DYNAMIC_LINK(fp, dl) ((fp)[2].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;
- ptrdiff_t fp_offset;
- ptrdiff_t sp_offset;
- uint32_t *ip;
- };
- enum scm_vm_frame_kind
- {
- SCM_VM_FRAME_KIND_VM,
- SCM_VM_FRAME_KIND_CONT
- };
- #define SCM_VM_FRAME_P(x) (SCM_HAS_TYP11 (x, scm_tc11_frame))
- #define SCM_VM_FRAME_KIND(x) ((enum scm_vm_frame_kind) (SCM_CELL_WORD_0 (x) >> 12))
- #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_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);
- /* 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_ */
|