aboutsummaryrefslogtreecommitdiffstats
path: root/forth/core.fth
diff options
context:
space:
mode:
Diffstat (limited to 'forth/core.fth')
-rw-r--r--forth/core.fth203
1 files changed, 203 insertions, 0 deletions
diff --git a/forth/core.fth b/forth/core.fth
new file mode 100644
index 0000000..a8ccb9d
--- /dev/null
+++ b/forth/core.fth
@@ -0,0 +1,203 @@
+: * m* drop ;
+: s>d 1 m* ;
+: / >r s>d r> _/ ;
+: % >r s>d r> _% ;
+
+: cell+ 2 + ;
+: cells 2 * ;
+
+: . 0 sys ;
+: emit 1 sys ;
+: u. 4 sys ;
+
+: 1+ 1 + ;
+: 1- 1 - ;
+
+: ' _' drop ;
+: ! 1 _! ;
+: @ 1 _@ ;
+: +! dup >r swap r> @ + swap ! ;
+
+: base 0 ;
+: here 1 cells @ ;
+: allot 1 cells +! ;
+: _latest 2 cells ;
+: imm _latest @ dup @ 1 5 << | swap ! ;
+: immediate imm ;
+: state 3 cells ;
+: _source 4 cells ;
+: _sourceu 5 cells ;
+: >in 6 cells ;
+
+: , here ! 1 cells allot ;
+
+: [ 0 state ! ; imm
+: ] 1 state ! ;
+
+: literal [ ' _lit dup , , ] , , ; imm
+: ['] ' [ ' literal , ] ; imm
+
+: if ['] _jmp0 , here 0 , ; imm
+: then here swap ! ; imm
+: else ['] _jmp , here 0 , swap here swap ! ; imm
+
+: postpone _' dup 0 = if exit then
+ 1 = swap ['] _lit , , if ['] execute ,
+ else ['] , , then ; imm
+
+: over 1 pick ;
+: rot >r swap r> swap ;
+: -rot rot rot ;
+
+: 2drop drop drop ;
+: 2dup over over ;
+: 2over 3 pick 3 pick ;
+: 2swap rot >r rot r> ;
+
+: c! 0 _! ;
+: c@ 0 _@ ;
+: c, here c! 1 allot ;
+: char+ 1+ ;
+: chars ;
+
+: decimal 10 base ! ;
+
+: 2r> ['] r> , ['] r> , ['] swap , ; imm
+: 2>r ['] swap , ['] >r , ['] >r , ; imm
+: r@ ['] r> , ['] dup , ['] >r , ; imm
+
+: 2! swap over ! cell+ ! ;
+: 2@ dup cell+ @ swap @ ;
+
+: 0= 0 = ;
+: 0< 0 < ;
+: <= 2dup < >r = r> | ;
+: > swap < ;
+: <> = 0= ;
+
+: begin 0 here ; imm
+: while swap 1+ swap postpone if -rot ; imm
+: repeat ['] _jmp , , if postpone then then ; imm
+: until ['] _jmp0 , , drop ; imm
+
+: do ['] _lit , here 0 , ['] >r , postpone 2>r here ; imm
+: unloop postpone 2r> ['] 2drop , ['] r> , ['] drop , ; imm
+: leave postpone 2r> ['] 2drop , postpone 2r>
+ ['] drop , ['] >r , ['] exit , ; imm
+: +loop postpone 2r> ['] 2dup , ['] swap , ['] < , ['] >r ,
+ ['] rot , ['] + , ['] 2dup , ['] swap , ['] < ,
+ ['] r> , ['] ^ , ['] -rot ,
+ postpone 2>r ['] _jmp0 , ,
+ postpone unloop here swap ! ; imm
+: loop postpone 2r> ['] 1+ , ['] 2dup ,
+ postpone 2>r ['] = , ['] _jmp0 , ,
+ postpone unloop here swap ! ; imm
+: i postpone r@ ; imm
+: j postpone 2r> ['] r> , postpone r@ ['] swap ,
+ ['] >r , ['] -rot , postpone 2>r ; imm
+
+: align here 1 cells 1- swap over & if 1 cells swap - allot else drop then ;
+: aligned dup 1 cells 1- swap over & if 1 cells swap - + else drop then ;
+
+: and & ;
+: or | ;
+: xor ^ ;
+: lshift << ;
+: rshift >> ;
+: invert -1 ^ ;
+: mod % ;
+: 2* 2 * ;
+: _msb 1 1 cells 8 * 1- << ;
+: 2/ dup 1 >> swap 0< if _msb or then ;
+
+: /mod 2dup % -rot / ;
+: */ >r m* r> _/ ;
+: sm/rem >r 2dup r@ _% -rot r> _/ ;
+: */mod >r m* r> sm/rem ;
+: fm/mod 2dup dup >r ^ >r sm/rem swap dup
+ if r> 0< if r> + swap 1- else swap r> drop then
+ else swap 2r> 2drop then ;
+
+: cr 10 emit ;
+: bl 32 ;
+: space bl emit ;
+: spaces begin dup 0 > while space 1- repeat drop ;
+
+: ?dup dup if dup then ;
+
+: negate -1 * ;
+: abs dup 0< if negate then ;
+: min 2dup <= if drop else swap drop then ;
+: max 2dup <= if swap drop else drop then ;
+
+: source _source @ 0 begin 2dup + c@ while char+ repeat ;
+: key _source @ >in @ +
+ begin dup c@ 0 = while _in repeat
+ c@ 1 >in +! ;
+: key? _source @ >in @ + c@ 0 <> ;
+: word here dup >r char+ >r
+ begin key? if key 2dup <> else 0 0 then while
+ r> swap over c! char+ >r repeat
+ 2drop r> r> swap over - 1- over c! ;
+: count dup char+ swap c@ ;
+: char bl word char+ c@ ;
+: [char] char postpone literal ; imm
+
+: ( begin [char] ) key <> while repeat ; imm
+
+: type begin dup 0 > while swap dup c@ emit char+ swap 1- repeat 2drop ;
+: s" state @ if ['] _jmp , here 0 , then
+ [char] " word count
+ state @ 0= if exit then
+ dup cell+ allot
+ rot here swap !
+ swap postpone literal postpone literal ; imm
+: ." postpone s" state @ if ['] type , else type then ; imm
+
+: create align here
+ 1 cells 1 chars - allot
+ bl word count swap drop
+ 1 chars allot
+ swap over over ! swap allot align
+ ['] _lit , here 3 cells + , ['] exit dup , ,
+ dup @ 31 & over _latest @ - 6 << or over ! _latest ! ;
+: _does> _latest @ dup @ 31 & + cell+ aligned 2 cells +
+ ['] _jmp over ! cell+
+ r@ 1 cells - @ swap ! ;
+: does> ['] _jmp , here 2 cells + dup , 2 cells + ,
+ ['] _does> , ['] exit , ; imm
+: >body cell+ @ ;
+
+: variable create 1 cells allot ;
+: constant create , does> @ ;
+
+: quit begin _rdepth 1 > while r> drop repeat postpone [ ;
+: abort begin depth 0 > while drop repeat quit ;
+: abort" postpone s" ['] rot ,
+ postpone if ['] type , ['] abort ,
+ postpone else ['] 2drop , postpone then ; imm
+
+: recurse depth 1- pick dup @ 31 & + cell+ aligned , ; imm
+
+: move dup 0 <= if drop 2drop exit then
+ >r 2dup < r> swap if
+ 1- 0 swap do over i + c@ over i + c! -1 +loop
+ else
+ 0 do over i + c@ over i + c! loop
+ then 2drop ;
+: fill -rot begin dup 0 > while
+ >r 2dup c! char+ r> 1- repeat
+ 2drop drop ;
+
+: environment? 2drop 1 0= ;
+
+: accept over >r begin dup 0 > while
+ key dup 32 < if 2drop 0
+ else dup emit rot 2dup c! char+ swap drop swap 1- then
+ repeat drop r> - 1 chars / ;
+
+: :noname 0 , here ] ;
+
+: evaluate _source @ >r _sourceu @ >r >in @ >r
+ 0 >in ! _sourceu ! _source ! _ev
+ r> >in ! r> _sourceu ! r> _source ! ;