Introduce scopes and compile-time known mods
This commit is contained in:
parent
38955a4d8b
commit
a2ad8e44bb
204
bootstrap/main.c
204
bootstrap/main.c
@ -2372,6 +2372,82 @@ fn void move_dependencies_to_worklist(Thread* thread, Node* node)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn String node_id_to_string(Node* node)
|
||||||
|
{
|
||||||
|
switch (node->id)
|
||||||
|
{
|
||||||
|
case_to_name(NODE_, START);
|
||||||
|
case_to_name(NODE_, STOP);
|
||||||
|
case_to_name(NODE_, CONTROL_PROJECTION);
|
||||||
|
case_to_name(NODE_, DEAD_CONTROL);
|
||||||
|
case_to_name(NODE_, SCOPE);
|
||||||
|
case_to_name(NODE_, PROJECTION);
|
||||||
|
case_to_name(NODE_, RETURN);
|
||||||
|
case_to_name(NODE_, REGION);
|
||||||
|
case_to_name(NODE_, REGION_LOOP);
|
||||||
|
case_to_name(NODE_, IF);
|
||||||
|
case_to_name(NODE_, PHI);
|
||||||
|
case_to_name(NODE_, INTEGER_ADD);
|
||||||
|
case_to_name(NODE_, INTEGER_SUBSTRACT);
|
||||||
|
case_to_name(NODE_, INTEGER_MULTIPLY);
|
||||||
|
case_to_name(NODE_, INTEGER_UNSIGNED_DIVIDE);
|
||||||
|
case_to_name(NODE_, INTEGER_SIGNED_DIVIDE);
|
||||||
|
case_to_name(NODE_, INTEGER_UNSIGNED_REMAINDER);
|
||||||
|
case_to_name(NODE_, INTEGER_SIGNED_REMAINDER);
|
||||||
|
case_to_name(NODE_, INTEGER_UNSIGNED_SHIFT_LEFT);
|
||||||
|
case_to_name(NODE_, INTEGER_SIGNED_SHIFT_LEFT);
|
||||||
|
case_to_name(NODE_, INTEGER_UNSIGNED_SHIFT_RIGHT);
|
||||||
|
case_to_name(NODE_, INTEGER_SIGNED_SHIFT_RIGHT);
|
||||||
|
case_to_name(NODE_, INTEGER_AND);
|
||||||
|
case_to_name(NODE_, INTEGER_OR);
|
||||||
|
case_to_name(NODE_, INTEGER_XOR);
|
||||||
|
case_to_name(NODE_, CONSTANT);
|
||||||
|
case_to_name(NODE_, COUNT);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn u8 node_is_unused(Node* node)
|
||||||
|
{
|
||||||
|
return node->output_count == 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn u8 node_is_dead(Node* node)
|
||||||
|
{
|
||||||
|
return node_is_unused(node) & ((node->input_count == 0) & (!validi(node->type)));
|
||||||
|
}
|
||||||
|
|
||||||
|
fn void node_kill(Thread* thread, NodeIndex node_index)
|
||||||
|
{
|
||||||
|
node_unlock(thread, node_index);
|
||||||
|
auto* node = thread_node_get(thread, node_index);
|
||||||
|
print("[NODE KILLING] (#{u32}, {s}) START\n", node_index.index, node_id_to_string(node));
|
||||||
|
assert(node_is_unused(node));
|
||||||
|
node->type = invalidi(Type);
|
||||||
|
|
||||||
|
auto inputs = node_get_inputs(thread, node);
|
||||||
|
while (node->input_count > 0)
|
||||||
|
{
|
||||||
|
auto input_index = node->input_count - 1;
|
||||||
|
node->input_count = input_index;
|
||||||
|
auto old_input_index = inputs.pointer[input_index];
|
||||||
|
|
||||||
|
print("[NODE KILLING] (#{u32}, {s}) Removing input #{u32} at slot {u32}\n", node_index.index, node_id_to_string(node), old_input_index.index, input_index);
|
||||||
|
if (validi(old_input_index))
|
||||||
|
{
|
||||||
|
thread_worklist_push(thread, old_input_index);
|
||||||
|
u8 no_more_outputs = node_remove_output(thread, old_input_index, node_index);
|
||||||
|
if (no_more_outputs)
|
||||||
|
{
|
||||||
|
print("[NODE KILLING] (#{u32}, {s}) (NO MORE OUTPUTS - KILLING) Input #{u32}\n", node_index.index, node_id_to_string(node), old_input_index.index);
|
||||||
|
node_kill(thread, old_input_index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
assert(node_is_dead(node));
|
||||||
|
// print("[NODE KILLING] (#{u32}, {s}) END\n", node_index.index, node_id_to_string(node));
|
||||||
|
}
|
||||||
|
|
||||||
fn NodeIndex node_set_input(Thread* thread, NodeIndex node_index, u16 index, NodeIndex new_input)
|
fn NodeIndex node_set_input(Thread* thread, NodeIndex node_index, u16 index, NodeIndex new_input)
|
||||||
{
|
{
|
||||||
auto* node = thread_node_get(thread, node_index);
|
auto* node = thread_node_get(thread, node_index);
|
||||||
@ -2392,7 +2468,7 @@ fn NodeIndex node_set_input(Thread* thread, NodeIndex node_index, u16 index, Nod
|
|||||||
{
|
{
|
||||||
if (node_remove_output(thread, old_input, node_index))
|
if (node_remove_output(thread, old_input, node_index))
|
||||||
{
|
{
|
||||||
trap();
|
node_kill(thread, old_input);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2432,39 +2508,6 @@ struct NodeCreate
|
|||||||
};
|
};
|
||||||
typedef struct NodeCreate NodeCreate;
|
typedef struct NodeCreate NodeCreate;
|
||||||
|
|
||||||
fn String node_id_to_string(Node* node)
|
|
||||||
{
|
|
||||||
switch (node->id)
|
|
||||||
{
|
|
||||||
case_to_name(NODE_, START);
|
|
||||||
case_to_name(NODE_, STOP);
|
|
||||||
case_to_name(NODE_, CONTROL_PROJECTION);
|
|
||||||
case_to_name(NODE_, DEAD_CONTROL);
|
|
||||||
case_to_name(NODE_, SCOPE);
|
|
||||||
case_to_name(NODE_, PROJECTION);
|
|
||||||
case_to_name(NODE_, RETURN);
|
|
||||||
case_to_name(NODE_, REGION);
|
|
||||||
case_to_name(NODE_, REGION_LOOP);
|
|
||||||
case_to_name(NODE_, IF);
|
|
||||||
case_to_name(NODE_, PHI);
|
|
||||||
case_to_name(NODE_, INTEGER_ADD);
|
|
||||||
case_to_name(NODE_, INTEGER_SUBSTRACT);
|
|
||||||
case_to_name(NODE_, INTEGER_MULTIPLY);
|
|
||||||
case_to_name(NODE_, INTEGER_UNSIGNED_DIVIDE);
|
|
||||||
case_to_name(NODE_, INTEGER_SIGNED_DIVIDE);
|
|
||||||
case_to_name(NODE_, INTEGER_UNSIGNED_REMAINDER);
|
|
||||||
case_to_name(NODE_, INTEGER_SIGNED_REMAINDER);
|
|
||||||
case_to_name(NODE_, INTEGER_UNSIGNED_SHIFT_LEFT);
|
|
||||||
case_to_name(NODE_, INTEGER_SIGNED_SHIFT_LEFT);
|
|
||||||
case_to_name(NODE_, INTEGER_UNSIGNED_SHIFT_RIGHT);
|
|
||||||
case_to_name(NODE_, INTEGER_SIGNED_SHIFT_RIGHT);
|
|
||||||
case_to_name(NODE_, INTEGER_AND);
|
|
||||||
case_to_name(NODE_, INTEGER_OR);
|
|
||||||
case_to_name(NODE_, INTEGER_XOR);
|
|
||||||
case_to_name(NODE_, CONSTANT);
|
|
||||||
case_to_name(NODE_, COUNT);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn NodeIndex thread_node_add(Thread* thread, NodeCreate data)
|
fn NodeIndex thread_node_add(Thread* thread, NodeCreate data)
|
||||||
{
|
{
|
||||||
@ -2596,7 +2639,8 @@ fn NodeIndex scope_update_extended(Thread* thread, FunctionBuilder* builder, Str
|
|||||||
|
|
||||||
if (validi(node_index))
|
if (validi(node_index))
|
||||||
{
|
{
|
||||||
trap();
|
auto result = node_set_input(thread, builder->scope, index, node_index);
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@ -2605,7 +2649,7 @@ fn NodeIndex scope_update_extended(Thread* thread, FunctionBuilder* builder, Str
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
trap();
|
return scope_update_extended(thread, builder, name, node_index, nesting_level - 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2618,6 +2662,13 @@ fn NodeIndex scope_lookup(Thread* thread, FunctionBuilder* builder, String name)
|
|||||||
return scope_update_extended(thread, builder, name, invalidi(Node), scope_node->scope.stack.length - 1);
|
return scope_update_extended(thread, builder, name, invalidi(Node), scope_node->scope.stack.length - 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn NodeIndex scope_update(Thread* thread, FunctionBuilder* builder, String name, NodeIndex value_node_index)
|
||||||
|
{
|
||||||
|
auto* scope_node = thread_node_get(thread, builder->scope);
|
||||||
|
auto result = scope_update_extended(thread, builder, name, value_node_index, scope_node->scope.stack.length - 1);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
fn u8 type_equal(Type* a, Type* b)
|
fn u8 type_equal(Type* a, Type* b)
|
||||||
{
|
{
|
||||||
u8 result = 0;
|
u8 result = 0;
|
||||||
@ -3166,17 +3217,6 @@ fn u8 type_is_constant(Type* type)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn u8 node_is_unused(Node* node)
|
|
||||||
{
|
|
||||||
return node->output_count == 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
fn u8 node_is_dead(Node* node)
|
|
||||||
{
|
|
||||||
return node_is_unused(node) & ((node->input_count == 0) & (!validi(node->type)));
|
|
||||||
}
|
|
||||||
|
|
||||||
fn TypeIndex compute_type_integer_binary(Thread* thread, NodeIndex node_index)
|
fn TypeIndex compute_type_integer_binary(Thread* thread, NodeIndex node_index)
|
||||||
{
|
{
|
||||||
auto* node = thread_node_get(thread, node_index);
|
auto* node = thread_node_get(thread, node_index);
|
||||||
@ -3868,38 +3908,6 @@ fn NodeIndex node_unkeep(Thread* thread, NodeIndex node_index)
|
|||||||
return node_index;
|
return node_index;
|
||||||
}
|
}
|
||||||
|
|
||||||
fn void node_kill(Thread* thread, NodeIndex node_index)
|
|
||||||
{
|
|
||||||
node_unlock(thread, node_index);
|
|
||||||
auto* node = thread_node_get(thread, node_index);
|
|
||||||
print("[NODE KILLING] (#{u32}, {s}) START\n", node_index.index, node_id_to_string(node));
|
|
||||||
assert(node_is_unused(node));
|
|
||||||
node->type = invalidi(Type);
|
|
||||||
|
|
||||||
auto inputs = node_get_inputs(thread, node);
|
|
||||||
while (node->input_count > 0)
|
|
||||||
{
|
|
||||||
auto input_index = node->input_count - 1;
|
|
||||||
node->input_count = input_index;
|
|
||||||
auto old_input_index = inputs.pointer[input_index];
|
|
||||||
|
|
||||||
print("[NODE KILLING] (#{u32}, {s}) Removing input #{u32} at slot {u32}\n", node_index.index, node_id_to_string(node), old_input_index.index, input_index);
|
|
||||||
if (validi(old_input_index))
|
|
||||||
{
|
|
||||||
thread_worklist_push(thread, old_input_index);
|
|
||||||
u8 no_more_outputs = node_remove_output(thread, old_input_index, node_index);
|
|
||||||
if (no_more_outputs)
|
|
||||||
{
|
|
||||||
print("[NODE KILLING] (#{u32}, {s}) (NO MORE OUTPUTS - KILLING) Input #{u32}\n", node_index.index, node_id_to_string(node), old_input_index.index);
|
|
||||||
node_kill(thread, old_input_index);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
assert(node_is_dead(node));
|
|
||||||
// print("[NODE KILLING] (#{u32}, {s}) END\n", node_index.index, node_id_to_string(node));
|
|
||||||
}
|
|
||||||
|
|
||||||
fn NodeIndex dead_code_elimination(Thread* thread, NodePair nodes)
|
fn NodeIndex dead_code_elimination(Thread* thread, NodePair nodes)
|
||||||
{
|
{
|
||||||
NodeIndex old = nodes.old;
|
NodeIndex old = nodes.old;
|
||||||
@ -4699,11 +4707,42 @@ fn void analyze_block(Thread* thread, Parser* parser, FunctionBuilder* builder,
|
|||||||
builder_add_return(thread, builder, return_node_index);
|
builder_add_return(thread, builder, return_node_index);
|
||||||
|
|
||||||
builder_set_control(thread, builder, builder->dead_control);
|
builder_set_control(thread, builder, builder->dead_control);
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
else
|
|
||||||
|
String left_name = statement_start_identifier;
|
||||||
|
|
||||||
|
skip_space(parser, src);
|
||||||
|
|
||||||
|
typedef enum AssignmentOperator
|
||||||
{
|
{
|
||||||
trap();
|
ASSIGNMENT_OPERATOR_NONE,
|
||||||
|
} AssignmentOperator;
|
||||||
|
|
||||||
|
AssignmentOperator assignment_operator;
|
||||||
|
switch (src.pointer[parser->i])
|
||||||
|
{
|
||||||
|
case '=':
|
||||||
|
assignment_operator = ASSIGNMENT_OPERATOR_NONE;
|
||||||
|
parser->i += 1;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
trap();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
skip_space(parser, src);
|
||||||
|
|
||||||
|
NodeIndex right = analyze_expression(thread, parser, builder, src, invalidi(Type));
|
||||||
|
|
||||||
|
expect_character(parser, src, ';');
|
||||||
|
|
||||||
|
auto left = scope_lookup(thread, builder, left_name);
|
||||||
|
if (!validi(left))
|
||||||
|
{
|
||||||
|
fail();
|
||||||
|
}
|
||||||
|
|
||||||
|
scope_update(thread, builder, left_name, right);
|
||||||
}
|
}
|
||||||
else if (is_decimal_digit(statement_start_ch))
|
else if (is_decimal_digit(statement_start_ch))
|
||||||
{
|
{
|
||||||
@ -4754,6 +4793,9 @@ fn void analyze_block(Thread* thread, Parser* parser, FunctionBuilder* builder,
|
|||||||
fail();
|
fail();
|
||||||
}
|
}
|
||||||
} break;
|
} break;
|
||||||
|
case block_start:
|
||||||
|
analyze_block(thread, parser, builder, src);
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
todo();
|
todo();
|
||||||
break;
|
break;
|
||||||
|
10
run_tests.sh
10
run_tests.sh
@ -8,7 +8,15 @@ build_dir="build"
|
|||||||
base_exe_name="nest"
|
base_exe_name="nest"
|
||||||
debug_flags="-g"
|
debug_flags="-g"
|
||||||
no_optimization_flags=""
|
no_optimization_flags=""
|
||||||
test_names=("first" "add_sub" "and" "or" "xor" "return_var")
|
test_names=(
|
||||||
|
"first"
|
||||||
|
"add_sub"
|
||||||
|
"and"
|
||||||
|
"or"
|
||||||
|
"xor"
|
||||||
|
"return_var"
|
||||||
|
"return_mod_scope"
|
||||||
|
)
|
||||||
|
|
||||||
if [ "$all" == "1" ]
|
if [ "$all" == "1" ]
|
||||||
then
|
then
|
||||||
|
10
tests/return_mod_scope.nat
Normal file
10
tests/return_mod_scope.nat
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
fn main() s32
|
||||||
|
{
|
||||||
|
>a: s32 = 1;
|
||||||
|
|
||||||
|
{
|
||||||
|
a = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return a;
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user