]>
Commit | Line | Data |
---|---|---|
c30ab7b3 SL |
1 | //! Implementation of the install aspects of the compiler. |
2 | //! | |
3 | //! This module is responsible for installing the standard library, | |
4 | //! compiler, and documentation. | |
5 | ||
32a655c1 | 6 | use std::env; |
c30ab7b3 | 7 | use std::fs; |
dc9dc135 | 8 | use std::path::{Component, Path, PathBuf}; |
c30ab7b3 SL |
9 | use std::process::Command; |
10 | ||
48663c56 XL |
11 | use build_helper::t; |
12 | ||
0731742a | 13 | use crate::dist::{self, pkgname, sanitize_sh, tmpdir}; |
dc9dc135 | 14 | use crate::Compiler; |
7cac9316 | 15 | |
0731742a XL |
16 | use crate::builder::{Builder, RunConfig, ShouldRun, Step}; |
17 | use crate::cache::Interned; | |
18 | use crate::config::Config; | |
7cac9316 | 19 | |
9fa01778 | 20 | pub fn install_docs(builder: &Builder<'_>, stage: u32, host: Interned<String>) { |
3b2f2976 XL |
21 | install_sh(builder, "docs", "rust-docs", stage, Some(host)); |
22 | } | |
cc61c64b | 23 | |
9fa01778 | 24 | pub fn install_std(builder: &Builder<'_>, stage: u32, target: Interned<String>) { |
abe05a73 | 25 | install_sh(builder, "std", "rust-std", stage, Some(target)); |
3b2f2976 | 26 | } |
c30ab7b3 | 27 | |
9fa01778 | 28 | pub fn install_cargo(builder: &Builder<'_>, stage: u32, host: Interned<String>) { |
3b2f2976 XL |
29 | install_sh(builder, "cargo", "cargo", stage, Some(host)); |
30 | } | |
7cac9316 | 31 | |
9fa01778 | 32 | pub fn install_rls(builder: &Builder<'_>, stage: u32, host: Interned<String>) { |
3b2f2976 XL |
33 | install_sh(builder, "rls", "rls", stage, Some(host)); |
34 | } | |
f035d41b XL |
35 | pub fn install_rust_analyzer(builder: &Builder<'_>, stage: u32, host: Interned<String>) { |
36 | install_sh(builder, "rust-analyzer", "rust-analyzer", stage, Some(host)); | |
37 | } | |
9fa01778 | 38 | pub fn install_clippy(builder: &Builder<'_>, stage: u32, host: Interned<String>) { |
8faf50e0 XL |
39 | install_sh(builder, "clippy", "clippy", stage, Some(host)); |
40 | } | |
9fa01778 | 41 | pub fn install_miri(builder: &Builder<'_>, stage: u32, host: Interned<String>) { |
0731742a XL |
42 | install_sh(builder, "miri", "miri", stage, Some(host)); |
43 | } | |
7cac9316 | 44 | |
9fa01778 | 45 | pub fn install_rustfmt(builder: &Builder<'_>, stage: u32, host: Interned<String>) { |
abe05a73 XL |
46 | install_sh(builder, "rustfmt", "rustfmt", stage, Some(host)); |
47 | } | |
48 | ||
9fa01778 | 49 | pub fn install_analysis(builder: &Builder<'_>, stage: u32, host: Interned<String>) { |
3b2f2976 XL |
50 | install_sh(builder, "analysis", "rust-analysis", stage, Some(host)); |
51 | } | |
7cac9316 | 52 | |
9fa01778 | 53 | pub fn install_src(builder: &Builder<'_>, stage: u32) { |
3b2f2976 XL |
54 | install_sh(builder, "src", "rust-src", stage, None); |
55 | } | |
9fa01778 | 56 | pub fn install_rustc(builder: &Builder<'_>, stage: u32, host: Interned<String>) { |
3b2f2976 XL |
57 | install_sh(builder, "rustc", "rustc", stage, Some(host)); |
58 | } | |
7cac9316 | 59 | |
3b2f2976 | 60 | fn install_sh( |
9fa01778 | 61 | builder: &Builder<'_>, |
3b2f2976 XL |
62 | package: &str, |
63 | name: &str, | |
64 | stage: u32, | |
dc9dc135 | 65 | host: Option<Interned<String>>, |
3b2f2976 | 66 | ) { |
83c7162d | 67 | builder.info(&format!("Install {} stage{} ({:?})", package, stage, host)); |
3b2f2976 XL |
68 | |
69 | let prefix_default = PathBuf::from("/usr/local"); | |
70 | let sysconfdir_default = PathBuf::from("/etc"); | |
2c00a5a8 XL |
71 | let datadir_default = PathBuf::from("share"); |
72 | let docdir_default = datadir_default.join("doc/rust"); | |
3b2f2976 | 73 | let libdir_default = PathBuf::from("lib"); |
2c00a5a8 | 74 | let mandir_default = datadir_default.join("man"); |
83c7162d | 75 | let prefix = builder.config.prefix.as_ref().map_or(prefix_default, |p| { |
f035d41b XL |
76 | fs::create_dir_all(p) |
77 | .unwrap_or_else(|err| panic!("could not create {}: {}", p.display(), err)); | |
78 | fs::canonicalize(p) | |
79 | .unwrap_or_else(|err| panic!("could not canonicalize {}: {}", p.display(), err)) | |
83c7162d XL |
80 | }); |
81 | let sysconfdir = builder.config.sysconfdir.as_ref().unwrap_or(&sysconfdir_default); | |
82 | let datadir = builder.config.datadir.as_ref().unwrap_or(&datadir_default); | |
83 | let docdir = builder.config.docdir.as_ref().unwrap_or(&docdir_default); | |
e1599b0c | 84 | let bindir = &builder.config.bindir; |
83c7162d XL |
85 | let libdir = builder.config.libdir.as_ref().unwrap_or(&libdir_default); |
86 | let mandir = builder.config.mandir.as_ref().unwrap_or(&mandir_default); | |
3b2f2976 XL |
87 | |
88 | let sysconfdir = prefix.join(sysconfdir); | |
2c00a5a8 | 89 | let datadir = prefix.join(datadir); |
3b2f2976 XL |
90 | let docdir = prefix.join(docdir); |
91 | let bindir = prefix.join(bindir); | |
92 | let libdir = prefix.join(libdir); | |
93 | let mandir = prefix.join(mandir); | |
94 | ||
95 | let destdir = env::var_os("DESTDIR").map(PathBuf::from); | |
96 | ||
97 | let prefix = add_destdir(&prefix, &destdir); | |
98 | let sysconfdir = add_destdir(&sysconfdir, &destdir); | |
2c00a5a8 | 99 | let datadir = add_destdir(&datadir, &destdir); |
3b2f2976 XL |
100 | let docdir = add_destdir(&docdir, &destdir); |
101 | let bindir = add_destdir(&bindir, &destdir); | |
102 | let libdir = add_destdir(&libdir, &destdir); | |
103 | let mandir = add_destdir(&mandir, &destdir); | |
104 | ||
83c7162d | 105 | let empty_dir = builder.out.join("tmp/empty_dir"); |
3b2f2976 XL |
106 | |
107 | t!(fs::create_dir_all(&empty_dir)); | |
108 | let package_name = if let Some(host) = host { | |
83c7162d | 109 | format!("{}-{}", pkgname(builder, name), host) |
3b2f2976 | 110 | } else { |
83c7162d | 111 | pkgname(builder, name) |
3b2f2976 | 112 | }; |
7cac9316 | 113 | |
3b2f2976 XL |
114 | let mut cmd = Command::new("sh"); |
115 | cmd.current_dir(&empty_dir) | |
83c7162d | 116 | .arg(sanitize_sh(&tmpdir(builder).join(&package_name).join("install.sh"))) |
3b2f2976 XL |
117 | .arg(format!("--prefix={}", sanitize_sh(&prefix))) |
118 | .arg(format!("--sysconfdir={}", sanitize_sh(&sysconfdir))) | |
2c00a5a8 | 119 | .arg(format!("--datadir={}", sanitize_sh(&datadir))) |
3b2f2976 XL |
120 | .arg(format!("--docdir={}", sanitize_sh(&docdir))) |
121 | .arg(format!("--bindir={}", sanitize_sh(&bindir))) | |
122 | .arg(format!("--libdir={}", sanitize_sh(&libdir))) | |
123 | .arg(format!("--mandir={}", sanitize_sh(&mandir))) | |
124 | .arg("--disable-ldconfig"); | |
83c7162d | 125 | builder.run(&mut cmd); |
3b2f2976 | 126 | t!(fs::remove_dir_all(&empty_dir)); |
c30ab7b3 | 127 | } |
32a655c1 SL |
128 | |
129 | fn add_destdir(path: &Path, destdir: &Option<PathBuf>) -> PathBuf { | |
130 | let mut ret = match *destdir { | |
131 | Some(ref dest) => dest.clone(), | |
132 | None => return path.to_path_buf(), | |
133 | }; | |
134 | for part in path.components() { | |
74b04a01 XL |
135 | if let Component::Normal(s) = part { |
136 | ret.push(s) | |
32a655c1 SL |
137 | } |
138 | } | |
041b39d2 | 139 | ret |
32a655c1 | 140 | } |
3b2f2976 XL |
141 | |
142 | macro_rules! install { | |
143 | (($sel:ident, $builder:ident, $_config:ident), | |
144 | $($name:ident, | |
145 | $path:expr, | |
146 | $default_cond:expr, | |
147 | only_hosts: $only_hosts:expr, | |
148 | $run_item:block $(, $c:ident)*;)+) => { | |
149 | $( | |
150 | #[derive(Debug, Copy, Clone, Hash, PartialEq, Eq)] | |
151 | pub struct $name { | |
dc9dc135 | 152 | pub compiler: Compiler, |
3b2f2976 | 153 | pub target: Interned<String>, |
3b2f2976 XL |
154 | } |
155 | ||
2c00a5a8 XL |
156 | impl $name { |
157 | #[allow(dead_code)] | |
158 | fn should_build(config: &Config) -> bool { | |
159 | config.extended && config.tools.as_ref() | |
160 | .map_or(true, |t| t.contains($path)) | |
161 | } | |
2c00a5a8 XL |
162 | } |
163 | ||
3b2f2976 XL |
164 | impl Step for $name { |
165 | type Output = (); | |
166 | const DEFAULT: bool = true; | |
3b2f2976 XL |
167 | const ONLY_HOSTS: bool = $only_hosts; |
168 | $(const $c: bool = true;)* | |
169 | ||
9fa01778 | 170 | fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> { |
3b2f2976 XL |
171 | let $_config = &run.builder.config; |
172 | run.path($path).default_condition($default_cond) | |
173 | } | |
174 | ||
9fa01778 | 175 | fn make_run(run: RunConfig<'_>) { |
3b2f2976 | 176 | run.builder.ensure($name { |
dc9dc135 | 177 | compiler: run.builder.compiler(run.builder.top_stage, run.builder.config.build), |
3b2f2976 | 178 | target: run.target, |
3b2f2976 XL |
179 | }); |
180 | } | |
181 | ||
9fa01778 | 182 | fn run($sel, $builder: &Builder<'_>) { |
3b2f2976 XL |
183 | $run_item |
184 | } | |
185 | })+ | |
186 | } | |
187 | } | |
188 | ||
189 | install!((self, builder, _config), | |
190 | Docs, "src/doc", _config.docs, only_hosts: false, { | |
dc9dc135 XL |
191 | builder.ensure(dist::Docs { host: self.target }); |
192 | install_docs(builder, self.compiler.stage, self.target); | |
3b2f2976 XL |
193 | }; |
194 | Std, "src/libstd", true, only_hosts: true, { | |
83c7162d | 195 | for target in &builder.targets { |
abe05a73 | 196 | builder.ensure(dist::Std { |
dc9dc135 | 197 | compiler: self.compiler, |
abe05a73 XL |
198 | target: *target |
199 | }); | |
dc9dc135 | 200 | install_std(builder, self.compiler.stage, *target); |
abe05a73 | 201 | } |
3b2f2976 | 202 | }; |
2c00a5a8 | 203 | Cargo, "cargo", Self::should_build(_config), only_hosts: true, { |
dc9dc135 XL |
204 | builder.ensure(dist::Cargo { compiler: self.compiler, target: self.target }); |
205 | install_cargo(builder, self.compiler.stage, self.target); | |
3b2f2976 | 206 | }; |
2c00a5a8 | 207 | Rls, "rls", Self::should_build(_config), only_hosts: true, { |
f9652781 | 208 | if builder.ensure(dist::Rls { compiler: self.compiler, target: self.target }).is_some() { |
dc9dc135 | 209 | install_rls(builder, self.compiler.stage, self.target); |
abe05a73 | 210 | } else { |
dc9dc135 XL |
211 | builder.info( |
212 | &format!("skipping Install RLS stage{} ({})", self.compiler.stage, self.target), | |
213 | ); | |
abe05a73 XL |
214 | } |
215 | }; | |
f035d41b XL |
216 | RustAnalyzer, "rust-analyzer", Self::should_build(_config), only_hosts: true, { |
217 | builder.ensure(dist::RustAnalyzer { compiler: self.compiler, target: self.target }); | |
f9652781 | 218 | install_rust_analyzer(builder, self.compiler.stage, self.target); |
f035d41b | 219 | }; |
8faf50e0 | 220 | Clippy, "clippy", Self::should_build(_config), only_hosts: true, { |
f9f354fc | 221 | builder.ensure(dist::Clippy { compiler: self.compiler, target: self.target }); |
f9652781 | 222 | install_clippy(builder, self.compiler.stage, self.target); |
8faf50e0 | 223 | }; |
0731742a | 224 | Miri, "miri", Self::should_build(_config), only_hosts: true, { |
f9652781 | 225 | if builder.ensure(dist::Miri { compiler: self.compiler, target: self.target }).is_some() { |
dc9dc135 | 226 | install_miri(builder, self.compiler.stage, self.target); |
0731742a | 227 | } else { |
dc9dc135 XL |
228 | builder.info( |
229 | &format!("skipping Install miri stage{} ({})", self.compiler.stage, self.target), | |
230 | ); | |
0731742a XL |
231 | } |
232 | }; | |
2c00a5a8 | 233 | Rustfmt, "rustfmt", Self::should_build(_config), only_hosts: true, { |
dc9dc135 XL |
234 | if builder.ensure(dist::Rustfmt { |
235 | compiler: self.compiler, | |
236 | target: self.target | |
f9652781 | 237 | }).is_some() { |
dc9dc135 | 238 | install_rustfmt(builder, self.compiler.stage, self.target); |
abe05a73 | 239 | } else { |
83c7162d | 240 | builder.info( |
dc9dc135 XL |
241 | &format!("skipping Install Rustfmt stage{} ({})", self.compiler.stage, self.target), |
242 | ); | |
abe05a73 | 243 | } |
3b2f2976 | 244 | }; |
2c00a5a8 | 245 | Analysis, "analysis", Self::should_build(_config), only_hosts: false, { |
3b2f2976 | 246 | builder.ensure(dist::Analysis { |
dc9dc135 XL |
247 | // Find the actual compiler (handling the full bootstrap option) which |
248 | // produced the save-analysis data because that data isn't copied | |
249 | // through the sysroot uplifting. | |
250 | compiler: builder.compiler_for(builder.top_stage, builder.config.build, self.target), | |
3b2f2976 XL |
251 | target: self.target |
252 | }); | |
dc9dc135 | 253 | install_analysis(builder, self.compiler.stage, self.target); |
3b2f2976 | 254 | }; |
3b2f2976 XL |
255 | Rustc, "src/librustc", true, only_hosts: true, { |
256 | builder.ensure(dist::Rustc { | |
60c5eb7d | 257 | compiler: builder.compiler(builder.top_stage, self.target), |
3b2f2976 | 258 | }); |
dc9dc135 | 259 | install_rustc(builder, self.compiler.stage, self.target); |
3b2f2976 XL |
260 | }; |
261 | ); | |
0531ce1d XL |
262 | |
263 | #[derive(Debug, Copy, Clone, Hash, PartialEq, Eq)] | |
264 | pub struct Src { | |
265 | pub stage: u32, | |
266 | } | |
267 | ||
268 | impl Step for Src { | |
269 | type Output = (); | |
270 | const DEFAULT: bool = true; | |
271 | const ONLY_HOSTS: bool = true; | |
272 | ||
9fa01778 | 273 | fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> { |
0531ce1d | 274 | let config = &run.builder.config; |
dc9dc135 | 275 | let cond = config.extended && config.tools.as_ref().map_or(true, |t| t.contains("src")); |
0531ce1d XL |
276 | run.path("src").default_condition(cond) |
277 | } | |
278 | ||
9fa01778 | 279 | fn make_run(run: RunConfig<'_>) { |
dc9dc135 | 280 | run.builder.ensure(Src { stage: run.builder.top_stage }); |
0531ce1d XL |
281 | } |
282 | ||
9fa01778 | 283 | fn run(self, builder: &Builder<'_>) { |
0531ce1d XL |
284 | builder.ensure(dist::Src); |
285 | install_src(builder, self.stage); | |
286 | } | |
287 | } |