Skip to content

Commit add6093

Browse files
committed
constify comparison traits on slices
1 parent 8c88ae6 commit add6093

File tree

1 file changed

+92
-43
lines changed

1 file changed

+92
-43
lines changed

library/core/src/slice/cmp.rs

Lines changed: 92 additions & 43 deletions
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,9 @@
33
use super::{from_raw_parts, memchr};
44
use crate::ascii;
55
use crate::cmp::{self, BytewiseEq, Ordering};
6+
use crate::convert::Infallible;
67
use crate::intrinsics::compare_bytes;
8+
use crate::marker::Destruct;
79
use crate::num::NonZero;
810
use crate::ops::ControlFlow;
911

@@ -23,18 +25,20 @@ where
2325
}
2426

2527
#[stable(feature = "rust1", since = "1.0.0")]
26-
impl<T: Eq> Eq for [T] {}
28+
#[rustc_const_unstable(feature = "const_cmp", issue = "143800")]
29+
impl<T: [const] Eq> const Eq for [T] {}
2730

2831
/// Implements comparison of slices [lexicographically](Ord#lexicographical-comparison).
2932
#[stable(feature = "rust1", since = "1.0.0")]
30-
impl<T: Ord> Ord for [T] {
33+
#[rustc_const_unstable(feature = "const_cmp", issue = "143800")]
34+
impl<T: [const] Ord> const Ord for [T] {
3135
fn cmp(&self, other: &[T]) -> Ordering {
3236
SliceOrd::compare(self, other)
3337
}
3438
}
3539

3640
#[inline]
37-
fn as_underlying(x: ControlFlow<bool>) -> u8 {
41+
const fn as_underlying(x: ControlFlow<bool>) -> u8 {
3842
// SAFETY: This will only compile if `bool` and `ControlFlow<bool>` have the same
3943
// size (which isn't guaranteed but this is libcore). Because they have the same
4044
// size, it's a niched implementation, which in one byte means there can't be
@@ -46,7 +50,8 @@ fn as_underlying(x: ControlFlow<bool>) -> u8 {
4650

4751
/// Implements comparison of slices [lexicographically](Ord#lexicographical-comparison).
4852
#[stable(feature = "rust1", since = "1.0.0")]
49-
impl<T: PartialOrd> PartialOrd for [T] {
53+
#[rustc_const_unstable(feature = "const_cmp", issue = "143800")]
54+
impl<T: [const] PartialOrd> const PartialOrd for [T] {
5055
#[inline]
5156
fn partial_cmp(&self, other: &[T]) -> Option<Ordering> {
5257
SlicePartialOrd::partial_compare(self, other)
@@ -154,35 +159,50 @@ where
154159
}
155160

156161
#[doc(hidden)]
162+
#[rustc_const_unstable(feature = "const_cmp", issue = "143800")]
157163
// intermediate trait for specialization of slice's PartialOrd
158-
trait SlicePartialOrd: Sized {
164+
const trait SlicePartialOrd: Sized {
159165
fn partial_compare(left: &[Self], right: &[Self]) -> Option<Ordering>;
160166
}
161167

162168
#[doc(hidden)]
169+
#[rustc_const_unstable(feature = "const_cmp", issue = "143800")]
163170
// intermediate trait for specialization of slice's PartialOrd chaining methods
164-
trait SliceChain: Sized {
171+
const trait SliceChain: Sized {
165172
fn chaining_lt(left: &[Self], right: &[Self]) -> ControlFlow<bool>;
166173
fn chaining_le(left: &[Self], right: &[Self]) -> ControlFlow<bool>;
167174
fn chaining_gt(left: &[Self], right: &[Self]) -> ControlFlow<bool>;
168175
fn chaining_ge(left: &[Self], right: &[Self]) -> ControlFlow<bool>;
169176
}
170177

171-
type AlwaysBreak<B> = ControlFlow<B, crate::convert::Infallible>;
178+
type AlwaysBreak<B> = ControlFlow<B, Infallible>;
172179

173-
impl<A: PartialOrd> SlicePartialOrd for A {
180+
#[rustc_const_unstable(feature = "const_cmp", issue = "143800")]
181+
const fn elem_chain<B>(a: &B, b: &B) -> ControlFlow<Option<Ordering>>
182+
where
183+
B: [const] PartialOrd,
184+
{
185+
match a.partial_cmp(b) {
186+
Some(Ordering::Equal) => ControlFlow::Continue(()),
187+
non_eq => ControlFlow::Break(non_eq),
188+
}
189+
}
190+
191+
#[rustc_const_unstable(feature = "const_cmp", issue = "143800")]
192+
const fn len_chain(a: &usize, b: &usize) -> ControlFlow<Option<Ordering>, Infallible> {
193+
ControlFlow::Break(usize::partial_cmp(a, b))
194+
}
195+
196+
#[rustc_const_unstable(feature = "const_cmp", issue = "143800")]
197+
impl<A: [const] PartialOrd> const SlicePartialOrd for A {
174198
default fn partial_compare(left: &[A], right: &[A]) -> Option<Ordering> {
175-
let elem_chain = |a, b| match PartialOrd::partial_cmp(a, b) {
176-
Some(Ordering::Equal) => ControlFlow::Continue(()),
177-
non_eq => ControlFlow::Break(non_eq),
178-
};
179-
let len_chain = |a: &_, b: &_| ControlFlow::Break(usize::partial_cmp(a, b));
180199
let AlwaysBreak::Break(b) = chaining_impl(left, right, elem_chain, len_chain);
181200
b
182201
}
183202
}
184203

185-
impl<A: PartialOrd> SliceChain for A {
204+
#[rustc_const_unstable(feature = "const_cmp", issue = "143800")]
205+
impl<A: [const] PartialOrd> const SliceChain for A {
186206
default fn chaining_lt(left: &[Self], right: &[Self]) -> ControlFlow<bool> {
187207
chaining_impl(left, right, PartialOrd::__chaining_lt, usize::__chaining_lt)
188208
}
@@ -198,21 +218,28 @@ impl<A: PartialOrd> SliceChain for A {
198218
}
199219

200220
#[inline]
201-
fn chaining_impl<'l, 'r, A: PartialOrd, B, C>(
221+
#[rustc_const_unstable(feature = "const_cmp", issue = "143800")]
222+
const fn chaining_impl<'l, 'r, A: PartialOrd, B, C, X, Y>(
202223
left: &'l [A],
203224
right: &'r [A],
204-
elem_chain: impl Fn(&'l A, &'r A) -> ControlFlow<B>,
205-
len_chain: impl for<'a> FnOnce(&'a usize, &'a usize) -> ControlFlow<B, C>,
206-
) -> ControlFlow<B, C> {
225+
elem_chain: X,
226+
len_chain: Y,
227+
) -> ControlFlow<B, C>
228+
where
229+
X: [const] Destruct + [const] Fn(&'l A, &'r A) -> ControlFlow<B>,
230+
Y: [const] Destruct + for<'a> [const] FnOnce(&'a usize, &'a usize) -> ControlFlow<B, C>,
231+
{
207232
let l = cmp::min(left.len(), right.len());
208233

209234
// Slice to the loop iteration range to enable bound check
210235
// elimination in the compiler
211236
let lhs = &left[..l];
212237
let rhs = &right[..l];
213238

214-
for i in 0..l {
239+
let mut i = 0;
240+
while i < l {
215241
elem_chain(&lhs[i], &rhs[i])?;
242+
i += 1;
216243
}
217244

218245
len_chain(&left.len(), &right.len())
@@ -231,46 +258,65 @@ where
231258
}
232259
*/
233260

234-
impl<A: AlwaysApplicableOrd> SlicePartialOrd for A {
261+
#[rustc_const_unstable(feature = "const_cmp", issue = "143800")]
262+
impl<A: [const] AlwaysApplicableOrd> const SlicePartialOrd for A {
235263
fn partial_compare(left: &[A], right: &[A]) -> Option<Ordering> {
236264
Some(SliceOrd::compare(left, right))
237265
}
238266
}
239267

240268
#[rustc_specialization_trait]
241-
trait AlwaysApplicableOrd: SliceOrd + Ord {}
269+
#[rustc_const_unstable(feature = "const_cmp", issue = "143800")]
270+
const trait AlwaysApplicableOrd: [const] SliceOrd + [const] Ord {}
242271

243272
macro_rules! always_applicable_ord {
244273
($([$($p:tt)*] $t:ty,)*) => {
245-
$(impl<$($p)*> AlwaysApplicableOrd for $t {})*
274+
$(
275+
#[rustc_const_unstable(feature = "const_cmp", issue = "143800")]
276+
impl<$($p)*> const AlwaysApplicableOrd for $t {}
277+
)*
246278
}
247279
}
248280

249281
always_applicable_ord! {
250282
[] u8, [] u16, [] u32, [] u64, [] u128, [] usize,
251283
[] i8, [] i16, [] i32, [] i64, [] i128, [] isize,
252284
[] bool, [] char,
253-
[T: ?Sized] *const T, [T: ?Sized] *mut T,
254-
[T: AlwaysApplicableOrd] &T,
255-
[T: AlwaysApplicableOrd] &mut T,
256-
[T: AlwaysApplicableOrd] Option<T>,
285+
[T: [const] AlwaysApplicableOrd] &T,
286+
[T: [const] AlwaysApplicableOrd] &mut T,
287+
[T: [const] AlwaysApplicableOrd] Option<T>,
257288
}
258289

290+
impl<T: ?Sized> AlwaysApplicableOrd for *const T {}
291+
impl<T: ?Sized> AlwaysApplicableOrd for *mut T {}
292+
259293
#[doc(hidden)]
260294
// intermediate trait for specialization of slice's Ord
261-
trait SliceOrd: Sized {
295+
#[rustc_const_unstable(feature = "const_cmp", issue = "143800")]
296+
const trait SliceOrd: Sized {
262297
fn compare(left: &[Self], right: &[Self]) -> Ordering;
263298
}
264299

265-
impl<A: Ord> SliceOrd for A {
300+
#[rustc_const_unstable(feature = "const_cmp", issue = "143800")]
301+
impl<A: [const] Ord> const SliceOrd for A {
266302
default fn compare(left: &[Self], right: &[Self]) -> Ordering {
267-
let elem_chain = |a, b| match Ord::cmp(a, b) {
268-
Ordering::Equal => ControlFlow::Continue(()),
269-
non_eq => ControlFlow::Break(non_eq),
270-
};
271-
let len_chain = |a: &_, b: &_| ControlFlow::Break(usize::cmp(a, b));
272-
let AlwaysBreak::Break(b) = chaining_impl(left, right, elem_chain, len_chain);
273-
b
303+
let l = cmp::min(left.len(), right.len());
304+
305+
// Slice to the loop iteration range to enable bound check
306+
// elimination in the compiler
307+
let lhs = &left[..l];
308+
let rhs = &right[..l];
309+
310+
let mut i = 0;
311+
while i < l {
312+
match Ord::cmp(&lhs[i], &rhs[i]) {
313+
Ordering::Equal => {}
314+
non_eq => return non_eq,
315+
}
316+
i += 1;
317+
}
318+
319+
usize::cmp(&left.len(), &right.len())
274320
}
275321
}
276322

@@ -282,17 +328,19 @@ impl<A: Ord> SliceOrd for A {
282328
/// * For every `x` and `y` of this type, `Ord(x, y)` must return the same
283329
/// value as `Ord::cmp(transmute::<_, u8>(x), transmute::<_, u8>(y))`.
284330
#[rustc_specialization_trait]
285-
unsafe trait UnsignedBytewiseOrd: Ord {}
331+
#[const_trait]
332+
unsafe trait UnsignedBytewiseOrd: [const] Ord {}
286333

287-
unsafe impl UnsignedBytewiseOrd for bool {}
288-
unsafe impl UnsignedBytewiseOrd for u8 {}
289-
unsafe impl UnsignedBytewiseOrd for NonZero<u8> {}
290-
unsafe impl UnsignedBytewiseOrd for Option<NonZero<u8>> {}
291-
unsafe impl UnsignedBytewiseOrd for ascii::Char {}
334+
unsafe impl const UnsignedBytewiseOrd for bool {}
335+
unsafe impl const UnsignedBytewiseOrd for u8 {}
336+
unsafe impl const UnsignedBytewiseOrd for NonZero<u8> {}
337+
unsafe impl const UnsignedBytewiseOrd for Option<NonZero<u8>> {}
338+
unsafe impl const UnsignedBytewiseOrd for ascii::Char {}
292339

293340
// `compare_bytes` compares a sequence of unsigned bytes lexicographically, so
294341
// use it if the requirements for `UnsignedBytewiseOrd` are fulfilled.
295-
impl<A: Ord + UnsignedBytewiseOrd> SliceOrd for A {
342+
#[rustc_const_unstable(feature = "const_cmp", issue = "143800")]
343+
impl<A: [const] Ord + [const] UnsignedBytewiseOrd> const SliceOrd for A {
296344
#[inline]
297345
fn compare(left: &[Self], right: &[Self]) -> Ordering {
298346
// Since the length of a slice is always less than or equal to
@@ -317,7 +365,8 @@ impl<A: Ord + UnsignedBytewiseOrd> SliceOrd for A {
317365
}
318366

319367
// Don't generate our own chaining loops for `memcmp`-able things either.
320-
impl<A: PartialOrd + UnsignedBytewiseOrd> SliceChain for A {
368+
#[rustc_const_unstable(feature = "const_cmp", issue = "143800")]
369+
impl<A: [const] PartialOrd + [const] UnsignedBytewiseOrd> const SliceChain for A {
321370
#[inline]
322371
fn chaining_lt(left: &[Self], right: &[Self]) -> ControlFlow<bool> {
323372
match SliceOrd::compare(left, right) {

0 commit comments

Comments
 (0)