aboutsummaryrefslogtreecommitdiff
path: root/src/codegen/llvm.zig
diff options
context:
space:
mode:
authorAndrew Kelley <andrew@ziglang.org>2022-03-29 22:19:06 -0700
committerAndrew Kelley <andrew@ziglang.org>2022-03-29 22:19:06 -0700
commit05947ea870f95ab90a75e174079492f546baaf72 (patch)
tree10c9a048bb1d55ce553c42c8f8e4f969241925cc /src/codegen/llvm.zig
parent83617eac5902a9e66449e8c409dfa9e560bf9f12 (diff)
downloadzig-05947ea870f95ab90a75e174079492f546baaf72.tar.gz
zig-05947ea870f95ab90a75e174079492f546baaf72.zip
stage2: implement `@intToError` with safety
This commit introduces a new AIR instruction `cmp_lt_errors_len`. It's specific to this use case for two reasons: * The total number of errors is not stable during semantic analysis; it can only be reliably checked when flush() is called. So the backend that is lowering the instruction must emit a relocation of some kind and then populate it during flush(). * The fewer AIR instructions in memory, the better for compiler performance, so we squish complex meanings into AIR tags without hesitation. The instruction is implemented only in the LLVM backend so far. It does this by creating a simple function which is gutted and re-populated with each flush(). AstGen now uses ResultLoc.coerced_ty for `@intToError` and Sema does the coercion.
Diffstat (limited to 'src/codegen/llvm.zig')
-rw-r--r--src/codegen/llvm.zig63
1 files changed, 63 insertions, 0 deletions
diff --git a/src/codegen/llvm.zig b/src/codegen/llvm.zig
index 74de485721..af68255fe8 100644
--- a/src/codegen/llvm.zig
+++ b/src/codegen/llvm.zig
@@ -440,8 +440,38 @@ pub const Object = struct {
error_name_table_ptr_global.setInitializer(error_name_table_ptr);
}
+ fn genCmpLtErrorsLenFunction(object: *Object, comp: *Compilation) !void {
+ // If there is no such function in the module, it means the source code does not need it.
+ const llvm_fn = object.llvm_module.getNamedFunction(lt_errors_fn_name) orelse return;
+ const mod = comp.bin_file.options.module.?;
+ const errors_len = mod.global_error_set.count();
+
+ // Delete previous implementation. We replace it with every flush() because the
+ // total number of errors may have changed.
+ while (llvm_fn.getFirstBasicBlock()) |bb| {
+ bb.deleteBasicBlock();
+ }
+
+ const builder = object.context.createBuilder();
+
+ const entry_block = object.context.appendBasicBlock(llvm_fn, "Entry");
+ builder.positionBuilderAtEnd(entry_block);
+ builder.clearCurrentDebugLocation();
+
+ // Example source of the following LLVM IR:
+ // fn __zig_lt_errors_len(index: u16) bool {
+ // return index < total_errors_len;
+ // }
+
+ const lhs = llvm_fn.getParam(0);
+ const rhs = lhs.typeOf().constInt(errors_len, .False);
+ const is_lt = builder.buildICmp(.ULT, lhs, rhs, "");
+ _ = builder.buildRet(is_lt);
+ }
+
pub fn flushModule(self: *Object, comp: *Compilation) !void {
try self.genErrorNameTable(comp);
+ try self.genCmpLtErrorsLenFunction(comp);
if (self.di_builder) |dib| {
// When lowering debug info for pointers, we emitted the element types as
@@ -3457,7 +3487,9 @@ pub const FuncGen = struct {
.cmp_lt => try self.airCmp(inst, .lt),
.cmp_lte => try self.airCmp(inst, .lte),
.cmp_neq => try self.airCmp(inst, .neq),
+
.cmp_vector => try self.airCmpVector(inst),
+ .cmp_lt_errors_len => try self.airCmpLtErrorsLen(inst),
.is_non_null => try self.airIsNonNull(inst, false, false, .NE),
.is_non_null_ptr => try self.airIsNonNull(inst, true , false, .NE),
@@ -3738,6 +3770,16 @@ pub const FuncGen = struct {
return self.cmp(lhs, rhs, vec_ty, cmp_op);
}
+ fn airCmpLtErrorsLen(self: *FuncGen, inst: Air.Inst.Index) !?*const llvm.Value {
+ if (self.liveness.isUnused(inst)) return null;
+
+ const un_op = self.air.instructions.items(.data)[inst].un_op;
+ const operand = try self.resolveInst(un_op);
+ const llvm_fn = try self.getCmpLtErrorsLenFunction();
+ const args: [1]*const llvm.Value = .{operand};
+ return self.builder.buildCall(llvm_fn, &args, args.len, .Fast, .Auto, "");
+ }
+
fn cmp(
self: *FuncGen,
lhs: *const llvm.Value,
@@ -6392,6 +6434,25 @@ pub const FuncGen = struct {
return fn_val;
}
+ fn getCmpLtErrorsLenFunction(self: *FuncGen) !*const llvm.Value {
+ if (self.dg.object.llvm_module.getNamedFunction(lt_errors_fn_name)) |llvm_fn| {
+ return llvm_fn;
+ }
+
+ // Function signature: fn (anyerror) bool
+
+ const ret_llvm_ty = try self.dg.llvmType(Type.bool);
+ const anyerror_llvm_ty = try self.dg.llvmType(Type.anyerror);
+ const param_types = [_]*const llvm.Type{anyerror_llvm_ty};
+
+ const fn_type = llvm.functionType(ret_llvm_ty, &param_types, param_types.len, .False);
+ const llvm_fn = self.dg.object.llvm_module.addFunction(lt_errors_fn_name, fn_type);
+ llvm_fn.setLinkage(.Internal);
+ llvm_fn.setFunctionCallConv(.Fast);
+ self.dg.addCommonFnAttributes(llvm_fn);
+ return llvm_fn;
+ }
+
fn airErrorName(self: *FuncGen, inst: Air.Inst.Index) !?*const llvm.Value {
if (self.liveness.isUnused(inst)) return null;
@@ -7663,3 +7724,5 @@ const AnnotatedDITypePtr = enum(usize) {
return @truncate(u1, @enumToInt(self)) != 0;
}
};
+
+const lt_errors_fn_name = "__zig_lt_errors_len";