1
1
mirror of https://github.com/kanaka/mal.git synced 2024-09-20 18:18:51 +03:00
mal/ps/step5_tco.ps
Joel Martin 3da90d3907 PS: add step8_macros.
Use dicts rather than array block for user defined mal function type.
Add fload function to setup call from a mal_function dict.
2014-03-31 23:05:41 -05:00

149 lines
3.8 KiB
PostScript

(types.ps) run
(reader.ps) run
% read
/READ {
/str exch def
str read_str
} def
% eval
/eval_ast { 2 dict begin
/env exch def
/ast exch def
%(eval_ast: ) print ast ==
ast _symbol? { %if symbol
env ast env_get
}{ ast _list? { %elseif list
[
ast {
env EVAL
} forall
]
}{ % else
ast
} ifelse } ifelse
end } def
/EVAL { 13 dict begin
{ %loop (TCO)
/env exch def
/ast exch def
/loop? false def
%(EVAL: ) print ast true _pr_str print (\n) print
ast _list? not { %if not a list
ast env eval_ast
}{ %else apply the list
/a0 ast 0 get def
/def! a0 eq { %if def!
/a1 ast 1 get def
/a2 ast 2 get def
env a1 a2 env EVAL env_set
}{ /let* a0 eq { %if let*
/a1 ast 1 get def
/a2 ast 2 get def
/let_env env [ ] [ ] env_new def
0 2 a1 length 1 sub { %for each pair
/idx exch def
let_env
a1 idx get
a1 idx 1 add get let_env EVAL
env_set
pop % discard the return value
} for
a2 let_env EVAL
}{ /do a0 eq { %if do
ast length 2 gt { %if ast has more than 2 elements
ast 1 ast length 2 sub getinterval env eval_ast pop
} if
ast ast length 1 sub get % last ast becomes new ast
env
/loop? true def % loop
}{ /if a0 eq { %if if
/a1 ast 1 get def
/cond a1 env EVAL def
cond null eq cond false eq or { % if cond is nil or false
ast length 3 gt { %if false branch with a3
ast 3 get env
/loop? true def
}{ % else false branch with no a3
null
} ifelse
}{ % true branch
ast 2 get env
/loop? true def
} ifelse
}{ /fn* a0 eq { %if fn*
/a1 ast 1 get def
/a2 ast 2 get def
<<
/type /_maltype_function % user defined function
/params null % close over parameters
/ast null % close over ast
/env null % close over environment
>>
dup length dict copy % make an actual copy/new instance
dup /params a1 put % insert closed over a1 into position 2
dup /ast a2 put % insert closed over a2 into position 3
dup /env env put % insert closed over env into position 4
}{
/el ast env eval_ast def
el _first _mal_function? { % if user defined function
el _rest el _first fload % stack: ast new_env
/loop? true def
}{ %else (regular procedure/function)
el _rest el _first exec % apply function to args
} ifelse
} ifelse } ifelse } ifelse } ifelse } ifelse
} ifelse
loop? not { exit } if
} loop % TCO
end } def
% print
/PRINT {
true _pr_str
} def
% repl
/repl_env null [ ] [ ] env_new def
/RE { READ repl_env EVAL } def
/REP { READ repl_env EVAL PRINT } def
/_ref { repl_env 3 1 roll env_set pop } def
types_ns { _ref } forall
(\(def! not \(fn* \(a\) \(if a false true\)\)\)) RE pop
/stdin (%stdin) (r) file def
{ % loop
(user> ) print flush
stdin 99 string readline
not { exit } if % exit if EOF
%(\ngot line: ) print dup print (\n) print flush
{ %try
REP print (\n) print
} stopped {
(Error: ) print
get_error_data false _pr_str print (\n) print
$error /newerror false put
$error /errorinfo null put
clear
} if
} bind loop
(\n) print % final newline before exit for cleanliness
quit