@@ -26,6 +26,7 @@ use crate::{check_inline, util};
26
26
27
27
pub ( crate ) mod cycle;
28
28
29
+ const HISTORY_DEPTH_LIMIT : usize = 20 ;
29
30
const TOP_DOWN_DEPTH_LIMIT : usize = 5 ;
30
31
31
32
#[ derive( Clone , Debug ) ]
@@ -128,10 +129,6 @@ trait Inliner<'tcx> {
128
129
callee_attrs : & CodegenFnAttrs ,
129
130
) -> Result < ( ) , & ' static str > ;
130
131
131
- // How many callsites in a body are we allowed to inline? We need to limit this in order
132
- // to prevent super-linear growth in MIR size.
133
- fn inline_limit_for_block ( & self ) -> Option < usize > ;
134
-
135
132
/// Called when inlining succeeds.
136
133
fn on_inline_success (
137
134
& mut self ,
@@ -142,9 +139,6 @@ trait Inliner<'tcx> {
142
139
143
140
/// Called when inlining failed or was not performed.
144
141
fn on_inline_failure ( & self , callsite : & CallSite < ' tcx > , reason : & ' static str ) ;
145
-
146
- /// Called when the inline limit for a body is reached.
147
- fn on_inline_limit_reached ( & self ) -> bool ;
148
142
}
149
143
150
144
struct ForceInliner < ' tcx > {
@@ -224,10 +218,6 @@ impl<'tcx> Inliner<'tcx> for ForceInliner<'tcx> {
224
218
}
225
219
}
226
220
227
- fn inline_limit_for_block ( & self ) -> Option < usize > {
228
- Some ( usize:: MAX )
229
- }
230
-
231
221
fn on_inline_success (
232
222
& mut self ,
233
223
callsite : & CallSite < ' tcx > ,
@@ -261,10 +251,6 @@ impl<'tcx> Inliner<'tcx> for ForceInliner<'tcx> {
261
251
justification : justification. map ( |sym| crate :: errors:: ForceInlineJustification { sym } ) ,
262
252
} ) ;
263
253
}
264
-
265
- fn on_inline_limit_reached ( & self ) -> bool {
266
- false
267
- }
268
254
}
269
255
270
256
struct NormalInliner < ' tcx > {
@@ -278,6 +264,10 @@ struct NormalInliner<'tcx> {
278
264
/// The number of `DefId`s is finite, so checking history is enough
279
265
/// to ensure that we do not loop endlessly while inlining.
280
266
history : Vec < DefId > ,
267
+ /// How many (multi-call) callsites have we inlined for the top-level call?
268
+ ///
269
+ /// We need to limit this in order to prevent super-linear growth in MIR size.
270
+ top_down_counter : usize ,
281
271
/// Indicates that the caller body has been modified.
282
272
changed : bool ,
283
273
/// Indicates that the caller is #[inline] and just calls another function,
@@ -295,6 +285,7 @@ impl<'tcx> Inliner<'tcx> for NormalInliner<'tcx> {
295
285
typing_env,
296
286
def_id,
297
287
history : Vec :: new ( ) ,
288
+ top_down_counter : 0 ,
298
289
changed : false ,
299
290
caller_is_inline_forwarder : matches ! (
300
291
codegen_fn_attrs. inline,
@@ -351,6 +342,15 @@ impl<'tcx> Inliner<'tcx> for NormalInliner<'tcx> {
351
342
return Err ( "body has errors" ) ;
352
343
}
353
344
345
+ if callee_body. basic_blocks . len ( ) > 1 {
346
+ if self . history . len ( ) > HISTORY_DEPTH_LIMIT {
347
+ return Err ( "Too many blocks for deep history" ) ;
348
+ }
349
+ if self . top_down_counter > TOP_DOWN_DEPTH_LIMIT {
350
+ return Err ( "Too many blocks for more top-down inlining" ) ;
351
+ }
352
+ }
353
+
354
354
let mut threshold = if self . caller_is_inline_forwarder {
355
355
tcx. sess . opts . unstable_opts . inline_mir_forwarder_threshold . unwrap_or ( 30 )
356
356
} else if tcx. cross_crate_inlinable ( callsite. callee . def_id ( ) ) {
@@ -431,14 +431,6 @@ impl<'tcx> Inliner<'tcx> for NormalInliner<'tcx> {
431
431
}
432
432
}
433
433
434
- fn inline_limit_for_block ( & self ) -> Option < usize > {
435
- match self . history . len ( ) {
436
- 0 => Some ( usize:: MAX ) ,
437
- 1 ..=TOP_DOWN_DEPTH_LIMIT => Some ( 1 ) ,
438
- _ => None ,
439
- }
440
- }
441
-
442
434
fn on_inline_success (
443
435
& mut self ,
444
436
callsite : & CallSite < ' tcx > ,
@@ -447,13 +439,26 @@ impl<'tcx> Inliner<'tcx> for NormalInliner<'tcx> {
447
439
) {
448
440
self . changed = true ;
449
441
442
+ let new_calls_count = new_blocks
443
+ . clone ( )
444
+ . filter ( |& bb| {
445
+ matches ! (
446
+ caller_body. basic_blocks[ bb] . terminator( ) . kind,
447
+ TerminatorKind :: Call { .. } ,
448
+ )
449
+ } )
450
+ . count ( ) ;
451
+ if new_calls_count > 1 {
452
+ self . top_down_counter += 1 ;
453
+ }
454
+
450
455
self . history . push ( callsite. callee . def_id ( ) ) ;
451
456
process_blocks ( self , caller_body, new_blocks) ;
452
457
self . history . pop ( ) ;
453
- }
454
458
455
- fn on_inline_limit_reached ( & self ) -> bool {
456
- true
459
+ if self . history . is_empty ( ) {
460
+ self . top_down_counter = 0 ;
461
+ }
457
462
}
458
463
459
464
fn on_inline_failure ( & self , _: & CallSite < ' tcx > , _: & ' static str ) { }
@@ -482,8 +487,6 @@ fn process_blocks<'tcx, I: Inliner<'tcx>>(
482
487
caller_body : & mut Body < ' tcx > ,
483
488
blocks : Range < BasicBlock > ,
484
489
) {
485
- let Some ( inline_limit) = inliner. inline_limit_for_block ( ) else { return } ;
486
- let mut inlined_count = 0 ;
487
490
for bb in blocks {
488
491
let bb_data = & caller_body[ bb] ;
489
492
if bb_data. is_cleanup {
@@ -505,13 +508,6 @@ fn process_blocks<'tcx, I: Inliner<'tcx>>(
505
508
Ok ( new_blocks) => {
506
509
debug ! ( "inlined {}" , callsite. callee) ;
507
510
inliner. on_inline_success ( & callsite, caller_body, new_blocks) ;
508
-
509
- inlined_count += 1 ;
510
- if inlined_count == inline_limit {
511
- if inliner. on_inline_limit_reached ( ) {
512
- return ;
513
- }
514
- }
515
511
}
516
512
}
517
513
}
0 commit comments