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(frontend): Correctly monomorphize turbofish functions #5049

Merged
merged 66 commits into from
May 21, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
66 commits
Select commit Hold shift + click to select a range
04ea5d0
Add turbofish operator; add stack overflow crash
jfecher Nov 21, 2023
3a95229
Fix parsing error
jfecher Nov 22, 2023
427cb95
Add test
jfecher Nov 22, 2023
3ab58b8
Add compiler error for incorrect generic count
jfecher Nov 22, 2023
8fc453a
Edit example to have a more problematic case
jfecher Nov 22, 2023
4050fe5
Merge branch 'master' into jf/turbofish
TomAFrench Dec 11, 2023
b9673b2
resolved merge conflicts
vezenovm May 14, 2024
b289b87
fixup remaining fmt stuff
vezenovm May 14, 2024
6127baf
cleanup
vezenovm May 14, 2024
9ebb3b5
cargo fmt
vezenovm May 14, 2024
c652541
working initial turbofish tests
vezenovm May 14, 2024
9660360
cargo fmt
vezenovm May 14, 2024
4cee817
nargo fmt
vezenovm May 15, 2024
e89cde2
clippy
vezenovm May 15, 2024
a2da705
fmt
vezenovm May 15, 2024
da23029
nargo fmt
vezenovm May 15, 2024
da26bc3
fix nargo fmt for turbofish on method calls
vezenovm May 15, 2024
77dfc19
clippy
vezenovm May 15, 2024
0e5a579
initial work to get calling trait methods working, decided this would…
vezenovm May 15, 2024
c17e834
working turbofish with implicit generics now
vezenovm May 16, 2024
8f2be2e
separate out function and implicit generic counts
vezenovm May 16, 2024
405683e
merge conflicts w/ jf/turbofish and working trait methods with mutabl…
vezenovm May 16, 2024
5ad8fb4
cleanup
vezenovm May 16, 2024
e7000d7
Merge branch 'master' into jf/turbofish
vezenovm May 16, 2024
3cbba0b
merge conflicts w/ parent jf/turbofish
vezenovm May 16, 2024
cc135e7
cargo fmt
vezenovm May 16, 2024
a0090c1
cleanup
vezenovm May 16, 2024
954aba7
nargo fmt
vezenovm May 16, 2024
1e22003
add todo comments
vezenovm May 16, 2024
44a4aee
specify type annotation for hasher
vezenovm May 17, 2024
8d5c072
rename
vezenovm May 17, 2024
c6f3057
brought back removed comment
vezenovm May 17, 2024
384e26a
chore: add test for specifying types on function with turbofish
TomAFrench May 17, 2024
98c5d89
chore: add test for using turbofish with generic methods
TomAFrench May 17, 2024
32d1714
chore: add turbofish to cspell
TomAFrench May 17, 2024
dde6d5a
correctly monomorphize turbofish functions
vezenovm May 17, 2024
e26ed36
missed save
vezenovm May 17, 2024
9ab7351
nargo fmt tests
vezenovm May 17, 2024
b73d263
fix noirc_frontend tests
vezenovm May 17, 2024
a886946
chore: update formatter test outputs
TomAFrench May 20, 2024
4125804
Revert "chore: update formatter test outputs"
TomAFrench May 20, 2024
bc7abf8
Update compiler/noirc_frontend/src/hir_def/expr.rs
vezenovm May 20, 2024
8b1bbc4
have atom_or_right_unary accept a type parser
vezenovm May 20, 2024
cd1f59e
Merge remote-tracking branch 'origin/jf/turbofish' into jf/turbofish
vezenovm May 20, 2024
bd075fa
clippy and fmt
vezenovm May 20, 2024
b2100ad
Update compiler/noirc_frontend/src/hir/type_check/expr.rs
vezenovm May 20, 2024
0d7fb67
Update test_programs/execution_success/trait_method_mut_self/src/main.nr
vezenovm May 20, 2024
c045540
Update test_programs/execution_success/trait_method_mut_self/src/main.nr
vezenovm May 20, 2024
c6aed8c
fetch implicit generic count from node interner
vezenovm May 20, 2024
9bf4ac2
remove unused method implicit generics map
vezenovm May 20, 2024
22e3cda
add new line to expected formatter tests
vezenovm May 20, 2024
af2c858
fixup after code review
vezenovm May 20, 2024
aa8e49e
fmy and clippy
vezenovm May 20, 2024
af3cc88
Merge branch 'jf/turbofish' into mv/trait-method-reference
vezenovm May 20, 2024
e6850e9
update comment
vezenovm May 20, 2024
e14738b
merge parent
vezenovm May 20, 2024
bf07ee8
move where implciit_generic_count is computed
vezenovm May 21, 2024
a9cb6c4
switch to single loop in instantiate_with
vezenovm May 21, 2024
daf151b
rename to turbofish_generics
vezenovm May 21, 2024
0d887de
Merge branch 'master' into jf/turbofish
vezenovm May 21, 2024
8d93a66
Merge branch 'jf/turbofish' into mv/trait-method-reference
vezenovm May 21, 2024
a44782b
Merge branch 'master' into jf/turbofish
vezenovm May 21, 2024
d174e13
Merge branch 'jf/turbofish' into mv/trait-method-reference
vezenovm May 21, 2024
29bb0ae
Merge branch 'mv/trait-method-reference' into mv/turbofish-monomorphi…
vezenovm May 21, 2024
12084ae
merge w/ master
vezenovm May 21, 2024
fa91128
remove extra function_generic_count
vezenovm May 21, 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
48 changes: 36 additions & 12 deletions compiler/noirc_frontend/src/monomorphization/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -55,9 +55,12 @@ struct LambdaContext {
struct Monomorphizer<'interner> {
/// Functions are keyed by their unique ID and expected type so that we can monomorphize
/// a new version of the function for each type.
/// We also key by any turbofish generics that are specified.
/// This is necessary for a case where we may have a trait generic that can be instantiated
/// outside of a function parameter or return value.
vezenovm marked this conversation as resolved.
Show resolved Hide resolved
///
/// Using nested HashMaps here lets us avoid cloning HirTypes when calling .get()
functions: HashMap<node_interner::FuncId, HashMap<HirType, FuncId>>,
functions: HashMap<node_interner::FuncId, HashMap<(HirType, Vec<HirType>), FuncId>>,

/// Unlike functions, locals are only keyed by their unique ID because they are never
/// duplicated during monomorphization. Doing so would allow them to be used polymorphically
Expand Down Expand Up @@ -198,10 +201,15 @@ impl<'interner> Monomorphizer<'interner> {
id: node_interner::FuncId,
expr_id: node_interner::ExprId,
typ: &HirType,
turbofish_generics: Vec<HirType>,
trait_method: Option<TraitMethodId>,
) -> Definition {
let typ = typ.follow_bindings();
match self.functions.get(&id).and_then(|inner_map| inner_map.get(&typ)) {
match self
.functions
.get(&id)
.and_then(|inner_map| inner_map.get(&(typ.clone(), turbofish_generics.clone())))
{
Some(id) => Definition::Function(*id),
None => {
// Function has not been monomorphized yet
Expand All @@ -222,7 +230,8 @@ impl<'interner> Monomorphizer<'interner> {
Definition::Builtin(opcode)
}
FunctionKind::Normal => {
let id = self.queue_function(id, expr_id, typ, trait_method);
let id =
self.queue_function(id, expr_id, typ, turbofish_generics, trait_method);
Definition::Function(id)
}
FunctionKind::Oracle => {
Expand All @@ -249,8 +258,14 @@ impl<'interner> Monomorphizer<'interner> {
}

/// Prerequisite: typ = typ.follow_bindings()
fn define_function(&mut self, id: node_interner::FuncId, typ: HirType, new_id: FuncId) {
self.functions.entry(id).or_default().insert(typ, new_id);
fn define_function(
&mut self,
id: node_interner::FuncId,
typ: HirType,
turbofish_generics: Vec<HirType>,
new_id: FuncId,
) {
self.functions.entry(id).or_default().insert((typ, turbofish_generics), new_id);
}

fn compile_main(
Expand Down Expand Up @@ -393,7 +408,7 @@ impl<'interner> Monomorphizer<'interner> {
use ast::Literal::*;

let expr = match self.interner.expression(&expr) {
HirExpression::Ident(ident, _) => self.ident(ident, expr)?,
HirExpression::Ident(ident, generics) => self.ident(ident, expr, generics)?,
HirExpression::Literal(HirLiteral::Str(contents)) => Literal(Str(contents)),
HirExpression::Literal(HirLiteral::FmtStr(contents, idents)) => {
let fields = try_vecmap(idents, |ident| self.expr(ident))?;
Expand Down Expand Up @@ -825,6 +840,7 @@ impl<'interner> Monomorphizer<'interner> {
&mut self,
ident: HirIdent,
expr_id: node_interner::ExprId,
generics: Option<Vec<HirType>>,
) -> Result<ast::Expression, MonomorphizationError> {
let typ = self.interner.id_type(expr_id);

Expand All @@ -838,7 +854,13 @@ impl<'interner> Monomorphizer<'interner> {
let mutable = definition.mutable;
let location = Some(ident.location);
let name = definition.name.clone();
let definition = self.lookup_function(*func_id, expr_id, &typ, None);
let definition = self.lookup_function(
*func_id,
expr_id,
&typ,
generics.unwrap_or_default(),
None,
);
let typ = Self::convert_type(&typ, ident.location)?;
let ident = ast::Ident { location, mutable, definition, name, typ: typ.clone() };
let ident_expression = ast::Expression::Ident(ident);
Expand Down Expand Up @@ -1063,10 +1085,11 @@ impl<'interner> Monomorphizer<'interner> {
}
};

let func_id = match self.lookup_function(func_id, expr_id, &function_type, Some(method)) {
Definition::Function(func_id) => func_id,
_ => unreachable!(),
};
let func_id =
match self.lookup_function(func_id, expr_id, &function_type, vec![], Some(method)) {
Definition::Function(func_id) => func_id,
_ => unreachable!(),
};

let the_trait = self.interner.get_trait(method.trait_id);
let location = self.interner.expr_location(&expr_id);
Expand Down Expand Up @@ -1292,10 +1315,11 @@ impl<'interner> Monomorphizer<'interner> {
id: node_interner::FuncId,
expr_id: node_interner::ExprId,
function_type: HirType,
turbofish_generics: Vec<HirType>,
trait_method: Option<TraitMethodId>,
) -> FuncId {
let new_id = self.next_function_id();
self.define_function(id, function_type.clone(), new_id);
self.define_function(id, function_type.clone(), turbofish_generics, new_id);

let bindings = self.interner.get_instantiation_bindings(expr_id);
let bindings = self.follow_bindings(bindings);
Expand Down
22 changes: 0 additions & 22 deletions test_programs/execution_success/trait_method_mut_self/src/main.nr
Original file line number Diff line number Diff line change
Expand Up @@ -12,14 +12,6 @@ fn main(x: Field, y: pub Field) {

pass_trait_by_mut_ref(&mut a_mut_ref, y);
assert(a_mut_ref.x == y);

let mut hasher = Poseidon2Hasher::default();
hasher.write(x);
hasher.write(y);
let expected_hash = hasher.finish();
// Check that we get the same result when using the hasher in a
// method that purely uses trait methods without a supplied implementation.
assert(hash_simple_array::<Poseidon2Hasher>([x, y]) == expected_hash);
}

trait SomeTrait {
Expand Down Expand Up @@ -58,17 +50,3 @@ fn pass_trait_by_mut_ref<T>(a_mut_ref: &mut T, value: Field) where T: SomeTrait
// We auto add a mutable reference to the object type if the method call expects a mutable self
a_mut_ref.set_value(value);
}

fn hash_simple_array<H>(input: [Field; 2]) -> Field where H: Hasher + Default {
// Check that we can call a trait method instead of a trait implementation
// TODO: Need to remove the need for this type annotation
// TODO: Curently, without the annotation we will get `Expression type is ambiguous` when trying to use the `hasher`
let mut hasher: H = H::default();
// Regression that the object is converted to a mutable reference type `&mut _`.
// Otherwise will see `Expected type &mut _, found type H`.
// Then we need to make sure to also auto dereference later in the type checking process
// when searching for a matching impl or else we will get `No matching impl found for `&mut H: Hasher`
hasher.write(input[0]);
hasher.write(input[1]);
hasher.finish()
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
[package]
name = "turbofish_call_func_diff_types"
type = "bin"
authors = [""]
compiler_version = ">=0.29.0"

[dependencies]
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
x = "5"
y = "10"
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
use dep::std::hash::Hasher;
use dep::std::hash::poseidon2::Poseidon2Hasher;
use dep::std::hash::poseidon::PoseidonHasher;

fn main(x: Field, y: pub Field) {
let mut hasher = PoseidonHasher::default();
hasher.write(x);
hasher.write(y);
let poseidon_expected_hash = hasher.finish();
// Check that we get the same result when using the hasher in a
// method that purely uses trait methods without a supplied implementation.
assert(hash_simple_array::<PoseidonHasher>([x, y]) == poseidon_expected_hash);

// Now let's do the same logic but with a different `Hasher` supplied to the turbofish operator
// We want to make sure that we have correctly monomorphized a function with a trait generic
// where the generic is not used on any function parameters or the return value.
let mut hasher = Poseidon2Hasher::default();
hasher.write(x);
hasher.write(y);
let poseidon2_expected_hash = hasher.finish();
assert(hash_simple_array::<Poseidon2Hasher>([x, y]) == poseidon2_expected_hash);
}

fn hash_simple_array<H>(input: [Field; 2]) -> Field where H: Hasher + Default {
// Check that we can call a trait method instead of a trait implementation
// TODO(https://github.com/noir-lang/noir/issues/5063): Need to remove the need for this type annotation
// Curently, without the annotation we will get `Expression type is ambiguous` when trying to use the `hasher`
let mut hasher: H = H::default();
// Regression that the object is converted to a mutable reference type `&mut _`.
// Otherwise will see `Expected type &mut _, found type H`.
// Then we need to make sure to also auto dereference later in the type checking process
// when searching for a matching impl or else we will get `No matching impl found for `&mut H: Hasher`
hasher.write(input[0]);
hasher.write(input[1]);
hasher.finish()
}
Loading