|
| 1 | +//! Job management (mostly for windows) |
| 2 | +//! |
| 3 | +//! Most of the time when you're running cargo you expect Ctrl-C to actually |
| 4 | +//! terminate the entire tree of processes in play, not just the one at the top |
| 5 | +//! (cago). This currently works "by default" on Unix platforms because Ctrl-C |
| 6 | +//! actually sends a signal to the *process group* rather than the parent |
| 7 | +//! process, so everything will get torn down. On Windows, however, this does |
| 8 | +//! not happen and Ctrl-C just kills cargo. |
| 9 | +//! |
| 10 | +//! To achieve the same semantics on Windows we use Job Objects to ensure that |
| 11 | +//! all processes die at the same time. Job objects have a mode of operation |
| 12 | +//! where when all handles to the object are closed it causes all child |
| 13 | +//! processes associated with the object to be terminated immediately. |
| 14 | +//! Conveniently whenever a process in the job object spawns a new process the |
| 15 | +//! child will be associated with the job object as well. This means if we add |
| 16 | +//! ourselves to the job object we create then everything will get torn down! |
| 17 | +
|
| 18 | +pub fn setup() { |
| 19 | + unsafe { imp::setup() } |
| 20 | +} |
| 21 | + |
| 22 | +#[cfg(unix)] |
| 23 | +mod imp { |
| 24 | + use std::env; |
| 25 | + use libc; |
| 26 | + |
| 27 | + pub unsafe fn setup() { |
| 28 | + // There's a test case for the behavior of |
| 29 | + // when-cargo-is-killed-subprocesses-are-also-killed, but that requires |
| 30 | + // one cargo spawned to become its own session leader, so we do that |
| 31 | + // here. |
| 32 | + if env::var("__CARGO_TEST_SETSID_PLEASE_DONT_USE_ELSEWHERE").is_ok() { |
| 33 | + libc::setsid(); |
| 34 | + } |
| 35 | + } |
| 36 | +} |
| 37 | + |
| 38 | +#[cfg(windows)] |
| 39 | +mod imp { |
| 40 | + extern crate kernel32; |
| 41 | + extern crate winapi; |
| 42 | + |
| 43 | + use std::mem; |
| 44 | + |
| 45 | + pub unsafe fn setup() { |
| 46 | + // Creates a new job object for us to use and then adds ourselves to it. |
| 47 | + // Note that all errors are basically ignored in this function, |
| 48 | + // intentionally. Job objects are "relatively new" in Windows, |
| 49 | + // particularly the ability to support nested job objects. Older |
| 50 | + // Windows installs don't support this ability. We probably don't want |
| 51 | + // to force Cargo to abort in this situation or force others to *not* |
| 52 | + // use job objects, so we instead just ignore errors and assume that |
| 53 | + // we're otherwise part of someone else's job object in this case. |
| 54 | + |
| 55 | + let job = kernel32::CreateJobObjectW(0 as *mut _, 0 as *const _); |
| 56 | + if job.is_null() { |
| 57 | + return |
| 58 | + } |
| 59 | + |
| 60 | + // Indicate that when all handles to the job object are gone that all |
| 61 | + // process in the object should be killed. Note that this includes our |
| 62 | + // entire process tree by default because we've added ourselves and and |
| 63 | + // our children will reside in the job once we spawn a process. |
| 64 | + let mut info: winapi::JOBOBJECT_EXTENDED_LIMIT_INFORMATION; |
| 65 | + info = mem::zeroed(); |
| 66 | + info.BasicLimitInformation.LimitFlags = |
| 67 | + winapi::JOB_OBJECT_LIMIT_KILL_ON_JOB_CLOSE; |
| 68 | + let r = kernel32::SetInformationJobObject(job, |
| 69 | + winapi::JobObjectExtendedLimitInformation, |
| 70 | + &mut info as *mut _ as winapi::LPVOID, |
| 71 | + mem::size_of_val(&info) as winapi::DWORD); |
| 72 | + if r == 0 { |
| 73 | + kernel32::CloseHandle(job); |
| 74 | + return |
| 75 | + } |
| 76 | + |
| 77 | + // Assign our process to this job object, meaning that our children will |
| 78 | + // now live or die based on our existence. |
| 79 | + let me = kernel32::GetCurrentProcess(); |
| 80 | + let r = kernel32::AssignProcessToJobObject(job, me); |
| 81 | + if r == 0 { |
| 82 | + kernel32::CloseHandle(job); |
| 83 | + return |
| 84 | + } |
| 85 | + |
| 86 | + // Intentionally leak the `job` handle here. We've got the only |
| 87 | + // reference to this job, so once it's gone we and all our children will |
| 88 | + // be killed. This typically won't happen unless Cargo itself is |
| 89 | + // ctrl-c'd. |
| 90 | + } |
| 91 | +} |
0 commit comments