1
1
mirror of https://github.com/kanaka/mal.git synced 2024-08-17 09:40:21 +03:00
mal/impls/zig/step3_env.zig
Joel Martin 8a19f60386 Move implementations into impls/ dir
- Reorder README to have implementation list after "learning tool"
  bullet.

- This also moves tests/ and libs/ into impls. It would be preferrable
  to have these directories at the top level.  However, this causes
  difficulties with the wasm implementations which need pre-open
  directories and have trouble with paths starting with "../../". So
  in lieu of that, symlink those directories to the top-level.

- Move the run_argv_test.sh script into the tests directory for
  general hygiene.
2020-02-10 23:50:16 -06:00

255 lines
8.8 KiB
Zig

const std = @import("std");
const warn = @import("std").debug.warn;
const reader = @import("reader.zig");
const pcre = reader.pcre;
const printer = @import("printer.zig");
const getline = @import("readline.zig").getline;
const string_eql = @import("utils.zig").string_eql;
const string_copy = @import("utils.zig").string_copy;
const string_concat = @import("utils.zig").string_concat;
const apply_function = @import("types.zig").apply_function;
const linked_list = @import("linked_list.zig");
const hash_map = @import("hmap.zig");
const Allocator = @import("std").heap.c_allocator;
const MalType = @import("types.zig").MalType;
const MalData = @import("types.zig").MalData;
const MalError = @import("error.zig").MalError;
const MalLinkedList = @import("linked_list.zig").MalLinkedList;
const Env = @import("env.zig").Env;
var repl_environment: *Env = undefined;
fn READ(a: [] u8) MalError!?*MalType {
var read = try reader.read_str(a);
var optional_mal = reader.read_form(&read);
return optional_mal;
}
fn EVAL(mal: *MalType, env: *Env) MalError!*MalType {
switch(mal.data) {
.List => |ll| {
if(ll.len == 0) {
return mal;
}
var first_mal = linked_list.first(&ll) orelse return MalError.ArgError;
var symbol = switch(first_mal.data) {
.Generic => |symbol| symbol,
else => "",
};
if(string_eql(symbol, "def!")) {
return EVAL_def(mal, env);
}
else if(string_eql(symbol, "let*")) {
return EVAL_let(mal, env);
}
else {
var new_list = try eval_ast(mal, env);
return apply_function(Allocator, (try new_list.sequence_linked_list()).*);
}
},
else => {
return eval_ast(mal, env);
},
}
}
fn EVAL_def(mal: *MalType, env: *Env) MalError!*MalType {
const first_arg = try mal.sequence_nth(1);
const second_arg = try mal.sequence_nth(2);
const second_arg_copy = try second_arg.copy(Allocator);
const symbol_name = try first_arg.as_symbol();
const new_value = try EVAL(second_arg_copy, env);
try env.set(symbol_name, new_value);
mal.delete(Allocator);
return new_value.copy(Allocator);
}
fn EVAL_let(mal: *MalType, env: *Env) MalError!*MalType {
const binding_arg = try mal.sequence_nth(1);
const eval_arg = try mal.sequence_nth(2);
const eval_arg_copy = try eval_arg.copy(Allocator);
const new_env = try Env.new(Allocator, env);
defer new_env.delete();
var binding_ll = switch(binding_arg.data) {
.List => |l| l,
.Vector => |v| v,
else => return MalError.TypeError,
};
var iterator = binding_ll.iterator();
var optional_node = iterator.next();
while(optional_node) |node| {
const key_mal = node;
const key = try key_mal.as_symbol();
const val_mal = iterator.next() orelse return MalError.ArgError;
const evaled_mal = try EVAL(val_mal, new_env);
try new_env.set(key, evaled_mal);
optional_node = iterator.next();
key_mal.delete(Allocator);
}
linked_list.destroy(Allocator, &binding_ll, true);
binding_arg.data = MalData{.Nil=undefined};
mal.delete(Allocator);
// We use eval_arg_copy, since we just deleted eval_arg above
return EVAL(eval_arg_copy, new_env);
}
fn PRINT(optional_mal: ?*MalType) MalError![] u8 {
return printer.print_str(optional_mal);
}
fn rep(environment: *Env, input: [] u8) MalError!?[] u8 {
var read_input = (try READ(input)) orelse return null;
var eval_input = try EVAL(read_input, environment);
var print_input = try PRINT(eval_input);
eval_input.delete(Allocator);
return print_input;
}
fn lookup(environment: *Env, symbol: []const u8, do_warn: bool) MalError!*MalType {
var mal = environment.get(symbol) catch |err| {
if(do_warn) {
const s1 = string_concat(Allocator, "'", symbol) catch return MalError.SystemError;
const s2 = string_concat(Allocator, s1, "' not found") catch return MalError.SystemError;
defer Allocator.free(s1);
defer Allocator.free(s2);
warn("'{}' not found.\n", symbol);
}
return MalError.KeyError;
};
var new_mal = try mal.copy(Allocator);
return new_mal;
}
fn eval_ast(mal: *MalType, env: *Env) MalError!*MalType {
switch(mal.data) {
.Generic => |symbol| {
defer mal.delete(Allocator);
return lookup(env, symbol, true);
},
.List => |*ll| {
var new_ll = MalLinkedList.init(Allocator);
var iterator = ll.iterator();
while(iterator.next()) |next_mal| {
const new_mal = try EVAL(next_mal, try env.copy(Allocator));
try linked_list.append_mal(Allocator, &new_ll, new_mal);
}
linked_list.destroy(Allocator, ll, true);
mal.shallow_destroy(Allocator);
const ret_mal = MalType.new_list(Allocator, new_ll);
return ret_mal;
},
.Vector => |*ll| {
var new_ll = MalLinkedList.init(Allocator);
var iterator = ll.iterator();
while(iterator.next()) |next_mal| {
const new_mal = try EVAL(next_mal, try env.copy(Allocator));
try linked_list.append_mal(Allocator, &new_ll, new_mal);
}
linked_list.destroy(Allocator, ll, true);
mal.shallow_destroy(Allocator);
const ret_mal = MalType.new_vector(Allocator, new_ll);
return ret_mal;
},
.HashMap => |hmap| {
var new_hashmap = try MalType.new_hashmap(Allocator);
var iterator = hmap.iterator();
var optional_pair = iterator.next();
while(true) {
const pair = optional_pair orelse break;
const key = pair.key;
const value = pair.value;
const evaled_value = try EVAL(value, try env.copy(Allocator));
try new_hashmap.hashmap_insert(key, evaled_value);
optional_pair = iterator.next();
}
hash_map.destroy(Allocator, hmap, true);
mal.shallow_destroy(Allocator);
return new_hashmap;
},
else => {
return mal;
}
}
}
const safeAdd = @import("std").math.add;
const safeSub = @import("std").math.sub;
const safeMul = @import("std").math.mul;
const safeDivFloor = @import("std").math.divFloor;
fn int_plus(a1: *MalType, a2: *MalType) MalError!*MalType {
const x = try a1.as_int();
const y = try a2.as_int();
const res = safeAdd(i64, x, y) catch return MalError.Overflow;
return MalType.new_int(Allocator, res);
}
fn int_minus(a1: *MalType, a2: *MalType) MalError!*MalType {
const x = try a1.as_int();
const y = try a2.as_int();
const res = safeSub(i64, x, y) catch return MalError.Overflow;
return MalType.new_int(Allocator, res);
}
fn int_mult(a1: *MalType, a2: *MalType) MalError!*MalType {
const x = try a1.as_int();
const y = try a2.as_int();
const res = safeMul(i64, x, y) catch return MalError.Overflow;
return MalType.new_int(Allocator, res);
}
fn int_div(a1: *MalType, a2: *MalType) MalError!*MalType {
const x = try a1.as_int();
const y = try a2.as_int();
const res = safeDivFloor(i64, x, y) catch |err| switch(err) {
error.DivisionByZero => return MalError.DivisionByZero,
else => return MalError.Overflow,
};
return MalType.new_int(Allocator, res);
}
fn make_environment() MalError!*Env {
repl_environment = try Env.new(Allocator, null);
var environment = repl_environment;
const plus_mal = try MalType.new_nil(Allocator);
plus_mal.data = MalData{.Fn2 = &int_plus};
try environment.set("+", plus_mal);
const minus_mal = try MalType.new_nil(Allocator);
minus_mal.data = MalData{.Fn2 = &int_minus};
try environment.set("-", minus_mal);
const mult_mal = try MalType.new_nil(Allocator);
mult_mal.data = MalData{.Fn2 = &int_mult};
try environment.set("*", mult_mal);
const div_mal = try MalType.new_nil(Allocator);
div_mal.data = MalData{.Fn2 = &int_div};
try environment.set("/", div_mal);
return environment;
}
pub fn main() !void {
const stdout_file = try std.io.getStdOut();
var environment = try make_environment();
while(true) {
var line = (try getline(Allocator)) orelse break;
var optional_output = rep(environment, line) catch |err| {
if(err == MalError.KeyError) {
continue;
} else {
return err;
}
};
if(optional_output) |output| {
try stdout_file.write(output);
Allocator.free(output);
try stdout_file.write("\n");
}
}
}