Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit b649346

Browse files
committedMar 3, 2025
Arbitrary self types v2: stabilize! [WIP]
1 parent 81d8edc commit b649346

File tree

15 files changed

+90
-257
lines changed

15 files changed

+90
-257
lines changed
 

‎compiler/rustc_feature/src/accepted.rs

+2
Original file line numberDiff line numberDiff line change
@@ -60,6 +60,8 @@ declare_features! (
6060
(accepted, adx_target_feature, "1.61.0", Some(44839)),
6161
/// Allows explicit discriminants on non-unit enum variants.
6262
(accepted, arbitrary_enum_discriminant, "1.66.0", Some(60553)),
63+
/// Allows inherent and trait methods with arbitrary self types.
64+
(accepted, arbitrary_self_types, "CURRENT_RUSTC_VERSION", Some(44874)),
6365
/// Allows using `const` operands in inline assembly.
6466
(accepted, asm_const, "1.82.0", Some(93332)),
6567
/// Allows using `sym` operands in inline assembly.

‎compiler/rustc_feature/src/unstable.rs

-2
Original file line numberDiff line numberDiff line change
@@ -363,8 +363,6 @@ declare_features! (
363363
(unstable, adt_const_params, "1.56.0", Some(95174)),
364364
/// Allows defining an `#[alloc_error_handler]`.
365365
(unstable, alloc_error_handler, "1.29.0", Some(51540)),
366-
/// Allows inherent and trait methods with arbitrary self types.
367-
(unstable, arbitrary_self_types, "1.23.0", Some(44874)),
368366
/// Allows inherent and trait methods with arbitrary self types that are raw pointers.
369367
(unstable, arbitrary_self_types_pointers, "1.83.0", Some(44874)),
370368
/// Enables experimental inline assembly support for additional architectures.

‎compiler/rustc_hir/src/lang_items.rs

-1
Original file line numberDiff line numberDiff line change
@@ -242,7 +242,6 @@ language_item_table! {
242242
DerefTarget, sym::deref_target, deref_target, Target::AssocTy, GenericRequirement::None;
243243
Receiver, sym::receiver, receiver_trait, Target::Trait, GenericRequirement::None;
244244
ReceiverTarget, sym::receiver_target, receiver_target, Target::AssocTy, GenericRequirement::None;
245-
LegacyReceiver, sym::legacy_receiver, legacy_receiver_trait, Target::Trait, GenericRequirement::None;
246245

247246
Fn, kw::Fn, fn_trait, Target::Trait, GenericRequirement::Exact(1);
248247
FnMut, sym::fn_mut, fn_mut_trait, Target::Trait, GenericRequirement::Exact(1);

‎compiler/rustc_hir_analysis/messages.ftl

-6
Original file line numberDiff line numberDiff line change
@@ -259,18 +259,12 @@ hir_analysis_invalid_receiver_ty = invalid `self` parameter type: `{$receiver_ty
259259
hir_analysis_invalid_receiver_ty_help =
260260
consider changing to `self`, `&self`, `&mut self`, or a type implementing `Receiver` such as `self: Box<Self>`, `self: Rc<Self>`, or `self: Arc<Self>`
261261
262-
hir_analysis_invalid_receiver_ty_help_no_arbitrary_self_types =
263-
consider changing to `self`, `&self`, `&mut self`, `self: Box<Self>`, `self: Rc<Self>`, `self: Arc<Self>`, or `self: Pin<P>` (where P is one of the previous types except `Self`)
264-
265262
hir_analysis_invalid_receiver_ty_help_nonnull_note =
266263
`NonNull` does not implement `Receiver` because it has methods that may shadow the referent; consider wrapping your `NonNull` in a newtype wrapper for which you implement `Receiver`
267264
268265
hir_analysis_invalid_receiver_ty_help_weak_note =
269266
`Weak` does not implement `Receiver` because it has methods that may shadow the referent; consider wrapping your `Weak` in a newtype wrapper for which you implement `Receiver`
270267
271-
hir_analysis_invalid_receiver_ty_no_arbitrary_self_types = invalid `self` parameter type: `{$receiver_ty}`
272-
.note = type of `self` must be `Self` or a type that dereferences to it
273-
274268
hir_analysis_invalid_union_field =
275269
field must implement `Copy` or be wrapped in `ManuallyDrop<...>` to be used in a union
276270
.note = union fields must not have drop side-effects, which is currently enforced via either `Copy` or `ManuallyDrop<...>`

‎compiler/rustc_hir_analysis/src/check/wfcheck.rs

+22-125
Original file line numberDiff line numberDiff line change
@@ -29,7 +29,6 @@ use rustc_trait_selection::regions::{InferCtxtRegionExt, OutlivesEnvironmentBuil
2929
use rustc_trait_selection::traits::misc::{
3030
ConstParamTyImplementationError, type_allowed_to_implement_const_param_ty,
3131
};
32-
use rustc_trait_selection::traits::query::evaluate_obligation::InferCtxtExt as _;
3332
use rustc_trait_selection::traits::{
3433
self, FulfillmentError, Obligation, ObligationCause, ObligationCauseCode, ObligationCtxt,
3534
WellFormedLoc,
@@ -1720,13 +1719,6 @@ fn check_sized_if_body<'tcx>(
17201719
}
17211720
}
17221721

1723-
/// The `arbitrary_self_types_pointers` feature implies `arbitrary_self_types`.
1724-
#[derive(Clone, Copy, PartialEq)]
1725-
enum ArbitrarySelfTypesLevel {
1726-
Basic, // just arbitrary_self_types
1727-
WithPointers, // both arbitrary_self_types and arbitrary_self_types_pointers
1728-
}
1729-
17301722
#[instrument(level = "debug", skip(wfcx))]
17311723
fn check_method_receiver<'tcx>(
17321724
wfcx: &WfCheckingCtxt<'_, 'tcx>,
@@ -1759,55 +1751,21 @@ fn check_method_receiver<'tcx>(
17591751
return Ok(());
17601752
}
17611753

1762-
let arbitrary_self_types_level = if tcx.features().arbitrary_self_types_pointers() {
1763-
Some(ArbitrarySelfTypesLevel::WithPointers)
1764-
} else if tcx.features().arbitrary_self_types() {
1765-
Some(ArbitrarySelfTypesLevel::Basic)
1766-
} else {
1767-
None
1768-
};
17691754
let generics = tcx.generics_of(method.def_id);
17701755

1771-
let receiver_validity =
1772-
receiver_is_valid(wfcx, span, receiver_ty, self_ty, arbitrary_self_types_level, generics);
1756+
let arbitrary_self_types_pointers_enabled = tcx.features().arbitrary_self_types_pointers();
1757+
let receiver_validity = receiver_is_valid(
1758+
wfcx,
1759+
span,
1760+
receiver_ty,
1761+
self_ty,
1762+
arbitrary_self_types_pointers_enabled,
1763+
generics,
1764+
);
17731765
if let Err(receiver_validity_err) = receiver_validity {
1774-
return Err(match arbitrary_self_types_level {
1775-
// Wherever possible, emit a message advising folks that the features
1776-
// `arbitrary_self_types` or `arbitrary_self_types_pointers` might
1777-
// have helped.
1778-
None if receiver_is_valid(
1779-
wfcx,
1780-
span,
1781-
receiver_ty,
1782-
self_ty,
1783-
Some(ArbitrarySelfTypesLevel::Basic),
1784-
generics,
1785-
)
1786-
.is_ok() =>
1787-
{
1788-
// Report error; would have worked with `arbitrary_self_types`.
1789-
feature_err(
1790-
&tcx.sess,
1791-
sym::arbitrary_self_types,
1792-
span,
1793-
format!(
1794-
"`{receiver_ty}` cannot be used as the type of `self` without \
1795-
the `arbitrary_self_types` feature",
1796-
),
1797-
)
1798-
.with_help(fluent::hir_analysis_invalid_receiver_ty_help)
1799-
.emit()
1800-
}
1801-
None | Some(ArbitrarySelfTypesLevel::Basic)
1802-
if receiver_is_valid(
1803-
wfcx,
1804-
span,
1805-
receiver_ty,
1806-
self_ty,
1807-
Some(ArbitrarySelfTypesLevel::WithPointers),
1808-
generics,
1809-
)
1810-
.is_ok() =>
1766+
return Err(
1767+
if !arbitrary_self_types_pointers_enabled
1768+
&& receiver_is_valid(wfcx, span, receiver_ty, self_ty, true, generics).is_ok()
18111769
{
18121770
// Report error; would have worked with `arbitrary_self_types_pointers`.
18131771
feature_err(
@@ -1816,17 +1774,15 @@ fn check_method_receiver<'tcx>(
18161774
span,
18171775
format!(
18181776
"`{receiver_ty}` cannot be used as the type of `self` without \
1819-
the `arbitrary_self_types_pointers` feature",
1777+
the `arbitrary_self_types_pointers` feature",
18201778
),
18211779
)
18221780
.with_help(fluent::hir_analysis_invalid_receiver_ty_help)
18231781
.emit()
1824-
}
1825-
_ =>
1826-
// Report error; would not have worked with `arbitrary_self_types[_pointers]`.
1827-
{
1782+
} else {
1783+
// Report error; would not have worked with `arbitrary_self_types_pointers`.
18281784
match receiver_validity_err {
1829-
ReceiverValidityError::DoesNotDeref if arbitrary_self_types_level.is_some() => {
1785+
ReceiverValidityError::DoesNotDeref => {
18301786
let hint = match receiver_ty
18311787
.builtin_deref(false)
18321788
.unwrap_or(receiver_ty)
@@ -1840,18 +1796,12 @@ fn check_method_receiver<'tcx>(
18401796

18411797
tcx.dcx().emit_err(errors::InvalidReceiverTy { span, receiver_ty, hint })
18421798
}
1843-
ReceiverValidityError::DoesNotDeref => {
1844-
tcx.dcx().emit_err(errors::InvalidReceiverTyNoArbitrarySelfTypes {
1845-
span,
1846-
receiver_ty,
1847-
})
1848-
}
18491799
ReceiverValidityError::MethodGenericParamUsed => {
18501800
tcx.dcx().emit_err(errors::InvalidGenericReceiverTy { span, receiver_ty })
18511801
}
18521802
}
1853-
}
1854-
});
1803+
},
1804+
);
18551805
}
18561806
Ok(())
18571807
}
@@ -1895,11 +1845,10 @@ fn receiver_is_valid<'tcx>(
18951845
span: Span,
18961846
receiver_ty: Ty<'tcx>,
18971847
self_ty: Ty<'tcx>,
1898-
arbitrary_self_types_enabled: Option<ArbitrarySelfTypesLevel>,
1848+
arbitrary_self_types_pointers_enabled: bool,
18991849
method_generics: &ty::Generics,
19001850
) -> Result<(), ReceiverValidityError> {
19011851
let infcx = wfcx.infcx;
1902-
let tcx = wfcx.tcx();
19031852
let cause =
19041853
ObligationCause::new(span, wfcx.body_def_id, traits::ObligationCauseCode::MethodReceiver);
19051854

@@ -1914,17 +1863,11 @@ fn receiver_is_valid<'tcx>(
19141863

19151864
confirm_type_is_not_a_method_generic_param(receiver_ty, method_generics)?;
19161865

1917-
let mut autoderef = Autoderef::new(infcx, wfcx.param_env, wfcx.body_def_id, span, receiver_ty);
1918-
1919-
// The `arbitrary_self_types` feature allows custom smart pointer
1920-
// types to be method receivers, as identified by following the Receiver<Target=T>
1921-
// chain.
1922-
if arbitrary_self_types_enabled.is_some() {
1923-
autoderef = autoderef.use_receiver_trait();
1924-
}
1866+
let mut autoderef = Autoderef::new(infcx, wfcx.param_env, wfcx.body_def_id, span, receiver_ty)
1867+
.use_receiver_trait();
19251868

19261869
// The `arbitrary_self_types_pointers` feature allows raw pointer receivers like `self: *const Self`.
1927-
if arbitrary_self_types_enabled == Some(ArbitrarySelfTypesLevel::WithPointers) {
1870+
if arbitrary_self_types_pointers_enabled {
19281871
autoderef = autoderef.include_raw_pointers();
19291872
}
19301873

@@ -1947,58 +1890,12 @@ fn receiver_is_valid<'tcx>(
19471890
wfcx.register_obligations(autoderef.into_obligations());
19481891
return Ok(());
19491892
}
1950-
1951-
// Without `feature(arbitrary_self_types)`, we require that each step in the
1952-
// deref chain implement `LegacyReceiver`.
1953-
if arbitrary_self_types_enabled.is_none() {
1954-
let legacy_receiver_trait_def_id =
1955-
tcx.require_lang_item(LangItem::LegacyReceiver, Some(span));
1956-
if !legacy_receiver_is_implemented(
1957-
wfcx,
1958-
legacy_receiver_trait_def_id,
1959-
cause.clone(),
1960-
potential_self_ty,
1961-
) {
1962-
// We cannot proceed.
1963-
break;
1964-
}
1965-
1966-
// Register the bound, in case it has any region side-effects.
1967-
wfcx.register_bound(
1968-
cause.clone(),
1969-
wfcx.param_env,
1970-
potential_self_ty,
1971-
legacy_receiver_trait_def_id,
1972-
);
1973-
}
19741893
}
19751894

19761895
debug!("receiver_is_valid: type `{:?}` does not deref to `{:?}`", receiver_ty, self_ty);
19771896
Err(ReceiverValidityError::DoesNotDeref)
19781897
}
19791898

1980-
fn legacy_receiver_is_implemented<'tcx>(
1981-
wfcx: &WfCheckingCtxt<'_, 'tcx>,
1982-
legacy_receiver_trait_def_id: DefId,
1983-
cause: ObligationCause<'tcx>,
1984-
receiver_ty: Ty<'tcx>,
1985-
) -> bool {
1986-
let tcx = wfcx.tcx();
1987-
let trait_ref = ty::TraitRef::new(tcx, legacy_receiver_trait_def_id, [receiver_ty]);
1988-
1989-
let obligation = Obligation::new(tcx, cause, wfcx.param_env, trait_ref);
1990-
1991-
if wfcx.infcx.predicate_must_hold_modulo_regions(&obligation) {
1992-
true
1993-
} else {
1994-
debug!(
1995-
"receiver_is_implemented: type `{:?}` does not implement `LegacyReceiver` trait",
1996-
receiver_ty
1997-
);
1998-
false
1999-
}
2000-
}
2001-
20021899
fn check_variances_for_type_defn<'tcx>(
20031900
tcx: TyCtxt<'tcx>,
20041901
item: &'tcx hir::Item<'tcx>,

‎compiler/rustc_hir_analysis/src/errors.rs

-10
Original file line numberDiff line numberDiff line change
@@ -1633,16 +1633,6 @@ pub(crate) enum InvalidReceiverTyHint {
16331633
NonNull,
16341634
}
16351635

1636-
#[derive(Diagnostic)]
1637-
#[diag(hir_analysis_invalid_receiver_ty_no_arbitrary_self_types, code = E0307)]
1638-
#[note]
1639-
#[help(hir_analysis_invalid_receiver_ty_help_no_arbitrary_self_types)]
1640-
pub(crate) struct InvalidReceiverTyNoArbitrarySelfTypes<'tcx> {
1641-
#[primary_span]
1642-
pub span: Span,
1643-
pub receiver_ty: Ty<'tcx>,
1644-
}
1645-
16461636
#[derive(Diagnostic)]
16471637
#[diag(hir_analysis_invalid_receiver_ty, code = E0307)]
16481638
#[note]

‎compiler/rustc_hir_typeck/src/method/confirm.rs

+4-27
Original file line numberDiff line numberDiff line change
@@ -352,15 +352,7 @@ impl<'a, 'tcx> ConfirmContext<'a, 'tcx> {
352352
// yield an object-type (e.g., `&Object` or `Box<Object>`
353353
// etc).
354354

355-
let mut autoderef = self.fcx.autoderef(self.span, self_ty);
356-
357-
// We don't need to gate this behind arbitrary self types
358-
// per se, but it does make things a bit more gated.
359-
if self.tcx.features().arbitrary_self_types()
360-
|| self.tcx.features().arbitrary_self_types_pointers()
361-
{
362-
autoderef = autoderef.use_receiver_trait();
363-
}
355+
let autoderef = self.fcx.autoderef(self.span, self_ty).use_receiver_trait();
364356

365357
autoderef
366358
.include_raw_pointers()
@@ -540,24 +532,9 @@ impl<'a, 'tcx> ConfirmContext<'a, 'tcx> {
540532
self.register_predicates(obligations);
541533
}
542534
Err(terr) => {
543-
if self.tcx.features().arbitrary_self_types() {
544-
self.err_ctxt()
545-
.report_mismatched_types(
546-
&cause,
547-
self.param_env,
548-
method_self_ty,
549-
self_ty,
550-
terr,
551-
)
552-
.emit();
553-
} else {
554-
// This has/will have errored in wfcheck, which we cannot depend on from here, as typeck on functions
555-
// may run before wfcheck if the function is used in const eval.
556-
self.dcx().span_delayed_bug(
557-
cause.span,
558-
format!("{self_ty} was a subtype of {method_self_ty} but now is not?"),
559-
);
560-
}
535+
self.err_ctxt()
536+
.report_mismatched_types(&cause, self.param_env, method_self_ty, self_ty, terr)
537+
.emit();
561538
}
562539
}
563540
}

0 commit comments

Comments
 (0)
Please sign in to comment.