const std = @import("std"); pub fn main() !void { var arena_instance = std.heap.ArenaAllocator.init(std.heap.page_allocator); defer arena_instance.deinit(); const arena = arena_instance.allocator(); const args = try std.cli.parse(struct { positional: struct { zig_src_lib_path: [:0]const u8, mingw_src_path: [:0]const u8, }, }, arena, .{}); const zig_src_lib_path = args.positional.zig_src_lib_path; const mingw_src_path = args.positional.mingw_src_path; const dest_mingw_crt_path = try std.fs.path.join(arena, &.{ zig_src_lib_path, "libc", "mingw", }); const src_mingw_crt_path = try std.fs.path.join(arena, &.{ mingw_src_path, "mingw-w64-crt", }); // Update only the set of existing files we have already chosen to include // in zig's installation. var dest_crt_dir = std.fs.cwd().openDir(dest_mingw_crt_path, .{ .iterate = true }) catch |err| { std.log.err("unable to open directory '{s}': {s}", .{ dest_mingw_crt_path, @errorName(err) }); std.process.exit(1); }; defer dest_crt_dir.close(); var src_crt_dir = std.fs.cwd().openDir(src_mingw_crt_path, .{ .iterate = true }) catch |err| { std.log.err("unable to open directory '{s}': {s}", .{ src_mingw_crt_path, @errorName(err) }); std.process.exit(1); }; defer src_crt_dir.close(); { var walker = try dest_crt_dir.walk(arena); defer walker.deinit(); var fail = false; while (try walker.next()) |entry| { if (entry.kind != .file) continue; src_crt_dir.copyFile(entry.path, dest_crt_dir, entry.path, .{}) catch |err| switch (err) { error.FileNotFound => { const keep = for (kept_crt_files) |item| { if (std.mem.eql(u8, entry.path, item)) break true; if (std.mem.startsWith(u8, entry.path, "winpthreads/")) break true; } else false; if (!keep) { std.log.warn("deleting {s}", .{entry.path}); try dest_crt_dir.deleteFile(entry.path); } }, else => { std.log.err("unable to copy {s}: {s}", .{ entry.path, @errorName(err) }); fail = true; }, }; } if (fail) std.process.exit(1); } { const dest_mingw_winpthreads_path = try std.fs.path.join(arena, &.{ zig_src_lib_path, "libc", "mingw", "winpthreads", }); const src_mingw_libraries_winpthreads_src_path = try std.fs.path.join(arena, &.{ mingw_src_path, "mingw-w64-libraries", "winpthreads", "src", }); var dest_winpthreads_dir = std.fs.cwd().openDir(dest_mingw_winpthreads_path, .{ .iterate = true }) catch |err| { std.log.err("unable to open directory '{s}': {s}", .{ dest_mingw_winpthreads_path, @errorName(err) }); std.process.exit(1); }; defer dest_winpthreads_dir.close(); var src_winpthreads_dir = std.fs.cwd().openDir(src_mingw_libraries_winpthreads_src_path, .{ .iterate = true }) catch |err| { std.log.err("unable to open directory '{s}': {s}", .{ src_mingw_libraries_winpthreads_src_path, @errorName(err) }); std.process.exit(1); }; defer src_winpthreads_dir.close(); { var walker = try dest_winpthreads_dir.walk(arena); defer walker.deinit(); var fail = false; while (try walker.next()) |entry| { if (entry.kind != .file) continue; src_winpthreads_dir.copyFile(entry.path, dest_winpthreads_dir, entry.path, .{}) catch |err| switch (err) { error.FileNotFound => { std.log.warn("deleting {s}", .{entry.path}); try dest_winpthreads_dir.deleteFile(entry.path); }, else => { std.log.err("unable to copy {s}: {s}", .{ entry.path, @errorName(err) }); fail = true; }, }; } if (fail) std.process.exit(1); } } { // Also add all new def and def.in files. var walker = try src_crt_dir.walk(arena); defer walker.deinit(); var fail = false; while (try walker.next()) |entry| { if (entry.kind != .file) continue; const ok_ext = for (def_exts) |ext| { if (std.mem.endsWith(u8, entry.path, ext)) break true; } else false; if (!ok_ext) continue; const ok_prefix = for (def_dirs) |p| { if (std.mem.startsWith(u8, entry.path, p)) break true; } else false; if (!ok_prefix) continue; const blacklisted = for (blacklisted_defs) |item| { if (std.mem.eql(u8, entry.basename, item)) break true; } else false; if (blacklisted) continue; if (std.mem.endsWith(u8, entry.basename, "_windowsapp.def")) continue; if (std.mem.endsWith(u8, entry.basename, "_onecore.def")) continue; src_crt_dir.copyFile(entry.path, dest_crt_dir, entry.path, .{}) catch |err| { std.log.err("unable to copy {s}: {s}", .{ entry.path, @errorName(err) }); fail = true; }; } if (fail) std.process.exit(1); } return std.process.cleanExit(); } const kept_crt_files = [_][]const u8{ "COPYING", "include" ++ std.fs.path.sep_str ++ "config.h", }; const def_exts = [_][]const u8{ ".def", ".def.in", }; const def_dirs = [_][]const u8{ "lib32" ++ std.fs.path.sep_str, "lib64" ++ std.fs.path.sep_str, "libarm32" ++ std.fs.path.sep_str, "libarm64" ++ std.fs.path.sep_str, "lib-common" ++ std.fs.path.sep_str, "def-include" ++ std.fs.path.sep_str, }; const blacklisted_defs = [_][]const u8{ "crtdll.def.in", "msvcp60.def", "msvcp110.def", "msvcp120_app.def.in", "msvcp120_clr0400.def", "msvcr40d.def.in", "msvcr70.def.in", "msvcr70d.def.in", "msvcr71.def.in", "msvcr71d.def.in", "msvcr80.def.in", "msvcr80d.def.in", "msvcr90.def.in", "msvcr90d.def.in", "msvcr100.def.in", "msvcr100d.def.in", "msvcr110.def.in", "msvcr110d.def.in", "msvcr120.def.in", "msvcr120d.def.in", "msvcr120_app.def.in", "msvcrt.def.in", "msvcrtd.def.in", "msvcrt10.def.in", "msvcrt20.def.in", "msvcrt40.def.in", };