Asm.js closer still.

This commit is contained in:
Brad Nelson
2022-07-12 23:36:45 -07:00
parent 550050d9c3
commit 422507c072
2 changed files with 213 additions and 53 deletions

View File

@ -179,7 +179,7 @@ static cell_t *evaluate1(cell_t *rp) {
if (len == 0) { DUP; tos = 0; PARK; return rp; } // ignore empty if (len == 0) { DUP; tos = 0; PARK; return rp; } // ignore empty
cell_t xt = find((const char *) name, len); cell_t xt = find((const char *) name, len);
if (xt) { if (xt) {
if (g_sys->state && !(((cell_t *) xt)[-1] & IMMEDIATE)) { if (g_sys->state && !(*TOFLAGS(xt) & IMMEDIATE)) {
COMMA(xt); COMMA(xt);
} else { } else {
call = xt; call = xt;
@ -210,7 +210,7 @@ static cell_t *evaluate1(cell_t *rp) {
#endif #endif
PUSH name; PUSH name;
PUSH len; PUSH len;
*++sp = -1; PUSH -1;
call = g_sys->notfound; call = g_sys->notfound;
} }
} }

View File

@ -28,6 +28,7 @@ const BUILTIN_MARK = 8;
{{boot}} {{boot}}
var heap = new ArrayBuffer(HEAP_SIZE); var heap = new ArrayBuffer(HEAP_SIZE);
var f32 = new Float32Array(heap);
var i32 = new Int32Array(heap); var i32 = new Int32Array(heap);
var u16 = new Uint16Array(heap); var u16 = new Uint16Array(heap);
var u8 = new Uint8Array(heap); var u8 = new Uint8Array(heap);
@ -58,23 +59,6 @@ function Load(addr, content) {
return addr; return addr;
} }
function UPPER(a) {
// a = 97, z = 122
return a >= 97 && a <= 122 ? a & 95 : a;
}
function Same(a, b) {
if (a.length != b.length) {
return false;
}
for (var i = 0; i < a.length; ++i) {
if (UPPER(a.charCodeAt(i)) != UPPER(b.charCodeAt(i))) {
return false;
}
}
return true;
}
function GetString(a, n) { function GetString(a, n) {
var ret = ''; var ret = '';
for (var i = 0; i < n; ++i) { for (var i = 0; i < n; ++i) {
@ -85,14 +69,17 @@ function GetString(a, n) {
function CELL_ALIGNED(n) { return (n + 3) & ~3; } function CELL_ALIGNED(n) { return (n + 3) & ~3; }
function UPPER(ch) {
return ch >= 'a'.charCodeAt(0) && ch <= 'z'.charCodeAt(0) ? (ch & 0x5F) : ch;
}
function TOFLAGS(xt) { return xt - 4; } function TOFLAGS(xt) { return xt - 1 * 4; }
function TONAMELEN(xt) { return xt + 1; } function TONAMELEN(xt) { return xt + 1; }
function TOPARAMS(xt) { return TOFLAGS(xt) + 2; } function TOPARAMS(xt) { return TOFLAGS(xt) + 2; }
function TOSIZE(xt) { return CELL_ALIGNED(u8[TONAMELEN(xt)>>2]) + 4 * i32[TOPARAMS(xt)>>2]; } function TOSIZE(xt) { return CELL_ALIGNED(u8[TONAMELEN(xt)>>2]) + 4 * i32[TOPARAMS(xt)>>2]; }
function TOLINK(xt) { return xt - 2; } function TOLINK(xt) { return xt - 2 * 4; }
function TONAME(xt) { function TONAME(xt) {
return (i32[TOFLAGS(xt)] & BUILTIN_MARK) return (i32[TOFLAGS(xt)>>2] & BUILTIN_MARK)
? u8[TOLINK(xt)] : TOLINK(xt) - CELL_ALIGNED(u8[TONAMELEN(xt)]); ? u8[TOLINK(xt)] : TOLINK(xt) - CELL_ALIGNED(u8[TONAMELEN(xt)]);
} }
function TOBODY(xt) { function TOBODY(xt) {
@ -109,7 +96,7 @@ function BUILTIN_FLAGS(i) {
return u8[BUILTIN_ITEM(i) + 1 * 4 + 0]; return u8[BUILTIN_ITEM(i) + 1 * 4 + 0];
} }
function BUILTIN_NAMELEN(i) { function BUILTIN_NAMELEN(i) {
return i32[BUILTIN_ITEM(i) + 1 * 4 + 1]; return u8[BUILTIN_ITEM(i) + 1 * 4 + 1];
} }
function BUILTIN_VOCAB(i) { function BUILTIN_VOCAB(i) {
return u16[(BUILTIN_ITEM(i) + 1 * 4 + 2)>>1]; return u16[(BUILTIN_ITEM(i) + 1 * 4 + 2)>>1];
@ -127,14 +114,14 @@ function Find(name) {
for (var i = 0; BUILTIN_NAME(i); ++i) { for (var i = 0; BUILTIN_NAME(i); ++i) {
if (BUILTIN_VOCAB(i) === vocab && if (BUILTIN_VOCAB(i) === vocab &&
name.length === BUILTIN_NAMELEN(i) && name.length === BUILTIN_NAMELEN(i) &&
name === GetString(BUILTIN_NAME(i), name.length)) { name.toUpperCase() === GetString(BUILTIN_NAME(i), name.length).toUpperCase()) {
return BUILTIN_CODE(i); return BUILTIN_CODE(i);
} }
} }
} }
if (!(u8[TOFLAGS(xt)] & SMUDGE) && if (!(u8[TOFLAGS(xt)] & SMUDGE) &&
name.length === u8[TONAMELEN(xt)] && name.length === u8[TONAMELEN(xt)] &&
name === GetString(TONAME(xt), name.length)) { name.toUpperCase() === GetString(TONAME(xt), name.length).toUpperCase()) {
return xt; return xt;
} }
xt = i32[TOLINK(xt)>>2]; xt = i32[TOLINK(xt)>>2];
@ -154,7 +141,20 @@ function CCOMMA(value) {
} }
function Finish() { function Finish() {
// TODO if (i32[g_sys_latestxt>>2] && !i32[TOPARAMS(i32[g_sys_latestxt>>2])>>2]) {
var sz = i32[g_sys_heap>>2] - (g_sys_latestxt + 4);
if (sz < 0 || sz > 0xffff) { sz = 0xffff; }
i32[TOPARAMS(i32[g_sys_latestxt>>2])>>2] = sz;
}
}
function UNSMUDGE() {
u8[TOFLAGS(i32[i32[g_sys_current>>2]>>2])] &= ~SMUDGE;
Finish();
}
function DOIMMEDIATE() {
u8[TOFLAGS(i32[i32[g_sys_current>>2]>>2])] |= IMMEDIATE;
} }
function Create(name, flags, op) { function Create(name, flags, op) {
@ -192,6 +192,138 @@ function SetupBuiltins() {
COMMA(0); COMMA(0);
} }
function Match(sep, ch) {
return sep == ch || (sep == 32 && (ch == 9 || ch == 10 || ch == 13));
}
function Parse(sep, ret) {
if (sep == 32) {
while (i32[g_sys_tin>>2] < i32[g_sys_ntib>>2] &&
Match(sep, u8[i32[g_sys_tib>>2] + i32[g_sys_tin>>2]])) { ++i32[g_sys_tin>>2]; }
}
i32[ret>>2] = i32[g_sys_tib>>2] + i32[g_sys_tin>>2];
while (i32[g_sys_tin>>2] < i32[g_sys_ntib>>2] &&
!Match(sep, u8[i32[g_sys_tib>>2] + i32[g_sys_tin>>2]])) { ++i32[g_sys_tin>>2]; }
var len = i32[g_sys_tin>>2] - (i32[ret>>2] - i32[g_sys_tib>>2]);
console.log('PARSE: ' + GetString(i32[ret>>2], len));
return len;
}
function Convert(pos, n, base, ret) {
i32[ret>>2] = 0;
var negate = 0;
if (!n) { return 0; }
if (u8[pos] == '-'.charCodeAt(0)) { negate = -1; ++pos; --n; }
if (u8[pos] == '$'.charCodeAt(0)) { base = 16; ++pos; --n; }
for (; n; --n) {
var d = UPPER(u8[pos]) - 48;
if (d > 9) {
d -= 7;
if (d < 10) { return 0; }
}
if (d >= base) { return 0; }
i32[ret>>2] = i32[ret>>2] * base + d;
++pos;
}
if (negate) { i32[ret>>2] = -i32[ret>>2]; }
return -1;
}
function FConvert(pos, n, ret) {
f32[ret>>2] = 0;
var negate = 0;
var has_dot = 0;
var exp = 0;
var shift = 1;
if (!n) { return 0; }
if (u8[pos] == '-'.charCodeAt(0)) { negate = -1; ++pos; --n; }
for (; n; --n) {
if (u8[pos] >= 48 && u8[pos] <= 48 + 9) {
if (has_dot) {
shift = shift * 0.1;
f32[ret>>2] = f32[ret>>2] + (u8[pos] - 48) * shift;
} else {
f32[ret>>2] = f32[ret>>2] * 10 + (u8[pos] - 48);
}
} else if (u8[pos] == 'e'.charCodeAt(0) || u8[pos] == 'E'.charCodeAt(0)) {
break;
} else if (u8[os] == '.'.charCodeAt(0)) {
if (has_dot) { return 0; }
has_dot = -1;
} else {
return 0;
}
++pos;
}
if (!n) { return 0; } // must have E
++pos; --n;
if (n) {
var tmp = f32[ret>>2];
if (!Convert(pos, n, 10, ret)) { return 0; }
exp = i32[ret>>2];
f32[ret>>2] = tmp;
}
if (exp < -128 || exp > 128) { return 0; }
for (; exp < 0; ++exp) { f32[ret>>2] *= 0.1; }
for (; exp > 0; --exp) { f32[ret>>2] *= 10.0; }
if (negate) { f32[ret>>2] = -f32[ret>>2]; }
return -1;
}
function Evaluate1(rp) {
var call = 0;
var tos, sp, ip, fp;
// UNPARK
ip = i32[rp>>2]; rp -= 4; fp = i32[rp>>2]; rp -= 4; sp = i32[rp>>2]; rp -= 4; tos = i32[sp>>2]; sp -= 4;
var name = sp + 8;
var len = Parse(32, name);
if (len == 0) { // ignore empty
sp += 4; i32[sp>>2] = tos; tos = 0;
// PARK
sp += 4; i32[sp>>2] = tos; rp += 4; i32[rp>>2] = sp; rp += 4; i32[rp>>2] = fp; rp += 4; i32[rp>>2] = ip;
return rp;
}
var xt = Find(GetString(i32[name>>2], len));
if (xt) {
if (i32[g_sys_state>>2] && !(u8[TOFLAGS(xt)] & IMMEDIATE)) {
COMMA(xt);
} else {
call = xt;
}
} else {
var n = sp + 16;
if (Convert(i32[name>>2], len, i32[g_sys_base>>2], n)) {
if (i32[g_sys_state>>2]) {
COMMA(i32[g_sys_DOLIT_XT>>2]);
COMMA(i32[n>>2]);
} else {
sp += 4; i32[sp>>2] = tos; tos = i32[n>>2];
}
} else {
if (FConvert(i32[name>>2], len, n)) {
if (i32[g_sys_state>>2]) {
COMMA(i32[g_sys_DOFLIT_XT>>2]);
f32[i32[g_sys_heap>>2]>>2] = f32[n>>2]; i32[g_sys_heap>>2] += 4;
} else {
fp += 4; f32[fp>>2] = f32[n>>2];
}
} else {
console.log('CANT FIND: ' + GetString(i32[name>>2], len));
sp += 4; i32[sp>>2] = i32[name>>2];
sp += 4; i32[sp>>2] = len;
sp += 4; i32[sp>>2] = -1;
call = i32[g_sys_notfound>>2];
}
}
}
sp += 4; i32[sp>>2] = call;
// PARK
sp += 4; i32[sp>>2] = tos; rp += 4; i32[rp>>2] = sp; rp += 4; i32[rp>>2] = fp; rp += 4; i32[rp>>2] = ip;
return rp;
}
function InitDictionary() { function InitDictionary() {
{{dict}} {{dict}}
SetupBuiltins(); SetupBuiltins();
@ -222,9 +354,9 @@ function Init() {
for (var i = 0; i < VOCABULARY_DEPTH; ++i) { COMMA(0); } for (var i = 0; i < VOCABULARY_DEPTH; ++i) { COMMA(0); }
// setup boot text. // setup boot text.
var source = g_sys_heap; var source = i32[g_sys_heap>>2];
i32[g_sys_heap>>2] = Load(i32[g_sys_heap>>2], boot); i32[g_sys_heap>>2] = Load(i32[g_sys_heap>>2], boot);
var source_len = g_sys_heap - source; var source_len = i32[g_sys_heap>>2] - source;
i32[g_sys_boot>>2] = source; i32[g_sys_boot>>2] = source;
i32[g_sys_boot_size>>2] = source_len; i32[g_sys_boot_size>>2] = source_len;
@ -247,6 +379,7 @@ function Init() {
i32[g_sys_base>>2] = 10; i32[g_sys_base>>2] = 10;
i32[g_sys_tib>>2] = source; i32[g_sys_tib>>2] = source;
i32[g_sys_ntib>>2] = source_len; i32[g_sys_ntib>>2] = source_len;
i32[g_sys_ntib>>2] = source_len;
rp += 4; i32[rp>>2] = fp; rp += 4; i32[rp>>2] = fp;
rp += 4; i32[rp>>2] = sp; rp += 4; i32[rp>>2] = sp;
@ -282,8 +415,6 @@ function VM(stdlib, foreign, heap) {
var create = foreign.create; var create = foreign.create;
var find = foreign.find; var find = foreign.find;
var parse = foreign.parse; var parse = foreign.parse;
var memset = foreign.memset;
var memmove = foreign.memmove;
var convert = foreign.convert; var convert = foreign.convert;
var fconvert = foreign.fconvert; var fconvert = foreign.fconvert;
var evaluate1 = foreign.evaluate1; var evaluate1 = foreign.evaluate1;
@ -296,6 +427,40 @@ function VM(stdlib, foreign, heap) {
{{sys}} {{sys}}
function memset(dst, ch, n) {
dst = dst | 0;
ch = ch | 0;
n = n | 0;
while (n | 0) {
u8[dst] = ch;
dst = (dst + 1) | 0;
n = (n - 1) | 0;
}
}
function memmove(src, dst, n) {
src = src | 0;
dst = dst | 0;
n = n | 0;
if ((src | 0) < (dst | 0)) {
src = (src + n - 1) | 0;
dst = (dst + n - 1) | 0;
while (n | 0) {
u8[dst] = u8[src];
src = (src - 1) | 0;
dst = (dst - 1) | 0;
n = (n - 1) | 0;
}
} else {
while (n | 0) {
u8[dst] = u8[src];
src = (src + 1) | 0;
dst = (dst + 1) | 0;
n = (n - 1) | 0;
}
}
}
function run() { function run() {
var tos = 0; var tos = 0;
var ip = 0; var ip = 0;
@ -314,11 +479,11 @@ function VM(stdlib, foreign, heap) {
tos = i32[sp>>2]|0; sp = (sp - 4)|0; tos = i32[sp>>2]|0; sp = (sp - 4)|0;
for (;;) { for (;;) {
w = i32[ip>>2]|0; w = i32[ip>>2]|0;
emitlog(ip|0); //emitlog(ip|0);
ip = (ip + 4)|0; ip = (ip + 4)|0;
decode: for (;;) { decode: for (;;) {
ir = u8[w]|0; ir = u8[w]|0;
emitlog(ir|0); //emitlog(ir|0);
switch (ir&0xff) { switch (ir&0xff) {
{{cases}} {{cases}}
default: default:
@ -332,27 +497,22 @@ function VM(stdlib, foreign, heap) {
} }
var ffi = { var ffi = {
Call: Call, 'Call': Call,
create: function() { console.log('create'); }, 'create': function(name, len, flags, op) { Create(GetString(name, len), flags, op); },
parse: function() { console.log('parse'); }, 'parse': function(sep, ret) { return Parse(sep, ret); },
COMMA: function(n) { COMMA(n); }, 'find': function(a, n) { return Find(GetString(a, n)); },
CCOMMA: function(n) { COMMA(n); }, 'convert': function(pos, n, base, ret) { return Convert(pos, n, base, ret); },
SSMOD: function() { console.log('ssmod'); }, 'fconvert': function(pos, n, ret) { return FConvert(pos, n, ret); },
DOES: function() { console.log('does'); }, 'evaluate1': function(rp) { return Evaluate1(rp); },
DOIMMEDIATE: function() { console.log('immediate'); }, 'log': function(n) { console.log(n); },
UNSMUDGE: function() { console.log('unsmudge'); }, 'COMMA': function(n) { COMMA(n); },
parse: function() { console.log('parse'); }, 'CCOMMA': function(n) { COMMA(n); },
find: function() { console.log('find'); }, 'SSMOD': function() { console.log('ssmod'); },
memmove: function() { console.log('memmove'); }, 'DOES': function() { console.log('does'); },
memset: function() { console.log('memset'); }, 'DOIMMEDIATE': function() { DOIMMEDIATE(); },
convert: function() { console.log('convert'); }, 'UNSMUDGE': function() { UNSMUDGE(); },
fconvert: function() { console.log('fconvert'); },
evaluate1: function() { console.log('evaluate1'); },
log: function(n) { console.log(n); }
}; };
heap[128 + 6] = 256 * 4; // set g_sys.heap = 256 * 4;
function getGlobalObj() { function getGlobalObj() {
return (function(g) { return (function(g) {
return g; return g;