From 95dc19fe680b628e5733b5be5899a851eb80745e Mon Sep 17 00:00:00 2001 From: Ico Doornekamp Date: Tue, 10 Dec 2024 16:00:04 +0100 Subject: [PATCH] Added VM context `zf_ctx` to allow multiple instances of the forth interpreter to be exist at the same time. --- src/atmega8/main.c | 20 +- src/linux/main.c | 56 ++--- src/zforth/zforth.c | 509 +++++++++++++++++++++----------------------- src/zforth/zforth.h | 44 ++-- 4 files changed, 318 insertions(+), 311 deletions(-) diff --git a/src/atmega8/main.c b/src/atmega8/main.c index cb5d9a3..55811c1 100644 --- a/src/atmega8/main.c +++ b/src/atmega8/main.c @@ -31,12 +31,14 @@ int main(void) uart_init(UART_BAUD(9600)); stdout = stdin = &f; + zf_ctx _ctx; + zf_ctx *ctx = &_ctx; /* Initialize zforth */ - zf_init(1); - zf_bootstrap(); - zf_eval(": . 1 sys ;"); + zf_init(ctx, 1); + zf_bootstrap(ctx); + zf_eval(ctx, ": . 1 sys ;"); /* Main loop: read words and eval */ @@ -47,7 +49,7 @@ int main(void) int c = getchar(); putchar(c); if(c == 10 || c == 13 || c == 32) { - zf_result r = zf_eval(buf); + zf_result r = zf_eval(ctx, buf); if(r != ZF_OK) puts("A"); l = 0; } else if(l < sizeof(buf)-1) { @@ -60,19 +62,19 @@ int main(void) } -zf_input_state zf_host_sys(zf_syscall_id id, const char *input) +zf_input_state zf_host_sys(zf_ctx *ctx, zf_syscall_id id, const char *input) { char buf[16]; switch((int)id) { case ZF_SYSCALL_EMIT: - putchar((char)zf_pop()); + putchar((char)zf_pop(ctx)); fflush(stdout); break; case ZF_SYSCALL_PRINT: - itoa(zf_pop(), buf, 10); + itoa(zf_pop(ctx), buf, 10); puts(buf); break; } @@ -81,12 +83,12 @@ zf_input_state zf_host_sys(zf_syscall_id id, const char *input) } -zf_cell zf_host_parse_num(const char *buf) +zf_cell zf_host_parse_num(zf_ctx *ctx, const char *buf) { char *end; zf_cell v = strtol(buf, &end, 0); if(*end != '\0') { - zf_abort(ZF_ABORT_NOT_A_WORD); + zf_abort(ctx, ZF_ABORT_NOT_A_WORD); } return v; } diff --git a/src/linux/main.c b/src/linux/main.c index dbb44c9..f4e43c7 100644 --- a/src/linux/main.c +++ b/src/linux/main.c @@ -20,11 +20,11 @@ * Evaluate buffer with code, check return value and report errors */ -zf_result do_eval(const char *src, int line, const char *buf) +zf_result do_eval(zf_ctx *ctx, const char *src, int line, const char *buf) { const char *msg = NULL; - zf_result rv = zf_eval(buf); + zf_result rv = zf_eval(ctx, buf); switch(rv) { @@ -56,7 +56,7 @@ zf_result do_eval(const char *src, int line, const char *buf) * Load given forth file */ -void include(const char *fname) +void include(zf_ctx *ctx, const char *fname) { char buf[256]; @@ -64,7 +64,7 @@ void include(const char *fname) int line = 1; if(f) { while(fgets(buf, sizeof(buf), f)) { - do_eval(fname, line++, buf); + do_eval(ctx, fname, line++, buf); } fclose(f); } else { @@ -77,10 +77,10 @@ void include(const char *fname) * Save dictionary */ -static void save(const char *fname) +static void save(zf_ctx *ctx, const char *fname) { size_t len; - void *p = zf_dump(&len); + void *p = zf_dump(ctx, &len); FILE *f = fopen(fname, "wb"); if(f) { fwrite(p, 1, len, f); @@ -93,10 +93,10 @@ static void save(const char *fname) * Load dictionary */ -static void load(const char *fname) +static void load(zf_ctx *ctx, const char *fname) { size_t len; - void *p = zf_dump(&len); + void *p = zf_dump(ctx, &len); FILE *f = fopen(fname, "rb"); if(f) { fread(p, 1, len, f); @@ -111,7 +111,7 @@ static void load(const char *fname) * Sys callback function */ -zf_input_state zf_host_sys(zf_syscall_id id, const char *input) +zf_input_state zf_host_sys(zf_ctx *ctx, zf_syscall_id id, const char *input) { switch((int)id) { @@ -119,21 +119,21 @@ zf_input_state zf_host_sys(zf_syscall_id id, const char *input) /* The core system callbacks */ case ZF_SYSCALL_EMIT: - putchar((char)zf_pop()); + putchar((char)zf_pop(ctx)); fflush(stdout); break; case ZF_SYSCALL_PRINT: - printf(ZF_CELL_FMT " ", zf_pop()); + printf(ZF_CELL_FMT " ", zf_pop(ctx)); break; case ZF_SYSCALL_TELL: { - zf_cell len = zf_pop(); - zf_cell addr = zf_pop(); + zf_cell len = zf_pop(ctx); + zf_cell addr = zf_pop(ctx); if(addr >= ZF_DICT_SIZE - len) { - zf_abort(ZF_ABORT_OUTSIDE_MEM); + zf_abort(ctx, ZF_ABORT_OUTSIDE_MEM); } - void *buf = (uint8_t *)zf_dump(NULL) + (int)addr; + void *buf = (uint8_t *)zf_dump(ctx, NULL) + (int)addr; (void)fwrite(buf, 1, len, stdout); fflush(stdout); } break; @@ -147,18 +147,18 @@ zf_input_state zf_host_sys(zf_syscall_id id, const char *input) break; case ZF_SYSCALL_USER + 1: - zf_push(sin(zf_pop())); + zf_push(ctx, sin(zf_pop(ctx))); break; case ZF_SYSCALL_USER + 2: if(input == NULL) { return ZF_INPUT_PASS_WORD; } - include(input); + include(ctx, input); break; case ZF_SYSCALL_USER + 3: - save("zforth.save"); + save(ctx, "zforth.save"); break; default: @@ -174,7 +174,7 @@ zf_input_state zf_host_sys(zf_syscall_id id, const char *input) * Tracing output */ -void zf_host_trace(const char *fmt, va_list va) +void zf_host_trace(zf_ctx *ctx, const char *fmt, va_list va) { fprintf(stderr, "\033[1;30m"); vfprintf(stderr, fmt, va); @@ -186,13 +186,13 @@ void zf_host_trace(const char *fmt, va_list va) * Parse number */ -zf_cell zf_host_parse_num(const char *buf) +zf_cell zf_host_parse_num(zf_ctx *ctx, const char *buf) { zf_cell v; int n = 0; int r = sscanf(buf, ZF_SCAN_FMT"%n", &v, &n); if(r != 1 || buf[n] != '\0') { - zf_abort(ZF_ABORT_NOT_A_WORD); + zf_abort(ctx, ZF_ABORT_NOT_A_WORD); } return v; } @@ -247,31 +247,33 @@ int main(int argc, char **argv) argc -= optind; argv += optind; + zf_ctx *ctx = malloc(sizeof(zf_ctx)); + printf("%p\n", (void *)ctx); /* Initialize zforth */ - zf_init(trace); + zf_init(ctx, trace); /* Load dict from disk if requested, otherwise bootstrap fort * dictionary */ if(fname_load) { - load(fname_load); + load(ctx, fname_load); } else { - zf_bootstrap(); + zf_bootstrap(ctx); } /* Include files from command line */ for(i=0; i 0) { - do_eval("stdin", ++line, buf); + do_eval(ctx, "stdin", ++line, buf); printf("\n"); add_history(buf); diff --git a/src/zforth/zforth.c b/src/zforth/zforth.c index 32456d6..3ca7626 100644 --- a/src/zforth/zforth.c +++ b/src/zforth/zforth.c @@ -17,9 +17,9 @@ * is set to 0, the boundary check code will not be compiled in to reduce size */ #if ZF_ENABLE_BOUNDARY_CHECKS -#define CHECK(exp, abort) if(!(exp)) zf_abort(abort); +#define CHECK(ctx, exp, abort) if(!(exp)) zf_abort(ctx, abort); #else -#define CHECK(exp, abort) +#define CHECK(ctx, exp, abort) #endif typedef enum { @@ -61,46 +61,30 @@ static const char prim_names[] = _("##") _("&") _("|") _("^") _("<<") _(">>"); -/* Stacks and dictionary memory */ - -static zf_cell rstack[ZF_RSTACK_SIZE]; -static zf_cell dstack[ZF_DSTACK_SIZE]; -static uint8_t dict[ZF_DICT_SIZE]; - -/* State and stack and interpreter pointers */ - -static zf_input_state input_state; -static zf_addr ip; - -/* setjmp env for handling aborts */ - -static jmp_buf jmpbuf; - /* User variables are variables which are shared between forth and C. From * forth these can be accessed with @ and ! at pseudo-indices in low memory, in * C they are stored in an array of zf_addr with friendly reference names * through some macros */ -#define HERE uservar[ZF_USERVAR_HERE] /* compilation pointer in dictionary */ -#define LATEST uservar[ZF_USERVAR_LATEST] /* pointer to last compiled word */ -#define TRACE uservar[ZF_USERVAR_TRACE] /* trace enable flag */ -#define COMPILING uservar[ZF_USERVAR_COMPILING] /* compiling flag */ -#define POSTPONE uservar[ZF_USERVAR_POSTPONE] /* flag to indicate next imm word should be compiled */ -#define DSP uservar[ZF_USERVAR_DSP] /* data stack pointer */ -#define RSP uservar[ZF_USERVAR_RSP] /* return stack pointer */ +#define HERE ctx->uservar[ZF_USERVAR_HERE] /* compilation pointer in dictionary */ +#define LATEST ctx->uservar[ZF_USERVAR_LATEST] /* pointer to last compiled word */ +#define TRACE ctx->uservar[ZF_USERVAR_TRACE] /* trace enable flag */ +#define COMPILING ctx->uservar[ZF_USERVAR_COMPILING] /* compiling flag */ +#define POSTPONE ctx->uservar[ZF_USERVAR_POSTPONE] /* flag to indicate next imm word should be compiled */ +#define DSP ctx->uservar[ZF_USERVAR_DSP] /* data stack pointer */ +#define RSP ctx->uservar[ZF_USERVAR_RSP] /* return stack pointer */ static const char uservar_names[] = _("h") _("latest") _("trace") _("compiling") _("_postpone") _("dsp") _("rsp"); -static zf_addr *uservar = (zf_addr *)dict; /* Prototypes */ -static void do_prim(zf_prim prim, const char *input); -static zf_addr dict_get_cell(zf_addr addr, zf_cell *v); -static void dict_get_bytes(zf_addr addr, void *buf, size_t len); +static void do_prim(zf_ctx *ctx, zf_prim prim, const char *input); +static zf_addr dict_get_cell(zf_ctx *ctx, zf_addr addr, zf_cell *v); +static void dict_get_bytes(zf_ctx *ctx, zf_addr addr, void *buf, size_t len); /* Tracing functions. If disabled, the trace() function is replaced by an empty @@ -109,19 +93,19 @@ static void dict_get_bytes(zf_addr addr, void *buf, size_t len); #if ZF_ENABLE_TRACE -static void do_trace(const char *fmt, ...) +static void do_trace(zf_ctx *ctx, const char *fmt, ...) { if(TRACE) { va_list va; va_start(va, fmt); - zf_host_trace(fmt, va); + zf_host_trace(ctx, fmt, va); va_end(va); } } -#define trace(...) if(TRACE) do_trace(__VA_ARGS__) +#define trace(ctx, ...) if(TRACE) do_trace(ctx, __VA_ARGS__) -static const char *op_name(zf_addr addr) +static const char *op_name(zf_ctx *ctx, zf_addr addr) { zf_addr w = LATEST; static char name[32]; @@ -131,15 +115,15 @@ static const char *op_name(zf_addr addr) zf_cell d, link, op2; int lenflags; - p += dict_get_cell(p, &d); + p += dict_get_cell(ctx, p, &d); lenflags = d; - p += dict_get_cell(p, &link); + p += dict_get_cell(ctx, p, &link); xt = p + ZF_FLAG_LEN(lenflags); - dict_get_cell(xt, &op2); + dict_get_cell(ctx, xt, &op2); if(((lenflags & ZF_FLAG_PRIM) && addr == (zf_addr)op2) || addr == w || addr == xt) { int l = ZF_FLAG_LEN(lenflags); - dict_get_bytes(p, name, l); + dict_get_bytes(ctx, p, name, l); name[l] = '\0'; return name; } @@ -150,8 +134,8 @@ static const char *op_name(zf_addr addr) } #else -static void trace(const char *fmt, ...) { } -static const char *op_name(zf_addr addr) { return NULL; } +static void trace(zf_ctx *ctx, const char *fmt, ...) { } +static const char *op_name(zf_ctx *ctx, zf_addr addr) { return NULL; } #endif @@ -160,9 +144,9 @@ static const char *op_name(zf_addr addr) { return NULL; } * zf_eval() */ -void zf_abort(zf_result reason) +void zf_abort(zf_ctx *ctx, zf_result reason) { - longjmp(jmpbuf, reason); + longjmp(ctx->jmpbuf, reason); } @@ -171,52 +155,52 @@ void zf_abort(zf_result reason) * Stack operations. */ -void zf_push(zf_cell v) +void zf_push(zf_ctx *ctx, zf_cell v) { - CHECK(DSP < ZF_DSTACK_SIZE, ZF_ABORT_DSTACK_OVERRUN); - trace("»" ZF_CELL_FMT " ", v); - dstack[DSP++] = v; + CHECK(ctx, DSP < ZF_DSTACK_SIZE, ZF_ABORT_DSTACK_OVERRUN); + trace(ctx, "»" ZF_CELL_FMT " ", v); + ctx->dstack[DSP++] = v; } -zf_cell zf_pop(void) +zf_cell zf_pop(zf_ctx *ctx) { zf_cell v; - CHECK(DSP > 0, ZF_ABORT_DSTACK_UNDERRUN); - v = dstack[--DSP]; - trace("«" ZF_CELL_FMT " ", v); + CHECK(ctx, DSP > 0, ZF_ABORT_DSTACK_UNDERRUN); + v = ctx->dstack[--DSP]; + trace(ctx, "«" ZF_CELL_FMT " ", v); return v; } -zf_cell zf_pick(zf_addr n) +zf_cell zf_pick(zf_ctx *ctx, zf_addr n) { - CHECK(n < DSP, ZF_ABORT_DSTACK_UNDERRUN); - return dstack[DSP-n-1]; + CHECK(ctx, n < DSP, ZF_ABORT_DSTACK_UNDERRUN); + return ctx->dstack[DSP-n-1]; } -static void zf_pushr(zf_cell v) +static void zf_pushr(zf_ctx *ctx, zf_cell v) { - CHECK(RSP < ZF_RSTACK_SIZE, ZF_ABORT_RSTACK_OVERRUN); - trace("r»" ZF_CELL_FMT " ", v); - rstack[RSP++] = v; + CHECK(ctx, RSP < ZF_RSTACK_SIZE, ZF_ABORT_RSTACK_OVERRUN); + trace(ctx, "r»" ZF_CELL_FMT " ", v); + ctx->rstack[RSP++] = v; } -static zf_cell zf_popr(void) +static zf_cell zf_popr(zf_ctx *ctx) { zf_cell v; - CHECK(RSP > 0, ZF_ABORT_RSTACK_UNDERRUN); - v = rstack[--RSP]; - trace("r«" ZF_CELL_FMT " ", v); + CHECK(ctx, RSP > 0, ZF_ABORT_RSTACK_UNDERRUN); + v = ctx->rstack[--RSP]; + trace(ctx, "r«" ZF_CELL_FMT " ", v); return v; } -zf_cell zf_pickr(zf_addr n) +zf_cell zf_pickr(zf_ctx *ctx, zf_addr n) { - CHECK(n < RSP, ZF_ABORT_RSTACK_UNDERRUN); - return rstack[RSP-n-1]; + CHECK(ctx, n < RSP, ZF_ABORT_RSTACK_UNDERRUN); + return ctx->rstack[RSP-n-1]; } @@ -225,21 +209,21 @@ zf_cell zf_pickr(zf_addr n) * All access to dictionary memory is done through these functions. */ -static zf_addr dict_put_bytes(zf_addr addr, const void *buf, size_t len) +static zf_addr dict_put_bytes(zf_ctx *ctx, zf_addr addr, const void *buf, size_t len) { const uint8_t *p = (const uint8_t *)buf; size_t i = len; - CHECK(addr < ZF_DICT_SIZE-len, ZF_ABORT_OUTSIDE_MEM); - while(i--) dict[addr++] = *p++; + CHECK(ctx, addr < ZF_DICT_SIZE-len, ZF_ABORT_OUTSIDE_MEM); + while(i--) ctx->dict[addr++] = *p++; return len; } -static void dict_get_bytes(zf_addr addr, void *buf, size_t len) +static void dict_get_bytes(zf_ctx *ctx, zf_addr addr, void *buf, size_t len) { uint8_t *p = (uint8_t *)buf; - CHECK(addr < ZF_DICT_SIZE-len, ZF_ABORT_OUTSIDE_MEM); - while(len--) *p++ = dict[addr++]; + CHECK(ctx, addr < ZF_DICT_SIZE-len, ZF_ABORT_OUTSIDE_MEM); + while(len--) *p++ = ctx->dict[addr++]; } @@ -254,41 +238,41 @@ static void dict_get_bytes(zf_addr addr, void *buf, size_t len) */ #if ZF_ENABLE_TYPED_MEM_ACCESS -#define GET(s, t) if(size == s) { t v ## t; dict_get_bytes(addr, &v ## t, sizeof(t)); *v = v ## t; return sizeof(t); }; -#define PUT(s, t, val) if(size == s) { t v ## t = val; return dict_put_bytes(addr, &v ## t, sizeof(t)); } +#define GET(s, t) if(size == s) { t v ## t; dict_get_bytes(ctx, addr, &v ## t, sizeof(t)); *v = v ## t; return sizeof(t); }; +#define PUT(s, t, val) if(size == s) { t v ## t = val; return dict_put_bytes(ctx, addr, &v ## t, sizeof(t)); } #else #define GET(s, t) #define PUT(s, t, val) #endif -static zf_addr dict_put_cell_typed(zf_addr addr, zf_cell v, zf_mem_size size) +static zf_addr dict_put_cell_typed(zf_ctx *ctx, zf_addr addr, zf_cell v, zf_mem_size size) { unsigned int vi = v; uint8_t t[2]; - trace("\n+" ZF_ADDR_FMT " " ZF_ADDR_FMT, addr, (zf_addr)v); + trace(ctx, "\n+" ZF_ADDR_FMT " " ZF_ADDR_FMT, addr, (zf_addr)v); if(size == ZF_MEM_SIZE_VAR) { if((v - vi) == 0) { if(vi < 128) { - trace(" ¹"); + trace(ctx, " ¹"); t[0] = vi; - return dict_put_bytes(addr, t, 1); + return dict_put_bytes(ctx, addr, t, 1); } if(vi < 16384) { - trace(" ²"); + trace(ctx, " ²"); t[0] = (vi >> 8) | 0x80; t[1] = vi; - return dict_put_bytes(addr, t, sizeof(t)); + return dict_put_bytes(ctx, addr, t, sizeof(t)); } } } if(size == ZF_MEM_SIZE_VAR || size == ZF_MEM_SIZE_VAR_MAX) { - trace(" ⁵"); + trace(ctx, " ⁵"); t[0] = 0xff; - return dict_put_bytes(addr+0, t, 1) + - dict_put_bytes(addr+1, &v, sizeof(v)); + return dict_put_bytes(ctx, addr+0, t, 1) + + dict_put_bytes(ctx, addr+1, &v, sizeof(v)); } PUT(ZF_MEM_SIZE_CELL, zf_cell, v); @@ -299,20 +283,20 @@ static zf_addr dict_put_cell_typed(zf_addr addr, zf_cell v, zf_mem_size size) PUT(ZF_MEM_SIZE_S16, int16_t, vi); PUT(ZF_MEM_SIZE_S32, int32_t, vi); - zf_abort(ZF_ABORT_INVALID_SIZE); + zf_abort(ctx, ZF_ABORT_INVALID_SIZE); return 0; } -static zf_addr dict_get_cell_typed(zf_addr addr, zf_cell *v, zf_mem_size size) +static zf_addr dict_get_cell_typed(zf_ctx *ctx, zf_addr addr, zf_cell *v, zf_mem_size size) { uint8_t t[2]; - dict_get_bytes(addr, t, sizeof(t)); + dict_get_bytes(ctx, addr, t, sizeof(t)); if(size == ZF_MEM_SIZE_VAR) { if(t[0] & 0x80) { if(t[0] == 0xff) { - dict_get_bytes(addr+1, v, sizeof(*v)); + dict_get_bytes(ctx, addr+1, v, sizeof(*v)); return 1 + sizeof(*v); } else { *v = ((t[0] & 0x3f) << 8) + t[1]; @@ -332,7 +316,7 @@ static zf_addr dict_get_cell_typed(zf_addr addr, zf_cell *v, zf_mem_size size) GET(ZF_MEM_SIZE_S16, int16_t); GET(ZF_MEM_SIZE_S32, int32_t); - zf_abort(ZF_ABORT_INVALID_SIZE); + zf_abort(ctx, ZF_ABORT_INVALID_SIZE); return 0; } @@ -341,15 +325,15 @@ static zf_addr dict_get_cell_typed(zf_addr addr, zf_cell *v, zf_mem_size size) * Shortcut functions for cell access with variable cell size */ -static zf_addr dict_put_cell(zf_addr addr, zf_cell v) +static zf_addr dict_put_cell(zf_ctx *ctx, zf_addr addr, zf_cell v) { - return dict_put_cell_typed(addr, v, ZF_MEM_SIZE_VAR); + return dict_put_cell_typed(ctx, addr, v, ZF_MEM_SIZE_VAR); } -static zf_addr dict_get_cell(zf_addr addr, zf_cell *v) +static zf_addr dict_get_cell(zf_ctx *ctx, zf_addr addr, zf_cell *v) { - return dict_get_cell_typed(addr, v, ZF_MEM_SIZE_VAR); + return dict_get_cell_typed(ctx, addr, v, ZF_MEM_SIZE_VAR); } @@ -358,39 +342,39 @@ static zf_addr dict_get_cell(zf_addr addr, zf_cell *v) * increase the pointer */ -static void dict_add_cell_typed(zf_cell v, zf_mem_size size) +static void dict_add_cell_typed(zf_ctx *ctx, zf_cell v, zf_mem_size size) { - HERE += dict_put_cell_typed(HERE, v, size); - trace(" "); + HERE += dict_put_cell_typed(ctx, HERE, v, size); + trace(ctx, " "); } -static void dict_add_cell(zf_cell v) +static void dict_add_cell(zf_ctx *ctx, zf_cell v) { - dict_add_cell_typed(v, ZF_MEM_SIZE_VAR); + dict_add_cell_typed(ctx, v, ZF_MEM_SIZE_VAR); } -static void dict_add_op(zf_addr op) +static void dict_add_op(zf_ctx *ctx, zf_addr op) { - dict_add_cell(op); - trace("+%s ", op_name(op)); + dict_add_cell(ctx, op); + trace(ctx, "+%s ", op_name(ctx, op)); } -static void dict_add_lit(zf_cell v) +static void dict_add_lit(zf_ctx *ctx, zf_cell v) { - dict_add_op(PRIM_LIT); - dict_add_cell(v); + dict_add_op(ctx, PRIM_LIT); + dict_add_cell(ctx, v); } -static void dict_add_str(const char *s) +static void dict_add_str(zf_ctx *ctx, const char *s) { size_t l; - trace("\n+" ZF_ADDR_FMT " " ZF_ADDR_FMT " s '%s'", HERE, 0, s); + trace(ctx, "\n+" ZF_ADDR_FMT " " ZF_ADDR_FMT " s '%s'", HERE, 0, s); l = strlen(s); - HERE += dict_put_bytes(HERE, s, l); + HERE += dict_put_bytes(ctx, HERE, s, l); } @@ -398,16 +382,16 @@ static void dict_add_str(const char *s) * Create new word, adjusting HERE and LATEST accordingly */ -static void create(const char *name, int flags) +static void create(zf_ctx *ctx, const char *name, int flags) { zf_addr here_prev; - trace("\n=== create '%s'", name); + trace(ctx, "\n=== create '%s'", name); here_prev = HERE; - dict_add_cell((strlen(name)) | flags); - dict_add_cell(LATEST); - dict_add_str(name); + dict_add_cell(ctx, (strlen(name)) | flags); + dict_add_cell(ctx, LATEST); + dict_add_str(ctx, name); LATEST = here_prev; - trace("\n==="); + trace(ctx, "\n==="); } @@ -415,7 +399,7 @@ static void create(const char *name, int flags) * Find word in dictionary, returning address and execution token */ -static int find_word(const char *name, zf_addr *word, zf_addr *code) +static int find_word(zf_ctx *ctx, const char *name, zf_addr *word, zf_addr *code) { zf_addr w = LATEST; size_t namelen = strlen(name); @@ -424,11 +408,11 @@ static int find_word(const char *name, zf_addr *word, zf_addr *code) zf_cell link, d; zf_addr p = w; size_t len; - p += dict_get_cell(p, &d); - p += dict_get_cell(p, &link); + p += dict_get_cell(ctx, p, &d); + p += dict_get_cell(ctx, p, &link); len = ZF_FLAG_LEN((int)d); if(len == namelen) { - const char *name2 = (const char *)&dict[p]; + const char *name2 = (const char *)&ctx->dict[p]; if(memcmp(name, name2, len) == 0) { *word = w; *code = p + len; @@ -446,11 +430,11 @@ static int find_word(const char *name, zf_addr *word, zf_addr *code) * Set 'immediate' flag in last compiled word */ -static void make_immediate(void) +static void make_immediate(zf_ctx *ctx) { zf_cell lenflags; - dict_get_cell(LATEST, &lenflags); - dict_put_cell(LATEST, (int)lenflags | ZF_FLAG_IMMEDIATE); + dict_get_cell(ctx, LATEST, &lenflags); + dict_put_cell(ctx, LATEST, (int)lenflags | ZF_FLAG_IMMEDIATE); } @@ -458,34 +442,34 @@ static void make_immediate(void) * Inner interpreter */ -static void run(const char *input) +static void run(zf_ctx *ctx, const char *input) { - while(ip != 0) { + while(ctx->ip != 0) { zf_cell d; - zf_addr i, ip_org = ip; - zf_addr l = dict_get_cell(ip, &d); + zf_addr i, ip_org = ctx->ip; + zf_addr l = dict_get_cell(ctx, ctx->ip, &d); zf_addr code = d; - trace("\n "ZF_ADDR_FMT " " ZF_ADDR_FMT " ", ip, code); - for(i=0; iip, code); + for(i=0; iip += l; if(code <= PRIM_COUNT) { - do_prim((zf_prim)code, input); + do_prim(ctx, (zf_prim)code, input); /* If the prim requests input, restore IP so that the * next time around we call the same prim again */ - if(input_state != ZF_INPUT_INTERPRET) { - ip = ip_org; + if(ctx->input_state != ZF_INPUT_INTERPRET) { + ctx->ip = ip_org; break; } } else { - trace("%s/" ZF_ADDR_FMT " ", op_name(code), code); - zf_pushr(ip); - ip = code; + trace(ctx, "%s/" ZF_ADDR_FMT " ", op_name(ctx, code), code); + zf_pushr(ctx, ctx->ip); + ctx->ip = code; } input = NULL; @@ -497,25 +481,25 @@ static void run(const char *input) * Execute bytecode from given address */ -static void execute(zf_addr addr) +static void execute(zf_ctx *ctx, zf_addr addr) { - ip = addr; + ctx->ip = addr; RSP = 0; - zf_pushr(0); + zf_pushr(ctx, 0); - trace("\n[%s/" ZF_ADDR_FMT "] ", op_name(ip), ip); - run(NULL); + trace(ctx, "\n[%s/" ZF_ADDR_FMT "] ", op_name(ctx, ctx->ip), ctx->ip); + run(ctx, NULL); } -static zf_addr peek(zf_addr addr, zf_cell *val, int len) +static zf_addr peek(zf_ctx *ctx, zf_addr addr, zf_cell *val, int len) { if(addr < ZF_USERVAR_COUNT) { - *val = uservar[addr]; + *val = ctx->uservar[addr]; return 1; } else { - return dict_get_cell_typed(addr, val, (zf_mem_size)len); + return dict_get_cell_typed(ctx, addr, val, (zf_mem_size)len); } } @@ -525,231 +509,231 @@ static zf_addr peek(zf_addr addr, zf_cell *val, int len) * Run primitive opcode */ -static void do_prim(zf_prim op, const char *input) +static void do_prim(zf_ctx *ctx, zf_prim op, const char *input) { zf_cell d1, d2, d3; zf_addr addr, len; - trace("(%s) ", op_name(op)); + trace(ctx, "(%s) ", op_name(ctx, op)); switch(op) { case PRIM_COL: if(input == NULL) { - input_state = ZF_INPUT_PASS_WORD; + ctx->input_state = ZF_INPUT_PASS_WORD; } else { - create(input, 0); + create(ctx, input, 0); COMPILING = 1; } break; case PRIM_LTZ: - zf_push(zf_pop() < 0); + zf_push(ctx, zf_pop(ctx) < 0); break; case PRIM_SEMICOL: - dict_add_op(PRIM_EXIT); - trace("\n==="); + dict_add_op(ctx, PRIM_EXIT); + trace(ctx, "\n==="); COMPILING = 0; break; case PRIM_LIT: - ip += dict_get_cell(ip, &d1); - zf_push(d1); + ctx->ip += dict_get_cell(ctx, ctx->ip, &d1); + zf_push(ctx, d1); break; case PRIM_EXIT: - ip = zf_popr(); + ctx->ip = zf_popr(ctx); break; case PRIM_LEN: - len = zf_pop(); - addr = zf_pop(); - zf_push(peek(addr, &d1, len)); + len = zf_pop(ctx); + addr = zf_pop(ctx); + zf_push(ctx, peek(ctx, addr, &d1, len)); break; case PRIM_PEEK: - len = zf_pop(); - addr = zf_pop(); - peek(addr, &d1, len); - zf_push(d1); + len = zf_pop(ctx); + addr = zf_pop(ctx); + peek(ctx, addr, &d1, len); + zf_push(ctx, d1); break; case PRIM_POKE: - d2 = zf_pop(); - addr = zf_pop(); - d1 = zf_pop(); + d2 = zf_pop(ctx); + addr = zf_pop(ctx); + d1 = zf_pop(ctx); if(addr < ZF_USERVAR_COUNT) { - uservar[addr] = d1; + ctx->uservar[addr] = d1; break; } - dict_put_cell_typed(addr, d1, (zf_mem_size)d2); + dict_put_cell_typed(ctx, addr, d1, (zf_mem_size)d2); break; case PRIM_SWAP: - d1 = zf_pop(); d2 = zf_pop(); - zf_push(d1); zf_push(d2); + d1 = zf_pop(ctx); d2 = zf_pop(ctx); + zf_push(ctx, d1); zf_push(ctx, d2); break; case PRIM_ROT: - d1 = zf_pop(); d2 = zf_pop(); d3 = zf_pop(); - zf_push(d2); zf_push(d1); zf_push(d3); + d1 = zf_pop(ctx); d2 = zf_pop(ctx); d3 = zf_pop(ctx); + zf_push(ctx, d2); zf_push(ctx, d1); zf_push(ctx, d3); break; case PRIM_DROP: - zf_pop(); + zf_pop(ctx); break; case PRIM_DUP: - d1 = zf_pop(); - zf_push(d1); zf_push(d1); + d1 = zf_pop(ctx); + zf_push(ctx, d1); zf_push(ctx, d1); break; case PRIM_ADD: - d1 = zf_pop(); d2 = zf_pop(); - zf_push(d1 + d2); + d1 = zf_pop(ctx); d2 = zf_pop(ctx); + zf_push(ctx, d1 + d2); break; case PRIM_SYS: - d1 = zf_pop(); - input_state = zf_host_sys((zf_syscall_id)d1, input); - if(input_state != ZF_INPUT_INTERPRET) { - zf_push(d1); /* re-push id to resume */ + d1 = zf_pop(ctx); + ctx->input_state = zf_host_sys(ctx, (zf_syscall_id)d1, input); + if(ctx->input_state != ZF_INPUT_INTERPRET) { + zf_push(ctx, d1); /* re-push id to resume */ } break; case PRIM_PICK: - addr = zf_pop(); - zf_push(zf_pick(addr)); + addr = zf_pop(ctx); + zf_push(ctx, zf_pick(ctx, addr)); break; case PRIM_PICKR: - addr = zf_pop(); - zf_push(zf_pickr(addr)); + addr = zf_pop(ctx); + zf_push(ctx, zf_pickr(ctx, addr)); break; case PRIM_SUB: - d1 = zf_pop(); d2 = zf_pop(); - zf_push(d2 - d1); + d1 = zf_pop(ctx); d2 = zf_pop(ctx); + zf_push(ctx, d2 - d1); break; case PRIM_MUL: - zf_push(zf_pop() * zf_pop()); + zf_push(ctx, zf_pop(ctx) * zf_pop(ctx)); break; case PRIM_DIV: - if((d2 = zf_pop()) == 0) { - zf_abort(ZF_ABORT_DIVISION_BY_ZERO); + if((d2 = zf_pop(ctx)) == 0) { + zf_abort(ctx, ZF_ABORT_DIVISION_BY_ZERO); } - d1 = zf_pop(); - zf_push(d1 / d2); + d1 = zf_pop(ctx); + zf_push(ctx, d1 / d2); break; case PRIM_MOD: - if((int)(d2 = zf_pop()) == 0) { - zf_abort(ZF_ABORT_DIVISION_BY_ZERO); + if((int)(d2 = zf_pop(ctx)) == 0) { + zf_abort(ctx, ZF_ABORT_DIVISION_BY_ZERO); } - d1 = zf_pop(); - zf_push((int)d1 % (int)d2); + d1 = zf_pop(ctx); + zf_push(ctx, (int)d1 % (int)d2); break; case PRIM_IMMEDIATE: - make_immediate(); + make_immediate(ctx); break; case PRIM_JMP: - ip += dict_get_cell(ip, &d1); - trace("ip " ZF_ADDR_FMT "=>" ZF_ADDR_FMT, ip, (zf_addr)d1); - ip = d1; + ctx->ip += dict_get_cell(ctx, ctx->ip, &d1); + trace(ctx, "ip " ZF_ADDR_FMT "=>" ZF_ADDR_FMT, ctx->ip, (zf_addr)d1); + ctx->ip = d1; break; case PRIM_JMP0: - ip += dict_get_cell(ip, &d1); - if(zf_pop() == 0) { - trace("ip " ZF_ADDR_FMT "=>" ZF_ADDR_FMT, ip, (zf_addr)d1); - ip = d1; + ctx->ip += dict_get_cell(ctx, ctx->ip, &d1); + if(zf_pop(ctx) == 0) { + trace(ctx, "ip " ZF_ADDR_FMT "=>" ZF_ADDR_FMT, ctx->ip, (zf_addr)d1); + ctx->ip = d1; } break; case PRIM_TICK: if (COMPILING) { - ip += dict_get_cell(ip, &d1); - trace("%s/", op_name(d1)); - zf_push(d1); + ctx->ip += dict_get_cell(ctx, ctx->ip, &d1); + trace(ctx, "%s/", op_name(ctx, d1)); + zf_push(ctx, d1); } else { if (input) { - if (find_word(input,&addr,&len)) zf_push(len); - else zf_abort(ZF_ABORT_INTERNAL_ERROR); + if (find_word(ctx, input,&addr,&len)) zf_push(ctx, len); + else zf_abort(ctx, ZF_ABORT_INTERNAL_ERROR); } - else input_state = ZF_INPUT_PASS_WORD; + else ctx->input_state = ZF_INPUT_PASS_WORD; } break; case PRIM_COMMA: - d2 = zf_pop(); - d1 = zf_pop(); - dict_add_cell_typed(d1, (zf_mem_size)d2); + d2 = zf_pop(ctx); + d1 = zf_pop(ctx); + dict_add_cell_typed(ctx, d1, (zf_mem_size)d2); break; case PRIM_COMMENT: if(!input || input[0] != ')') { - input_state = ZF_INPUT_PASS_CHAR; + ctx->input_state = ZF_INPUT_PASS_CHAR; } break; case PRIM_PUSHR: - zf_pushr(zf_pop()); + zf_pushr(ctx, zf_pop(ctx)); break; case PRIM_POPR: - zf_push(zf_popr()); + zf_push(ctx, zf_popr(ctx)); break; case PRIM_EQUAL: - zf_push(zf_pop() == zf_pop()); + zf_push(ctx, zf_pop(ctx) == zf_pop(ctx)); break; case PRIM_KEY: if(input == NULL) { - input_state = ZF_INPUT_PASS_CHAR; + ctx->input_state = ZF_INPUT_PASS_CHAR; } else { - zf_push(input[0]); + zf_push(ctx, input[0]); } break; case PRIM_LITS: - ip += dict_get_cell(ip, &d1); - zf_push(ip); - zf_push(d1); - ip += d1; + ctx->ip += dict_get_cell(ctx, ctx->ip, &d1); + zf_push(ctx, ctx->ip); + zf_push(ctx, d1); + ctx->ip += d1; break; case PRIM_AND: - zf_push((zf_int)zf_pop() & (zf_int)zf_pop()); + zf_push(ctx, (zf_int)zf_pop(ctx) & (zf_int)zf_pop(ctx)); break; case PRIM_OR: - zf_push((zf_int)zf_pop() | (zf_int)zf_pop()); + zf_push(ctx, (zf_int)zf_pop(ctx) | (zf_int)zf_pop(ctx)); break; case PRIM_XOR: - zf_push((zf_int)zf_pop() ^ (zf_int)zf_pop()); + zf_push(ctx, (zf_int)zf_pop(ctx) ^ (zf_int)zf_pop(ctx)); break; case PRIM_SHL: - d1 = zf_pop(); - zf_push((zf_int)zf_pop() << (zf_int)d1); + d1 = zf_pop(ctx); + zf_push(ctx, (zf_int)zf_pop(ctx) << (zf_int)d1); break; case PRIM_SHR: - d1 = zf_pop(); - zf_push((zf_int)zf_pop() >> (zf_int)d1); + d1 = zf_pop(ctx); + zf_push(ctx, (zf_int)zf_pop(ctx) >> (zf_int)d1); break; default: - zf_abort(ZF_ABORT_INTERNAL_ERROR); + zf_abort(ctx, ZF_ABORT_INTERNAL_ERROR); break; } } @@ -760,7 +744,7 @@ static void do_prim(zf_prim op, const char *input) * deferred primitive if it requested a word from the input stream. */ -static void handle_word(const char *buf) +static void handle_word(zf_ctx *ctx, const char *buf) { zf_addr w, c = 0; int found; @@ -768,15 +752,15 @@ static void handle_word(const char *buf) /* If a word was requested by an earlier operation, resume with the new * word */ - if(input_state == ZF_INPUT_PASS_WORD) { - input_state = ZF_INPUT_INTERPRET; - run(buf); + if(ctx->input_state == ZF_INPUT_PASS_WORD) { + ctx->input_state = ZF_INPUT_INTERPRET; + run(ctx, buf); return; } /* Look up the word in the dictionary */ - found = find_word(buf, &w, &c); + found = find_word(ctx, buf, &w, &c); if(found) { @@ -784,31 +768,31 @@ static void handle_word(const char *buf) zf_cell d; int flags; - dict_get_cell(w, &d); + dict_get_cell(ctx, w, &d); flags = d; if(COMPILING && (POSTPONE || !(flags & ZF_FLAG_IMMEDIATE))) { if(flags & ZF_FLAG_PRIM) { - dict_get_cell(c, &d); - dict_add_op(d); + dict_get_cell(ctx, c, &d); + dict_add_op(ctx, d); } else { - dict_add_op(c); + dict_add_op(ctx, c); } POSTPONE = 0; } else { - execute(c); + execute(ctx, c); } } else { /* Word not found: try to convert to a number and compile or push, depending * on state */ - zf_cell v = zf_host_parse_num(buf); + zf_cell v = zf_host_parse_num(ctx, buf); if(COMPILING) { - dict_add_lit(v); + dict_add_lit(ctx, v); } else { - zf_push(v); + zf_push(ctx, v); } } } @@ -819,15 +803,15 @@ static void handle_word(const char *buf) * char to a deferred prim if it requested a character from the input stream */ -static void handle_char(char c) +static void handle_char(zf_ctx *ctx, char c) { static char buf[32]; static size_t len = 0; - if(input_state == ZF_INPUT_PASS_CHAR) { + if(ctx->input_state == ZF_INPUT_PASS_CHAR) { - input_state = ZF_INPUT_INTERPRET; - run(&c); + ctx->input_state = ZF_INPUT_INTERPRET; + run(ctx, &c); } else if(c != '\0' && !isspace(c)) { @@ -840,7 +824,7 @@ static void handle_char(char c) if(len > 0) { len = 0; - handle_word(buf); + handle_word(ctx, buf); } } } @@ -850,8 +834,9 @@ static void handle_char(char c) * Initialisation */ -void zf_init(int enable_trace) +void zf_init(zf_ctx *ctx, int enable_trace) { + ctx->uservar = (zf_addr *)ctx->dict; HERE = ZF_USERVAR_COUNT * sizeof(zf_addr); TRACE = enable_trace; LATEST = 0; @@ -868,7 +853,7 @@ void zf_init(int enable_trace) * user variables. */ -static void add_prim(const char *name, zf_prim op) +static void add_prim(zf_ctx *ctx, const char *name, zf_prim op) { int imm = 0; @@ -877,21 +862,21 @@ static void add_prim(const char *name, zf_prim op) imm = 1; } - create(name, ZF_FLAG_PRIM); - dict_add_op(op); - dict_add_op(PRIM_EXIT); - if(imm) make_immediate(); + create(ctx, name, ZF_FLAG_PRIM); + dict_add_op(ctx, op); + dict_add_op(ctx, PRIM_EXIT); + if(imm) make_immediate(ctx); } -static void add_uservar(const char *name, zf_addr addr) +static void add_uservar(zf_ctx *ctx, const char *name, zf_addr addr) { - create(name, 0); - dict_add_lit(addr); - dict_add_op(PRIM_EXIT); + create(ctx, name, 0); + dict_add_lit(ctx, addr); + dict_add_op(ctx, PRIM_EXIT); } -void zf_bootstrap(void) +void zf_bootstrap(zf_ctx *ctx) { /* Add primitives and user variables to dictionary */ @@ -899,12 +884,12 @@ void zf_bootstrap(void) zf_addr i = 0; const char *p; for(p=prim_names; *p; p+=strlen(p)+1) { - add_prim(p, (zf_prim)i++); + add_prim(ctx, p, (zf_prim)i++); } i = 0; for(p=uservar_names; *p; p+=strlen(p)+1) { - add_uservar(p, i++); + add_uservar(ctx, p, i++); } } @@ -917,13 +902,13 @@ void zf_bootstrap(void) {} * Eval forth string */ -zf_result zf_eval(const char *buf) +zf_result zf_eval(zf_ctx *ctx, const char *buf) { - zf_result r = (zf_result)setjmp(jmpbuf); + zf_result r = (zf_result)setjmp(ctx->jmpbuf); if(r == ZF_OK) { for(;;) { - handle_char(*buf); + handle_char(ctx, *buf); if(*buf == '\0') { return ZF_OK; } @@ -938,31 +923,31 @@ zf_result zf_eval(const char *buf) } -void *zf_dump(size_t *len) +void *zf_dump(zf_ctx *ctx, size_t *len) { - if(len) *len = sizeof(dict); - return dict; + if(len) *len = sizeof(ctx->dict); + return ctx->dict; } -zf_result zf_uservar_set(zf_uservar_id uv, zf_cell v) +zf_result zf_uservar_set(zf_ctx *ctx, zf_uservar_id uv, zf_cell v) { zf_result result = ZF_ABORT_INVALID_USERVAR; if (uv < ZF_USERVAR_COUNT) { - uservar[uv] = v; + ctx->uservar[uv] = v; result = ZF_OK; } return result; } -zf_result zf_uservar_get(zf_uservar_id uv, zf_cell *v) +zf_result zf_uservar_get(zf_ctx *ctx, zf_uservar_id uv, zf_cell *v) { zf_result result = ZF_ABORT_INVALID_USERVAR; if (uv < ZF_USERVAR_COUNT) { if (v != NULL) { - *v = uservar[uv]; + *v = ctx->uservar[uv]; } result = ZF_OK; } diff --git a/src/zforth/zforth.h b/src/zforth/zforth.h index 8b02e95..af8eb21 100644 --- a/src/zforth/zforth.h +++ b/src/zforth/zforth.h @@ -9,6 +9,7 @@ extern "C" #include #include #include +#include #include "zfconf.h" @@ -56,27 +57,44 @@ typedef enum { } zf_uservar_id; +typedef struct { + /* Stacks and dictionary memory */ + zf_cell rstack[ZF_RSTACK_SIZE]; + zf_cell dstack[ZF_DSTACK_SIZE]; + uint8_t dict[ZF_DICT_SIZE]; + + /* State and stack and interpreter pointers */ + zf_input_state input_state; + zf_addr ip; + + /* setjmp env for handling aborts */ + jmp_buf jmpbuf; + + zf_addr *uservar; +} zf_ctx; + + /* ZForth API functions */ -void zf_init(int trace); -void zf_bootstrap(void); -void *zf_dump(size_t *len); -zf_result zf_eval(const char *buf); -void zf_abort(zf_result reason); +void zf_init(zf_ctx *ctx, int trace); +void zf_bootstrap(zf_ctx *ctx); +void *zf_dump(zf_ctx *ctx, size_t *len); +zf_result zf_eval(zf_ctx *ctx, const char *buf); +void zf_abort(zf_ctx *ctx, zf_result reason); -void zf_push(zf_cell v); -zf_cell zf_pop(void); -zf_cell zf_pick(zf_addr n); +void zf_push(zf_ctx *ctx, zf_cell v); +zf_cell zf_pop(zf_ctx *ctx); +zf_cell zf_pick(zf_ctx *ctx, zf_addr n); -zf_result zf_uservar_set(zf_uservar_id uv, zf_cell v); -zf_result zf_uservar_get(zf_uservar_id uv, zf_cell *v); +zf_result zf_uservar_set(zf_ctx *ctx, zf_uservar_id uv, zf_cell v); +zf_result zf_uservar_get(zf_ctx *ctx, zf_uservar_id uv, zf_cell *v); /* Host provides these functions */ -zf_input_state zf_host_sys(zf_syscall_id id, const char *last_word); -void zf_host_trace(const char *fmt, va_list va); -zf_cell zf_host_parse_num(const char *buf); +zf_input_state zf_host_sys(zf_ctx *ctx, zf_syscall_id id, const char *last_word); +void zf_host_trace(zf_ctx *ctx, const char *fmt, va_list va); +zf_cell zf_host_parse_num(zf_ctx *ctx, const char *buf); #ifdef __cplusplus }