Skip to content

Commit fbd930b

Browse files
committed
rewrite cross-lang-lto to rmake
1 parent 4db3d12 commit fbd930b

File tree

5 files changed

+148
-60
lines changed

5 files changed

+148
-60
lines changed

src/tools/run-make-support/src/external_deps/llvm.rs

+36
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,12 @@ pub fn llvm_ar() -> LlvmAr {
3636
LlvmAr::new()
3737
}
3838

39+
/// Construct a new `llvm-bcanalyzer` invocation. This assumes that `llvm-bcanalyzer` is available
40+
/// at `$LLVM_BIN_DIR/llvm-bcanalyzer`.
41+
pub fn llvm_bcanalyzer() -> LlvmBcanalyzer {
42+
LlvmBcanalyzer::new()
43+
}
44+
3945
/// A `llvm-readobj` invocation builder.
4046
#[derive(Debug)]
4147
#[must_use]
@@ -71,11 +77,19 @@ pub struct LlvmAr {
7177
cmd: Command,
7278
}
7379

80+
/// A `llvm-bcanalyzer` invocation builder.
81+
#[derive(Debug)]
82+
#[must_use]
83+
pub struct LlvmBcanalyzer {
84+
cmd: Command,
85+
}
86+
7487
crate::macros::impl_common_helpers!(LlvmReadobj);
7588
crate::macros::impl_common_helpers!(LlvmProfdata);
7689
crate::macros::impl_common_helpers!(LlvmFilecheck);
7790
crate::macros::impl_common_helpers!(LlvmObjdump);
7891
crate::macros::impl_common_helpers!(LlvmAr);
92+
crate::macros::impl_common_helpers!(LlvmBcanalyzer);
7993

8094
/// Generate the path to the bin directory of LLVM.
8195
#[must_use]
@@ -236,6 +250,12 @@ impl LlvmAr {
236250
self
237251
}
238252

253+
/// Extract archive members back to files.
254+
pub fn extract(&mut self) -> &mut Self {
255+
self.cmd.arg("x");
256+
self
257+
}
258+
239259
/// Provide an output, then an input file. Bundled in one function, as llvm-ar has
240260
/// no "--output"-style flag.
241261
pub fn output_input(&mut self, out: impl AsRef<Path>, input: impl AsRef<Path>) -> &mut Self {
@@ -244,3 +264,19 @@ impl LlvmAr {
244264
self
245265
}
246266
}
267+
268+
impl LlvmBcanalyzer {
269+
/// Construct a new `llvm-bcanalyzer` invocation. This assumes that `llvm-bcanalyzer` is available
270+
/// at `$LLVM_BIN_DIR/llvm-bcanalyzer`.
271+
pub fn new() -> Self {
272+
let llvm_bcanalyzer = llvm_bin_dir().join("llvm-bcanalyzer");
273+
let cmd = Command::new(llvm_bcanalyzer);
274+
Self { cmd }
275+
}
276+
277+
/// Provide an input file.
278+
pub fn input<P: AsRef<Path>>(&mut self, path: P) -> &mut Self {
279+
self.cmd.arg(path.as_ref());
280+
self
281+
}
282+
}

src/tools/run-make-support/src/lib.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -48,8 +48,8 @@ pub use cc::{cc, cxx, extra_c_flags, extra_cxx_flags, Cc};
4848
pub use clang::{clang, Clang};
4949
pub use htmldocck::htmldocck;
5050
pub use llvm::{
51-
llvm_ar, llvm_filecheck, llvm_objdump, llvm_profdata, llvm_readobj, LlvmAr, LlvmFilecheck,
52-
LlvmObjdump, LlvmProfdata, LlvmReadobj,
51+
llvm_ar, llvm_bcanalyzer, llvm_filecheck, llvm_objdump, llvm_profdata, llvm_readobj, LlvmAr,
52+
LlvmBcanalyzer, LlvmFilecheck, LlvmObjdump, LlvmProfdata, LlvmReadobj,
5353
};
5454
pub use python::python_command;
5555
pub use rustc::{aux_build, bare_rustc, rustc, Rustc};

src/tools/tidy/src/allowed_run_make_makefiles.txt

-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,6 @@ run-make/cdylib-dylib-linkage/Makefile
44
run-make/cross-lang-lto-clang/Makefile
55
run-make/cross-lang-lto-pgo-smoketest/Makefile
66
run-make/cross-lang-lto-upstream-rlibs/Makefile
7-
run-make/cross-lang-lto/Makefile
87
run-make/dep-info-doesnt-run-much/Makefile
98
run-make/dep-info-spaces/Makefile
109
run-make/dep-info/Makefile

tests/run-make/cross-lang-lto/Makefile

-57
This file was deleted.
+110
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,110 @@
1+
// This test checks that the object files we generate are actually
2+
// LLVM bitcode files (as used by linker LTO plugins) when compiling with
3+
// -Clinker-plugin-lto.
4+
// See https://github.com/rust-lang/rust/pull/50000
5+
6+
#![feature(path_file_prefix)]
7+
8+
use std::path::PathBuf;
9+
10+
use run_make_support::{
11+
cwd, has_extension, has_prefix, llvm_ar, llvm_bcanalyzer, path, rfs, rust_lib_name, rustc,
12+
shallow_find_files, static_lib_name,
13+
};
14+
15+
fn main() {
16+
check_bitcode(LibBuild {
17+
source: path("lib.rs"),
18+
crate_type: Some("staticlib"),
19+
output: path(static_lib_name("liblib")),
20+
lto: None,
21+
emit_obj: false,
22+
});
23+
check_bitcode(LibBuild {
24+
source: path("lib.rs"),
25+
crate_type: Some("staticlib"),
26+
output: path(static_lib_name("liblib-fat-lto")),
27+
lto: Some("fat"),
28+
emit_obj: false,
29+
});
30+
check_bitcode(LibBuild {
31+
source: path("lib.rs"),
32+
crate_type: Some("staticlib"),
33+
output: path(static_lib_name("liblib-thin-lto")),
34+
lto: Some("thin"),
35+
emit_obj: false,
36+
});
37+
check_bitcode(LibBuild {
38+
source: path("lib.rs"),
39+
crate_type: Some("rlib"),
40+
output: path(rust_lib_name("liblib")),
41+
lto: None,
42+
emit_obj: false,
43+
});
44+
check_bitcode(LibBuild {
45+
source: path("lib.rs"),
46+
crate_type: Some("cdylib"),
47+
output: path("cdylib.o"),
48+
lto: None,
49+
emit_obj: true,
50+
});
51+
check_bitcode(LibBuild {
52+
source: path("lib.rs"),
53+
crate_type: Some("dylib"),
54+
output: path("rdylib.o"),
55+
lto: None,
56+
emit_obj: true,
57+
});
58+
check_bitcode(LibBuild {
59+
source: path("main.rs"),
60+
crate_type: None,
61+
output: path("exe.o"),
62+
lto: None,
63+
emit_obj: true,
64+
});
65+
}
66+
67+
#[track_caller]
68+
fn check_bitcode(instructions: LibBuild) {
69+
let mut rustc = rustc();
70+
rustc
71+
.input(instructions.source)
72+
.output(&instructions.output)
73+
.opt_level("2")
74+
.codegen_units(1)
75+
.arg("-Clinker-plugin-lto");
76+
if instructions.emit_obj {
77+
rustc.emit("obj");
78+
}
79+
if let Some(crate_type) = instructions.crate_type {
80+
rustc.crate_type(crate_type);
81+
}
82+
if let Some(lto) = instructions.lto {
83+
rustc.arg(format!("-Clto={lto}"));
84+
}
85+
rustc.run();
86+
87+
if instructions.output.extension().unwrap() != "o" {
88+
// Remove all potential leftover object files, then turn the output into an object file.
89+
for object in shallow_find_files(cwd(), |path| has_extension(path, "o")) {
90+
rfs::remove_file(object);
91+
}
92+
llvm_ar().extract().arg(&instructions.output).run();
93+
}
94+
95+
for object in shallow_find_files(cwd(), |path| {
96+
has_prefix(path, instructions.output.file_prefix().unwrap().to_str().unwrap())
97+
&& has_extension(path, "o")
98+
}) {
99+
// All generated object files should be LLVM bitcode files - this will fail otherwise.
100+
llvm_bcanalyzer().input(object).run();
101+
}
102+
}
103+
104+
struct LibBuild {
105+
source: PathBuf,
106+
crate_type: Option<&'static str>,
107+
output: PathBuf,
108+
lto: Option<&'static str>,
109+
emit_obj: bool,
110+
}

0 commit comments

Comments
 (0)