Skip to content

Commit

Permalink
Auto merge of #13021 - N3xed:fix-gat-panics, r=N3xed
Browse files Browse the repository at this point in the history
fix: Fix panics on GATs involving const generics

This workaround avoids constant crashing of rust analyzer when using GATs with const generics,
even when the const generics are only on the `impl` block.

The workaround treats GATs as non-existing if either itself or the parent has const generics and
removes relevant panicking code-paths.
~~Additionally, I've added the `is_gat` field to `utils::Generics` that determines whether missing
generics params crashes rust analyzer. Another solution would have been to remove all panics
from the relevant code path regardless of whether the generics are on a GAT, but this could
change behavior outside of GATs.~~

Fixes #11989, fixes #12193
  • Loading branch information
bors committed Aug 21, 2022
2 parents a670ff8 + ad7a1ed commit 1b8e1c6
Show file tree
Hide file tree
Showing 3 changed files with 39 additions and 9 deletions.
9 changes: 8 additions & 1 deletion crates/hir-ty/src/lower.rs
Original file line number Diff line number Diff line change
Expand Up @@ -479,7 +479,14 @@ impl<'a> TyLoweringContext<'a> {
TyKind::Placeholder(to_placeholder_idx(self.db, param_id.into()))
}
ParamLoweringMode::Variable => {
let idx = generics.param_idx(param_id.into()).expect("matching generics");
let idx = match generics.param_idx(param_id.into()) {
None => {
never!("no matching generics");
return (TyKind::Error.intern(Interner), None);
}
Some(idx) => idx,
};

TyKind::BoundVar(BoundVar::new(self.in_binders, idx))
}
}
Expand Down
21 changes: 21 additions & 0 deletions crates/hir-ty/src/tests/regression.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1526,6 +1526,27 @@ unsafe impl Storage for InlineStorage {
);
}

#[test]
fn gat_crash_3() {
cov_mark::check!(ignore_gats);
check_no_mismatches(
r#"
trait Collection {
type Item;
type Member<T>: Collection<Item = T>;
fn add(&mut self, value: Self::Item) -> Result<(), Self::Error>;
}
struct ConstGen<T, const N: usize> {
data: [T; N],
}
impl<T, const N: usize> Collection for ConstGen<T, N> {
type Item = T;
type Member<U> = ConstGen<U, N>;
}
"#,
);
}

#[test]
fn cfgd_out_self_param() {
cov_mark::check!(cfgd_out_self_param);
Expand Down
18 changes: 10 additions & 8 deletions crates/hir-ty/src/utils.rs
Original file line number Diff line number Diff line change
Expand Up @@ -176,10 +176,16 @@ pub(crate) fn generics(db: &dyn DefDatabase, def: GenericDefId) -> Generics {
let parent_generics = parent_generic_def(db, def).map(|def| Box::new(generics(db, def)));
if parent_generics.is_some() && matches!(def, GenericDefId::TypeAliasId(_)) {
let params = db.generic_params(def);
let parent_params = &parent_generics.as_ref().unwrap().params;
let has_consts =
params.iter().any(|(_, x)| matches!(x, TypeOrConstParamData::ConstParamData(_)));
return if has_consts {
// XXX: treat const generic associated types as not existing to avoid crashes (#11769)
let parent_has_consts =
parent_params.iter().any(|(_, x)| matches!(x, TypeOrConstParamData::ConstParamData(_)));
return if has_consts || parent_has_consts {
// XXX: treat const generic associated types as not existing to avoid crashes
// (#11769, #12193)
// Note: also crashes when the parent has const generics (also even if the GAT
// doesn't use them), see `tests::regression::gat_crash_3` for an example.
//
// Chalk expects the inner associated type's parameters to come
// *before*, not after the trait's generics as we've always done it.
Expand Down Expand Up @@ -264,12 +270,8 @@ impl Generics {

fn find_param(&self, param: TypeOrConstParamId) -> Option<(usize, &TypeOrConstParamData)> {
if param.parent == self.def {
let (idx, (_local_id, data)) = self
.params
.iter()
.enumerate()
.find(|(_, (idx, _))| *idx == param.local_id)
.unwrap();
let (idx, (_local_id, data)) =
self.params.iter().enumerate().find(|(_, (idx, _))| *idx == param.local_id)?;
let parent_len = self.parent_generics().map_or(0, Generics::len);
Some((parent_len + idx, data))
} else {
Expand Down

0 comments on commit 1b8e1c6

Please sign in to comment.