Implement more core words

This commit is contained in:
Remko Tronçon 2018-06-01 10:13:13 +02:00
parent 027caab71a
commit eb1971de73
2 changed files with 422 additions and 35 deletions

View file

@ -94,7 +94,7 @@
;; Predefined table indices ;; Predefined table indices
(define !pushIndex 1) (define !pushIndex 1)
(define !popIndex 2) (define !popIndex 2)
(define !displayIndex 3) (define !typeIndex 3)
(define !pushDataAddressIndex 4) (define !pushDataAddressIndex 4)
(define !pushDataValueIndex 5) (define !pushDataValueIndex 5)
(define !tableStartIndex 6) (define !tableStartIndex 6)
@ -102,14 +102,14 @@
(define !dictionaryLatest 0) (define !dictionaryLatest 0)
(define !dictionaryTop !dictionaryBase) (define !dictionaryTop !dictionaryBase)
(define (!def_word name f (flags 0)) (define (!def_word name f (flags 0) (idx !tableStartIndex))
(let* ((idx !tableStartIndex) (let* ((base !dictionaryTop)
(base !dictionaryTop)
(previous !dictionaryLatest) (previous !dictionaryLatest)
(name-entry-length (* (ceiling (/ (+ (string-length name) 1) 4)) 4)) (name-entry-length (* (ceiling (/ (+ (string-length name) 1) 4)) 4))
(size (+ 8 name-entry-length))) (size (+ 8 name-entry-length)))
(set! !tableStartIndex (+ !tableStartIndex 1)) (cond ((= idx !tableStartIndex)
(set! !dictionaryLatest !dictionaryTop) (set! !tableStartIndex (+ !tableStartIndex 1))
(set! !dictionaryLatest !dictionaryTop)))
(set! !dictionaryTop (+ !dictionaryTop size)) (set! !dictionaryTop (+ !dictionaryTop size))
`((elem (i32.const ,(eval idx)) ,(string->symbol f)) `((elem (i32.const ,(eval idx)) ,(string->symbol f))
(data (data
@ -148,6 +148,7 @@
(type $word (func (param i32))) (type $word (func (param i32)))
(global $tos (mut i32) (i32.const !stackBase)) (global $tos (mut i32) (i32.const !stackBase))
(global $tors (mut i32) (i32.const !returnStackBase))
(global $state (mut i32) (i32.const 0)) (global $state (mut i32) (i32.const 0))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
@ -162,6 +163,14 @@
(set_global $tos (get_local $bbtos))) (set_global $tos (get_local $bbtos)))
(!def_word "!" "$!") (!def_word "!" "$!")
;; 6.1.0070
(func $tick (param i32)
(call $word (i32.const -1))
(if (i32.eqz (i32.load (i32.const !wordBase))) (then (unreachable)))
(call $find (i32.const -1))
(drop (call $pop)))
(!def_word "'" "$tick")
;; 6.1.0090 ;; 6.1.0090
(func $star (param i32) (func $star (param i32)
(local $btos i32) (local $btos i32)
@ -182,9 +191,19 @@
(set_global $tos (get_local $btos))) (set_global $tos (get_local $btos)))
(!def_word "+" "$plus") (!def_word "+" "$plus")
;; 6.1.0130
(func $+! (param i32)
(local $addr i32)
(local $bbtos i32)
(i32.store (tee_local $addr (i32.load (i32.sub (get_global $tos) (i32.const 4))))
(i32.add (i32.load (get_local $addr))
(i32.load (tee_local $bbtos (i32.sub (get_global $tos) (i32.const 8))))))
(set_global $tos (get_local $bbtos)))
(!def_word "+!" "$+!")
;; 6.1.0140 ;; 6.1.0140
(func $plus-loop (param i32) (func $plus-loop (param i32)
(if (i32.eqz (get_global $state)) (unreachable)) (call $ensureCompiling)
(call $compilePlusLoop)) (call $compilePlusLoop))
(!def_word "+LOOP" "$plus-loop" !fImmediate) (!def_word "+LOOP" "$plus-loop" !fImmediate)
@ -210,7 +229,7 @@
;; 6.1.0180 ;; 6.1.0180
(func $.q (param i32) (func $.q (param i32)
(call $Sq (i32.const -1)) (call $Sq (i32.const -1))
(call $emitICall (i32.const 0) (i32.const !displayIndex))) (call $emitICall (i32.const 0) (i32.const !typeIndex)))
(!def_word ".\"" "$.q" !fImmediate) (!def_word ".\"" "$.q" !fImmediate)
;; 6.1.0230 ;; 6.1.0230
@ -270,6 +289,21 @@
(i32.sub (i32.load (get_local $btos)) (i32.const 1)))) (i32.sub (i32.load (get_local $btos)) (i32.const 1))))
(!def_word "1-" "$one-minus") (!def_word "1-" "$one-minus")
;; 6.1.0320
(func $2* (param i32)
(local $btos i32)
(i32.store (tee_local $btos (i32.sub (get_global $tos) (i32.const 4)))
(i32.shl (i32.load (get_local $btos)) (i32.const 1))))
(!def_word "2*" "$2*")
;; 6.1.0330
(func $2/ (param i32)
(local $btos i32)
(i32.store (tee_local $btos (i32.sub (get_global $tos) (i32.const 4)))
(i32.shr_s (i32.load (get_local $btos)) (i32.const 1))))
(!def_word "2/" "$2/")
;; 6.1.0370 ;; 6.1.0370
(func $two-drop (param i32) (func $two-drop (param i32)
(set_global $tos (i32.sub (get_global $tos) (i32.const 8)))) (set_global $tos (i32.sub (get_global $tos) (i32.const 8))))
@ -284,6 +318,31 @@
(set_global $tos (i32.add (get_global $tos) (i32.const 8)))) (set_global $tos (i32.add (get_global $tos) (i32.const 8))))
(!def_word "2DUP" "$two-dupe") (!def_word "2DUP" "$two-dupe")
;; 6.1.0400
(func $2OVER (param i32)
(i32.store (get_global $tos)
(i32.load (i32.sub (get_global $tos) (i32.const 16))))
(i32.store (i32.add (get_global $tos) (i32.const 4))
(i32.load (i32.sub (get_global $tos) (i32.const 12))))
(set_global $tos (i32.add (get_global $tos) (i32.const 8))))
(!def_word "2OVER" "$2OVER")
;; 6.1.0430
(func $2SWAP (param i32)
(local $x1 i32)
(local $x2 i32)
(set_local $x1 (i32.load (i32.sub (get_global $tos) (i32.const 16))))
(set_local $x2 (i32.load (i32.sub (get_global $tos) (i32.const 12))))
(i32.store (i32.sub (get_global $tos) (i32.const 16))
(i32.load (i32.sub (get_global $tos) (i32.const 8))))
(i32.store (i32.sub (get_global $tos) (i32.const 12))
(i32.load (i32.sub (get_global $tos) (i32.const 4))))
(i32.store (i32.sub (get_global $tos) (i32.const 8))
(get_local $x1))
(i32.store (i32.sub (get_global $tos) (i32.const 4))
(get_local $x2)))
(!def_word "2SWAP" "$2SWAP")
;; 6.1.0450 ;; 6.1.0450
(func $colon (param i32) (func $colon (param i32)
(call $create (i32.const -1)) (call $create (i32.const -1))
@ -327,8 +386,7 @@
(i32.const !lengthMask))) (i32.const !lengthMask)))
(i32.store8 (get_global $cp) (i32.const 0)) (i32.store8 (get_global $cp) (i32.const 0))
(i32.store8 (i32.add (get_global $cp) (i32.const 1)) (i32.store8 (i32.add (get_global $cp) (i32.const 1))
(i32.add (i32.const 13) (i32.mul (i32.const 2) (i32.add (i32.const 13) (i32.mul (i32.const 2) (get_local $nameLength))))
(get_local $nameLength))))
(i32.store8 (i32.add (get_global $cp) (i32.const 2)) (i32.const 0x04)) (i32.store8 (i32.add (get_global $cp) (i32.const 2)) (i32.const 0x04))
(i32.store8 (i32.add (get_global $cp) (i32.const 3)) (i32.const 0x6e)) (i32.store8 (i32.add (get_global $cp) (i32.const 3)) (i32.const 0x6e))
(i32.store8 (i32.add (get_global $cp) (i32.const 4)) (i32.const 0x61)) (i32.store8 (i32.add (get_global $cp) (i32.const 4)) (i32.const 0x61))
@ -380,6 +438,17 @@
(set_global $tos (get_local $btos))) (set_global $tos (get_local $btos)))
(!def_word "<" "$less-than") (!def_word "<" "$less-than")
;; 6.1.0530
(func $= (param i32)
(local $btos i32)
(local $bbtos i32)
(if (i32.eq (i32.load (tee_local $bbtos (i32.sub (get_global $tos) (i32.const 8))))
(i32.load (tee_local $btos (i32.sub (get_global $tos) (i32.const 4)))))
(then (i32.store (get_local $bbtos) (i32.const -1)))
(else (i32.store (get_local $bbtos) (i32.const 0))))
(set_global $tos (get_local $btos)))
(!def_word "=" "$=")
;; 6.1.0540 ;; 6.1.0540
(func $greater-than (param i32) (func $greater-than (param i32)
(local $btos i32) (local $btos i32)
@ -391,6 +460,21 @@
(set_global $tos (get_local $btos))) (set_global $tos (get_local $btos)))
(!def_word ">" "$greater-than") (!def_word ">" "$greater-than")
;; 6.1.0550
(func $>BODY (param i32)
(local $btos i32)
(i32.store (tee_local $btos (i32.sub (get_global $tos) (i32.const 4)))
(i32.add (call $body (i32.load (get_local $btos)))
(i32.const 4))))
(!def_word ">BODY" "$>BODY")
;; 6.1.0580
(func $>R (param i32)
(set_global $tos (i32.sub (get_global $tos) (i32.const 4)))
(i32.store (get_global $tors) (i32.load (get_global $tos)))
(set_global $tors (i32.add (get_global $tors) (i32.const 4))))
(!def_word ">R" "$>R")
;; 6.1.0630 ;; 6.1.0630
(func $?DUP (param i32) (func $?DUP (param i32)
(local $btos i32) (local $btos i32)
@ -409,11 +493,32 @@
(i32.load (i32.load (get_local $btos))))) (i32.load (i32.load (get_local $btos)))))
(!def_word "@" "$@") (!def_word "@" "$@")
;; 6.1.0690
(func $ABS (param i32)
(local $btos i32)
(local $v i32)
(local $y i32)
(i32.store (tee_local $btos (i32.sub (get_global $tos) (i32.const 4)))
(i32.sub (i32.xor (tee_local $v (i32.load (get_local $btos)))
(tee_local $y (i32.shr_s (get_local $v) (i32.const 31))))
(get_local $y))))
(!def_word "ABS" "$ABS")
;; 6.1.0710 ;; 6.1.0710
(func $ALLOT (param i32) (func $ALLOT (param i32)
(set_global $here (i32.add (get_global $here) (call $pop)))) (set_global $here (i32.add (get_global $here) (call $pop))))
(!def_word "ALLOT" "$ALLOT") (!def_word "ALLOT" "$ALLOT")
;; 6.1.0720
(func $AND (param i32)
(local $btos i32)
(local $bbtos i32)
(i32.store (tee_local $bbtos (i32.sub (get_global $tos) (i32.const 8)))
(i32.and (i32.load (tee_local $btos (i32.sub (get_global $tos) (i32.const 4))))
(i32.load (get_local $bbtos))))
(set_global $tos (get_local $btos)))
(!def_word "AND" "$AND")
;; 6.1.0705 ;; 6.1.0705
(func $ALIGN (param i32) (func $ALIGN (param i32)
(set_global $here (i32.and (set_global $here (i32.and
@ -421,6 +526,14 @@
(i32.const -4 #| ~3 |#)))) (i32.const -4 #| ~3 |#))))
(!def_word "ALIGN" "$ALIGN") (!def_word "ALIGN" "$ALIGN")
;; 6.1.0706
(func $ALIGNED (param i32)
(local $btos i32)
(i32.store (tee_local $btos (i32.sub (get_global $tos) (i32.const 4)))
(i32.and (i32.add (i32.load (get_local $btos)) (i32.const 3))
(i32.const -4 #| ~3 |#))))
(!def_word "ALIGNED" "$ALIGNED")
;; 6.1.0750 ;; 6.1.0750
(func $BASE (param i32) (func $BASE (param i32)
(i32.store (get_global $tos) (i32.const !baseBase)) (i32.store (get_global $tos) (i32.const !baseBase))
@ -429,7 +542,7 @@
;; 6.1.0760 ;; 6.1.0760
(func $begin (param i32) (func $begin (param i32)
(if (i32.eqz (get_global $state)) (unreachable)) (call $ensureCompiling)
(call $compileBegin)) (call $compileBegin))
(!def_word "BEGIN" "$begin" !fImmediate) (!def_word "BEGIN" "$begin" !fImmediate)
@ -445,6 +558,14 @@
(set_global $tos (get_local $bbtos))) (set_global $tos (get_local $bbtos)))
(!def_word "C!" "$c-store") (!def_word "C!" "$c-store")
;; 6.1.0860
(func $c-comma (param i32)
(i32.store8 (get_global $here)
(i32.load (i32.sub (get_global $tos) (i32.const 4))))
(set_global $here (i32.add (get_global $here) (i32.const 1)))
(set_global $tos (i32.sub (get_global $tos) (i32.const 4))))
(!def_word "C," "$c-comma")
;; 6.1.0870 ;; 6.1.0870
(func $c-fetch (param i32) (func $c-fetch (param i32)
(local $btos i32) (local $btos i32)
@ -467,6 +588,17 @@
(set_global $here (i32.add (get_global $here) (i32.const 4)))) (set_global $here (i32.add (get_global $here) (i32.const 4))))
(!def_word "CONSTANT" "$CONSTANT") (!def_word "CONSTANT" "$CONSTANT")
;; 6.1.0980
(func $COUNT (param i32)
(local $btos i32)
(local $addr i32)
(i32.store (get_global $tos)
(i32.load (tee_local $addr (i32.load (tee_local $btos (i32.sub (get_global $tos)
(i32.const 4)))))))
(i32.store (get_local $btos) (i32.add (get_local $addr) (i32.const 4)))
(set_global $tos (i32.add (get_global $tos) (i32.const 4))))
(!def_word "COUNT" "$COUNT")
;; 6.1.1000 ;; 6.1.1000
(func $create (param i32) (func $create (param i32)
(local $length i32) (local $length i32)
@ -491,9 +623,17 @@
(set_global $here (i32.add (get_global $here) (i32.const 4)))) (set_global $here (i32.add (get_global $here) (i32.const 4))))
(!def_word "CREATE" "$create") (!def_word "CREATE" "$create")
;; 6.1.1200
(func $DEPTH (param i32)
(i32.store (get_global $tos)
(i32.shr_u (i32.sub (get_global $tos) (i32.const !stackBase)) (i32.const 2)))
(set_global $tos (i32.add (get_global $tos) (i32.const 4))))
(!def_word "DEPTH" "$DEPTH")
;; 6.1.1240 ;; 6.1.1240
(func $do (param i32) (func $do (param i32)
(if (i32.eqz (get_global $state)) (unreachable)) (call $ensureCompiling)
(call $compileDo)) (call $compileDo))
(!def_word "DO" "$do" !fImmediate) (!def_word "DO" "$do" !fImmediate)
@ -516,16 +656,38 @@
;; 6.1.1310 ;; 6.1.1310
(func $else (param i32) (func $else (param i32)
(if (i32.eqz (get_global $state)) (unreachable)) (call $ensureCompiling)
(call $compileElse)) (call $compileElse))
(!def_word "ELSE" "$else" !fImmediate) (!def_word "ELSE" "$else" !fImmediate)
;; 6.1.1320 ;; 6.1.1320
(func $emit (param i32) (func $emit (param i32)
(call $shell_emit (i32.load (i32.sub (get_global $tos) (i32.const 4)))) (call $shell_emit (i32.load (i32.sub (get_global $tos) (i32.const 4))))
(set_global $tos (i32.sub (get_global $tos) (i32.const 4)))) (set_global $tos (i32.sub (get_global $tos) (i32.const 4))))
(!def_word "EMIT" "$emit") (!def_word "EMIT" "$emit")
;; 6.1.1370
(func $EXECUTE (param i32)
(local $body i32)
(set_local $body (call $body (call $pop)))
(call_indirect (type $word) (i32.add (get_local $body) (i32.const 4))
(i32.load (get_local $body))))
(!def_word "EXECUTE" "$EXECUTE")
;; 6.1.1380
(func $EXIT (param i32)
(call $emitReturn))
(!def_word "EXIT" "$EXIT" !fImmediate)
;; 6.1.1540
(func $FILL (param i32)
(local $bbbtos i32)
(call $memset (i32.load (tee_local $bbbtos (i32.sub (get_global $tos) (i32.const 12))))
(i32.load (i32.sub (get_global $tos) (i32.const 4)))
(i32.load (i32.sub (get_global $tos) (i32.const 8))))
(set_global $tos (get_local $bbbtos)))
(!def_word "FILL" "$FILL")
;; 6.1.1550 ;; 6.1.1550
(func $find (export "FIND") (param i32) (func $find (export "FIND") (param i32)
(local $entryP i32) (local $entryP i32)
@ -585,13 +747,13 @@
;; 6.1.1680 ;; 6.1.1680
(func $i (param i32) (func $i (param i32)
(if (i32.eqz (get_global $state)) (unreachable)) (call $ensureCompiling)
(call $compilePushLocal (i32.sub (get_global $currentLocal) (i32.const 1)))) (call $compilePushLocal (i32.sub (get_global $currentLocal) (i32.const 1))))
(!def_word "I" "$i" !fImmediate) (!def_word "I" "$i" !fImmediate)
;; 6.1.1700 ;; 6.1.1700
(func $if (param i32) (func $if (param i32)
(if (i32.eqz (get_global $state)) (unreachable)) (call $ensureCompiling)
(call $compileIf)) (call $compileIf))
(!def_word "IF" "$if" !fImmediate) (!def_word "IF" "$if" !fImmediate)
@ -604,9 +766,16 @@
(i32.const !fImmediate)))) (i32.const !fImmediate))))
(!def_word "IMMEDIATE" "$immediate") (!def_word "IMMEDIATE" "$immediate")
;; 6.1.1720
(func $INVERT (param i32)
(local $btos i32)
(i32.store (tee_local $btos (i32.sub (get_global $tos) (i32.const 4)))
(i32.xor (i32.load (get_local $btos)) (i32.const -1))))
(!def_word "INVERT" "$INVERT")
;; 6.1.1730 ;; 6.1.1730
(func $j (param i32) (func $j (param i32)
(if (i32.eqz (get_global $state)) (unreachable)) (call $ensureCompiling)
(call $compilePushLocal (i32.sub (get_global $currentLocal) (i32.const 3)))) (call $compilePushLocal (i32.sub (get_global $currentLocal) (i32.const 3))))
(!def_word "J" "$j" !fImmediate) (!def_word "J" "$j" !fImmediate)
@ -616,6 +785,13 @@
(set_global $tos (i32.add (get_global $tos) (i32.const 4)))) (set_global $tos (i32.add (get_global $tos) (i32.const 4))))
(!def_word "KEY" "$key") (!def_word "KEY" "$key")
;; 6.1.1760
(func $LEAVE (param i32)
(call $ensureCompiling)
(call $compileLeave))
(!def_word "LEAVE" "$LEAVE" !fImmediate)
;; 6.1.1780 ;; 6.1.1780
(func $literal (param i32) (func $literal (param i32)
(call $compilePushConst (call $pop))) (call $compilePushConst (call $pop)))
@ -623,10 +799,56 @@
;; 6.1.1800 ;; 6.1.1800
(func $loop (param i32) (func $loop (param i32)
(if (i32.eqz (get_global $state)) (unreachable)) (call $ensureCompiling)
(call $compileLoop)) (call $compileLoop))
(!def_word "LOOP" "$loop" !fImmediate) (!def_word "LOOP" "$loop" !fImmediate)
;; 6.1.1805
(func $LSHIFT (param i32)
(local $btos i32)
(local $bbtos i32)
(i32.store (tee_local $bbtos (i32.sub (get_global $tos) (i32.const 8)))
(i32.shl (i32.load (get_local $bbtos))
(i32.load (tee_local $btos (i32.sub (get_global $tos) (i32.const 4))))))
(set_global $tos (get_local $btos)))
(!def_word "LSHIFT" "$LSHIFT")
;; 6.1.1870
(func $MAX (param i32)
(local $btos i32)
(local $bbtos i32)
(local $v i32)
(if (i32.lt_s (i32.load (tee_local $bbtos (i32.sub (get_global $tos) (i32.const 8))))
(tee_local $v (i32.load (tee_local $btos (i32.sub (get_global $tos)
(i32.const 4))))))
(then
(i32.store (get_local $bbtos) (get_local $v))))
(set_global $tos (get_local $btos)))
(!def_word "MAX" "$MAX")
;; 6.1.1880
(func $MIN (param i32)
(local $btos i32)
(local $bbtos i32)
(local $v i32)
(if (i32.gt_s (i32.load (tee_local $bbtos (i32.sub (get_global $tos) (i32.const 8))))
(tee_local $v (i32.load (tee_local $btos (i32.sub (get_global $tos)
(i32.const 4))))))
(then
(i32.store (get_local $bbtos) (get_local $v))))
(set_global $tos (get_local $btos)))
(!def_word "MIN" "$MIN")
;; 6.1.1880
(func $MOD (param i32)
(local $btos i32)
(local $bbtos i32)
(i32.store (tee_local $bbtos (i32.sub (get_global $tos) (i32.const 8)))
(i32.rem_s (i32.load (get_local $bbtos))
(i32.load (tee_local $btos (i32.sub (get_global $tos) (i32.const 4))))))
(set_global $tos (get_local $btos)))
(!def_word "MOD" "$MOD")
;; 6.1.1910 ;; 6.1.1910
(func $negate (param i32) (func $negate (param i32)
(local $btos i32) (local $btos i32)
@ -634,6 +856,16 @@
(i32.sub (i32.const 0) (i32.load (get_local $btos))))) (i32.sub (i32.const 0) (i32.load (get_local $btos)))))
(!def_word "NEGATE" "$negate") (!def_word "NEGATE" "$negate")
;; 6.1.1980
(func $OR (param i32)
(local $btos i32)
(local $bbtos i32)
(i32.store (tee_local $bbtos (i32.sub (get_global $tos) (i32.const 8)))
(i32.or (i32.load (tee_local $btos (i32.sub (get_global $tos) (i32.const 4))))
(i32.load (get_local $bbtos))))
(set_global $tos (get_local $btos)))
(!def_word "OR" "$OR")
;; 6.1.1990 ;; 6.1.1990
(func $over (param i32) (func $over (param i32)
(i32.store (get_global $tos) (i32.store (get_global $tos)
@ -641,6 +873,19 @@
(set_global $tos (i32.add (get_global $tos) (i32.const 4)))) (set_global $tos (i32.add (get_global $tos) (i32.const 4))))
(!def_word "OVER" "$over") (!def_word "OVER" "$over")
;; 6.1.2060
(func $R> (param i32)
(set_global $tors (i32.sub (get_global $tors) (i32.const 4)))
(i32.store (get_global $tos) (i32.load (get_global $tors)))
(set_global $tos (i32.add (get_global $tos) (i32.const 4))))
(!def_word "R>" "$R>")
;; 6.1.2070
(func $R@ (param i32)
(i32.store (get_global $tos) (i32.load (i32.sub (get_global $tors) (i32.const 4))))
(set_global $tos (i32.add (get_global $tos) (i32.const 4))))
(!def_word "R@" "$R@")
;; 6.1.2120 ;; 6.1.2120
(func $RECURSE (param i32) (func $RECURSE (param i32)
(call $compileRecurse)) (call $compileRecurse))
@ -649,7 +894,7 @@
;; 6.1.2140 ;; 6.1.2140
(func $repeat (param i32) (func $repeat (param i32)
(if (i32.eqz (get_global $state)) (unreachable)) (call $ensureCompiling)
(call $compileRepeat)) (call $compileRepeat))
(!def_word "REPEAT" "$repeat" !fImmediate) (!def_word "REPEAT" "$repeat" !fImmediate)
@ -668,6 +913,16 @@
(get_local $tmp))) (get_local $tmp)))
(!def_word "ROT" "$ROT") (!def_word "ROT" "$ROT")
;; 6.1.2162
(func $RSHIFT (param i32)
(local $btos i32)
(local $bbtos i32)
(i32.store (tee_local $bbtos (i32.sub (get_global $tos) (i32.const 8)))
(i32.shr_u (i32.load (get_local $bbtos))
(i32.load (tee_local $btos (i32.sub (get_global $tos) (i32.const 4))))))
(set_global $tos (get_local $btos)))
(!def_word "RSHIFT" "$RSHIFT")
;; 6.1.2165 ;; 6.1.2165
(func $Sq (param i32) (func $Sq (param i32)
(local $c i32) (local $c i32)
@ -705,7 +960,7 @@
;; 6.1.2270 ;; 6.1.2270
(func $then (param i32) (func $then (param i32)
(if (i32.eqz (get_global $state)) (unreachable)) (call $ensureCompiling)
(call $compileThen)) (call $compileThen))
(!def_word "THEN" "$then" !fImmediate) (!def_word "THEN" "$then" !fImmediate)
@ -718,6 +973,27 @@
(i32.store (i32.add (call $body (call $pop)) (i32.const 4)) (call $pop))) (i32.store (i32.add (call $body (call $pop)) (i32.const 4)) (call $pop)))
(!def_word "TO" "$TO") (!def_word "TO" "$TO")
;; 6.1.2340
(func $U< (param i32)
(local $btos i32)
(local $bbtos i32)
(if (i32.lt_u (i32.load (tee_local $bbtos (i32.sub (get_global $tos) (i32.const 8))))
(i32.load (tee_local $btos (i32.sub (get_global $tos) (i32.const 4)))))
(then (i32.store (get_local $bbtos) (i32.const -1)))
(else (i32.store (get_local $bbtos) (i32.const 0))))
(set_global $tos (get_local $btos)))
(!def_word "U<" "$U<")
;; 6.1.2380
(func $UNLOOP (param i32))
(!def_word "UNLOOP" "$UNLOOP" !fImmediate)
;; 6.1.2390
(func $UNTIL (param i32)
(call $ensureCompiling)
(call $compileUntil))
(!def_word "UNTIL" "$UNTIL" !fImmediate)
;; 6.2.2405 ;; 6.2.2405
(!def_word "VALUE" "$CONSTANT") (!def_word "VALUE" "$CONSTANT")
@ -731,11 +1007,10 @@
;; 6.1.2430 ;; 6.1.2430
(func $while (param i32) (func $while (param i32)
(if (i32.eqz (get_global $state)) (unreachable)) (call $ensureCompiling)
(call $compileWhile)) (call $compileWhile))
(!def_word "WHILE" "$while" !fImmediate) (!def_word "WHILE" "$while" !fImmediate)
;; 6.1.2450
(func $word (export "WORD") (param i32) (func $word (export "WORD") (param i32)
(local $char i32) (local $char i32)
(local $stringPtr i32) (local $stringPtr i32)
@ -781,7 +1056,16 @@
(i32.sub (get_local $stringPtr) (i32.const (!+ !wordBase 4)))) (i32.sub (get_local $stringPtr) (i32.const (!+ !wordBase 4))))
(call $push (i32.const !wordBase))) (call $push (i32.const !wordBase)))
(!def_word "WORD" "$word")
;; 6.1.2490
(func $XOR (param i32)
(local $btos i32)
(local $bbtos i32)
(i32.store (tee_local $bbtos (i32.sub (get_global $tos) (i32.const 8)))
(i32.xor (i32.load (tee_local $btos (i32.sub (get_global $tos) (i32.const 4))))
(i32.load (get_local $bbtos))))
(set_global $tos (get_local $btos)))
(!def_word "XOR" "$XOR")
;; 6.1.2500 ;; 6.1.2500
(func $left-bracket (param i32) (func $left-bracket (param i32)
@ -831,11 +1115,48 @@
;; High-level words ;; High-level words
(!prelude #<<EOF (!prelude #<<EOF
\ 6.1.1170
: DECIMAL 10 BASE ! ;
\ 6.1.0897
: CHAR+ 1+ ;
\ 6.1.0898
: CHARS ;
\ 6.1.0880
: CELL+ 4 + ;
\ 6.1.0890
: CELLS 4 * ;
\ 6.1.0350
: 2@ DUP CELL+ @ SWAP @ ;
\ 6.1.0310
: 2! SWAP OVER ! CELL+ ! ;
: UWIDTH BASE @ / ?DUP IF RECURSE 1+ ELSE 1 THEN ; : UWIDTH BASE @ / ?DUP IF RECURSE 1+ ELSE 1 THEN ;
: '\n' 10 ; : '\n' 10 ;
\ : 'A' [ CHAR A ] LITERAL ; : 'A' [ CHAR A ] LITERAL ;
\ : '0' [ CHAR 0 ] LITERAL ; : '0' [ CHAR 0 ] LITERAL ;
: '(' [ CHAR ( ] LITERAL ;
: ')' [ CHAR ) ] LITERAL ;
: (
1
BEGIN
KEY
DUP '(' = IF
DROP
1+
ELSE
')' = IF 1- THEN
THEN
DUP 0= UNTIL
DROP
; IMMEDIATE
\ 6.1.0990 \ 6.1.0990
: CR '\n' EMIT ; : CR '\n' EMIT ;
@ -1047,17 +1368,16 @@ EOF
(call $emitEnd) (call $emitEnd)
(set_global $currentLocal (i32.sub (get_global $currentLocal) (i32.const 2)))) (set_global $currentLocal (i32.sub (get_global $currentLocal) (i32.const 2))))
(func $compileLeave
(call $emitBr (i32.const 1)))
(func $compileBegin (func $compileBegin
(call $emitBlock) (call $emitBlock)
(call $emitLoop)) (call $emitLoop))
(func $compileWhile (func $compileWhile
(call $compilePop) (call $compilePop)
(call $emitEqualsZero)
;; eqz
(i32.store8 (get_global $cp) (i32.const 0x45))
(set_global $cp (i32.add (get_global $cp) (i32.const 1)))
(call $emitBrIf (i32.const 1))) (call $emitBrIf (i32.const 1)))
(func $compileRepeat (func $compileRepeat
@ -1065,6 +1385,14 @@ EOF
(call $emitEnd) (call $emitEnd)
(call $emitEnd)) (call $emitEnd))
(func $compileUntil
(call $compilePop)
(call $emitEqualsZero)
(call $emitBrIf (i32.const 0))
(call $emitBr (i32.const 1))
(call $emitEnd)
(call $emitEnd))
(func $compileRecurse (func $compileRecurse
;; get_local 0 ;; get_local 0
(i32.store8 (get_global $cp) (i32.const 0x20)) (i32.store8 (get_global $cp) (i32.const 0x20))
@ -1138,10 +1466,18 @@ EOF
(i32.store8 (get_global $cp) (i32.const 0x6a)) (i32.store8 (get_global $cp) (i32.const 0x6a))
(set_global $cp (i32.add (get_global $cp) (i32.const 1)))) (set_global $cp (i32.add (get_global $cp) (i32.const 1))))
(func $emitEqualsZero
(i32.store8 (get_global $cp) (i32.const 0x45))
(set_global $cp (i32.add (get_global $cp) (i32.const 1))))
(func $emitGreaterEqualSigned (func $emitGreaterEqualSigned
(i32.store8 (get_global $cp) (i32.const 0x4e)) (i32.store8 (get_global $cp) (i32.const 0x4e))
(set_global $cp (i32.add (get_global $cp) (i32.const 1)))) (set_global $cp (i32.add (get_global $cp) (i32.const 1))))
(func $emitReturn
(i32.store8 (get_global $cp) (i32.const 0x0f))
(set_global $cp (i32.add (get_global $cp) (i32.const 1))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;; Word helper function ;; Word helper function
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
@ -1156,7 +1492,7 @@ EOF
(i32.load (get_global $tos))) (i32.load (get_global $tos)))
(elem (i32.const !popIndex) $pop) (elem (i32.const !popIndex) $pop)
(func $display (func $TYPE
(local $p i32) (local $p i32)
(local $end i32) (local $end i32)
(set_local $end (i32.add (call $pop) (tee_local $p (call $pop)))) (set_local $end (i32.add (call $pop) (tee_local $p (call $pop))))
@ -1166,7 +1502,7 @@ EOF
(call $shell_emit (i32.load8_u (get_local $p))) (call $shell_emit (i32.load8_u (get_local $p)))
(set_local $p (i32.add (get_local $p) (i32.const 1))) (set_local $p (i32.add (get_local $p) (i32.const 1)))
(br $loop)))) (br $loop))))
(elem (i32.const !displayIndex) $display) (!def_word "TYPE" "$TYPE" !fNone !typeIndex)
(func $pushDataAddress (param $d i32) (func $pushDataAddress (param $d i32)
(call $push (get_local $d))) (call $push (get_local $d)))
@ -1180,6 +1516,9 @@ EOF
;; Helper functions ;; Helper functions
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
(func $ensureCompiling
(if (i32.eqz (get_global $state)) (unreachable)))
;; Toggle the hidden flag ;; Toggle the hidden flag
(func $hidden (func $hidden
(i32.store (i32.store

View file

@ -583,6 +583,16 @@ describe("WAForth", () => {
}); });
}); });
describe("LEAVE", () => {
it("should leave", () => {
run(`: FOO 4 0 DO 3 LEAVE 6 LOOP 4 ;`);
run("FOO 5");
expect(stack[0]).to.eql(3);
expect(stack[1]).to.eql(4);
expect(stack[2]).to.eql(5);
});
});
describe("+LOOP", () => { describe("+LOOP", () => {
it("should increment a loop", () => { it("should increment a loop", () => {
run(`: FOO 10 0 DO 3 2 +LOOP ;`); run(`: FOO 10 0 DO 3 2 +LOOP ;`);
@ -671,6 +681,44 @@ describe("WAForth", () => {
}); });
}); });
describe("BEGIN / UNTIL", () => {
it("should work", () => {
run(`: FOO BEGIN DUP 2 * DUP 16 > UNTIL 7 ;`);
run("1 FOO 5");
expect(stack[0]).to.eql(1);
expect(stack[1]).to.eql(2);
expect(stack[2]).to.eql(4);
expect(stack[3]).to.eql(8);
expect(stack[4]).to.eql(16);
expect(stack[5]).to.eql(32);
expect(stack[6]).to.eql(7);
expect(stack[7]).to.eql(5);
});
});
describe("EXIT", () => {
it("should work", () => {
run(`: FOO IF 3 EXIT 4 THEN 5 ;`);
run("1 FOO 6");
expect(stack[0]).to.eql(3);
expect(stack[1]).to.eql(6);
});
});
describe("( / )", () => {
beforeEach(() => {
core.loadPrelude();
});
it("should work", () => {
run(": FOO ( bad -- x ) 7 ;");
run("1 FOO 5");
expect(stack[0]).to.eql(1);
expect(stack[1]).to.eql(7);
expect(stack[2]).to.eql(5);
});
});
describe("CHAR", () => { describe("CHAR", () => {
it("should work with a single character", () => { it("should work with a single character", () => {
run("CHAR A 5"); run("CHAR A 5");
@ -735,7 +783,7 @@ describe("WAForth", () => {
forth.read("DUP"); forth.read("DUP");
core.WORD(); core.WORD();
core.FIND(); core.FIND();
expect(stack[0]).to.eql(131524); expect(stack[0]).to.eql(131728);
expect(stack[1]).to.eql(-1); expect(stack[1]).to.eql(-1);
}); });
@ -751,7 +799,7 @@ describe("WAForth", () => {
forth.read("+LOOP"); forth.read("+LOOP");
core.WORD(); core.WORD();
core.FIND(); core.FIND();
expect(stack[0]).to.eql(131108); expect(stack[0]).to.eql(131132);
expect(stack[1]).to.eql(1); expect(stack[1]).to.eql(1);
}); });