Properly check constrainedness of gen params in the presence of weak alias types
This commit is contained in:
parent
8677d64c72
commit
fde4556785
9 changed files with 105 additions and 17 deletions
|
@ -307,7 +307,7 @@ fn gather_explicit_predicates_of(tcx: TyCtxt<'_>, def_id: LocalDefId) -> ty::Gen
|
||||||
tcx,
|
tcx,
|
||||||
&mut predicates,
|
&mut predicates,
|
||||||
trait_ref,
|
trait_ref,
|
||||||
&mut cgp::parameters_for_impl(self_ty, trait_ref),
|
&mut cgp::parameters_for_impl(tcx, self_ty, trait_ref),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,5 @@
|
||||||
use rustc_data_structures::fx::FxHashSet;
|
use rustc_data_structures::fx::FxHashSet;
|
||||||
|
use rustc_data_structures::stack::ensure_sufficient_stack;
|
||||||
use rustc_middle::ty::visit::{TypeSuperVisitable, TypeVisitable, TypeVisitor};
|
use rustc_middle::ty::visit::{TypeSuperVisitable, TypeVisitable, TypeVisitor};
|
||||||
use rustc_middle::ty::{self, Ty, TyCtxt};
|
use rustc_middle::ty::{self, Ty, TyCtxt};
|
||||||
use rustc_span::Span;
|
use rustc_span::Span;
|
||||||
|
@ -27,12 +28,13 @@ impl From<ty::ParamConst> for Parameter {
|
||||||
|
|
||||||
/// Returns the set of parameters constrained by the impl header.
|
/// Returns the set of parameters constrained by the impl header.
|
||||||
pub fn parameters_for_impl<'tcx>(
|
pub fn parameters_for_impl<'tcx>(
|
||||||
|
tcx: TyCtxt<'tcx>,
|
||||||
impl_self_ty: Ty<'tcx>,
|
impl_self_ty: Ty<'tcx>,
|
||||||
impl_trait_ref: Option<ty::TraitRef<'tcx>>,
|
impl_trait_ref: Option<ty::TraitRef<'tcx>>,
|
||||||
) -> FxHashSet<Parameter> {
|
) -> FxHashSet<Parameter> {
|
||||||
let vec = match impl_trait_ref {
|
let vec = match impl_trait_ref {
|
||||||
Some(tr) => parameters_for(&tr, false),
|
Some(tr) => parameters_for(tcx, &tr, false),
|
||||||
None => parameters_for(&impl_self_ty, false),
|
None => parameters_for(tcx, &impl_self_ty, false),
|
||||||
};
|
};
|
||||||
vec.into_iter().collect()
|
vec.into_iter().collect()
|
||||||
}
|
}
|
||||||
|
@ -43,26 +45,47 @@ pub fn parameters_for_impl<'tcx>(
|
||||||
/// of parameters whose values are needed in order to constrain `ty` - these
|
/// of parameters whose values are needed in order to constrain `ty` - these
|
||||||
/// differ, with the latter being a superset, in the presence of projections.
|
/// differ, with the latter being a superset, in the presence of projections.
|
||||||
pub fn parameters_for<'tcx>(
|
pub fn parameters_for<'tcx>(
|
||||||
|
tcx: TyCtxt<'tcx>,
|
||||||
t: &impl TypeVisitable<TyCtxt<'tcx>>,
|
t: &impl TypeVisitable<TyCtxt<'tcx>>,
|
||||||
include_nonconstraining: bool,
|
include_nonconstraining: bool,
|
||||||
) -> Vec<Parameter> {
|
) -> Vec<Parameter> {
|
||||||
let mut collector = ParameterCollector { parameters: vec![], include_nonconstraining };
|
let mut collector =
|
||||||
|
ParameterCollector { tcx, parameters: vec![], include_nonconstraining, depth: 0 };
|
||||||
t.visit_with(&mut collector);
|
t.visit_with(&mut collector);
|
||||||
collector.parameters
|
collector.parameters
|
||||||
}
|
}
|
||||||
|
|
||||||
struct ParameterCollector {
|
struct ParameterCollector<'tcx> {
|
||||||
|
tcx: TyCtxt<'tcx>,
|
||||||
parameters: Vec<Parameter>,
|
parameters: Vec<Parameter>,
|
||||||
include_nonconstraining: bool,
|
include_nonconstraining: bool,
|
||||||
|
depth: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'tcx> TypeVisitor<TyCtxt<'tcx>> for ParameterCollector {
|
impl<'tcx> TypeVisitor<TyCtxt<'tcx>> for ParameterCollector<'tcx> {
|
||||||
fn visit_ty(&mut self, t: Ty<'tcx>) -> ControlFlow<Self::BreakTy> {
|
fn visit_ty(&mut self, t: Ty<'tcx>) -> ControlFlow<Self::BreakTy> {
|
||||||
match *t.kind() {
|
match *t.kind() {
|
||||||
ty::Alias(..) if !self.include_nonconstraining => {
|
ty::Alias(ty::Projection | ty::Inherent | ty::Opaque, _)
|
||||||
// projections are not injective
|
if !self.include_nonconstraining =>
|
||||||
|
{
|
||||||
|
// Projections are not injective in general.
|
||||||
return ControlFlow::Continue(());
|
return ControlFlow::Continue(());
|
||||||
}
|
}
|
||||||
|
ty::Alias(ty::Weak, alias) if !self.include_nonconstraining => {
|
||||||
|
if !self.tcx.recursion_limit().value_within_limit(self.depth) {
|
||||||
|
// Other constituent types may still constrain some generic params, consider
|
||||||
|
// `<T> (Overflow, T)` for example. Therefore we want to continue instead of
|
||||||
|
// breaking. Only affects diagnostics.
|
||||||
|
return ControlFlow::Continue(());
|
||||||
|
}
|
||||||
|
self.depth += 1;
|
||||||
|
return ensure_sufficient_stack(|| {
|
||||||
|
self.tcx
|
||||||
|
.type_of(alias.def_id)
|
||||||
|
.instantiate(self.tcx, alias.args)
|
||||||
|
.visit_with(self)
|
||||||
|
});
|
||||||
|
}
|
||||||
ty::Param(data) => {
|
ty::Param(data) => {
|
||||||
self.parameters.push(Parameter::from(data));
|
self.parameters.push(Parameter::from(data));
|
||||||
}
|
}
|
||||||
|
@ -82,7 +105,7 @@ impl<'tcx> TypeVisitor<TyCtxt<'tcx>> for ParameterCollector {
|
||||||
fn visit_const(&mut self, c: ty::Const<'tcx>) -> ControlFlow<Self::BreakTy> {
|
fn visit_const(&mut self, c: ty::Const<'tcx>) -> ControlFlow<Self::BreakTy> {
|
||||||
match c.kind() {
|
match c.kind() {
|
||||||
ty::ConstKind::Unevaluated(..) if !self.include_nonconstraining => {
|
ty::ConstKind::Unevaluated(..) if !self.include_nonconstraining => {
|
||||||
// Constant expressions are not injective
|
// Constant expressions are not injective in general.
|
||||||
return c.ty().visit_with(self);
|
return c.ty().visit_with(self);
|
||||||
}
|
}
|
||||||
ty::ConstKind::Param(data) => {
|
ty::ConstKind::Param(data) => {
|
||||||
|
@ -201,12 +224,12 @@ pub fn setup_constraining_predicates<'tcx>(
|
||||||
// `<<T as Bar>::Baz as Iterator>::Output = <U as Iterator>::Output`
|
// `<<T as Bar>::Baz as Iterator>::Output = <U as Iterator>::Output`
|
||||||
// Then the projection only applies if `T` is known, but it still
|
// Then the projection only applies if `T` is known, but it still
|
||||||
// does not determine `U`.
|
// does not determine `U`.
|
||||||
let inputs = parameters_for(&projection.projection_ty, true);
|
let inputs = parameters_for(tcx, &projection.projection_ty, true);
|
||||||
let relies_only_on_inputs = inputs.iter().all(|p| input_parameters.contains(p));
|
let relies_only_on_inputs = inputs.iter().all(|p| input_parameters.contains(p));
|
||||||
if !relies_only_on_inputs {
|
if !relies_only_on_inputs {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
input_parameters.extend(parameters_for(&projection.term, false));
|
input_parameters.extend(parameters_for(tcx, &projection.term, false));
|
||||||
} else {
|
} else {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
|
@ -94,7 +94,7 @@ fn enforce_impl_params_are_constrained(
|
||||||
let impl_predicates = tcx.predicates_of(impl_def_id);
|
let impl_predicates = tcx.predicates_of(impl_def_id);
|
||||||
let impl_trait_ref = tcx.impl_trait_ref(impl_def_id).map(ty::EarlyBinder::instantiate_identity);
|
let impl_trait_ref = tcx.impl_trait_ref(impl_def_id).map(ty::EarlyBinder::instantiate_identity);
|
||||||
|
|
||||||
let mut input_parameters = cgp::parameters_for_impl(impl_self_ty, impl_trait_ref);
|
let mut input_parameters = cgp::parameters_for_impl(tcx, impl_self_ty, impl_trait_ref);
|
||||||
cgp::identify_constrained_generic_params(
|
cgp::identify_constrained_generic_params(
|
||||||
tcx,
|
tcx,
|
||||||
impl_predicates,
|
impl_predicates,
|
||||||
|
@ -111,7 +111,7 @@ fn enforce_impl_params_are_constrained(
|
||||||
match item.kind {
|
match item.kind {
|
||||||
ty::AssocKind::Type => {
|
ty::AssocKind::Type => {
|
||||||
if item.defaultness(tcx).has_value() {
|
if item.defaultness(tcx).has_value() {
|
||||||
cgp::parameters_for(&tcx.type_of(def_id).instantiate_identity(), true)
|
cgp::parameters_for(tcx, &tcx.type_of(def_id).instantiate_identity(), true)
|
||||||
} else {
|
} else {
|
||||||
vec![]
|
vec![]
|
||||||
}
|
}
|
||||||
|
|
|
@ -266,15 +266,15 @@ fn unconstrained_parent_impl_args<'tcx>(
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
unconstrained_parameters.extend(cgp::parameters_for(&projection_ty, true));
|
unconstrained_parameters.extend(cgp::parameters_for(tcx, &projection_ty, true));
|
||||||
|
|
||||||
for param in cgp::parameters_for(&projected_ty, false) {
|
for param in cgp::parameters_for(tcx, &projected_ty, false) {
|
||||||
if !unconstrained_parameters.contains(¶m) {
|
if !unconstrained_parameters.contains(¶m) {
|
||||||
constrained_params.insert(param.0);
|
constrained_params.insert(param.0);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
unconstrained_parameters.extend(cgp::parameters_for(&projected_ty, true));
|
unconstrained_parameters.extend(cgp::parameters_for(tcx, &projected_ty, true));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -312,7 +312,7 @@ fn check_duplicate_params<'tcx>(
|
||||||
parent_args: &Vec<GenericArg<'tcx>>,
|
parent_args: &Vec<GenericArg<'tcx>>,
|
||||||
span: Span,
|
span: Span,
|
||||||
) -> Result<(), ErrorGuaranteed> {
|
) -> Result<(), ErrorGuaranteed> {
|
||||||
let mut base_params = cgp::parameters_for(parent_args, true);
|
let mut base_params = cgp::parameters_for(tcx, parent_args, true);
|
||||||
base_params.sort_by_key(|param| param.0);
|
base_params.sort_by_key(|param| param.0);
|
||||||
if let (_, [duplicate, ..]) = base_params.partition_dedup() {
|
if let (_, [duplicate, ..]) = base_params.partition_dedup() {
|
||||||
let param = impl1_args[duplicate.0 as usize];
|
let param = impl1_args[duplicate.0 as usize];
|
||||||
|
|
27
tests/ui/lazy-type-alias/constrained-params.rs
Normal file
27
tests/ui/lazy-type-alias/constrained-params.rs
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
//@ check-pass
|
||||||
|
|
||||||
|
#![feature(lazy_type_alias)]
|
||||||
|
#![allow(incomplete_features)]
|
||||||
|
|
||||||
|
type Injective<T> = Local<T>;
|
||||||
|
struct Local<T>(T);
|
||||||
|
|
||||||
|
impl<T> Injective<T> {
|
||||||
|
fn take(_: T) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
trait Trait {
|
||||||
|
type Out;
|
||||||
|
fn produce() -> Self::Out;
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T: Default> Trait for Injective<T> {
|
||||||
|
type Out = T;
|
||||||
|
fn produce() -> Self::Out { T::default() }
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
Injective::take(0);
|
||||||
|
let _: String = Injective::produce();
|
||||||
|
let _: bool = Local::produce();
|
||||||
|
}
|
|
@ -0,0 +1,8 @@
|
||||||
|
#![feature(lazy_type_alias)]
|
||||||
|
#![allow(incomplete_features)]
|
||||||
|
|
||||||
|
impl<T> Loop<T> {} //~ ERROR the type parameter `T` is not constrained
|
||||||
|
|
||||||
|
type Loop<T> = Loop<T>;
|
||||||
|
|
||||||
|
fn main() {}
|
|
@ -0,0 +1,9 @@
|
||||||
|
error[E0207]: the type parameter `T` is not constrained by the impl trait, self type, or predicates
|
||||||
|
--> $DIR/unconstrained-param-due-to-overflow.rs:4:6
|
||||||
|
|
|
||||||
|
LL | impl<T> Loop<T> {}
|
||||||
|
| ^ unconstrained type parameter
|
||||||
|
|
||||||
|
error: aborting due to 1 previous error
|
||||||
|
|
||||||
|
For more information about this error, try `rustc --explain E0207`.
|
12
tests/ui/lazy-type-alias/unconstrained-params.rs
Normal file
12
tests/ui/lazy-type-alias/unconstrained-params.rs
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
#![feature(lazy_type_alias)]
|
||||||
|
#![allow(incomplete_features)]
|
||||||
|
|
||||||
|
impl<T> NotInjective<T> {} //~ ERROR the type parameter `T` is not constrained
|
||||||
|
|
||||||
|
type NotInjective<T: ?Sized> = Local<<T as Discard>::Out>;
|
||||||
|
struct Local<T>(T);
|
||||||
|
|
||||||
|
trait Discard { type Out; }
|
||||||
|
impl<T: ?Sized> Discard for T { type Out = (); }
|
||||||
|
|
||||||
|
fn main() {}
|
9
tests/ui/lazy-type-alias/unconstrained-params.stderr
Normal file
9
tests/ui/lazy-type-alias/unconstrained-params.stderr
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
error[E0207]: the type parameter `T` is not constrained by the impl trait, self type, or predicates
|
||||||
|
--> $DIR/unconstrained-params.rs:4:6
|
||||||
|
|
|
||||||
|
LL | impl<T> NotInjective<T> {}
|
||||||
|
| ^ unconstrained type parameter
|
||||||
|
|
||||||
|
error: aborting due to 1 previous error
|
||||||
|
|
||||||
|
For more information about this error, try `rustc --explain E0207`.
|
Loading…
Add table
Add a link
Reference in a new issue