Skip to content

Commit 0781e50

Browse files
committed
pass ast_index to LoweringContext and lower impls correctly
plus some other fixes.
1 parent e80f33b commit 0781e50

File tree

9 files changed

+100
-15
lines changed

9 files changed

+100
-15
lines changed

compiler/rustc_ast_lowering/src/item.rs

+20-5
Original file line numberDiff line numberDiff line change
@@ -54,7 +54,7 @@ impl<'a, 'hir> ItemLowerer<'a, 'hir> {
5454
owner: NodeId,
5555
f: impl FnOnce(&mut LoweringContext<'_, 'hir>) -> hir::OwnerNode<'hir>,
5656
) {
57-
let mut lctx = LoweringContext::new(self.tcx, self.resolver);
57+
let mut lctx = LoweringContext::new(self.tcx, self.resolver, self.ast_index);
5858
lctx.with_hir_id_owner(owner, |lctx| f(lctx));
5959

6060
for (def_id, info) in lctx.children {
@@ -579,10 +579,25 @@ impl<'hir> LoweringContext<'_, 'hir> {
579579
let constness = match parent_item.kind {
580580
hir::ItemKind::Impl(impl_) => {
581581
self.is_in_trait_impl = impl_.of_trait.is_some();
582-
match impl_.constness {
583-
// TODO bad span
584-
hir::Constness::Const => Const::Yes(impl_.self_ty.span),
585-
hir::Constness::NotConst => Const::No,
582+
// N.B. the impl should always lower to methods that have `const host: bool` params if the trait
583+
// is const. It doesn't matter whether the `impl` itself is const. Disallowing const fn from
584+
// calling non-const impls are done through associated types.
585+
if let Some(def_id) = impl_.of_trait.and_then(|tr| tr.trait_def_id()) {
586+
if let Some(local_def) = def_id.as_local() {
587+
match &self.ast_index[local_def] {
588+
AstOwner::Item(ast::Item { attrs, .. }) => attrs
589+
.iter()
590+
.find(|attr| attr.has_name(sym::const_trait))
591+
.map_or(Const::No, |attr| Const::Yes(attr.span)),
592+
_ => Const::No,
593+
}
594+
} else {
595+
self.tcx
596+
.get_attr(def_id, sym::const_trait)
597+
.map_or(Const::No, |attr| Const::Yes(attr.span))
598+
}
599+
} else {
600+
Const::No
586601
}
587602
}
588603
hir::ItemKind::Trait(_, _, _, _, _) => parent_hir

compiler/rustc_ast_lowering/src/lib.rs

+8-2
Original file line numberDiff line numberDiff line change
@@ -139,14 +139,19 @@ struct LoweringContext<'a, 'hir> {
139139
generics_def_id_map: Vec<LocalDefIdMap<LocalDefId>>,
140140

141141
host_param_id: Option<LocalDefId>,
142+
ast_index: &'a IndexSlice<LocalDefId, AstOwner<'a>>,
142143
}
143144

144145
impl<'a, 'hir> LoweringContext<'a, 'hir> {
145-
fn new(tcx: TyCtxt<'hir>, resolver: &'a mut ResolverAstLowering) -> Self {
146+
fn new(
147+
tcx: TyCtxt<'hir>,
148+
resolver: &'a mut ResolverAstLowering,
149+
ast_index: &'a IndexSlice<LocalDefId, AstOwner<'a>>,
150+
) -> Self {
146151
Self {
147152
// Pseudo-globals.
148153
tcx,
149-
resolver: resolver,
154+
resolver,
150155
arena: tcx.hir_arena,
151156

152157
// HirId handling.
@@ -181,6 +186,7 @@ impl<'a, 'hir> LoweringContext<'a, 'hir> {
181186
allow_async_iterator: [sym::gen_future, sym::async_iterator].into(),
182187
generics_def_id_map: Default::default(),
183188
host_param_id: None,
189+
ast_index,
184190
}
185191
}
186192

compiler/rustc_hir/src/lang_items.rs

+1
Original file line numberDiff line numberDiff line change
@@ -346,6 +346,7 @@ language_item_table! {
346346
EffectsNoRuntime, sym::EffectsNoRuntime, effects_no_runtime, Target::Struct, GenericRequirement::None;
347347
EffectsMaybe, sym::EffectsMaybe, effects_maybe, Target::Struct, GenericRequirement::None;
348348
EffectsCompat, sym::EffectsCompat, effects_compat, Target::Trait, GenericRequirement::Exact(1);
349+
EffectsEq, sym::EffectsEq, effects_eq, Target::Trait, GenericRequirement::Exact(1);
349350
}
350351

351352
pub enum GenericRequirement {

compiler/rustc_hir_analysis/src/bounds.rs

+46-7
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
//! Bounds are restrictions applied to some types after they've been converted into the
22
//! `ty` form from the HIR.
33
4-
use rustc_hir::LangItem;
4+
use rustc_hir::{def::DefKind, LangItem};
55
use rustc_middle::ty::{self, ToPredicate, Ty, TyCtxt};
66
use rustc_span::{def_id::DefId, Span};
77

@@ -54,15 +54,54 @@ impl<'tcx> Bounds<'tcx> {
5454
.to_predicate(tcx),
5555
span,
5656
));
57-
// For `T: ~const Tr` or `T: const Tr`, we need to add an additional bound on the
58-
// associated type of `<T as Tr>` and make sure that the effect is compatible.
59-
if let Some(compat_val) = match constness {
57+
if let Some(compat_val) = match (tcx.def_kind(defining_def_id), constness) {
6058
// TODO: do we need `T: const Trait` anymore?
61-
ty::BoundConstness::Const => Some(tcx.consts.false_),
62-
ty::BoundConstness::ConstIfConst => {
59+
(_, ty::BoundConstness::Const) => Some(tcx.consts.false_),
60+
// body owners that can have trait bounds
61+
(DefKind::Const | DefKind::Fn | DefKind::AssocFn, ty::BoundConstness::ConstIfConst) => {
6362
Some(tcx.expected_host_effect_param_for_body(defining_def_id))
6463
}
65-
ty::BoundConstness::NotConst => None,
64+
65+
(_, ty::BoundConstness::NotConst) => None,
66+
67+
// if the defining_def_id is a trait, we wire it differently than others by equating the effects.
68+
(
69+
kind @ (DefKind::Trait | DefKind::Impl { of_trait: true }),
70+
ty::BoundConstness::ConstIfConst,
71+
) => {
72+
let trait_we_are_in = if let DefKind::Trait = kind {
73+
ty::TraitRef::identity(tcx, defining_def_id)
74+
} else {
75+
tcx.impl_trait_ref(defining_def_id).unwrap().instantiate_identity()
76+
};
77+
// create a new projection type `<T as TraitForBound>::Effects`
78+
let assoc = tcx.associated_type_for_effects(trait_ref.def_id()).unwrap();
79+
let self_ty = Ty::new_projection(tcx, assoc, trait_ref.skip_binder().args);
80+
// we might have `~const Tr` where `Tr` isn't a `#[const_trait]`.
81+
let Some(assoc_def) = tcx.associated_type_for_effects(trait_we_are_in.def_id)
82+
else {
83+
tcx.dcx().span_delayed_bug(
84+
span,
85+
"`~const` bound trait has no effect param yet no errors encountered?",
86+
);
87+
return;
88+
};
89+
let fx_ty_trait_we_are_in =
90+
Ty::new_projection(tcx, assoc_def, trait_we_are_in.args);
91+
// make `<T as TraitForBound>::Effects: EffectsEq<<Self as TraitWeAreIn>::Effects>`
92+
let new_trait_ref = ty::TraitRef::new(
93+
tcx,
94+
tcx.require_lang_item(LangItem::EffectsEq, Some(span)),
95+
[self_ty, fx_ty_trait_we_are_in],
96+
);
97+
self.clauses.push((trait_ref.rebind(new_trait_ref).to_predicate(tcx), span));
98+
return;
99+
}
100+
// probably illegal in this position.
101+
(_, ty::BoundConstness::ConstIfConst) => {
102+
tcx.dcx().span_delayed_bug(span, "invalid `~const` encountered");
103+
return;
104+
}
66105
} {
67106
// create a new projection type `<T as Tr>::Effects`
68107
let assoc = tcx.associated_type_for_effects(trait_ref.def_id()).unwrap();

compiler/rustc_hir_analysis/src/collect.rs

+1
Original file line numberDiff line numberDiff line change
@@ -1342,6 +1342,7 @@ fn impl_trait_ref(
13421342
tcx.is_const_trait_impl_raw(def_id.to_def_id()),
13431343
ast_trait_ref,
13441344
) {
1345+
// TODO this is wrong
13451346
// we have a const impl, but for a trait without `#[const_trait]`, so
13461347
// without the host param. If we continue with the HIR trait ref, we get
13471348
// ICEs for generic arg count mismatch. We do a little HIR editing to

compiler/rustc_span/src/symbol.rs

+1
Original file line numberDiff line numberDiff line change
@@ -190,6 +190,7 @@ symbols! {
190190
DoubleEndedIterator,
191191
Duration,
192192
EffectsCompat,
193+
EffectsEq,
193194
EffectsMaybe,
194195
EffectsNoRuntime,
195196
EffectsRuntime,

compiler/rustc_trait_selection/src/traits/error_reporting/suggestions.rs

+6
Original file line numberDiff line numberDiff line change
@@ -5112,6 +5112,12 @@ fn point_at_assoc_type_restriction(
51125112
let ty::ClauseKind::Projection(proj) = clause else {
51135113
return;
51145114
};
5115+
// avoid ICEing since effects desugared associated types don't have names.
5116+
// this path should only be hit for `~const` on invalid places, so they
5117+
// will have an informative error already.
5118+
if tcx.is_effects_desugared_assoc_ty(proj.projection_ty.def_id) {
5119+
return;
5120+
}
51155121
let name = tcx.item_name(proj.projection_ty.def_id);
51165122
let mut predicates = generics.predicates.iter().peekable();
51175123
let mut prev: Option<&hir::WhereBoundPredicate<'_>> = None;

library/core/src/marker.rs

+7
Original file line numberDiff line numberDiff line change
@@ -1104,6 +1104,7 @@ pub trait FnPtr: Copy + Clone {
11041104
)]
11051105
#[allow(missing_debug_implementations)] // these unit structs don't need `Debug` impls.
11061106
#[cfg(not(bootstrap))]
1107+
// TODO docs
11071108
pub mod effects {
11081109
#[lang = "EffectsNoRuntime"]
11091110
pub struct NoRuntime;
@@ -1118,4 +1119,10 @@ pub mod effects {
11181119
impl Compat<false> for NoRuntime {}
11191120
impl Compat<true> for Runtime {}
11201121
impl<const RUNTIME: bool> Compat<RUNTIME> for Maybe {}
1122+
1123+
#[lang = "EffectsEq"]
1124+
pub trait EffectsEq<T: ?Sized> {}
1125+
impl EffectsEq<NoRuntime> for NoRuntime {}
1126+
impl EffectsEq<Maybe> for Maybe {}
1127+
impl EffectsEq<Runtime> for Runtime {}
11211128
}

tests/ui/rfcs/rfc-2632-const-trait-impl/tilde-const-invalid-places.stderr

+10-1
Original file line numberDiff line numberDiff line change
@@ -292,12 +292,21 @@ help: wrap the field type in `ManuallyDrop<...>`
292292
LL | union Union<T: ~const Trait> { field: std::mem::ManuallyDrop<T> }
293293
| +++++++++++++++++++++++ +
294294

295-
error[E0275]: overflow evaluating the requirement `(): Trait`
295+
error[E0275]: overflow evaluating the requirement `Trait::{opaque#0} == _`
296296
--> $DIR/tilde-const-invalid-places.rs:34:34
297297
|
298298
LL | type Type<T: ~const Trait> = ();
299299
| ^^
300300
|
301+
note: required for `()` to implement `Trait`
302+
--> $DIR/tilde-const-invalid-places.rs:56:23
303+
|
304+
LL | impl<T: ~const Trait> Trait for T {}
305+
| ------------ ^^^^^ ^
306+
| |
307+
| unsatisfied trait bound introduced here
308+
= note: 1 redundant requirement hidden
309+
= note: required for `()` to implement `Trait`
301310
note: required by a bound in `NonConstTrait::Type`
302311
--> $DIR/tilde-const-invalid-places.rs:25:33
303312
|

0 commit comments

Comments
 (0)