From f530a29944ff1eba9a146704ba2f13b94331be5d Mon Sep 17 00:00:00 2001
From: Peter Jaszkowiak
Date: Mon, 27 Jan 2025 13:32:10 -0700
Subject: [PATCH] implement unstable `new_range` feature
for RFC 3550, tracking issue #123741
---
compiler/rustc_ast_lowering/src/expr.rs | 35 +++++++++++---
compiler/rustc_feature/src/unstable.rs | 2 +
compiler/rustc_hir/src/hir.rs | 41 +++++++++++++++-
compiler/rustc_hir/src/lang_items.rs | 5 ++
.../rustc_hir_typeck/src/method/suggest.rs | 4 ++
compiler/rustc_span/src/symbol.rs | 4 ++
library/core/src/range.rs | 3 ++
.../src/language-features/new-range.md | 9 ++++
.../feature-gates/feature-gate-new_range.rs | 10 ++++
.../feature-gate-new_range.stderr | 48 +++++++++++++++++++
...rg-type-mismatch-issue-45727.current.fixed | 2 +-
...-arg-type-mismatch-issue-45727.next.stderr | 8 ++--
.../closure-arg-type-mismatch-issue-45727.rs | 2 +-
tests/ui/new-range/disabled.rs | 27 +++++++++++
tests/ui/new-range/enabled.rs | 24 ++++++++++
15 files changed, 210 insertions(+), 14 deletions(-)
create mode 100644 src/doc/unstable-book/src/language-features/new-range.md
create mode 100644 tests/ui/feature-gates/feature-gate-new_range.rs
create mode 100644 tests/ui/feature-gates/feature-gate-new_range.stderr
create mode 100644 tests/ui/new-range/disabled.rs
create mode 100644 tests/ui/new-range/enabled.rs
diff --git a/compiler/rustc_ast_lowering/src/expr.rs b/compiler/rustc_ast_lowering/src/expr.rs
index 1267281f73eb..98d6372ca896 100644
--- a/compiler/rustc_ast_lowering/src/expr.rs
+++ b/compiler/rustc_ast_lowering/src/expr.rs
@@ -284,9 +284,6 @@ pub(super) fn lower_expr_mut(&mut self, e: &Expr) -> hir::Expr<'hir> {
ExprKind::Index(el, er, brackets_span) => {
hir::ExprKind::Index(self.lower_expr(el), self.lower_expr(er), *brackets_span)
}
- ExprKind::Range(Some(e1), Some(e2), RangeLimits::Closed) => {
- self.lower_expr_range_closed(e.span, e1, e2)
- }
ExprKind::Range(e1, e2, lims) => {
self.lower_expr_range(e.span, e1.as_deref(), e2.as_deref(), *lims)
}
@@ -1512,15 +1509,39 @@ fn lower_expr_range(
let lang_item = match (e1, e2, lims) {
(None, None, HalfOpen) => hir::LangItem::RangeFull,
- (Some(..), None, HalfOpen) => hir::LangItem::RangeFrom,
+ (Some(..), None, HalfOpen) => {
+ if self.tcx.features().new_range() {
+ hir::LangItem::RangeFromCopy
+ } else {
+ hir::LangItem::RangeFrom
+ }
+ }
(None, Some(..), HalfOpen) => hir::LangItem::RangeTo,
- (Some(..), Some(..), HalfOpen) => hir::LangItem::Range,
+ (Some(..), Some(..), HalfOpen) => {
+ if self.tcx.features().new_range() {
+ hir::LangItem::RangeCopy
+ } else {
+ hir::LangItem::Range
+ }
+ }
(None, Some(..), Closed) => hir::LangItem::RangeToInclusive,
- (Some(..), Some(..), Closed) => unreachable!(),
+ (Some(e1), Some(e2), Closed) => {
+ if self.tcx.features().new_range() {
+ hir::LangItem::RangeInclusiveCopy
+ } else {
+ return self.lower_expr_range_closed(span, e1, e2);
+ }
+ }
(start, None, Closed) => {
self.dcx().emit_err(InclusiveRangeWithNoEnd { span });
match start {
- Some(..) => hir::LangItem::RangeFrom,
+ Some(..) => {
+ if self.tcx.features().new_range() {
+ hir::LangItem::RangeFromCopy
+ } else {
+ hir::LangItem::RangeFrom
+ }
+ }
None => hir::LangItem::RangeFull,
}
}
diff --git a/compiler/rustc_feature/src/unstable.rs b/compiler/rustc_feature/src/unstable.rs
index 1a216ebf117c..7a30c8d4737b 100644
--- a/compiler/rustc_feature/src/unstable.rs
+++ b/compiler/rustc_feature/src/unstable.rs
@@ -568,6 +568,8 @@ pub fn internal(&self, feature: Symbol) -> bool {
(unstable, never_type, "1.13.0", Some(35121)),
/// Allows diverging expressions to fall back to `!` rather than `()`.
(unstable, never_type_fallback, "1.41.0", Some(65992)),
+ /// Switch `..` syntax to use the new (`Copy + IntoIterator`) range types.
+ (unstable, new_range, "CURRENT_RUSTC_VERSION", Some(123741)),
/// Allows `#![no_core]`.
(unstable, no_core, "1.3.0", Some(29639)),
/// Allows the use of `no_sanitize` attribute.
diff --git a/compiler/rustc_hir/src/hir.rs b/compiler/rustc_hir/src/hir.rs
index af2f86b67e00..5483b97a7f4d 100644
--- a/compiler/rustc_hir/src/hir.rs
+++ b/compiler/rustc_hir/src/hir.rs
@@ -2313,6 +2313,18 @@ pub fn equivalent_for_indexing(&self, other: &Expr<'_>) -> bool {
[val2],
StructTailExpr::None,
),
+ )
+ | (
+ ExprKind::Struct(
+ QPath::LangItem(LangItem::RangeFromCopy, _),
+ [val1],
+ StructTailExpr::None,
+ ),
+ ExprKind::Struct(
+ QPath::LangItem(LangItem::RangeFromCopy, _),
+ [val2],
+ StructTailExpr::None,
+ ),
) => val1.expr.equivalent_for_indexing(val2.expr),
(
ExprKind::Struct(
@@ -2325,6 +2337,30 @@ pub fn equivalent_for_indexing(&self, other: &Expr<'_>) -> bool {
[val2, val4],
StructTailExpr::None,
),
+ )
+ | (
+ ExprKind::Struct(
+ QPath::LangItem(LangItem::RangeCopy, _),
+ [val1, val3],
+ StructTailExpr::None,
+ ),
+ ExprKind::Struct(
+ QPath::LangItem(LangItem::RangeCopy, _),
+ [val2, val4],
+ StructTailExpr::None,
+ ),
+ )
+ | (
+ ExprKind::Struct(
+ QPath::LangItem(LangItem::RangeInclusiveCopy, _),
+ [val1, val3],
+ StructTailExpr::None,
+ ),
+ ExprKind::Struct(
+ QPath::LangItem(LangItem::RangeInclusiveCopy, _),
+ [val2, val4],
+ StructTailExpr::None,
+ ),
) => {
val1.expr.equivalent_for_indexing(val2.expr)
&& val3.expr.equivalent_for_indexing(val4.expr)
@@ -2354,7 +2390,10 @@ pub fn is_range_literal(expr: &Expr<'_>) -> bool {
| LangItem::RangeTo
| LangItem::RangeFrom
| LangItem::RangeFull
- | LangItem::RangeToInclusive,
+ | LangItem::RangeToInclusive
+ | LangItem::RangeCopy
+ | LangItem::RangeFromCopy
+ | LangItem::RangeInclusiveCopy,
..
)
),
diff --git a/compiler/rustc_hir/src/lang_items.rs b/compiler/rustc_hir/src/lang_items.rs
index 02bc069fc5f2..3edf1370d2b4 100644
--- a/compiler/rustc_hir/src/lang_items.rs
+++ b/compiler/rustc_hir/src/lang_items.rs
@@ -415,6 +415,11 @@ pub fn extract(attrs: &[impl AttributeExt]) -> Option<(Symbol, Span)> {
RangeToInclusive, sym::RangeToInclusive, range_to_inclusive_struct, Target::Struct, GenericRequirement::None;
RangeTo, sym::RangeTo, range_to_struct, Target::Struct, GenericRequirement::None;
+ // `new_range` types that are `Copy + IntoIterator`
+ RangeFromCopy, sym::RangeFromCopy, range_from_copy_struct, Target::Struct, GenericRequirement::None;
+ RangeCopy, sym::RangeCopy, range_copy_struct, Target::Struct, GenericRequirement::None;
+ RangeInclusiveCopy, sym::RangeInclusiveCopy, range_inclusive_copy_struct, Target::Struct, GenericRequirement::None;
+
String, sym::String, string, Target::Struct, GenericRequirement::None;
CStr, sym::CStr, c_str, Target::Struct, GenericRequirement::None;
}
diff --git a/compiler/rustc_hir_typeck/src/method/suggest.rs b/compiler/rustc_hir_typeck/src/method/suggest.rs
index e4a6a0fedc5a..fb48f5dca894 100644
--- a/compiler/rustc_hir_typeck/src/method/suggest.rs
+++ b/compiler/rustc_hir_typeck/src/method/suggest.rs
@@ -2410,6 +2410,10 @@ fn report_failed_method_call_on_range_end(
let lang_item = match parent_expr.kind {
ExprKind::Struct(qpath, _, _) => match *qpath {
QPath::LangItem(LangItem::Range, ..) => Some(LangItem::Range),
+ QPath::LangItem(LangItem::RangeCopy, ..) => Some(LangItem::RangeCopy),
+ QPath::LangItem(LangItem::RangeInclusiveCopy, ..) => {
+ Some(LangItem::RangeInclusiveCopy)
+ }
QPath::LangItem(LangItem::RangeTo, ..) => Some(LangItem::RangeTo),
QPath::LangItem(LangItem::RangeToInclusive, ..) => {
Some(LangItem::RangeToInclusive)
diff --git a/compiler/rustc_span/src/symbol.rs b/compiler/rustc_span/src/symbol.rs
index 6f1d3a74a816..dca96b74174d 100644
--- a/compiler/rustc_span/src/symbol.rs
+++ b/compiler/rustc_span/src/symbol.rs
@@ -294,9 +294,12 @@
ProceduralMasqueradeDummyType,
Range,
RangeBounds,
+ RangeCopy,
RangeFrom,
+ RangeFromCopy,
RangeFull,
RangeInclusive,
+ RangeInclusiveCopy,
RangeTo,
RangeToInclusive,
Rc,
@@ -1367,6 +1370,7 @@
new_lower_hex,
new_octal,
new_pointer,
+ new_range,
new_unchecked,
new_upper_exp,
new_upper_hex,
diff --git a/library/core/src/range.rs b/library/core/src/range.rs
index 427526fd14b9..6a62928873fe 100644
--- a/library/core/src/range.rs
+++ b/library/core/src/range.rs
@@ -48,6 +48,7 @@
/// assert_eq!(Range::from(3..5), Range { start: 3, end: 5 });
/// assert_eq!(3 + 4 + 5, Range::from(3..6).into_iter().sum());
/// ```
+#[cfg_attr(not(bootstrap), lang = "RangeCopy")]
#[derive(Clone, Copy, Default, PartialEq, Eq, Hash)]
#[unstable(feature = "new_range_api", issue = "125687")]
pub struct Range {
@@ -205,6 +206,7 @@ fn from(value: legacy::Range) -> Self {
/// assert_eq!(RangeInclusive::from(3..=5), RangeInclusive { start: 3, end: 5 });
/// assert_eq!(3 + 4 + 5, RangeInclusive::from(3..=5).into_iter().sum());
/// ```
+#[cfg_attr(not(bootstrap), lang = "RangeInclusiveCopy")]
#[derive(Clone, Copy, PartialEq, Eq, Hash)]
#[unstable(feature = "new_range_api", issue = "125687")]
pub struct RangeInclusive {
@@ -388,6 +390,7 @@ fn from(value: legacy::RangeInclusive) -> Self {
/// assert_eq!(RangeFrom::from(2..), core::range::RangeFrom { start: 2 });
/// assert_eq!(2 + 3 + 4, RangeFrom::from(2..).into_iter().take(3).sum());
/// ```
+#[cfg_attr(not(bootstrap), lang = "RangeFromCopy")]
#[derive(Clone, Copy, PartialEq, Eq, Hash)]
#[unstable(feature = "new_range_api", issue = "125687")]
pub struct RangeFrom {
diff --git a/src/doc/unstable-book/src/language-features/new-range.md b/src/doc/unstable-book/src/language-features/new-range.md
new file mode 100644
index 000000000000..e7464f31e537
--- /dev/null
+++ b/src/doc/unstable-book/src/language-features/new-range.md
@@ -0,0 +1,9 @@
+# `new_range`
+
+The tracking issue for this feature is: [#123741]
+
+[#123741]: https://github.com/rust-lang/rust/issues/123741
+
+---
+
+Switch the syntaxes `a..`, `a..b`, and `a..=b` to resolve the new range types.
diff --git a/tests/ui/feature-gates/feature-gate-new_range.rs b/tests/ui/feature-gates/feature-gate-new_range.rs
new file mode 100644
index 000000000000..ecb73546d6a6
--- /dev/null
+++ b/tests/ui/feature-gates/feature-gate-new_range.rs
@@ -0,0 +1,10 @@
+#![feature(new_range_api)]
+
+fn main() {
+ let a: core::range::RangeFrom = 1..;
+ //~^ mismatched types
+ let b: core::range::Range = 2..3;
+ //~^ mismatched types
+ let c: core::range::RangeInclusive = 4..=5;
+ //~^ mismatched types
+}
diff --git a/tests/ui/feature-gates/feature-gate-new_range.stderr b/tests/ui/feature-gates/feature-gate-new_range.stderr
new file mode 100644
index 000000000000..c4241390418b
--- /dev/null
+++ b/tests/ui/feature-gates/feature-gate-new_range.stderr
@@ -0,0 +1,48 @@
+error[E0308]: mismatched types
+ --> $DIR/feature-gate-new_range.rs:4:41
+ |
+LL | let a: core::range::RangeFrom = 1..;
+ | -------------------------- ^^^ expected `RangeFrom`, found `RangeFrom<{integer}>`
+ | |
+ | expected due to this
+ |
+ = note: expected struct `std::range::RangeFrom`
+ found struct `std::ops::RangeFrom<{integer}>`
+help: call `Into::into` on this expression to convert `std::ops::RangeFrom<{integer}>` into `std::range::RangeFrom`
+ |
+LL | let a: core::range::RangeFrom = 1...into();
+ | +++++++
+
+error[E0308]: mismatched types
+ --> $DIR/feature-gate-new_range.rs:6:37
+ |
+LL | let b: core::range::Range = 2..3;
+ | ---------------------- ^^^^ expected `Range`, found `Range<{integer}>`
+ | |
+ | expected due to this
+ |
+ = note: expected struct `std::range::Range`
+ found struct `std::ops::Range<{integer}>`
+help: call `Into::into` on this expression to convert `std::ops::Range<{integer}>` into `std::range::Range`
+ |
+LL | let b: core::range::Range = 2..3.into();
+ | +++++++
+
+error[E0308]: mismatched types
+ --> $DIR/feature-gate-new_range.rs:8:46
+ |
+LL | let c: core::range::RangeInclusive = 4..=5;
+ | ------------------------------- ^^^^^ expected `RangeInclusive`, found `RangeInclusive<{integer}>`
+ | |
+ | expected due to this
+ |
+ = note: expected struct `std::range::RangeInclusive`
+ found struct `std::ops::RangeInclusive<{integer}>`
+help: call `Into::into` on this expression to convert `std::ops::RangeInclusive<{integer}>` into `std::range::RangeInclusive`
+ |
+LL | let c: core::range::RangeInclusive = 4..=5.into();
+ | +++++++
+
+error: aborting due to 3 previous errors
+
+For more information about this error, try `rustc --explain E0308`.
diff --git a/tests/ui/mismatched_types/closure-arg-type-mismatch-issue-45727.current.fixed b/tests/ui/mismatched_types/closure-arg-type-mismatch-issue-45727.current.fixed
index fc44c824043b..ba46a447802c 100644
--- a/tests/ui/mismatched_types/closure-arg-type-mismatch-issue-45727.current.fixed
+++ b/tests/ui/mismatched_types/closure-arg-type-mismatch-issue-45727.current.fixed
@@ -9,5 +9,5 @@ fn main() {
let _ = (-10..=10).find(|x: &i32| x.signum() == 0);
//[current]~^ ERROR type mismatch in closure arguments
//[next]~^^ ERROR expected `RangeInclusive<{integer}>` to be an iterator that yields `&&i32`, but it yields `{integer}`
- //[next]~| ERROR expected a `FnMut(& as Iterator>::Item)` closure, found
+ //[next]~| ERROR expected a `FnMut(& as Iterator>::Item)` closure, found
}
diff --git a/tests/ui/mismatched_types/closure-arg-type-mismatch-issue-45727.next.stderr b/tests/ui/mismatched_types/closure-arg-type-mismatch-issue-45727.next.stderr
index 973fe7ade602..b71613106199 100644
--- a/tests/ui/mismatched_types/closure-arg-type-mismatch-issue-45727.next.stderr
+++ b/tests/ui/mismatched_types/closure-arg-type-mismatch-issue-45727.next.stderr
@@ -18,17 +18,17 @@ error[E0271]: expected `RangeInclusive<{integer}>` to be an iterator that yields
LL | let _ = (-10..=10).find(|x: &&&i32| x.signum() == 0);
| ^^^^ expected `&&i32`, found integer
-error[E0277]: expected a `FnMut(& as Iterator>::Item)` closure, found `{closure@$DIR/closure-arg-type-mismatch-issue-45727.rs:9:29: 9:40}`
+error[E0277]: expected a `FnMut(& as Iterator>::Item)` closure, found `{closure@$DIR/closure-arg-type-mismatch-issue-45727.rs:9:29: 9:40}`
--> $DIR/closure-arg-type-mismatch-issue-45727.rs:9:29
|
LL | let _ = (-10..=10).find(|x: &&&i32| x.signum() == 0);
- | ---- ^^^^^^^^^^^^^^^^^^^^^^^^^^^ expected an `FnMut(& as Iterator>::Item)` closure, found `{closure@$DIR/closure-arg-type-mismatch-issue-45727.rs:9:29: 9:40}`
+ | ---- ^^^^^^^^^^^^^^^^^^^^^^^^^^^ expected an `FnMut(& as Iterator>::Item)` closure, found `{closure@$DIR/closure-arg-type-mismatch-issue-45727.rs:9:29: 9:40}`
| |
| required by a bound introduced by this call
|
- = help: the trait `for<'a> FnMut(&'a as Iterator>::Item)` is not implemented for closure `{closure@$DIR/closure-arg-type-mismatch-issue-45727.rs:9:29: 9:40}`
+ = help: the trait `for<'a> FnMut(&'a as Iterator>::Item)` is not implemented for closure `{closure@$DIR/closure-arg-type-mismatch-issue-45727.rs:9:29: 9:40}`
= note: expected a closure with arguments `(&&&i32,)`
- found a closure with arguments `(& as Iterator>::Item,)`
+ found a closure with arguments `(& as Iterator>::Item,)`
note: required by a bound in `find`
--> $SRC_DIR/core/src/iter/traits/iterator.rs:LL:COL
diff --git a/tests/ui/mismatched_types/closure-arg-type-mismatch-issue-45727.rs b/tests/ui/mismatched_types/closure-arg-type-mismatch-issue-45727.rs
index a96df10db355..0fd56707763e 100644
--- a/tests/ui/mismatched_types/closure-arg-type-mismatch-issue-45727.rs
+++ b/tests/ui/mismatched_types/closure-arg-type-mismatch-issue-45727.rs
@@ -9,5 +9,5 @@ fn main() {
let _ = (-10..=10).find(|x: &&&i32| x.signum() == 0);
//[current]~^ ERROR type mismatch in closure arguments
//[next]~^^ ERROR expected `RangeInclusive<{integer}>` to be an iterator that yields `&&i32`, but it yields `{integer}`
- //[next]~| ERROR expected a `FnMut(& as Iterator>::Item)` closure, found
+ //[next]~| ERROR expected a `FnMut(& as Iterator>::Item)` closure, found
}
diff --git a/tests/ui/new-range/disabled.rs b/tests/ui/new-range/disabled.rs
new file mode 100644
index 000000000000..1a5fe3f97436
--- /dev/null
+++ b/tests/ui/new-range/disabled.rs
@@ -0,0 +1,27 @@
+//@ check-pass
+
+#![feature(new_range_api)]
+
+fn main() {
+ // Unchanged
+ let a: core::range::RangeFull = ..;
+ let b: core::range::RangeTo = ..2;
+ let c: core::range::RangeToInclusive = ..=3;
+
+ let _: core::ops::RangeFull = a;
+ let _: core::ops::RangeTo = b;
+ let _: core::ops::RangeToInclusive = c;
+
+ // Changed
+ let a: core::range::legacy::RangeFrom = 1..;
+ let b: core::range::legacy::Range = 2..3;
+ let c: core::range::legacy::RangeInclusive = 4..=5;
+
+ let a: core::ops::RangeFrom = a;
+ let b: core::ops::Range = b;
+ let c: core::ops::RangeInclusive = c;
+
+ let _: core::ops::RangeFrom = a.into_iter();
+ let _: core::ops::Range = b.into_iter();
+ let _: core::ops::RangeInclusive = c.into_iter();
+}
diff --git a/tests/ui/new-range/enabled.rs b/tests/ui/new-range/enabled.rs
new file mode 100644
index 000000000000..a5fb76ad52b7
--- /dev/null
+++ b/tests/ui/new-range/enabled.rs
@@ -0,0 +1,24 @@
+//@ check-pass
+
+#![feature(new_range_api)]
+#![feature(new_range)]
+
+fn main() {
+ // Unchanged
+ let a: core::range::RangeFull = ..;
+ let b: core::range::RangeTo = ..2;
+ let c: core::range::RangeToInclusive = ..=3;
+
+ let _: core::ops::RangeFull = a;
+ let _: core::ops::RangeTo = b;
+ let _: core::ops::RangeToInclusive = c;
+
+ // Changed
+ let a: core::range::RangeFrom = 1..;
+ let b: core::range::Range = 2..3;
+ let c: core::range::RangeInclusive = 4..=5;
+
+ let _: core::range::IterRangeFrom = a.into_iter();
+ let _: core::range::IterRange = b.into_iter();
+ let _: core::range::IterRangeInclusive = c.into_iter();
+}