]> git.ipfire.org Git - thirdparty/gcc.git/commitdiff
gccrs: Fix ICE on invalid match arms
authorPhilip Herron <herron.philip@googlemail.com>
Thu, 3 Apr 2025 14:37:40 +0000 (15:37 +0100)
committerArthur Cohen <arthur.cohen@embecosm.com>
Tue, 8 Apr 2025 08:17:13 +0000 (10:17 +0200)
We hit assertions on empty enum or unknown variant, this catches the error
and emits a new diagnostic.

Fixes Rust-GCC#3656

gcc/rust/ChangeLog:

* typecheck/rust-hir-type-check-pattern.cc (TypeCheckPattern::visit): emit error

gcc/testsuite/ChangeLog:

* rust/compile/issue-3656.rs: New test.

Signed-off-by: Philip Herron <herron.philip@googlemail.com>
gcc/rust/typecheck/rust-hir-type-check-pattern.cc
gcc/testsuite/rust/compile/issue-3656.rs [new file with mode: 0644]

index 3f9557a5c3e1f39b1437f47b87cd99a02a4e29cc..bd13f7a561856614690a99cc5e80c09144ee4874 100644 (file)
@@ -277,7 +277,15 @@ TypeCheckPattern::visit (HIR::StructPattern &pattern)
 
   infered = pattern_ty;
   TyTy::ADTType *adt = static_cast<TyTy::ADTType *> (infered);
-  rust_assert (adt->number_of_variants () > 0);
+  if (adt->number_of_variants () == 0)
+    {
+      HIR::PathInExpression &path = pattern.get_path ();
+      const AST::SimplePath &sp = path.as_simple_path ();
+      rust_error_at (pattern.get_locus (), ErrorCode::E0574,
+                    "expected struct, variant or union type, found enum %qs",
+                    sp.as_string ().c_str ());
+      return;
+    }
 
   TyTy::VariantDef *variant = adt->get_variants ().at (0);
   if (adt->is_enum ())
@@ -285,7 +293,16 @@ TypeCheckPattern::visit (HIR::StructPattern &pattern)
       HirId variant_id = UNKNOWN_HIRID;
       bool ok = context->lookup_variant_definition (
        pattern.get_path ().get_mappings ().get_hirid (), &variant_id);
-      rust_assert (ok);
+      if (!ok)
+       {
+         HIR::PathInExpression &path = pattern.get_path ();
+         const AST::SimplePath &sp = path.as_simple_path ();
+         rust_error_at (
+           pattern.get_locus (), ErrorCode::E0574,
+           "expected struct, variant or union type, found enum %qs",
+           sp.as_string ().c_str ());
+         return;
+       }
 
       ok = adt->lookup_variant_by_id (variant_id, &variant);
       rust_assert (ok);
diff --git a/gcc/testsuite/rust/compile/issue-3656.rs b/gcc/testsuite/rust/compile/issue-3656.rs
new file mode 100644 (file)
index 0000000..e0bec2f
--- /dev/null
@@ -0,0 +1,10 @@
+enum Foo {
+    Bar(isize),
+}
+
+fn main() {
+    match Foo::Bar(205) {
+        Foo { i } => (),
+        // { dg-error "expected struct, variant or union type, found enum .Foo. .E0574." "" { target *-*-* } .-1 }
+    }
+}