feat: use CREATE_NO_WINDOW to avoid child processes creating a console window
closes #2
This commit is contained in:
parent
321abd2006
commit
7850be31f8
1 changed files with 936 additions and 6 deletions
|
@ -3,14 +3,14 @@ const cbor = @import("cbor");
|
|||
const tp = @import("thespian.zig");
|
||||
|
||||
pid: ?tp.pid,
|
||||
stdin_behavior: std.process.Child.StdIo,
|
||||
stdin_behavior: Child.StdIo,
|
||||
|
||||
const Self = @This();
|
||||
pub const max_chunk_size = 4096 - 32;
|
||||
pub const Writer = std.io.Writer(*Self, error{Exit}, write);
|
||||
pub const BufferedWriter = std.io.BufferedWriter(max_chunk_size, Writer);
|
||||
|
||||
pub fn init(a: std.mem.Allocator, argv: tp.message, tag: [:0]const u8, stdin_behavior: std.process.Child.StdIo) !Self {
|
||||
pub fn init(a: std.mem.Allocator, argv: tp.message, tag: [:0]const u8, stdin_behavior: Child.StdIo) !Self {
|
||||
return .{
|
||||
.pid = try Proc.create(a, argv, tag, stdin_behavior),
|
||||
.stdin_behavior = stdin_behavior,
|
||||
|
@ -69,7 +69,7 @@ const Proc = struct {
|
|||
receiver: Receiver,
|
||||
args: std.heap.ArenaAllocator,
|
||||
parent: tp.pid,
|
||||
child: std.process.Child,
|
||||
child: Child,
|
||||
tag: [:0]const u8,
|
||||
stdin_buffer: std.ArrayList(u8),
|
||||
stream_stdout: ?tp.file_stream = null,
|
||||
|
@ -77,7 +77,7 @@ const Proc = struct {
|
|||
|
||||
const Receiver = tp.Receiver(*Proc);
|
||||
|
||||
fn create(a: std.mem.Allocator, argv: tp.message, tag: [:0]const u8, stdin_behavior: std.process.Child.StdIo) !tp.pid {
|
||||
fn create(a: std.mem.Allocator, argv: tp.message, tag: [:0]const u8, stdin_behavior: Child.StdIo) !tp.pid {
|
||||
const self: *Proc = try a.create(Proc);
|
||||
|
||||
var args = std.heap.ArenaAllocator.init(a);
|
||||
|
@ -94,7 +94,7 @@ const Proc = struct {
|
|||
i += 1;
|
||||
}
|
||||
|
||||
var child = std.process.Child.init(argv_, a);
|
||||
var child = Child.init(argv_, a);
|
||||
child.stdin_behavior = stdin_behavior;
|
||||
child.stdout_behavior = .Pipe;
|
||||
child.stderr_behavior = .Pipe;
|
||||
|
@ -205,7 +205,7 @@ const Proc = struct {
|
|||
return self.handle_term(self.child.wait() catch |e| return self.handle_error(e));
|
||||
}
|
||||
|
||||
fn handle_term(self: *Proc, term_: std.process.Child.Term) error{Exit} {
|
||||
fn handle_term(self: *Proc, term_: Child.Term) error{Exit} {
|
||||
(switch (term_) {
|
||||
.Exited => |val| self.parent.send(.{ self.tag, "term", "exited", val }),
|
||||
.Signal => |val| self.parent.send(.{ self.tag, "term", "signal", val }),
|
||||
|
@ -220,3 +220,933 @@ const Proc = struct {
|
|||
return tp.exit_normal();
|
||||
}
|
||||
};
|
||||
|
||||
const Child = struct {
|
||||
const windows = std.os.windows;
|
||||
const posix = std.posix;
|
||||
const unicode = std.unicode;
|
||||
const process = std.process;
|
||||
const fs = std.fs;
|
||||
const mem = std.mem;
|
||||
const File = std.fs.File;
|
||||
const EnvMap = std.process.EnvMap;
|
||||
const StdIo = std.process.Child.StdIo;
|
||||
|
||||
pub const Term = union(enum) {
|
||||
Exited: u8,
|
||||
Signal: u32,
|
||||
Stopped: u32,
|
||||
Unknown: u32,
|
||||
};
|
||||
|
||||
pub const SpawnError = error{
|
||||
OutOfMemory,
|
||||
InvalidWtf8,
|
||||
CurrentWorkingDirectoryUnlinked,
|
||||
InvalidBatchScriptArg,
|
||||
} ||
|
||||
posix.ExecveError ||
|
||||
posix.SetIdError ||
|
||||
posix.ChangeCurDirError ||
|
||||
windows.CreateProcessError ||
|
||||
windows.GetProcessMemoryInfoError ||
|
||||
windows.WaitForSingleObjectError;
|
||||
|
||||
id: std.process.Child.Id,
|
||||
thread_handle: windows.HANDLE,
|
||||
allocator: mem.Allocator,
|
||||
stdin: ?File,
|
||||
stdout: ?File,
|
||||
stderr: ?File,
|
||||
term: ?(SpawnError!Term),
|
||||
argv: []const []const u8,
|
||||
env_map: ?*const EnvMap,
|
||||
stdin_behavior: StdIo,
|
||||
stdout_behavior: StdIo,
|
||||
stderr_behavior: StdIo,
|
||||
cwd: ?[]const u8,
|
||||
cwd_dir: ?fs.Dir = null,
|
||||
expand_arg0: std.process.Child.Arg0Expand,
|
||||
|
||||
pub fn init(argv: []const []const u8, allocator: mem.Allocator) @This() {
|
||||
return .{
|
||||
.allocator = allocator,
|
||||
.argv = argv,
|
||||
.id = undefined,
|
||||
.thread_handle = undefined,
|
||||
.term = null,
|
||||
.env_map = null,
|
||||
.cwd = null,
|
||||
.stdin = null,
|
||||
.stdout = null,
|
||||
.stderr = null,
|
||||
.stdin_behavior = .Inherit,
|
||||
.stdout_behavior = .Inherit,
|
||||
.stderr_behavior = .Inherit,
|
||||
.expand_arg0 = .no_expand,
|
||||
};
|
||||
}
|
||||
|
||||
pub fn spawn(self: *@This()) !void {
|
||||
var saAttr = windows.SECURITY_ATTRIBUTES{
|
||||
.nLength = @sizeOf(windows.SECURITY_ATTRIBUTES),
|
||||
.bInheritHandle = windows.TRUE,
|
||||
.lpSecurityDescriptor = null,
|
||||
};
|
||||
|
||||
const any_ignore = (self.stdin_behavior == StdIo.Ignore or self.stdout_behavior == StdIo.Ignore or self.stderr_behavior == StdIo.Ignore);
|
||||
|
||||
const nul_handle = if (any_ignore)
|
||||
// "\Device\Null" or "\??\NUL"
|
||||
windows.OpenFile(&[_]u16{ '\\', 'D', 'e', 'v', 'i', 'c', 'e', '\\', 'N', 'u', 'l', 'l' }, .{
|
||||
.access_mask = windows.GENERIC_READ | windows.GENERIC_WRITE | windows.SYNCHRONIZE,
|
||||
.share_access = windows.FILE_SHARE_READ | windows.FILE_SHARE_WRITE | windows.FILE_SHARE_DELETE,
|
||||
.sa = &saAttr,
|
||||
.creation = windows.OPEN_EXISTING,
|
||||
}) catch |err| switch (err) {
|
||||
error.PathAlreadyExists => return error.Unexpected, // not possible for "NUL"
|
||||
error.PipeBusy => return error.Unexpected, // not possible for "NUL"
|
||||
error.FileNotFound => return error.Unexpected, // not possible for "NUL"
|
||||
error.AccessDenied => return error.Unexpected, // not possible for "NUL"
|
||||
error.NameTooLong => return error.Unexpected, // not possible for "NUL"
|
||||
error.WouldBlock => return error.Unexpected, // not possible for "NUL"
|
||||
error.NetworkNotFound => return error.Unexpected, // not possible for "NUL"
|
||||
error.AntivirusInterference => return error.Unexpected, // not possible for "NUL"
|
||||
else => |e| return e,
|
||||
}
|
||||
else
|
||||
undefined;
|
||||
defer {
|
||||
if (any_ignore) posix.close(nul_handle);
|
||||
}
|
||||
|
||||
var g_hChildStd_IN_Rd: ?windows.HANDLE = null;
|
||||
var g_hChildStd_IN_Wr: ?windows.HANDLE = null;
|
||||
switch (self.stdin_behavior) {
|
||||
StdIo.Pipe => {
|
||||
try makePipeIn(&g_hChildStd_IN_Rd, &g_hChildStd_IN_Wr, &saAttr);
|
||||
},
|
||||
StdIo.Ignore => {
|
||||
g_hChildStd_IN_Rd = nul_handle;
|
||||
},
|
||||
StdIo.Inherit => {
|
||||
g_hChildStd_IN_Rd = windows.GetStdHandle(windows.STD_INPUT_HANDLE) catch null;
|
||||
},
|
||||
StdIo.Close => {
|
||||
g_hChildStd_IN_Rd = null;
|
||||
},
|
||||
}
|
||||
errdefer if (self.stdin_behavior == StdIo.Pipe) {
|
||||
destroyPipe(g_hChildStd_IN_Rd, g_hChildStd_IN_Wr);
|
||||
};
|
||||
|
||||
var g_hChildStd_OUT_Rd: ?windows.HANDLE = null;
|
||||
var g_hChildStd_OUT_Wr: ?windows.HANDLE = null;
|
||||
switch (self.stdout_behavior) {
|
||||
StdIo.Pipe => {
|
||||
try makeAsyncPipe(&g_hChildStd_OUT_Rd, &g_hChildStd_OUT_Wr, &saAttr);
|
||||
},
|
||||
StdIo.Ignore => {
|
||||
g_hChildStd_OUT_Wr = nul_handle;
|
||||
},
|
||||
StdIo.Inherit => {
|
||||
g_hChildStd_OUT_Wr = windows.GetStdHandle(windows.STD_OUTPUT_HANDLE) catch null;
|
||||
},
|
||||
StdIo.Close => {
|
||||
g_hChildStd_OUT_Wr = null;
|
||||
},
|
||||
}
|
||||
errdefer if (self.stdout_behavior == StdIo.Pipe) {
|
||||
destroyPipe(g_hChildStd_OUT_Rd, g_hChildStd_OUT_Wr);
|
||||
};
|
||||
|
||||
var g_hChildStd_ERR_Rd: ?windows.HANDLE = null;
|
||||
var g_hChildStd_ERR_Wr: ?windows.HANDLE = null;
|
||||
switch (self.stderr_behavior) {
|
||||
StdIo.Pipe => {
|
||||
try makeAsyncPipe(&g_hChildStd_ERR_Rd, &g_hChildStd_ERR_Wr, &saAttr);
|
||||
},
|
||||
StdIo.Ignore => {
|
||||
g_hChildStd_ERR_Wr = nul_handle;
|
||||
},
|
||||
StdIo.Inherit => {
|
||||
g_hChildStd_ERR_Wr = windows.GetStdHandle(windows.STD_ERROR_HANDLE) catch null;
|
||||
},
|
||||
StdIo.Close => {
|
||||
g_hChildStd_ERR_Wr = null;
|
||||
},
|
||||
}
|
||||
errdefer if (self.stderr_behavior == StdIo.Pipe) {
|
||||
destroyPipe(g_hChildStd_ERR_Rd, g_hChildStd_ERR_Wr);
|
||||
};
|
||||
|
||||
var siStartInfo = windows.STARTUPINFOW{
|
||||
.cb = @sizeOf(windows.STARTUPINFOW),
|
||||
.hStdError = g_hChildStd_ERR_Wr,
|
||||
.hStdOutput = g_hChildStd_OUT_Wr,
|
||||
.hStdInput = g_hChildStd_IN_Rd,
|
||||
.dwFlags = windows.STARTF_USESTDHANDLES,
|
||||
|
||||
.lpReserved = null,
|
||||
.lpDesktop = null,
|
||||
.lpTitle = null,
|
||||
.dwX = 0,
|
||||
.dwY = 0,
|
||||
.dwXSize = 0,
|
||||
.dwYSize = 0,
|
||||
.dwXCountChars = 0,
|
||||
.dwYCountChars = 0,
|
||||
.dwFillAttribute = 0,
|
||||
.wShowWindow = 0,
|
||||
.cbReserved2 = 0,
|
||||
.lpReserved2 = null,
|
||||
};
|
||||
var piProcInfo: windows.PROCESS_INFORMATION = undefined;
|
||||
|
||||
const cwd_w = if (self.cwd) |cwd| try unicode.wtf8ToWtf16LeAllocZ(self.allocator, cwd) else null;
|
||||
defer if (cwd_w) |cwd| self.allocator.free(cwd);
|
||||
const cwd_w_ptr = if (cwd_w) |cwd| cwd.ptr else null;
|
||||
|
||||
const maybe_envp_buf = if (self.env_map) |env_map| try process.createWindowsEnvBlock(self.allocator, env_map) else null;
|
||||
defer if (maybe_envp_buf) |envp_buf| self.allocator.free(envp_buf);
|
||||
const envp_ptr = if (maybe_envp_buf) |envp_buf| envp_buf.ptr else null;
|
||||
|
||||
const app_name_wtf8 = self.argv[0];
|
||||
const app_name_is_absolute = fs.path.isAbsolute(app_name_wtf8);
|
||||
|
||||
var cwd_path_w_needs_free = false;
|
||||
const cwd_path_w = x: {
|
||||
if (app_name_is_absolute) {
|
||||
cwd_path_w_needs_free = true;
|
||||
const dir = fs.path.dirname(app_name_wtf8).?;
|
||||
break :x try unicode.wtf8ToWtf16LeAllocZ(self.allocator, dir);
|
||||
} else if (self.cwd) |cwd| {
|
||||
cwd_path_w_needs_free = true;
|
||||
break :x try unicode.wtf8ToWtf16LeAllocZ(self.allocator, cwd);
|
||||
} else {
|
||||
break :x &[_:0]u16{}; // empty for cwd
|
||||
}
|
||||
};
|
||||
defer if (cwd_path_w_needs_free) self.allocator.free(cwd_path_w);
|
||||
|
||||
const app_basename_wtf8 = fs.path.basename(app_name_wtf8);
|
||||
const maybe_app_dirname_wtf8 = if (!app_name_is_absolute) fs.path.dirname(app_name_wtf8) else null;
|
||||
const app_dirname_w: ?[:0]u16 = x: {
|
||||
if (maybe_app_dirname_wtf8) |app_dirname_wtf8| {
|
||||
break :x try unicode.wtf8ToWtf16LeAllocZ(self.allocator, app_dirname_wtf8);
|
||||
}
|
||||
break :x null;
|
||||
};
|
||||
defer if (app_dirname_w != null) self.allocator.free(app_dirname_w.?);
|
||||
|
||||
const app_name_w = try unicode.wtf8ToWtf16LeAllocZ(self.allocator, app_basename_wtf8);
|
||||
defer self.allocator.free(app_name_w);
|
||||
|
||||
run: {
|
||||
const PATH: [:0]const u16 = process.getenvW(unicode.utf8ToUtf16LeStringLiteral("PATH")) orelse &[_:0]u16{};
|
||||
const PATHEXT: [:0]const u16 = process.getenvW(unicode.utf8ToUtf16LeStringLiteral("PATHEXT")) orelse &[_:0]u16{};
|
||||
|
||||
var cmd_line_cache = CommandLineCache.init(self.allocator, self.argv);
|
||||
defer cmd_line_cache.deinit();
|
||||
|
||||
var app_buf = std.ArrayListUnmanaged(u16){};
|
||||
defer app_buf.deinit(self.allocator);
|
||||
|
||||
try app_buf.appendSlice(self.allocator, app_name_w);
|
||||
|
||||
var dir_buf = std.ArrayListUnmanaged(u16){};
|
||||
defer dir_buf.deinit(self.allocator);
|
||||
|
||||
if (cwd_path_w.len > 0) {
|
||||
try dir_buf.appendSlice(self.allocator, cwd_path_w);
|
||||
}
|
||||
if (app_dirname_w) |app_dir| {
|
||||
if (dir_buf.items.len > 0) try dir_buf.append(self.allocator, fs.path.sep);
|
||||
try dir_buf.appendSlice(self.allocator, app_dir);
|
||||
}
|
||||
if (dir_buf.items.len > 0) {
|
||||
const normalized_len = windows.normalizePath(u16, dir_buf.items) catch return error.BadPathName;
|
||||
dir_buf.shrinkRetainingCapacity(normalized_len);
|
||||
}
|
||||
|
||||
createProcessPathExt(self.allocator, &dir_buf, &app_buf, PATHEXT, &cmd_line_cache, envp_ptr, cwd_w_ptr, &siStartInfo, &piProcInfo) catch |no_path_err| {
|
||||
const original_err = switch (no_path_err) {
|
||||
error.InvalidArg0 => return error.FileNotFound,
|
||||
error.FileNotFound, error.InvalidExe, error.AccessDenied => |e| e,
|
||||
error.UnrecoverableInvalidExe => return error.InvalidExe,
|
||||
else => |e| return e,
|
||||
};
|
||||
|
||||
if (app_dirname_w != null or app_name_is_absolute) {
|
||||
return original_err;
|
||||
}
|
||||
|
||||
var it = mem.tokenizeScalar(u16, PATH, ';');
|
||||
while (it.next()) |search_path| {
|
||||
dir_buf.clearRetainingCapacity();
|
||||
try dir_buf.appendSlice(self.allocator, search_path);
|
||||
const normalized_len = windows.normalizePath(u16, dir_buf.items) catch continue;
|
||||
dir_buf.shrinkRetainingCapacity(normalized_len);
|
||||
|
||||
if (createProcessPathExt(self.allocator, &dir_buf, &app_buf, PATHEXT, &cmd_line_cache, envp_ptr, cwd_w_ptr, &siStartInfo, &piProcInfo)) {
|
||||
break :run;
|
||||
} else |err| switch (err) {
|
||||
error.InvalidArg0 => return error.FileNotFound,
|
||||
error.FileNotFound, error.AccessDenied, error.InvalidExe => continue,
|
||||
error.UnrecoverableInvalidExe => return error.InvalidExe,
|
||||
else => |e| return e,
|
||||
}
|
||||
} else {
|
||||
return original_err;
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
if (g_hChildStd_IN_Wr) |h| {
|
||||
self.stdin = File{ .handle = h };
|
||||
} else {
|
||||
self.stdin = null;
|
||||
}
|
||||
if (g_hChildStd_OUT_Rd) |h| {
|
||||
self.stdout = File{ .handle = h };
|
||||
} else {
|
||||
self.stdout = null;
|
||||
}
|
||||
if (g_hChildStd_ERR_Rd) |h| {
|
||||
self.stderr = File{ .handle = h };
|
||||
} else {
|
||||
self.stderr = null;
|
||||
}
|
||||
|
||||
self.id = piProcInfo.hProcess;
|
||||
self.thread_handle = piProcInfo.hThread;
|
||||
self.term = null;
|
||||
|
||||
if (self.stdin_behavior == StdIo.Pipe) {
|
||||
posix.close(g_hChildStd_IN_Rd.?);
|
||||
}
|
||||
if (self.stderr_behavior == StdIo.Pipe) {
|
||||
posix.close(g_hChildStd_ERR_Wr.?);
|
||||
}
|
||||
if (self.stdout_behavior == StdIo.Pipe) {
|
||||
posix.close(g_hChildStd_OUT_Wr.?);
|
||||
}
|
||||
}
|
||||
|
||||
fn makePipeIn(rd: *?windows.HANDLE, wr: *?windows.HANDLE, sattr: *const windows.SECURITY_ATTRIBUTES) !void {
|
||||
var rd_h: windows.HANDLE = undefined;
|
||||
var wr_h: windows.HANDLE = undefined;
|
||||
try windows.CreatePipe(&rd_h, &wr_h, sattr);
|
||||
errdefer destroyPipe(rd_h, wr_h);
|
||||
try windows.SetHandleInformation(wr_h, windows.HANDLE_FLAG_INHERIT, 0);
|
||||
rd.* = rd_h;
|
||||
wr.* = wr_h;
|
||||
}
|
||||
|
||||
fn destroyPipe(rd: ?windows.HANDLE, wr: ?windows.HANDLE) void {
|
||||
if (rd) |h| posix.close(h);
|
||||
if (wr) |h| posix.close(h);
|
||||
}
|
||||
|
||||
var pipe_name_counter = std.atomic.Value(u32).init(1);
|
||||
|
||||
fn makeAsyncPipe(rd: *?windows.HANDLE, wr: *?windows.HANDLE, sattr: *const windows.SECURITY_ATTRIBUTES) !void {
|
||||
var tmp_bufw: [128]u16 = undefined;
|
||||
|
||||
const pipe_path = blk: {
|
||||
var tmp_buf: [128]u8 = undefined;
|
||||
const pipe_path = std.fmt.bufPrintZ(
|
||||
&tmp_buf,
|
||||
"\\\\.\\pipe\\zig-childprocess-{d}-{d}",
|
||||
.{ windows.GetCurrentProcessId(), pipe_name_counter.fetchAdd(1, .monotonic) },
|
||||
) catch unreachable;
|
||||
const len = std.unicode.wtf8ToWtf16Le(&tmp_bufw, pipe_path) catch unreachable;
|
||||
tmp_bufw[len] = 0;
|
||||
break :blk tmp_bufw[0..len :0];
|
||||
};
|
||||
|
||||
const read_handle = windows.kernel32.CreateNamedPipeW(
|
||||
pipe_path.ptr,
|
||||
windows.PIPE_ACCESS_INBOUND | windows.FILE_FLAG_OVERLAPPED,
|
||||
windows.PIPE_TYPE_BYTE,
|
||||
1,
|
||||
4096,
|
||||
4096,
|
||||
0,
|
||||
sattr,
|
||||
);
|
||||
if (read_handle == windows.INVALID_HANDLE_VALUE) {
|
||||
switch (windows.kernel32.GetLastError()) {
|
||||
else => |err| return windows.unexpectedError(err),
|
||||
}
|
||||
}
|
||||
errdefer posix.close(read_handle);
|
||||
|
||||
var sattr_copy = sattr.*;
|
||||
const write_handle = windows.kernel32.CreateFileW(
|
||||
pipe_path.ptr,
|
||||
windows.GENERIC_WRITE,
|
||||
0,
|
||||
&sattr_copy,
|
||||
windows.OPEN_EXISTING,
|
||||
windows.FILE_ATTRIBUTE_NORMAL,
|
||||
null,
|
||||
);
|
||||
if (write_handle == windows.INVALID_HANDLE_VALUE) {
|
||||
switch (windows.kernel32.GetLastError()) {
|
||||
else => |err| return windows.unexpectedError(err),
|
||||
}
|
||||
}
|
||||
errdefer posix.close(write_handle);
|
||||
|
||||
try windows.SetHandleInformation(read_handle, windows.HANDLE_FLAG_INHERIT, 0);
|
||||
|
||||
rd.* = read_handle;
|
||||
wr.* = write_handle;
|
||||
}
|
||||
|
||||
const CommandLineCache = struct {
|
||||
cmd_line: ?[:0]u16 = null,
|
||||
script_cmd_line: ?[:0]u16 = null,
|
||||
cmd_exe_path: ?[:0]u16 = null,
|
||||
argv: []const []const u8,
|
||||
allocator: mem.Allocator,
|
||||
|
||||
fn init(allocator: mem.Allocator, argv: []const []const u8) CommandLineCache {
|
||||
return .{
|
||||
.allocator = allocator,
|
||||
.argv = argv,
|
||||
};
|
||||
}
|
||||
|
||||
fn deinit(self: *CommandLineCache) void {
|
||||
if (self.cmd_line) |cmd_line| self.allocator.free(cmd_line);
|
||||
if (self.script_cmd_line) |script_cmd_line| self.allocator.free(script_cmd_line);
|
||||
if (self.cmd_exe_path) |cmd_exe_path| self.allocator.free(cmd_exe_path);
|
||||
}
|
||||
|
||||
fn commandLine(self: *CommandLineCache) ![:0]u16 {
|
||||
if (self.cmd_line == null) {
|
||||
self.cmd_line = try argvToCommandLine(self.allocator, self.argv);
|
||||
}
|
||||
return self.cmd_line.?;
|
||||
}
|
||||
|
||||
fn scriptCommandLine(self: *CommandLineCache, script_path: []const u16) ![:0]u16 {
|
||||
if (self.script_cmd_line) |v| self.allocator.free(v);
|
||||
self.script_cmd_line = try argvToScriptCommandLine(
|
||||
self.allocator,
|
||||
script_path,
|
||||
self.argv[1..],
|
||||
);
|
||||
return self.script_cmd_line.?;
|
||||
}
|
||||
|
||||
fn cmdExePath(self: *CommandLineCache) ![:0]u16 {
|
||||
if (self.cmd_exe_path == null) {
|
||||
self.cmd_exe_path = try wcdExePath(self.allocator);
|
||||
}
|
||||
return self.cmd_exe_path.?;
|
||||
}
|
||||
};
|
||||
|
||||
fn createProcessPathExt(
|
||||
allocator: mem.Allocator,
|
||||
dir_buf: *std.ArrayListUnmanaged(u16),
|
||||
app_buf: *std.ArrayListUnmanaged(u16),
|
||||
pathext: [:0]const u16,
|
||||
cmd_line_cache: *CommandLineCache,
|
||||
envp_ptr: ?[*]u16,
|
||||
cwd_ptr: ?[*:0]u16,
|
||||
lpStartupInfo: *windows.STARTUPINFOW,
|
||||
lpProcessInformation: *windows.PROCESS_INFORMATION,
|
||||
) !void {
|
||||
const app_name_len = app_buf.items.len;
|
||||
const dir_path_len = dir_buf.items.len;
|
||||
|
||||
if (app_name_len == 0) return error.FileNotFound;
|
||||
|
||||
defer app_buf.shrinkRetainingCapacity(app_name_len);
|
||||
defer dir_buf.shrinkRetainingCapacity(dir_path_len);
|
||||
|
||||
var dir = dir: {
|
||||
try dir_buf.append(allocator, 0);
|
||||
defer dir_buf.shrinkRetainingCapacity(dir_path_len);
|
||||
const dir_path_z = dir_buf.items[0 .. dir_buf.items.len - 1 :0];
|
||||
const prefixed_path = try windows.wToPrefixedFileW(null, dir_path_z);
|
||||
break :dir fs.cwd().openDirW(prefixed_path.span().ptr, .{ .iterate = true }) catch
|
||||
return error.FileNotFound;
|
||||
};
|
||||
defer dir.close();
|
||||
|
||||
try app_buf.append(allocator, '*');
|
||||
try app_buf.append(allocator, 0);
|
||||
const app_name_wildcard = app_buf.items[0 .. app_buf.items.len - 1 :0];
|
||||
|
||||
var file_information_buf: [2048]u8 align(@alignOf(windows.FILE_DIRECTORY_INFORMATION)) = undefined;
|
||||
const file_info_maximum_single_entry_size = @sizeOf(windows.FILE_DIRECTORY_INFORMATION) + (windows.NAME_MAX * 2);
|
||||
if (file_information_buf.len < file_info_maximum_single_entry_size) {
|
||||
@compileError("file_information_buf must be large enough to contain at least one maximum size FILE_DIRECTORY_INFORMATION entry");
|
||||
}
|
||||
var io_status: windows.IO_STATUS_BLOCK = undefined;
|
||||
|
||||
const num_supported_pathext = @typeInfo(CreateProcessSupportedExtension).Enum.fields.len;
|
||||
var pathext_seen = [_]bool{false} ** num_supported_pathext;
|
||||
var any_pathext_seen = false;
|
||||
var unappended_exists = false;
|
||||
|
||||
while (true) {
|
||||
const app_name_len_bytes = std.math.cast(u16, app_name_wildcard.len * 2) orelse return error.NameTooLong;
|
||||
var app_name_unicode_string = windows.UNICODE_STRING{
|
||||
.Length = app_name_len_bytes,
|
||||
.MaximumLength = app_name_len_bytes,
|
||||
.Buffer = @constCast(app_name_wildcard.ptr),
|
||||
};
|
||||
const rc = windows.ntdll.NtQueryDirectoryFile(
|
||||
dir.fd,
|
||||
null,
|
||||
null,
|
||||
null,
|
||||
&io_status,
|
||||
&file_information_buf,
|
||||
file_information_buf.len,
|
||||
.FileDirectoryInformation,
|
||||
windows.FALSE, // single result
|
||||
&app_name_unicode_string,
|
||||
windows.FALSE, // restart iteration
|
||||
);
|
||||
|
||||
switch (rc) {
|
||||
.SUCCESS => {},
|
||||
.NO_SUCH_FILE => return error.FileNotFound,
|
||||
.NO_MORE_FILES => break,
|
||||
.ACCESS_DENIED => return error.AccessDenied,
|
||||
else => return windows.unexpectedStatus(rc),
|
||||
}
|
||||
|
||||
std.debug.assert(io_status.Information != 0);
|
||||
|
||||
var it = windows.FileInformationIterator(windows.FILE_DIRECTORY_INFORMATION){ .buf = &file_information_buf };
|
||||
while (it.next()) |info| {
|
||||
if (info.FileAttributes & windows.FILE_ATTRIBUTE_DIRECTORY != 0) continue;
|
||||
const filename = @as([*]u16, @ptrCast(&info.FileName))[0 .. info.FileNameLength / 2];
|
||||
if (filename.len == app_name_len) {
|
||||
unappended_exists = true;
|
||||
} else if (createProcessSupportsExtension(filename[app_name_len..])) |pathext_ext| {
|
||||
pathext_seen[@intFromEnum(pathext_ext)] = true;
|
||||
any_pathext_seen = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
const unappended_err = unappended: {
|
||||
if (unappended_exists) {
|
||||
if (dir_path_len != 0) switch (dir_buf.items[dir_buf.items.len - 1]) {
|
||||
'/', '\\' => {},
|
||||
else => try dir_buf.append(allocator, fs.path.sep),
|
||||
};
|
||||
try dir_buf.appendSlice(allocator, app_buf.items[0..app_name_len]);
|
||||
try dir_buf.append(allocator, 0);
|
||||
const full_app_name = dir_buf.items[0 .. dir_buf.items.len - 1 :0];
|
||||
|
||||
const is_bat_or_cmd = bat_or_cmd: {
|
||||
const app_name = app_buf.items[0..app_name_len];
|
||||
const ext_start = std.mem.lastIndexOfScalar(u16, app_name, '.') orelse break :bat_or_cmd false;
|
||||
const ext = app_name[ext_start..];
|
||||
const ext_enum = createProcessSupportsExtension(ext) orelse break :bat_or_cmd false;
|
||||
switch (ext_enum) {
|
||||
.cmd, .bat => break :bat_or_cmd true,
|
||||
else => break :bat_or_cmd false,
|
||||
}
|
||||
};
|
||||
const cmd_line_w = if (is_bat_or_cmd)
|
||||
try cmd_line_cache.scriptCommandLine(full_app_name)
|
||||
else
|
||||
try cmd_line_cache.commandLine();
|
||||
const app_name_w = if (is_bat_or_cmd)
|
||||
try cmd_line_cache.cmdExePath()
|
||||
else
|
||||
full_app_name;
|
||||
|
||||
if (createProcess(app_name_w.ptr, cmd_line_w.ptr, envp_ptr, cwd_ptr, lpStartupInfo, lpProcessInformation)) |_| {
|
||||
return;
|
||||
} else |err| switch (err) {
|
||||
error.FileNotFound,
|
||||
error.AccessDenied,
|
||||
=> break :unappended err,
|
||||
error.InvalidExe => {
|
||||
const app_name = app_buf.items[0..app_name_len];
|
||||
const ext_start = std.mem.lastIndexOfScalar(u16, app_name, '.') orelse break :unappended err;
|
||||
const ext = app_name[ext_start..];
|
||||
if (windows.eqlIgnoreCaseWTF16(ext, unicode.utf8ToUtf16LeStringLiteral(".EXE"))) {
|
||||
return error.UnrecoverableInvalidExe;
|
||||
}
|
||||
break :unappended err;
|
||||
},
|
||||
else => return err,
|
||||
}
|
||||
}
|
||||
break :unappended error.FileNotFound;
|
||||
};
|
||||
|
||||
if (!any_pathext_seen) return unappended_err;
|
||||
|
||||
var ext_it = mem.tokenizeScalar(u16, pathext, ';');
|
||||
while (ext_it.next()) |ext| {
|
||||
const ext_enum = createProcessSupportsExtension(ext) orelse continue;
|
||||
if (!pathext_seen[@intFromEnum(ext_enum)]) continue;
|
||||
|
||||
dir_buf.shrinkRetainingCapacity(dir_path_len);
|
||||
if (dir_path_len != 0) switch (dir_buf.items[dir_buf.items.len - 1]) {
|
||||
'/', '\\' => {},
|
||||
else => try dir_buf.append(allocator, fs.path.sep),
|
||||
};
|
||||
try dir_buf.appendSlice(allocator, app_buf.items[0..app_name_len]);
|
||||
try dir_buf.appendSlice(allocator, ext);
|
||||
try dir_buf.append(allocator, 0);
|
||||
const full_app_name = dir_buf.items[0 .. dir_buf.items.len - 1 :0];
|
||||
|
||||
const is_bat_or_cmd = switch (ext_enum) {
|
||||
.cmd, .bat => true,
|
||||
else => false,
|
||||
};
|
||||
const cmd_line_w = if (is_bat_or_cmd)
|
||||
try cmd_line_cache.scriptCommandLine(full_app_name)
|
||||
else
|
||||
try cmd_line_cache.commandLine();
|
||||
const app_name_w = if (is_bat_or_cmd)
|
||||
try cmd_line_cache.cmdExePath()
|
||||
else
|
||||
full_app_name;
|
||||
|
||||
if (createProcess(app_name_w.ptr, cmd_line_w.ptr, envp_ptr, cwd_ptr, lpStartupInfo, lpProcessInformation)) |_| {
|
||||
return;
|
||||
} else |err| switch (err) {
|
||||
error.FileNotFound => continue,
|
||||
error.AccessDenied => continue,
|
||||
error.InvalidExe => {
|
||||
if (windows.eqlIgnoreCaseWTF16(ext, unicode.utf8ToUtf16LeStringLiteral(".EXE"))) {
|
||||
return error.UnrecoverableInvalidExe;
|
||||
}
|
||||
continue;
|
||||
},
|
||||
else => return err,
|
||||
}
|
||||
}
|
||||
|
||||
return unappended_err;
|
||||
}
|
||||
|
||||
fn argvToCommandLine(
|
||||
allocator: mem.Allocator,
|
||||
argv: []const []const u8,
|
||||
) ![:0]u16 {
|
||||
var buf = std.ArrayList(u8).init(allocator);
|
||||
defer buf.deinit();
|
||||
|
||||
if (argv.len != 0) {
|
||||
const arg0 = argv[0];
|
||||
|
||||
var needs_quotes = arg0.len == 0;
|
||||
for (arg0) |c| {
|
||||
if (c <= ' ') {
|
||||
needs_quotes = true;
|
||||
} else if (c == '"') {
|
||||
return error.InvalidArg0;
|
||||
}
|
||||
}
|
||||
if (needs_quotes) {
|
||||
try buf.append('"');
|
||||
try buf.appendSlice(arg0);
|
||||
try buf.append('"');
|
||||
} else {
|
||||
try buf.appendSlice(arg0);
|
||||
}
|
||||
|
||||
for (argv[1..]) |arg| {
|
||||
try buf.append(' ');
|
||||
|
||||
needs_quotes = for (arg) |c| {
|
||||
if (c <= ' ' or c == '"') {
|
||||
break true;
|
||||
}
|
||||
} else arg.len == 0;
|
||||
if (!needs_quotes) {
|
||||
try buf.appendSlice(arg);
|
||||
continue;
|
||||
}
|
||||
|
||||
try buf.append('"');
|
||||
var backslash_count: usize = 0;
|
||||
for (arg) |byte| {
|
||||
switch (byte) {
|
||||
'\\' => {
|
||||
backslash_count += 1;
|
||||
},
|
||||
'"' => {
|
||||
try buf.appendNTimes('\\', backslash_count * 2 + 1);
|
||||
try buf.append('"');
|
||||
backslash_count = 0;
|
||||
},
|
||||
else => {
|
||||
try buf.appendNTimes('\\', backslash_count);
|
||||
try buf.append(byte);
|
||||
backslash_count = 0;
|
||||
},
|
||||
}
|
||||
}
|
||||
try buf.appendNTimes('\\', backslash_count * 2);
|
||||
try buf.append('"');
|
||||
}
|
||||
}
|
||||
|
||||
return try unicode.wtf8ToWtf16LeAllocZ(allocator, buf.items);
|
||||
}
|
||||
|
||||
fn argvToScriptCommandLine(
|
||||
allocator: mem.Allocator,
|
||||
script_path: []const u16,
|
||||
script_args: []const []const u8,
|
||||
) ![:0]u16 {
|
||||
var buf = try std.ArrayList(u8).initCapacity(allocator, 64);
|
||||
defer buf.deinit();
|
||||
|
||||
buf.appendSliceAssumeCapacity("cmd.exe /d /e:ON /v:OFF /c \"");
|
||||
|
||||
buf.appendAssumeCapacity('"');
|
||||
if (mem.indexOfAny(u16, script_path, &[_]u16{ mem.nativeToLittle(u16, '\\'), mem.nativeToLittle(u16, '/') }) == null) {
|
||||
try buf.appendSlice(".\\");
|
||||
}
|
||||
try unicode.wtf16LeToWtf8ArrayList(&buf, script_path);
|
||||
buf.appendAssumeCapacity('"');
|
||||
|
||||
for (script_args) |arg| {
|
||||
if (std.mem.indexOfAny(u8, arg, "\x00\r\n") != null) {
|
||||
return error.InvalidBatchScriptArg;
|
||||
}
|
||||
|
||||
try buf.append(' ');
|
||||
|
||||
var needs_quotes = arg.len == 0 or arg[arg.len - 1] == '\\';
|
||||
if (!needs_quotes) {
|
||||
for (arg) |c| {
|
||||
switch (c) {
|
||||
'A'...'Z', 'a'...'z', '0'...'9', '#', '$', '*', '+', '-', '.', '/', ':', '?', '@', '\\', '_' => {},
|
||||
else => {
|
||||
needs_quotes = true;
|
||||
break;
|
||||
},
|
||||
}
|
||||
}
|
||||
}
|
||||
if (needs_quotes) {
|
||||
try buf.append('"');
|
||||
}
|
||||
var backslashes: usize = 0;
|
||||
for (arg) |c| {
|
||||
switch (c) {
|
||||
'\\' => {
|
||||
backslashes += 1;
|
||||
},
|
||||
'"' => {
|
||||
try buf.appendNTimes('\\', backslashes);
|
||||
try buf.append('"');
|
||||
backslashes = 0;
|
||||
},
|
||||
'%' => {
|
||||
try buf.appendSlice("%%cd:~,");
|
||||
backslashes = 0;
|
||||
},
|
||||
else => {
|
||||
backslashes = 0;
|
||||
},
|
||||
}
|
||||
try buf.append(c);
|
||||
}
|
||||
if (needs_quotes) {
|
||||
try buf.appendNTimes('\\', backslashes);
|
||||
try buf.append('"');
|
||||
}
|
||||
}
|
||||
|
||||
try buf.append('"');
|
||||
|
||||
return try unicode.wtf8ToWtf16LeAllocZ(allocator, buf.items);
|
||||
}
|
||||
|
||||
fn wcdExePath(allocator: mem.Allocator) error{ OutOfMemory, Unexpected }![:0]u16 {
|
||||
var buf = try std.ArrayListUnmanaged(u16).initCapacity(allocator, 128);
|
||||
errdefer buf.deinit(allocator);
|
||||
while (true) {
|
||||
const unused_slice = buf.unusedCapacitySlice();
|
||||
const len = windows.kernel32.GetSystemDirectoryW(@ptrCast(unused_slice), @intCast(unused_slice.len));
|
||||
if (len == 0) {
|
||||
switch (windows.kernel32.GetLastError()) {
|
||||
else => |err| return windows.unexpectedError(err),
|
||||
}
|
||||
}
|
||||
if (len > unused_slice.len) {
|
||||
try buf.ensureUnusedCapacity(allocator, len);
|
||||
} else {
|
||||
buf.items.len = len;
|
||||
break;
|
||||
}
|
||||
}
|
||||
switch (buf.items[buf.items.len - 1]) {
|
||||
'/', '\\' => {},
|
||||
else => try buf.append(allocator, fs.path.sep),
|
||||
}
|
||||
try buf.appendSlice(allocator, unicode.utf8ToUtf16LeStringLiteral("cmd.exe"));
|
||||
return try buf.toOwnedSliceSentinel(allocator, 0);
|
||||
}
|
||||
|
||||
const CreateProcessSupportedExtension = enum {
|
||||
bat,
|
||||
cmd,
|
||||
com,
|
||||
exe,
|
||||
};
|
||||
|
||||
fn createProcessSupportsExtension(ext: []const u16) ?CreateProcessSupportedExtension {
|
||||
if (ext.len != 4) return null;
|
||||
const State = enum {
|
||||
start,
|
||||
dot,
|
||||
b,
|
||||
ba,
|
||||
c,
|
||||
cm,
|
||||
co,
|
||||
e,
|
||||
ex,
|
||||
};
|
||||
var state: State = .start;
|
||||
for (ext) |c| switch (state) {
|
||||
.start => switch (c) {
|
||||
'.' => state = .dot,
|
||||
else => return null,
|
||||
},
|
||||
.dot => switch (c) {
|
||||
'b', 'B' => state = .b,
|
||||
'c', 'C' => state = .c,
|
||||
'e', 'E' => state = .e,
|
||||
else => return null,
|
||||
},
|
||||
.b => switch (c) {
|
||||
'a', 'A' => state = .ba,
|
||||
else => return null,
|
||||
},
|
||||
.c => switch (c) {
|
||||
'm', 'M' => state = .cm,
|
||||
'o', 'O' => state = .co,
|
||||
else => return null,
|
||||
},
|
||||
.e => switch (c) {
|
||||
'x', 'X' => state = .ex,
|
||||
else => return null,
|
||||
},
|
||||
.ba => switch (c) {
|
||||
't', 'T' => return .bat,
|
||||
else => return null,
|
||||
},
|
||||
.cm => switch (c) {
|
||||
'd', 'D' => return .cmd,
|
||||
else => return null,
|
||||
},
|
||||
.co => switch (c) {
|
||||
'm', 'M' => return .com,
|
||||
else => return null,
|
||||
},
|
||||
.ex => switch (c) {
|
||||
'e', 'E' => return .exe,
|
||||
else => return null,
|
||||
},
|
||||
};
|
||||
return null;
|
||||
}
|
||||
|
||||
pub const CREATE_NO_WINDOW = 0x08000000;
|
||||
|
||||
fn createProcess(
|
||||
app_name: [*:0]u16,
|
||||
cmd_line: [*:0]u16,
|
||||
envp_ptr: ?[*]u16,
|
||||
cwd_ptr: ?[*:0]u16,
|
||||
lpStartupInfo: *windows.STARTUPINFOW,
|
||||
lpProcessInformation: *windows.PROCESS_INFORMATION,
|
||||
) !void {
|
||||
return windows.CreateProcessW(
|
||||
app_name,
|
||||
cmd_line,
|
||||
null,
|
||||
null,
|
||||
windows.TRUE,
|
||||
windows.CREATE_UNICODE_ENVIRONMENT | CREATE_NO_WINDOW,
|
||||
@as(?*anyopaque, @ptrCast(envp_ptr)),
|
||||
cwd_ptr,
|
||||
lpStartupInfo,
|
||||
lpProcessInformation,
|
||||
);
|
||||
}
|
||||
|
||||
pub fn wait(self: *@This()) !Term {
|
||||
if (self.term) |term_| {
|
||||
self.cleanupStreams();
|
||||
return term_;
|
||||
}
|
||||
|
||||
defer self.id = undefined;
|
||||
try self.waitUnwrapped();
|
||||
return self.term.?;
|
||||
}
|
||||
|
||||
fn waitUnwrapped(self: *@This()) !void {
|
||||
const result = windows.WaitForSingleObjectEx(self.id, windows.INFINITE, false);
|
||||
|
||||
self.term = @as(SpawnError!Term, x: {
|
||||
var exit_code: windows.DWORD = undefined;
|
||||
if (windows.kernel32.GetExitCodeProcess(self.id, &exit_code) == 0) {
|
||||
break :x Term{ .Unknown = 0 };
|
||||
} else {
|
||||
break :x Term{ .Exited = @as(u8, @truncate(exit_code)) };
|
||||
}
|
||||
});
|
||||
|
||||
posix.close(self.id);
|
||||
posix.close(self.thread_handle);
|
||||
self.cleanupStreams();
|
||||
return result;
|
||||
}
|
||||
|
||||
fn cleanupStreams(self: *@This()) void {
|
||||
if (self.stdin) |*stdin| {
|
||||
stdin.close();
|
||||
self.stdin = null;
|
||||
}
|
||||
if (self.stdout) |*stdout| {
|
||||
stdout.close();
|
||||
self.stdout = null;
|
||||
}
|
||||
if (self.stderr) |*stderr| {
|
||||
stderr.close();
|
||||
self.stderr = null;
|
||||
}
|
||||
}
|
||||
|
||||
pub fn kill(self: *@This()) !Term {
|
||||
if (self.term) |term_| {
|
||||
self.cleanupStreams();
|
||||
return term_;
|
||||
}
|
||||
|
||||
windows.TerminateProcess(self.id, 1) catch |err| switch (err) {
|
||||
error.PermissionDenied => {
|
||||
windows.WaitForSingleObjectEx(self.id, 0, false) catch return err;
|
||||
return error.AlreadyTerminated;
|
||||
},
|
||||
else => return err,
|
||||
};
|
||||
try self.waitUnwrapped();
|
||||
return self.term.?;
|
||||
}
|
||||
};
|
||||
|
|
Loading…
Add table
Reference in a new issue