From 004f8dd733cb23da4ed57b160f6b86d53bc0b5f1 Mon Sep 17 00:00:00 2001 From: Nikolay Nikolov Date: Thu, 12 Oct 2023 23:38:32 +0300 Subject: [PATCH] feat(traits): allow multiple traits to share the same associated function name and to be implemented for the same type (#3126) --- .../src/hir/def_collector/dc_crate.rs | 10 +- .../src/hir/def_map/item_scope.rs | 106 +++++++++++++++--- .../src/hir/def_map/module_data.rs | 38 +++++-- .../Nargo.toml | 7 ++ .../Prover.toml | 0 .../src/main.nr | 36 ++++++ 6 files changed, 163 insertions(+), 34 deletions(-) create mode 100644 tooling/nargo_cli/tests/execution_success/trait_associated_member_names_clashes/Nargo.toml create mode 100644 tooling/nargo_cli/tests/execution_success/trait_associated_member_names_clashes/Prover.toml create mode 100644 tooling/nargo_cli/tests/execution_success/trait_associated_member_names_clashes/src/main.nr diff --git a/compiler/noirc_frontend/src/hir/def_collector/dc_crate.rs b/compiler/noirc_frontend/src/hir/def_collector/dc_crate.rs index 86cdd936b0d..13610b8ff47 100644 --- a/compiler/noirc_frontend/src/hir/def_collector/dc_crate.rs +++ b/compiler/noirc_frontend/src/hir/def_collector/dc_crate.rs @@ -499,17 +499,18 @@ fn add_method_to_struct_namespace( struct_type: &Shared, func_id: FuncId, name_ident: &Ident, + trait_id: TraitId, ) -> Result<(), DefCollectorErrorKind> { let struct_type = struct_type.borrow(); let type_module = struct_type.id.local_module_id(); let module = &mut current_def_map.modules[type_module.0]; - module.declare_function(name_ident.clone(), func_id).map_err(|(first_def, second_def)| { - DefCollectorErrorKind::Duplicate { + module.declare_trait_function(name_ident.clone(), func_id, trait_id).map_err( + |(first_def, second_def)| DefCollectorErrorKind::Duplicate { typ: DuplicateType::TraitImplementation, first_def, second_def, - } - }) + }, + ) } fn collect_trait_impl( @@ -550,6 +551,7 @@ fn collect_trait_impl( struct_type, *func_id, ast.name_ident(), + trait_id, ) { Ok(()) => {} Err(err) => { diff --git a/compiler/noirc_frontend/src/hir/def_map/item_scope.rs b/compiler/noirc_frontend/src/hir/def_map/item_scope.rs index 78375ab2bc0..42cca550651 100644 --- a/compiler/noirc_frontend/src/hir/def_map/item_scope.rs +++ b/compiler/noirc_frontend/src/hir/def_map/item_scope.rs @@ -1,5 +1,8 @@ use super::{namespace::PerNs, ModuleDefId, ModuleId}; -use crate::{node_interner::FuncId, Ident}; +use crate::{ + node_interner::{FuncId, TraitId}, + Ident, +}; use std::collections::{hash_map::Entry, HashMap}; #[derive(Debug, PartialEq, Eq, Copy, Clone)] @@ -9,8 +12,8 @@ pub enum Visibility { #[derive(Default, Debug, PartialEq, Eq)] pub struct ItemScope { - types: HashMap, - values: HashMap, + types: HashMap, (ModuleDefId, Visibility)>>, + values: HashMap, (ModuleDefId, Visibility)>>, defs: Vec, } @@ -20,8 +23,9 @@ impl ItemScope { &mut self, name: Ident, mod_def: ModuleDefId, + trait_id: Option, ) -> Result<(), (Ident, Ident)> { - self.add_item_to_namespace(name, mod_def)?; + self.add_item_to_namespace(name, mod_def, trait_id)?; self.defs.push(mod_def); Ok(()) } @@ -33,16 +37,26 @@ impl ItemScope { &mut self, name: Ident, mod_def: ModuleDefId, + trait_id: Option, ) -> Result<(), (Ident, Ident)> { - let add_item = |map: &mut HashMap| { - if let Entry::Occupied(o) = map.entry(name.clone()) { - let old_ident = o.key(); - Err((old_ident.clone(), name)) - } else { - map.insert(name, (mod_def, Visibility::Public)); - Ok(()) - } - }; + let add_item = + |map: &mut HashMap, (ModuleDefId, Visibility)>>| { + if let Entry::Occupied(mut o) = map.entry(name.clone()) { + let trait_hashmap = o.get_mut(); + if let Entry::Occupied(_) = trait_hashmap.entry(trait_id) { + let old_ident = o.key(); + Err((old_ident.clone(), name)) + } else { + trait_hashmap.insert(trait_id, (mod_def, Visibility::Public)); + Ok(()) + } + } else { + let mut trait_hashmap = HashMap::new(); + trait_hashmap.insert(trait_id, (mod_def, Visibility::Public)); + map.insert(name, trait_hashmap); + Ok(()) + } + }; match mod_def { ModuleDefId::ModuleId(_) => add_item(&mut self.types), @@ -55,7 +69,7 @@ impl ItemScope { } pub fn find_module_with_name(&self, mod_name: &Ident) -> Option<&ModuleId> { - let (module_def, _) = self.types.get(mod_name)?; + let (module_def, _) = self.types.get(mod_name)?.get(&None)?; match module_def { ModuleDefId::ModuleId(id) => Some(id), _ => None, @@ -63,7 +77,35 @@ impl ItemScope { } pub fn find_func_with_name(&self, func_name: &Ident) -> Option { - let (module_def, _) = self.values.get(func_name)?; + let trait_hashmap = self.values.get(func_name)?; + // methods introduced without trait take priority and hide methods with the same name that come from a trait + let a = trait_hashmap.get(&None); + match a { + Some((module_def, _)) => match module_def { + ModuleDefId::FunctionId(id) => Some(*id), + _ => None, + }, + None => { + if trait_hashmap.len() == 1 { + let (module_def, _) = trait_hashmap.get(trait_hashmap.keys().last()?)?; + match module_def { + ModuleDefId::FunctionId(id) => Some(*id), + _ => None, + } + } else { + // ambiguous name (multiple traits, containing the same function name) + None + } + } + } + } + + pub fn find_func_with_name_and_trait_id( + &self, + func_name: &Ident, + trait_id: &Option, + ) -> Option { + let (module_def, _) = self.values.get(func_name)?.get(trait_id)?; match module_def { ModuleDefId::FunctionId(id) => Some(*id), _ => None, @@ -71,18 +113,46 @@ impl ItemScope { } pub fn find_name(&self, name: &Ident) -> PerNs { - PerNs { types: self.types.get(name).cloned(), values: self.values.get(name).cloned() } + // Names, not associated with traits are searched first. If not found, we search for name, coming from a trait. + // If we find only one name from trait, we return it. If there are multiple traits, providing the same name, we return None. + let find_name_in = + |a: &HashMap, (ModuleDefId, Visibility)>>| { + if let Some(t) = a.get(name) { + if let Some(tt) = t.get(&None) { + Some(*tt) + } else if t.len() == 1 { + t.values().last().cloned() + } else { + None + } + } else { + None + } + }; + + PerNs { types: find_name_in(&self.types), values: find_name_in(&self.values) } + } + + pub fn find_name_for_trait_id(&self, name: &Ident, trait_id: &Option) -> PerNs { + PerNs { + types: if let Some(t) = self.types.get(name) { t.get(trait_id).cloned() } else { None }, + values: if let Some(v) = self.values.get(name) { + v.get(trait_id).cloned() + } else { + None + }, + } } pub fn definitions(&self) -> Vec { self.defs.clone() } - pub fn types(&self) -> &HashMap { + pub fn types(&self) -> &HashMap, (ModuleDefId, Visibility)>> { &self.types } - pub fn values(&self) -> &HashMap { + pub fn values(&self) -> &HashMap, (ModuleDefId, Visibility)>> { &self.values } diff --git a/compiler/noirc_frontend/src/hir/def_map/module_data.rs b/compiler/noirc_frontend/src/hir/def_map/module_data.rs index 5528312c0fc..29b11e92c01 100644 --- a/compiler/noirc_frontend/src/hir/def_map/module_data.rs +++ b/compiler/noirc_frontend/src/hir/def_map/module_data.rs @@ -41,16 +41,30 @@ impl ModuleData { } } - fn declare(&mut self, name: Ident, item_id: ModuleDefId) -> Result<(), (Ident, Ident)> { - self.scope.add_definition(name.clone(), item_id)?; + fn declare( + &mut self, + name: Ident, + item_id: ModuleDefId, + trait_id: Option, + ) -> Result<(), (Ident, Ident)> { + self.scope.add_definition(name.clone(), item_id, trait_id)?; // definitions is a subset of self.scope so it is expected if self.scope.define_func_def // returns without error, so will self.definitions.define_func_def. - self.definitions.add_definition(name, item_id) + self.definitions.add_definition(name, item_id, trait_id) } pub fn declare_function(&mut self, name: Ident, id: FuncId) -> Result<(), (Ident, Ident)> { - self.declare(name, id.into()) + self.declare(name, id.into(), None) + } + + pub fn declare_trait_function( + &mut self, + name: Ident, + id: FuncId, + trait_id: TraitId, + ) -> Result<(), (Ident, Ident)> { + self.declare(name, id.into(), Some(trait_id)) } pub fn remove_function(&mut self, name: &Ident) { @@ -59,11 +73,11 @@ impl ModuleData { } pub fn declare_global(&mut self, name: Ident, id: StmtId) -> Result<(), (Ident, Ident)> { - self.declare(name, id.into()) + self.declare(name, id.into(), None) } pub fn declare_struct(&mut self, name: Ident, id: StructId) -> Result<(), (Ident, Ident)> { - self.declare(name, ModuleDefId::TypeId(id)) + self.declare(name, ModuleDefId::TypeId(id), None) } pub fn declare_type_alias( @@ -71,11 +85,11 @@ impl ModuleData { name: Ident, id: TypeAliasId, ) -> Result<(), (Ident, Ident)> { - self.declare(name, id.into()) + self.declare(name, id.into(), None) } pub fn declare_trait(&mut self, name: Ident, id: TraitId) -> Result<(), (Ident, Ident)> { - self.declare(name, ModuleDefId::TraitId(id)) + self.declare(name, ModuleDefId::TraitId(id), None) } pub fn declare_child_module( @@ -83,7 +97,7 @@ impl ModuleData { name: Ident, child_id: ModuleId, ) -> Result<(), (Ident, Ident)> { - self.declare(name, child_id.into()) + self.declare(name, child_id.into(), None) } pub fn find_func_with_name(&self, name: &Ident) -> Option { @@ -91,7 +105,7 @@ impl ModuleData { } pub fn import(&mut self, name: Ident, id: ModuleDefId) -> Result<(), (Ident, Ident)> { - self.scope.add_item_to_namespace(name, id) + self.scope.add_item_to_namespace(name, id, None) } pub fn find_name(&self, name: &Ident) -> PerNs { @@ -99,12 +113,12 @@ impl ModuleData { } pub fn type_definitions(&self) -> impl Iterator + '_ { - self.definitions.types().values().map(|(id, _)| *id) + self.definitions.types().values().flat_map(|a| a.values().map(|(id, _)| *id)) } /// Return an iterator over all definitions defined within this module, /// excluding any type definitions. pub fn value_definitions(&self) -> impl Iterator + '_ { - self.definitions.values().values().map(|(id, _)| *id) + self.definitions.values().values().flat_map(|a| a.values().map(|(id, _)| *id)) } } diff --git a/tooling/nargo_cli/tests/execution_success/trait_associated_member_names_clashes/Nargo.toml b/tooling/nargo_cli/tests/execution_success/trait_associated_member_names_clashes/Nargo.toml new file mode 100644 index 00000000000..585dcde7351 --- /dev/null +++ b/tooling/nargo_cli/tests/execution_success/trait_associated_member_names_clashes/Nargo.toml @@ -0,0 +1,7 @@ +[package] +name = "trait_associated_member_names_clashes" +type = "bin" +authors = [""] +compiler_version = "0.16.0" + +[dependencies] \ No newline at end of file diff --git a/tooling/nargo_cli/tests/execution_success/trait_associated_member_names_clashes/Prover.toml b/tooling/nargo_cli/tests/execution_success/trait_associated_member_names_clashes/Prover.toml new file mode 100644 index 00000000000..e69de29bb2d diff --git a/tooling/nargo_cli/tests/execution_success/trait_associated_member_names_clashes/src/main.nr b/tooling/nargo_cli/tests/execution_success/trait_associated_member_names_clashes/src/main.nr new file mode 100644 index 00000000000..412a75010f6 --- /dev/null +++ b/tooling/nargo_cli/tests/execution_success/trait_associated_member_names_clashes/src/main.nr @@ -0,0 +1,36 @@ +trait Trait1 { + fn tralala() -> Field; +} + +trait Trait2 { + fn tralala() -> Field; +} + +struct Struct1 { +} + +impl Struct1 { + fn tralala() -> Field { + 123456 + } +} + +impl Trait1 for Struct1 { + fn tralala() -> Field { + 111111 + } +} + +impl Trait2 for Struct1 { + fn tralala() -> Field { + 222222 + } +} + +fn main() { + // the struct impl takes priority over trait methods + assert(Struct1::tralala() == 123456); + // TODO: uncomment these, once we support the :: syntax + //assert(::tralala() == 111111); + //assert(::tralala() == 222222); +}