stage2 cbe: block results

This commit is contained in:
Veikka Tuominen
2021-01-27 11:05:22 +02:00
parent bdfe3aeab8
commit 258f3ec5ec
2 changed files with 48 additions and 11 deletions

View File

@@ -325,7 +325,7 @@ pub fn genDecl(o: *Object) !void {
const func: *Module.Fn = func_payload.data;
try o.indent_writer.insertNewline();
try o.dg.renderFunctionSignature(o.writer(), is_global);
try o.writer().writeByte(' ');
try genBody(o, func.body);
@@ -586,28 +586,44 @@ fn genDbgStmt(o: *Object, inst: *Inst.NoOp) !CValue {
fn genBlock(o: *Object, inst: *Inst.Block) !CValue {
const block_id: usize = o.next_block_index;
o.next_block_index += 1;
// abuse codegen.msv to store the block's id
inst.codegen.mcv.a = block_id;
const writer = o.writer();
// store the block id in relocs.capacity as it is not used for anything else in the C backend.
inst.codegen.relocs.capacity = block_id;
const result = if (inst.base.ty.tag() != .void and !inst.base.isUnused()) blk: {
// allocate a location for the result
const local = try o.allocLocal(inst.base.ty, .Mut);
try writer.writeAll(";\n");
break :blk local;
} else
CValue{ .none = {} };
inst.codegen.mcv = @bitCast(@import("../codegen.zig").AnyMCValue, result);
try genBody(o, inst.body);
try o.indent_writer.insertNewline();
// label must be followed by an expression, add an empty one.
try o.writer().print("zig_block_{d}:;\n", .{block_id});
// blocks in C cannot result in values
// TODO we need some other way to pass the result of the block
return CValue.none;
try writer.print("zig_block_{d}:;\n", .{block_id});
return result;
}
fn genBr(o: *Object, inst: *Inst.Br) !CValue {
if (inst.operand.ty.tag() != .void) {
return o.dg.fail(o.dg.decl.src(), "TODO: C backend: implement block return values", .{});
const result = @bitCast(CValue, inst.block.codegen.mcv);
const writer = o.writer();
// If result is .none then the value of the block is unused.
if (inst.operand.ty.tag() != .void and result != .none) {
const operand = try o.resolveInst(inst.operand);
try o.writeCValue(writer, result);
try writer.writeAll(" = ");
try o.writeCValue(writer, operand);
try writer.writeAll(";\n");
}
return genBrVoid(o, inst.block);
}
fn genBrVoid(o: *Object, block: *Inst.Block) !CValue {
try o.writer().print("goto zig_block_{d};\n", .{block.codegen.mcv.a});
try o.writer().print("goto zig_block_{d};\n", .{block.codegen.relocs.capacity});
return CValue.none;
}

View File

@@ -151,6 +151,27 @@ pub fn addCases(ctx: *TestContext) !void {
\\ unreachable;
\\}
, "");
// If expression
case.addCompareOutput(
\\export fn main() c_int {
\\ var cond: c_int = 0;
\\ var a: c_int = @as(c_int, if (cond == 0)
\\ 2
\\ else
\\ 3) + 9;
\\ exit(a - 11);
\\}
\\
\\fn exit(code: usize) noreturn {
\\ asm volatile ("syscall"
\\ :
\\ : [number] "{rax}" (231),
\\ [arg1] "{rdi}" (code)
\\ );
\\ unreachable;
\\}
, "");
}
{