Skip to content

Commit 1f49df9

Browse files
authored
Pause child processes on spawn. (#1042)
This PR pauses child processes we spawn before resuming them, which makes it easier to set a breakpoint and attach a debugger to those processes. This functionality is available on macOS and Windows; as far as I know, neither Linux nor POSIX-in-general has API for this. ### Checklist: - [x] Code and documentation should follow the style of the [Style Guide](https://github.com/apple/swift-testing/blob/main/Documentation/StyleGuide.md). - [x] If public symbols are renamed or modified, DocC references should be updated.
1 parent 4e7a085 commit 1f49df9

File tree

1 file changed

+22
-2
lines changed

1 file changed

+22
-2
lines changed

Sources/Testing/ExitTests/SpawnProcess.swift

+22-2
Original file line numberDiff line numberDiff line change
@@ -176,6 +176,11 @@ func spawnExecutable(
176176
#warning("Platform-specific implementation missing: cannot close unused file descriptors")
177177
#endif
178178

179+
#if SWT_TARGET_OS_APPLE && DEBUG
180+
// Start the process suspended so we can attach a debugger if needed.
181+
flags |= CShort(POSIX_SPAWN_START_SUSPENDED)
182+
#endif
183+
179184
// Set flags; make sure to keep this call below any code that might modify
180185
// the flags mask!
181186
_ = posix_spawnattr_setflags(attrs, flags)
@@ -202,6 +207,10 @@ func spawnExecutable(
202207
guard 0 == processSpawned else {
203208
throw CError(rawValue: processSpawned)
204209
}
210+
#if SWT_TARGET_OS_APPLE && DEBUG
211+
// Resume the process.
212+
_ = kill(pid, SIGCONT)
213+
#endif
205214
return pid
206215
}
207216
}
@@ -254,6 +263,12 @@ func spawnExecutable(
254263
let commandLine = _escapeCommandLine(CollectionOfOne(executablePath) + arguments)
255264
let environ = environment.map { "\($0.key)=\($0.value)" }.joined(separator: "\0") + "\0\0"
256265

266+
var flags = DWORD(CREATE_NO_WINDOW | CREATE_UNICODE_ENVIRONMENT | EXTENDED_STARTUPINFO_PRESENT)
267+
#if DEBUG
268+
// Start the process suspended so we can attach a debugger if needed.
269+
flags |= DWORD(CREATE_SUSPENDED)
270+
#endif
271+
257272
return try commandLine.withCString(encodedAs: UTF16.self) { commandLine in
258273
try environ.withCString(encodedAs: UTF16.self) { environ in
259274
var processInfo = PROCESS_INFORMATION()
@@ -264,16 +279,21 @@ func spawnExecutable(
264279
nil,
265280
nil,
266281
true, // bInheritHandles
267-
DWORD(CREATE_NO_WINDOW | CREATE_UNICODE_ENVIRONMENT | EXTENDED_STARTUPINFO_PRESENT),
282+
flags,
268283
.init(mutating: environ),
269284
nil,
270285
startupInfo.pointer(to: \.StartupInfo)!,
271286
&processInfo
272287
) else {
273288
throw Win32Error(rawValue: GetLastError())
274289
}
275-
_ = CloseHandle(processInfo.hThread)
276290

291+
#if DEBUG
292+
// Resume the process.
293+
_ = ResumeThread(processInfo.hThread!)
294+
#endif
295+
296+
_ = CloseHandle(processInfo.hThread)
277297
return processInfo.hProcess!
278298
}
279299
}

0 commit comments

Comments
 (0)