mirror of
				https://github.com/rust-lang/rust-analyzer.git
				synced 2025-10-25 01:17:45 +00:00 
			
		
		
		
	Merge pull request #19182 from ShoyuVanilla/issue-19177
fix: Binding wrong associated type when lowering bounds like `T: Trait<Assoc = U>`
This commit is contained in:
		
						commit
						9df88ff0f6
					
				
					 2 changed files with 47 additions and 7 deletions
				
			
		|  | @ -10,7 +10,7 @@ use hir_def::{ | |||
|     generics::{TypeParamProvenance, WherePredicate, WherePredicateTypeTarget}, | ||||
|     path::{GenericArg, GenericArgs, Path, PathSegment, PathSegments}, | ||||
|     resolver::{ResolveValueResult, TypeNs, ValueNs}, | ||||
|     type_ref::{TypeBound, TypeRef}, | ||||
|     type_ref::{TypeBound, TypeRef, TypesMap}, | ||||
|     GenericDefId, GenericParamId, ItemContainerId, Lookup, TraitId, | ||||
| }; | ||||
| use smallvec::SmallVec; | ||||
|  | @ -838,15 +838,21 @@ impl<'a, 'b> PathLoweringContext<'a, 'b> { | |||
|                         (_, ImplTraitLoweringMode::Param | ImplTraitLoweringMode::Variable) => { | ||||
|                             // Find the generic index for the target of our `bound`
 | ||||
|                             let target_param_idx = | ||||
|                                 self.ctx.resolver.where_predicates_in_scope().find_map(|(p, _)| { | ||||
|                                     match p { | ||||
|                                 self.ctx.resolver.where_predicates_in_scope().find_map( | ||||
|                                     |(p, (_, types_map))| match p { | ||||
|                                         WherePredicate::TypeBound { | ||||
|                                             target: WherePredicateTypeTarget::TypeOrConstParam(idx), | ||||
|                                             bound: b, | ||||
|                                         } if b == bound => Some(idx), | ||||
|                                         _ => None, | ||||
|                                         } if std::ptr::eq::<TypesMap>( | ||||
|                                             self.ctx.types_map, | ||||
|                                             types_map, | ||||
|                                         ) && bound == b => | ||||
|                                         { | ||||
|                                             Some(idx) | ||||
|                                         } | ||||
|                                 }); | ||||
|                                         _ => None, | ||||
|                                     }, | ||||
|                                 ); | ||||
|                             let ty = if let Some(target_param_idx) = target_param_idx { | ||||
|                                 let mut counter = 0; | ||||
|                                 let generics = self.ctx.generics().expect("generics in scope"); | ||||
|  |  | |||
|  | @ -3,7 +3,7 @@ use span::{Edition, EditionedFileId}; | |||
| use syntax::{TextRange, TextSize}; | ||||
| use test_fixture::WithFixture; | ||||
| 
 | ||||
| use crate::{db::HirDatabase, test_db::TestDB, Interner, Substitution}; | ||||
| use crate::{db::HirDatabase, mir::MirLowerError, test_db::TestDB, Interner, Substitution}; | ||||
| 
 | ||||
| use super::{interpret_mir, MirEvalError}; | ||||
| 
 | ||||
|  | @ -84,6 +84,16 @@ fn check_panic(#[rust_analyzer::rust_fixture] ra_fixture: &str, expected_panic: | |||
|     assert_eq!(e.is_panic().unwrap_or_else(|| panic!("unexpected error: {e:?}")), expected_panic); | ||||
| } | ||||
| 
 | ||||
| fn check_error_with( | ||||
|     #[rust_analyzer::rust_fixture] ra_fixture: &str, | ||||
|     expect_err: impl FnOnce(MirEvalError) -> bool, | ||||
| ) { | ||||
|     let (db, file_ids) = TestDB::with_many_files(ra_fixture); | ||||
|     let file_id = *file_ids.last().unwrap(); | ||||
|     let e = eval_main(&db, file_id).unwrap_err(); | ||||
|     assert!(expect_err(e)); | ||||
| } | ||||
| 
 | ||||
| #[test] | ||||
| fn function_with_extern_c_abi() { | ||||
|     check_pass( | ||||
|  | @ -945,3 +955,27 @@ fn main() { | |||
| "#,
 | ||||
|     ); | ||||
| } | ||||
| 
 | ||||
| #[test] | ||||
| fn regression_19177() { | ||||
|     check_error_with( | ||||
|         r#" | ||||
| //- minicore: copy
 | ||||
| trait Foo {} | ||||
| trait Bar {} | ||||
| trait Baz {} | ||||
| trait Qux { | ||||
|     type Assoc; | ||||
| } | ||||
| 
 | ||||
| fn main<'a, T: Foo + Bar + Baz>( | ||||
|     x: &T, | ||||
|     y: (), | ||||
|     z: &'a dyn Qux<Assoc = T>, | ||||
|     w: impl Foo + Bar, | ||||
| ) { | ||||
| } | ||||
| "#,
 | ||||
|         |e| matches!(e, MirEvalError::MirLowerError(_, MirLowerError::GenericArgNotProvided(..))), | ||||
|     ); | ||||
| } | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Chayim Refael Friedman
						Chayim Refael Friedman