This commit is contained in:
nub31
2025-09-20 19:19:40 +02:00
parent ac40648c9b
commit 8660c88278
4 changed files with 59 additions and 63 deletions

View File

@@ -467,7 +467,7 @@ public class QBEGenerator
private void EmitBlock(BlockNode block, Scope? scope = null)
{
scope ??= Scope.SubScope();
scope ??= new Scope();
_scopes.Push(scope);
foreach (var statement in block.Statements)
@@ -1295,12 +1295,6 @@ public class QBEGenerator
return new NubStructType(definition.Module, definition.Name, fieldTypes, functionTypes);
}
private NubFuncType TypeOfFunc(FuncSignatureNode signature)
{
var parameters = signature.Parameters.Select(x => x.Type).ToList();
return new NubFuncType(parameters, signature.ReturnType);
}
private string TmpName()
{
return $"%.t{++_tmpIndex}";
@@ -1342,16 +1336,10 @@ public class QBEGenerator
}
}
// todo(nub31): Parent is not used when getting variables and deferred statements
public class Scope(Scope? parent = null)
public class Scope
{
public readonly Stack<StatementNode> DeferredStatements = [];
public readonly Stack<Variable> Variables = [];
public Scope SubScope()
{
return new Scope(this);
}
}
public record Variable(string Name, NubType Type);

View File

@@ -38,6 +38,4 @@ public record ModuleStruct(bool Exported, string Name, List<ModuleStructField> F
public record ModuleFunctionParameter(string Name, TypeSyntax Type);
public record ModuleFunction(bool Exported, string Name, string? ExternSymbol, List<ModuleFunctionParameter> Parameters, TypeSyntax ReturnType);
public record ModuleTemplateStruct(bool Exported, string? Name);
public record ModuleFunction(bool Exported, string Name, string? ExternSymbol, List<ModuleFunctionParameter> Parameters, TypeSyntax ReturnType);

View File

@@ -70,19 +70,18 @@ public sealed class TypeChecker
}
}
private Scope BeginScope(bool root)
private void BeginScope(bool root)
{
var scope = root
? _globalScope.SubScope()
: _scopes.Peek().SubScope();
_scopes.Push(scope);
return scope;
}
private Scope EndScope()
private void EndScope()
{
return _scopes.Pop();
_scopes.Pop();
}
private StructNode CheckStructDefinition(StructSyntax node)
@@ -841,13 +840,24 @@ public sealed class TypeChecker
private bool AlwaysReturns(StatementNode statement)
{
return statement switch
switch (statement)
{
ReturnNode => true,
BlockNode block => block.Statements.Count != 0 && AlwaysReturns(block.Statements.Last()),
IfNode ifNode => AlwaysReturns(ifNode.Body) && ifNode.Else.TryGetValue(out var elseStatement) ? elseStatement.Match(AlwaysReturns, AlwaysReturns) : true,
_ => false
};
case ReturnNode:
return true;
case BlockNode block:
return block.Statements.Count != 0 && AlwaysReturns(block.Statements.Last());
case IfNode ifNode:
{
if (!AlwaysReturns(ifNode.Body))
{
return false;
}
return !ifNode.Else.TryGetValue(out var elseStatement) || elseStatement.Match(AlwaysReturns, AlwaysReturns);
}
default:
return false;
}
}
private NubType ResolveType(TypeSyntax type)

View File

@@ -11,49 +11,49 @@ struct Human
extern "main" func main(args: []cstring): i64
{
// let x: ref<Human> = {}
let x: ref<Human> = {}
// test(x)
test(x)
return 0
}
// func test(x: ref<Human>)
// {
func test(x: ref<Human>)
{
// }
}
// struct ref<T>
// {
// value: ^T
// count: ^u64
struct ref<T>
{
value: ^T
count: ^u64
// @oncreate
// func on_create()
// {
// puts("on_create")
// this.value = @interpret(^T, malloc(@size(T)))
// this.count = @interpret(^u64, malloc(@size(u64)))
// this.count^ = 1
// }
@oncreate
func on_create()
{
puts("on_create")
this.value = @interpret(^T, malloc(@size(T)))
this.count = @interpret(^u64, malloc(@size(u64)))
this.count^ = 1
}
// @oncopy
// func on_copy()
// {
// puts("on_copy")
// this.count^ = this.count^ + 1
// }
@oncopy
func on_copy()
{
puts("on_copy")
this.count^ = this.count^ + 1
}
// @ondestroy
// func on_destroy()
// {
// puts("on_destroy")
// this.count^ = this.count^ - 1
// if this.count^ <= 0
// {
// puts("free")
// free(@interpret(^void, this.value))
// free(@interpret(^void, this.count))
// }
// }
// }
@ondestroy
func on_destroy()
{
puts("on_destroy")
this.count^ = this.count^ - 1
if this.count^ <= 0
{
puts("free")
free(@interpret(^void, this.value))
free(@interpret(^void, this.count))
}
}
}