@@ -4,7 +4,6 @@ use crate::core::compiler::context::Metadata;
4
4
use crate :: core:: compiler:: job_queue:: JobState ;
5
5
use crate :: core:: { profiles:: ProfileRoot , PackageId } ;
6
6
use crate :: util:: errors:: { CargoResult , CargoResultExt } ;
7
- use crate :: util:: interning:: InternedString ;
8
7
use crate :: util:: machine_message:: { self , Message } ;
9
8
use crate :: util:: { self , internal, paths, profile} ;
10
9
use cargo_platform:: Cfg ;
@@ -268,7 +267,7 @@ fn build_work(cx: &mut Context<'_, '_>, unit: &Unit) -> CargoResult<Job> {
268
267
}
269
268
} )
270
269
. collect :: < Vec < _ > > ( ) ;
271
- let pkg_name = unit. pkg . name ( ) ;
270
+ let library_name = unit. pkg . library ( ) . map ( |t| t . crate_name ( ) ) ;
272
271
let pkg_descr = unit. pkg . to_string ( ) ;
273
272
let build_script_outputs = Arc :: clone ( & cx. build_script_outputs ) ;
274
273
let id = unit. pkg . package_id ( ) ;
@@ -278,7 +277,7 @@ fn build_work(cx: &mut Context<'_, '_>, unit: &Unit) -> CargoResult<Job> {
278
277
let host_target_root = cx. files ( ) . host_dest ( ) . to_path_buf ( ) ;
279
278
let all = (
280
279
id,
281
- pkg_name ,
280
+ library_name . clone ( ) ,
282
281
pkg_descr. clone ( ) ,
283
282
Arc :: clone ( & build_script_outputs) ,
284
283
output_file. clone ( ) ,
@@ -398,7 +397,7 @@ fn build_work(cx: &mut Context<'_, '_>, unit: &Unit) -> CargoResult<Job> {
398
397
paths:: write ( & root_output_file, util:: path2bytes ( & script_out_dir) ?) ?;
399
398
let parsed_output = BuildOutput :: parse (
400
399
& output. stdout ,
401
- pkg_name ,
400
+ library_name ,
402
401
& pkg_descr,
403
402
& script_out_dir,
404
403
& script_out_dir,
@@ -420,12 +419,12 @@ fn build_work(cx: &mut Context<'_, '_>, unit: &Unit) -> CargoResult<Job> {
420
419
// itself to run when we actually end up just discarding what we calculated
421
420
// above.
422
421
let fresh = Work :: new ( move |state| {
423
- let ( id, pkg_name , pkg_descr, build_script_outputs, output_file, script_out_dir) = all;
422
+ let ( id, library_name , pkg_descr, build_script_outputs, output_file, script_out_dir) = all;
424
423
let output = match prev_output {
425
424
Some ( output) => output,
426
425
None => BuildOutput :: parse_file (
427
426
& output_file,
428
- pkg_name ,
427
+ library_name ,
429
428
& pkg_descr,
430
429
& prev_script_out_dir,
431
430
& script_out_dir,
@@ -477,7 +476,7 @@ fn insert_warnings_in_build_outputs(
477
476
impl BuildOutput {
478
477
pub fn parse_file (
479
478
path : & Path ,
480
- pkg_name : InternedString ,
479
+ library_name : Option < String > ,
481
480
pkg_descr : & str ,
482
481
script_out_dir_when_generated : & Path ,
483
482
script_out_dir : & Path ,
@@ -487,7 +486,7 @@ impl BuildOutput {
487
486
let contents = paths:: read_bytes ( path) ?;
488
487
BuildOutput :: parse (
489
488
& contents,
490
- pkg_name ,
489
+ library_name ,
491
490
pkg_descr,
492
491
script_out_dir_when_generated,
493
492
script_out_dir,
@@ -497,10 +496,12 @@ impl BuildOutput {
497
496
}
498
497
499
498
// Parses the output of a script.
500
- // The `pkg_name` is used for error messages.
499
+ // The `pkg_descr` is used for error messages.
500
+ // The `library_name` is used for determining if RUSTC_BOOTSTRAP should be allowed.
501
501
pub fn parse (
502
502
input : & [ u8 ] ,
503
- pkg_name : InternedString ,
503
+ // Takes String instead of InternedString so passing `unit.pkg.name()` will give a compile error.
504
+ library_name : Option < String > ,
504
505
pkg_descr : & str ,
505
506
script_out_dir_when_generated : & Path ,
506
507
script_out_dir : & Path ,
@@ -587,7 +588,23 @@ impl BuildOutput {
587
588
// to set RUSTC_BOOTSTRAP.
588
589
// If this is a nightly build, setting RUSTC_BOOTSTRAP wouldn't affect the
589
590
// behavior, so still only give a warning.
590
- if nightly_features_allowed {
591
+ // NOTE: cargo only allows nightly features on RUSTC_BOOTSTRAP=1, but we
592
+ // want setting any value of RUSTC_BOOTSTRAP to downgrade this to a warning
593
+ // (so that `RUSTC_BOOTSTRAP=library_name` will work)
594
+ let rustc_bootstrap_allows = |name : Option < & str > | {
595
+ let name = match name {
596
+ // as of 2021, no binaries on crates.io use RUSTC_BOOTSTRAP, so
597
+ // fine-grained opt-outs aren't needed. end-users can always use
598
+ // RUSTC_BOOTSTRAP=1 from the top-level if it's really a problem.
599
+ None => return false ,
600
+ Some ( n) => n,
601
+ } ;
602
+ std:: env:: var ( "RUSTC_BOOTSTRAP" )
603
+ . map_or ( false , |var| var. split ( ',' ) . any ( |s| s == name) )
604
+ } ;
605
+ if nightly_features_allowed
606
+ || rustc_bootstrap_allows ( library_name. as_deref ( ) )
607
+ {
591
608
warnings. push ( format ! ( "Cannot set `RUSTC_BOOTSTRAP={}` from {}.\n \
592
609
note: Crates cannot set `RUSTC_BOOTSTRAP` themselves, as doing so would subvert the stability guarantees of Rust for your project.",
593
610
val, whence
@@ -600,7 +617,7 @@ impl BuildOutput {
600
617
help: If you're sure you want to do this in your project, set the environment variable `RUSTC_BOOTSTRAP={}` before running cargo instead.",
601
618
val,
602
619
whence,
603
- pkg_name ,
620
+ library_name . as_deref ( ) . unwrap_or ( "1" ) ,
604
621
) ;
605
622
}
606
623
} else {
@@ -857,7 +874,7 @@ fn prev_build_output(cx: &mut Context<'_, '_>, unit: &Unit) -> (Option<BuildOutp
857
874
(
858
875
BuildOutput :: parse_file (
859
876
& output_file,
860
- unit. pkg . name ( ) ,
877
+ unit. pkg . library ( ) . map ( |t| t . crate_name ( ) ) ,
861
878
& unit. pkg . to_string ( ) ,
862
879
& prev_script_out_dir,
863
880
& script_out_dir,
0 commit comments