zig/test/src/LlvmIr.zig
Linus Groh eb37552536 Remove numerous things deprecated during the 0.14 release cycle
Basically everything that has a direct replacement or no uses left.

Notable omissions:

- std.ArrayHashMap: Too much fallout, needs a separate cleanup.
- std.debug.runtime_safety: Too much fallout.
- std.heap.GeneralPurposeAllocator: Lots of references to it remain, not
  a simple find and replace as "debug allocator" is not equivalent to
  "general purpose allocator".
- std.io.Reader: Is being reworked at the moment.
- std.unicode.utf8Decode(): No replacement, needs a new API first.
- Manifest backwards compat options: Removal would break test data used
  by TestFetchBuilder.
- panic handler needs to be a namespace: Many tests still rely on it
  being a function, needs a separate cleanup.
2025-07-11 08:17:43 +02:00

135 lines
4.4 KiB
Zig

b: *std.Build,
options: Options,
root_step: *std.Build.Step,
pub const Options = struct {
enable_llvm: bool,
test_filters: []const []const u8,
test_target_filters: []const []const u8,
};
const TestCase = struct {
name: []const u8,
source: []const u8,
check: union(enum) {
matches: []const []const u8,
exact: []const u8,
},
params: Params,
pub const Params = struct {
code_model: std.builtin.CodeModel = .default,
dll_export_fns: ?bool = null,
dwarf_format: ?std.dwarf.Format = null,
error_tracing: ?bool = null,
no_builtin: ?bool = null,
omit_frame_pointer: ?bool = null,
// For most cases, we want to test the LLVM IR that we output; we don't want to be in the
// business of testing LLVM's optimization passes. `Debug` gets us the closest to that as it
// disables the vast majority of passes in LLVM.
optimize: std.builtin.OptimizeMode = .Debug,
pic: ?bool = null,
pie: ?bool = null,
red_zone: ?bool = null,
sanitize_thread: ?bool = null,
single_threaded: ?bool = null,
stack_check: ?bool = null,
stack_protector: ?bool = null,
strip: ?bool = null,
target: std.Target.Query = .{},
unwind_tables: ?std.builtin.UnwindTables = null,
valgrind: ?bool = null,
};
};
pub fn addMatches(
self: *LlvmIr,
name: []const u8,
source: []const u8,
matches: []const []const u8,
params: TestCase.Params,
) void {
self.addCase(.{
.name = name,
.source = source,
.check = .{ .matches = matches },
.params = params,
});
}
pub fn addExact(
self: *LlvmIr,
name: []const u8,
source: []const u8,
expected: []const []const u8,
params: TestCase.Params,
) void {
self.addCase(.{
.name = name,
.source = source,
.check = .{ .exact = expected },
.params = params,
});
}
pub fn addCase(self: *LlvmIr, case: TestCase) void {
const target = self.b.resolveTargetQuery(case.params.target);
if (self.options.test_target_filters.len > 0) {
const triple_txt = target.query.zigTriple(self.b.allocator) catch @panic("OOM");
for (self.options.test_target_filters) |filter| {
if (std.mem.indexOf(u8, triple_txt, filter) != null) break;
} else return;
}
const name = std.fmt.allocPrint(self.b.allocator, "check llvm-ir {s}", .{case.name}) catch @panic("OOM");
if (self.options.test_filters.len > 0) {
for (self.options.test_filters) |filter| {
if (std.mem.indexOf(u8, name, filter) != null) break;
} else return;
}
const obj = self.b.addObject(.{
.name = "test",
.root_module = self.b.createModule(.{
.root_source_file = self.b.addWriteFiles().add("test.zig", case.source),
.code_model = case.params.code_model,
.error_tracing = case.params.error_tracing,
.omit_frame_pointer = case.params.omit_frame_pointer,
.optimize = case.params.optimize,
.pic = case.params.pic,
.sanitize_thread = case.params.sanitize_thread,
.single_threaded = case.params.single_threaded,
.strip = case.params.strip,
.target = target,
.unwind_tables = case.params.unwind_tables,
}),
.use_llvm = true,
});
obj.dll_export_fns = case.params.dll_export_fns;
obj.pie = case.params.pie;
obj.root_module.dwarf_format = case.params.dwarf_format;
obj.root_module.no_builtin = case.params.no_builtin;
obj.root_module.red_zone = case.params.red_zone;
obj.root_module.stack_check = case.params.stack_check;
obj.root_module.stack_protector = case.params.stack_protector;
obj.root_module.valgrind = case.params.valgrind;
// This is not very sophisticated at the moment. Eventually, we should move towards something
// like LLVM's `FileCheck` utility (https://llvm.org/docs/CommandGuide/FileCheck.html), though
// likely a more simplified version as we probably don't want a full-blown regex engine in the
// standard library...
const check = self.b.addCheckFile(obj.getEmittedLlvmIr(), switch (case.check) {
.matches => |m| .{ .expected_matches = m },
.exact => |e| .{ .expected_exact = e },
});
check.setName(name);
self.root_step.dependOn(&check.step);
}
const LlvmIr = @This();
const std = @import("std");