@@ -77,6 +77,7 @@ use crate::core::compiler::future_incompat::{
77
77
use crate :: core:: resolver:: ResolveBehavior ;
78
78
use crate :: core:: { PackageId , Shell , TargetKind } ;
79
79
use crate :: util:: diagnostic_server:: { self , DiagnosticPrinter } ;
80
+ use crate :: util:: errors:: AlreadyPrintedError ;
80
81
use crate :: util:: machine_message:: { self , Message as _} ;
81
82
use crate :: util:: CargoResult ;
82
83
use crate :: util:: { self , internal, profile} ;
@@ -169,6 +170,41 @@ struct DrainState<'cfg> {
169
170
per_package_future_incompat_reports : Vec < FutureIncompatReportPackage > ,
170
171
}
171
172
173
+ pub struct ErrorsDuringDrain {
174
+ pub count : usize ,
175
+ }
176
+
177
+ struct ErrorToHandle {
178
+ error : anyhow:: Error ,
179
+
180
+ /// This field is true for "interesting" errors and false for "mundane"
181
+ /// errors. If false, we print the above error only if it's the first one
182
+ /// encountered so far while draining the job queue.
183
+ ///
184
+ /// At most places that an error is propagated, we set this to false to
185
+ /// avoid scenarios where Cargo might end up spewing tons of redundant error
186
+ /// messages. For example if an i/o stream got closed somewhere, we don't
187
+ /// care about individually reporting every thread that it broke; just the
188
+ /// first is enough.
189
+ ///
190
+ /// The exception where print_always is true is that we do report every
191
+ /// instance of a rustc invocation that failed with diagnostics. This
192
+ /// corresponds to errors from Message::Finish.
193
+ print_always : bool ,
194
+ }
195
+
196
+ impl < E > From < E > for ErrorToHandle
197
+ where
198
+ anyhow:: Error : From < E > ,
199
+ {
200
+ fn from ( error : E ) -> Self {
201
+ ErrorToHandle {
202
+ error : anyhow:: Error :: from ( error) ,
203
+ print_always : false ,
204
+ }
205
+ }
206
+ }
207
+
172
208
#[ derive( Debug , Copy , Clone , PartialEq , Eq , Hash , PartialOrd , Ord ) ]
173
209
pub struct JobId ( pub u32 ) ;
174
210
@@ -613,7 +649,7 @@ impl<'cfg> DrainState<'cfg> {
613
649
jobserver_helper : & HelperThread ,
614
650
plan : & mut BuildPlan ,
615
651
event : Message ,
616
- ) -> CargoResult < ( ) > {
652
+ ) -> Result < ( ) , ErrorToHandle > {
617
653
match event {
618
654
Message :: Run ( id, cmd) => {
619
655
cx. bcx
@@ -678,11 +714,14 @@ impl<'cfg> DrainState<'cfg> {
678
714
debug ! ( "end ({:?}): {:?}" , unit, result) ;
679
715
match result {
680
716
Ok ( ( ) ) => self . finish ( id, & unit, artifact, cx) ?,
681
- Err ( e ) => {
717
+ Err ( error ) => {
682
718
let msg = "The following warnings were emitted during compilation:" ;
683
719
self . emit_warnings ( Some ( msg) , & unit, cx) ?;
684
720
self . back_compat_notice ( cx, & unit) ?;
685
- return Err ( e) ;
721
+ return Err ( ErrorToHandle {
722
+ error,
723
+ print_always : true ,
724
+ } ) ;
686
725
}
687
726
}
688
727
}
@@ -787,14 +826,14 @@ impl<'cfg> DrainState<'cfg> {
787
826
// After a job has finished we update our internal state if it was
788
827
// successful and otherwise wait for pending work to finish if it failed
789
828
// and then immediately return.
790
- let mut error = None ;
829
+ let mut errors = ErrorsDuringDrain { count : 0 } ;
791
830
// CAUTION! Do not use `?` or break out of the loop early. Every error
792
831
// must be handled in such a way that the loop is still allowed to
793
832
// drain event messages.
794
833
loop {
795
- if error . is_none ( ) {
834
+ if errors . count == 0 {
796
835
if let Err ( e) = self . spawn_work_if_possible ( cx, jobserver_helper, scope) {
797
- self . handle_error ( & mut cx. bcx . config . shell ( ) , & mut error , e) ;
836
+ self . handle_error ( & mut cx. bcx . config . shell ( ) , & mut errors , e) ;
798
837
}
799
838
}
800
839
@@ -805,7 +844,7 @@ impl<'cfg> DrainState<'cfg> {
805
844
}
806
845
807
846
if let Err ( e) = self . grant_rustc_token_requests ( ) {
808
- self . handle_error ( & mut cx. bcx . config . shell ( ) , & mut error , e) ;
847
+ self . handle_error ( & mut cx. bcx . config . shell ( ) , & mut errors , e) ;
809
848
}
810
849
811
850
// And finally, before we block waiting for the next event, drop any
@@ -815,7 +854,7 @@ impl<'cfg> DrainState<'cfg> {
815
854
// to the jobserver itself.
816
855
for event in self . wait_for_events ( ) {
817
856
if let Err ( event_err) = self . handle_event ( cx, jobserver_helper, plan, event) {
818
- self . handle_error ( & mut cx. bcx . config . shell ( ) , & mut error , event_err) ;
857
+ self . handle_error ( & mut cx. bcx . config . shell ( ) , & mut errors , event_err) ;
819
858
}
820
859
}
821
860
}
@@ -840,30 +879,24 @@ impl<'cfg> DrainState<'cfg> {
840
879
}
841
880
842
881
let time_elapsed = util:: elapsed ( cx. bcx . config . creation_time ( ) . elapsed ( ) ) ;
843
- if let Err ( e) = self . timings . finished ( cx, & error) {
844
- if error. is_some ( ) {
845
- crate :: display_error ( & e, & mut cx. bcx . config . shell ( ) ) ;
846
- } else {
847
- return Some ( e) ;
848
- }
882
+ if let Err ( e) = self . timings . finished ( cx, & errors. to_error ( ) ) {
883
+ self . handle_error ( & mut cx. bcx . config . shell ( ) , & mut errors, e) ;
849
884
}
850
885
if cx. bcx . build_config . emit_json ( ) {
851
886
let mut shell = cx. bcx . config . shell ( ) ;
852
887
let msg = machine_message:: BuildFinished {
853
- success : error . is_none ( ) ,
888
+ success : errors . count == 0 ,
854
889
}
855
890
. to_json_string ( ) ;
856
891
if let Err ( e) = writeln ! ( shell. out( ) , "{}" , msg) {
857
- if error. is_some ( ) {
858
- crate :: display_error ( & e. into ( ) , & mut shell) ;
859
- } else {
860
- return Some ( e. into ( ) ) ;
861
- }
892
+ self . handle_error ( & mut shell, & mut errors, e) ;
862
893
}
863
894
}
864
895
865
- if let Some ( e) = error {
866
- Some ( e)
896
+ if let Some ( error) = errors. to_error ( ) {
897
+ // Any errors up to this point have already been printed via the
898
+ // `display_error` inside `handle_error`.
899
+ Some ( anyhow:: Error :: new ( AlreadyPrintedError :: new ( error) ) )
867
900
} else if self . queue . is_empty ( ) && self . pending_queue . is_empty ( ) {
868
901
let message = format ! (
869
902
"{} [{}] target(s) in {}" ,
@@ -888,18 +921,18 @@ impl<'cfg> DrainState<'cfg> {
888
921
fn handle_error (
889
922
& self ,
890
923
shell : & mut Shell ,
891
- err_state : & mut Option < anyhow :: Error > ,
892
- new_err : anyhow :: Error ,
924
+ err_state : & mut ErrorsDuringDrain ,
925
+ new_err : impl Into < ErrorToHandle > ,
893
926
) {
894
- if err_state . is_some ( ) {
895
- // Already encountered one error.
896
- log :: warn! ( "{:?}" , new_err ) ;
897
- } else if !self . active . is_empty ( ) {
898
- crate :: display_error ( & new_err , shell ) ;
899
- drop ( shell . warn ( "build failed, waiting for other jobs to finish..." ) ) ;
900
- * err_state = Some ( anyhow :: format_err! ( "build failed" ) ) ;
927
+ let new_err = new_err . into ( ) ;
928
+ if new_err . print_always || err_state . count == 0 {
929
+ crate :: display_error ( & new_err . error , shell ) ;
930
+ if err_state . count == 0 && !self . active . is_empty ( ) {
931
+ drop ( shell . warn ( "build failed, waiting for other jobs to finish..." ) ) ;
932
+ }
933
+ err_state. count += 1 ;
901
934
} else {
902
- * err_state = Some ( new_err) ;
935
+ log :: warn! ( "{:?}" , new_err. error ) ;
903
936
}
904
937
}
905
938
@@ -1217,3 +1250,13 @@ feature resolver. Try updating to diesel 1.4.8 to fix this error.
1217
1250
Ok ( ( ) )
1218
1251
}
1219
1252
}
1253
+
1254
+ impl ErrorsDuringDrain {
1255
+ fn to_error ( & self ) -> Option < anyhow:: Error > {
1256
+ match self . count {
1257
+ 0 => None ,
1258
+ 1 => Some ( format_err ! ( "1 job failed" ) ) ,
1259
+ n => Some ( format_err ! ( "{} jobs failed" , n) ) ,
1260
+ }
1261
+ }
1262
+ }
0 commit comments