#ifndef SCM_DYNSTACK_H #define SCM_DYNSTACK_H /* Copyright 2012-2013,2018,2025 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 . */ #ifndef _WIN64 #include #else #include "libguile/setjump-win.h" #endif #include #include "libguile/scm.h" typedef struct scm_dynstack { scm_t_bits tag; scm_t_bits *base; scm_t_bits *top; scm_t_bits *limit; scm_t_bits inline_storage[]; } scm_t_dynstack; /* Items on the dynstack are preceded by two-word headers, giving the offset of the preceding item (or 0 if there is none) and the type, flags, and length of the following dynstack entry, in words. In addition, there is a "null header" at the top of the stack, indicating the length of the previous item, but with a tag of zero. For example, consider an empty dynstack, with a capacity of 6 words: +----------+----------+ + |prev=0 |tag=0 | | +----------+----------+ + ^base ^top limit^ Now we evaluate (dynamic-wind enter thunk leave). That will result in a dynstack of: / the len=2 words \ +----------+----------+----------+----------+----------+----------+ |prev=0 |tag:len=2 |enter |leave |prev=4 |tag=0 | +----------+----------+----------+----------+----------+----------+ ^base top,limit^ The tag is a combination of the type of the dynstack item, some flags associated with the item, and the length of the item. See scm_make_dynstack_tag below for the details. This arrangement makes it possible to have variable-length dynstack items, and yet be able to traverse them forwards or backwards. */ #define SCM_DYNSTACK_HEADER_LEN 2 static inline ptrdiff_t scm_dynstack_prev_offset (scm_t_bits *top) { return top[-2]; } static inline void scm_dynstack_set_prev_offset (scm_t_bits *top, ptrdiff_t offset) { top[-2] = offset; } static inline scm_t_bits scm_dynstack_tag (scm_t_bits *top) { return top[-1]; } static inline void scm_dynstack_set_tag (scm_t_bits *top, scm_t_bits tag) { top[-1] = tag; } typedef enum { SCM_DYNSTACK_TYPE_NONE = 0, SCM_DYNSTACK_TYPE_FRAME, SCM_DYNSTACK_TYPE_UNWINDER, SCM_DYNSTACK_TYPE_REWINDER, SCM_DYNSTACK_TYPE_WITH_FLUID, SCM_DYNSTACK_TYPE_PROMPT, SCM_DYNSTACK_TYPE_DYNWIND, SCM_DYNSTACK_TYPE_DYNAMIC_STATE, } scm_t_dynstack_item_type; #define SCM_DYNSTACK_TAG_TYPE_MASK 0xf #define SCM_DYNSTACK_TAG_FLAGS_MASK 0xf0 #define SCM_DYNSTACK_TAG_FLAGS_SHIFT 4 #define SCM_DYNSTACK_TAG_LEN_SHIFT 8 static inline scm_t_bits scm_make_dynstack_tag (scm_t_dynstack_item_type type, scm_t_bits flags, size_t len) { if (type & ~SCM_DYNSTACK_TAG_TYPE_MASK) abort (); if (flags & ~SCM_DYNSTACK_TAG_FLAGS_MASK) abort (); scm_t_bits ret = type; ret |= flags; ret |= (len << SCM_DYNSTACK_TAG_LEN_SHIFT); return ret; } static inline scm_t_dynstack_item_type scm_dynstack_tag_type (scm_t_bits tag) { return tag & SCM_DYNSTACK_TAG_TYPE_MASK; } static inline scm_t_bits scm_dynstack_tag_flags (scm_t_bits tag) { return tag & SCM_DYNSTACK_TAG_FLAGS_MASK; } static inline size_t scm_dynstack_tag_len (scm_t_bits tag) { return tag >> SCM_DYNSTACK_TAG_LEN_SHIFT; } static inline scm_t_bits * scm_dynstack_prev (scm_t_bits *top) { ptrdiff_t offset = scm_dynstack_prev_offset (top); return offset ? top - offset : NULL; } static inline scm_t_bits * scm_dynstack_next (scm_t_bits *top) { scm_t_bits tag = scm_dynstack_tag (top); return tag ? top + scm_dynstack_tag_len (tag) + SCM_DYNSTACK_HEADER_LEN : NULL; } static inline scm_t_bits * scm_dynstack_first (struct scm_dynstack *dynstack) { return dynstack->base + SCM_DYNSTACK_HEADER_LEN; } static inline size_t scm_dynstack_capacity (struct scm_dynstack *dynstack) { return dynstack->limit - dynstack->base; } static inline size_t scm_dynstack_space (struct scm_dynstack *dynstack) { return dynstack->limit - dynstack->top; } static inline size_t scm_dynstack_height (struct scm_dynstack *dynstack) { return dynstack->top - dynstack->base; } static inline int scm_dynstack_has_space (struct scm_dynstack *dynstack, size_t n) { return scm_dynstack_space (dynstack) >= n + SCM_DYNSTACK_HEADER_LEN; } typedef enum { SCM_F_DYNSTACK_FRAME_REWINDABLE = (1 << SCM_DYNSTACK_TAG_FLAGS_SHIFT) } scm_t_dynstack_frame_flags; typedef enum { SCM_F_DYNSTACK_WINDER_EXPLICIT = (1 << SCM_DYNSTACK_TAG_FLAGS_SHIFT), SCM_F_DYNSTACK_WINDER_MANAGED = (2 << SCM_DYNSTACK_TAG_FLAGS_SHIFT) } scm_t_dynstack_winder_flags; typedef enum { SCM_F_DYNSTACK_PROMPT_ESCAPE_ONLY = (1 << SCM_DYNSTACK_TAG_FLAGS_SHIFT) } scm_t_dynstack_prompt_flags; typedef void (*scm_t_guard) (void *); SCM_INTERNAL void scm_dynstack_init_for_thread (scm_t_dynstack *); /* Pushing and popping entries on the dynamic stack. */ SCM_INTERNAL void scm_dynstack_push_frame (scm_t_dynstack *, scm_t_dynstack_frame_flags); SCM_INTERNAL void scm_dynstack_push_rewinder (scm_t_dynstack *, scm_t_dynstack_winder_flags, scm_t_guard, void *); SCM_INTERNAL void scm_dynstack_push_unwinder (scm_t_dynstack *, scm_t_dynstack_winder_flags, scm_t_guard, void *); SCM_INTERNAL void scm_dynstack_push_fluid (scm_t_dynstack *, SCM fluid, SCM value, scm_thread *thread); SCM_INTERNAL void scm_dynstack_push_dynamic_state (scm_t_dynstack *, SCM, scm_t_dynamic_state *); SCM_INTERNAL void scm_dynstack_push_prompt (scm_t_dynstack *, scm_t_dynstack_prompt_flags, SCM key, ptrdiff_t fp_offset, ptrdiff_t sp_offset, uint32_t *vra, uint8_t *mra, jmp_buf *registers); SCM_INTERNAL void scm_dynstack_push_dynwind (scm_t_dynstack *, SCM enter, SCM leave); SCM_INTERNAL void scm_dynstack_pop (scm_t_dynstack *); /* Capturing, winding, and unwinding. */ SCM_INTERNAL scm_t_dynstack* scm_dynstack_capture_all (scm_t_dynstack *dynstack); SCM_INTERNAL scm_t_dynstack* scm_dynstack_capture (scm_t_dynstack *dynstack, scm_t_bits *item); SCM_INTERNAL void scm_dynstack_wind_1 (scm_t_dynstack *, scm_t_bits *); SCM_INTERNAL scm_t_bits scm_dynstack_unwind_1 (scm_t_dynstack *); SCM_INTERNAL void scm_dynstack_wind (scm_t_dynstack *, scm_t_bits *); SCM_INTERNAL void scm_dynstack_unwind (scm_t_dynstack *, scm_t_bits *); /* Miscellany. */ SCM_INTERNAL scm_t_bits* scm_dynstack_unwind_fork (scm_t_dynstack *, scm_t_dynstack *); SCM_INTERNAL void scm_dynstack_unwind_frame (scm_t_dynstack *); SCM_INTERNAL void scm_dynstack_unwind_fluid (scm_t_dynstack *dynstack, scm_thread *thread); SCM_INTERNAL void scm_dynstack_unwind_dynamic_state (scm_t_dynstack *dynstack, scm_t_dynamic_state *dynamic_state); SCM_INTERNAL scm_t_bits* scm_dynstack_find_prompt (scm_t_dynstack *, SCM, scm_t_dynstack_prompt_flags *, ptrdiff_t *, ptrdiff_t *, uint32_t **, uint8_t **, jmp_buf **); SCM_INTERNAL SCM scm_dynstack_find_old_fluid_value (scm_t_dynstack *, SCM, size_t, SCM); SCM_INTERNAL void scm_dynstack_relocate_prompts (scm_t_dynstack *, ptrdiff_t); SCM_INTERNAL void scm_dynstack_wind_prompt (scm_t_dynstack *, scm_t_bits *, ptrdiff_t, jmp_buf *); #endif /* SCM_DYNSTACK_H */