@@ -18,6 +18,7 @@ use rustc_mir_dataflow::impls::MaybeStorageLive;
18
18
use rustc_mir_dataflow:: storage:: always_storage_live_locals;
19
19
use rustc_mir_dataflow:: { Analysis , ResultsCursor } ;
20
20
use rustc_target:: abi:: { Size , FIRST_VARIANT } ;
21
+ use rustc_target:: spec:: abi:: Abi ;
21
22
22
23
#[ derive( Copy , Clone , Debug , PartialEq , Eq ) ]
23
24
enum EdgeKind {
@@ -58,6 +59,25 @@ impl<'tcx> MirPass<'tcx> for Validator {
58
59
. iterate_to_fixpoint ( )
59
60
. into_results_cursor ( body) ;
60
61
62
+ let can_unwind = if mir_phase <= MirPhase :: Runtime ( RuntimePhase :: Initial ) {
63
+ // In this case `AbortUnwindingCalls` haven't yet been executed.
64
+ true
65
+ } else if !tcx. def_kind ( def_id) . is_fn_like ( ) {
66
+ true
67
+ } else {
68
+ let body_ty = tcx. type_of ( def_id) . skip_binder ( ) ;
69
+ let body_abi = match body_ty. kind ( ) {
70
+ ty:: FnDef ( ..) => body_ty. fn_sig ( tcx) . abi ( ) ,
71
+ ty:: Closure ( ..) => Abi :: RustCall ,
72
+ ty:: Generator ( ..) => Abi :: Rust ,
73
+ _ => {
74
+ span_bug ! ( body. span, "unexpected body ty: {:?} phase {:?}" , body_ty, mir_phase)
75
+ }
76
+ } ;
77
+
78
+ ty:: layout:: fn_can_unwind ( tcx, Some ( def_id) , body_abi)
79
+ } ;
80
+
61
81
let mut cfg_checker = CfgChecker {
62
82
when : & self . when ,
63
83
body,
@@ -68,6 +88,7 @@ impl<'tcx> MirPass<'tcx> for Validator {
68
88
storage_liveness,
69
89
place_cache : FxHashSet :: default ( ) ,
70
90
value_cache : FxHashSet :: default ( ) ,
91
+ can_unwind,
71
92
} ;
72
93
cfg_checker. visit_body ( body) ;
73
94
cfg_checker. check_cleanup_control_flow ( ) ;
@@ -99,6 +120,9 @@ struct CfgChecker<'a, 'tcx> {
99
120
storage_liveness : ResultsCursor < ' a , ' tcx , MaybeStorageLive < ' static > > ,
100
121
place_cache : FxHashSet < PlaceRef < ' tcx > > ,
101
122
value_cache : FxHashSet < u128 > ,
123
+ // If `false`, then the MIR must not contain `UnwindAction::Continue` or
124
+ // `TerminatorKind::Resume`.
125
+ can_unwind : bool ,
102
126
}
103
127
104
128
impl < ' a , ' tcx > CfgChecker < ' a , ' tcx > {
@@ -237,13 +261,17 @@ impl<'a, 'tcx> CfgChecker<'a, 'tcx> {
237
261
match unwind {
238
262
UnwindAction :: Cleanup ( unwind) => {
239
263
if is_cleanup {
240
- self . fail ( location, "unwind on cleanup block" ) ;
264
+ self . fail ( location, "`UnwindAction::Cleanup` in cleanup block" ) ;
241
265
}
242
266
self . check_edge ( location, unwind, EdgeKind :: Unwind ) ;
243
267
}
244
268
UnwindAction :: Continue => {
245
269
if is_cleanup {
246
- self . fail ( location, "unwind on cleanup block" ) ;
270
+ self . fail ( location, "`UnwindAction::Continue` in cleanup block" ) ;
271
+ }
272
+
273
+ if !self . can_unwind {
274
+ self . fail ( location, "`UnwindAction::Continue` in no-unwind function" ) ;
247
275
}
248
276
}
249
277
UnwindAction :: Unreachable | UnwindAction :: Terminate => ( ) ,
@@ -464,13 +492,19 @@ impl<'a, 'tcx> Visitor<'tcx> for CfgChecker<'a, 'tcx> {
464
492
) ;
465
493
}
466
494
}
467
- TerminatorKind :: Resume | TerminatorKind :: Terminate => {
495
+ TerminatorKind :: Resume => {
468
496
let bb = location. block ;
469
497
if !self . body . basic_blocks [ bb] . is_cleanup {
470
- self . fail (
471
- location,
472
- "Cannot `Resume` or `Terminate` from non-cleanup basic block" ,
473
- )
498
+ self . fail ( location, "Cannot `Resume` from non-cleanup basic block" )
499
+ }
500
+ if !self . can_unwind {
501
+ self . fail ( location, "Cannot `Resume` in a function that cannot unwind" )
502
+ }
503
+ }
504
+ TerminatorKind :: Terminate => {
505
+ let bb = location. block ;
506
+ if !self . body . basic_blocks [ bb] . is_cleanup {
507
+ self . fail ( location, "Cannot `Terminate` from non-cleanup basic block" )
474
508
}
475
509
}
476
510
TerminatorKind :: Return => {
@@ -665,7 +699,7 @@ impl<'a, 'tcx> Visitor<'tcx> for TypeChecker<'a, 'tcx> {
665
699
return ;
666
700
} ;
667
701
668
- f_ty. ty
702
+ ty :: EarlyBinder :: bind ( f_ty. ty ) . instantiate ( self . tcx , args )
669
703
} else {
670
704
let Some ( & f_ty) = args. as_generator ( ) . prefix_tys ( ) . get ( f. index ( ) )
671
705
else {
0 commit comments