Type checking basics
This commit is contained in:
@@ -2,14 +2,15 @@
|
||||
|
||||
namespace Compiler;
|
||||
|
||||
public sealed class Generator(List<NodeDefinition> nodes)
|
||||
public sealed class Generator(TypedAst ast)
|
||||
{
|
||||
public static string Emit(List<NodeDefinition> nodes)
|
||||
public static string Emit(TypedAst ast)
|
||||
{
|
||||
return new Generator(nodes).Emit();
|
||||
return new Generator(ast).Emit();
|
||||
}
|
||||
|
||||
private IndentedTextWriter writer = new();
|
||||
private Dictionary<NubTypeStruct, string> structTypeNames = new();
|
||||
|
||||
private string Emit()
|
||||
{
|
||||
@@ -27,23 +28,22 @@ public sealed class Generator(List<NodeDefinition> nodes)
|
||||
|
||||
""");
|
||||
|
||||
foreach (var node in nodes.OfType<NodeDefinitionFunc>())
|
||||
for (var i = 0; i < ast.StructTypes.Count; i++)
|
||||
{
|
||||
var parameters = node.Parameters.Select(x => CType(x.Type, x.Name.Ident));
|
||||
writer.WriteLine($"{CType(node.ReturnType, node.Name.Ident)}({string.Join(", ", parameters)});");
|
||||
var structType = ast.StructTypes[i];
|
||||
structTypeNames[structType] = $"s{i}";
|
||||
}
|
||||
|
||||
writer.WriteLine();
|
||||
|
||||
foreach (var node in nodes.OfType<NodeDefinitionStruct>())
|
||||
foreach (var structType in ast.StructTypes)
|
||||
{
|
||||
writer.WriteLine($"struct {node.Name.Ident}");
|
||||
var name = structTypeNames[structType];
|
||||
writer.WriteLine($"struct {name}");
|
||||
writer.WriteLine("{");
|
||||
using (writer.Indent())
|
||||
{
|
||||
foreach (var field in node.Fields)
|
||||
foreach (var field in structType.Fields)
|
||||
{
|
||||
writer.WriteLine($"{CType(field.Type, field.Name.Ident)};");
|
||||
writer.WriteLine($"{CType(field.Type, field.Name)};");
|
||||
}
|
||||
}
|
||||
|
||||
@@ -52,7 +52,15 @@ public sealed class Generator(List<NodeDefinition> nodes)
|
||||
|
||||
writer.WriteLine();
|
||||
|
||||
foreach (var node in nodes.OfType<NodeDefinitionFunc>())
|
||||
foreach (var node in ast.Functions)
|
||||
{
|
||||
var parameters = node.Parameters.Select(x => CType(x.Type, x.Name.Ident));
|
||||
writer.WriteLine($"{CType(node.ReturnType, node.Name.Ident)}({string.Join(", ", parameters)});");
|
||||
}
|
||||
|
||||
writer.WriteLine();
|
||||
|
||||
foreach (var node in ast.Functions)
|
||||
{
|
||||
var parameters = node.Parameters.Select(x => CType(x.Type, x.Name.Ident));
|
||||
writer.WriteLine($"{CType(node.ReturnType, node.Name.Ident)}({string.Join(", ", parameters)})");
|
||||
@@ -69,29 +77,29 @@ public sealed class Generator(List<NodeDefinition> nodes)
|
||||
return writer.ToString();
|
||||
}
|
||||
|
||||
private void EmitStatement(NodeStatement node)
|
||||
private void EmitStatement(TypedNodeStatement node)
|
||||
{
|
||||
switch (node)
|
||||
{
|
||||
case NodeStatementBlock statement:
|
||||
case TypedNodeStatementBlock statement:
|
||||
EmitStatementBlock(statement);
|
||||
break;
|
||||
case NodeStatementFuncCall statement:
|
||||
case TypedNodeStatementFuncCall statement:
|
||||
EmitStatementFuncCall(statement);
|
||||
break;
|
||||
case NodeStatementReturn statement:
|
||||
case TypedNodeStatementReturn statement:
|
||||
EmitStatementReturn(statement);
|
||||
break;
|
||||
case NodeStatementVariableDeclaration statement:
|
||||
case TypedNodeStatementVariableDeclaration statement:
|
||||
EmitStatementVariableDeclaration(statement);
|
||||
break;
|
||||
case NodeStatementAssignment statement:
|
||||
case TypedNodeStatementAssignment statement:
|
||||
EmitStatementAssignment(statement);
|
||||
break;
|
||||
case NodeStatementIf statement:
|
||||
case TypedNodeStatementIf statement:
|
||||
EmitStatementIf(statement);
|
||||
break;
|
||||
case NodeStatementWhile statement:
|
||||
case TypedNodeStatementWhile statement:
|
||||
EmitStatementWhile(statement);
|
||||
break;
|
||||
default:
|
||||
@@ -99,7 +107,7 @@ public sealed class Generator(List<NodeDefinition> nodes)
|
||||
}
|
||||
}
|
||||
|
||||
private void EmitStatementBlock(NodeStatementBlock node)
|
||||
private void EmitStatementBlock(TypedNodeStatementBlock node)
|
||||
{
|
||||
writer.WriteLine("{");
|
||||
using (writer.Indent())
|
||||
@@ -111,33 +119,33 @@ public sealed class Generator(List<NodeDefinition> nodes)
|
||||
writer.WriteLine("}");
|
||||
}
|
||||
|
||||
private void EmitStatementFuncCall(NodeStatementFuncCall node)
|
||||
private void EmitStatementFuncCall(TypedNodeStatementFuncCall node)
|
||||
{
|
||||
var name = EmitExpression(node.Target);
|
||||
var parameterValues = node.Parameters.Select(EmitExpression).ToList();
|
||||
writer.WriteLine($"{name}({string.Join(", ", parameterValues)});");
|
||||
}
|
||||
|
||||
private void EmitStatementReturn(NodeStatementReturn statement)
|
||||
private void EmitStatementReturn(TypedNodeStatementReturn statement)
|
||||
{
|
||||
var value = EmitExpression(statement.Value);
|
||||
writer.WriteLine($"return {value};");
|
||||
}
|
||||
|
||||
private void EmitStatementVariableDeclaration(NodeStatementVariableDeclaration statement)
|
||||
private void EmitStatementVariableDeclaration(TypedNodeStatementVariableDeclaration statement)
|
||||
{
|
||||
var value = EmitExpression(statement.Value);
|
||||
writer.WriteLine($"{CType(statement.Type)} {statement.Name.Ident} = {value};");
|
||||
}
|
||||
|
||||
private void EmitStatementAssignment(NodeStatementAssignment statement)
|
||||
private void EmitStatementAssignment(TypedNodeStatementAssignment statement)
|
||||
{
|
||||
var target = EmitExpression(statement.Target);
|
||||
var value = EmitExpression(statement.Value);
|
||||
writer.WriteLine($"{target} = {value};");
|
||||
}
|
||||
|
||||
private void EmitStatementIf(NodeStatementIf statement)
|
||||
private void EmitStatementIf(TypedNodeStatementIf statement)
|
||||
{
|
||||
var condition = EmitExpression(statement.Condition);
|
||||
writer.WriteLine($"if ({condition})");
|
||||
@@ -152,7 +160,7 @@ public sealed class Generator(List<NodeDefinition> nodes)
|
||||
if (statement.ElseBlock != null)
|
||||
{
|
||||
writer.Write("else");
|
||||
if (statement.ElseBlock is NodeStatementIf)
|
||||
if (statement.ElseBlock is TypedNodeStatementIf)
|
||||
writer.Write(" ");
|
||||
else
|
||||
writer.WriteLine();
|
||||
@@ -167,7 +175,7 @@ public sealed class Generator(List<NodeDefinition> nodes)
|
||||
}
|
||||
}
|
||||
|
||||
private void EmitStatementWhile(NodeStatementWhile statement)
|
||||
private void EmitStatementWhile(TypedNodeStatementWhile statement)
|
||||
{
|
||||
var condition = EmitExpression(statement.Condition);
|
||||
writer.WriteLine($"while ({condition})");
|
||||
@@ -180,61 +188,61 @@ public sealed class Generator(List<NodeDefinition> nodes)
|
||||
writer.WriteLine("}");
|
||||
}
|
||||
|
||||
private string EmitExpression(NodeExpression node)
|
||||
private string EmitExpression(TypedNodeExpression node)
|
||||
{
|
||||
return node switch
|
||||
{
|
||||
NodeExpressionBinary expression => EmitExpressionBinary(expression),
|
||||
NodeExpressionUnary expression => EmitExpressionUnary(expression),
|
||||
NodeExpressionBoolLiteral expression => expression.Value.Value ? "true" : "false",
|
||||
NodeExpressionIntLiteral expression => expression.Value.Value.ToString(),
|
||||
NodeExpressionStringLiteral expression => $"(struct string){{ \"{expression.Value.Value}\", {expression.Value.Value.Length} }}",
|
||||
NodeExpressionStructLiteral expression => EmitExpressionStructLiteral(expression),
|
||||
NodeExpressionMemberAccess expression => EmitExpressionMemberAccess(expression),
|
||||
NodeExpressionIdent expression => expression.Value.Ident,
|
||||
TypedNodeExpressionBinary expression => EmitExpressionBinary(expression),
|
||||
TypedNodeExpressionUnary expression => EmitExpressionUnary(expression),
|
||||
TypedNodeExpressionBoolLiteral expression => expression.Value.Value ? "true" : "false",
|
||||
TypedNodeExpressionIntLiteral expression => expression.Value.Value.ToString(),
|
||||
TypedNodeExpressionStringLiteral expression => $"(struct string){{ \"{expression.Value.Value}\", {expression.Value.Value.Length} }}",
|
||||
TypedNodeExpressionStructLiteral expression => EmitExpressionStructLiteral(expression),
|
||||
TypedNodeExpressionMemberAccess expression => EmitExpressionMemberAccess(expression),
|
||||
TypedNodeExpressionIdent expression => expression.Value.Ident,
|
||||
_ => throw new ArgumentOutOfRangeException(nameof(node), node, null)
|
||||
};
|
||||
}
|
||||
|
||||
private string EmitExpressionBinary(NodeExpressionBinary expression)
|
||||
private string EmitExpressionBinary(TypedNodeExpressionBinary expression)
|
||||
{
|
||||
var left = EmitExpression(expression.Left);
|
||||
var right = EmitExpression(expression.Right);
|
||||
|
||||
return expression.Operation switch
|
||||
{
|
||||
NodeExpressionBinary.Op.Add => $"({left} + {right})",
|
||||
NodeExpressionBinary.Op.Subtract => $"({left} - {right})",
|
||||
NodeExpressionBinary.Op.Multiply => $"({left} * {right})",
|
||||
NodeExpressionBinary.Op.Divide => $"({left} / {right})",
|
||||
NodeExpressionBinary.Op.Modulo => $"({left} % {right})",
|
||||
NodeExpressionBinary.Op.Equal => $"({left} == {right})",
|
||||
NodeExpressionBinary.Op.NotEqual => $"({left} != {right})",
|
||||
NodeExpressionBinary.Op.LessThan => $"({left} < {right})",
|
||||
NodeExpressionBinary.Op.LessThanOrEqual => $"({left} <= {right})",
|
||||
NodeExpressionBinary.Op.GreaterThan => $"({left} > {right})",
|
||||
NodeExpressionBinary.Op.GreaterThanOrEqual => $"({left} >= {right})",
|
||||
NodeExpressionBinary.Op.LeftShift => $"({left} << {right})",
|
||||
NodeExpressionBinary.Op.RightShift => $"({left} >> {right})",
|
||||
NodeExpressionBinary.Op.LogicalAnd => $"({left} && {right})",
|
||||
NodeExpressionBinary.Op.LogicalOr => $"({left} || {right})",
|
||||
TypedNodeExpressionBinary.Op.Add => $"({left} + {right})",
|
||||
TypedNodeExpressionBinary.Op.Subtract => $"({left} - {right})",
|
||||
TypedNodeExpressionBinary.Op.Multiply => $"({left} * {right})",
|
||||
TypedNodeExpressionBinary.Op.Divide => $"({left} / {right})",
|
||||
TypedNodeExpressionBinary.Op.Modulo => $"({left} % {right})",
|
||||
TypedNodeExpressionBinary.Op.Equal => $"({left} == {right})",
|
||||
TypedNodeExpressionBinary.Op.NotEqual => $"({left} != {right})",
|
||||
TypedNodeExpressionBinary.Op.LessThan => $"({left} < {right})",
|
||||
TypedNodeExpressionBinary.Op.LessThanOrEqual => $"({left} <= {right})",
|
||||
TypedNodeExpressionBinary.Op.GreaterThan => $"({left} > {right})",
|
||||
TypedNodeExpressionBinary.Op.GreaterThanOrEqual => $"({left} >= {right})",
|
||||
TypedNodeExpressionBinary.Op.LeftShift => $"({left} << {right})",
|
||||
TypedNodeExpressionBinary.Op.RightShift => $"({left} >> {right})",
|
||||
TypedNodeExpressionBinary.Op.LogicalAnd => $"({left} && {right})",
|
||||
TypedNodeExpressionBinary.Op.LogicalOr => $"({left} || {right})",
|
||||
_ => throw new ArgumentOutOfRangeException()
|
||||
};
|
||||
}
|
||||
|
||||
private string EmitExpressionUnary(NodeExpressionUnary expression)
|
||||
private string EmitExpressionUnary(TypedNodeExpressionUnary expression)
|
||||
{
|
||||
var target = EmitExpression(expression.Target);
|
||||
|
||||
return expression.Operation switch
|
||||
{
|
||||
NodeExpressionUnary.Op.Negate => $"(-{target})",
|
||||
NodeExpressionUnary.Op.Invert => $"(!{target})",
|
||||
TypedNodeExpressionUnary.Op.Negate => $"(-{target})",
|
||||
TypedNodeExpressionUnary.Op.Invert => $"(!{target})",
|
||||
_ => throw new ArgumentOutOfRangeException()
|
||||
};
|
||||
}
|
||||
|
||||
private string EmitExpressionStructLiteral(NodeExpressionStructLiteral expression)
|
||||
private string EmitExpressionStructLiteral(TypedNodeExpressionStructLiteral expression)
|
||||
{
|
||||
var initializerValues = new Dictionary<string, string>();
|
||||
|
||||
@@ -246,27 +254,27 @@ public sealed class Generator(List<NodeDefinition> nodes)
|
||||
|
||||
var initializerStrings = initializerValues.Select(x => $".{x.Key} = {x.Value}");
|
||||
|
||||
return $"(struct {expression.Name.Ident}){{ {string.Join(", ", initializerStrings)} }}";
|
||||
return $"(struct {structTypeNames[(NubTypeStruct)expression.Type]}){{ {string.Join(", ", initializerStrings)} }}";
|
||||
}
|
||||
|
||||
private string EmitExpressionMemberAccess(NodeExpressionMemberAccess expression)
|
||||
private string EmitExpressionMemberAccess(TypedNodeExpressionMemberAccess expression)
|
||||
{
|
||||
var target = EmitExpression(expression.Target);
|
||||
return $"{target}.{expression.Name.Ident}";
|
||||
}
|
||||
|
||||
private static string CType(NodeType node, string? varName = null)
|
||||
private string CType(NubType node, string? varName = null)
|
||||
{
|
||||
return node switch
|
||||
{
|
||||
NodeTypeVoid => "void" + (varName != null ? $" {varName}" : ""),
|
||||
NodeTypeBool => "bool" + (varName != null ? $" {varName}" : ""),
|
||||
NodeTypeCustom type => $"struct {type.Name.Ident}" + (varName != null ? $" {varName}" : ""),
|
||||
NodeTypeSInt type => $"int{type.Width}_t" + (varName != null ? $" {varName}" : ""),
|
||||
NodeTypeUInt type => $"uint{type.Width}_t" + (varName != null ? $" {varName}" : ""),
|
||||
NodeTypePointer type => CType(type.To) + (varName != null ? $" *{varName}" : "*"),
|
||||
NodeTypeString => "struct string" + (varName != null ? $" {varName}" : ""),
|
||||
NodeTypeFunc type => $"{CType(type.ReturnType)} (*{varName})({string.Join(", ", type.Parameters.Select(p => CType(p)))})",
|
||||
NubTypeVoid => "void" + (varName != null ? $" {varName}" : ""),
|
||||
NubTypeBool => "bool" + (varName != null ? $" {varName}" : ""),
|
||||
NubTypeStruct type => $"struct {structTypeNames[type]}" + (varName != null ? $" {varName}" : ""),
|
||||
NubTypeSInt type => $"int{type.Width}_t" + (varName != null ? $" {varName}" : ""),
|
||||
NubTypeUInt type => $"uint{type.Width}_t" + (varName != null ? $" {varName}" : ""),
|
||||
NubTypePointer type => CType(type.To) + (varName != null ? $" *{varName}" : "*"),
|
||||
NubTypeString => "struct string" + (varName != null ? $" {varName}" : ""),
|
||||
NubTypeFunc type => $"{CType(type.ReturnType)} (*{varName})({string.Join(", ", type.Parameters.Select(p => CType(p)))})",
|
||||
_ => throw new ArgumentOutOfRangeException(nameof(node), node, null)
|
||||
};
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user