b: *std.Build, step: *std.Build.Step, test_index: usize, test_filters: []const []const u8, test_target_filters: []const []const u8, const TestCase = struct { name: []const u8, sources: std.array_list.Managed(SourceFile), expected_lines: std.array_list.Managed([]const u8), allow_warnings: bool, target: std.Target.Query = .{}, const SourceFile = struct { filename: []const u8, source: []const u8, }; pub fn addSourceFile(self: *TestCase, filename: []const u8, source: []const u8) void { self.sources.append(SourceFile{ .filename = filename, .source = source, }) catch unreachable; } pub fn addExpectedLine(self: *TestCase, text: []const u8) void { self.expected_lines.append(text) catch unreachable; } }; pub fn create( self: *TranslateCContext, allow_warnings: bool, filename: []const u8, name: []const u8, source: []const u8, expected_lines: []const []const u8, ) *TestCase { const tc = self.b.allocator.create(TestCase) catch unreachable; tc.* = TestCase{ .name = name, .sources = std.array_list.Managed(TestCase.SourceFile).init(self.b.allocator), .expected_lines = std.array_list.Managed([]const u8).init(self.b.allocator), .allow_warnings = allow_warnings, }; tc.addSourceFile(filename, source); var arg_i: usize = 0; while (arg_i < expected_lines.len) : (arg_i += 1) { tc.addExpectedLine(expected_lines[arg_i]); } return tc; } pub fn add( self: *TranslateCContext, name: []const u8, source: []const u8, expected_lines: []const []const u8, ) void { const tc = self.create(false, "source.h", name, source, expected_lines); self.addCase(tc); } pub fn addWithTarget( self: *TranslateCContext, name: []const u8, target: std.Target.Query, source: []const u8, expected_lines: []const []const u8, ) void { const tc = self.create(false, "source.h", name, source, expected_lines); tc.target = target; self.addCase(tc); } pub fn addAllowWarnings( self: *TranslateCContext, name: []const u8, source: []const u8, expected_lines: []const []const u8, ) void { const tc = self.create(true, "source.h", name, source, expected_lines); self.addCase(tc); } pub fn addCase(self: *TranslateCContext, case: *const TestCase) void { const b = self.b; const translate_c_cmd = "translate-c"; const annotated_case_name = fmt.allocPrint(self.b.allocator, "{s} {s}", .{ translate_c_cmd, case.name }) catch unreachable; for (self.test_filters) |test_filter| { if (mem.indexOf(u8, annotated_case_name, test_filter)) |_| break; } else if (self.test_filters.len > 0) return; const target = b.resolveTargetQuery(case.target); if (self.test_target_filters.len > 0) { const triple_txt = target.query.zigTriple(b.allocator) catch @panic("OOM"); for (self.test_target_filters) |filter| { if (std.mem.indexOf(u8, triple_txt, filter) != null) break; } else return; } const write_src = b.addWriteFiles(); const first_src = case.sources.items[0]; const root_source_file = write_src.add(first_src.filename, first_src.source); for (case.sources.items[1..]) |src_file| { _ = write_src.add(src_file.filename, src_file.source); } const translate_c = b.addTranslateC(.{ .root_source_file = root_source_file, .target = target, .optimize = .Debug, }); translate_c.step.name = annotated_case_name; const check_file = translate_c.addCheckFile(case.expected_lines.items); self.step.dependOn(&check_file.step); } const TranslateCContext = @This(); const std = @import("std"); const fmt = std.fmt; const mem = std.mem; const fs = std.fs;