2021-04-04 10:55:47 +00:00
|
|
|
//! `TyBuilder`, a helper for building instances of `Ty` and related types.
|
|
|
|
|
|
|
|
use std::iter;
|
|
|
|
|
|
|
|
use chalk_ir::{
|
|
|
|
cast::{Cast, CastTo, Caster},
|
2022-07-03 07:22:10 +00:00
|
|
|
fold::TypeFoldable,
|
2021-04-04 10:55:47 +00:00
|
|
|
interner::HasInterner,
|
2022-10-02 09:39:42 +00:00
|
|
|
AdtId, DebruijnIndex, Scalar,
|
2021-04-04 10:55:47 +00:00
|
|
|
};
|
2022-03-09 18:50:24 +00:00
|
|
|
use hir_def::{
|
2022-09-09 06:20:18 +00:00
|
|
|
builtin_type::BuiltinType, generics::TypeOrConstParamData, ConstParamId, DefWithBodyId,
|
|
|
|
GenericDefId, TraitId, TypeAliasId,
|
2022-03-09 18:50:24 +00:00
|
|
|
};
|
2021-04-04 10:55:47 +00:00
|
|
|
use smallvec::SmallVec;
|
|
|
|
|
|
|
|
use crate::{
|
2022-03-17 11:39:42 +00:00
|
|
|
consteval::unknown_const_as_generic, db::HirDatabase, infer::unify::InferenceTable, primitive,
|
2022-10-02 09:39:42 +00:00
|
|
|
to_assoc_type_id, to_chalk_trait_id, utils::generics, Binders, BoundVar, CallableSig,
|
|
|
|
GenericArg, Interner, ProjectionTy, Substitution, TraitRef, Ty, TyDefId, TyExt, TyKind,
|
|
|
|
ValueTyDefId,
|
2021-04-04 10:55:47 +00:00
|
|
|
};
|
|
|
|
|
2022-03-09 18:50:24 +00:00
|
|
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
|
|
|
pub enum ParamKind {
|
|
|
|
Type,
|
|
|
|
Const(Ty),
|
|
|
|
}
|
|
|
|
|
2021-04-04 11:23:22 +00:00
|
|
|
/// This is a builder for `Ty` or anything that needs a `Substitution`.
|
2021-04-04 10:55:47 +00:00
|
|
|
pub struct TyBuilder<D> {
|
2021-04-04 11:23:22 +00:00
|
|
|
/// The `data` field is used to keep track of what we're building (e.g. an
|
|
|
|
/// ADT, a `TraitRef`, ...).
|
2021-04-04 10:55:47 +00:00
|
|
|
data: D,
|
|
|
|
vec: SmallVec<[GenericArg; 2]>,
|
2022-03-09 18:50:24 +00:00
|
|
|
param_kinds: SmallVec<[ParamKind; 2]>,
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
parent_subst: Substitution,
|
2022-03-09 18:50:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<A> TyBuilder<A> {
|
|
|
|
fn with_data<B>(self, data: B) -> TyBuilder<B> {
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
TyBuilder {
|
|
|
|
data,
|
|
|
|
vec: self.vec,
|
|
|
|
param_kinds: self.param_kinds,
|
|
|
|
parent_subst: self.parent_subst,
|
|
|
|
}
|
2022-03-09 18:50:24 +00:00
|
|
|
}
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<D> TyBuilder<D> {
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
fn new(
|
|
|
|
data: D,
|
|
|
|
param_kinds: SmallVec<[ParamKind; 2]>,
|
|
|
|
parent_subst: Option<Substitution>,
|
|
|
|
) -> Self {
|
|
|
|
let parent_subst = parent_subst.unwrap_or_else(|| Substitution::empty(Interner));
|
|
|
|
Self { data, vec: SmallVec::with_capacity(param_kinds.len()), param_kinds, parent_subst }
|
|
|
|
}
|
|
|
|
|
|
|
|
fn new_empty(data: D) -> Self {
|
|
|
|
TyBuilder::new(data, SmallVec::new(), None)
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn build_internal(self) -> (D, Substitution) {
|
2022-03-09 18:50:24 +00:00
|
|
|
assert_eq!(self.vec.len(), self.param_kinds.len());
|
|
|
|
for (a, e) in self.vec.iter().zip(self.param_kinds.iter()) {
|
|
|
|
self.assert_match_kind(a, e);
|
|
|
|
}
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
let subst = Substitution::from_iter(
|
|
|
|
Interner,
|
|
|
|
self.vec.into_iter().chain(self.parent_subst.iter(Interner).cloned()),
|
|
|
|
);
|
2021-04-04 10:55:47 +00:00
|
|
|
(self.data, subst)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn push(mut self, arg: impl CastTo<GenericArg>) -> Self {
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
assert!(self.remaining() > 0);
|
2022-03-09 18:50:24 +00:00
|
|
|
let arg = arg.cast(Interner);
|
|
|
|
let expected_kind = &self.param_kinds[self.vec.len()];
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
|
2022-03-09 18:50:24 +00:00
|
|
|
let arg_kind = match arg.data(Interner) {
|
|
|
|
chalk_ir::GenericArgData::Ty(_) => ParamKind::Type,
|
|
|
|
chalk_ir::GenericArgData::Lifetime(_) => panic!("Got lifetime in TyBuilder::push"),
|
|
|
|
chalk_ir::GenericArgData::Const(c) => {
|
|
|
|
let c = c.data(Interner);
|
|
|
|
ParamKind::Const(c.ty.clone())
|
|
|
|
}
|
|
|
|
};
|
|
|
|
assert_eq!(*expected_kind, arg_kind);
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
|
2022-03-09 18:50:24 +00:00
|
|
|
self.vec.push(arg);
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
|
2021-04-04 10:55:47 +00:00
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn remaining(&self) -> usize {
|
2022-03-09 18:50:24 +00:00
|
|
|
self.param_kinds.len() - self.vec.len()
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn fill_with_bound_vars(self, debruijn: DebruijnIndex, starting_from: usize) -> Self {
|
2022-03-09 18:50:24 +00:00
|
|
|
// self.fill is inlined to make borrow checker happy
|
|
|
|
let mut this = self;
|
2022-10-02 09:39:42 +00:00
|
|
|
let other = &this.param_kinds[this.vec.len()..];
|
2022-03-09 18:50:24 +00:00
|
|
|
let filler = (starting_from..).zip(other).map(|(idx, kind)| match kind {
|
2022-10-02 09:39:42 +00:00
|
|
|
ParamKind::Type => BoundVar::new(debruijn, idx).to_ty(Interner).cast(Interner),
|
|
|
|
ParamKind::Const(ty) => {
|
|
|
|
BoundVar::new(debruijn, idx).to_const(Interner, ty.clone()).cast(Interner)
|
2022-03-09 18:50:24 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
this.vec.extend(filler.take(this.remaining()).casted(Interner));
|
|
|
|
assert_eq!(this.remaining(), 0);
|
|
|
|
this
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn fill_with_unknown(self) -> Self {
|
2022-03-09 18:50:24 +00:00
|
|
|
// self.fill is inlined to make borrow checker happy
|
|
|
|
let mut this = self;
|
2022-10-02 09:39:42 +00:00
|
|
|
let filler = this.param_kinds[this.vec.len()..].iter().map(|x| match x {
|
|
|
|
ParamKind::Type => TyKind::Error.intern(Interner).cast(Interner),
|
2022-03-09 18:50:24 +00:00
|
|
|
ParamKind::Const(ty) => unknown_const_as_generic(ty.clone()),
|
|
|
|
});
|
|
|
|
this.vec.extend(filler.casted(Interner));
|
|
|
|
assert_eq!(this.remaining(), 0);
|
|
|
|
this
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
|
2022-07-20 13:02:08 +00:00
|
|
|
pub(crate) fn fill_with_inference_vars(self, table: &mut InferenceTable<'_>) -> Self {
|
2022-03-17 11:39:42 +00:00
|
|
|
self.fill(|x| match x {
|
2022-10-02 09:39:42 +00:00
|
|
|
ParamKind::Type => table.new_type_var().cast(Interner),
|
|
|
|
ParamKind::Const(ty) => table.new_const_var(ty.clone()).cast(Interner),
|
2022-03-17 11:39:42 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2022-03-09 18:50:24 +00:00
|
|
|
pub fn fill(mut self, filler: impl FnMut(&ParamKind) -> GenericArg) -> Self {
|
2022-10-02 09:39:42 +00:00
|
|
|
self.vec.extend(self.param_kinds[self.vec.len()..].iter().map(filler));
|
2021-04-04 10:55:47 +00:00
|
|
|
assert_eq!(self.remaining(), 0);
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
2022-03-09 18:50:24 +00:00
|
|
|
fn assert_match_kind(&self, a: &chalk_ir::GenericArg<Interner>, e: &ParamKind) {
|
|
|
|
match (a.data(Interner), e) {
|
|
|
|
(chalk_ir::GenericArgData::Ty(_), ParamKind::Type)
|
|
|
|
| (chalk_ir::GenericArgData::Const(_), ParamKind::Const(_)) => (),
|
|
|
|
_ => panic!("Mismatched kinds: {:?}, {:?}, {:?}", a, self.vec, self.param_kinds),
|
|
|
|
}
|
|
|
|
}
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl TyBuilder<()> {
|
|
|
|
pub fn unit() -> Ty {
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::Tuple(0, Substitution::empty(Interner)).intern(Interner)
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
|
2022-04-07 01:00:33 +00:00
|
|
|
pub fn usize() -> Ty {
|
|
|
|
TyKind::Scalar(chalk_ir::Scalar::Uint(chalk_ir::UintTy::Usize)).intern(Interner)
|
|
|
|
}
|
|
|
|
|
2021-04-04 10:55:47 +00:00
|
|
|
pub fn fn_ptr(sig: CallableSig) -> Ty {
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::Function(sig.to_fn_ptr()).intern(Interner)
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn builtin(builtin: BuiltinType) -> Ty {
|
|
|
|
match builtin {
|
2021-12-19 16:58:39 +00:00
|
|
|
BuiltinType::Char => TyKind::Scalar(Scalar::Char).intern(Interner),
|
|
|
|
BuiltinType::Bool => TyKind::Scalar(Scalar::Bool).intern(Interner),
|
|
|
|
BuiltinType::Str => TyKind::Str.intern(Interner),
|
2021-04-04 10:55:47 +00:00
|
|
|
BuiltinType::Int(t) => {
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::Scalar(Scalar::Int(primitive::int_ty_from_builtin(t))).intern(Interner)
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
BuiltinType::Uint(t) => {
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::Scalar(Scalar::Uint(primitive::uint_ty_from_builtin(t))).intern(Interner)
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
BuiltinType::Float(t) => {
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::Scalar(Scalar::Float(primitive::float_ty_from_builtin(t))).intern(Interner)
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-09-29 12:04:32 +00:00
|
|
|
pub fn slice(argument: Ty) -> Ty {
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::Slice(argument).intern(Interner)
|
2021-09-29 12:04:32 +00:00
|
|
|
}
|
|
|
|
|
2022-03-09 18:50:24 +00:00
|
|
|
pub fn placeholder_subst(db: &dyn HirDatabase, def: impl Into<GenericDefId>) -> Substitution {
|
2021-04-04 11:16:16 +00:00
|
|
|
let params = generics(db.upcast(), def.into());
|
2022-03-09 18:50:24 +00:00
|
|
|
params.placeholder_subst(db)
|
2021-04-04 11:16:16 +00:00
|
|
|
}
|
|
|
|
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
pub fn subst_for_def(
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
def: impl Into<GenericDefId>,
|
|
|
|
parent_subst: Option<Substitution>,
|
|
|
|
) -> TyBuilder<()> {
|
|
|
|
let generics = generics(db.upcast(), def.into());
|
|
|
|
// FIXME: this assertion should hold but some adjustment around
|
|
|
|
// `ValueTyDefId::EnumVariantId` is needed.
|
|
|
|
// assert!(generics.parent_generics().is_some() == parent_subst.is_some());
|
|
|
|
let params = generics
|
|
|
|
.iter_self()
|
|
|
|
.map(|(id, data)| match data {
|
|
|
|
TypeOrConstParamData::TypeParamData(_) => ParamKind::Type,
|
|
|
|
TypeOrConstParamData::ConstParamData(_) => {
|
|
|
|
ParamKind::Const(db.const_param_ty(ConstParamId::from_unchecked(id)))
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.collect();
|
|
|
|
TyBuilder::new((), params, parent_subst)
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
|
2022-09-09 06:20:18 +00:00
|
|
|
/// Creates a `TyBuilder` to build `Substitution` for a generator defined in `parent`.
|
|
|
|
///
|
|
|
|
/// A generator's substitution consists of:
|
|
|
|
/// - resume type of generator
|
|
|
|
/// - yield type of generator ([`Generator::Yield`](std::ops::Generator::Yield))
|
|
|
|
/// - return type of generator ([`Generator::Return`](std::ops::Generator::Return))
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
/// - generic parameters in scope on `parent`
|
2022-09-09 06:20:18 +00:00
|
|
|
/// in this order.
|
|
|
|
///
|
|
|
|
/// This method prepopulates the builder with placeholder substitution of `parent`, so you
|
|
|
|
/// should only push exactly 3 `GenericArg`s before building.
|
|
|
|
pub fn subst_for_generator(db: &dyn HirDatabase, parent: DefWithBodyId) -> TyBuilder<()> {
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
let parent_subst =
|
|
|
|
parent.as_generic_def_id().map(|p| generics(db.upcast(), p).placeholder_subst(db));
|
|
|
|
// These represent resume type, yield type, and return type of generator.
|
|
|
|
let params = std::iter::repeat(ParamKind::Type).take(3).collect();
|
|
|
|
TyBuilder::new((), params, parent_subst)
|
2022-09-09 06:20:18 +00:00
|
|
|
}
|
|
|
|
|
2021-04-04 10:55:47 +00:00
|
|
|
pub fn build(self) -> Substitution {
|
|
|
|
let ((), subst) = self.build_internal();
|
|
|
|
subst
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TyBuilder<hir_def::AdtId> {
|
2022-03-09 18:50:24 +00:00
|
|
|
pub fn adt(db: &dyn HirDatabase, def: hir_def::AdtId) -> TyBuilder<hir_def::AdtId> {
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
TyBuilder::subst_for_def(db, def, None).with_data(def)
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn fill_with_defaults(
|
|
|
|
mut self,
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
mut fallback: impl FnMut() -> Ty,
|
|
|
|
) -> Self {
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
// Note that we're building ADT, so we never have parent generic parameters.
|
2021-04-04 10:55:47 +00:00
|
|
|
let defaults = db.generic_defaults(self.data.into());
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
let dummy_ty = TyKind::Error.intern(Interner).cast(Interner);
|
2021-04-04 10:55:47 +00:00
|
|
|
for default_ty in defaults.iter().skip(self.vec.len()) {
|
2022-10-02 09:39:42 +00:00
|
|
|
// NOTE(skip_binders): we only check if the arg type is error type.
|
|
|
|
if let Some(x) = default_ty.skip_binders().ty(Interner) {
|
2022-03-09 18:50:24 +00:00
|
|
|
if x.is_unknown() {
|
|
|
|
self.vec.push(fallback().cast(Interner));
|
|
|
|
continue;
|
|
|
|
}
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
}
|
|
|
|
// Each default can only depend on the previous parameters.
|
|
|
|
// FIXME: we don't handle const generics here.
|
|
|
|
let subst_so_far = Substitution::from_iter(
|
|
|
|
Interner,
|
|
|
|
self.vec
|
|
|
|
.iter()
|
|
|
|
.cloned()
|
|
|
|
.chain(iter::repeat(dummy_ty.clone()))
|
|
|
|
.take(self.param_kinds.len()),
|
|
|
|
);
|
2022-03-09 18:50:24 +00:00
|
|
|
self.vec.push(default_ty.clone().substitute(Interner, &subst_so_far).cast(Interner));
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
self
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn build(self) -> Ty {
|
|
|
|
let (adt, subst) = self.build_internal();
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::Adt(AdtId(adt), subst).intern(Interner)
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Tuple(usize);
|
|
|
|
impl TyBuilder<Tuple> {
|
|
|
|
pub fn tuple(size: usize) -> TyBuilder<Tuple> {
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
TyBuilder::new(Tuple(size), iter::repeat(ParamKind::Type).take(size).collect(), None)
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn build(self) -> Ty {
|
|
|
|
let (Tuple(size), subst) = self.build_internal();
|
2021-12-19 16:58:39 +00:00
|
|
|
TyKind::Tuple(size, subst).intern(Interner)
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TyBuilder<TraitId> {
|
2022-03-09 18:50:24 +00:00
|
|
|
pub fn trait_ref(db: &dyn HirDatabase, def: TraitId) -> TyBuilder<TraitId> {
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
TyBuilder::subst_for_def(db, def, None).with_data(def)
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn build(self) -> TraitRef {
|
|
|
|
let (trait_id, substitution) = self.build_internal();
|
|
|
|
TraitRef { trait_id: to_chalk_trait_id(trait_id), substitution }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TyBuilder<TypeAliasId> {
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
pub fn assoc_type_projection(
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
def: TypeAliasId,
|
|
|
|
parent_subst: Option<Substitution>,
|
|
|
|
) -> TyBuilder<TypeAliasId> {
|
|
|
|
TyBuilder::subst_for_def(db, def, parent_subst).with_data(def)
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn build(self) -> ProjectionTy {
|
|
|
|
let (type_alias, substitution) = self.build_internal();
|
|
|
|
ProjectionTy { associated_ty_id: to_assoc_type_id(type_alias), substitution }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-03 07:22:10 +00:00
|
|
|
impl<T: HasInterner<Interner = Interner> + TypeFoldable<Interner>> TyBuilder<Binders<T>> {
|
|
|
|
pub fn build(self) -> T {
|
2021-04-04 10:55:47 +00:00
|
|
|
let (b, subst) = self.build_internal();
|
2021-12-19 16:58:39 +00:00
|
|
|
b.substitute(Interner, &subst)
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TyBuilder<Binders<Ty>> {
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
pub fn def_ty(
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
def: TyDefId,
|
|
|
|
parent_subst: Option<Substitution>,
|
|
|
|
) -> TyBuilder<Binders<Ty>> {
|
|
|
|
let poly_ty = db.ty(def);
|
|
|
|
let id: GenericDefId = match def {
|
|
|
|
TyDefId::BuiltinType(_) => {
|
|
|
|
assert!(parent_subst.is_none());
|
|
|
|
return TyBuilder::new_empty(poly_ty);
|
|
|
|
}
|
|
|
|
TyDefId::AdtId(id) => id.into(),
|
|
|
|
TyDefId::TypeAliasId(id) => id.into(),
|
|
|
|
};
|
|
|
|
TyBuilder::subst_for_def(db, id, parent_subst).with_data(poly_ty)
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn impl_self_ty(db: &dyn HirDatabase, def: hir_def::ImplId) -> TyBuilder<Binders<Ty>> {
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
TyBuilder::subst_for_def(db, def, None).with_data(db.impl_self_ty(def))
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
|
Change generic parameter/argument order
This commit "inverts" the order of generic parameters/arguments of an
item and its parent. This is to fulfill chalk's expectation on the
order of `Substitution` for generic associated types and it's one step
forward for their support (hopefully).
Although chalk doesn't put any constraint on the order of `Substitution`
for other items, it feels natural to get everything aligned rather than
special casing GATs.
One complication is that `TyBuilder` now demands its users to pass in
parent's `Substitution` upon construction unless it's obvious that the
the item has no parent (e.g. an ADT never has parent). All users
*should* already know the parent of the item in question, and without
this, it cannot be easily reasoned about whether we're pushing the
argument for the item or for its parent.
Quick comparison of how this commit changes `Substitution`:
```rust
trait Trait<TP, const CP: usize> {
type Type<TC, const CC: usize> = ();
fn f<TC, const CC: usize>() {}
}
```
- before this commit: `[Self, TP, CP, TC, CC]` for each trait item
- after this commit: `[TC, CC, Self, TP, CP]` for each trait item
2022-10-02 12:13:21 +00:00
|
|
|
pub fn value_ty(
|
|
|
|
db: &dyn HirDatabase,
|
|
|
|
def: ValueTyDefId,
|
|
|
|
parent_subst: Option<Substitution>,
|
|
|
|
) -> TyBuilder<Binders<Ty>> {
|
|
|
|
let poly_value_ty = db.value_ty(def);
|
|
|
|
let id = match def.to_generic_def_id() {
|
|
|
|
Some(id) => id,
|
|
|
|
None => {
|
|
|
|
// static items
|
|
|
|
assert!(parent_subst.is_none());
|
|
|
|
return TyBuilder::new_empty(poly_value_ty);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
TyBuilder::subst_for_def(db, id, parent_subst).with_data(poly_value_ty)
|
2021-04-04 10:55:47 +00:00
|
|
|
}
|
|
|
|
}
|