/** * See Copyright Notice in picrin.h */ #include "picrin.h" void pic_set_argv(pic_state *pic, int argc, char *argv[], char **envp) { pic->argc = argc; pic->argv = argv; pic->envp = envp; } void pic_add_feature(pic_state *pic, const char *feature) { pic_push(pic, pic_obj_value(pic_intern(pic, feature)), pic->features); } 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_macro(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 *); void pic_init_reg(pic_state *); extern const char pic_boot[][80]; static void pic_init_features(pic_state *pic) { pic_add_feature(pic, "picrin"); #if __STDC_IEC_559__ pic_add_feature(pic, "ieee-float"); #endif #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 } static pic_value pic_features(pic_state *pic) { pic_get_args(pic, ""); return pic->features; } #define DONE pic_gc_arena_restore(pic, ai); #define define_builtin_syntax(uid, name) \ pic_put_variable(pic, pic->lib->env, pic_obj_value(pic_intern(pic, name)), uid) #define VM(uid, name) \ pic_put_variable(pic, pic->lib->env, pic_obj_value(pic_intern(pic, name)), uid) #define VM3(name) \ pic->c##name = pic_vm_gref_slot(pic, pic->u##name); #define VM2(proc, name) \ proc = pic_ref(pic, pic->lib, name) static void pic_init_core(pic_state *pic) { struct pic_box *pic_vm_gref_slot(pic_state *, pic_sym *); pic_init_features(pic); pic_deflibrary (pic, "(picrin base)") { size_t ai = pic_gc_arena_preserve(pic); define_builtin_syntax(pic->uDEFINE, "builtin:define"); define_builtin_syntax(pic->uSETBANG, "builtin:set!"); define_builtin_syntax(pic->uQUOTE, "builtin:quote"); define_builtin_syntax(pic->uLAMBDA, "builtin:lambda"); define_builtin_syntax(pic->uIF, "builtin:if"); define_builtin_syntax(pic->uBEGIN, "builtin:begin"); define_builtin_syntax(pic->uDEFINE_MACRO, "builtin:define-macro"); pic_defun(pic, "features", pic_features); VM(pic->uCONS, "cons"); VM(pic->uCAR, "car"); VM(pic->uCDR, "cdr"); VM(pic->uNILP, "null?"); VM(pic->uSYMBOLP, "symbol?"); VM(pic->uPAIRP, "pair?"); VM(pic->uNOT, "not"); VM(pic->uADD, "+"); VM(pic->uSUB, "-"); VM(pic->uMUL, "*"); VM(pic->uDIV, "/"); VM(pic->uEQ, "="); VM(pic->uLT, "<"); VM(pic->uLE, "<="); VM(pic->uGT, ">"); VM(pic->uGE, ">="); 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_macro(pic); DONE; pic_init_var(pic); DONE; pic_init_write(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_reg(pic); DONE; VM3(CONS); VM3(CAR); VM3(CDR); VM3(NILP); VM3(SYMBOLP); VM3(PAIRP); VM3(NOT); VM3(ADD); VM3(SUB); VM3(MUL); VM3(DIV); VM3(EQ); VM3(LT); VM3(LE); VM3(GT); VM3(GE); VM2(pic->pCONS, "cons"); VM2(pic->pCAR, "car"); VM2(pic->pCDR, "cdr"); VM2(pic->pNILP, "null?"); VM2(pic->pSYMBOLP, "symbol?"); VM2(pic->pPAIRP, "pair?"); VM2(pic->pNOT, "not"); VM2(pic->pADD, "+"); VM2(pic->pSUB, "-"); VM2(pic->pMUL, "*"); VM2(pic->pDIV, "/"); VM2(pic->pEQ, "="); VM2(pic->pLT, "<"); VM2(pic->pLE, "<="); VM2(pic->pGT, ">"); VM2(pic->pGE, ">="); pic_try { pic_load_cstr(pic, &pic_boot[0][0]); } pic_catch { pic_print_backtrace(pic, xstdout); pic_panic(pic, ""); } } } pic_state * pic_open(pic_allocf allocf, void *userdata) { struct pic_port *pic_make_standard_port(pic_state *, xFILE *, short); char t; pic_state *pic; size_t ai; pic = allocf(userdata, NULL, sizeof(pic_state)); if (! pic) { goto EXIT_PIC; } /* allocator */ pic->allocf = allocf; /* user data */ pic->userdata = userdata; /* turn off GC */ pic->gc_enable = false; /* continuation chain */ pic->cc = NULL; pic->ccnt = 0; /* root block */ pic->cp = NULL; /* command line */ pic->argc = 0; pic->argv = NULL; pic->envp = NULL; /* prepare VM stack */ pic->stbase = pic->sp = allocf(userdata, NULL, PIC_STACK_SIZE * sizeof(pic_value)); pic->stend = pic->stbase + PIC_STACK_SIZE; if (! pic->sp) { goto EXIT_SP; } /* callinfo */ pic->cibase = pic->ci = allocf(userdata, NULL, PIC_STACK_SIZE * sizeof(pic_callinfo)); pic->ciend = pic->cibase + PIC_STACK_SIZE; if (! pic->ci) { goto EXIT_CI; } /* exception handler */ pic->xpbase = pic->xp = allocf(userdata, NULL, PIC_RESCUE_SIZE * sizeof(struct pic_proc *)); pic->xpend = pic->xpbase + PIC_RESCUE_SIZE; if (! pic->xp) { goto EXIT_XP; } /* GC arena */ pic->arena = allocf(userdata, NULL, PIC_ARENA_SIZE * sizeof(struct pic_object *)); pic->arena_size = PIC_ARENA_SIZE; pic->arena_idx = 0; if (! pic->arena) { goto EXIT_ARENA; } /* memory heap */ pic->heap = pic_heap_open(pic); /* symbol table */ kh_init(s, &pic->syms); /* unique symbol count */ pic->ucnt = 0; /* global variables */ pic->globals = NULL; /* macros */ pic->macros = NULL; /* features */ pic->features = pic_nil_value(); /* libraries */ pic->libs = pic_nil_value(); pic->lib = NULL; /* ireps */ pic->ireps.next = &pic->ireps; pic->ireps.prev = &pic->ireps; /* raised error object */ pic->err = pic_invalid_value(); /* file pool */ memset(pic->files, 0, sizeof pic->files); /* parameter table */ pic->ptable = pic_nil_value(); /* native stack marker */ pic->native_stack_start = &t; ai = pic_gc_arena_preserve(pic); #define S(slot,name) pic->slot = pic_intern(pic, name) S(sQUOTE, "quote"); S(sQUASIQUOTE, "quasiquote"); S(sUNQUOTE, "unquote"); S(sUNQUOTE_SPLICING, "unquote-splicing"); S(sSYNTAX_QUOTE, "syntax-quote"); S(sSYNTAX_QUASIQUOTE, "syntax-quasiquote"); S(sSYNTAX_UNQUOTE, "syntax-unquote"); S(sSYNTAX_UNQUOTE_SPLICING, "syntax-unquote-splicing"); S(sIMPORT, "import"); S(sEXPORT, "export"); S(sDEFINE_LIBRARY, "define-library"); S(sCOND_EXPAND, "cond-expand"); pic_gc_arena_restore(pic, ai); #define U(slot,name) pic->slot = pic_uniq(pic, pic_obj_value(pic_intern(pic, name))) U(uDEFINE, "define"); U(uLAMBDA, "lambda"); U(uIF, "if"); U(uBEGIN, "begin"); U(uSETBANG, "set!"); U(uQUOTE, "quote"); U(uDEFINE_MACRO, "define-macro"); U(uIMPORT, "import"); U(uEXPORT, "export"); U(uDEFINE_LIBRARY, "define-library"); U(uCOND_EXPAND, "cond-expand"); U(uCONS, "cons"); U(uCAR, "car"); U(uCDR, "cdr"); U(uNILP, "null?"); U(uSYMBOLP, "symbol?"); U(uPAIRP, "pair?"); U(uADD, "+"); U(uSUB, "-"); U(uMUL, "*"); U(uDIV, "/"); U(uEQ, "="); U(uLT, "<"); U(uLE, "<="); U(uGT, ">"); U(uGE, ">="); U(uNOT, "not"); pic_gc_arena_restore(pic, ai); /* system procedures */ pic->pCONS = pic_invalid_value(); pic->pCAR = pic_invalid_value(); pic->pCDR = pic_invalid_value(); pic->pNILP = pic_invalid_value(); pic->pSYMBOLP = pic_invalid_value(); pic->pPAIRP = pic_invalid_value(); pic->pNOT = pic_invalid_value(); pic->pADD = pic_invalid_value(); pic->pSUB = pic_invalid_value(); pic->pMUL = pic_invalid_value(); pic->pDIV = pic_invalid_value(); pic->pEQ = pic_invalid_value(); pic->pLT = pic_invalid_value(); pic->pLE = pic_invalid_value(); pic->pGT = pic_invalid_value(); pic->pGE = pic_invalid_value(); /* root tables */ pic->globals = pic_make_reg(pic); pic->macros = pic_make_reg(pic); /* root block */ pic->cp = (pic_checkpoint *)pic_obj_alloc(pic, sizeof(pic_checkpoint), PIC_TT_CP); pic->cp->prev = NULL; pic->cp->depth = 0; pic->cp->in = pic->cp->out = NULL; /* reader */ pic_reader_init(pic); /* parameter table */ pic->ptable = pic_cons(pic, pic_obj_value(pic_make_reg(pic)), pic->ptable); /* standard libraries */ pic->PICRIN_BASE = pic_make_library(pic, pic_read_cstr(pic, "(picrin base)")); pic->PICRIN_USER = pic_make_library(pic, pic_read_cstr(pic, "(picrin user)")); pic->lib = pic->PICRIN_USER; pic->prev_lib = NULL; pic_gc_arena_restore(pic, ai); /* turn on GC */ pic->gc_enable = true; pic->cCONS = pic_box(pic, pic_invalid_value()); pic->cCAR = pic_box(pic, pic_invalid_value()); pic->cCDR = pic_box(pic, pic_invalid_value()); pic->cNILP = pic_box(pic, pic_invalid_value()); pic->cSYMBOLP = pic_box(pic, pic_invalid_value()); pic->cPAIRP = pic_box(pic, pic_invalid_value()); pic->cNOT = pic_box(pic, pic_invalid_value()); pic->cADD = pic_box(pic, pic_invalid_value()); pic->cSUB = pic_box(pic, pic_invalid_value()); pic->cMUL = pic_box(pic, pic_invalid_value()); pic->cDIV = pic_box(pic, pic_invalid_value()); pic->cEQ = pic_box(pic, pic_invalid_value()); pic->cLT = pic_box(pic, pic_invalid_value()); pic->cLE = pic_box(pic, pic_invalid_value()); pic->cGT = pic_box(pic, pic_invalid_value()); pic->cGE = pic_box(pic, pic_invalid_value()); pic_init_core(pic); pic_gc_arena_restore(pic, ai); return pic; EXIT_ARENA: allocf(userdata, pic->xp, 0); EXIT_XP: allocf(userdata, pic->ci, 0); EXIT_CI: allocf(userdata, pic->sp, 0); EXIT_SP: allocf(userdata, pic, 0); EXIT_PIC: return NULL; } void pic_close(pic_state *pic) { khash_t(s) *h = &pic->syms; pic_allocf allocf = pic->allocf; /* clear out root objects */ pic->sp = pic->stbase; pic->ci = pic->cibase; pic->xp = pic->xpbase; pic->arena_idx = 0; pic->err = pic_invalid_value(); pic->globals = NULL; pic->macros = NULL; pic->features = pic_nil_value(); pic->libs = pic_nil_value(); /* free all heap objects */ pic_gc_run(pic); #if 0 { /* FIXME */ int i = 0; struct pic_list *list; for (list = pic->ireps.next; list != &pic->ireps; list = list->next) { i++; } printf("%d\n", i); } #endif /* flush all xfiles */ xfflush(pic, NULL); /* free heaps */ pic_heap_close(pic, pic->heap); /* free reader struct */ pic_reader_destroy(pic); /* free runtime context */ allocf(pic->userdata, pic->stbase, 0); allocf(pic->userdata, pic->cibase, 0); allocf(pic->userdata, pic->xpbase, 0); /* free global stacks */ kh_destroy(s, h); /* free GC arena */ allocf(pic->userdata, pic->arena, 0); allocf(pic->userdata, pic, 0); }