picrin/lib/state.c

399 lines
8.4 KiB
C
Raw Normal View History

2014-08-25 00:38:09 -04:00
/**
* See Copyright Notice in picrin.h
*/
#include "picrin.h"
2017-03-28 10:09:40 -04:00
#include "object.h"
#include "state.h"
2014-08-25 00:38:09 -04:00
2016-02-23 13:59:51 -05:00
static pic_value
pic_features(pic_state *pic)
{
pic_get_args(pic, "");
return pic->features;
}
2015-01-26 00:31:36 -05:00
static void
pic_init_features(pic_state *pic)
{
2016-02-23 13:59:51 -05:00
pic_defun(pic, "features", pic_features);
2015-01-26 00:31:36 -05:00
pic_add_feature(pic, "picrin");
2015-07-19 13:19:41 -04:00
#if __STDC_IEC_559__
2015-01-26 00:31:36 -05:00
pic_add_feature(pic, "ieee-float");
#endif
2015-01-26 00:31:36 -05:00
#if _POSIX_SOURCE
pic_add_feature(pic, "posix");
#endif
#if _WIN32
pic_add_feature(pic, "windows");
#endif
#if __unix__
pic_add_feature(pic, "unix");
#endif
#if __gnu_linux__
pic_add_feature(pic, "gnu-linux");
#endif
#if __FreeBSD__
pic_add_feature(pic, "freebsd");
#endif
#if __i386__
pic_add_feature(pic, "i386");
#elif __x86_64__
pic_add_feature(pic, "x86-64");
#elif __ppc__
pic_add_feature(pic, "ppc");
#elif __sparc__
pic_add_feature(pic, "sparc");
#endif
#if __ILP32__
pic_add_feature(pic, "ilp32");
#elif __LP64__
pic_add_feature(pic, "lp64");
#endif
#if defined(__BYTE_ORDER__)
# if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
pic_add_feature(pic, "little-endian");
# elif __BYTE_ORDER__ == __ORDER_BIG_ENDIAN__
pic_add_feature(pic, "big-endian");
# endif
#else
# if __LITTLE_ENDIAN__
pic_add_feature(pic, "little-endian");
# elif __BIG_ENDIAN__
pic_add_feature(pic, "big-endian");
# endif
#endif
}
2016-02-09 11:38:05 -05:00
void
pic_add_feature(pic_state *pic, const char *feature)
{
2016-02-20 01:31:14 -05:00
pic_push(pic, pic_intern_cstr(pic, feature), pic->features);
2016-02-09 11:38:05 -05:00
}
2016-02-23 13:59:51 -05:00
#define import_builtin_syntax(name) do { \
pic_value nick, real; \
nick = pic_intern_lit(pic, "builtin:" name); \
real = pic_intern_lit(pic, name); \
pic_put_identifier(pic, nick, real, env); \
} while (0)
2016-02-09 11:38:05 -05:00
void pic_init_bool(pic_state *);
void pic_init_pair(pic_state *);
void pic_init_port(pic_state *);
void pic_init_number(pic_state *);
void pic_init_proc(pic_state *);
void pic_init_symbol(pic_state *);
void pic_init_vector(pic_state *);
void pic_init_blob(pic_state *);
void pic_init_cont(pic_state *);
void pic_init_char(pic_state *);
void pic_init_error(pic_state *);
void pic_init_str(pic_state *);
void pic_init_var(pic_state *);
void pic_init_write(pic_state *);
void pic_init_read(pic_state *);
void pic_init_dict(pic_state *);
void pic_init_record(pic_state *);
void pic_init_eval(pic_state *);
void pic_init_lib(pic_state *);
2016-02-10 07:50:39 -05:00
void pic_init_weak(pic_state *);
2016-02-09 11:38:05 -05:00
2016-03-03 04:59:07 -05:00
void pic_boot(pic_state *);
#define DONE pic_leave(pic, ai);
2016-02-09 11:38:05 -05:00
2015-01-26 00:31:36 -05:00
static void
pic_init_core(pic_state *pic)
{
2016-03-03 04:59:07 -05:00
size_t ai = pic_enter(pic);
2016-02-23 13:59:51 -05:00
pic_value env;
2015-01-26 00:31:36 -05:00
pic_deflibrary(pic, "picrin.base");
2016-02-23 13:59:51 -05:00
env = pic_library_environment(pic, pic->lib);
import_builtin_syntax("define");
import_builtin_syntax("set!");
import_builtin_syntax("quote");
import_builtin_syntax("lambda");
import_builtin_syntax("if");
import_builtin_syntax("begin");
import_builtin_syntax("define-macro");
2016-03-03 04:59:07 -05:00
pic_init_features(pic); DONE;
pic_init_bool(pic); DONE;
pic_init_pair(pic); DONE;
pic_init_port(pic); DONE;
pic_init_number(pic); DONE;
pic_init_proc(pic); DONE;
pic_init_symbol(pic); DONE;
pic_init_vector(pic); DONE;
pic_init_blob(pic); DONE;
pic_init_cont(pic); DONE;
pic_init_char(pic); DONE;
pic_init_error(pic); DONE;
pic_init_str(pic); DONE;
pic_init_var(pic); DONE;
pic_init_read(pic); DONE;
pic_init_dict(pic); DONE;
pic_init_record(pic); DONE;
pic_init_eval(pic); DONE;
pic_init_lib(pic); DONE;
pic_init_weak(pic); DONE;
2016-02-22 14:37:58 -05:00
#if PIC_USE_WRITE
pic_init_write(pic); DONE;
#endif
2016-03-03 04:59:07 -05:00
pic_boot(pic); DONE;
2015-01-26 00:31:36 -05:00
}
2014-08-25 00:38:09 -04:00
pic_state *
2015-06-25 13:09:06 -04:00
pic_open(pic_allocf allocf, void *userdata)
2014-08-25 00:38:09 -04:00
{
pic_state *pic;
2015-08-10 08:13:23 -04:00
pic = allocf(userdata, NULL, sizeof(pic_state));
2014-08-25 00:38:09 -04:00
2015-05-27 10:08:47 -04:00
if (! pic) {
goto EXIT_PIC;
}
2015-05-27 13:12:26 -04:00
/* allocator */
pic->allocf = allocf;
2015-06-25 13:09:06 -04:00
/* user data */
pic->userdata = userdata;
2015-01-18 22:14:29 -05:00
/* turn off GC */
pic->gc_enable = false;
2015-06-22 14:02:12 -04:00
/* continuation chain */
pic->cc = NULL;
2014-08-25 00:38:09 -04:00
/* prepare VM stack */
2015-08-10 08:13:23 -04:00
pic->stbase = pic->sp = allocf(userdata, NULL, PIC_STACK_SIZE * sizeof(pic_value));
2014-08-25 00:38:09 -04:00
pic->stend = pic->stbase + PIC_STACK_SIZE;
2015-05-27 10:08:47 -04:00
if (! pic->sp) {
goto EXIT_SP;
}
2014-08-25 00:38:09 -04:00
/* callinfo */
2016-02-21 06:32:00 -05:00
pic->cibase = pic->ci = allocf(userdata, NULL, PIC_STACK_SIZE * sizeof(struct callinfo));
2014-08-25 00:38:09 -04:00
pic->ciend = pic->cibase + PIC_STACK_SIZE;
2015-05-27 10:08:47 -04:00
if (! pic->ci) {
goto EXIT_CI;
}
/* GC arena */
2016-02-21 06:32:00 -05:00
pic->arena = allocf(userdata, NULL, PIC_ARENA_SIZE * sizeof(struct object *));
2015-05-27 10:08:47 -04:00
pic->arena_size = PIC_ARENA_SIZE;
pic->arena_idx = 0;
if (! pic->arena) {
goto EXIT_ARENA;
}
2014-08-25 00:38:09 -04:00
/* memory heap */
pic->heap = pic_heap_open(pic);
2014-08-25 00:38:09 -04:00
/* symbol table */
2016-02-14 22:59:58 -05:00
kh_init(oblist, &pic->oblist);
2014-08-25 00:38:09 -04:00
/* unique symbol count */
pic->ucnt = 0;
2014-08-25 00:38:09 -04:00
/* global variables */
2016-02-20 14:34:26 -05:00
pic->globals = pic_invalid_value(pic);
2014-08-25 00:38:09 -04:00
/* macros */
2016-02-20 14:34:26 -05:00
pic->macros = pic_invalid_value(pic);
2014-08-25 00:38:09 -04:00
2014-09-09 12:41:10 -04:00
/* features */
pic->features = pic_nil_value(pic);
2014-09-09 12:41:10 -04:00
2017-03-31 01:39:01 -04:00
/* dynamic environment */
pic->dyn_env = pic_invalid_value(pic);
2014-08-25 00:38:09 -04:00
/* libraries */
kh_init(ltable, &pic->ltable);
2014-08-25 00:38:09 -04:00
pic->lib = NULL;
2014-09-18 01:50:01 -04:00
/* raised error object */
2016-02-21 04:59:18 -05:00
pic->panicf = NULL;
2016-02-20 05:00:41 -05:00
pic->err = pic_invalid_value(pic);
2014-08-25 00:38:09 -04:00
/* root tables */
2016-02-10 07:50:39 -05:00
pic->globals = pic_make_weak(pic);
pic->macros = pic_make_weak(pic);
2017-03-31 01:39:01 -04:00
pic->dyn_env = pic_list(pic, 1, pic_make_weak(pic));
2016-03-01 14:34:33 -05:00
/* user land */
pic_deflibrary(pic, "picrin.user");
2014-08-25 00:38:09 -04:00
2015-01-18 22:14:29 -05:00
/* turn on GC */
pic->gc_enable = true;
pic_init_core(pic);
2014-08-25 00:38:09 -04:00
2016-03-01 14:34:33 -05:00
pic_in_library(pic, "picrin.user");
2016-02-20 14:34:26 -05:00
pic_leave(pic, 0); /* empty arena */
2015-05-27 10:08:47 -04:00
2014-08-25 00:38:09 -04:00
return pic;
2015-05-27 10:08:47 -04:00
EXIT_ARENA:
2015-08-10 08:13:23 -04:00
allocf(userdata, pic->ci, 0);
2015-05-27 10:08:47 -04:00
EXIT_CI:
2015-08-10 08:13:23 -04:00
allocf(userdata, pic->sp, 0);
2015-05-27 10:08:47 -04:00
EXIT_SP:
2015-08-10 08:13:23 -04:00
allocf(userdata, pic, 0);
2015-05-27 10:08:47 -04:00
EXIT_PIC:
return NULL;
2014-08-25 00:38:09 -04:00
}
void
pic_close(pic_state *pic)
{
2015-05-27 13:12:26 -04:00
pic_allocf allocf = pic->allocf;
2014-08-25 00:38:09 -04:00
/* clear out root objects */
pic->sp = pic->stbase;
pic->ci = pic->cibase;
pic->arena_idx = 0;
2016-02-20 05:00:41 -05:00
pic->err = pic_invalid_value(pic);
pic->globals = pic_invalid_value(pic);
pic->macros = pic_invalid_value(pic);
2017-03-31 01:39:01 -04:00
pic->features = pic_invalid_value(pic);
pic->dyn_env = pic_invalid_value(pic);
/* free all libraries */
kh_clear(ltable, &pic->ltable);
2014-08-25 00:38:09 -04:00
/* free all heap objects */
2016-02-13 23:17:12 -05:00
pic_gc(pic);
2014-08-25 00:38:09 -04:00
/* free heaps */
pic_heap_close(pic, pic->heap);
2014-08-25 00:38:09 -04:00
/* free runtime context */
2015-08-10 08:13:23 -04:00
allocf(pic->userdata, pic->stbase, 0);
allocf(pic->userdata, pic->cibase, 0);
2014-08-25 00:38:09 -04:00
/* free global stacks */
kh_destroy(oblist, &pic->oblist);
kh_destroy(ltable, &pic->ltable);
2014-08-25 00:38:09 -04:00
/* free GC arena */
2015-08-10 08:13:23 -04:00
allocf(pic->userdata, pic->arena, 0);
2014-08-25 00:38:09 -04:00
2015-08-10 08:13:23 -04:00
allocf(pic->userdata, pic, 0);
2014-08-25 00:38:09 -04:00
}
2017-03-28 11:03:23 -04:00
pic_value
pic_global_ref(pic_state *pic, pic_value uid)
{
pic_value val;
if (! pic_weak_has(pic, pic->globals, uid)) {
pic_error(pic, "undefined variable", 1, uid);
}
val = pic_weak_ref(pic, pic->globals, uid);;
if (pic_invalid_p(pic, val)) {
pic_error(pic, "uninitialized global variable", 1, uid);
}
return val;
}
void
pic_global_set(pic_state *pic, pic_value uid, pic_value value)
{
if (! pic_weak_has(pic, pic->globals, uid)) {
pic_error(pic, "undefined variable", 1, uid);
}
pic_weak_set(pic, pic->globals, uid, value);
}
pic_value
pic_ref(pic_state *pic, const char *lib, const char *name)
{
pic_value sym, env;
sym = pic_intern_cstr(pic, name);
env = pic_library_environment(pic, lib);
return pic_global_ref(pic, pic_find_identifier(pic, sym, env));
}
void
pic_set(pic_state *pic, const char *lib, const char *name, pic_value val)
{
pic_value sym, env;
sym = pic_intern_cstr(pic, name);
env = pic_library_environment(pic, lib);
pic_global_set(pic, pic_find_identifier(pic, sym, env), val);
}
void
pic_define(pic_state *pic, const char *lib, const char *name, pic_value val)
{
pic_value sym, uid, env;
sym = pic_intern_cstr(pic, name);
env = pic_library_environment(pic, lib);
uid = pic_find_identifier(pic, sym, env);
if (pic_weak_has(pic, pic->globals, uid)) {
pic_warnf(pic, "redefining variable: %s", pic_str(pic, pic_sym_name(pic, uid), NULL));
}
pic_weak_set(pic, pic->globals, uid, val);
}
void
pic_defun(pic_state *pic, const char *name, pic_func_t f)
{
pic_define(pic, pic_current_library(pic), name, pic_make_proc(pic, f, 0, NULL));
pic_export(pic, pic_intern_cstr(pic, name));
}
void
pic_defvar(pic_state *pic, const char *name, pic_value init)
{
pic_define(pic, pic_current_library(pic), name, pic_make_var(pic, init, pic_false_value(pic)));
pic_export(pic, pic_intern_cstr(pic, name));
}
pic_value
pic_funcall(pic_state *pic, const char *lib, const char *name, int n, ...)
{
pic_value proc, r;
va_list ap;
proc = pic_ref(pic, lib, name);
TYPE_CHECK(pic, proc, proc);
va_start(ap, n);
r = pic_vcall(pic, proc, n, ap);
va_end(ap);
return r;
}