mirror of
https://gitlab.com/embeddable-common-lisp/ecl.git
synced 2026-03-13 08:20:31 -07:00
ecl_alloc_object, ecl_free_object ecl_alloc, ecl_alloc_manual, ecl_alloc_atomic, ecl_dealloc Moreover move all top-level ops to memory.d so they are not reliant on mem_gc. The stubbed allocator uses manual memory managament for all ops. [wip] because we should adjust ecl_make_stack too
565 lines
15 KiB
C
565 lines
15 KiB
C
/* -*- Mode: C; c-basic-offset: 2; indent-tabs-mode: nil -*- */
|
|
/* vim: set filetype=c tabstop=2 shiftwidth=2 expandtab: */
|
|
|
|
/*
|
|
* instance.d - CLOS interface
|
|
*
|
|
* Copyright (c) 1990 Giuseppe Attardi
|
|
* Copyright (c) 2001 Juan Jose Garcia Ripoll
|
|
*
|
|
* See file 'LICENSE' for the copyright details.
|
|
*
|
|
*/
|
|
|
|
#include <string.h>
|
|
#include <ecl/ecl.h>
|
|
#include <ecl/internal.h>
|
|
#include <ecl/ecl-inl.h>
|
|
|
|
static cl_index stamp = 0;
|
|
cl_index ecl_next_stamp() {
|
|
#if ECL_THREADS
|
|
return AO_fetch_and_add((AO_t*)&stamp, 1) + 1;
|
|
#else
|
|
return ++stamp;
|
|
#endif
|
|
}
|
|
|
|
cl_object
|
|
ecl_alloc_instance(cl_index slots)
|
|
{
|
|
cl_object i;
|
|
i = ecl_alloc_object(t_instance);
|
|
i->instance.slots = (cl_object *)ecl_alloc(sizeof(cl_object) * slots);
|
|
i->instance.length = slots;
|
|
i->instance.isgf = ECL_NOT_FUNCALLABLE;
|
|
i->instance.entry = FEnot_funcallable_vararg;
|
|
i->instance.slotds = ECL_UNBOUND;
|
|
return i;
|
|
}
|
|
|
|
cl_object
|
|
ecl_allocate_instance(cl_object clas, cl_index size)
|
|
{
|
|
cl_object x = ecl_alloc_instance(size);
|
|
cl_index i;
|
|
ECL_CLASS_OF(x) = clas;
|
|
for (i = 0; i < size; i++)
|
|
x->instance.slots[i] = ECL_UNBOUND;
|
|
return x;
|
|
}
|
|
|
|
cl_object
|
|
si_allocate_raw_instance(cl_object orig, cl_object clas, cl_object size)
|
|
{
|
|
cl_object output = ecl_allocate_instance(clas, ecl_to_size(size));
|
|
if (orig == ECL_NIL) {
|
|
orig = output;
|
|
} else {
|
|
orig->instance.clas = clas;
|
|
orig->instance.length = output->instance.length;
|
|
orig->instance.slots = output->instance.slots;
|
|
}
|
|
@(return orig);
|
|
}
|
|
|
|
cl_object
|
|
si_instance_obsolete_p(cl_object x)
|
|
{
|
|
return (x->instance.stamp != ECL_CLASS_OF(x)->instance.class_stamp)
|
|
? ECL_T : ECL_NIL;
|
|
}
|
|
|
|
cl_object
|
|
si_instance_new_stamp(cl_object x)
|
|
{
|
|
x->instance.class_stamp = ecl_next_stamp();
|
|
return ecl_make_fixnum(x->instance.class_stamp);
|
|
}
|
|
|
|
cl_object
|
|
si_instance_get_stamp(cl_object x)
|
|
{
|
|
cl_object a = ecl_make_fixnum(x->instance.stamp);
|
|
cl_object b = ecl_make_fixnum(ECL_CLASS_OF(x)->instance.class_stamp);
|
|
@(return a b);
|
|
}
|
|
|
|
cl_object
|
|
si_instance_slotds(cl_object x)
|
|
{
|
|
@(return x->instance.slotds);
|
|
}
|
|
|
|
cl_object
|
|
si_instance_sig_set(cl_object x)
|
|
{
|
|
x->instance.stamp = ECL_CLASS_OF(x)->instance.class_stamp;
|
|
@(return (x->instance.slotds = ECL_CLASS_SLOTS(ECL_CLASS_OF(x))));
|
|
}
|
|
|
|
cl_object
|
|
si_instance_class(cl_object x)
|
|
{
|
|
unlikely_if (!ECL_INSTANCEP(x)) {
|
|
FEwrong_type_only_arg(@[class-of], x, @[ext::instance]);
|
|
}
|
|
@(return ECL_CLASS_OF(x));
|
|
}
|
|
|
|
cl_object
|
|
si_instance_class_set(cl_object x, cl_object y)
|
|
{
|
|
unlikely_if (!ECL_INSTANCEP(x)) {
|
|
FEwrong_type_nth_arg(@[si::instance-class-set], 1, x, @[ext::instance]);
|
|
}
|
|
unlikely_if (!ECL_INSTANCEP(y)) {
|
|
FEwrong_type_nth_arg(@[si::instance-class-set], 2, y, @[ext::instance]);
|
|
}
|
|
ECL_CLASS_OF(x) = y;
|
|
@(return x);
|
|
}
|
|
|
|
cl_object
|
|
ecl_instance_ref(cl_object x, cl_fixnum i)
|
|
{
|
|
unlikely_if (!ECL_INSTANCEP(x)) {
|
|
FEwrong_type_nth_arg(@[si::instance-ref], 1, x, @[ext::instance]);
|
|
}
|
|
unlikely_if (i < 0 || i >= (cl_fixnum)x->instance.length) {
|
|
FEtype_error_index(x, i);
|
|
}
|
|
return(x->instance.slots[i]);
|
|
}
|
|
|
|
cl_object
|
|
si_instance_ref(cl_object x, cl_object index)
|
|
{
|
|
cl_fixnum i;
|
|
|
|
unlikely_if (!ECL_INSTANCEP(x)) {
|
|
FEwrong_type_nth_arg(@[si::instance-ref], 1, x, @[ext::instance]);
|
|
}
|
|
unlikely_if (!ECL_FIXNUMP(index)) {
|
|
FEwrong_type_nth_arg(@[si::instance-ref], 2, index, @[fixnum]);
|
|
}
|
|
i = ecl_fixnum(index);
|
|
unlikely_if (i < 0 || i >= (cl_fixnum)x->instance.length) {
|
|
FEtype_error_index(x, i);
|
|
}
|
|
@(return x->instance.slots[i]);
|
|
}
|
|
|
|
cl_object
|
|
clos_safe_instance_ref(cl_object x, cl_object index)
|
|
{
|
|
cl_fixnum i;
|
|
|
|
unlikely_if (!ECL_INSTANCEP(x)) {
|
|
FEwrong_type_nth_arg(@[si::instance-ref], 1, x, @[ext::instance]);
|
|
}
|
|
unlikely_if (!ECL_FIXNUMP(index)) {
|
|
FEwrong_type_nth_arg(@[si::instance-ref], 2, index, @[fixnum]);
|
|
}
|
|
i = ecl_fixnum(index);
|
|
unlikely_if (i < 0 || i >= x->instance.length) {
|
|
FEtype_error_index(x, i);
|
|
}
|
|
x = x->instance.slots[i];
|
|
unlikely_if (x == ECL_UNBOUND) {
|
|
x = _ecl_funcall4(@'slot-unbound', ECL_NIL, x, index);
|
|
}
|
|
@(return x);
|
|
}
|
|
|
|
cl_object
|
|
ecl_instance_set(cl_object x, cl_fixnum i, cl_object v)
|
|
{
|
|
unlikely_if (!ECL_INSTANCEP(x)) {
|
|
FEwrong_type_nth_arg(@[si::instance-set], 1, x, @[ext::instance]);
|
|
}
|
|
unlikely_if (i >= x->instance.length || i < 0) {
|
|
FEtype_error_index(x, i);
|
|
}
|
|
x->instance.slots[i] = v;
|
|
return(v);
|
|
}
|
|
|
|
cl_object
|
|
si_instance_set(cl_object x, cl_object index, cl_object value)
|
|
{
|
|
cl_fixnum i;
|
|
|
|
unlikely_if (!ECL_INSTANCEP(x)) {
|
|
FEwrong_type_nth_arg(@[si::instance-set], 1, x, @[ext::instance]);
|
|
}
|
|
unlikely_if (!ECL_FIXNUMP(index)) {
|
|
FEwrong_type_nth_arg(@[si::instance-set], 2, index, @[fixnum]);
|
|
}
|
|
i = ecl_fixnum(index);
|
|
unlikely_if (i >= (cl_fixnum)x->instance.length || i < 0) {
|
|
FEtype_error_index(x, i);
|
|
}
|
|
x->instance.slots[i] = value;
|
|
@(return value);
|
|
}
|
|
|
|
#ifdef ECL_THREADS
|
|
cl_object
|
|
ecl_compare_and_swap_instance(cl_object x, cl_fixnum i, cl_object old, cl_object new)
|
|
{
|
|
unlikely_if (!ECL_INSTANCEP(x)) {
|
|
FEwrong_type_nth_arg(@[mp::compare-and-swap-instance], 1, x, @[ext::instance]);
|
|
}
|
|
unlikely_if (i >= x->instance.length || i < 0) {
|
|
FEtype_error_index(x, i);
|
|
}
|
|
return ecl_compare_and_swap(x->instance.slots + i, old, new);
|
|
}
|
|
|
|
cl_object
|
|
mp_compare_and_swap_instance(cl_object x, cl_object index, cl_object old, cl_object new)
|
|
{
|
|
cl_fixnum i;
|
|
|
|
unlikely_if (!ECL_INSTANCEP(x)) {
|
|
FEwrong_type_nth_arg(@[mp::compare-and-swap-instance], 1, x, @[ext::instance]);
|
|
}
|
|
unlikely_if (!ECL_FIXNUMP(index)) {
|
|
FEwrong_type_nth_arg(@[mp::compare-and-swap-instance], 2, index, @[fixnum]);
|
|
}
|
|
i = ecl_fixnum(index);
|
|
unlikely_if (i >= (cl_fixnum)x->instance.length || i < 0) {
|
|
FEtype_error_index(x, i);
|
|
}
|
|
return ecl_compare_and_swap(x->instance.slots + i, old, new);
|
|
}
|
|
|
|
cl_object
|
|
ecl_atomic_incf_instance(cl_object x, cl_fixnum i, cl_object increment)
|
|
{
|
|
unlikely_if (!ECL_INSTANCEP(x)) {
|
|
FEwrong_type_nth_arg(@[mp::atomic-incf-instance], 1, x, @[ext::instance]);
|
|
}
|
|
unlikely_if (i >= x->instance.length || i < 0) {
|
|
FEtype_error_index(x, i);
|
|
}
|
|
return ecl_atomic_incf(x->instance.slots + i, increment);
|
|
}
|
|
|
|
cl_object
|
|
mp_atomic_incf_instance(cl_object x, cl_object index, cl_object increment)
|
|
{
|
|
cl_fixnum i;
|
|
|
|
unlikely_if (!ECL_INSTANCEP(x)) {
|
|
FEwrong_type_nth_arg(@[mp::atomic-incf-instance], 1, x, @[ext::instance]);
|
|
}
|
|
unlikely_if (!ECL_FIXNUMP(index)) {
|
|
FEwrong_type_nth_arg(@[mp::atomic-incf-instance], 2, index, @[fixnum]);
|
|
}
|
|
i = ecl_fixnum(index);
|
|
unlikely_if (i >= (cl_fixnum)x->instance.length || i < 0) {
|
|
FEtype_error_index(x, i);
|
|
}
|
|
return ecl_atomic_incf(x->instance.slots + i, increment);
|
|
}
|
|
#endif /* ECL_THREADS */
|
|
|
|
cl_object
|
|
si_instancep(cl_object x)
|
|
{
|
|
@(return (ECL_INSTANCEP(x) ? ecl_make_fixnum(x->instance.length) : ECL_NIL));
|
|
}
|
|
|
|
cl_object
|
|
si_unbound()
|
|
{
|
|
/* Returns an object that cannot be read or written and which
|
|
is used to represent an unitialized slot */
|
|
@(return ECL_UNBOUND);
|
|
}
|
|
|
|
cl_object
|
|
si_sl_boundp(cl_object x)
|
|
{
|
|
@(return ((x == ECL_UNBOUND) ? ECL_NIL : ECL_T));
|
|
}
|
|
|
|
cl_object
|
|
si_sl_makunbound(cl_object x, cl_object index)
|
|
{
|
|
cl_fixnum i;
|
|
|
|
unlikely_if (!ECL_INSTANCEP(x)) {
|
|
FEwrong_type_nth_arg(@[si::sl-makunbound], 1, x, @[ext::instance]);
|
|
}
|
|
unlikely_if (!ECL_FIXNUMP(index)) {
|
|
FEwrong_type_nth_arg(@[si::sl-makunbound], 2, index, @[fixnum]);
|
|
}
|
|
i = ecl_fixnum(index);
|
|
unlikely_if (i >= x->instance.length || i < 0) {
|
|
FEtype_error_index(x, i);
|
|
}
|
|
x->instance.slots[i] = ECL_UNBOUND;
|
|
@(return x);
|
|
}
|
|
|
|
cl_object
|
|
si_copy_instance(cl_object x)
|
|
{
|
|
cl_object y;
|
|
|
|
unlikely_if (!ECL_INSTANCEP(x)) {
|
|
FEwrong_type_nth_arg(@[si::copy-instance], 1, x, @[ext::instance]);
|
|
}
|
|
y = ecl_allocate_instance(x->instance.clas, x->instance.length);
|
|
y->instance.slotds = x->instance.slotds;
|
|
y->instance.stamp = x->instance.stamp;
|
|
y->instance.class_stamp = x->instance.class_stamp;
|
|
memcpy(y->instance.slots, x->instance.slots,
|
|
x->instance.length * sizeof(cl_object));
|
|
@(return y);
|
|
}
|
|
|
|
@(defun find-class (name &optional (errorp ECL_T) env)
|
|
cl_object class = ECL_NIL, hash;
|
|
@
|
|
if (ECL_CONSP(env)) {
|
|
env = ecl_car(env);
|
|
loop_for_in(env) {
|
|
if (ECL_CONSP(env)) {
|
|
cl_object record = ECL_CONS_CAR(env);
|
|
if (ECL_CONSP(record) && ecl_car(record) == @':type'
|
|
&& ecl_cadr(record) == name && ECL_INSTANCEP(ecl_caddr(record))) {
|
|
class = ecl_caddr(record);
|
|
break;
|
|
}
|
|
}
|
|
} end_loop_for_in;
|
|
}
|
|
if (class == ECL_NIL) {
|
|
hash = ECL_SYM_VAL(the_env, @'si::*class-name-hash-table*');
|
|
class = ecl_gethash_safe(name, hash, ECL_NIL);
|
|
}
|
|
if (class == ECL_NIL && errorp != ECL_NIL) {
|
|
FEerror("No class named ~S.", 1, name);
|
|
}
|
|
@(return class);
|
|
@)
|
|
|
|
cl_object
|
|
ecl_slot_value(cl_object x, const char *slot)
|
|
{
|
|
cl_object slot_name = ecl_read_from_cstring(slot);
|
|
return funcall(3, @'slot-value', x, slot_name);
|
|
}
|
|
|
|
cl_object
|
|
ecl_slot_value_set(cl_object x, const char *slot, cl_object value)
|
|
{
|
|
cl_object slot_name = ecl_read_from_cstring(slot);
|
|
cl_object slot_setter = ecl_read_from_cstring("(SETF SLOT-VALUE)");
|
|
return funcall(4, ecl_fdefinition(slot_setter), value, x, slot_name);
|
|
}
|
|
|
|
/**********************************************************************
|
|
* IMPORTANT: THE FOLLOWING LIST IS LINKED TO src/clos/hierarchy.lsp
|
|
**********************************************************************/
|
|
enum ecl_built_in_classes {
|
|
ECL_BUILTIN_T = 0,
|
|
ECL_BUILTIN_SEQUENCE,
|
|
ECL_BUILTIN_LIST,
|
|
ECL_BUILTIN_CONS,
|
|
ECL_BUILTIN_ARRAY,
|
|
ECL_BUILTIN_VECTOR,
|
|
ECL_BUILTIN_STRING,
|
|
ECL_BUILTIN_BASE_STRING,
|
|
ECL_BUILTIN_BIT_VECTOR,
|
|
ECL_BUILTIN_STREAM,
|
|
ECL_BUILTIN_ANSI_STREAM,
|
|
ECL_BUILTIN_FILE_STREAM,
|
|
ECL_BUILTIN_ECHO_STREAM,
|
|
ECL_BUILTIN_STRING_STREAM,
|
|
ECL_BUILTIN_TWO_WAY_STREAM,
|
|
ECL_BUILTIN_SYNONYM_STREAM,
|
|
ECL_BUILTIN_BROADCAST_STREAM,
|
|
ECL_BUILTIN_CONCATENATED_STREAM,
|
|
ECL_BUILTIN_SEQUENCE_STREAM,
|
|
ECL_BUILTIN_CHARACTER,
|
|
ECL_BUILTIN_NUMBER,
|
|
ECL_BUILTIN_REAL,
|
|
ECL_BUILTIN_RATIONAL,
|
|
ECL_BUILTIN_INTEGER,
|
|
ECL_BUILTIN_FIXNUM,
|
|
ECL_BUILTIN_BIGNUM,
|
|
ECL_BUILTIN_RATIO,
|
|
ECL_BUILTIN_FLOAT,
|
|
ECL_BUILTIN_SINGLE_FLOAT,
|
|
ECL_BUILTIN_DOUBLE_FLOAT,
|
|
ECL_BUILTIN_LONG_FLOAT,
|
|
ECL_BUILTIN_COMPLEX,
|
|
ECL_BUILTIN_COMPLEX_FLOAT,
|
|
ECL_BUILTIN_COMPLEX_SINGLE_FLOAT,
|
|
ECL_BUILTIN_COMPLEX_DOUBLE_FLOAT,
|
|
ECL_BUILTIN_COMPLEX_LONG_FLOAT,
|
|
ECL_BUILTIN_SYMBOL,
|
|
ECL_BUILTIN_NULL,
|
|
ECL_BUILTIN_KEYWORD,
|
|
ECL_BUILTIN_PACKAGE,
|
|
ECL_BUILTIN_FUNCTION,
|
|
ECL_BUILTIN_PATHNAME,
|
|
ECL_BUILTIN_LOGICAL_PATHNAME,
|
|
ECL_BUILTIN_HASH_TABLE,
|
|
ECL_BUILTIN_RANDOM_STATE,
|
|
ECL_BUILTIN_READTABLE,
|
|
ECL_BUILTIN_CODE_BLOCK,
|
|
ECL_BUILTIN_FOREIGN_DATA,
|
|
ECL_BUILTIN_FRAME,
|
|
ECL_BUILTIN_TOKEN,
|
|
ECL_BUILTIN_MODULE,
|
|
ECL_BUILTIN_EXCEPTION,
|
|
ECL_BUILTIN_WEAK_POINTER,
|
|
ECL_BUILTIN_PROCESS,
|
|
ECL_BUILTIN_LOCK,
|
|
ECL_BUILTIN_RWLOCK,
|
|
ECL_BUILTIN_CONDITION_VARIABLE,
|
|
ECL_BUILTIN_SEMAPHORE,
|
|
ECL_BUILTIN_BARRIER,
|
|
ECL_BUILTIN_MAILBOX,
|
|
ECL_BUILTIN_SSE_PACK
|
|
};
|
|
|
|
cl_object
|
|
cl_class_of(cl_object x)
|
|
{
|
|
size_t index;
|
|
switch (ecl_t_of(x)) {
|
|
case t_instance:
|
|
@(return ECL_CLASS_OF(x));
|
|
case t_fixnum:
|
|
index = ECL_BUILTIN_FIXNUM; break;
|
|
case t_bignum:
|
|
index = ECL_BUILTIN_BIGNUM; break;
|
|
case t_ratio:
|
|
index = ECL_BUILTIN_RATIO; break;
|
|
case t_singlefloat:
|
|
index = ECL_BUILTIN_SINGLE_FLOAT; break;
|
|
case t_doublefloat:
|
|
index = ECL_BUILTIN_DOUBLE_FLOAT; break;
|
|
case t_longfloat:
|
|
index = ECL_BUILTIN_LONG_FLOAT; break;
|
|
case t_complex:
|
|
index = ECL_BUILTIN_COMPLEX; break;
|
|
#ifdef ECL_COMPLEX_FLOAT
|
|
case t_csfloat:
|
|
index = ECL_BUILTIN_COMPLEX_SINGLE_FLOAT; break;
|
|
case t_cdfloat:
|
|
index = ECL_BUILTIN_COMPLEX_DOUBLE_FLOAT; break;
|
|
case t_clfloat:
|
|
index = ECL_BUILTIN_COMPLEX_LONG_FLOAT; break;
|
|
#endif
|
|
case t_character:
|
|
index = ECL_BUILTIN_CHARACTER; break;
|
|
case t_symbol:
|
|
if (x->symbol.hpack == cl_core.keyword_package)
|
|
index = ECL_BUILTIN_KEYWORD;
|
|
else
|
|
index = ECL_BUILTIN_SYMBOL;
|
|
break;
|
|
case t_package:
|
|
index = ECL_BUILTIN_PACKAGE; break;
|
|
case t_list:
|
|
index = Null(x)? ECL_BUILTIN_NULL : ECL_BUILTIN_CONS; break;
|
|
case t_hashtable:
|
|
index = ECL_BUILTIN_HASH_TABLE; break;
|
|
case t_array:
|
|
index = ECL_BUILTIN_ARRAY; break;
|
|
case t_vector:
|
|
index = ECL_BUILTIN_VECTOR; break;
|
|
#ifdef ECL_UNICODE
|
|
case t_string:
|
|
index = ECL_BUILTIN_STRING; break;
|
|
case t_base_string:
|
|
index = ECL_BUILTIN_BASE_STRING; break;
|
|
#else
|
|
case t_base_string:
|
|
index = ECL_BUILTIN_STRING; break;
|
|
#endif
|
|
case t_bitvector:
|
|
index = ECL_BUILTIN_BIT_VECTOR; break;
|
|
case t_stream:
|
|
switch (x->stream.mode) {
|
|
case ecl_smm_synonym: index = ECL_BUILTIN_SYNONYM_STREAM; break;
|
|
case ecl_smm_broadcast: index = ECL_BUILTIN_BROADCAST_STREAM; break;
|
|
case ecl_smm_concatenated: index = ECL_BUILTIN_CONCATENATED_STREAM; break;
|
|
case ecl_smm_two_way: index = ECL_BUILTIN_TWO_WAY_STREAM; break;
|
|
case ecl_smm_string_input:
|
|
case ecl_smm_string_output: index = ECL_BUILTIN_STRING_STREAM; break;
|
|
case ecl_smm_echo: index = ECL_BUILTIN_ECHO_STREAM; break;
|
|
case ecl_smm_sequence_input:
|
|
case ecl_smm_sequence_output: index = ECL_BUILTIN_SEQUENCE_STREAM; break;
|
|
default: index = ECL_BUILTIN_FILE_STREAM; break;
|
|
}
|
|
break;
|
|
case t_readtable:
|
|
index = ECL_BUILTIN_READTABLE; break;
|
|
case t_pathname:
|
|
index = ECL_BUILTIN_PATHNAME; break;
|
|
case t_random:
|
|
index = ECL_BUILTIN_RANDOM_STATE; break;
|
|
case t_bytecodes:
|
|
case t_bclosure:
|
|
case t_cfun:
|
|
case t_cfunfixed:
|
|
case t_cclosure:
|
|
index = ECL_BUILTIN_FUNCTION; break;
|
|
#ifdef ECL_THREADS
|
|
case t_process:
|
|
index = ECL_BUILTIN_PROCESS; break;
|
|
case t_lock:
|
|
index = ECL_BUILTIN_LOCK; break;
|
|
case t_rwlock:
|
|
index = ECL_BUILTIN_RWLOCK; break;
|
|
case t_condition_variable:
|
|
index = ECL_BUILTIN_CONDITION_VARIABLE; break;
|
|
case t_semaphore:
|
|
index = ECL_BUILTIN_SEMAPHORE; break;
|
|
case t_barrier:
|
|
index = ECL_BUILTIN_BARRIER; break;
|
|
case t_mailbox:
|
|
index = ECL_BUILTIN_MAILBOX; break;
|
|
#endif
|
|
case t_codeblock:
|
|
index = ECL_BUILTIN_CODE_BLOCK; break;
|
|
case t_foreign:
|
|
index = ECL_BUILTIN_FOREIGN_DATA; break;
|
|
case t_frame:
|
|
index = ECL_BUILTIN_FRAME; break;
|
|
case t_token:
|
|
index = ECL_BUILTIN_TOKEN; break;
|
|
case t_module:
|
|
index = ECL_BUILTIN_MODULE; break;
|
|
case t_exception:
|
|
index = ECL_BUILTIN_EXCEPTION; break;
|
|
case t_weak_pointer:
|
|
index = ECL_BUILTIN_WEAK_POINTER; break;
|
|
#ifdef ECL_SSE2
|
|
case t_sse_pack:
|
|
index = ECL_BUILTIN_SSE_PACK; break;
|
|
#endif
|
|
default:
|
|
ecl_internal_error("not a lisp data object");
|
|
}
|
|
{
|
|
/* We have to be careful because +builtin-classes+ might be empty! */
|
|
/* In any case, since +builtin-classes+ is a constant, we may
|
|
* optimize the slot access */
|
|
cl_object v = @'clos::+builtin-classes+'->symbol.value;
|
|
cl_object output = Null(v)?
|
|
cl_find_class(1, ECL_T) :
|
|
v->vector.self.t[index];
|
|
@(return output);
|
|
}
|
|
}
|
|
|