]> git.ipfire.org Git - thirdparty/gcc.git/commitdiff
c++: constexpr, empty base after non-empty [PR106369]
authorJason Merrill <jason@redhat.com>
Tue, 26 Jul 2022 15:02:21 +0000 (11:02 -0400)
committerRichard Biener <rguenther@suse.de>
Fri, 12 Aug 2022 07:10:17 +0000 (09:10 +0200)
Here the CONSTRUCTOR we were providing for D{} had an entry for the B base
subobject at offset 0 following the entry for the C base, causing
output_constructor_regular_field to ICE due to going backwards.  It might be
nice for that function to be more tolerant of empty fields, but it also
seems reasonable for the front end to prune the useless entry.

PR c++/106369

gcc/cp/ChangeLog:

* constexpr.cc (reduced_constant_expression_p): Return false
if a CONSTRUCTOR initializes an empty field.

gcc/testsuite/ChangeLog:

* g++.dg/cpp1z/constexpr-lambda27.C: New test.

(cherry picked from commit 9efe4e153d994974afcbba09c3c683f5f4a19c63)

gcc/cp/constexpr.cc
gcc/testsuite/g++.dg/cpp1z/constexpr-lambda27.C [new file with mode: 0644]

index 1c17f0d14b875699e70c7ddfe950fcb74fa6d89f..28f8785dea3286c8ae1067a704df43dfe21e8828 100644 (file)
@@ -3063,7 +3063,13 @@ reduced_constant_expression_p (tree t)
             element.  */
          if (!reduced_constant_expression_p (e.value))
            return false;
-         /* Empty class field may or may not have an initializer.  */
+         /* We want to remove initializers for empty fields in a struct to
+            avoid confusing output_constructor.  */
+         if (is_empty_field (e.index)
+             && TREE_CODE (TREE_TYPE (t)) == RECORD_TYPE)
+           return false;
+         /* Check for non-empty fields between initialized fields when
+            CONSTRUCTOR_NO_CLEARING.  */
          for (; field && e.index != field;
               field = next_subobject_field (DECL_CHAIN (field)))
            if (!is_really_empty_class (TREE_TYPE (field),
diff --git a/gcc/testsuite/g++.dg/cpp1z/constexpr-lambda27.C b/gcc/testsuite/g++.dg/cpp1z/constexpr-lambda27.C
new file mode 100644 (file)
index 0000000..24e2e9b
--- /dev/null
@@ -0,0 +1,26 @@
+// PR c++/106369
+// { dg-do compile { target c++17 } }
+
+struct A {
+  int a[256];
+  constexpr int &operator[] (int n) noexcept { return a[n]; }
+  constexpr const int &operator[] (int n) const noexcept { return a[n]; }
+};
+struct B {};
+template <typename T>
+struct C {
+  constexpr T &foo (const char x) noexcept { c = T::d[x]; return static_cast<T &>(*this); }
+  int c;
+};
+struct D : public C<D>, public B
+{
+  D () noexcept = default;
+  static constexpr char e[9] { 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I' };
+  static constexpr A d = [] () constexpr {
+    A f {};
+    for (int i = 0; i < 9; ++i)
+      f[e[i]] = 1;
+    return f;
+  } ();
+};
+constexpr auto g = D{}.foo ('E');