X-Git-Url: https://thelambdalab.xyz/gitweb/index.cgi?a=blobdiff_plain;f=scheme.4th;h=ecbb91baef6bd015f722321c3fe684100e245c21;hb=871b6800c52715e27f968bf1203e7f3144893a9c;hp=1f0211464f792819dcdc204d71295e33ce4527c3;hpb=4f600fd1014c0684aad06ec8fdbc8d3ee9c324a4;p=scheme.forth.jl.git diff --git a/scheme.4th b/scheme.4th index 1f02114..ecbb91b 100644 --- a/scheme.4th +++ b/scheme.4th @@ -6,13 +6,14 @@ include defer-is.4th \ ------ Types ------ -0 constant number-type +0 constant fixnum-type 1 constant boolean-type 2 constant character-type 3 constant string-type 4 constant nil-type 5 constant pair-type 6 constant symbol-type +7 constant primitive-type : istype? ( obj type -- obj bool ) over = ; @@ -257,8 +258,63 @@ global-env setobj \ }}} +\ ---- Primitives ---- {{{ + +: make-primitive ( cfa -- ) + bl word + count + + (create-symbol) + drop symbol-type + + 2dup + + symbol-table fetchobj + cons + symbol-table setobj + + rot primitive-type ( var prim ) + global-env fetchobj define-var +; + +: arg-count-error + bold fg red ." Incorrect argument count." reset-term cr + abort +; + +: ensure-arg-count ( args n -- ) + dup 0= if + drop nil objeq? false = if + arg-count-error + then + else + -rot 2dup nil objeq? if + arg-count-error + then + + cdr rot 1- recurse + then +; + +: arg-type-error + bold fg red ." Incorrect argument type." reset-term cr + abort +; + +: ensure-arg-type ( arg type -- arg ) + istype? false = if + arg-type-error + then +; + +include scheme-primitives.4th + +\ }}} + \ ---- Read ---- {{{ +defer read + variable parse-idx variable stored-parse-idx create parse-str 161 allot @@ -337,10 +393,20 @@ parse-idx-stack parse-idx-sp ! : minus? ( -- bool ) nextchar [char] - = ; -: number? ( -- bool ) - digit? minus? or false = if - false - exit +: fixnum? ( -- bool ) + minus? if + inc-parse-idx + + delim? if + dec-parse-idx + false exit + else + dec-parse-idx + then + else + digit? false = if + false exit + then then push-parse-idx @@ -440,7 +506,7 @@ parse-idx-stack parse-idx-sp ! swap if negate then - number-type + fixnum-type ; : readbool ( -- bool-atom ) @@ -566,8 +632,6 @@ parse-idx-stack parse-idx-sp ! symbol-table setobj ; -defer read - : readpair ( -- pairobj ) eatspaces @@ -615,7 +679,7 @@ defer read eatspaces - number? if + fixnum? if readnum exit then @@ -688,7 +752,7 @@ defer eval : self-evaluating? ( obj -- obj bool ) boolean-type istype? if true exit then - number-type istype? if true exit then + fixnum-type istype? if true exit then character-type istype? if true exit then string-type istype? if true exit then nil-type istype? if true exit then @@ -784,9 +848,39 @@ defer eval then ; -: true? ( boolobj -- boolean ) +: true? ( boolobj -- bool ) false? invert ; +: application? ( obj -- obj bool) + pair-type istype? ; + +: operator ( obj -- operator ) + car ; + +: operands ( obj -- operands ) + cdr ; + +: nooperands? ( operands -- bool ) + nil objeq? ; + +: first-operand ( operands -- operand ) + car ; + +: rest-operands ( operands -- other-operands ) + cdr ; + +: list-of-vals ( args env -- vals ) + 2swap + + 2dup nooperands? if + 2swap 2drop + else + 2over 2over first-operand 2swap eval + -2rot rest-operands 2swap recurse + cons + then +; + :noname ( obj env -- result ) 2swap @@ -830,6 +924,22 @@ defer eval 2swap ['] eval goto then + application? if + 2over 2over + operator 2swap eval + + primitive-type istype? false = if + bold fg red ." Object not applicable. Aboring." reset-term cr + abort + then + + -2rot + operands 2swap list-of-vals + + 2swap drop execute + exit + then + bold fg red ." Error evaluating expression - unrecognized type. Aborting." reset-term cr abort ; is eval @@ -838,6 +948,8 @@ defer eval \ ---- Print ---- {{{ +defer print + : printnum ( numobj -- ) drop 0 .R ; : printbool ( numobj -- ) @@ -887,7 +999,6 @@ defer eval : printnil ( nilobj -- ) 2drop ." ()" ; -defer print : printpair ( pairobj -- ) 2dup car print @@ -897,14 +1008,18 @@ defer print ." . " print ; +: printprim ( primobj -- ) + 2drop ." " ; + :noname ( obj -- ) - number-type istype? if printnum exit then + fixnum-type istype? if printnum exit then boolean-type istype? if printbool exit then character-type istype? if printchar exit then string-type istype? if printstring exit then symbol-type istype? if printsymbol exit then nil-type istype? if printnil exit then pair-type istype? if ." (" printpair ." )" exit then + primitive-type istype? if printprim exit then bold fg red ." Error printing expression - unrecognized type. Aborting" reset-term cr abort