Kaynağa Gözat

brefore implementing envi stack

master
Felix Brendel 6 yıl önce
ebeveyn
işleme
e266667607
8 değiştirilmiş dosya ile 98 ekleme ve 75 silme
  1. +1
    -1
      bin/pre.slime
  2. +1
    -1
      src/built_ins.cpp
  3. +11
    -1
      src/env.cpp
  4. +27
    -34
      src/eval.cpp
  5. +9
    -6
      src/forward_decls.cpp
  6. +27
    -13
      src/io.cpp
  7. +6
    -3
      src/memory.cpp
  8. +16
    -16
      src/testing.cpp

+ 1
- 1
bin/pre.slime Dosyayı Görüntüle

@@ -21,7 +21,7 @@ condition is true.
`(if ,condition (begin @body) nil)))

(define-syntax (unless condition :rest body)
"Special form for when multiple actions should be done if a
"Special form for when multiple actions should be done if a
condition is false."
(if (= (rest body) ())
`(if ,condition nil @body)


+ 1
- 1
src/built_ins.cpp Dosyayı Görüntüle

@@ -72,7 +72,7 @@ proc built_in_load(String* file_name, Environment* env) -> Lisp_Object* {
proc built_in_import(String* file_name, Environment* env) -> Lisp_Object* {
// create new empty environment
Environment* new_env;
try new_env = Memory::create_child_environment(Globals::root_environment);
try new_env = Memory::create_child_environment(get_root_environment());
append_to_array_list(env->parents, new_env);

Environment* old_macro_env = Parser::environment_for_macros;


+ 11
- 1
src/env.cpp Dosyayı Görüntüle

@@ -48,6 +48,16 @@ proc try_lookup_symbol(Lisp_Object* node, Environment* env) -> Lisp_Object* {
return nullptr;
}

inline proc get_root_environment() -> Environment* {
using namespace Globals::Current_Execution;
return envi_stack->data[0];
}

inline proc get_current_environment() -> Environment* {
using namespace Globals::Current_Execution;
return envi_stack->data[envi_stack->next_index-1];
}

proc lookup_symbol(Lisp_Object* node, Environment* env) -> Lisp_Object* {
Lisp_Object* result = try_lookup_symbol(node, env);

@@ -68,7 +78,7 @@ proc print_indent(int indent) -> void {
}

proc print_environment_indent(Environment* env, int indent) -> void {
if(env == Globals::root_environment) {
if(env == get_root_environment()) {
print_indent(indent);
printf("[built-ins]-Environment (%lld)\n", (long long)env);
return;


+ 27
- 34
src/eval.cpp Dosyayı Görüntüle

@@ -216,7 +216,7 @@ proc parse_argument_list(Lisp_Object* arguments, Function* function) -> void {
else {
create_parsing_error("A non recognized marker was found "
"in the lambda list: ':%s'",
&arguments->value.pair.first->value.symbol.identifier);
&arguments->value.pair.first->value.symbol.identifier->data);
return;
}
}
@@ -402,7 +402,16 @@ proc eval_arguments(Lisp_Object* arguments, Environment* env, int *out_arguments
}

proc eval_expr(Lisp_Object* node, Environment* env) -> Lisp_Object* {
Globals::current_source_code = node;
using namespace Globals::Current_Execution;
append_to_array_list(call_stack, node);
defer {
// NOTE(Felix): We only delete the current entry from the call
// stack, if we did not encounter an error, otherwise we neet
// to preserve the callstack to print it later. it will be
// cleared in log_error().
if (!Globals::error)
--call_stack->next_index;
};

switch (Memory::get_type(node)) {
case Lisp_Object_Type::T:
@@ -414,12 +423,11 @@ proc eval_expr(Lisp_Object* node, Environment* env) -> Lisp_Object* {
case Lisp_Object_Type::CFunction:
return node;
case Lisp_Object_Type::Symbol: {
Lisp_Object* symbol;
try symbol = lookup_symbol(node, env);
return symbol;
Lisp_Object* value;
try value = lookup_symbol(node, env);
return value;
}
case Lisp_Object_Type::Pair: {

Lisp_Object* lispOperator;
if (Memory::get_type(node->value.pair.first) != Lisp_Object_Type::CFunction &&
Memory::get_type(node->value.pair.first) != Lisp_Object_Type::Function)
@@ -482,7 +490,7 @@ proc is_truthy(Lisp_Object* expression, Environment* env) -> bool {

proc interprete_file (char* file_name) -> Lisp_Object* {
Memory::init(4096 * 256, 1024, 4096 * 256);
Environment* root_env = Globals::root_environment;
Environment* root_env = get_root_environment();
Environment* user_env;
try user_env = Memory::create_child_environment(root_env);
Parser::environment_for_macros = user_env;
@@ -494,14 +502,12 @@ proc interprete_file (char* file_name) -> Lisp_Object* {
delete_error();
return nullptr;
}

// print(result);
return result;
}

proc interprete_stdin(bool is_emacs_repl = false) -> void {
proc interprete_stdin() -> void {
Memory::init(4096 * 256, 1024, 4096 * 256);
Environment* root_env = Globals::root_environment;
Environment* root_env = get_root_environment();
Environment* user_env = Memory::create_child_environment(root_env);
if (Globals::error) {
log_error();
@@ -515,11 +521,6 @@ proc interprete_stdin(bool is_emacs_repl = false) -> void {

char* line;

if (Globals::error) {
log_error();
delete_error();
}

Lisp_Object* parsed, * evaluated;
while (true) {
printf("> ");
@@ -534,24 +535,16 @@ proc interprete_stdin(bool is_emacs_repl = false) -> void {
continue;
}
evaluated = eval_expr(parsed, user_env);
if (is_emacs_repl) {
if (Globals::error) {
printf("((error \"%s\"))", &Globals::error->message->data);
} else {
printf("((result \"");
print(evaluated);
printf("\"))");
}
} else {
if (Globals::error) {
log_error();
delete_error();
continue;
}
if (evaluated != Memory::nil) {
print(evaluated);
printf("\n");
}

if (Globals::error) {
log_error();
delete_error();
continue;
}
if (evaluated != Memory::nil) {
print(evaluated);
printf("\n");
}

}
}

+ 9
- 6
src/forward_decls.cpp Dosyayı Görüntüle

@@ -10,9 +10,12 @@ proc is_truthy (Lisp_Object*, Environment*) -> bool;
proc list_length(Lisp_Object*) -> int;
proc load_built_ins_into_environment(Environment*) -> void;
proc parse_argument_list(Lisp_Object*, Function*) -> void;


proc print_environment(Environment*) -> void;
inline proc get_root_environment() -> Environment*;
inline proc get_current_environment() -> Environment*;

// proc get_exe_dir() -> char*;

proc Lisp_Object_Type_to_string(Lisp_Object_Type type) -> const char*;

@@ -25,7 +28,6 @@ namespace Memory {
inline proc get_type(Lisp_Object* node) -> Lisp_Object_Type;
}


namespace Parser {
extern Environment* environment_for_macros;

@@ -36,11 +38,12 @@ namespace Parser {
}

namespace Globals {
Environment* root_environment; // contains the built-ins
Log_Level log_level = Log_Level::Debug;

// TODO(Felix): make this the callstack by using a arraylist
// instead
Lisp_Object* current_source_code = nullptr;
namespace Current_Execution {
Lisp_Object_Array_List* call_stack = create_Lisp_Object_array_list();
Environment_Array_List* envi_stack = create_Environment_array_list();
}

Error* error = nullptr;
}

+ 27
- 13
src/io.cpp Dosyayı Görüntüle

@@ -340,9 +340,9 @@ proc print(Lisp_Object* node, bool print_repr = false, FILE* file = stdout) -> v

putc('(', file);

// NOTE(Felix): We cold do a while true here, however in case
// NOTE(Felix): We cuold do a while true here, however in case
// we want to print a broken list (for logging the error) we
// should do mo checks.
// should do more checks.
while (head) {
print(head->value.pair.first, print_repr, file);
head = head->value.pair.rest;
@@ -363,27 +363,41 @@ proc print(Lisp_Object* node, bool print_repr = false, FILE* file = stdout) -> v
}
}


proc print_error_location() -> void {
if (Globals::current_source_code) {
printf("%s (line %d, position %d) code:" console_red "\n ",
proc print_single_call(Lisp_Object* obj) -> void {
printf(console_cyan);
print(obj);
printf(console_normal);
printf("\n at ");
if (obj->sourceCodeLocation) {
printf("%s (line %d, position %d)",
Memory::get_c_str(
Globals::current_source_code->sourceCodeLocation->file),
Globals::current_source_code->sourceCodeLocation->line,
Globals::current_source_code->sourceCodeLocation->column);
print(Globals::current_source_code);
obj->sourceCodeLocation->file),
obj->sourceCodeLocation->line,
obj->sourceCodeLocation->column);
} else {
fputs("no source code location avaliable", stdout);
}
}

proc print_call_stack() -> void {
using Globals::Current_Execution::call_stack;

printf("callstack [%d] (most recent call last):\n", call_stack->next_index);
for (int i = 0; i < call_stack->next_index; ++i) {
printf("%2d -> ", i);
print_single_call(call_stack->data[i]);
printf("\n");
}
}

proc log_error() -> void {
fputs(console_red, stdout);
fputs(Memory::get_c_str(Globals::error->message), stdout);
puts(console_normal);

fputs(" in: " console_cyan, stdout);
print_error_location();
fputs(" in: ", stdout);
print_call_stack();
puts(console_normal);
Globals::Current_Execution::call_stack->next_index = 0;
}


+ 6
- 3
src/memory.cpp Dosyayı Görüntüle

@@ -182,8 +182,10 @@ namespace Memory {
try_void t = create_lisp_object();
set_type(t, Lisp_Object_Type::T);

try_void Globals::root_environment = create_built_ins_environment();
try_void Parser::standard_in = create_string("stdin");
try_void Parser::standard_in = create_string("stdin");

try_void Globals::Current_Execution::envi_stack->data[0] = create_built_ins_environment();
try_void Globals::Current_Execution::envi_stack->next_index = 1;
}

proc reset() -> void {
@@ -196,7 +198,8 @@ namespace Memory {
next_index_in_environment_memory = 0;
next_free_spot_in_string_memory = string_memory;

Globals::root_environment = create_built_ins_environment();
try_void Globals::Current_Execution::envi_stack->data[0] = create_built_ins_environment();
try_void Globals::Current_Execution::envi_stack->next_index = 1;
}

proc create_lisp_object_number(double number) -> Lisp_Object* {


+ 16
- 16
src/testing.cpp Dosyayı Görüntüle

@@ -197,7 +197,7 @@ proc test_eval_operands() -> testresult {
char operands_string[] = "((eval 1) (+ 1 2) \"okay\" (eval :haha))";
Lisp_Object* operands = Parser::parse_single_expression(operands_string);
int operands_length;
try operands = eval_arguments(operands, Globals::root_environment, &operands_length);
try operands = eval_arguments(operands, get_root_environment(), &operands_length);

assert_no_error();
assert_equal_int(list_length(operands), 4);
@@ -342,7 +342,7 @@ proc test_built_in_add() -> testresult {
char exp_string[] = "(+ 10 4)";
Lisp_Object* expression = Parser::parse_single_expression(exp_string);
Lisp_Object* result;
try result = eval_expr(expression, Globals::root_environment);
try result = eval_expr(expression, get_root_environment());

assert_no_error();
assert_not_null(result);
@@ -357,7 +357,7 @@ proc test_built_in_substract() -> testresult {
Lisp_Object* expression = Parser::parse_single_expression(exp_string);
Lisp_Object* result;

try result = eval_expr(expression, Globals::root_environment);
try result = eval_expr(expression, get_root_environment());

assert_no_error();
assert_not_null(result);
@@ -372,7 +372,7 @@ proc test_built_in_multiply() -> testresult {
char exp_string[] = "(* 10 4)";
Lisp_Object* expression = Parser::parse_single_expression(exp_string);
Lisp_Object* result;
try result = eval_expr(expression, Globals::root_environment);
try result = eval_expr(expression, get_root_environment());

assert_no_error();
assert_not_null(result);
@@ -387,7 +387,7 @@ proc test_built_in_divide() -> testresult {
char exp_string[] = "(/ 20 4)";
Lisp_Object* expression = Parser::parse_single_expression(exp_string);
Lisp_Object* result;
try result = eval_expr(expression, Globals::root_environment);
try result = eval_expr(expression, get_root_environment());

assert_no_error();
assert_not_null(result);
@@ -402,7 +402,7 @@ proc test_built_in_if() -> testresult {
char exp_string1[] = "(if 1 4 5)";
Lisp_Object* expression = Parser::parse_single_expression(exp_string1);
Lisp_Object* result;
try result = eval_expr(expression, Globals::root_environment);
try result = eval_expr(expression, get_root_environment());

assert_no_error();
assert_not_null(result);
@@ -411,7 +411,7 @@ proc test_built_in_if() -> testresult {

char exp_string2[] = "(if () 4 5)";
expression = Parser::parse_single_expression(exp_string2);
try result = eval_expr(expression, Globals::root_environment);
try result = eval_expr(expression, get_root_environment());

assert_no_error();
assert_not_null(result);
@@ -425,7 +425,7 @@ proc test_built_in_and() -> testresult {
char exp_string1[] = "(and 1 \"asd\" 4)";
Lisp_Object* expression = Parser::parse_single_expression(exp_string1);
Lisp_Object* result;
try result = eval_expr(expression, Globals::root_environment);
try result = eval_expr(expression, get_root_environment());

assert_no_error();
assert_not_null(result);
@@ -434,7 +434,7 @@ proc test_built_in_and() -> testresult {
// a false case
char exp_string2[] = "(and () \"asd\" 4)";
expression = Parser::parse_single_expression(exp_string2);
try result = eval_expr(expression, Globals::root_environment);
try result = eval_expr(expression, get_root_environment());

assert_no_error();
assert_not_null(result);
@@ -447,7 +447,7 @@ proc test_built_in_or() -> testresult {
char exp_string1[] = "(or \"asd\" nil)";
Lisp_Object* expression = Parser::parse_single_expression(exp_string1);
Lisp_Object* result;
try result = eval_expr(expression, Globals::root_environment);
try result = eval_expr(expression, get_root_environment());

assert_no_error();
assert_not_null(result);
@@ -456,7 +456,7 @@ proc test_built_in_or() -> testresult {
// a false case
char exp_string2[] = "(or () ())";
expression = Parser::parse_single_expression(exp_string2);
try result = eval_expr(expression, Globals::root_environment);
try result = eval_expr(expression, get_root_environment());

assert_no_error();
assert_not_null(result);
@@ -470,7 +470,7 @@ proc test_built_in_not() -> testresult {
char exp_string1[] = "(not ())";
Lisp_Object* expression = Parser::parse_single_expression(exp_string1);
Lisp_Object* result;
try result = eval_expr(expression, Globals::root_environment);
try result = eval_expr(expression, get_root_environment());

// a true case
assert_no_error();
@@ -480,7 +480,7 @@ proc test_built_in_not() -> testresult {
// a false case
char exp_string2[] = "(not \"asd xD\")";
expression = Parser::parse_single_expression(exp_string2);
try result = eval_expr(expression, Globals::root_environment);
try result = eval_expr(expression, get_root_environment());

assert_no_error();
assert_not_null(result);
@@ -491,7 +491,7 @@ proc test_built_in_not() -> testresult {

proc test_built_in_type() -> testresult {
Environment* env;
try env = Globals::root_environment;
try env = get_root_environment();

// normal type testing
char exp_string1[] = "(begin (define a 10)(type a))";
@@ -539,7 +539,7 @@ proc test_built_in_type() -> testresult {

proc test_singular_t_and_nil() -> testresult {
Environment* env;
try env = Globals::root_environment;
try env = get_root_environment();

// nil testing
char exp_string1[] = "()";
@@ -576,7 +576,7 @@ proc test_file(const char* file) -> testresult {
Memory::reset();
assert_no_error();

Environment* root_env = Globals::root_environment;
Environment* root_env = get_root_environment();
Environment* user_env = Memory::create_child_environment(root_env);
assert_no_error();



Yükleniyor…
İptal
Kaydet