1
+ // Seemingly inconsequential code changes to this file can lead to measurable
2
+ // performance impact on compilation times, due at least in part to the fact
3
+ // that the layout code gets called from many instantiations of the various
4
+ // collections, resulting in having to optimize down excess IR multiple times.
5
+ // Your performance intuition is useless. Run perf.
6
+
1
7
use crate :: cmp;
2
8
use crate :: fmt;
3
9
use crate :: mem:: { self , ValidAlign } ;
@@ -62,6 +68,13 @@ impl Layout {
62
68
return Err ( LayoutError ) ;
63
69
}
64
70
71
+ // SAFETY: just checked that align is a power of two.
72
+ Layout :: from_size_valid_align ( size, unsafe { ValidAlign :: new_unchecked ( align) } )
73
+ }
74
+
75
+ /// Internal helper constructor to skip revalidating alignment validity.
76
+ #[ inline]
77
+ const fn from_size_valid_align ( size : usize , align : ValidAlign ) -> Result < Self , LayoutError > {
65
78
// (power-of-two implies align != 0.)
66
79
67
80
// Rounded up size is:
@@ -76,13 +89,12 @@ impl Layout {
76
89
//
77
90
// Above implies that checking for summation overflow is both
78
91
// necessary and sufficient.
79
- if size > isize:: MAX as usize - ( align - 1 ) {
92
+ if size > isize:: MAX as usize - ( align. as_nonzero ( ) . get ( ) - 1 ) {
80
93
return Err ( LayoutError ) ;
81
94
}
82
95
83
- // SAFETY: the conditions for `from_size_align_unchecked` have been
84
- // checked above.
85
- unsafe { Ok ( Layout :: from_size_align_unchecked ( size, align) ) }
96
+ // SAFETY: Layout::size invariants checked above.
97
+ Ok ( Layout { size, align } )
86
98
}
87
99
88
100
/// Creates a layout, bypassing all checks.
@@ -96,8 +108,8 @@ impl Layout {
96
108
#[ must_use]
97
109
#[ inline]
98
110
pub const unsafe fn from_size_align_unchecked ( size : usize , align : usize ) -> Self {
99
- // SAFETY: the caller must ensure that `align` is a power of two .
100
- Layout { size, align : unsafe { ValidAlign :: new_unchecked ( align) } }
111
+ // SAFETY: the caller is required to uphold the preconditions .
112
+ unsafe { Layout { size, align : ValidAlign :: new_unchecked ( align) } }
101
113
}
102
114
103
115
/// The minimum size in bytes for a memory block of this layout.
@@ -126,10 +138,9 @@ impl Layout {
126
138
#[ inline]
127
139
pub const fn new < T > ( ) -> Self {
128
140
let ( size, align) = size_align :: < T > ( ) ;
129
- // SAFETY: the align is guaranteed by Rust to be a power of two and
130
- // the size+align combo is guaranteed to fit in our address space. As a
131
- // result use the unchecked constructor here to avoid inserting code
132
- // that panics if it isn't optimized well enough.
141
+ // SAFETY: if the type is instantiated, rustc already ensures that its
142
+ // layout is valid. Use the unchecked constructor to avoid inserting a
143
+ // panicking codepath that needs to be optimized out.
133
144
unsafe { Layout :: from_size_align_unchecked ( size, align) }
134
145
}
135
146
@@ -141,7 +152,6 @@ impl Layout {
141
152
#[ inline]
142
153
pub fn for_value < T : ?Sized > ( t : & T ) -> Self {
143
154
let ( size, align) = ( mem:: size_of_val ( t) , mem:: align_of_val ( t) ) ;
144
- debug_assert ! ( Layout :: from_size_align( size, align) . is_ok( ) ) ;
145
155
// SAFETY: see rationale in `new` for why this is using the unsafe variant
146
156
unsafe { Layout :: from_size_align_unchecked ( size, align) }
147
157
}
@@ -176,7 +186,6 @@ impl Layout {
176
186
pub unsafe fn for_value_raw < T : ?Sized > ( t : * const T ) -> Self {
177
187
// SAFETY: we pass along the prerequisites of these functions to the caller
178
188
let ( size, align) = unsafe { ( mem:: size_of_val_raw ( t) , mem:: align_of_val_raw ( t) ) } ;
179
- debug_assert ! ( Layout :: from_size_align( size, align) . is_ok( ) ) ;
180
189
// SAFETY: see rationale in `new` for why this is using the unsafe variant
181
190
unsafe { Layout :: from_size_align_unchecked ( size, align) }
182
191
}
@@ -280,8 +289,7 @@ impl Layout {
280
289
// > less than or equal to `isize::MAX`)
281
290
let new_size = self . size ( ) + pad;
282
291
283
- // SAFETY: self.align is already known to be valid and new_size has been
284
- // padded already.
292
+ // SAFETY: padded size is guaranteed to not exceed `isize::MAX`.
285
293
unsafe { Layout :: from_size_align_unchecked ( new_size, self . align ( ) ) }
286
294
}
287
295
@@ -304,7 +312,7 @@ impl Layout {
304
312
let alloc_size = padded_size. checked_mul ( n) . ok_or ( LayoutError ) ?;
305
313
306
314
// The safe constructor is called here to enforce the isize size limit.
307
- Layout :: from_size_align ( alloc_size, self . align ( ) ) . map ( |layout| ( layout, padded_size) )
315
+ Layout :: from_size_valid_align ( alloc_size, self . align ) . map ( |layout| ( layout, padded_size) )
308
316
}
309
317
310
318
/// Creates a layout describing the record for `self` followed by
@@ -355,14 +363,14 @@ impl Layout {
355
363
#[ stable( feature = "alloc_layout_manipulation" , since = "1.44.0" ) ]
356
364
#[ inline]
357
365
pub fn extend ( & self , next : Self ) -> Result < ( Self , usize ) , LayoutError > {
358
- let new_align = cmp:: max ( self . align ( ) , next. align ( ) ) ;
366
+ let new_align = cmp:: max ( self . align , next. align ) ;
359
367
let pad = self . padding_needed_for ( next. align ( ) ) ;
360
368
361
369
let offset = self . size ( ) . checked_add ( pad) . ok_or ( LayoutError ) ?;
362
370
let new_size = offset. checked_add ( next. size ( ) ) . ok_or ( LayoutError ) ?;
363
371
364
372
// The safe constructor is called here to enforce the isize size limit.
365
- let layout = Layout :: from_size_align ( new_size, new_align) ?;
373
+ let layout = Layout :: from_size_valid_align ( new_size, new_align) ?;
366
374
Ok ( ( layout, offset) )
367
375
}
368
376
@@ -383,7 +391,7 @@ impl Layout {
383
391
pub fn repeat_packed ( & self , n : usize ) -> Result < Self , LayoutError > {
384
392
let size = self . size ( ) . checked_mul ( n) . ok_or ( LayoutError ) ?;
385
393
// The safe constructor is called here to enforce the isize size limit.
386
- Layout :: from_size_align ( size, self . align ( ) )
394
+ Layout :: from_size_valid_align ( size, self . align )
387
395
}
388
396
389
397
/// Creates a layout describing the record for `self` followed by
@@ -397,7 +405,7 @@ impl Layout {
397
405
pub fn extend_packed ( & self , next : Self ) -> Result < Self , LayoutError > {
398
406
let new_size = self . size ( ) . checked_add ( next. size ( ) ) . ok_or ( LayoutError ) ?;
399
407
// The safe constructor is called here to enforce the isize size limit.
400
- Layout :: from_size_align ( new_size, self . align ( ) )
408
+ Layout :: from_size_valid_align ( new_size, self . align )
401
409
}
402
410
403
411
/// Creates a layout describing the record for a `[T; n]`.
@@ -408,7 +416,7 @@ impl Layout {
408
416
pub fn array < T > ( n : usize ) -> Result < Self , LayoutError > {
409
417
let array_size = mem:: size_of :: < T > ( ) . checked_mul ( n) . ok_or ( LayoutError ) ?;
410
418
// The safe constructor is called here to enforce the isize size limit.
411
- Layout :: from_size_align ( array_size, mem :: align_of :: < T > ( ) )
419
+ Layout :: from_size_valid_align ( array_size, ValidAlign :: of :: < T > ( ) )
412
420
}
413
421
}
414
422
0 commit comments