@@ -22,7 +22,9 @@ use rustc_session::utils::NativeLibKind;
22
22
/// need out of the shared crate context before we get rid of it.
23
23
use rustc_session:: { filesearch, Session } ;
24
24
use rustc_span:: symbol:: Symbol ;
25
- use rustc_target:: spec:: crt_objects:: { CrtObjects , LinkSelfContainedDefault } ;
25
+ use rustc_target:: spec:: crt_objects:: CrtObjects ;
26
+ use rustc_target:: spec:: LinkSelfContainedComponents ;
27
+ use rustc_target:: spec:: LinkSelfContainedDefault ;
26
28
use rustc_target:: spec:: { Cc , LinkOutputKind , LinkerFlavor , Lld , PanicStrategy } ;
27
29
use rustc_target:: spec:: { RelocModel , RelroLevel , SanitizerSet , SplitDebuginfo } ;
28
30
@@ -728,6 +730,7 @@ fn link_natively<'a>(
728
730
) -> Result < ( ) , ErrorGuaranteed > {
729
731
info ! ( "preparing {:?} to {:?}" , crate_type, out_filename) ;
730
732
let ( linker_path, flavor) = linker_and_flavor ( sess) ;
733
+ let self_contained_components = self_contained_components ( sess, crate_type) ;
731
734
let mut cmd = linker_with_args (
732
735
& linker_path,
733
736
flavor,
@@ -737,6 +740,7 @@ fn link_natively<'a>(
737
740
tmpdir,
738
741
out_filename,
739
742
codegen_results,
743
+ self_contained_components,
740
744
) ?;
741
745
742
746
linker:: disable_localization ( & mut cmd) ;
@@ -812,14 +816,14 @@ fn link_natively<'a>(
812
816
"Linker does not support -static-pie command line option. Retrying with -static instead."
813
817
) ;
814
818
// Mirror `add_(pre,post)_link_objects` to replace CRT objects.
815
- let self_contained = self_contained ( sess , crate_type ) ;
819
+ let self_contained_crt_objects = self_contained_components . is_crt_objects_enabled ( ) ;
816
820
let opts = & sess. target ;
817
- let pre_objects = if self_contained {
821
+ let pre_objects = if self_contained_crt_objects {
818
822
& opts. pre_link_objects_self_contained
819
823
} else {
820
824
& opts. pre_link_objects
821
825
} ;
822
- let post_objects = if self_contained {
826
+ let post_objects = if self_contained_crt_objects {
823
827
& opts. post_link_objects_self_contained
824
828
} else {
825
829
& opts. post_link_objects
@@ -830,7 +834,9 @@ fn link_natively<'a>(
830
834
. iter ( )
831
835
. copied ( )
832
836
. flatten ( )
833
- . map ( |obj| get_object_file_path ( sess, obj, self_contained) . into_os_string ( ) )
837
+ . map ( |obj| {
838
+ get_object_file_path ( sess, obj, self_contained_crt_objects) . into_os_string ( )
839
+ } )
834
840
. collect :: < Vec < _ > > ( )
835
841
} ;
836
842
let pre_objects_static_pie = get_objects ( pre_objects, LinkOutputKind :: StaticPicExe ) ;
@@ -1710,26 +1716,43 @@ fn detect_self_contained_mingw(sess: &Session) -> bool {
1710
1716
/// Various toolchain components used during linking are used from rustc distribution
1711
1717
/// instead of being found somewhere on the host system.
1712
1718
/// We only provide such support for a very limited number of targets.
1713
- fn self_contained ( sess : & Session , crate_type : CrateType ) -> bool {
1714
- if let Some ( self_contained) = sess. opts . cg . link_self_contained . explicitly_set {
1715
- if sess. target . link_self_contained == LinkSelfContainedDefault :: False {
1716
- sess. emit_err ( errors:: UnsupportedLinkSelfContained ) ;
1717
- }
1718
- return self_contained;
1719
- }
1719
+ fn self_contained_components ( sess : & Session , crate_type : CrateType ) -> LinkSelfContainedComponents {
1720
+ // Turn the backwards compatible bool values for `self_contained` into fully inferred
1721
+ // `LinkSelfContainedComponents`.
1722
+ let self_contained =
1723
+ if let Some ( self_contained) = sess. opts . cg . link_self_contained . explicitly_set {
1724
+ // Emit an error if the user requested self-contained mode on the CLI but the target
1725
+ // explicitly refuses it.
1726
+ if sess. target . link_self_contained . is_disabled ( ) {
1727
+ sess. emit_err ( errors:: UnsupportedLinkSelfContained ) ;
1728
+ }
1729
+ self_contained
1730
+ } else {
1731
+ match sess. target . link_self_contained {
1732
+ LinkSelfContainedDefault :: False => false ,
1733
+ LinkSelfContainedDefault :: True => true ,
1734
+
1735
+ LinkSelfContainedDefault :: WithComponents ( components) => {
1736
+ // For target specs with explicitly enabled components, we can return them
1737
+ // directly.
1738
+ return components;
1739
+ }
1720
1740
1721
- match sess. target . link_self_contained {
1722
- LinkSelfContainedDefault :: False => false ,
1723
- LinkSelfContainedDefault :: True => true ,
1724
- // FIXME: Find a better heuristic for "native musl toolchain is available",
1725
- // based on host and linker path, for example.
1726
- // (https://github.com/rust-lang/rust/pull/71769#issuecomment-626330237).
1727
- LinkSelfContainedDefault :: Musl => sess. crt_static ( Some ( crate_type) ) ,
1728
- LinkSelfContainedDefault :: Mingw => {
1729
- sess. host == sess. target
1730
- && sess. target . vendor != "uwp"
1731
- && detect_self_contained_mingw ( & sess)
1732
- }
1741
+ // FIXME: Find a better heuristic for "native musl toolchain is available",
1742
+ // based on host and linker path, for example.
1743
+ // (https://github.com/rust-lang/rust/pull/71769#issuecomment-626330237).
1744
+ LinkSelfContainedDefault :: InferredForMusl => sess. crt_static ( Some ( crate_type) ) ,
1745
+ LinkSelfContainedDefault :: InferredForMingw => {
1746
+ sess. host == sess. target
1747
+ && sess. target . vendor != "uwp"
1748
+ && detect_self_contained_mingw ( & sess)
1749
+ }
1750
+ }
1751
+ } ;
1752
+ if self_contained {
1753
+ LinkSelfContainedComponents :: all ( )
1754
+ } else {
1755
+ LinkSelfContainedComponents :: empty ( )
1733
1756
}
1734
1757
}
1735
1758
@@ -2030,13 +2053,14 @@ fn linker_with_args<'a>(
2030
2053
tmpdir : & Path ,
2031
2054
out_filename : & Path ,
2032
2055
codegen_results : & CodegenResults ,
2056
+ self_contained_components : LinkSelfContainedComponents ,
2033
2057
) -> Result < Command , ErrorGuaranteed > {
2034
- let self_contained = self_contained ( sess , crate_type ) ;
2058
+ let self_contained_crt_objects = self_contained_components . is_crt_objects_enabled ( ) ;
2035
2059
let cmd = & mut * super :: linker:: get_linker (
2036
2060
sess,
2037
2061
path,
2038
2062
flavor,
2039
- self_contained ,
2063
+ self_contained_components . are_any_components_enabled ( ) ,
2040
2064
& codegen_results. crate_info . target_cpu ,
2041
2065
) ;
2042
2066
let link_output_kind = link_output_kind ( sess, crate_type) ;
@@ -2063,7 +2087,7 @@ fn linker_with_args<'a>(
2063
2087
// ------------ Object code and libraries, order-dependent ------------
2064
2088
2065
2089
// Pre-link CRT objects.
2066
- add_pre_link_objects ( cmd, sess, flavor, link_output_kind, self_contained ) ;
2090
+ add_pre_link_objects ( cmd, sess, flavor, link_output_kind, self_contained_crt_objects ) ;
2067
2091
2068
2092
add_linked_symbol_object (
2069
2093
cmd,
@@ -2206,7 +2230,7 @@ fn linker_with_args<'a>(
2206
2230
cmd,
2207
2231
sess,
2208
2232
link_output_kind,
2209
- self_contained ,
2233
+ self_contained_components ,
2210
2234
flavor,
2211
2235
crate_type,
2212
2236
codegen_results,
@@ -2222,7 +2246,7 @@ fn linker_with_args<'a>(
2222
2246
// ------------ Object code and libraries, order-dependent ------------
2223
2247
2224
2248
// Post-link CRT objects.
2225
- add_post_link_objects ( cmd, sess, link_output_kind, self_contained ) ;
2249
+ add_post_link_objects ( cmd, sess, link_output_kind, self_contained_crt_objects ) ;
2226
2250
2227
2251
// ------------ Late order-dependent options ------------
2228
2252
@@ -2239,15 +2263,15 @@ fn add_order_independent_options(
2239
2263
cmd : & mut dyn Linker ,
2240
2264
sess : & Session ,
2241
2265
link_output_kind : LinkOutputKind ,
2242
- self_contained : bool ,
2266
+ self_contained_components : LinkSelfContainedComponents ,
2243
2267
flavor : LinkerFlavor ,
2244
2268
crate_type : CrateType ,
2245
2269
codegen_results : & CodegenResults ,
2246
2270
out_filename : & Path ,
2247
2271
tmpdir : & Path ,
2248
2272
) {
2249
2273
// Take care of the flavors and CLI options requesting the `lld` linker.
2250
- add_lld_args ( cmd, sess, flavor) ;
2274
+ add_lld_args ( cmd, sess, flavor, self_contained_components ) ;
2251
2275
2252
2276
add_apple_sdk ( cmd, sess, flavor) ;
2253
2277
@@ -2272,7 +2296,7 @@ fn add_order_independent_options(
2272
2296
// Make the binary compatible with data execution prevention schemes.
2273
2297
cmd. add_no_exec ( ) ;
2274
2298
2275
- if self_contained {
2299
+ if self_contained_components . is_crt_objects_enabled ( ) {
2276
2300
cmd. no_crt_objects ( ) ;
2277
2301
}
2278
2302
@@ -2303,7 +2327,7 @@ fn add_order_independent_options(
2303
2327
2304
2328
cmd. linker_plugin_lto ( ) ;
2305
2329
2306
- add_library_search_dirs ( cmd, sess, self_contained ) ;
2330
+ add_library_search_dirs ( cmd, sess, self_contained_components . are_any_components_enabled ( ) ) ;
2307
2331
2308
2332
cmd. output_filename ( out_filename) ;
2309
2333
@@ -2953,8 +2977,16 @@ fn get_apple_sdk_root(sdk_name: &str) -> Result<String, errors::AppleSdkRootErro
2953
2977
/// invoke it:
2954
2978
/// - when the self-contained linker flag is active: the build of `lld` distributed with rustc,
2955
2979
/// - or any `lld` available to `cc`.
2956
- fn add_lld_args ( cmd : & mut dyn Linker , sess : & Session , flavor : LinkerFlavor ) {
2957
- debug ! ( "add_lld_args requested, flavor: '{flavor:?}'" ) ;
2980
+ fn add_lld_args (
2981
+ cmd : & mut dyn Linker ,
2982
+ sess : & Session ,
2983
+ flavor : LinkerFlavor ,
2984
+ self_contained_components : LinkSelfContainedComponents ,
2985
+ ) {
2986
+ debug ! (
2987
+ "add_lld_args requested, flavor: '{:?}', target self-contained components: {:?}" ,
2988
+ flavor, self_contained_components,
2989
+ ) ;
2958
2990
2959
2991
// If the flavor doesn't use a C/C++ compiler to invoke the linker, or doesn't opt in to `lld`,
2960
2992
// we don't need to do anything.
@@ -2963,9 +2995,32 @@ fn add_lld_args(cmd: &mut dyn Linker, sess: &Session, flavor: LinkerFlavor) {
2963
2995
}
2964
2996
2965
2997
// 1. Implement the "self-contained" part of this feature by adding rustc distribution
2966
- // directories to the tool's search path:
2967
- // - if the self-contained linker is enabled on the CLI.
2968
- if sess. opts . cg . link_self_contained . is_linker_enabled ( ) {
2998
+ // directories to the tool's search path, depending on a mix between what users can specify on
2999
+ // the CLI, and what the target spec enables (as it can't disable components):
3000
+ // - if the self-contained linker is enabled on the CLI or by the target spec,
3001
+ // - and if the self-contained linker is not disabled on the CLI.
3002
+ let self_contained_cli = sess. opts . cg . link_self_contained . is_linker_enabled ( ) ;
3003
+ let self_contained_target = self_contained_components. is_linker_enabled ( ) ;
3004
+
3005
+ // FIXME: in the future, codegen backends may need to have more control over this process: they
3006
+ // don't always support all the features the linker expects here, and vice versa. For example,
3007
+ // at the time of writing this, lld expects a newer style of aarch64 TLS relocations that
3008
+ // cranelift doesn't implement yet. That in turn can impact whether linking would succeed on
3009
+ // such a target when using the `cg_clif` backend and lld.
3010
+ //
3011
+ // Until interactions between backends and linker features are expressible, we limit target
3012
+ // specs to opt-in to lld only when we're on the llvm backend, where it's expected to work and
3013
+ // tested on CI. As usual, the CLI still has precedence over this, so that users and developers
3014
+ // can still override this default when needed (e.g. for tests).
3015
+ let uses_llvm_backend =
3016
+ matches ! ( sess. opts. unstable_opts. codegen_backend. as_deref( ) , None | Some ( "llvm" ) ) ;
3017
+ if !uses_llvm_backend && !self_contained_cli && sess. opts . cg . linker_flavor . is_none ( ) {
3018
+ // We bail if we're not using llvm and lld was not explicitly requested on the CLI.
3019
+ return ;
3020
+ }
3021
+
3022
+ let self_contained_linker = self_contained_cli || self_contained_target;
3023
+ if self_contained_linker && !sess. opts . cg . link_self_contained . is_linker_disabled ( ) {
2969
3024
for path in sess. get_tools_search_paths ( false ) {
2970
3025
cmd. arg ( {
2971
3026
let mut arg = OsString :: from ( "-B" ) ;
0 commit comments