mirror of
https://gitlab.com/embeddable-common-lisp/ecl.git
synced 2025-12-05 18:30:24 -08:00
783 lines
25 KiB
C
Executable file
783 lines
25 KiB
C
Executable file
/* -*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*- */
|
|
/* vim: set filetype=c tabstop=8 shiftwidth=4 expandtab: */
|
|
|
|
/*
|
|
* Copyright (c) 2001, Juan Jose Garcia Ripoll.
|
|
*
|
|
* See file 'LICENSE' for the copyright details.
|
|
*
|
|
*/
|
|
|
|
/* internal.h -- Structures and functions that are not meant for the end user */
|
|
#ifndef ECL_INTERNAL_H
|
|
#define ECL_INTERNAL_H
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
/* -------------------------------------------------------------------- *
|
|
* FUNCTIONS, VARIABLES AND TYPES NOT FOR GENERAL USE *
|
|
* -------------------------------------------------------------------- */
|
|
|
|
#define unlikely_if(x) if (ecl_unlikely(x))
|
|
|
|
/* booting */
|
|
extern void init_all_symbols(void);
|
|
extern void init_alloc(void);
|
|
extern void init_backq(void);
|
|
extern void init_big();
|
|
extern void init_clos(void);
|
|
extern void init_error(void);
|
|
extern void init_eval(void);
|
|
extern void init_file(void);
|
|
#ifndef GBC_BOEHM
|
|
extern void init_GC(void);
|
|
#endif
|
|
extern void init_macros(void);
|
|
extern void init_read(void);
|
|
extern void init_stacks(cl_env_ptr);
|
|
extern void init_unixint(int pass);
|
|
extern void init_unixtime(void);
|
|
extern void init_compiler(void);
|
|
#ifdef ECL_THREADS
|
|
extern void init_threads(cl_env_ptr);
|
|
#else
|
|
#define init_threads(env) cl_env_p = env
|
|
#endif
|
|
extern void ecl_init_env(cl_env_ptr);
|
|
extern void init_lib_LSP(cl_object);
|
|
|
|
extern cl_env_ptr _ecl_alloc_env(cl_env_ptr parent);
|
|
extern void _ecl_dealloc_env(cl_env_ptr);
|
|
|
|
/* alloc.d/alloc_2.d */
|
|
|
|
#ifdef GBC_BOEHM
|
|
#define ECL_COMPACT_OBJECT_EXTRA(x) ((void*)((x)->array.displaced))
|
|
#endif
|
|
extern void _ecl_set_max_heap_size(size_t new_size);
|
|
extern cl_object ecl_alloc_bytecodes(cl_index data_size, cl_index code_size);
|
|
extern cl_index ecl_object_byte_size(cl_type t);
|
|
extern cl_index ecl_next_stamp();
|
|
|
|
/* array.d */
|
|
|
|
#ifdef ECL_DEFINE_AET_SIZE
|
|
#undef ECL_DEFINE_AET_SIZE
|
|
static const cl_index ecl_aet_size[] = {
|
|
sizeof(cl_object), /* ecl_aet_object */
|
|
sizeof(float), /* ecl_aet_sf */
|
|
sizeof(double), /* ecl_aet_df */
|
|
sizeof(long double), /* ecl_aet_lf */
|
|
#ifdef ECL_COMPLEX_FLOAT
|
|
sizeof(_Complex float), /* ecl_aet_csf */
|
|
sizeof(_Complex double), /* ecl_aet_cdf */
|
|
sizeof(_Complex long double), /* ecl_aet_clf */
|
|
#endif
|
|
0, /* ecl_aet_bit: cannot be handled with this code */
|
|
sizeof(cl_fixnum), /* ecl_aet_fix */
|
|
sizeof(cl_index), /* ecl_aet_index */
|
|
sizeof(uint8_t), /* ecl_aet_b8 */
|
|
sizeof(int8_t), /* ecl_aet_i8 */
|
|
#ifdef ecl_uint16_t
|
|
sizeof(ecl_uint16_t),
|
|
sizeof(ecl_int16_t),
|
|
#endif
|
|
#ifdef ecl_uint32_t
|
|
sizeof(ecl_uint32_t),
|
|
sizeof(ecl_int32_t),
|
|
#endif
|
|
#ifdef ecl_uint64_t
|
|
sizeof(ecl_uint64_t),
|
|
sizeof(ecl_int64_t),
|
|
#endif
|
|
#ifdef ECL_UNICODE
|
|
sizeof(ecl_character), /* ecl_aet_ch */
|
|
#endif
|
|
sizeof(unsigned char) /* ecl_aet_bc */
|
|
};
|
|
#endif /* ECL_DEFINE_AET_SIZE */
|
|
|
|
extern void ecl_displace(cl_object from, cl_object to, cl_object offset);
|
|
|
|
/* compiler.d */
|
|
|
|
struct cl_compiler_env {
|
|
cl_object variables; /* Variables, tags, functions, etc: the env. */
|
|
cl_object macros; /* Macros and function bindings */
|
|
cl_fixnum lexical_level; /* =0 if toplevel form */
|
|
cl_object constants; /* Constants for this form */
|
|
cl_object load_time_forms; /* Constants that have to be rebuilt */
|
|
cl_object ltf_being_created; /* Load time objects being compiled */
|
|
cl_object ltf_defer_init_until; /* Defer evaluation of current
|
|
* load time init form until
|
|
* this object has been created */
|
|
cl_object ltf_locations; /* Locations of constants externalized
|
|
* with make-load-form */
|
|
cl_object lex_env; /* Lexical env. for eval-when */
|
|
cl_object code_walker; /* Value of SI:*CODE-WALKER* */
|
|
cl_index env_depth;
|
|
cl_index env_size;
|
|
int mode;
|
|
bool stepping;
|
|
bool function_boundary_crossed;
|
|
};
|
|
|
|
typedef struct cl_compiler_env *cl_compiler_env_ptr;
|
|
|
|
/* character.d */
|
|
|
|
#ifdef ECL_UNICODE
|
|
#define ECL_UCS_NONCHARACTER(c) \
|
|
(((c) >= 0xFDD0 && (c) <= 0xFDEF) || \
|
|
(((c) & 0xFFFF) >= 0xFFFE && (((c) & 0xFFFF) <= 0xFFFF)))
|
|
#define ECL_UCS_PRIVATE(c) \
|
|
(((c) >= 0xE000 && (c) <= 0xF8FF) || \
|
|
((c) >= 0xF0000 && (c) <= 0xFFFD) || \
|
|
((c) >= 0x100000 && (c) <= 0x10FFFD))
|
|
#define ECL_UCS_HIGH_SURROGATE(c) ((c) >= 0xD800 && (c) <= 0xDBFF)
|
|
#define ECL_UCS_LOW_SURROGATE(c) ((c) >= 0xDC00 && (c) <= 0xDFFF)
|
|
#endif
|
|
|
|
/* error.d */
|
|
|
|
extern void _ecl_unexpected_return() ecl_attr_noreturn;
|
|
extern cl_object _ecl_strerror(int code);
|
|
extern ECL_API cl_object si_serror _ECL_ARGS
|
|
((cl_narg narg, cl_object cformat, cl_object eformat, ...));
|
|
|
|
|
|
/* eval.d */
|
|
|
|
#define _ecl_funcall5(fun, a, b, c, d) \
|
|
ecl_function_dispatch(ecl_process_env(), (fun))(4, (a),(b),(c),(d))
|
|
#define _ecl_funcall4(fun, a, b, c) \
|
|
ecl_function_dispatch(ecl_process_env(), (fun))(3, (a),(b),(c))
|
|
#define _ecl_funcall3(fun, a, b) \
|
|
ecl_function_dispatch(ecl_process_env(), (fun))(2, (a),(b))
|
|
#define _ecl_funcall2(fun, a) \
|
|
ecl_function_dispatch(ecl_process_env(), (fun))(1, (a))
|
|
#define _ecl_funcall1(fun) \
|
|
ecl_function_dispatch(ecl_process_env(), (fun))(0)
|
|
|
|
extern cl_object si_constantp_inner _ECL_ARGS((cl_narg narg, cl_object form, ...));
|
|
extern cl_object si_constant_form_value _ECL_ARGS((cl_narg narg, cl_object form, ...));
|
|
|
|
/* interpreter.d */
|
|
|
|
#define ECL_BUILD_STACK_FRAME(env,name,frame) \
|
|
struct ecl_stack_frame frame;\
|
|
cl_object name = ecl_stack_frame_open(env, (cl_object)&frame, 0);
|
|
|
|
#define ECL_STACK_FRAME_FROM_VA_LIST(e,f,va) do { \
|
|
const cl_object __frame = (f); \
|
|
cl_index i, __nargs = va[0].narg; \
|
|
ecl_stack_frame_open((e), __frame, __nargs); \
|
|
for (i = 0; i < __nargs; i++) { \
|
|
__frame->frame.base[i] = ecl_va_arg(va); \
|
|
} \
|
|
} while (0)
|
|
|
|
#define ECL_STACK_FRAME_VARARGS_BEGIN(narg,lastarg,frame) \
|
|
struct ecl_stack_frame __ecl_frame; \
|
|
const cl_object frame = (cl_object)&__ecl_frame; \
|
|
const cl_env_ptr env = ecl_process_env(); \
|
|
frame->frame.t = t_frame; \
|
|
frame->frame.env = env; \
|
|
frame->frame.size = narg; \
|
|
if (narg <= ECL_C_ARGUMENTS_LIMIT) { \
|
|
cl_object *p = frame->frame.base = env->values; \
|
|
va_list args; \
|
|
va_start(args, lastarg); \
|
|
while (narg--) { \
|
|
*p = va_arg(args, cl_object); \
|
|
++p; \
|
|
} \
|
|
va_end(args); \
|
|
frame->frame.stack = (cl_object*)0x1; \
|
|
} else { \
|
|
frame->frame.base = env->stack_top - narg; \
|
|
frame->frame.stack = 0; \
|
|
}
|
|
#define ECL_STACK_FRAME_VARARGS_END(frame) \
|
|
/* No stack consumed, no need to close frame */
|
|
|
|
extern cl_object _ecl_bytecodes_dispatch_vararg(cl_narg narg, ...);
|
|
extern cl_object _ecl_bclosure_dispatch_vararg(cl_narg narg, ...);
|
|
extern cl_object ecl_close_around(cl_object fun, cl_object env);
|
|
|
|
/* ffi/backtrace.d */
|
|
|
|
extern void _ecl_dump_c_backtrace();
|
|
|
|
/* ffi.d */
|
|
|
|
extern enum ecl_ffi_tag ecl_foreign_type_code(cl_object type);
|
|
|
|
/* file.d */
|
|
|
|
/* Windows does not have this flag (POSIX thing) */
|
|
#ifndef O_CLOEXEC
|
|
#define O_CLOEXEC 0
|
|
#endif
|
|
#ifndef O_NONBLOCK
|
|
#define O_NONBLOCK 0
|
|
#endif
|
|
/* Windows needs to be told explicitely to open files in binary mode */
|
|
#ifndef O_BINARY
|
|
#define O_BINARY 0
|
|
#endif
|
|
|
|
#define ECL_FILE_STREAM_P(strm) \
|
|
(ECL_ANSI_STREAM_P(strm) && (strm)->stream.mode < ecl_smm_synonym)
|
|
#define STRING_OUTPUT_STRING(strm) (strm)->stream.object0
|
|
#define STRING_INPUT_STRING(strm) (strm)->stream.object0
|
|
#define STRING_INPUT_POSITION(strm) (strm)->stream.int0
|
|
#define STRING_INPUT_LIMIT(strm) (strm)->stream.int1
|
|
#define TWO_WAY_STREAM_INPUT(strm) (strm)->stream.object0
|
|
#define TWO_WAY_STREAM_OUTPUT(strm) (strm)->stream.object1
|
|
#define SYNONYM_STREAM_SYMBOL(strm) (strm)->stream.object0
|
|
#define SYNONYM_STREAM_STREAM(strm) ecl_symbol_value((strm)->stream.object0)
|
|
#define BROADCAST_STREAM_LIST(strm) (strm)->stream.object0
|
|
#define ECHO_STREAM_INPUT(strm) (strm)->stream.object0
|
|
#define ECHO_STREAM_OUTPUT(strm) (strm)->stream.object1
|
|
#define CONCATENATED_STREAM_LIST(strm) (strm)->stream.object0
|
|
#define IO_STREAM_FILE(strm) ((strm)->stream.file.stream)
|
|
#define IO_STREAM_ELT_TYPE(strm) (strm)->stream.object0
|
|
#define IO_STREAM_FILENAME(strm) (strm)->stream.object1
|
|
#define IO_FILE_DESCRIPTOR(strm) (strm)->stream.file.descriptor
|
|
#define IO_FILE_ELT_TYPE(strm) (strm)->stream.object0
|
|
#define IO_FILE_FILENAME(strm) (strm)->stream.object1
|
|
#define SEQ_OUTPUT_VECTOR(strm) (strm)->stream.object1
|
|
#define SEQ_OUTPUT_POSITION(strm) (strm)->stream.int0
|
|
#define SEQ_INPUT_VECTOR(strm) (strm)->stream.object1
|
|
#define SEQ_INPUT_POSITION(strm) (strm)->stream.int0
|
|
#define SEQ_INPUT_LIMIT(strm) (strm)->stream.int1
|
|
|
|
#ifndef HAVE_FSEEKO
|
|
#define ecl_off_t int
|
|
#define ecl_fseeko fseek
|
|
#define ecl_ftello ftell
|
|
#else
|
|
#define ecl_off_t off_t
|
|
#define ecl_fseeko fseeko
|
|
#define ecl_ftello ftello
|
|
#endif
|
|
|
|
extern cl_object ecl_off_t_to_integer(ecl_off_t offset);
|
|
extern ecl_off_t ecl_integer_to_off_t(cl_object offset);
|
|
|
|
/* format.d */
|
|
|
|
#ifndef ECL_CMU_FORMAT
|
|
extern cl_object si_formatter_aux _ECL_ARGS((cl_narg narg, cl_object strm, cl_object string, ...));
|
|
#endif
|
|
|
|
/* hash.d */
|
|
extern cl_object ecl_extend_hashtable(cl_object hashtable);
|
|
#ifdef ECL_EXTERNALIZABLE
|
|
extern void ecl_reconstruct_serialized_hashtable(cl_object h);
|
|
#endif
|
|
|
|
/* gfun.d, kernel.lsp */
|
|
|
|
#define GFUN_NAME(x) ((x)->instance.slots[0])
|
|
#define GFUN_SPEC(x) ((x)->instance.slots[1])
|
|
#define GFUN_COMB(x) ((x)->instance.slots[2])
|
|
|
|
extern cl_object FEnot_funcallable_vararg(cl_narg narg, ...);
|
|
extern cl_object ecl_slot_reader_dispatch(cl_narg narg, ... /* cl_object instance */);
|
|
extern cl_object ecl_slot_writer_dispatch(cl_narg narg, ... /* cl_object value, cl_object instance */);
|
|
|
|
/* load.d */
|
|
|
|
extern cl_object _ecl_library_init_prefix(void);
|
|
extern cl_object _ecl_library_default_entry(void);
|
|
|
|
/* number.d */
|
|
|
|
extern cl_object _ecl_double_to_integer(double d);
|
|
extern cl_object _ecl_float_to_integer(float d);
|
|
extern cl_object _ecl_long_double_to_integer(long double d);
|
|
|
|
/* main.d */
|
|
|
|
extern cl_fixnum ecl_option_values[ECL_OPT_LIMIT+1];
|
|
|
|
extern void ecl_init_bignum_registers(cl_env_ptr env);
|
|
extern void ecl_clear_bignum_registers(cl_env_ptr env);
|
|
|
|
/* threads/mutex.d */
|
|
|
|
extern cl_object si_mutex_timeout();
|
|
|
|
/* print.d */
|
|
|
|
extern cl_object _ecl_stream_or_default_output(cl_object stream);
|
|
extern void _ecl_write_addr(void *x, cl_object stream);
|
|
extern void _ecl_write_array(cl_object o, cl_object stream);
|
|
extern void _ecl_write_vector(cl_object o, cl_object stream);
|
|
extern void _ecl_write_bitvector(cl_object o, cl_object stream);
|
|
extern void _ecl_write_string(cl_object o, cl_object stream);
|
|
extern void _ecl_write_base_string(cl_object o, cl_object stream);
|
|
extern void _ecl_write_list(cl_object o, cl_object stream);
|
|
extern void _ecl_write_bclosure(cl_object o, cl_object stream);
|
|
extern void _ecl_write_bytecodes(cl_object o, cl_object stream);
|
|
extern void _ecl_write_symbol(cl_object o, cl_object stream);
|
|
extern void _ecl_write_fixnum(cl_fixnum o, cl_object stream);
|
|
extern void _ecl_write_sse(cl_object o, cl_object stream);
|
|
extern void _ecl_write_unreadable(cl_object x, const char *prefix, cl_object name, cl_object stream);
|
|
extern bool _ecl_will_print_as_hash(cl_object o);
|
|
extern cl_object _ecl_ensure_buffer(cl_object buffer, cl_fixnum length);
|
|
extern void _ecl_string_push_c_string(cl_object s, const char *c);
|
|
|
|
#define ECL_PPRINT_QUEUE_SIZE 128
|
|
#define ECL_PPRINT_INDENTATION_STACK_SIZE 256
|
|
|
|
extern void cl_write_object(cl_object x, cl_object stream);
|
|
|
|
/* threads/rwlock.d */
|
|
|
|
#ifdef ECL_THREADS
|
|
extern cl_object mp_get_rwlock_read_wait(cl_object lock);
|
|
extern cl_object mp_get_rwlock_write_wait(cl_object lock);
|
|
#endif
|
|
|
|
/* read.d */
|
|
#ifdef ECL_UNICODE
|
|
#define RTABSIZE 256 /* read table size */
|
|
#else
|
|
#define RTABSIZE ECL_CHAR_CODE_LIMIT /* read table size */
|
|
#endif
|
|
|
|
/* package.d */
|
|
|
|
extern cl_object _ecl_package_to_be_created(const cl_env_ptr env, cl_object name);
|
|
|
|
/* pathname.d */
|
|
|
|
extern bool ecl_wild_string_p(cl_object item);
|
|
|
|
/* sequence.d */
|
|
typedef struct { cl_index start, end, length; } cl_index_pair;
|
|
extern ECL_API cl_index_pair ecl_sequence_start_end(cl_object fun, cl_object s, cl_object start, cl_object end);
|
|
|
|
#ifdef ECL_EXTERNALIZABLE
|
|
/* serialize.d */
|
|
|
|
extern cl_object ecl_deserialize(uint8_t *data);
|
|
#endif
|
|
|
|
/* string.d */
|
|
#define ecl_vector_start_end ecl_sequence_start_end
|
|
|
|
/* stacks.d */
|
|
#define CL_NEWENV_BEGIN {\
|
|
const cl_env_ptr the_env = ecl_process_env(); \
|
|
cl_index __i = ecl_stack_push_values(the_env); \
|
|
|
|
#define CL_NEWENV_END \
|
|
ecl_stack_pop_values(the_env,__i); }
|
|
|
|
extern void ecl_cs_set_org(cl_env_ptr env);
|
|
|
|
#ifndef RLIM_SAVED_MAX
|
|
# define RLIM_SAVED_MAX RLIM_INFINITY
|
|
#endif
|
|
|
|
#ifndef RLIM_SAVED_CUR
|
|
# define RLIM_SAVED_CUR RLIM_INFINITY
|
|
#endif
|
|
|
|
/* threads.d */
|
|
|
|
#ifdef ECL_THREADS
|
|
extern ECL_API cl_object mp_suspend_loop();
|
|
extern ECL_API cl_object mp_break_suspend_loop();
|
|
|
|
# ifdef ECL_WINDOWS_THREADS
|
|
# define ecl_thread_exit() ExitThread(0);
|
|
# else
|
|
# define ecl_thread_exit() pthread_exit(NULL);
|
|
# endif /* ECL_WINDOWS_THREADS */
|
|
|
|
#endif
|
|
|
|
/* time.d */
|
|
|
|
struct ecl_timeval {
|
|
cl_index tv_usec;
|
|
cl_index tv_sec;
|
|
};
|
|
|
|
extern void ecl_get_internal_real_time(struct ecl_timeval *time);
|
|
extern void ecl_get_internal_run_time(struct ecl_timeval *time);
|
|
extern void ecl_musleep(double time);
|
|
|
|
#define UTC_time_to_universal_time(x) ecl_plus(ecl_make_integer(x),cl_core.Jan1st1970UT)
|
|
extern cl_fixnum ecl_runtime(void);
|
|
|
|
/* unixfsys.d */
|
|
|
|
/* Filename encodings: on Unix we use ordinary chars encoded in a user
|
|
* specified format (usually utf8), while on Windows we use a wchar_t
|
|
* type.
|
|
*
|
|
* Naming conventions:
|
|
* fstr: null-terminated raw C array with element type char or wchar_t
|
|
* filename: Lisp base string or vector with element type byte16,
|
|
* also null-terminated
|
|
*/
|
|
#if defined(ECL_MS_WINDOWS_HOST) && defined(ECL_UNICODE)
|
|
#include <wchar.h>
|
|
|
|
typedef wchar_t ecl_filename_char;
|
|
#define ecl_fstrlen(x) wcslen(x)
|
|
#define ecl_fstrcpy(x,y) wcscpy(x,y)
|
|
#define ecl_fstrcat(x,y) wcscat(x,y)
|
|
#define ecl_fstr(x) L ## x /* wchar_t string constructor prefixed with L */
|
|
|
|
cl_object ecl_make_simple_filename(const ecl_filename_char *x, cl_fixnum size);
|
|
#define ecl_make_constant_filename(x,y) ecl_make_simple_filename(x,y)
|
|
cl_object ecl_alloc_filename(cl_index len, cl_object adjustable);
|
|
#define ecl_alloc_adjustable_filename(len) ecl_alloc_filename(len, ECL_T)
|
|
#define ecl_alloc_simple_filename(len) ecl_alloc_filename(len, ECL_NIL)
|
|
cl_object ecl_concatenate_filename(cl_object x, cl_object y);
|
|
#define ecl_filename_self(x) ((ecl_filename_char*)((x)->vector.self.b16))
|
|
|
|
#define ecl_chdir _wchdir
|
|
#define ecl_stat _wstat64
|
|
#define ecl_fstat _fstat64
|
|
typedef struct __stat64 ecl_stat_struct;
|
|
#define ecl_getcwd _wgetcwd
|
|
#define ecl_access _waccess
|
|
#define ecl_unlink _wunlink
|
|
#define ecl_rename _wrename
|
|
#define ecl_open _wopen
|
|
#define ecl_fopen _wfopen
|
|
#define ecl_fdopen _wfdopen
|
|
#define ecl_rmdir _wrmdir
|
|
#define ecl_mkdir _wmkdir
|
|
#define ecl_chmod _wchmod
|
|
#define ecl_getenv _wgetenv
|
|
#define ecl_GetFileAttributes GetFileAttributesW
|
|
#define ecl_MoveFile MoveFileW
|
|
#define ecl_MoveFileEx MoveFileExW
|
|
#define ecl_DeleteFile DeleteFileW
|
|
#define ecl_FindFirstFile FindFirstFileW
|
|
#define ecl_FindNextFile FindNextFileW
|
|
#define ecl_WIN32_FIND_DATA WIN32_FIND_DATAW
|
|
#define ecl_GetTempFileName GetTempFileNameW
|
|
#define ecl_CopyFile CopyFileW
|
|
#define ecl_LoadLibrary LoadLibraryW
|
|
#define ecl_GetModuleFileName GetModuleFileNameW
|
|
|
|
#else
|
|
|
|
typedef char ecl_filename_char;
|
|
#define ecl_fstrlen(x) strlen(x)
|
|
#define ecl_fstrcpy(x,y) strcpy(x,y)
|
|
#define ecl_fstrcat(x,y) strcat(x,y)
|
|
#define ecl_fstr(x) x
|
|
|
|
#define ecl_make_simple_filename(x,y) ecl_make_simple_base_string((char *)x,y)
|
|
#define ecl_make_constant_filename(x,y) ecl_make_constant_base_string((char *)x,y)
|
|
#define ecl_alloc_adjustable_filename(len) ecl_alloc_adjustable_base_string(len)
|
|
#define ecl_alloc_simple_filename(len) ecl_alloc_simple_base_string(len)
|
|
#define ecl_concatenate_filename(x,y) si_base_string_concatenate(2,x,y)
|
|
#define ecl_filename_self(x) ((ecl_filename_char*)((x)->base_string.self))
|
|
|
|
#define ecl_chdir chdir
|
|
#define ecl_stat stat
|
|
#define ecl_fstat fstat
|
|
typedef struct stat ecl_stat_struct;
|
|
#define ecl_getcwd getcwd
|
|
#define ecl_access access
|
|
#define ecl_unlink unlink
|
|
#define ecl_rename rename
|
|
#define ecl_open open
|
|
#define ecl_fopen fopen
|
|
#define ecl_fdopen fdopen
|
|
#define ecl_rmdir rmdir
|
|
#define ecl_mkdir mkdir
|
|
#define ecl_chmod chmod
|
|
#define ecl_getenv getenv
|
|
#define ecl_GetFileAttributes GetFileAttributesA
|
|
#define ecl_MoveFile MoveFileA
|
|
#define ecl_MoveFileEx MoveFileExA
|
|
#define ecl_DeleteFile DeleteFileA
|
|
#define ecl_FindFirstFile FindFirstFileA
|
|
#define ecl_FindNextFile FindNextFileA
|
|
#define ecl_WIN32_FIND_DATA WIN32_FIND_DATAA
|
|
#define ecl_GetTempFileName GetTempFileNameA
|
|
#define ecl_CopyFile CopyFileA
|
|
#define ecl_LoadLibrary LoadLibraryA
|
|
#define ecl_GetModuleFileName GetModuleFileNameA
|
|
|
|
#endif
|
|
|
|
/*
|
|
* POSIX specifies that the "b" flag is ignored. This is good, because
|
|
* under MSDOS and Apple's OS we need to open text files in binary mode,
|
|
* so that we get both the carriage return and the linefeed characters.
|
|
* Otherwise, it would be complicated to implement file-position and
|
|
* seek operations.
|
|
*/
|
|
#define OPEN_R ecl_fstr("rb")
|
|
#define OPEN_W ecl_fstr("wb")
|
|
#define OPEN_RW ecl_fstr("r+b")
|
|
#define OPEN_A ecl_fstr("ab")
|
|
#define OPEN_RA ecl_fstr("a+b")
|
|
|
|
int ecl_backup_open(const ecl_filename_char *filename, int option, int mode);
|
|
cl_object ecl_decode_filename(cl_object x, cl_object len);
|
|
cl_object ecl_encode_filename(cl_object x, cl_object len);
|
|
|
|
|
|
/* unixint.d */
|
|
|
|
#define ECL_PI_D 3.14159265358979323846264338327950288
|
|
#define ECL_PI_L 3.14159265358979323846264338327950288l
|
|
#define ECL_PI2_D 1.57079632679489661923132169163975144
|
|
#define ECL_PI2_L 1.57079632679489661923132169163975144l
|
|
|
|
extern void ecl_interrupt_process(cl_object process, cl_object function);
|
|
|
|
/* disabling interrupts on the lisp side */
|
|
|
|
#define ECL_WITHOUT_INTERRUPTS_BEGIN(the_env) do { \
|
|
cl_env_ptr __the_env = (the_env); \
|
|
ecl_bds_bind(__the_env, ECL_INTERRUPTS_ENABLED, ECL_NIL);
|
|
|
|
#define ECL_WITHOUT_INTERRUPTS_END \
|
|
ecl_bds_unwind1(__the_env); \
|
|
ecl_check_pending_interrupts(__the_env); } while(0)
|
|
|
|
/* unixsys.d */
|
|
|
|
/* Some old BSD systems don't have WCONTINUED / WIFCONTINUED */
|
|
|
|
#ifndef ECL_MS_WINDOWS_HOST
|
|
# ifndef WCONTINUED
|
|
# define WCONTINUED 0
|
|
# endif
|
|
|
|
# ifndef WIFCONTINUED
|
|
# define WIFCONTINUED(x) 0
|
|
# endif
|
|
#endif /* ECL_MS_WINDOWS_HOST */
|
|
|
|
/* global locks */
|
|
|
|
#include <ecl/threads.h>
|
|
|
|
#ifdef ECL_THREADS
|
|
# define ECL_WITH_GLOBAL_LOCK_BEGIN(the_env) \
|
|
ECL_WITH_NATIVE_LOCK_BEGIN(the_env, &cl_core.global_lock)
|
|
# define ECL_WITH_GLOBAL_LOCK_END \
|
|
ECL_WITH_NATIVE_LOCK_END
|
|
# define ECL_WITH_LOCK_BEGIN(the_env,lock) { \
|
|
const cl_env_ptr __ecl_the_env = the_env; \
|
|
const cl_object __ecl_the_lock = lock; \
|
|
ecl_disable_interrupts_env(__ecl_the_env); \
|
|
mp_get_lock_wait(__ecl_the_lock); \
|
|
ECL_UNWIND_PROTECT_BEGIN(__ecl_the_env); \
|
|
ecl_enable_interrupts_env(__ecl_the_env);
|
|
# define ECL_WITH_LOCK_END \
|
|
ECL_UNWIND_PROTECT_THREAD_SAFE_EXIT { \
|
|
mp_giveup_lock(__ecl_the_lock); \
|
|
} ECL_UNWIND_PROTECT_THREAD_SAFE_END; }
|
|
# define ECL_WITH_NATIVE_LOCK_BEGIN(the_env,lock) { \
|
|
const cl_env_ptr __ecl_the_env = (the_env); \
|
|
ecl_mutex_t* __ecl_the_lock = (lock); \
|
|
ecl_disable_interrupts_env(__ecl_the_env); \
|
|
ecl_mutex_lock(__ecl_the_lock); \
|
|
ECL_UNWIND_PROTECT_BEGIN(__ecl_the_env); \
|
|
ecl_enable_interrupts_env(__ecl_the_env);
|
|
# define ECL_WITH_NATIVE_LOCK_END \
|
|
ECL_UNWIND_PROTECT_THREAD_SAFE_EXIT { \
|
|
ecl_mutex_unlock(__ecl_the_lock); \
|
|
} ECL_UNWIND_PROTECT_THREAD_SAFE_END; }
|
|
# define ECL_WITH_GLOBAL_ENV_RDLOCK_BEGIN(the_env) { \
|
|
const cl_env_ptr __ecl_pack_env = the_env; \
|
|
ecl_bds_bind(__ecl_pack_env, ECL_INTERRUPTS_ENABLED, ECL_NIL); \
|
|
ecl_rwlock_lock_read(&cl_core.global_env_lock);
|
|
# define ECL_WITH_GLOBAL_ENV_RDLOCK_END \
|
|
ecl_rwlock_unlock_read(&cl_core.global_env_lock); \
|
|
ecl_bds_unwind1(__ecl_pack_env); \
|
|
ecl_check_pending_interrupts(__ecl_pack_env); }
|
|
# define ECL_WITH_GLOBAL_ENV_WRLOCK_BEGIN(the_env) { \
|
|
const cl_env_ptr __ecl_pack_env = the_env; \
|
|
ecl_bds_bind(__ecl_pack_env, ECL_INTERRUPTS_ENABLED, ECL_NIL); \
|
|
ecl_rwlock_lock_write(&cl_core.global_env_lock);
|
|
# define ECL_WITH_GLOBAL_ENV_WRLOCK_END \
|
|
ecl_rwlock_unlock_write(&cl_core.global_env_lock); \
|
|
ecl_bds_unwind1(__ecl_pack_env); \
|
|
ecl_check_pending_interrupts(__ecl_pack_env); }
|
|
#else
|
|
# define ECL_WITH_GLOBAL_LOCK_BEGIN(the_env)
|
|
# define ECL_WITH_GLOBAL_LOCK_END
|
|
# define ECL_WITH_LOCK_BEGIN(the_env,lock)
|
|
# define ECL_WITH_LOCK_END
|
|
# define ECL_WITH_NATIVE_LOCK_BEGIN(the_env,lock)
|
|
# define ECL_WITH_NATIVE_LOCK_END
|
|
# define ECL_WITH_GLOBAL_ENV_RDLOCK_BEGIN(the_env)
|
|
# define ECL_WITH_GLOBAL_ENV_RDLOCK_END
|
|
# define ECL_WITH_GLOBAL_ENV_WRLOCK_BEGIN(the_env)
|
|
# define ECL_WITH_GLOBAL_ENV_WRLOCK_END
|
|
#endif /* ECL_THREADS */
|
|
|
|
#include <ecl/ecl_atomics.h>
|
|
|
|
/*
|
|
* Fake several ISO C99 mathematical functions if not available
|
|
*/
|
|
#include <math.h>
|
|
|
|
#ifndef HAVE_EXPF
|
|
# ifdef expf
|
|
# undef expf
|
|
# endif
|
|
# define expf(x) exp((float)x)
|
|
#endif
|
|
#ifndef HAVE_POWF
|
|
# ifdef powf
|
|
# undef powf
|
|
# endif
|
|
# define powf(x,y) pow((float)x,(float)y)
|
|
#endif
|
|
#ifndef HAVE_LOGF
|
|
# ifdef logf
|
|
# undef logf
|
|
# endif
|
|
# define logf(x) log((float)x)
|
|
#endif
|
|
#ifndef HAVE_SQRTF
|
|
# ifdef sqrtf
|
|
# undef sqrtf
|
|
# endif
|
|
# define sqrtf(x) sqrt((float)x)
|
|
#endif
|
|
#ifndef HAVE_SINF
|
|
# ifdef sinf
|
|
# undef sinf
|
|
# endif
|
|
# define sinf(x) sin((float)x)
|
|
#endif
|
|
#ifndef HAVE_COSF
|
|
# ifdef cosf
|
|
# undef cosf
|
|
# endif
|
|
# define cosf(x) cos((float)x)
|
|
#endif
|
|
#ifndef HAVE_TANF
|
|
# ifdef tanf
|
|
# undef tanf
|
|
# endif
|
|
# define tanf(x) tan((float)x)
|
|
#endif
|
|
#ifndef HAVE_SINHF
|
|
# ifdef sinhf
|
|
# undef sinhf
|
|
# endif
|
|
# define sinhf(x) sinh((float)x)
|
|
#endif
|
|
#ifndef HAVE_COSHF
|
|
# ifdef coshf
|
|
# undef coshf
|
|
# endif
|
|
# define coshf(x) cosh((float)x)
|
|
#endif
|
|
#ifndef HAVE_TANHF
|
|
# ifdef tanhf
|
|
# undef tanhf
|
|
# endif
|
|
# define tanhf(x) tanh((float)x)
|
|
#endif
|
|
|
|
#ifndef HAVE_CEILF
|
|
# define ceilf(x) ceil((float)x)
|
|
#endif
|
|
#ifndef HAVE_FLOORF
|
|
# define floorf(x) floor((float)x)
|
|
#endif
|
|
#ifndef HAVE_FABSF
|
|
# define fabsf(x) fabs((float)x)
|
|
#endif
|
|
#ifndef HAVE_FREXPF
|
|
# define frexpf(x,y) frexp((float)x,y)
|
|
#endif
|
|
#ifndef HAVE_LDEXPF
|
|
# define ldexpf(x,y) ldexp((float)x,y)
|
|
#endif
|
|
|
|
#ifndef HAVE_LOG1PF
|
|
# ifdef log1pf
|
|
# undef log1pf
|
|
# endif
|
|
# define log1pf(x) logf(x+1.0f)
|
|
#endif
|
|
#ifndef HAVE_LOG1P
|
|
# ifdef log1p
|
|
# undef log1p
|
|
# endif
|
|
# define log1p(x) log(x+1.0)
|
|
#endif
|
|
#ifndef HAVE_LOG1PL
|
|
# ifdef log1pl
|
|
# undef log1pl
|
|
# endif
|
|
# define log1pl(x) logl(x+1.0l)
|
|
#endif
|
|
|
|
/*
|
|
* Fake INFINITY and NAN defined in ISO C99 (portably)
|
|
*/
|
|
|
|
#ifndef INFINITY
|
|
# if _MSC_VER == 1600
|
|
static union {
|
|
uint8_t bytes [ sizeof ( float ) ];
|
|
float inf;
|
|
} __ecl_inf = {
|
|
{ 0, 0, 0xf0, 0x7f }
|
|
};
|
|
# define INFINITY (__ecl_inf.inf)
|
|
# else
|
|
# define INFINITY (1.0/0.0)
|
|
# endif /* _MSC_VER == 1600 */
|
|
#endif /* INFINITY */
|
|
|
|
#ifndef NAN
|
|
# if _MSC_VER == 1600
|
|
static union {
|
|
uint8_t bytes [ sizeof ( float ) ];
|
|
float nan;
|
|
} __ecl_nan = {
|
|
{ 0, 0, 0xc0, 0x7f }
|
|
};
|
|
# define NAN (__ecl_nan.nan)
|
|
# else
|
|
# define NAN (0.0/0.0)
|
|
# endif /* _MSC_VER == 1600 */
|
|
#endif /* ~NAN */
|
|
|
|
#ifdef ECL_COMPLEX_FLOAT
|
|
#include <complex.h>
|
|
#ifndef CMPLXF
|
|
# define CMPLXF(x, y) ((float complex)((float)(x) + I * (float)(y)))
|
|
#endif
|
|
#ifndef CMPLX
|
|
# define CMPLX(x, y) ((double complex)((double)(x) + I * (double)(y)))
|
|
#endif
|
|
#ifndef CMPLXL
|
|
# define CMPLXL(x, y) ((long double complex)((long double)(x) + I * (long double)(y)))
|
|
#endif
|
|
#endif
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif
|
|
|
|
#endif /* ECL_INTERNAL_H */
|