Skip to content

Commit 0bbb1fd

Browse files
committed
Auto merge of rust-lang#124362 - tmandry:panic-abort-rustc, r=<try>
[experiment] Build the compiler with panic=abort This wasn't possible until rust-lang#122362 (which this is based on, still not merged), and I want to see what performance benefits there are. This is only a proof-of-concept to get an upper bound. Some functionality around fatal errors needs to be restored using panic hooks, and without that it's likely that important code paths are being optimized out. About 4% of the ui tests are failing. With that said, in my local stage 2 builds this cuts the size of librustc_driver by 24%. That's pretty in line with the results I observed when turning this on for Fuchsia – savings were around 20%.
2 parents 9e6c4fd + 3d781b9 commit 0bbb1fd

File tree

14 files changed

+97
-19
lines changed

14 files changed

+97
-19
lines changed

compiler/rustc/src/main.rs

+2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,6 @@
11
#![feature(unix_sigpipe)]
2+
// We need this feature as it changes `dylib` linking behavior and allows us to link to `rustc_driver`.
3+
#![feature(rustc_private)]
24

35
// A note about jemalloc: rustc uses jemalloc when built for CI and
46
// distribution. The obvious way to do this is with the `#[global_allocator]`

compiler/rustc_metadata/messages.ftl

+3
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,9 @@ metadata_crate_dep_multiple =
4141
metadata_crate_dep_not_static =
4242
`{$crate_name}` was unavailable as a static crate, preventing fully static linking
4343
44+
metadata_crate_dep_rustc_driver =
45+
`feature(rustc_private)` is needed to link to the compiler's `rustc_driver` library
46+
4447
metadata_crate_location_unknown_type =
4548
extern location for {$crate_name} is of an unknown type: {$path}
4649

compiler/rustc_metadata/src/dependency_format.rs

+42-13
Original file line numberDiff line numberDiff line change
@@ -54,16 +54,18 @@
5454
use crate::creader::CStore;
5555
use crate::errors::{
5656
BadPanicStrategy, CrateDepMultiple, IncompatiblePanicInDropStrategy, LibRequired,
57-
NonStaticCrateDep, RequiredPanicStrategy, RlibRequired, RustcLibRequired, TwoPanicRuntimes,
57+
NonStaticCrateDep, RequiredPanicStrategy, RlibRequired, RustcDriverHelp, RustcLibRequired,
58+
TwoPanicRuntimes,
5859
};
5960

60-
use rustc_data_structures::fx::FxHashMap;
61+
use rustc_data_structures::fx::{FxHashMap, FxHashSet};
6162
use rustc_hir::def_id::CrateNum;
6263
use rustc_middle::middle::dependency_format::{Dependencies, DependencyList, Linkage};
6364
use rustc_middle::ty::TyCtxt;
6465
use rustc_session::config::CrateType;
6566
use rustc_session::cstore::CrateDepKind;
6667
use rustc_session::cstore::LinkagePreference::{self, RequireDynamic, RequireStatic};
68+
use rustc_span::sym;
6769

6870
pub(crate) fn calculate(tcx: TyCtxt<'_>) -> Dependencies {
6971
tcx.crate_types()
@@ -158,25 +160,49 @@ fn calculate_type(tcx: TyCtxt<'_>, ty: CrateType) -> DependencyList {
158160
Linkage::Dynamic | Linkage::IncludedFromDylib => {}
159161
}
160162

163+
let all_dylibs = || {
164+
tcx.crates(()).iter().filter(|&&cnum| {
165+
!tcx.dep_kind(cnum).macros_only() && tcx.used_crate_source(cnum).dylib.is_some()
166+
})
167+
};
168+
169+
let mut upstream_in_dylibs = FxHashSet::default();
170+
171+
if tcx.features().rustc_private {
172+
// We need this to prevent users of `rustc_driver` from linking to dynamically to `std`
173+
// which does not work as `std` is also statically linked into `rustc_driver`.
174+
175+
// Find all libraries statically linked to upstream dylibs.
176+
for &cnum in all_dylibs() {
177+
let deps = tcx.dylib_dependency_formats(cnum);
178+
for &(depnum, style) in deps.iter() {
179+
if let RequireStatic = style {
180+
upstream_in_dylibs.insert(depnum);
181+
}
182+
}
183+
}
184+
}
185+
161186
let mut formats = FxHashMap::default();
162187

163188
// Sweep all crates for found dylibs. Add all dylibs, as well as their
164189
// dependencies, ensuring there are no conflicts. The only valid case for a
165190
// dependency to be relied upon twice is for both cases to rely on a dylib.
166-
for &cnum in tcx.crates(()).iter() {
167-
if tcx.dep_kind(cnum).macros_only() {
191+
for &cnum in all_dylibs() {
192+
if upstream_in_dylibs.contains(&cnum) {
193+
info!("skipping dylib: {}", tcx.crate_name(cnum));
194+
// If this dylib is also available statically linked to another dylib
195+
// we try to use that instead.
168196
continue;
169197
}
198+
170199
let name = tcx.crate_name(cnum);
171-
let src = tcx.used_crate_source(cnum);
172-
if src.dylib.is_some() {
173-
info!("adding dylib: {}", name);
174-
add_library(tcx, cnum, RequireDynamic, &mut formats, &mut unavailable_as_static);
175-
let deps = tcx.dylib_dependency_formats(cnum);
176-
for &(depnum, style) in deps.iter() {
177-
info!("adding {:?}: {}", style, tcx.crate_name(depnum));
178-
add_library(tcx, depnum, style, &mut formats, &mut unavailable_as_static);
179-
}
200+
info!("adding dylib: {}", name);
201+
add_library(tcx, cnum, RequireDynamic, &mut formats, &mut unavailable_as_static);
202+
let deps = tcx.dylib_dependency_formats(cnum);
203+
for &(depnum, style) in deps.iter() {
204+
info!("adding {:?}: {}", style, tcx.crate_name(depnum));
205+
add_library(tcx, depnum, style, &mut formats, &mut unavailable_as_static);
180206
}
181207
}
182208

@@ -266,12 +292,15 @@ fn add_library(
266292
// This error is probably a little obscure, but I imagine that it
267293
// can be refined over time.
268294
if link2 != link || link == RequireStatic {
295+
let linking_to_rustc_driver = tcx.sess.psess.unstable_features.is_nightly_build()
296+
&& tcx.crates(()).iter().any(|&cnum| tcx.crate_name(cnum) == sym::rustc_driver);
269297
tcx.dcx().emit_err(CrateDepMultiple {
270298
crate_name: tcx.crate_name(cnum),
271299
non_static_deps: unavailable_as_static
272300
.drain(..)
273301
.map(|cnum| NonStaticCrateDep { crate_name: tcx.crate_name(cnum) })
274302
.collect(),
303+
rustc_driver_help: linking_to_rustc_driver.then_some(RustcDriverHelp),
275304
});
276305
}
277306
}

compiler/rustc_metadata/src/errors.rs

+6
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,8 @@ pub struct CrateDepMultiple {
4040
pub crate_name: Symbol,
4141
#[subdiagnostic]
4242
pub non_static_deps: Vec<NonStaticCrateDep>,
43+
#[subdiagnostic]
44+
pub rustc_driver_help: Option<RustcDriverHelp>,
4345
}
4446

4547
#[derive(Subdiagnostic)]
@@ -48,6 +50,10 @@ pub struct NonStaticCrateDep {
4850
pub crate_name: Symbol,
4951
}
5052

53+
#[derive(Subdiagnostic)]
54+
#[help(metadata_crate_dep_rustc_driver)]
55+
pub struct RustcDriverHelp;
56+
5157
#[derive(Diagnostic)]
5258
#[diag(metadata_two_panic_runtimes)]
5359
pub struct TwoPanicRuntimes {

compiler/rustc_span/src/symbol.rs

+1
Original file line numberDiff line numberDiff line change
@@ -1573,6 +1573,7 @@ symbols! {
15731573
rustc_dirty,
15741574
rustc_do_not_const_check,
15751575
rustc_doc_primitive,
1576+
rustc_driver,
15761577
rustc_dummy,
15771578
rustc_dump_env_program_clauses,
15781579
rustc_dump_program_clauses,

src/bootstrap/src/bin/rustc.rs

+17-3
Original file line numberDiff line numberDiff line change
@@ -91,6 +91,23 @@ fn main() {
9191
rustc_real
9292
};
9393

94+
// Get the name of the crate we're compiling, if any.
95+
let crate_name = arg("--crate-name");
96+
97+
// We want everything statically linked into `rustc_driver`, so remove `-C prefer-dynamic`
98+
if crate_name == Some("rustc_driver") && stage != "0" {
99+
// Remove `-C prefer-dynamic` to link `std` statically into `rustc_driver`
100+
if let Some(pos) = args.iter().enumerate().position(|(i, a)| {
101+
a == "-C" && args.get(i + 1).map(|a| a == "prefer-dynamic").unwrap_or(false)
102+
}) {
103+
args.remove(pos);
104+
args.remove(pos);
105+
}
106+
if let Some(pos) = args.iter().position(|a| a == "-Cprefer-dynamic") {
107+
args.remove(pos);
108+
}
109+
}
110+
94111
let mut cmd = if let Some(wrapper) = env::var_os("RUSTC_WRAPPER_REAL") {
95112
let mut cmd = Command::new(wrapper);
96113
cmd.arg(rustc_driver);
@@ -100,9 +117,6 @@ fn main() {
100117
};
101118
cmd.args(&args).env(dylib_path_var(), env::join_paths(&dylib_path).unwrap());
102119

103-
// Get the name of the crate we're compiling, if any.
104-
let crate_name = arg("--crate-name");
105-
106120
if let Some(crate_name) = crate_name {
107121
if let Some(target) = env::var_os("RUSTC_TIME") {
108122
if target == "all"

src/bootstrap/src/core/build_steps/compile.rs

+11-1
Original file line numberDiff line numberDiff line change
@@ -1079,6 +1079,11 @@ pub fn rustc_cargo(
10791079
));
10801080
}
10811081

1082+
if compiler.stage != 0 {
1083+
cargo.rustflag("-Cpanic=abort");
1084+
cargo.rustflag("-Cforce-unwind-tables=yes");
1085+
}
1086+
10821087
rustc_cargo_env(builder, cargo, target, compiler.stage);
10831088
}
10841089

@@ -1780,7 +1785,12 @@ impl Step for Assemble {
17801785
let src_libdir = builder.sysroot_libdir(build_compiler, host);
17811786
for f in builder.read_dir(&src_libdir) {
17821787
let filename = f.file_name().into_string().unwrap();
1783-
if (is_dylib(&filename) || is_debug_info(&filename)) && !proc_macros.contains(&filename)
1788+
let can_be_rustc_dep = filename.starts_with("rustc_driver-")
1789+
|| filename.starts_with("librustc_driver-")
1790+
|| build_compiler.stage == 0;
1791+
if can_be_rustc_dep
1792+
&& (is_dylib(&filename) || is_debug_info(&filename))
1793+
&& !proc_macros.contains(&filename)
17841794
{
17851795
builder.copy_link(&f.path(), &rustc_libdir.join(&filename));
17861796
}

src/bootstrap/src/core/builder.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -2064,7 +2064,7 @@ impl<'a> Builder<'a> {
20642064
// When we build Rust dylibs they're all intended for intermediate
20652065
// usage, so make sure we pass the -Cprefer-dynamic flag instead of
20662066
// linking all deps statically into the dylib.
2067-
if matches!(mode, Mode::Std | Mode::Rustc) {
2067+
if matches!(mode, Mode::Std) {
20682068
rustflags.arg("-Cprefer-dynamic");
20692069
}
20702070

src/ci/github-actions/jobs.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -84,7 +84,7 @@ pr:
8484
try:
8585
- image: dist-x86_64-linux
8686
env:
87-
CODEGEN_BACKENDS: llvm,cranelift
87+
CODEGEN_BACKENDS: llvm
8888
<<: *job-linux-16c
8989

9090
# Main CI jobs that have to be green to merge a commit into master

src/tools/clippy/src/main.rs

+3
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,6 @@
1+
// We need this feature as it changes `dylib` linking behavior and allows us to link to
2+
// `rustc_driver`.
3+
#![feature(rustc_private)]
14
#![cfg_attr(feature = "deny-warnings", deny(warnings))]
25
// warn on lints, that are included in `rust-lang/rust`s bootstrap
36
#![warn(rust_2018_idioms, unused_lifetimes)]

src/tools/clippy/tests/compile-test.rs

+3
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,6 @@
1+
// We need this feature as it changes `dylib` linking behavior and allows us to link to
2+
// `rustc_driver`.
3+
#![feature(rustc_private)]
14
#![feature(lazy_cell)]
25
#![feature(is_sorted)]
36
#![cfg_attr(feature = "deny-warnings", deny(warnings))]

src/tools/opt-dist/src/main.rs

+1
Original file line numberDiff line numberDiff line change
@@ -376,6 +376,7 @@ fn main() -> anyhow::Result<()> {
376376
"clippy",
377377
"miri",
378378
"rustfmt",
379+
"rustdoc",
379380
] {
380381
build_args.extend(["--skip".to_string(), target.to_string()]);
381382
}

src/tools/rustdoc/main.rs

+2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,6 @@
11
#![feature(unix_sigpipe)]
2+
// We need this feature as it changes `dylib` linking behavior and allows us to link to `rustc_driver`.
3+
#![feature(rustc_private)]
24

35
#[unix_sigpipe = "sig_dfl"]
46
fn main() {

src/tools/rustfmt/src/git-rustfmt/main.rs

+4
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,7 @@
1+
// We need this feature as it changes `dylib` linking behavior and allows us to link to
2+
// `rustc_driver`.
3+
#![feature(rustc_private)]
4+
15
#[macro_use]
26
extern crate tracing;
37

0 commit comments

Comments
 (0)