-
Notifications
You must be signed in to change notification settings - Fork 63
/
win32exelink.zig
109 lines (98 loc) · 4.07 KB
/
win32exelink.zig
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
const builtin = @import("builtin");
const std = @import("std");
const log = std.log.scoped(.zigexelink);
// NOTE: to prevent the exe from having multiple markers, I can't create a separate string literal
// for the marker and get the length from that, I have to hardcode the length
const exe_marker_len = 42;
// I'm exporting this and making it mutable to make sure the compiler keeps it around
// and prevent it from evaluting its contents at comptime
export var zig_exe_string: [exe_marker_len + std.fs.max_path_bytes + 1]u8 =
("!!!THIS MARKS THE zig_exe_string MEMORY!!#" ++ ([1]u8{0} ** (std.fs.max_path_bytes + 1))).*;
const global = struct {
var child: std.process.Child = undefined;
var arena_instance = std.heap.ArenaAllocator.init(std.heap.page_allocator);
const arena = arena_instance.allocator();
};
pub fn main() !u8 {
// Sanity check that the exe_marker_len is right (note: not fullproof)
std.debug.assert(zig_exe_string[exe_marker_len - 1] == '#');
if (zig_exe_string[exe_marker_len] == 0) {
log.err("the zig target executable has not been set in the exelink", .{});
return 0xff; // fail
}
var zig_exe_len: usize = 1;
while (zig_exe_string[exe_marker_len + zig_exe_len] != 0) {
zig_exe_len += 1;
if (exe_marker_len + zig_exe_len > std.fs.max_path_bytes) {
log.err("the zig target execuable is either too big (over {}) or the exe is corrupt", .{std.fs.max_path_bytes});
return 1;
}
}
const zig_exe = zig_exe_string[exe_marker_len .. exe_marker_len + zig_exe_len :0];
const args = try std.process.argsAlloc(global.arena);
if (args.len >= 2 and std.mem.eql(u8, args[1], "exelink")) {
try std.io.getStdOut().writer().writeAll(zig_exe);
return 0;
}
args[0] = zig_exe;
// NOTE: create the process.child before calling SetConsoleCtrlHandler because it uses it
global.child = std.process.Child.init(args, global.arena);
if (0 == win32.SetConsoleCtrlHandler(consoleCtrlHandler, 1)) {
log.err("SetConsoleCtrlHandler failed, error={}", .{@intFromEnum(win32.GetLastError())});
return 0xff; // fail
}
try global.child.spawn();
return switch (try global.child.wait()) {
.Exited => |e| e,
.Signal => 0xff,
.Stopped => 0xff,
.Unknown => 0xff,
};
}
fn consoleCtrlHandler(ctrl_type: u32) callconv(@import("std").os.windows.WINAPI) win32.BOOL {
//
// NOTE: Do I need to synchronize this with the main thread?
//
const name: []const u8 = switch (ctrl_type) {
win32.CTRL_C_EVENT => "Control-C",
win32.CTRL_BREAK_EVENT => "Break",
win32.CTRL_CLOSE_EVENT => "Close",
win32.CTRL_LOGOFF_EVENT => "Logoff",
win32.CTRL_SHUTDOWN_EVENT => "Shutdown",
else => "Unknown",
};
// TODO: should we stop the process on a break event?
log.info("caught ctrl signal {d} ({s}), stopping process...", .{ ctrl_type, name });
const exit_code = switch (global.child.kill() catch |err| {
log.err("failed to kill process, error={s}", .{@errorName(err)});
std.process.exit(0xff);
}) {
.Exited => |e| e,
.Signal => 0xff,
.Stopped => 0xff,
.Unknown => 0xff,
};
std.process.exit(exit_code);
unreachable;
}
const win32 = struct {
pub const BOOL = i32;
pub const CTRL_C_EVENT = @as(u32, 0);
pub const CTRL_BREAK_EVENT = @as(u32, 1);
pub const CTRL_CLOSE_EVENT = @as(u32, 2);
pub const CTRL_LOGOFF_EVENT = @as(u32, 5);
pub const CTRL_SHUTDOWN_EVENT = @as(u32, 6);
pub const GetLastError = std.os.windows.kernel32.GetLastError;
pub const PHANDLER_ROUTINE = switch (builtin.zig_backend) {
.stage1 => fn (
CtrlType: u32,
) callconv(@import("std").os.windows.WINAPI) BOOL,
else => *const fn (
CtrlType: u32,
) callconv(@import("std").os.windows.WINAPI) BOOL,
};
pub extern "kernel32" fn SetConsoleCtrlHandler(
HandlerRoutine: ?PHANDLER_ROUTINE,
Add: BOOL,
) callconv(@import("std").os.windows.WINAPI) BOOL;
};