1
- use crate :: convert:: TryFrom ;
1
+ use crate :: convert:: { TryFrom , TryInto } ;
2
2
use crate :: intrinsics:: assert_unsafe_precondition;
3
3
use crate :: num:: NonZeroUsize ;
4
4
use crate :: { cmp, fmt, hash, mem, num} ;
@@ -8,134 +8,178 @@ use crate::{cmp, fmt, hash, mem, num};
8
8
///
9
9
/// Note that particularly large alignments, while representable in this type,
10
10
/// are likely not to be supported by actual allocators and linkers.
11
- #[ derive( Copy , Clone ) ]
11
+ #[ unstable( feature = "ptr_alignment_type" , issue = "102070" ) ]
12
+ #[ derive( Copy , Clone , Eq , PartialEq ) ]
12
13
#[ repr( transparent) ]
13
- pub ( crate ) struct ValidAlign ( ValidAlignEnum ) ;
14
+ pub struct Alignment ( AlignmentEnum ) ;
14
15
15
- // ValidAlign is `repr(usize)`, but via extra steps.
16
- const _: ( ) = assert ! ( mem:: size_of:: <ValidAlign >( ) == mem:: size_of:: <usize >( ) ) ;
17
- const _: ( ) = assert ! ( mem:: align_of:: <ValidAlign >( ) == mem:: align_of:: <usize >( ) ) ;
16
+ // Alignment is `repr(usize)`, but via extra steps.
17
+ const _: ( ) = assert ! ( mem:: size_of:: <Alignment >( ) == mem:: size_of:: <usize >( ) ) ;
18
+ const _: ( ) = assert ! ( mem:: align_of:: <Alignment >( ) == mem:: align_of:: <usize >( ) ) ;
18
19
19
- impl ValidAlign {
20
- /// Creates a `ValidAlign` from a power-of-two `usize`.
20
+ fn _alignment_can_be_structurally_matched ( a : Alignment ) -> bool {
21
+ matches ! ( a, Alignment :: MIN )
22
+ }
23
+
24
+ impl Alignment {
25
+ /// The smallest possible alignment, 1.
26
+ ///
27
+ /// All addresses are always aligned at least this much.
28
+ ///
29
+ /// # Examples
30
+ ///
31
+ /// ```
32
+ /// #![feature(ptr_alignment_type)]
33
+ /// use std::ptr::Alignment;
34
+ ///
35
+ /// assert_eq!(Alignment::MIN.as_usize(), 1);
36
+ /// ```
37
+ #[ unstable( feature = "ptr_alignment_type" , issue = "102070" ) ]
38
+ pub const MIN : Self = Self ( AlignmentEnum :: _Align1Shl0) ;
39
+
40
+ /// Returns the alignment for a type.
41
+ ///
42
+ /// This provides the same numerical value as [`mem::align_of`],
43
+ /// but in an `Alignment` instead of a `usize.
44
+ #[ unstable( feature = "ptr_alignment_type" , issue = "102070" ) ]
45
+ #[ inline]
46
+ pub const fn of < T > ( ) -> Self {
47
+ // SAFETY: rustc ensures that type alignment is always a power of two.
48
+ unsafe { Alignment :: new_unchecked ( mem:: align_of :: < T > ( ) ) }
49
+ }
50
+
51
+ /// Creates an `Alignment` from a `usize`, or returns `None` if it's
52
+ /// not a power of two.
53
+ ///
54
+ /// Note that `0` is not a power of two, nor a valid alignment.
55
+ #[ unstable( feature = "ptr_alignment_type" , issue = "102070" ) ]
56
+ #[ inline]
57
+ pub const fn new ( align : usize ) -> Option < Self > {
58
+ if align. is_power_of_two ( ) {
59
+ // SAFETY: Just checked it only has one bit set
60
+ Some ( unsafe { Self :: new_unchecked ( align) } )
61
+ } else {
62
+ None
63
+ }
64
+ }
65
+
66
+ /// Creates an `Alignment` from a power-of-two `usize`.
21
67
///
22
68
/// # Safety
23
69
///
24
70
/// `align` must be a power of two.
25
71
///
26
72
/// Equivalently, it must be `1 << exp` for some `exp` in `0..usize::BITS`.
27
73
/// It must *not* be zero.
74
+ #[ unstable( feature = "ptr_alignment_type" , issue = "102070" ) ]
75
+ #[ rustc_const_unstable( feature = "ptr_alignment_type" , issue = "102070" ) ]
28
76
#[ inline]
29
- pub ( crate ) const unsafe fn new_unchecked ( align : usize ) -> Self {
77
+ pub const unsafe fn new_unchecked ( align : usize ) -> Self {
30
78
// SAFETY: Precondition passed to the caller.
31
79
unsafe { assert_unsafe_precondition ! ( ( align: usize ) => align. is_power_of_two( ) ) } ;
32
80
33
81
// SAFETY: By precondition, this must be a power of two, and
34
82
// our variants encompass all possible powers of two.
35
- unsafe { mem:: transmute :: < usize , ValidAlign > ( align) }
83
+ unsafe { mem:: transmute :: < usize , Alignment > ( align) }
36
84
}
37
85
86
+ /// Returns the alignment as a [`NonZeroUsize`]
87
+ #[ unstable( feature = "ptr_alignment_type" , issue = "102070" ) ]
88
+ #[ rustc_const_unstable( feature = "ptr_alignment_type" , issue = "102070" ) ]
38
89
#[ inline]
39
- pub ( crate ) const fn as_usize ( self ) -> usize {
90
+ pub const fn as_usize ( self ) -> usize {
40
91
self . 0 as usize
41
92
}
42
93
94
+ /// Returns the alignment as a [`usize`]
95
+ #[ unstable( feature = "ptr_alignment_type" , issue = "102070" ) ]
43
96
#[ inline]
44
- pub ( crate ) const fn as_nonzero ( self ) -> NonZeroUsize {
97
+ pub const fn as_nonzero ( self ) -> NonZeroUsize {
45
98
// SAFETY: All the discriminants are non-zero.
46
99
unsafe { NonZeroUsize :: new_unchecked ( self . as_usize ( ) ) }
47
100
}
48
101
49
- /// Returns the base 2 logarithm of the alignment.
102
+ /// Returns the base- 2 logarithm of the alignment.
50
103
///
51
104
/// This is always exact, as `self` represents a power of two.
105
+ ///
106
+ /// # Examples
107
+ ///
108
+ /// ```
109
+ /// #![feature(ptr_alignment_type)]
110
+ /// use std::ptr::Alignment;
111
+ ///
112
+ /// assert_eq!(Alignment::of::<u8>().log2(), 0);
113
+ /// assert_eq!(Alignment::new(1024).unwrap().log2(), 10);
114
+ /// ```
115
+ #[ unstable( feature = "ptr_alignment_type" , issue = "102070" ) ]
52
116
#[ inline]
53
- pub ( crate ) fn log2 ( self ) -> u32 {
117
+ pub fn log2 ( self ) -> u32 {
54
118
self . as_nonzero ( ) . trailing_zeros ( )
55
119
}
56
-
57
- /// Returns the alignment for a type.
58
- #[ inline]
59
- pub ( crate ) fn of < T > ( ) -> Self {
60
- // SAFETY: rustc ensures that type alignment is always a power of two.
61
- unsafe { ValidAlign :: new_unchecked ( mem:: align_of :: < T > ( ) ) }
62
- }
63
120
}
64
121
65
- impl fmt:: Debug for ValidAlign {
122
+ #[ unstable( feature = "ptr_alignment_type" , issue = "102070" ) ]
123
+ impl fmt:: Debug for Alignment {
66
124
fn fmt ( & self , f : & mut fmt:: Formatter < ' _ > ) -> fmt:: Result {
67
125
write ! ( f, "{:?} (1 << {:?})" , self . as_nonzero( ) , self . log2( ) )
68
126
}
69
127
}
70
128
71
- impl TryFrom < NonZeroUsize > for ValidAlign {
129
+ #[ unstable( feature = "ptr_alignment_type" , issue = "102070" ) ]
130
+ impl TryFrom < NonZeroUsize > for Alignment {
72
131
type Error = num:: TryFromIntError ;
73
132
74
133
#[ inline]
75
- fn try_from ( align : NonZeroUsize ) -> Result < ValidAlign , Self :: Error > {
76
- if align. is_power_of_two ( ) {
77
- // SAFETY: Just checked for power-of-two
78
- unsafe { Ok ( ValidAlign :: new_unchecked ( align. get ( ) ) ) }
79
- } else {
80
- Err ( num:: TryFromIntError ( ( ) ) )
81
- }
134
+ fn try_from ( align : NonZeroUsize ) -> Result < Alignment , Self :: Error > {
135
+ align. get ( ) . try_into ( )
82
136
}
83
137
}
84
138
85
- impl TryFrom < usize > for ValidAlign {
139
+ #[ unstable( feature = "ptr_alignment_type" , issue = "102070" ) ]
140
+ impl TryFrom < usize > for Alignment {
86
141
type Error = num:: TryFromIntError ;
87
142
88
143
#[ inline]
89
- fn try_from ( align : usize ) -> Result < ValidAlign , Self :: Error > {
90
- if align. is_power_of_two ( ) {
91
- // SAFETY: Just checked for power-of-two
92
- unsafe { Ok ( ValidAlign :: new_unchecked ( align) ) }
93
- } else {
94
- Err ( num:: TryFromIntError ( ( ) ) )
95
- }
96
- }
97
- }
98
-
99
- impl cmp:: Eq for ValidAlign { }
100
-
101
- impl cmp:: PartialEq for ValidAlign {
102
- #[ inline]
103
- fn eq ( & self , other : & Self ) -> bool {
104
- self . as_nonzero ( ) == other. as_nonzero ( )
144
+ fn try_from ( align : usize ) -> Result < Alignment , Self :: Error > {
145
+ Self :: new ( align) . ok_or ( num:: TryFromIntError ( ( ) ) )
105
146
}
106
147
}
107
148
108
- impl cmp:: Ord for ValidAlign {
149
+ #[ unstable( feature = "ptr_alignment_type" , issue = "102070" ) ]
150
+ impl cmp:: Ord for Alignment {
109
151
#[ inline]
110
152
fn cmp ( & self , other : & Self ) -> cmp:: Ordering {
111
153
self . as_nonzero ( ) . cmp ( & other. as_nonzero ( ) )
112
154
}
113
155
}
114
156
115
- impl cmp:: PartialOrd for ValidAlign {
157
+ #[ unstable( feature = "ptr_alignment_type" , issue = "102070" ) ]
158
+ impl cmp:: PartialOrd for Alignment {
116
159
#[ inline]
117
160
fn partial_cmp ( & self , other : & Self ) -> Option < cmp:: Ordering > {
118
161
Some ( self . cmp ( other) )
119
162
}
120
163
}
121
164
122
- impl hash:: Hash for ValidAlign {
165
+ #[ unstable( feature = "ptr_alignment_type" , issue = "102070" ) ]
166
+ impl hash:: Hash for Alignment {
123
167
#[ inline]
124
168
fn hash < H : hash:: Hasher > ( & self , state : & mut H ) {
125
169
self . as_nonzero ( ) . hash ( state)
126
170
}
127
171
}
128
172
129
173
#[ cfg( target_pointer_width = "16" ) ]
130
- type ValidAlignEnum = ValidAlignEnum16 ;
174
+ type AlignmentEnum = AlignmentEnum16 ;
131
175
#[ cfg( target_pointer_width = "32" ) ]
132
- type ValidAlignEnum = ValidAlignEnum32 ;
176
+ type AlignmentEnum = AlignmentEnum32 ;
133
177
#[ cfg( target_pointer_width = "64" ) ]
134
- type ValidAlignEnum = ValidAlignEnum64 ;
178
+ type AlignmentEnum = AlignmentEnum64 ;
135
179
136
- #[ derive( Copy , Clone ) ]
180
+ #[ derive( Copy , Clone , Eq , PartialEq ) ]
137
181
#[ repr( u16 ) ]
138
- enum ValidAlignEnum16 {
182
+ enum AlignmentEnum16 {
139
183
_Align1Shl0 = 1 << 0 ,
140
184
_Align1Shl1 = 1 << 1 ,
141
185
_Align1Shl2 = 1 << 2 ,
@@ -154,9 +198,9 @@ enum ValidAlignEnum16 {
154
198
_Align1Shl15 = 1 << 15 ,
155
199
}
156
200
157
- #[ derive( Copy , Clone ) ]
201
+ #[ derive( Copy , Clone , Eq , PartialEq ) ]
158
202
#[ repr( u32 ) ]
159
- enum ValidAlignEnum32 {
203
+ enum AlignmentEnum32 {
160
204
_Align1Shl0 = 1 << 0 ,
161
205
_Align1Shl1 = 1 << 1 ,
162
206
_Align1Shl2 = 1 << 2 ,
@@ -191,9 +235,9 @@ enum ValidAlignEnum32 {
191
235
_Align1Shl31 = 1 << 31 ,
192
236
}
193
237
194
- #[ derive( Copy , Clone ) ]
238
+ #[ derive( Copy , Clone , Eq , PartialEq ) ]
195
239
#[ repr( u64 ) ]
196
- enum ValidAlignEnum64 {
240
+ enum AlignmentEnum64 {
197
241
_Align1Shl0 = 1 << 0 ,
198
242
_Align1Shl1 = 1 << 1 ,
199
243
_Align1Shl2 = 1 << 2 ,
0 commit comments