( Usage: uxnfor.rom source.tal ) |00 @System &vector $2 &expansion $2 &wst $1 &rst $1 &metadata $2 &r $2 &g $2 &b $2 &debug $1 &state $1 |10 @Console &vector $2 &read $1 &pad $4 &type $1 &write $1 &error $1 |a0 @File &vector $2 &success $2 &stat $2 &delete $1 &append $1 &name $2 &length $2 &read $2 &write $2 |b0 @File2 &vector $2 &success $2 &stat $2 &delete $1 &append $1 &name $2 &length $2 &read $2 &write $2 |000 @pad $20 @src $40 @err $1 @nobrk $1 @inzp $1 |100 @on-reset ( -> ) ;meta #06 DEO2 ;await-src .Console/vector DEO2 .Console/type DEI ?{ ;dict/usage #01 ;await-src/interactive STA } BRK @meta $1 ( name ) "Uxnfor 0a ( desc ) "Uxntal 20 "Formatter 0a ( auth ) "By 20 "Devine 20 "Lu 20 "Linvega 0a ( date ) "25 20 "Aug 20 "2024 $2 @await-src ( -> ) .Console/read DEI DUP #20 GTH ?{ POP src/ [ LIT &interactive $1 ] ?{ #800f DEO } BRK } src/ BRK @src/ ( char -- ) [ LIT2 00 &ptr -src ] INCk ,&ptr STR STZ2 JMP2r @src/ ( -- ) [ LIT2 -src _&ptr ] STR #0000 .src STZ2 JMP2r ( @|core ) @ ( -- ) ;src .File/name DEO2 #0001 .File/length DEO2 [ LIT2 00 -err ] STZ Result/new &>s ( -- ) ;&c feof ?{ .err LDZ ?&fail [ LIT &c $1 ] walk-char !&>s } eval-scope ! &fail ( -- ) ;dict/err ;mem/scope / #0a19 DEO JMP2r @walk-char ( c -- c ) ( norm ws ) #20 GTHk [ JMP SWP POP ] ( join ws ) [ LIT &last 20 ] OVR ,&last STR ( no repeat ) DUP2 #2020 NEQ2 ?{ POP2 JMP2r } #20 NEQ ?scope/put DUP #28 NEQ ?{ [ LIT2 01 _&mute ] STR } DUP #29 NEQ ?{ [ LIT2 00 _&mute ] STR } [ LIT &mute $1 ] ?scope/put DUP [ LIT "@ ] NEQ OVR [ LIT "% ] NEQ AND ?scope/put eval-scope !scope/put @eval-scope ( -- ) ;mem/scope ( | pad zp ) .inzp LDZ #00 EQU ?{ #09 } ( | label/macro ) / #20 ( | definition ) INC2k LDA #28 NEQ ?{ ( | definition for macro joins brackets ) INC2k LDA [ LIT "{ ] NEQ ?{ #20 INC2 LDAk } INC2 !&w } INC2 LDAk [ LIT "& ] NEQ ?{ !&w } LDAk #5b NEQ ?{ !&w } LDAk [ LIT "$ ] NEQ ?{ !&w } &w ( -- ) DUP2 #0001 SUB2 LDA #20 NEQ ?{ LDAk #28 EQU ?handle-comment LDAk #5b EQU ?handle-block LDAk [ LIT "| ] EQU ?handle-padabs LDAk [ LIT "$ ] EQU ?handle-padrel LDAk [ LIT "& ] EQU ?handle-sublab LDAk [ LIT "{ ] EQU ?handle-lambda LDAk [ LIT "} ] EQU ?handle-lambda-end LDAk [ LIT "? ] EQU ?handle-jxi DUP2 ;dict/jmp scmp3 ?handle-jmp DUP2 ;dict/brk scmp3 ?handle-jmp LDAk [ LIT "! ] EQU ?handle-jmp LDAk [ LIT "~ ] EQU ?handle-include } LDAk #20 NEQ ?&collapse DUP2 wrew INC2 is-breaking #00 EQU ?&collapse !&continue &collapse LDAk &continue INC2 LDAk ?&w .nobrk LDZ ?&join .inzp LDZ ?{ #0a } #0a POP2 !scope/new &join ( -- ) [ LIT2 00 -nobrk ] STZ #20 POP2 !scope/new @handle-padabs ( addr* -- addr* ) #0a INC2k slen #0004 NEQ2 ?{ #0a } DUP2 wrew wrew INC2 LDA [ LIT ") ] NEQ ?{ #0a } INC2k slen #0004 EQU2 ?{ [ LIT2 01 -nobrk ] STZ / !eval-scope/continue } INC2k LDA2 [ LIT2 "00 ] EQU2 .inzp STZ / #0a !eval-scope/continue @handle-padrel ( addr* -- addr* ) / ;mem/scope INC2 LDA ciuc ?&space-after INC2k LDA #00 EQU ?&space-after DUP2 wcap/ INC2 LDA [ LIT "& ] NEQ ?&break-after DUP2 wrew wrew INC2 LDA [ LIT "& ] EQU ?&space-after DUP2 wrew wrew LDA [ LIT "@ ] EQU ?&space-after &break-after ( addr* -- addr* ) !eval-scope/continue &space-after ( addr* -- addr* ) !eval-scope/continue @handle-sublab ( addr* -- addr* ) DUP2 wcap/ INC2 LDA ( a method ) DUP [ LIT "( ] NEQ ?{ } ( a string ) DUP [ LIT "" ] NEQ ?{ } ( a block ) DUP [ LIT "[ ] NEQ ?{ } POP ( | continue ) LDA2k [ LIT2 "&> ] NEQ2 ?{ } / INC2 LDAk [ LIT "( ] NEQ ?{ } !eval-scope/w @handle-block ( addr* -- addr* ) !eval-scope/continue @handle-comment ( addr* -- addr* ) DUP2 #0002 SUB2 LDA [ LIT ") ] EQU ?{ } !eval-scope/continue @handle-lambda ( addr* -- addr* ) / INC2k LDA [ LIT "& ] EQU ?eval-scope/continue DUP2 count-lambda #0006 LTH2 ?{ } !eval-scope/continue @handle-lambda-end ( addr* -- addr* ) / INC2k LDA [ LIT "} ] NEQ ?{ !eval-scope/continue } INC2k LDA [ LIT "& ] EQU ?eval-scope/continue INC2k LDA #00 EQU ?eval-scope/continue !eval-scope/continue @handle-jxi ( addr* -- addr* ) INC2k LDA [ LIT "{ ] EQU ?handle-lambda INC2k LDA2 [ LIT2 "&> ] NEQ2 ?{ } / INC2k LDA [ LIT "} ] EQU ?eval-scope/continue INC2k LDA [ LIT "& ] EQU ?eval-scope/continue INC2k LDA #00 EQU ?eval-scope/continue !eval-scope/continue @handle-jmp ( addr* -- addr* ) INC2k LDA2 [ LIT2 "&> ] NEQ2 ?{ } / INC2k LDA #00 EQU ?eval-scope/continue INC2k LDA [ LIT "} ] EQU ?{ } !eval-scope/continue @handle-include ( addr* -- addr* ) #0a / !eval-scope/continue @ ( -- ) .err LDZ ?{ ;src .File/name DEO2 ;mem/result ;Result/ptr LDA2 OVR2 SUB2 .File/length DEO2 .File/write DEO2 } JMP2r ( @|utils ) @is-breaking ( str* -- bool ) DUP2 wcap/ INC2 LDA [ LIT "! ] EQU ?&ignore DUP2 wcap/ INC2 LDA [ LIT "} ] EQU ?&ignore DUP2 wcap/ INC2 LDA #00 EQU ?&ignore LDAk [ LIT "" ] EQU ?&ignore DUP2 wcap/ INC2 LDA [ LIT "& ] EQU ?&ignore LDAk [ LIT "< ] EQU ?&pass DUP2 wcap/ #0002 SUB2 LDA2 [ LIT "> ] EQU SWP [ LIT "> ] EQU ORA ?&pass DUP2 ;dict/sth wcmp ?&special-opcode DUP2 ;dict/sth2 wcmp ?&special-opcode DUP2 ;dict/deo scmp3 ?&special-opcode DUP2 ;dict/sta scmp3 ?&special-opcode DUP2 ;dict/str scmp3 ?&special-opcode DUP2 ;dict/stz scmp3 ?&special-opcode DUP2 ;dict/jcn scmp3 ?&special-opcode &ignore ( -- ) POP2 #00 JMP2r &special-opcode ( -- ) DUP2 wcap/ #0001 SUB2 LDA [ LIT "k ] EQU ?&ignore &pass ( -- ) POP2 #01 JMP2r @count-block ( str* -- length* ) [ LIT2r 0000 ] &>w ( -- ) LDAk #5d EQU ?&end INC2r INC2 LDAk ?&>w &end POP2 STH2r JMP2r @count-lambda ( str* -- length* ) LIT2r 0000 &>w ( -- ) LDAk [ LIT "} ] EQU ?&end LDAk #20 NEQ ?{ INC2r } INC2 LDAk ?&>w &end POP2 STH2r JMP2r @get-block-width ( len* -- and ) DUP2 #0002 EQU2 ?&byte DUP2 #0004 EQU2 ?&short DUP2 #000c GTH2 ?&long POP2 #03 JMP2r &byte POP2 #0f JMP2r &short POP2 #07 JMP2r &long POP2 #01 JMP2r @chex ( c -: ) [ LIT "0 ] SUB DUP #0a LTH ?&end #27 SUB DUP #10 LTH ?&end POP #ff &end JMP2r ( @|emitters ) @ ( str* -- str* ) DUP2 count-block #002a GTH2 ? &>w ( -- ) LDAk LDAk #5d EQU ?&end INC2 LDAk ?&>w LDAk &end INC2 JMP2r @ ( str* -- str* ) ( get largest of two first chunks ) ( a ) INC2k wcap/ INC2 DUP2 wlen STH2 ( b ) wcap/ INC2 wlen STH2r GTH2k [ JMP SWP2 POP2 ] ( res ) get-block-width ,&lb STR LIT2r 0000 &>w ( -- ) LDAk LDAk #20 NEQ ?{ STHkr [ LIT &lb $1 ] AND ?&no-spacer INC2k LDA #5d EQU ?&no-spacer &no-spacer INC2r } LDAk #00 EQU ?&end INC2 LDAk #5d NEQ ?&>w LDAk &end INC2 POP2r JMP2r @ ( str* -- str* ) INC2k INC2 LDA [ LIT "| ] NEQ ,&break-after STR INC2k INC2 LDA2 [ LIT2 "@| ] NEQ2 ?{ #0a LDAk #0a INC2 INC2 } &>w ( -- ) LDAk #00 EQU ?&end LDA2k [ LIT2 "| 20 ] NEQ2 ?{ DUP2 #0002 SUB2 LDA2 [ LIT2 "( 20 ] EQU2 ?{ } } LDAk INC2 LDAk #29 NEQ ?&>w LDAk &end ( str* -- str* ) [ LIT &break-after $1 ] ?{ } INC2 JMP2r @ ( -: ) ;Result/ptr LDA2 #0001 SUB2 LDA #1f GTH ? JMP2r @ ( -- ) #00 ;/depth STA JMP2r @ ( -- ) ;/depth LDA INC ;/depth STA JMP2r @ ( -- ) ;/depth LDA DUP ?{ POP JMP2r } #01 SUB ;/depth STA JMP2r @ ( -- ) #0a #09 ( | depth ) [ LIT2 &depth $1 00 ] EQUk ?{ OVR #10 GTH ?{ &>l ( -- ) #09 INC GTHk ?&>l } } POP2 JMP2r @ ( str* -: str* ) LDAk INC2 & LDAk #20 GTH ? JMP2r @ ( -- ) LDAk ?{ JMP2r } #20 ( >> ) @ ( c -- ) DUP ?{ POP JMP2r } DUP #20 NEQ ?{ ( ws ws ) Result/get-last #20 EQU ?&ignore ( tb ws ) Result/get-last #09 EQU ?&ignore } DUP #0a NEQ ?{ ( ws lb <- ) Result/get-last #20 NEQ ?{ Result/rew } ( tb lb <- ) Result/get-last #09 NEQ ?{ Result/rew } [ LIT &lb $1 ] INCk ,&lb STR #02 LTH ?Result/put POP JMP2r } Result/put [ LIT2 00 _&lb ] STR JMP2r &ignore ( c last -- ) POP JMP2r ( @|stdlib ) @ciuc ( c -: f ) [ LIT "A ] SUB #19 LTH JMP2r @slen ( str* -: len* ) DUP2 scap/ SWP2 SUB2 JMP2r @scap ( str* -: end* ) INC2 & LDAk ?scap JMP2r @ ( str* -: ) #00 ROT ROT &>w ( cap str* -- ) STAk INC2 LDAk ?&>w STA JMP2r @scmp3 ( a* b* -- f ) STH2 LDAkr LDAk STHr NEQ ?{ INC2r INC2 } LDA2r LDA2 STH2r EQU2 JMP2r @wcmp ( a* b* -- f ) STH2 &>w ( -- ) LDAk LDAkr STHr DUP2 #2020 EQU2 ?&end NEQk ?&end POP2 INC2 INC2r !&>w &end ( a b cc -- f ) NIP2 POP2r EQU JMP2r @wrew ( w* -- rew* ) DUP2 ;mem/scope EQU2 ?{ #0001 SUB2 LDAk #20 GTH ?wrew } JMP2r @wcap ( w* -: cap* ) INC2 & LDAk #20 GTH ?wcap JMP2r @wlen ( w* -: len* ) DUP2 wcap/ SWP2 SUB2 JMP2r @feof ( addr* -: f ) .File/read DEO2 .File/success DEI2 #0000 EQU2 JMP2r @ ( str* -: ) LDAk #19 DEO INC2 LDAk ? POP2 JMP2r @ ( str* -: ) LDAk #19 DEO INC2 & LDAk #20 GTH ? POP2 JMP2r ( @|memory ) @scope &new ( -- ) ;mem/scope DUP2 ,&ptr STR2 ! &put ( c -- ) #00 [ LIT2 &ptr =mem/scope ] ( ) DUP2 ;mem/scope-cap EQU2 ?&overflow ( ) INC2k ,&ptr STR2 STA2 JMP2r &overflow ( c* ptr* -- ) POP2 POP2 [ LIT2 01 -err ] STZ JMP2r @Result &new ( -- ) ;mem/result ,&ptr STR2 JMP2r &put ( c -- ) #00 [ LIT2 &ptr =mem/result ] INC2k ,&ptr STR2 STA2 JMP2r &rew ( -- ) ,&ptr LDR2 #0001 SUB2 ,&ptr STR2 JMP2r &get-last ( -- c ) ,&ptr LDR2 #0001 SUB2 LDA JMP2r @dict &usage "usage: 20 "uxnfor.rom 20 "source.tal 0a $1 &err "!! 20 "Error: 20 "Scope 20 "too 20 "large: 20 $1 &deo "DEO $1 &sta "STA $1 &stz "STZ $1 &str "STR $1 &sth "STH 20 $1 &sth2 "STH2 20 $1 &jmp "JMP $1 &jcn "JCN $1 20 &brk "BRK $1 @mem ( static buffers ) &scope $4000 &scope-cap $2 &result