-
Notifications
You must be signed in to change notification settings - Fork 13.2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Link std
statically in rustc_driver
#122362
Changes from all commits
3ee4325
736a249
486864f
56beb1d
dd3f757
b22253c
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -51,20 +51,22 @@ | |
//! Additionally, the algorithm is geared towards finding *any* solution rather | ||
//! than finding a number of solutions (there are normally quite a few). | ||
|
||
use rustc_data_structures::fx::FxHashMap; | ||
use rustc_data_structures::fx::{FxHashMap, FxHashSet}; | ||
use rustc_hir::def_id::CrateNum; | ||
use rustc_middle::bug; | ||
use rustc_middle::middle::dependency_format::{Dependencies, DependencyList, Linkage}; | ||
use rustc_middle::ty::TyCtxt; | ||
use rustc_session::config::CrateType; | ||
use rustc_session::cstore::CrateDepKind; | ||
use rustc_session::cstore::LinkagePreference::{self, RequireDynamic, RequireStatic}; | ||
use rustc_span::sym; | ||
use tracing::info; | ||
|
||
use crate::creader::CStore; | ||
use crate::errors::{ | ||
BadPanicStrategy, CrateDepMultiple, IncompatiblePanicInDropStrategy, LibRequired, | ||
NonStaticCrateDep, RequiredPanicStrategy, RlibRequired, RustcLibRequired, TwoPanicRuntimes, | ||
NonStaticCrateDep, RequiredPanicStrategy, RlibRequired, RustcDriverHelp, RustcLibRequired, | ||
TwoPanicRuntimes, | ||
}; | ||
|
||
pub(crate) fn calculate(tcx: TyCtxt<'_>) -> Dependencies { | ||
|
@@ -160,25 +162,49 @@ fn calculate_type(tcx: TyCtxt<'_>, ty: CrateType) -> DependencyList { | |
Linkage::Dynamic | Linkage::IncludedFromDylib => {} | ||
} | ||
|
||
let all_dylibs = || { | ||
tcx.crates(()).iter().filter(|&&cnum| { | ||
!tcx.dep_kind(cnum).macros_only() && tcx.used_crate_source(cnum).dylib.is_some() | ||
}) | ||
}; | ||
|
||
let mut upstream_in_dylibs = FxHashSet::default(); | ||
|
||
if tcx.features().rustc_private { | ||
// We need this to prevent users of `rustc_driver` from linking dynamically to `std` | ||
// which does not work as `std` is also statically linked into `rustc_driver`. | ||
|
||
// Find all libraries statically linked to upstream dylibs. | ||
for &cnum in all_dylibs() { | ||
let deps = tcx.dylib_dependency_formats(cnum); | ||
for &(depnum, style) in deps.iter() { | ||
if let RequireStatic = style { | ||
upstream_in_dylibs.insert(depnum); | ||
} | ||
} | ||
} | ||
} | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Instead of all this interspersed logic would it be possible to at the end iterate over all dylibs and mark all crates they have linked into them as There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Not sure if that's correct. The current logic starts by picking the dylibs then adding other crates as needed. That seems less likely to end up with missing crates than removing them after. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. My suggestion is not to remove crates, but to change their linkage to IncludeFromDylib after it has determined which crates to include. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. That's what I mean by remove as |
||
|
||
let mut formats = FxHashMap::default(); | ||
|
||
// Sweep all crates for found dylibs. Add all dylibs, as well as their | ||
// dependencies, ensuring there are no conflicts. The only valid case for a | ||
// dependency to be relied upon twice is for both cases to rely on a dylib. | ||
for &cnum in tcx.crates(()).iter() { | ||
if tcx.dep_kind(cnum).macros_only() { | ||
for &cnum in all_dylibs() { | ||
if upstream_in_dylibs.contains(&cnum) { | ||
info!("skipping dylib: {}", tcx.crate_name(cnum)); | ||
// If this dylib is also available statically linked to another dylib | ||
// we try to use that instead. | ||
continue; | ||
} | ||
|
||
let name = tcx.crate_name(cnum); | ||
let src = tcx.used_crate_source(cnum); | ||
if src.dylib.is_some() { | ||
info!("adding dylib: {}", name); | ||
add_library(tcx, cnum, RequireDynamic, &mut formats, &mut unavailable_as_static); | ||
let deps = tcx.dylib_dependency_formats(cnum); | ||
for &(depnum, style) in deps.iter() { | ||
info!("adding {:?}: {}", style, tcx.crate_name(depnum)); | ||
add_library(tcx, depnum, style, &mut formats, &mut unavailable_as_static); | ||
} | ||
info!("adding dylib: {}", name); | ||
add_library(tcx, cnum, RequireDynamic, &mut formats, &mut unavailable_as_static); | ||
let deps = tcx.dylib_dependency_formats(cnum); | ||
for &(depnum, style) in deps.iter() { | ||
info!("adding {:?}: {}", style, tcx.crate_name(depnum)); | ||
add_library(tcx, depnum, style, &mut formats, &mut unavailable_as_static); | ||
} | ||
} | ||
|
||
|
@@ -268,12 +294,15 @@ fn add_library( | |
// This error is probably a little obscure, but I imagine that it | ||
// can be refined over time. | ||
if link2 != link || link == RequireStatic { | ||
let linking_to_rustc_driver = tcx.sess.psess.unstable_features.is_nightly_build() | ||
&& tcx.crates(()).iter().any(|&cnum| tcx.crate_name(cnum) == sym::rustc_driver); | ||
tcx.dcx().emit_err(CrateDepMultiple { | ||
crate_name: tcx.crate_name(cnum), | ||
non_static_deps: unavailable_as_static | ||
.drain(..) | ||
.map(|cnum| NonStaticCrateDep { crate_name: tcx.crate_name(cnum) }) | ||
.collect(), | ||
rustc_driver_help: linking_to_rustc_driver.then_some(RustcDriverHelp), | ||
}); | ||
} | ||
} | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -1106,6 +1106,12 @@ impl<'a> Builder<'a> { | |
StepDescription::run(v, self, paths); | ||
} | ||
|
||
/// Returns if `std` should be statically linked into `rustc_driver`. | ||
/// It's currently not done on `windows-gnu` due to linker bugs. | ||
pub fn link_std_into_rustc_driver(&self, target: TargetSelection) -> bool { | ||
!target.triple.ends_with("-windows-gnu") | ||
} | ||
|
||
/// Obtain a compiler at a given stage and for a given host (i.e., this is the target that the | ||
/// compiler will run on, *not* the target it will build code for). Explicitly does not take | ||
/// `Compiler` since all `Compiler` instances are meant to be obtained through this function, | ||
|
@@ -2162,10 +2168,18 @@ impl<'a> Builder<'a> { | |
// When we build Rust dylibs they're all intended for intermediate | ||
// usage, so make sure we pass the -Cprefer-dynamic flag instead of | ||
// linking all deps statically into the dylib. | ||
if matches!(mode, Mode::Std | Mode::Rustc) { | ||
if matches!(mode, Mode::Std) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think you will need to add There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I don't see how there's any change to |
||
rustflags.arg("-Cprefer-dynamic"); | ||
} | ||
if matches!(mode, Mode::Rustc) && !self.link_std_into_rustc_driver(target) { | ||
rustflags.arg("-Cprefer-dynamic"); | ||
} | ||
|
||
cargo.env( | ||
"RUSTC_LINK_STD_INTO_RUSTC_DRIVER", | ||
Kobzol marked this conversation as resolved.
Show resolved
Hide resolved
|
||
if self.link_std_into_rustc_driver(target) { "1" } else { "0" }, | ||
); | ||
|
||
// When building incrementally we default to a lower ThinLTO import limit | ||
// (unless explicitly specified otherwise). This will produce a somewhat | ||
// slower code but give way better compile times. | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,11 @@ | ||
# `rustc_private` | ||
|
||
The tracking issue for this feature is: [#27812] | ||
|
||
[#27812]: https://github.com/rust-lang/rust/issues/27812 | ||
|
||
------------------------ | ||
|
||
This feature allows access to unstable internal compiler crates. | ||
|
||
Additionally it changes the linking behavior of crates which have this feature enabled. It will prevent linking to a dylib if there's a static variant of it already statically linked into another dylib dependency. This is required to successfully link to `rustc_driver`. |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,3 +1,6 @@ | ||
// We need this feature as it changes `dylib` linking behavior and allows us to link to `rustc_driver`. | ||
#![feature(rustc_private)] | ||
|
||
fn main() { | ||
rustdoc::main() | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I'm surprised we didn't need it before to invoke
rustc_driver::main
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We compile it with an unstable flag which both marks all unmarked items as rustc_private and allows calling rustc_private functions without enabling the feature gate.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That's also the reason why we should be able to do this change without modifying the rustc/rustdoc/clippy/rustfmt binaries, right? (like also checking for
-Zforce-unstable-if-unmarked
when deciding to look up the statically linked libraries).There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That didn't work out as some tool binaries only indirectly use rustc libraries and thus didn't have
feature(rustc_private)
.