Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix(experimental elaborator): Fix duplicate resolve_type on self type and don't leak a trait impl's generics #5102

Merged
merged 53 commits into from
May 28, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
53 commits
Select commit Hold shift + click to select a range
d0afc12
Add elaborator
jfecher May 2, 2024
1c433ec
Merge branch 'master' into jf/elaborator
jfecher May 3, 2024
225303f
More expressions
jfecher May 6, 2024
29e67a1
Finish each expression kind
jfecher May 7, 2024
0177c12
Merge branch 'master' into jf/elaborator
jfecher May 7, 2024
6b30028
Add compiler flag
jfecher May 8, 2024
7550fb8
Move expressions into their own file
jfecher May 8, 2024
39d23dc
Merge branch 'jf/elaborator' into jf/integrate-elaborator
jfecher May 8, 2024
2e77d87
Connect elaborator
jfecher May 8, 2024
e542d32
Code review
jfecher May 8, 2024
fd287df
Finish removing Scope
jfecher May 8, 2024
73dc473
Merge branch 'jf/elaborator' into jf/integrate-elaborator
jfecher May 8, 2024
9614b87
Finish functions
jfecher May 8, 2024
b44c98a
Add should_panic
jfecher May 8, 2024
9579646
clippy
jfecher May 8, 2024
e4d5f61
Fix test
jfecher May 8, 2024
556e6ef
Fix test
jfecher May 9, 2024
7487b03
Elaborate impls
jfecher May 9, 2024
82d5871
Fix yet another missed line
jfecher May 9, 2024
a3872e0
Merge branch 'jf/integrate-elaborator' into jf/elaborator-impls
jfecher May 9, 2024
3fb05dc
Merge branch 'master' into jf/integrate-elaborator
jfecher May 9, 2024
4fa4677
Merge branch 'jf/integrate-elaborator' into jf/elaborator-impls
jfecher May 9, 2024
e9e69f4
Resolve weird git merge
jfecher May 9, 2024
d7d91ae
Wrong arg order
jfecher May 9, 2024
e9afa1a
Merge branch 'jf/integrate-elaborator' into jf/elaborator-impls
jfecher May 9, 2024
13564f8
Merge branch 'master' into jf/elaborator-impls
jfecher May 9, 2024
75c6451
Code review
jfecher May 10, 2024
bbecf41
Start work on traits + types (+ globals)?
jfecher May 10, 2024
0029558
Merge branch 'master' into jf/elaborator-types
jfecher May 21, 2024
7a37bf3
Add globals
jfecher May 21, 2024
464d58c
Format
jfecher May 21, 2024
ba893ae
Merge branch 'jf/elaborator-types' into jf/elaborator-globals
jfecher May 21, 2024
397fe78
Copy over comments from dc_crate
jfecher May 22, 2024
afcf26e
Remove unneeded comment
jfecher May 22, 2024
1f3e79b
Merge branch 'master' into jf/elaborator-globals
jfecher May 22, 2024
8522936
Re-add accidentally removed collect_traits call
jfecher May 22, 2024
bbb7228
Fix panic in the elaborator
jfecher May 22, 2024
1f9b6d0
Merge branch 'master' into jf/elaborator-fixes
jfecher May 22, 2024
78e6a27
Undo unnecessary change
jfecher May 22, 2024
3b59749
Remove more unnecessary code
jfecher May 22, 2024
bf999d9
Remove unnecessary import
jfecher May 22, 2024
7be1561
Remove outdated assert
jfecher May 22, 2024
f48280d
Merge branch 'master' into jf/elaborator-fixes
jfecher May 22, 2024
fadc085
Fix issue with function generics
jfecher May 23, 2024
f0d249a
Clippy
jfecher May 23, 2024
68ccf9c
Add missed field in test
jfecher May 23, 2024
aa6d581
Fix structs not recovering generics length
jfecher May 23, 2024
f519ced
Don't check return type for trait methods
jfecher May 23, 2024
4e6490c
Fix more errors
jfecher May 24, 2024
38f20e9
Clippy
jfecher May 24, 2024
31e4228
fix(experimental elaborator): Avoid defining globals twice (#5103)
jfecher May 28, 2024
0fed42c
fix(experimental elaborator): Move code to declare trait impls earlie…
jfecher May 28, 2024
6267857
Merge branch 'master' into jf/elaborator-fixes4
jfecher May 28, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
153 changes: 63 additions & 90 deletions compiler/noirc_frontend/src/elaborator/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -223,8 +223,8 @@ impl<'context> Elaborator<'context> {
//
// These are resolved after trait impls so that struct methods are chosen
// over trait methods if there are name conflicts.
for ((typ, module), impls) in &mut items.impls {
this.collect_impls(typ, *module, impls);
for ((_self_type, module), impls) in &mut items.impls {
this.collect_impls(*module, impls);
}

// We must wait to resolve non-literal globals until after we resolve structs since struct
Expand Down Expand Up @@ -284,8 +284,6 @@ impl<'context> Elaborator<'context> {
self.scopes.start_function();
self.current_item = Some(DependencyId::Function(id));

// Check whether the function has globals in the local module and add them to the scope
self.resolve_local_globals();
self.trait_bounds = function.def.where_clause.clone();

if function.def.is_unconstrained {
Expand Down Expand Up @@ -474,18 +472,6 @@ impl<'context> Elaborator<'context> {
None
}

fn resolve_local_globals(&mut self) {
let globals = vecmap(self.interner.get_all_globals(), |global| {
(global.id, global.local_id, global.ident.clone())
});
for (id, local_module_id, name) in globals {
if local_module_id == self.local_module {
let definition = DefinitionKind::Global(id);
self.add_global_variable_decl(name, definition);
}
}
}

/// TODO: This is currently only respected for generic free functions
/// there's a bunch of other places where trait constraints can pop up
fn resolve_trait_constraints(
Expand All @@ -494,21 +480,20 @@ impl<'context> Elaborator<'context> {
) -> Vec<TraitConstraint> {
where_clause
.iter()
.cloned()
.filter_map(|constraint| self.resolve_trait_constraint(constraint))
.collect()
}

pub fn resolve_trait_constraint(
&mut self,
constraint: UnresolvedTraitConstraint,
constraint: &UnresolvedTraitConstraint,
) -> Option<TraitConstraint> {
let typ = self.resolve_type(constraint.typ);
let typ = self.resolve_type(constraint.typ.clone());
let trait_generics =
vecmap(constraint.trait_bound.trait_generics, |typ| self.resolve_type(typ));
vecmap(&constraint.trait_bound.trait_generics, |typ| self.resolve_type(typ.clone()));

let span = constraint.trait_bound.trait_path.span();
let the_trait = self.lookup_trait_or_error(constraint.trait_bound.trait_path)?;
let the_trait = self.lookup_trait_or_error(constraint.trait_bound.trait_path.clone())?;
let trait_id = the_trait.id;

let expected_generics = the_trait.generics.len();
Expand Down Expand Up @@ -548,9 +533,6 @@ impl<'context> Elaborator<'context> {
self.scopes.start_function();
self.current_item = Some(DependencyId::Function(func_id));

// Check whether the function has globals in the local module and add them to the scope
self.resolve_local_globals();

let location = Location::new(func.name_ident().span(), self.file);
let id = self.interner.function_definition_id(func_id);
let name_ident = HirIdent::non_trait_method(id, location);
Expand Down Expand Up @@ -864,40 +846,69 @@ impl<'context> Elaborator<'context> {
self.file = trait_impl.file_id;
self.local_module = trait_impl.module_id;

let unresolved_type = trait_impl.object_type;
let self_type_span = unresolved_type.span;
let old_generics_length = self.generics.len();
self.generics = trait_impl.resolved_generics;
self.current_trait_impl = trait_impl.impl_id;

let trait_generics =
vecmap(&trait_impl.trait_generics, |generic| self.resolve_type(generic.clone()));
self.elaborate_functions(trait_impl.methods);

let self_type = trait_impl.resolved_object_type.unwrap_or(Type::Error);
let impl_id =
trait_impl.impl_id.expect("An impls' id should be set during define_function_metas");
self.self_type = None;
self.current_trait_impl = None;
self.generics.clear();
}

self.current_trait_impl = trait_impl.impl_id;
fn collect_impls(
&mut self,
module: LocalModuleId,
impls: &mut [(Vec<Ident>, Span, UnresolvedFunctions)],
) {
self.local_module = module;

let methods = trait_impl.methods.function_ids();
for (generics, span, unresolved) in impls {
self.file = unresolved.file_id;
let old_generic_count = self.generics.len();
self.add_generics(generics);
self.declare_methods_on_struct(false, unresolved, *span);
self.generics.truncate(old_generic_count);
}
}

self.elaborate_functions(trait_impl.methods);
fn collect_trait_impl(&mut self, trait_impl: &mut UnresolvedTraitImpl) {
self.local_module = trait_impl.module_id;
self.file = trait_impl.file_id;
trait_impl.trait_id = self.resolve_trait_by_path(trait_impl.trait_path.clone());

let self_type = trait_impl.methods.self_type.clone();
let self_type =
self_type.expect("Expected struct type to be set before collect_trait_impl");

let self_type_span = trait_impl.object_type.span;

if matches!(self_type, Type::MutableReference(_)) {
let span = self_type_span.unwrap_or_else(|| trait_impl.trait_path.span());
self.push_err(DefCollectorErrorKind::MutableReferenceInTraitImpl { span });
}

assert!(trait_impl.trait_id.is_some());
if let Some(trait_id) = trait_impl.trait_id {
self.collect_trait_impl_methods(trait_id, trait_impl);

let span = trait_impl.object_type.span.expect("All trait self types should have spans");
self.generics = trait_impl.resolved_generics.clone();
self.declare_methods_on_struct(true, &mut trait_impl.methods, span);

let methods = trait_impl.methods.function_ids();
for func_id in &methods {
self.interner.set_function_trait(*func_id, self_type.clone(), trait_id);
}

let where_clause = trait_impl
.where_clause
.into_iter()
.iter()
.flat_map(|item| self.resolve_trait_constraint(item))
.collect();

let trait_generics = trait_impl.resolved_trait_generics.clone();

let resolved_trait_impl = Shared::new(TraitImpl {
ident: trait_impl.trait_path.last_segment().clone(),
typ: self_type.clone(),
Expand All @@ -914,7 +925,7 @@ impl<'context> Elaborator<'context> {
self_type.clone(),
trait_id,
trait_generics,
impl_id,
trait_impl.impl_id.expect("impl_id should be set in define_function_metas"),
generics,
resolved_trait_impl,
) {
Expand All @@ -931,45 +942,7 @@ impl<'context> Elaborator<'context> {
}
}

self.self_type = None;
self.current_trait_impl = None;
self.generics.truncate(old_generics_length);
}

fn collect_impls(
&mut self,
self_type: &UnresolvedType,
module: LocalModuleId,
impls: &mut [(Vec<Ident>, Span, UnresolvedFunctions)],
) {
self.local_module = module;

for (generics, span, unresolved) in impls {
self.file = unresolved.file_id;
self.recover_generics(|this| {
this.add_generics(generics);
this.declare_methods_on_struct(self_type, false, unresolved, *span);
});
}
}

fn collect_trait_impl(&mut self, trait_impl: &mut UnresolvedTraitImpl) {
self.local_module = trait_impl.module_id;
self.file = trait_impl.file_id;
trait_impl.trait_id = self.resolve_trait_by_path(trait_impl.trait_path.clone());

if let Some(trait_id) = trait_impl.trait_id {
self.collect_trait_impl_methods(trait_id, trait_impl);

let span = trait_impl.object_type.span.expect("All trait self types should have spans");
let object_type = &trait_impl.object_type;

self.recover_generics(|this| {
this.add_generics(&trait_impl.generics);
trait_impl.resolved_generics = this.generics.clone();
this.declare_methods_on_struct(object_type, true, &mut trait_impl.methods, span);
});
}
self.generics.clear();
}

fn get_module_mut(&mut self, module: ModuleId) -> &mut ModuleData {
Expand All @@ -979,14 +952,13 @@ impl<'context> Elaborator<'context> {

fn declare_methods_on_struct(
&mut self,
self_type: &UnresolvedType,
is_trait_impl: bool,
functions: &mut UnresolvedFunctions,
span: Span,
) {
let self_type = self.resolve_type(self_type.clone());

functions.self_type = Some(self_type.clone());
let self_type = functions.self_type.as_ref();
let self_type =
self_type.expect("Expected struct type to be set before declare_methods_on_struct");

let function_ids = functions.function_ids();

Expand Down Expand Up @@ -1016,11 +988,11 @@ impl<'context> Elaborator<'context> {
}
}

self.declare_struct_methods(&self_type, &function_ids);
self.declare_struct_methods(self_type, &function_ids);
// We can define methods on primitive types only if we're in the stdlib
} else if !is_trait_impl && self_type != Type::Error {
} else if !is_trait_impl && *self_type != Type::Error {
if self.crate_id.is_stdlib() {
self.declare_struct_methods(&self_type, &function_ids);
self.declare_struct_methods(self_type, &function_ids);
} else {
self.push_err(DefCollectorErrorKind::NonStructTypeInImpl { span });
}
Expand Down Expand Up @@ -1145,8 +1117,8 @@ impl<'context> Elaborator<'context> {
self.local_module = trait_impl.module_id;
self.file = trait_impl.file_id;

let object_crate = match self.resolve_type(trait_impl.object_type.clone()) {
Type::Struct(struct_type, _) => struct_type.borrow().id.krate(),
let object_crate = match &trait_impl.resolved_object_type {
Some(Type::Struct(struct_type, _)) => struct_type.borrow().id.krate(),
_ => CrateId::Dummy,
};

Expand All @@ -1163,7 +1135,6 @@ impl<'context> Elaborator<'context> {
self.local_module = alias.module_id;

let generics = self.add_generics(&alias.type_alias_def.generics);
self.resolve_local_globals();
self.current_item = Some(DependencyId::Alias(alias_id));
let typ = self.resolve_type(alias.type_alias_def.typ);
self.interner.set_type_alias(alias_id, typ, generics);
Expand Down Expand Up @@ -1215,9 +1186,6 @@ impl<'context> Elaborator<'context> {
self.recover_generics(|this| {
let generics = this.add_generics(&unresolved.generics);

// Check whether the struct definition has globals in the local module and add them to the scope
this.resolve_local_globals();

this.current_item = Some(DependencyId::Struct(struct_id));

this.resolving_ids.insert(struct_id);
Expand Down Expand Up @@ -1253,7 +1221,7 @@ impl<'context> Elaborator<'context> {
let (let_statement, _typ) = self.elaborate_let(let_stmt);

let statement_id = self.interner.get_global(global_id).let_statement;
self.interner.get_global_definition_mut(global_id).kind = definition_kind;
self.interner.get_global_definition_mut(global_id).kind = definition_kind.clone();
self.interner.replace_statement(statement_id, let_statement);
}

Expand Down Expand Up @@ -1286,6 +1254,11 @@ impl<'context> Elaborator<'context> {

let unresolved_type = &trait_impl.object_type;
self.add_generics(&trait_impl.generics);
trait_impl.resolved_generics = self.generics.clone();
jfecher marked this conversation as resolved.
Show resolved Hide resolved

let trait_generics =
vecmap(&trait_impl.trait_generics, |generic| self.resolve_type(generic.clone()));
trait_impl.resolved_trait_generics = trait_generics;

let self_type = self.resolve_type(unresolved_type.clone());

Expand Down
Loading
Loading