d: Fix ICE in gimplify_expr, at gimplify.c (PR98277)
authorIain Buclaw <ibuclaw@gdcproject.org>
Tue, 15 Dec 2020 09:36:00 +0000 (10:36 +0100)
committerIain Buclaw <ibuclaw@gdcproject.org>
Tue, 15 Dec 2020 21:20:35 +0000 (22:20 +0100)
The DMD front-end shouldn't, but can sometimes leak manifest constants
in the AST passed to the code generator.  To prevent this being an
issue, the setting of DECL_INITIAL has been moved to the point where the
CONST_DECL is used, rather than in the declaration handler.

gcc/d/ChangeLog:

PR d/98277
* decl.cc (DeclVisitor::visit (VarDeclaration *)): Move setting of
DECL_INITIAL for manifest constants to ...
(get_symbol_decl): ... here.

gcc/testsuite/ChangeLog:

PR d/98277
* gdc.dg/pr98277.d: New test.

gcc/d/decl.cc
gcc/testsuite/gdc.dg/pr98277.d [new file with mode: 0644]

index 218f35838fdd3c4bb30efceecfb4f5dfe0c48c2a..a3db4ae6d8949b1e0aeed114257a2085117d463d 100644 (file)
@@ -653,31 +653,13 @@ public:
        return;
       }
 
-    /* Do not store variables we cannot take the address of,
-       but keep the values for purposes of debugging.  */
     if (!d->canTakeAddressOf ())
       {
-       /* Don't know if there is a good way to handle instantiations.  */
-       if (d->isInstantiated ())
-         return;
-
-       /* Cannot make an expression out of a void initializer.  */
-       if (!d->_init || d->_init->isVoidInitializer ())
-         return;
-
-       tree decl = get_symbol_decl (d);
-       Expression *ie = initializerToExpression (d->_init);
-
-       /* CONST_DECL was initially intended for enumerals and may be used for
-          scalars in general, but not for aggregates.  Here a non-constant
-          value is generated anyway so as the CONST_DECL only serves as a
-          placeholder for the value, however the DECL itself should never be
-          referenced in any generated code, or passed to the back-end.  */
+       /* Do not store variables we cannot take the address of,
+          but keep the values for purposes of debugging.  */
        if (!d->type->isscalar ())
-         DECL_INITIAL (decl) = build_expr (ie, false);
-       else
          {
-           DECL_INITIAL (decl) = build_expr (ie, true);
+           tree decl = get_symbol_decl (d);
            d_pushdecl (decl);
            rest_of_decl_compilation (decl, 1, 0);
          }
@@ -1106,6 +1088,25 @@ get_symbol_decl (Declaration *decl)
 
       if (vd->storage_class & STCextern)
        DECL_EXTERNAL (decl->csym) = 1;
+
+      /* CONST_DECL was initially intended for enumerals and may be used for
+        scalars in general, but not for aggregates.  Here a non-constant
+        value is generated anyway so as the CONST_DECL only serves as a
+        placeholder for the value, however the DECL itself should never be
+        referenced in any generated code, or passed to the back-end.  */
+      if (vd->storage_class & STCmanifest)
+       {
+         /* Cannot make an expression out of a void initializer.  */
+         if (vd->_init && !vd->_init->isVoidInitializer ())
+           {
+             Expression *ie = initializerToExpression (vd->_init);
+
+             if (!vd->type->isscalar ())
+               DECL_INITIAL (decl->csym) = build_expr (ie, false);
+             else
+               DECL_INITIAL (decl->csym) = build_expr (ie, true);
+           }
+       }
     }
 
   /* Set the declaration mangled identifier if static.  */
diff --git a/gcc/testsuite/gdc.dg/pr98277.d b/gcc/testsuite/gdc.dg/pr98277.d
new file mode 100644 (file)
index 0000000..d3b859f
--- /dev/null
@@ -0,0 +1,13 @@
+// https://gcc.gnu.org/bugzilla/show_bug.cgi?id=98277
+// { dg-do compile }
+
+enum Side
+{
+    left,
+    right
+}
+
+ref int getSide(Side side, ref int left, ref int right)
+{
+    return side == Side.left ? left : right;
+}