From 909b22bb20645761aabf8b40622aa79c6a96ed6a Mon Sep 17 00:00:00 2001 From: jfecher Date: Mon, 9 Dec 2024 15:05:15 -0600 Subject: [PATCH] fix: Improve type error when indexing a variable of unknown type (#6744) --- .../src/elaborator/expressions.rs | 4 +++ .../src/elaborator/statements.rs | 28 ++++++++++--------- .../src/hir/type_check/errors.rs | 9 ++++++ 3 files changed, 28 insertions(+), 13 deletions(-) diff --git a/compiler/noirc_frontend/src/elaborator/expressions.rs b/compiler/noirc_frontend/src/elaborator/expressions.rs index b5fab6faf9b..b4ea06f1030 100644 --- a/compiler/noirc_frontend/src/elaborator/expressions.rs +++ b/compiler/noirc_frontend/src/elaborator/expressions.rs @@ -346,6 +346,10 @@ impl<'context> Elaborator<'context> { Type::Array(_, base_type) => *base_type, Type::Slice(base_type) => *base_type, Type::Error => Type::Error, + Type::TypeVariable(_) => { + self.push_err(TypeCheckError::TypeAnnotationsNeededForIndex { span: lhs_span }); + Type::Error + } typ => { self.push_err(TypeCheckError::TypeMismatch { expected_typ: "Array".to_owned(), diff --git a/compiler/noirc_frontend/src/elaborator/statements.rs b/compiler/noirc_frontend/src/elaborator/statements.rs index 6ed8fee753c..93009f49071 100644 --- a/compiler/noirc_frontend/src/elaborator/statements.rs +++ b/compiler/noirc_frontend/src/elaborator/statements.rs @@ -183,20 +183,20 @@ impl<'context> Elaborator<'context> { } pub(super) fn elaborate_assign(&mut self, assign: AssignStatement) -> (HirStatement, Type) { - let span = assign.expression.span; + let expr_span = assign.expression.span; let (expression, expr_type) = self.elaborate_expression(assign.expression); - let (lvalue, lvalue_type, mutable) = self.elaborate_lvalue(assign.lvalue, span); + let (lvalue, lvalue_type, mutable) = self.elaborate_lvalue(assign.lvalue); if !mutable { let (name, span) = self.get_lvalue_name_and_span(&lvalue); self.push_err(TypeCheckError::VariableMustBeMutable { name, span }); } - self.unify_with_coercions(&expr_type, &lvalue_type, expression, span, || { + self.unify_with_coercions(&expr_type, &lvalue_type, expression, expr_span, || { TypeCheckError::TypeMismatchWithSource { actual: expr_type.clone(), expected: lvalue_type.clone(), - span, + span: expr_span, source: Source::Assignment, } }); @@ -296,7 +296,7 @@ impl<'context> Elaborator<'context> { } } - fn elaborate_lvalue(&mut self, lvalue: LValue, assign_span: Span) -> (HirLValue, Type, bool) { + fn elaborate_lvalue(&mut self, lvalue: LValue) -> (HirLValue, Type, bool) { match lvalue { LValue::Ident(ident) => { let mut mutable = true; @@ -330,7 +330,7 @@ impl<'context> Elaborator<'context> { (HirLValue::Ident(ident.clone(), typ.clone()), typ, mutable) } LValue::MemberAccess { object, field_name, span } => { - let (object, lhs_type, mut mutable) = self.elaborate_lvalue(*object, assign_span); + let (object, lhs_type, mut mutable) = self.elaborate_lvalue(*object); let mut object = Box::new(object); let field_name = field_name.clone(); @@ -374,8 +374,7 @@ impl<'context> Elaborator<'context> { expr_span, }); - let (mut lvalue, mut lvalue_type, mut mutable) = - self.elaborate_lvalue(*array, assign_span); + let (mut lvalue, mut lvalue_type, mut mutable) = self.elaborate_lvalue(*array); // Before we check that the lvalue is an array, try to dereference it as many times // as needed to unwrap any &mut wrappers. @@ -397,12 +396,15 @@ impl<'context> Elaborator<'context> { self.push_err(TypeCheckError::StringIndexAssign { span: lvalue_span }); Type::Error } + Type::TypeVariable(_) => { + self.push_err(TypeCheckError::TypeAnnotationsNeededForIndex { span }); + Type::Error + } other => { - // TODO: Need a better span here self.push_err(TypeCheckError::TypeMismatch { expected_typ: "array".to_string(), expr_typ: other.to_string(), - expr_span: assign_span, + expr_span: span, }); Type::Error } @@ -413,7 +415,7 @@ impl<'context> Elaborator<'context> { (HirLValue::Index { array, index, typ, location }, array_type, mutable) } LValue::Dereference(lvalue, span) => { - let (lvalue, reference_type, _) = self.elaborate_lvalue(*lvalue, assign_span); + let (lvalue, reference_type, _) = self.elaborate_lvalue(*lvalue); let lvalue = Box::new(lvalue); let location = Location::new(span, self.file); @@ -423,7 +425,7 @@ impl<'context> Elaborator<'context> { self.unify(&reference_type, &expected_type, || TypeCheckError::TypeMismatch { expected_typ: expected_type.to_string(), expr_typ: reference_type.to_string(), - expr_span: assign_span, + expr_span: span, }); // Dereferences are always mutable since we already type checked against a &mut T @@ -433,7 +435,7 @@ impl<'context> Elaborator<'context> { } LValue::Interned(id, span) => { let lvalue = self.interner.get_lvalue(id, span).clone(); - self.elaborate_lvalue(lvalue, assign_span) + self.elaborate_lvalue(lvalue) } } } diff --git a/compiler/noirc_frontend/src/hir/type_check/errors.rs b/compiler/noirc_frontend/src/hir/type_check/errors.rs index dfa431157e3..15b8d50c78b 100644 --- a/compiler/noirc_frontend/src/hir/type_check/errors.rs +++ b/compiler/noirc_frontend/src/hir/type_check/errors.rs @@ -206,6 +206,8 @@ pub enum TypeCheckError { UnspecifiedType { span: Span }, #[error("Binding `{typ}` here to the `_` inside would create a cyclic type")] CyclicType { typ: Type, span: Span }, + #[error("Type annotations required before indexing this array or slice")] + TypeAnnotationsNeededForIndex { span: Span }, } #[derive(Debug, Clone, PartialEq, Eq)] @@ -520,6 +522,13 @@ impl<'a> From<&'a TypeCheckError> for Diagnostic { *span, ) }, + TypeCheckError::TypeAnnotationsNeededForIndex { span } => { + Diagnostic::simple_error( + "Type annotations required before indexing this array or slice".into(), + "Type annotations needed before this point, can't decide if this is an array or slice".into(), + *span, + ) + }, } } }