1
1
mirror of https://github.com/kanaka/mal.git synced 2024-09-20 01:57:09 +03:00
mal/objpascal/step3_env.pas
2016-04-02 17:27:43 -05:00

174 lines
4.3 KiB
ObjectPascal

program Mal;
{$H+} // Use AnsiString
Uses sysutils,
CMem,
fgl,
mal_readline,
mal_types,
mal_func,
reader,
printer,
mal_env;
var
Repl_Env : TEnv;
Line : string;
// read
function READ(const Str: string) : TMal;
begin
READ := read_str(Str);
end;
// eval
// Forward declation since eval_ast call it
function EVAL(Ast: TMal; Env: TEnv) : TMal; forward;
function eval_ast(Ast: TMal; Env: TEnv) : TMal;
var
OldArr, NewArr : TMalArray;
OldDict, NewDict : TMalDict;
I : longint;
begin
if Ast is TMalSymbol then
begin
eval_ast := Env.Get((Ast as TMalSymbol));
end
else if Ast is TMalList then
begin
OldArr := (Ast as TMalList).Val;
SetLength(NewArr, Length(OldArr));
for I := 0 to Length(OldArr)-1 do
begin
NewArr[I] := EVAL(OldArr[I], Env);
end;
if Ast is TMalVector then
eval_ast := TMalVector.Create(NewArr)
else
eval_ast := TMalList.Create(NewArr);
end
else if Ast is TMalHashMap then
begin
OldDict := (Ast as TMalHashMap).Val;
NewDict := TMalDict.Create;
I := 0;
while I < OldDict.Count do
begin
NewDict[OldDict.Keys[I]] := EVAL(OldDict[OldDict.Keys[I]], Env);
I := I + 1;
end;
eval_ast := TMalHashMap.Create(NewDict);
end
else
eval_ast := Ast;
end;
function EVAL(Ast: TMal; Env: TEnv) : TMal;
var
Arr : TMalArray;
Arr1 : TMalArray;
A0Sym : string;
LetEnv : TEnv;
I : longint;
Fn : TMalCallable;
begin
if Ast.ClassType <> TMalList then
Exit(eval_ast(Ast, Env));
// Apply list
Arr := (Ast as TMalList).Val;
if Length(Arr) = 0 then
Exit(Ast);
if Arr[0] is TMalSymbol then
A0Sym := (Arr[0] as TMalSymbol).Val
else
A0Sym := '__<*fn*>__';
case A0Sym of
'def!':
EVAL := Env.Add((Arr[1] as TMalSymbol), EVAL(Arr[2], ENV));
'let*':
begin
LetEnv := TEnv.Create(Env);
Arr1 := (Arr[1] as TMalList).Val;
I := 0;
while I < Length(Arr1) do
begin
LetEnv.Add((Arr1[I] as TMalSymbol), EVAL(Arr1[I+1], LetEnv));
Inc(I,2);
end;
EVAL := EVAL(Arr[2], LetEnv);
end;
else
begin
Arr := (eval_ast(Ast, Env) as TMalList).Val;
if Arr[0] is TMalFunc then
begin
Fn := (Arr[0] as TMalFunc).Val;
EVAL := Fn(copy(Arr, 1, Length(Arr)-1));
end
else
raise Exception.Create('invalid apply');
end;
end;
end;
// print
function PRINT(Exp: TMal) : string;
begin
PRINT := pr_str(Exp, True);
end;
// repl
function REP(Str: string) : string;
begin
REP := PRINT(EVAL(READ(Str), Repl_Env));
end;
function add(Args: TMalArray) : TMal;
begin
add := TMalInt.Create((Args[0] as TMalInt).Val +
(Args[1] as TMalInt).Val);
end;
function subtract(Args: TMalArray) : TMal;
begin
subtract := TMalInt.Create((Args[0] as TMalInt).Val -
(Args[1] as TMalInt).Val);
end;
function multiply(Args: TMalArray) : TMal;
begin
multiply := TMalInt.Create((Args[0] as TMalInt).Val *
(Args[1] as TMalInt).Val);
end;
function divide(Args: TMalArray) : TMal;
begin
divide := TMalInt.Create((Args[0] as TMalInt).Val div
(Args[1] as TMalInt).Val);
end;
begin
Repl_Env := TEnv.Create;
Repl_Env.Add(TMalSymbol.Create('+'), TMalFunc.Create(@add));
Repl_Env.Add(TMalSymbol.Create('-'), TMalFunc.Create(@subtract));
Repl_Env.Add(TMalSymbol.Create('*'), TMalFunc.Create(@multiply));
Repl_Env.Add(TMalSymbol.Create('/'), TMalFunc.Create(@divide));
while True do
begin
try
Line := _readline('user> ');
if Line = '' then continue;
WriteLn(REP(Line))
except
On E : MalEOF do Halt(0);
On E : Exception do
begin
WriteLn('Error: ' + E.message);
WriteLn('Backtrace:');
WriteLn(GetBacktrace(E));
end;
end;
end;
end.