@@ -361,9 +361,12 @@ pub fn report_error<'tcx, 'mir>(
361
361
} ;
362
362
363
363
let stacktrace = ecx. generate_stacktrace ( ) ;
364
- let ( stacktrace, was_pruned ) = prune_stacktrace ( stacktrace, & ecx. machine ) ;
364
+ let ( stacktrace, mut any_pruned ) = prune_stacktrace ( stacktrace, & ecx. machine ) ;
365
365
366
- // We want to dump the allocation if this is `InvalidUninitBytes`. Since `format_error` consumes `e`, we compute the outut early.
366
+ let mut show_all_threads = false ;
367
+
368
+ // We want to dump the allocation if this is `InvalidUninitBytes`.
369
+ // Since `format_interp_error` consumes `e`, we compute the outut early.
367
370
let mut extra = String :: new ( ) ;
368
371
match e. kind ( ) {
369
372
UndefinedBehavior ( InvalidUninitBytes ( Some ( ( alloc_id, access) ) ) ) => {
@@ -375,6 +378,15 @@ pub fn report_error<'tcx, 'mir>(
375
378
. unwrap ( ) ;
376
379
writeln ! ( extra, "{:?}" , ecx. dump_alloc( * alloc_id) ) . unwrap ( ) ;
377
380
}
381
+ MachineStop ( info) => {
382
+ let info = info. downcast_ref :: < TerminationInfo > ( ) . expect ( "invalid MachineStop payload" ) ;
383
+ match info {
384
+ TerminationInfo :: Deadlock => {
385
+ show_all_threads = true ;
386
+ }
387
+ _ => { }
388
+ }
389
+ }
378
390
_ => { }
379
391
}
380
392
@@ -387,18 +399,39 @@ pub fn report_error<'tcx, 'mir>(
387
399
vec ! [ ] ,
388
400
helps,
389
401
& stacktrace,
402
+ Some ( ecx. get_active_thread ( ) ) ,
390
403
& ecx. machine ,
391
404
) ;
392
405
406
+ eprint ! ( "{extra}" ) ; // newlines are already in the string
407
+
408
+ if show_all_threads {
409
+ for ( thread, stack) in ecx. machine . threads . all_stacks ( ) {
410
+ if thread != ecx. get_active_thread ( ) {
411
+ let stacktrace = Frame :: generate_stacktrace_from_stack ( stack) ;
412
+ let ( stacktrace, was_pruned) = prune_stacktrace ( stacktrace, & ecx. machine ) ;
413
+ any_pruned |= was_pruned;
414
+ report_msg (
415
+ DiagLevel :: Error ,
416
+ format ! ( "deadlock: the evaluated program deadlocked" ) ,
417
+ vec ! [ format!( "the evaluated program deadlocked" ) ] ,
418
+ vec ! [ ] ,
419
+ vec ! [ ] ,
420
+ & stacktrace,
421
+ Some ( thread) ,
422
+ & ecx. machine ,
423
+ )
424
+ }
425
+ }
426
+ }
427
+
393
428
// Include a note like `std` does when we omit frames from a backtrace
394
- if was_pruned {
429
+ if any_pruned {
395
430
ecx. tcx . dcx ( ) . note (
396
431
"some details are omitted, run with `MIRIFLAGS=-Zmiri-backtrace=full` for a verbose backtrace" ,
397
432
) ;
398
433
}
399
434
400
- eprint ! ( "{extra}" ) ; // newlines are already in the string
401
-
402
435
// Debug-dump all locals.
403
436
for ( i, frame) in ecx. active_thread_stack ( ) . iter ( ) . enumerate ( ) {
404
437
trace ! ( "-------------------" ) ;
@@ -435,6 +468,7 @@ pub fn report_leaks<'mir, 'tcx>(
435
468
vec ! [ ] ,
436
469
vec ! [ ] ,
437
470
& backtrace,
471
+ None , // we don't know the thread this is from
438
472
& ecx. machine ,
439
473
) ;
440
474
}
@@ -457,6 +491,7 @@ pub fn report_msg<'tcx>(
457
491
notes : Vec < ( Option < SpanData > , String ) > ,
458
492
helps : Vec < ( Option < SpanData > , String ) > ,
459
493
stacktrace : & [ FrameInfo < ' tcx > ] ,
494
+ thread : Option < ThreadId > ,
460
495
machine : & MiriMachine < ' _ , ' tcx > ,
461
496
) {
462
497
let span = stacktrace. first ( ) . map_or ( DUMMY_SP , |fi| fi. span ) ;
@@ -506,12 +541,13 @@ pub fn report_msg<'tcx>(
506
541
if extra_span {
507
542
write ! ( backtrace_title, " (of the first span)" ) . unwrap ( ) ;
508
543
}
509
- let thread_name =
510
- machine. threads . get_thread_display_name ( machine. threads . get_active_thread_id ( ) ) ;
511
- if thread_name != "main" {
512
- // Only print thread name if it is not `main`.
513
- write ! ( backtrace_title, " on thread `{thread_name}`" ) . unwrap ( ) ;
514
- } ;
544
+ if let Some ( thread) = thread {
545
+ let thread_name = machine. threads . get_thread_display_name ( thread) ;
546
+ if thread_name != "main" {
547
+ // Only print thread name if it is not `main`.
548
+ write ! ( backtrace_title, " on thread `{thread_name}`" ) . unwrap ( ) ;
549
+ } ;
550
+ }
515
551
write ! ( backtrace_title, ":" ) . unwrap ( ) ;
516
552
err. note ( backtrace_title) ;
517
553
for ( idx, frame_info) in stacktrace. iter ( ) . enumerate ( ) {
@@ -628,7 +664,16 @@ impl<'mir, 'tcx> MiriMachine<'mir, 'tcx> {
628
664
_ => vec ! [ ] ,
629
665
} ;
630
666
631
- report_msg ( diag_level, title, vec ! [ msg] , notes, helps, & stacktrace, self ) ;
667
+ report_msg (
668
+ diag_level,
669
+ title,
670
+ vec ! [ msg] ,
671
+ notes,
672
+ helps,
673
+ & stacktrace,
674
+ Some ( self . threads . get_active_thread_id ( ) ) ,
675
+ self ,
676
+ ) ;
632
677
}
633
678
}
634
679
@@ -654,6 +699,7 @@ pub trait EvalContextExt<'mir, 'tcx: 'mir>: crate::MiriInterpCxExt<'mir, 'tcx> {
654
699
vec ! [ ] ,
655
700
vec ! [ ] ,
656
701
& stacktrace,
702
+ Some ( this. get_active_thread ( ) ) ,
657
703
& this. machine ,
658
704
) ;
659
705
}
0 commit comments