mirror of
https://gitlab.com/embeddable-common-lisp/ecl.git
synced 2026-01-18 23:32:17 -08:00
711 lines
18 KiB
C
711 lines
18 KiB
C
/* -*- Mode: C; c-basic-offset: 2; indent-tabs-mode: nil -*- */
|
|
/* vim: set filetype=c tabstop=2 shiftwidth=2 expandtab: */
|
|
|
|
/*
|
|
* stacks.d - binding/history/frame stacks
|
|
*
|
|
* Copyright (c) 1984 Taiichi Yuasa and Masami Hagiya
|
|
* Copyright (c) 1990 Giuseppe Attardi
|
|
* Copyright (c) 2001 Juan Jose Garcia Ripoll
|
|
*
|
|
* See file 'LICENSE' for the copyright details.
|
|
*
|
|
*/
|
|
|
|
#include <ecl/ecl.h>
|
|
#include <signal.h>
|
|
#include <string.h>
|
|
#ifdef HAVE_SYS_RESOURCE_H
|
|
# include <sys/time.h>
|
|
# include <sys/resource.h>
|
|
#endif
|
|
#include <ecl/internal.h>
|
|
|
|
/************************ C STACK ***************************/
|
|
|
|
static void
|
|
cs_set_size(cl_env_ptr env, cl_index new_size)
|
|
{
|
|
volatile char foo = 0;
|
|
cl_index margin = ecl_option_values[ECL_OPT_C_STACK_SAFETY_AREA];
|
|
env->cs_limit_size = new_size - (2*margin);
|
|
#ifdef ECL_DOWN_STACK
|
|
if (&foo > (env->cs_org - new_size) + 16) {
|
|
env->cs_limit = (env->cs_org - new_size) + (2*margin);
|
|
if (env->cs_limit < env->cs_barrier)
|
|
env->cs_barrier = env->cs_limit;
|
|
}
|
|
#else
|
|
if (&foo < (env->cs_org + new_size) - 16) {
|
|
env->cs_limit = (env->cs_org + new_size) - (2*margin);
|
|
if (env->cs_limit > env->cs_barrier)
|
|
env->cs_barrier = env->cs_limit;
|
|
}
|
|
#endif
|
|
else
|
|
ecl_internal_error("can't reset env->cs_limit.");
|
|
env->cs_size = new_size;
|
|
}
|
|
|
|
void
|
|
ecl_cs_overflow(void)
|
|
{
|
|
static const char *stack_overflow_msg =
|
|
"\n;;;\n;;; Stack overflow.\n"
|
|
";;; Jumping to the outermost toplevel prompt\n"
|
|
";;;\n\n";
|
|
cl_env_ptr env = ecl_process_env();
|
|
cl_index margin = ecl_option_values[ECL_OPT_C_STACK_SAFETY_AREA];
|
|
cl_index size = env->cs_size;
|
|
#ifdef ECL_DOWN_STACK
|
|
if (env->cs_limit > env->cs_org - size)
|
|
env->cs_limit -= margin;
|
|
#else
|
|
if (env->cs_limit < env->cs_org + size)
|
|
env->cs_limit += margin;
|
|
#endif
|
|
else
|
|
ecl_unrecoverable_error(env, stack_overflow_msg);
|
|
|
|
if (env->cs_max_size == (cl_index)0 || env->cs_size < env->cs_max_size)
|
|
si_serror(6, make_constant_base_string("Extend stack size"),
|
|
@'ext::stack-overflow',
|
|
@':size', ecl_make_fixnum(size),
|
|
@':type', @'ext::c-stack');
|
|
else
|
|
si_serror(6, ECL_NIL,
|
|
@'ext::stack-overflow',
|
|
@':size', ECL_NIL,
|
|
@':type', @'ext::c-stack');
|
|
size += size/2;
|
|
if (size > env->cs_max_size)
|
|
size = env->cs_max_size;
|
|
cs_set_size(env, size);
|
|
}
|
|
|
|
void
|
|
ecl_cs_set_org(cl_env_ptr env)
|
|
{
|
|
/* Rough estimate. Not very safe. We assume that cl_boot()
|
|
* is invoked from the main() routine of the program.
|
|
*/
|
|
env->cs_org = (char*)(&env);
|
|
env->cs_barrier = env->cs_org;
|
|
env->cs_max_size = 0;
|
|
#if defined(HAVE_SYS_RESOURCE_H) && defined(RLIMIT_STACK) && !defined(NACL)
|
|
{
|
|
struct rlimit rl;
|
|
cl_index size;
|
|
|
|
if (!getrlimit(RLIMIT_STACK, &rl) &&
|
|
( rl.rlim_cur != RLIM_INFINITY
|
|
|| rl.rlim_cur !=RLIM_SAVED_MAX
|
|
|| rl.rlim_cur != RLIM_SAVED_CUR) ) {
|
|
env->cs_max_size = rl.rlim_cur;
|
|
size = rl.rlim_cur / 2;
|
|
if (size < (cl_index)ecl_option_values[ECL_OPT_C_STACK_SIZE])
|
|
ecl_set_option(ECL_OPT_C_STACK_SIZE, size);
|
|
#ifdef ECL_DOWN_STACK
|
|
env->cs_barrier = (env->cs_org - rl.rlim_cur) - 1024;
|
|
#else
|
|
env->cs_barrier = (env->cs_org + rl.rlim_cur) + 1024;
|
|
#endif
|
|
}
|
|
}
|
|
#endif
|
|
cs_set_size(env, ecl_option_values[ECL_OPT_C_STACK_SIZE]);
|
|
}
|
|
|
|
|
|
/********************* BINDING STACK ************************/
|
|
|
|
void
|
|
ecl_bds_unwind_n(cl_env_ptr env, int n)
|
|
{
|
|
while (n--) ecl_bds_unwind1(env);
|
|
}
|
|
|
|
static void
|
|
ecl_bds_set_size(cl_env_ptr env, cl_index new_size)
|
|
{
|
|
ecl_bds_ptr old_org = env->bds_org;
|
|
cl_index limit = env->bds_top - old_org;
|
|
if (new_size <= limit) {
|
|
FEerror("Cannot shrink the binding stack below ~D.", 1,
|
|
ecl_make_unsigned_integer(limit));
|
|
} else {
|
|
cl_index margin = ecl_option_values[ECL_OPT_BIND_STACK_SAFETY_AREA];
|
|
ecl_bds_ptr org;
|
|
env->bds_limit_size = new_size - 2*margin;
|
|
org = ecl_alloc_atomic(new_size * sizeof(*org));
|
|
|
|
ecl_disable_interrupts_env(env);
|
|
memcpy(org, old_org, (limit + 1) * sizeof(*org));
|
|
env->bds_top = org + limit;
|
|
env->bds_org = org;
|
|
env->bds_limit = org + (new_size - 2*margin);
|
|
env->bds_size = new_size;
|
|
ecl_enable_interrupts_env(env);
|
|
|
|
ecl_dealloc(old_org);
|
|
}
|
|
}
|
|
|
|
ecl_bds_ptr
|
|
ecl_bds_overflow(void)
|
|
{
|
|
static const char *stack_overflow_msg =
|
|
"\n;;;\n;;; Binding stack overflow.\n"
|
|
";;; Jumping to the outermost toplevel prompt\n"
|
|
";;;\n\n";
|
|
cl_env_ptr env = ecl_process_env();
|
|
cl_index margin = ecl_option_values[ECL_OPT_BIND_STACK_SAFETY_AREA];
|
|
cl_index size = env->bds_size;
|
|
ecl_bds_ptr org = env->bds_org;
|
|
ecl_bds_ptr last = org + size;
|
|
if (env->bds_limit >= last) {
|
|
ecl_unrecoverable_error(env, stack_overflow_msg);
|
|
}
|
|
env->bds_limit += margin;
|
|
si_serror(6, make_constant_base_string("Extend stack size"),
|
|
@'ext::stack-overflow', @':size', ecl_make_fixnum(size),
|
|
@':type', @'ext::binding-stack');
|
|
ecl_bds_set_size(env, size + (size / 2));
|
|
return env->bds_top;
|
|
}
|
|
|
|
void
|
|
ecl_bds_unwind(cl_env_ptr env, cl_index new_bds_top_index)
|
|
{
|
|
ecl_bds_ptr new_bds_top = new_bds_top_index + env->bds_org;
|
|
ecl_bds_ptr bds = env->bds_top;
|
|
for (; bds > new_bds_top; bds--)
|
|
#ifdef ECL_THREADS
|
|
ecl_bds_unwind1(env);
|
|
#else
|
|
bds->symbol->symbol.value = bds->value;
|
|
#endif
|
|
env->bds_top = new_bds_top;
|
|
}
|
|
|
|
cl_index
|
|
ecl_progv(cl_env_ptr env, cl_object vars0, cl_object values0)
|
|
{
|
|
cl_object vars = vars0, values = values0;
|
|
cl_index n = env->bds_top - env->bds_org;
|
|
for (; LISTP(vars) && LISTP(values); vars = ECL_CONS_CDR(vars)) {
|
|
if (Null(vars)) {
|
|
return n;
|
|
} else {
|
|
cl_object var = ECL_CONS_CAR(vars);
|
|
if (Null(values)) {
|
|
ecl_bds_bind(env, var, OBJNULL);
|
|
} else {
|
|
ecl_bds_bind(env, var, ECL_CONS_CAR(values));
|
|
values = ECL_CONS_CDR(values);
|
|
}
|
|
}
|
|
}
|
|
FEerror("Wrong arguments to special form PROGV. Either~%"
|
|
"~A~%or~%~A~%are not proper lists",
|
|
2, vars0, values0);
|
|
}
|
|
|
|
static ecl_bds_ptr
|
|
get_bds_ptr(cl_object x)
|
|
{
|
|
if (ECL_FIXNUMP(x)) {
|
|
cl_env_ptr env = ecl_process_env();
|
|
ecl_bds_ptr p = env->bds_org + ecl_fixnum(x);
|
|
if (env->bds_org <= p && p <= env->bds_top)
|
|
return(p);
|
|
}
|
|
FEerror("~S is an illegal bds index.", 1, x);
|
|
}
|
|
|
|
cl_object
|
|
si_bds_top()
|
|
{
|
|
cl_env_ptr env = ecl_process_env();
|
|
@(return ecl_make_fixnum(env->bds_top - env->bds_org));
|
|
}
|
|
|
|
cl_object
|
|
si_bds_var(cl_object arg)
|
|
{
|
|
@(return get_bds_ptr(arg)->symbol);
|
|
}
|
|
|
|
cl_object
|
|
si_bds_val(cl_object arg)
|
|
{
|
|
cl_object v = get_bds_ptr(arg)->value;
|
|
@(return ((v == OBJNULL)? ECL_UNBOUND : v));
|
|
}
|
|
|
|
#ifdef ecl_bds_bind
|
|
# undef ecl_bds_bind
|
|
# undef ecl_bds_push
|
|
# undef ecl_bds_unwind1
|
|
#endif
|
|
#ifdef ecl_bds_read
|
|
# undef ecl_bds_read
|
|
# undef ecl_bds_set
|
|
# undef ecl_bds_ref
|
|
#endif
|
|
|
|
#ifdef ECL_THREADS
|
|
static cl_index
|
|
ecl_new_binding_index(cl_env_ptr env, cl_object symbol)
|
|
{
|
|
cl_object pool;
|
|
cl_index new_index = symbol->symbol.binding;
|
|
if (new_index == ECL_MISSING_SPECIAL_BINDING) {
|
|
pool = ecl_atomic_pop(&cl_core.reused_indices);
|
|
if (!Null(pool)) {
|
|
new_index = ecl_fixnum(ECL_CONS_CAR(pool));
|
|
} else {
|
|
new_index = ecl_atomic_index_incf(&cl_core.last_var_index);
|
|
}
|
|
symbol->symbol.binding = new_index;
|
|
symbol->symbol.dynamic |= 1;
|
|
}
|
|
si_set_finalizer(symbol, ECL_T);
|
|
return new_index;
|
|
}
|
|
|
|
static cl_object
|
|
ecl_extend_bindings_array(cl_object vector)
|
|
{
|
|
cl_index new_size = cl_core.last_var_index * 1.25;
|
|
cl_object new_vector = si_make_vector(ECL_T, ecl_make_fixnum(new_size), ECL_NIL,
|
|
ECL_NIL, ECL_NIL, ECL_NIL);
|
|
si_fill_array_with_elt(new_vector, ECL_NO_TL_BINDING, ecl_make_fixnum(0), ECL_NIL);
|
|
ecl_copy_subarray(new_vector, 0, vector, 0, vector->vector.dim);
|
|
return new_vector;
|
|
}
|
|
|
|
static cl_index
|
|
invalid_or_too_large_binding_index(cl_env_ptr env, cl_object s)
|
|
{
|
|
cl_index index = s->symbol.binding;
|
|
if (index == ECL_MISSING_SPECIAL_BINDING) {
|
|
index = ecl_new_binding_index(env, s);
|
|
}
|
|
if (index >= env->thread_local_bindings_size) {
|
|
cl_object vector = env->bindings_array;
|
|
env->bindings_array = vector = ecl_extend_bindings_array(vector);
|
|
env->thread_local_bindings_size = vector->vector.dim;
|
|
env->thread_local_bindings = vector->vector.self.t;
|
|
}
|
|
return index;
|
|
}
|
|
#endif /* ECL_THREADS */
|
|
|
|
/*
|
|
* The following routines must match the inline forms in stacks.h
|
|
*/
|
|
void
|
|
ecl_bds_bind(cl_env_ptr env, cl_object s, cl_object v)
|
|
{
|
|
#ifdef ECL_THREADS
|
|
cl_object *location;
|
|
ecl_bds_ptr slot;
|
|
cl_index index = s->symbol.binding;
|
|
if (index >= env->thread_local_bindings_size) {
|
|
index = invalid_or_too_large_binding_index(env,s);
|
|
}
|
|
location = env->thread_local_bindings + index;
|
|
slot = ++env->bds_top;
|
|
if (slot >= env->bds_limit) slot = ecl_bds_overflow();
|
|
slot->symbol = s;
|
|
slot->value = *location;
|
|
*location = v;
|
|
#else
|
|
ecl_bds_check(env);
|
|
(++(env->bds_top))->symbol = s;
|
|
env->bds_top->value = s->symbol.value; \
|
|
s->symbol.value = v;
|
|
#endif
|
|
}
|
|
|
|
void
|
|
ecl_bds_push(cl_env_ptr env, cl_object s)
|
|
{
|
|
#ifdef ECL_THREADS
|
|
cl_object *location;
|
|
ecl_bds_ptr slot;
|
|
cl_index index = s->symbol.binding;
|
|
if (index >= env->thread_local_bindings_size) {
|
|
index = invalid_or_too_large_binding_index(env,s);
|
|
}
|
|
location = env->thread_local_bindings + index;
|
|
slot = ++env->bds_top;
|
|
if (slot >= env->bds_limit) slot = ecl_bds_overflow();
|
|
slot->symbol = s;
|
|
slot->value = *location;
|
|
if (*location == ECL_NO_TL_BINDING) *location = s->symbol.value;
|
|
#else
|
|
ecl_bds_check(env);
|
|
(++(env->bds_top))->symbol = s;
|
|
env->bds_top->value = s->symbol.value;
|
|
#endif
|
|
}
|
|
|
|
void
|
|
ecl_bds_unwind1(cl_env_ptr env)
|
|
{
|
|
ecl_bds_ptr slot = env->bds_top--;
|
|
cl_object s = slot->symbol;
|
|
#ifdef ECL_THREADS
|
|
cl_object *location = env->thread_local_bindings + s->symbol.binding;
|
|
*location = slot->value;
|
|
#else
|
|
s->symbol.value = slot->value;
|
|
#endif
|
|
}
|
|
|
|
#ifdef ECL_THREADS
|
|
cl_object
|
|
ecl_bds_read(cl_env_ptr env, cl_object s)
|
|
{
|
|
cl_index index = s->symbol.binding;
|
|
if (index < env->thread_local_bindings_size) {
|
|
cl_object x = env->thread_local_bindings[index];
|
|
if (x != ECL_NO_TL_BINDING) return x;
|
|
}
|
|
return s->symbol.value;
|
|
}
|
|
|
|
cl_object *
|
|
ecl_bds_ref(cl_env_ptr env, cl_object s)
|
|
{
|
|
cl_index index = s->symbol.binding;
|
|
if (index < env->thread_local_bindings_size) {
|
|
cl_object *location = env->thread_local_bindings + index;
|
|
if (*location != ECL_NO_TL_BINDING)
|
|
return location;
|
|
}
|
|
return &(s->symbol.value);
|
|
}
|
|
|
|
cl_object
|
|
ecl_bds_set(cl_env_ptr env, cl_object s, cl_object value)
|
|
{
|
|
return *ecl_bds_ref(env, s) = value;
|
|
}
|
|
#endif /* ECL_THREADS */
|
|
|
|
/******************** INVOCATION STACK **********************/
|
|
|
|
static cl_object
|
|
ihs_function_name(cl_object x)
|
|
{
|
|
cl_object y;
|
|
|
|
switch (ecl_t_of(x)) {
|
|
case t_symbol:
|
|
return(x);
|
|
|
|
case t_bclosure:
|
|
x = x->bclosure.code;
|
|
|
|
case t_bytecodes:
|
|
y = x->bytecodes.name;
|
|
if (Null(y))
|
|
return(@'lambda');
|
|
else
|
|
return y;
|
|
|
|
case t_cfun:
|
|
case t_cfunfixed:
|
|
return(x->cfun.name);
|
|
|
|
default:
|
|
return(ECL_NIL);
|
|
}
|
|
}
|
|
|
|
static ecl_ihs_ptr
|
|
get_ihs_ptr(cl_index n)
|
|
{
|
|
cl_env_ptr env = ecl_process_env();
|
|
ecl_ihs_ptr p = env->ihs_top;
|
|
if (n > p->index)
|
|
FEerror("~D is an illegal IHS index.", 1, ecl_make_fixnum(n));
|
|
while (n < p->index)
|
|
p = p->next;
|
|
return p;
|
|
}
|
|
|
|
cl_object
|
|
si_ihs_top(void)
|
|
{
|
|
cl_env_ptr env = ecl_process_env();
|
|
@(return ecl_make_fixnum(env->ihs_top->index));
|
|
}
|
|
|
|
cl_object
|
|
si_ihs_prev(cl_object x)
|
|
{
|
|
@(return cl_1M(x));
|
|
}
|
|
|
|
cl_object
|
|
si_ihs_next(cl_object x)
|
|
{
|
|
@(return cl_1P(x));
|
|
}
|
|
|
|
cl_object
|
|
si_ihs_bds(cl_object arg)
|
|
{
|
|
@(return ecl_make_fixnum(get_ihs_ptr(ecl_to_size(arg))->bds));
|
|
}
|
|
|
|
cl_object
|
|
si_ihs_fun(cl_object arg)
|
|
{
|
|
@(return get_ihs_ptr(ecl_to_size(arg))->function);
|
|
}
|
|
|
|
cl_object
|
|
si_ihs_env(cl_object arg)
|
|
{
|
|
@(return get_ihs_ptr(ecl_to_size(arg))->lex_env);
|
|
}
|
|
|
|
/********************** FRAME STACK *************************/
|
|
|
|
static void
|
|
frs_set_size(cl_env_ptr env, cl_index new_size)
|
|
{
|
|
ecl_frame_ptr old_org = env->frs_org;
|
|
cl_index limit = env->frs_top - old_org;
|
|
if (new_size <= limit) {
|
|
FEerror("Cannot shrink frame stack below ~D.", 1,
|
|
ecl_make_unsigned_integer(limit));
|
|
} else {
|
|
cl_index margin = ecl_option_values[ECL_OPT_FRAME_STACK_SAFETY_AREA];
|
|
ecl_frame_ptr org;
|
|
env->frs_limit_size = new_size - 2*margin;
|
|
org = ecl_alloc_atomic(new_size * sizeof(*org));
|
|
|
|
ecl_disable_interrupts_env(env);
|
|
memcpy(org, old_org, (limit + 1) * sizeof(*org));
|
|
env->frs_top = org + limit;
|
|
env->frs_org = org;
|
|
env->frs_limit = org + (new_size - 2*margin);
|
|
env->frs_size = new_size;
|
|
ecl_enable_interrupts_env(env);
|
|
|
|
ecl_dealloc(old_org);
|
|
}
|
|
}
|
|
|
|
static void
|
|
frs_overflow(void) /* used as condition in list.d */
|
|
{
|
|
static const char *stack_overflow_msg =
|
|
"\n;;;\n;;; Frame stack overflow.\n"
|
|
";;; Jumping to the outermost toplevel prompt\n"
|
|
";;;\n\n";
|
|
cl_env_ptr env = ecl_process_env();
|
|
cl_index margin = ecl_option_values[ECL_OPT_FRAME_STACK_SAFETY_AREA];
|
|
cl_index size = env->frs_size;
|
|
ecl_frame_ptr org = env->frs_org;
|
|
ecl_frame_ptr last = org + size;
|
|
if (env->frs_limit >= last) {
|
|
ecl_unrecoverable_error(env, stack_overflow_msg);
|
|
}
|
|
env->frs_limit += margin;
|
|
si_serror(6, make_constant_base_string("Extend stack size"),
|
|
@'ext::stack-overflow', @':size', ecl_make_fixnum(size),
|
|
@':type', @'ext::frame-stack');
|
|
frs_set_size(env, size + size / 2);
|
|
}
|
|
|
|
ecl_frame_ptr
|
|
_ecl_frs_push(register cl_env_ptr env, register cl_object val)
|
|
{
|
|
ecl_frame_ptr output = ++env->frs_top;
|
|
if (output >= env->frs_limit) {
|
|
frs_overflow();
|
|
output = env->frs_top;
|
|
}
|
|
output->frs_bds_top_index = env->bds_top - env->bds_org;
|
|
output->frs_val = val;
|
|
output->frs_ihs = env->ihs_top;
|
|
output->frs_sp = ECL_STACK_INDEX(env);
|
|
return output;
|
|
}
|
|
|
|
void
|
|
ecl_unwind(cl_env_ptr env, ecl_frame_ptr fr)
|
|
{
|
|
env->nlj_fr = fr;
|
|
while (env->frs_top != fr && env->frs_top->frs_val != ECL_PROTECT_TAG)
|
|
--env->frs_top;
|
|
env->ihs_top = env->frs_top->frs_ihs;
|
|
ecl_bds_unwind(env, env->frs_top->frs_bds_top_index);
|
|
ECL_STACK_SET_INDEX(env, env->frs_top->frs_sp);
|
|
ecl_longjmp(env->frs_top->frs_jmpbuf, 1);
|
|
/* never reached */
|
|
}
|
|
|
|
ecl_frame_ptr
|
|
frs_sch (cl_object frame_id)
|
|
{
|
|
cl_env_ptr env = ecl_process_env();
|
|
ecl_frame_ptr top;
|
|
for (top = env->frs_top; top >= env->frs_org; top--)
|
|
if (top->frs_val == frame_id)
|
|
return(top);
|
|
return(NULL);
|
|
}
|
|
|
|
static ecl_frame_ptr
|
|
get_frame_ptr(cl_object x)
|
|
{
|
|
if (ECL_FIXNUMP(x)) {
|
|
cl_env_ptr env = ecl_process_env();
|
|
ecl_frame_ptr p = env->frs_org + ecl_fixnum(x);
|
|
if (env->frs_org <= p && p <= env->frs_top)
|
|
return p;
|
|
}
|
|
FEerror("~S is an illegal frs index.", 1, x);
|
|
}
|
|
|
|
cl_object
|
|
si_frs_top()
|
|
{
|
|
cl_env_ptr env = ecl_process_env();
|
|
@(return ecl_make_fixnum(env->frs_top - env->frs_org));
|
|
}
|
|
|
|
cl_object
|
|
si_frs_bds(cl_object arg)
|
|
{
|
|
@(return ecl_make_fixnum(get_frame_ptr(arg)->frs_bds_top_index));
|
|
}
|
|
|
|
cl_object
|
|
si_frs_tag(cl_object arg)
|
|
{
|
|
@(return get_frame_ptr(arg)->frs_val);
|
|
}
|
|
|
|
cl_object
|
|
si_frs_ihs(cl_object arg)
|
|
{
|
|
@(return ecl_make_fixnum(get_frame_ptr(arg)->frs_ihs->index));
|
|
}
|
|
|
|
cl_object
|
|
si_sch_frs_base(cl_object fr, cl_object ihs)
|
|
{
|
|
cl_env_ptr env = ecl_process_env();
|
|
ecl_frame_ptr x;
|
|
cl_index y = ecl_to_size(ihs);
|
|
for (x = get_frame_ptr(fr);
|
|
x <= env->frs_top && x->frs_ihs->index < y;
|
|
x++);
|
|
@(return ((x > env->frs_top) ? ECL_NIL : ecl_make_fixnum(x - env->frs_org)));
|
|
}
|
|
|
|
/********************* INITIALIZATION ***********************/
|
|
|
|
cl_object
|
|
si_set_limit(cl_object type, cl_object limit)
|
|
{
|
|
cl_env_ptr env = ecl_process_env();
|
|
cl_index margin;
|
|
if (type == @'ext::frame-stack') {
|
|
cl_index the_size = ecl_to_size(limit);
|
|
margin = ecl_option_values[ECL_OPT_FRAME_STACK_SAFETY_AREA];
|
|
frs_set_size(env, the_size + 2*margin);
|
|
} else if (type == @'ext::binding-stack') {
|
|
cl_index the_size = ecl_to_size(limit);
|
|
margin = ecl_option_values[ECL_OPT_BIND_STACK_SAFETY_AREA];
|
|
ecl_bds_set_size(env, the_size + 2*margin);
|
|
} else if (type == @'ext::c-stack') {
|
|
cl_index the_size = ecl_to_size(limit);
|
|
margin = ecl_option_values[ECL_OPT_C_STACK_SAFETY_AREA];
|
|
cs_set_size(env, the_size + 2*margin);
|
|
} else if (type == @'ext::lisp-stack') {
|
|
cl_index the_size = ecl_to_size(limit);
|
|
ecl_stack_set_size(env, the_size);
|
|
} else {
|
|
/*
|
|
* size_t can be larger than cl_index, and ecl_to_size()
|
|
* creates a fixnum which is too small for size_t on 32-bit.
|
|
*/
|
|
size_t the_size = (size_t)ecl_to_ulong(limit);
|
|
_ecl_set_max_heap_size(the_size);
|
|
}
|
|
|
|
return si_get_limit(type);
|
|
}
|
|
|
|
cl_object
|
|
si_get_limit(cl_object type)
|
|
{
|
|
cl_env_ptr env = ecl_process_env();
|
|
cl_index output;
|
|
if (type == @'ext::frame-stack')
|
|
output = env->frs_limit_size;
|
|
else if (type == @'ext::binding-stack')
|
|
output = env->bds_limit_size;
|
|
else if (type == @'ext::c-stack')
|
|
output = env->cs_limit_size;
|
|
else if (type == @'ext::lisp-stack')
|
|
output = env->stack_limit_size;
|
|
else {
|
|
/* size_t can be larger than cl_index */
|
|
@(return ecl_make_unsigned_integer(cl_core.max_heap_size));
|
|
}
|
|
|
|
@(return ecl_make_unsigned_integer(output));
|
|
}
|
|
|
|
cl_object
|
|
si_reset_margin(cl_object type)
|
|
{
|
|
cl_env_ptr env = ecl_process_env();
|
|
if (type == @'ext::frame-stack')
|
|
frs_set_size(env, env->frs_size);
|
|
else if (type == @'ext::binding-stack')
|
|
ecl_bds_set_size(env, env->bds_size);
|
|
else if (type == @'ext::c-stack')
|
|
cs_set_size(env, env->cs_size);
|
|
else
|
|
return ECL_NIL;
|
|
|
|
return ECL_T;
|
|
}
|
|
|
|
void
|
|
init_stacks(cl_env_ptr env)
|
|
{
|
|
static struct ecl_ihs_frame ihs_org = { NULL, NULL, NULL, 0};
|
|
cl_index size, margin;
|
|
|
|
margin = ecl_option_values[ECL_OPT_FRAME_STACK_SAFETY_AREA];
|
|
size = ecl_option_values[ECL_OPT_FRAME_STACK_SIZE] + 2 * margin;
|
|
env->frs_size = size;
|
|
env->frs_org = (ecl_frame_ptr)ecl_alloc_atomic(size * sizeof(*env->frs_org));
|
|
env->frs_top = env->frs_org-1;
|
|
env->frs_limit = &env->frs_org[size - 2*margin];
|
|
|
|
margin = ecl_option_values[ECL_OPT_BIND_STACK_SAFETY_AREA];
|
|
size = ecl_option_values[ECL_OPT_BIND_STACK_SIZE] + 2 * margin;
|
|
env->bds_size = size;
|
|
env->bds_org = (ecl_bds_ptr)ecl_alloc_atomic(size * sizeof(*env->bds_org));
|
|
env->bds_top = env->bds_org-1;
|
|
env->bds_limit = &env->bds_org[size - 2*margin];
|
|
|
|
env->ihs_top = &ihs_org;
|
|
ihs_org.function = ECL_NIL;
|
|
ihs_org.lex_env = ECL_NIL;
|
|
ihs_org.index = 0;
|
|
}
|