Propagate the resolved type of assoc const bindings via query feeding
This commit is contained in:
parent
23a3d777c8
commit
8bb49e22b5
@ -408,7 +408,7 @@ impl<'tcx> dyn AstConv<'tcx> + '_ {
|
|||||||
// Create the generic arguments for the associated type or constant by joining the
|
// Create the generic arguments for the associated type or constant by joining the
|
||||||
// parent arguments (the arguments of the trait) and the own arguments (the ones of
|
// parent arguments (the arguments of the trait) and the own arguments (the ones of
|
||||||
// the associated item itself) and construct an alias type using them.
|
// the associated item itself) and construct an alias type using them.
|
||||||
candidate.map_bound(|trait_ref| {
|
let alias_ty = candidate.map_bound(|trait_ref| {
|
||||||
let ident = Ident::new(assoc_item.name, binding.ident.span);
|
let ident = Ident::new(assoc_item.name, binding.ident.span);
|
||||||
let item_segment = hir::PathSegment {
|
let item_segment = hir::PathSegment {
|
||||||
ident,
|
ident,
|
||||||
@ -430,7 +430,18 @@ impl<'tcx> dyn AstConv<'tcx> + '_ {
|
|||||||
// *constants* to represent *const projections*. Alias *term* would be a more
|
// *constants* to represent *const projections*. Alias *term* would be a more
|
||||||
// appropriate name but alas.
|
// appropriate name but alas.
|
||||||
ty::AliasTy::new(tcx, assoc_item.def_id, alias_args)
|
ty::AliasTy::new(tcx, assoc_item.def_id, alias_args)
|
||||||
})
|
});
|
||||||
|
|
||||||
|
// Provide the resolved type of the associated constant to `type_of(AnonConst)`.
|
||||||
|
if !speculative && let ty::AssocKind::Const = assoc_kind {
|
||||||
|
let ty = alias_ty.map_bound(|ty| tcx.type_of(ty.def_id).instantiate(tcx, ty.args));
|
||||||
|
// Since the arguments passed to the alias type above may contain early-bound
|
||||||
|
// generic parameters, the instantiated type may contain some as well.
|
||||||
|
// Therefore wrap it in `EarlyBinder`.
|
||||||
|
tcx.feed_type_of_assoc_const_binding(binding.hir_id, ty::EarlyBinder::bind(ty));
|
||||||
|
}
|
||||||
|
|
||||||
|
alias_ty
|
||||||
};
|
};
|
||||||
|
|
||||||
match binding.kind {
|
match binding.kind {
|
||||||
|
@ -63,6 +63,7 @@ pub fn provide(providers: &mut Providers) {
|
|||||||
*providers = Providers {
|
*providers = Providers {
|
||||||
type_of: type_of::type_of,
|
type_of: type_of::type_of,
|
||||||
type_of_opaque: type_of::type_of_opaque,
|
type_of_opaque: type_of::type_of_opaque,
|
||||||
|
type_of_assoc_const_binding: type_of::type_of_assoc_const_binding,
|
||||||
type_alias_is_lazy: type_of::type_alias_is_lazy,
|
type_alias_is_lazy: type_of::type_alias_is_lazy,
|
||||||
item_bounds: item_bounds::item_bounds,
|
item_bounds: item_bounds::item_bounds,
|
||||||
explicit_item_bounds: item_bounds::explicit_item_bounds,
|
explicit_item_bounds: item_bounds::explicit_item_bounds,
|
||||||
|
@ -78,35 +78,10 @@ fn anon_const_type_of<'tcx>(tcx: TyCtxt<'tcx>, def_id: LocalDefId) -> Ty<'tcx> {
|
|||||||
.expect("const parameter types cannot be generic");
|
.expect("const parameter types cannot be generic");
|
||||||
}
|
}
|
||||||
|
|
||||||
Node::TypeBinding(binding @ &TypeBinding { hir_id: binding_id, .. })
|
Node::TypeBinding(&TypeBinding { hir_id, .. }) => {
|
||||||
if let Node::TraitRef(trait_ref) = tcx.parent_hir_node(binding_id) =>
|
// FIXME(fmease): Reject “escaping” early-bound generic parameters.
|
||||||
{
|
// FIXME(fmease): Reject escaping late-bound vars.
|
||||||
let Some(trait_def_id) = trait_ref.trait_def_id() else {
|
return tcx.type_of_assoc_const_binding(hir_id).skip_binder().skip_binder();
|
||||||
return Ty::new_error_with_message(
|
|
||||||
tcx,
|
|
||||||
tcx.def_span(def_id),
|
|
||||||
"Could not find trait",
|
|
||||||
);
|
|
||||||
};
|
|
||||||
let assoc_items = tcx.associated_items(trait_def_id);
|
|
||||||
let assoc_item = assoc_items.find_by_name_and_kind(
|
|
||||||
tcx,
|
|
||||||
binding.ident,
|
|
||||||
ty::AssocKind::Const,
|
|
||||||
def_id.to_def_id(),
|
|
||||||
);
|
|
||||||
return if let Some(assoc_item) = assoc_item {
|
|
||||||
tcx.type_of(assoc_item.def_id)
|
|
||||||
.no_bound_vars()
|
|
||||||
.expect("const parameter types cannot be generic")
|
|
||||||
} else {
|
|
||||||
// FIXME(associated_const_equality): add a useful error message here.
|
|
||||||
Ty::new_error_with_message(
|
|
||||||
tcx,
|
|
||||||
tcx.def_span(def_id),
|
|
||||||
"Could not find associated const on trait",
|
|
||||||
)
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// This match arm is for when the def_id appears in a GAT whose
|
// This match arm is for when the def_id appears in a GAT whose
|
||||||
@ -315,6 +290,18 @@ fn anon_const_type_of<'tcx>(tcx: TyCtxt<'tcx>, def_id: LocalDefId) -> Ty<'tcx> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(super) fn type_of_assoc_const_binding<'tcx>(
|
||||||
|
tcx: TyCtxt<'tcx>,
|
||||||
|
hir_id: HirId,
|
||||||
|
) -> ty::EarlyBinder<ty::Binder<'tcx, Ty<'tcx>>> {
|
||||||
|
let reported = tcx.dcx().delayed_bug(format!(
|
||||||
|
"attempt to obtain type of assoc const binding `{hir_id}` before \
|
||||||
|
it was resolved by `add_predicates_for_ast_type_binding`"
|
||||||
|
));
|
||||||
|
|
||||||
|
ty::EarlyBinder::bind(ty::Binder::dummy(Ty::new_error(tcx, reported)))
|
||||||
|
}
|
||||||
|
|
||||||
fn get_path_containing_arg_in_pat<'hir>(
|
fn get_path_containing_arg_in_pat<'hir>(
|
||||||
pat: &'hir hir::Pat<'hir>,
|
pat: &'hir hir::Pat<'hir>,
|
||||||
arg_id: HirId,
|
arg_id: HirId,
|
||||||
|
@ -193,6 +193,10 @@ impl<T: EraseType> EraseType for ty::EarlyBinder<T> {
|
|||||||
type Result = T::Result;
|
type Result = T::Result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl EraseType for ty::Binder<'_, Ty<'_>> {
|
||||||
|
type Result = [u8; size_of::<ty::Binder<'static, Ty<'static>>>()];
|
||||||
|
}
|
||||||
|
|
||||||
impl EraseType for ty::Binder<'_, ty::FnSig<'_>> {
|
impl EraseType for ty::Binder<'_, ty::FnSig<'_>> {
|
||||||
type Result = [u8; size_of::<ty::Binder<'static, ty::FnSig<'static>>>()];
|
type Result = [u8; size_of::<ty::Binder<'static, ty::FnSig<'static>>>()];
|
||||||
}
|
}
|
||||||
|
@ -248,6 +248,11 @@ rustc_queries! {
|
|||||||
cycle_stash
|
cycle_stash
|
||||||
}
|
}
|
||||||
|
|
||||||
|
query type_of_assoc_const_binding(key: hir::HirId) -> ty::EarlyBinder<ty::Binder<'tcx, Ty<'tcx>>> {
|
||||||
|
desc { |tcx| "getting type of associated constant binding `{key:?}`" }
|
||||||
|
feedable
|
||||||
|
}
|
||||||
|
|
||||||
query type_alias_is_lazy(key: DefId) -> bool {
|
query type_alias_is_lazy(key: DefId) -> bool {
|
||||||
desc { |tcx|
|
desc { |tcx|
|
||||||
"computing whether `{path}` is a lazy type alias",
|
"computing whether `{path}` is a lazy type alias",
|
||||||
|
@ -71,6 +71,7 @@ use rustc_type_ir::TyKind::*;
|
|||||||
use rustc_type_ir::WithCachedTypeInfo;
|
use rustc_type_ir::WithCachedTypeInfo;
|
||||||
use rustc_type_ir::{CollectAndApply, Interner, TypeFlags};
|
use rustc_type_ir::{CollectAndApply, Interner, TypeFlags};
|
||||||
|
|
||||||
|
use std::assert_matches::debug_assert_matches;
|
||||||
use std::borrow::Borrow;
|
use std::borrow::Borrow;
|
||||||
use std::cmp::Ordering;
|
use std::cmp::Ordering;
|
||||||
use std::fmt;
|
use std::fmt;
|
||||||
@ -540,6 +541,22 @@ impl<'tcx> TyCtxt<'tcx> {
|
|||||||
debug_assert_eq!(self.def_kind(key), DefKind::AnonConst);
|
debug_assert_eq!(self.def_kind(key), DefKind::AnonConst);
|
||||||
TyCtxtFeed { tcx: self, key }.type_of(value)
|
TyCtxtFeed { tcx: self, key }.type_of(value)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn feed_type_of_assoc_const_binding(
|
||||||
|
self,
|
||||||
|
key: hir::HirId,
|
||||||
|
value: ty::EarlyBinder<ty::Binder<'tcx, Ty<'tcx>>>,
|
||||||
|
) {
|
||||||
|
debug_assert_matches!(
|
||||||
|
self.hir_node(key),
|
||||||
|
hir::Node::TypeBinding(hir::TypeBinding {
|
||||||
|
kind: hir::TypeBindingKind::Equality { term: hir::Term::Const(_) },
|
||||||
|
..
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
TyCtxtFeed { tcx: self, key }.type_of_assoc_const_binding(value)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'tcx, KEY: Copy> TyCtxtFeed<'tcx, KEY> {
|
impl<'tcx, KEY: Copy> TyCtxtFeed<'tcx, KEY> {
|
||||||
|
16
tests/ui/associated-consts/assoc-const-eq-supertraits.rs
Normal file
16
tests/ui/associated-consts/assoc-const-eq-supertraits.rs
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
// Regression test for issue #118040.
|
||||||
|
// Ensure that we support assoc const eq bounds where the assoc const comes from a supertrait.
|
||||||
|
|
||||||
|
//@ check-pass
|
||||||
|
|
||||||
|
#![feature(associated_const_equality)]
|
||||||
|
|
||||||
|
trait Trait: SuperTrait {}
|
||||||
|
trait SuperTrait: SuperSuperTrait<i32> {}
|
||||||
|
trait SuperSuperTrait<T> {
|
||||||
|
const K: T;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn take(_: impl Trait<K = 0>) {}
|
||||||
|
|
||||||
|
fn main() {}
|
@ -1,21 +1,25 @@
|
|||||||
// Regression test for issue #112560.
|
// Regression test for issue #112560.
|
||||||
// Respect the fact that (associated) types and constants live in different namespaces and
|
// Respect the fact that (associated) types and constants live in different namespaces and
|
||||||
// therefore equality bounds involving identically named associated items don't conflict if
|
// therefore equality bounds involving identically named associated items don't conflict if
|
||||||
// their kind (type vs. const) differs.
|
// their kind (type vs. const) differs. This obviously extends to supertraits.
|
||||||
|
|
||||||
// FIXME(fmease): Extend this test to cover supertraits again
|
|
||||||
// once #118040 is fixed. See initial version of PR #118360.
|
|
||||||
|
|
||||||
//@ check-pass
|
//@ check-pass
|
||||||
|
|
||||||
#![feature(associated_const_equality)]
|
#![feature(associated_const_equality)]
|
||||||
|
|
||||||
trait Trait {
|
trait Trait: SuperTrait {
|
||||||
type N;
|
type N;
|
||||||
|
type Q;
|
||||||
|
|
||||||
const N: usize;
|
const N: usize;
|
||||||
}
|
}
|
||||||
|
|
||||||
fn take(_: impl Trait<N = 0, N = ()>) {}
|
trait SuperTrait {
|
||||||
|
const Q: &'static str;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn take0(_: impl Trait<N = 0, N = ()>) {}
|
||||||
|
|
||||||
|
fn take1(_: impl Trait<Q = "...", Q = [()]>) {}
|
||||||
|
|
||||||
fn main() {}
|
fn main() {}
|
||||||
|
@ -1,22 +1,31 @@
|
|||||||
//@ check-pass
|
//@ check-pass
|
||||||
|
|
||||||
#![feature(generic_const_items, associated_const_equality)]
|
#![feature(generic_const_items, associated_const_equality, adt_const_params)]
|
||||||
#![allow(incomplete_features)]
|
#![allow(incomplete_features)]
|
||||||
|
|
||||||
trait Owner {
|
trait Owner {
|
||||||
const C<const N: u32>: u32;
|
const C<const N: u32>: u32;
|
||||||
const K<const N: u32>: u32;
|
const K<const N: u32>: u32;
|
||||||
|
const Q<T>: Maybe<T>;
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Owner for () {
|
impl Owner for () {
|
||||||
const C<const N: u32>: u32 = N;
|
const C<const N: u32>: u32 = N;
|
||||||
const K<const N: u32>: u32 = N + 1;
|
const K<const N: u32>: u32 = N + 1;
|
||||||
|
const Q<T>: Maybe<T> = Maybe::Nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
fn take0<const N: u32>(_: impl Owner<C<N> = { N }>) {}
|
fn take0<const N: u32>(_: impl Owner<C<N> = { N }>) {}
|
||||||
fn take1(_: impl Owner<K<99> = 100>) {}
|
fn take1(_: impl Owner<K<99> = 100>) {}
|
||||||
|
fn take2(_: impl Owner<Q<()> = { Maybe::Just(()) }>) {}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
take0::<128>(());
|
take0::<128>(());
|
||||||
take1(());
|
take1(());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq, Eq, std::marker::ConstParamTy)]
|
||||||
|
enum Maybe<T> {
|
||||||
|
Nothing,
|
||||||
|
Just(T),
|
||||||
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user