Skip to content

Commit 93b116a

Browse files
committed
Append Windows bin directory to PATH
1 parent 21d4b27 commit 93b116a

File tree

2 files changed

+99
-4
lines changed

2 files changed

+99
-4
lines changed

src/env_var.rs

+86-2
Original file line numberDiff line numberDiff line change
@@ -7,9 +7,10 @@ use crate::process::Process;
77

88
pub const RUST_RECURSION_COUNT_MAX: u32 = 20;
99

10-
pub(crate) fn prepend_path(
10+
pub(crate) fn insert_path(
1111
name: &str,
1212
prepend: Vec<PathBuf>,
13+
append: Option<PathBuf>,
1314
cmd: &mut Command,
1415
process: &Process,
1516
) {
@@ -21,6 +22,11 @@ pub(crate) fn prepend_path(
2122
tail.push_front(path);
2223
}
2324
}
25+
if let Some(path) = append {
26+
if !tail.contains(&path) {
27+
tail.push_back(path);
28+
}
29+
}
2430
tail
2531
} else {
2632
prepend.into()
@@ -77,7 +83,7 @@ mod tests {
7783
let path_z = PathBuf::from(z);
7884
path_entries.push(path_z);
7985

80-
prepend_path("PATH", path_entries, &mut cmd, &tp.process);
86+
insert_path("PATH", path_entries, None, &mut cmd, &tp.process);
8187
let envs: Vec<_> = cmd.get_envs().collect();
8288

8389
assert_eq!(
@@ -99,4 +105,82 @@ mod tests {
99105
),]
100106
);
101107
}
108+
109+
#[test]
110+
fn append_unique_path() {
111+
let mut vars = HashMap::new();
112+
vars.env(
113+
"PATH",
114+
env::join_paths(["/home/a/.cargo/bin", "/home/b/.cargo/bin"].iter()).unwrap(),
115+
);
116+
let tp = TestProcess::with_vars(vars);
117+
#[cfg(windows)]
118+
let _path_guard = RegistryGuard::new(&USER_PATH).unwrap();
119+
120+
#[track_caller]
121+
fn check(tp: &TestProcess, path_entries: Vec<PathBuf>, append: &str, expected: &[&str]) {
122+
let mut cmd = Command::new("test");
123+
124+
insert_path(
125+
"PATH",
126+
path_entries,
127+
Some(append.into()),
128+
&mut cmd,
129+
&tp.process,
130+
);
131+
let envs: Vec<_> = cmd.get_envs().collect();
132+
133+
assert_eq!(
134+
envs,
135+
&[(
136+
OsStr::new("PATH"),
137+
Some(env::join_paths(expected.iter()).unwrap().as_os_str())
138+
),]
139+
);
140+
}
141+
142+
check(
143+
&tp,
144+
Vec::new(),
145+
"/home/z/.cargo/bin",
146+
&[
147+
"/home/a/.cargo/bin",
148+
"/home/b/.cargo/bin",
149+
"/home/z/.cargo/bin",
150+
],
151+
);
152+
check(
153+
&tp,
154+
Vec::new(),
155+
"/home/a/.cargo/bin",
156+
&["/home/a/.cargo/bin", "/home/b/.cargo/bin"],
157+
);
158+
check(
159+
&tp,
160+
Vec::new(),
161+
"/home/b/.cargo/bin",
162+
&["/home/a/.cargo/bin", "/home/b/.cargo/bin"],
163+
);
164+
check(
165+
&tp,
166+
Vec::from(["/home/c/.cargo/bin".into()]),
167+
"/home/c/.cargo/bin",
168+
&[
169+
"/home/c/.cargo/bin",
170+
"/home/a/.cargo/bin",
171+
"/home/b/.cargo/bin",
172+
],
173+
);
174+
check(
175+
&tp,
176+
Vec::from(["/home/c/.cargo/bin".into()]),
177+
"/home/z/.cargo/bin",
178+
&[
179+
"/home/c/.cargo/bin",
180+
"/home/a/.cargo/bin",
181+
"/home/b/.cargo/bin",
182+
"/home/z/.cargo/bin",
183+
],
184+
);
185+
}
102186
}

src/toolchain.rs

+13-2
Original file line numberDiff line numberDiff line change
@@ -217,7 +217,7 @@ impl<'a> Toolchain<'a> {
217217
new_path.push(PathBuf::from("/usr/lib"));
218218
}
219219

220-
env_var::prepend_path(sysenv::LOADER_PATH, new_path, cmd, self.cfg.process);
220+
env_var::insert_path(sysenv::LOADER_PATH, new_path, None, cmd, self.cfg.process);
221221

222222
// Prepend CARGO_HOME/bin to the PATH variable so that we're sure to run
223223
// cargo/rustc via the proxy bins. There is no fallback case for if the
@@ -251,7 +251,18 @@ impl<'a> Toolchain<'a> {
251251
path_entries.push(self.path.join("bin"));
252252
}
253253

254-
env_var::prepend_path("PATH", path_entries, cmd, self.cfg.process);
254+
// On Windows, we append the "bin" directory to PATH.
255+
// Windows loads DLLs from PATH and the "bin" directory contains DLLs
256+
// that proc macros and other tools not in the sysroot use.
257+
// It's appended rather than prepended so that the exe files in "bin"
258+
// do not take precedence over anything else in PATH.
259+
let append = if cfg!(target_os = "windows") {
260+
Some(self.path.join("bin"))
261+
} else {
262+
None
263+
};
264+
265+
env_var::insert_path("PATH", path_entries, append, cmd, self.cfg.process);
255266
}
256267

257268
/// Infallible function that describes the version of rustc in an installed distribution

0 commit comments

Comments
 (0)