]> git.ipfire.org Git - thirdparty/gcc.git/commitdiff
c++/reflection: consteval-only type in splice-spec [PR125900]
authorMarek Polacek <polacek@redhat.com>
Fri, 26 Jun 2026 18:09:38 +0000 (14:09 -0400)
committerMarek Polacek <polacek@redhat.com>
Fri, 26 Jun 2026 19:11:48 +0000 (15:11 -0400)
In this test in the template for the

  std::println("{}", static_cast<int>(s.[:member:]));

line provokes a bogus "consteval-only expressions are only allowed
in a constant-evaluated context" error.  (Not the line before
because there we have an OVERLOAD because the second argument's
type isn't obvious, and so the whole thing is considered type-dep.)

A splice-specifier is [: constant-expression :], and [expr.const.defns]:
an expression or conversion is manifestly constant-evaluated if it is
a constant-expression.

So I think we shouldn't walk SPLICE_EXPRs in
check_out_of_consteval_use_r.

PR c++/125900

gcc/cp/ChangeLog:

* reflect.cc (check_out_of_consteval_use_r): Don't walk
SPLICE_EXPR.

gcc/testsuite/ChangeLog:

* g++.dg/reflect/expr18.C: New test.

Reviewed-by: Jason Merrill <jason@redhat.com>
gcc/cp/reflect.cc
gcc/testsuite/g++.dg/reflect/expr18.C [new file with mode: 0644]

index 3d6d24eb23acdb9adaf302415e488b6dbe935bda..73e160d700f85db44d1fa1cf499b96213e0b896c 100644 (file)
@@ -8760,6 +8760,10 @@ check_out_of_consteval_use_r (tree *tp, int *walk_subtrees, void *pset)
       || TREE_CODE (t) == DECL_EXPR
       /* Neither into USING_STMT.  */
       || TREE_CODE (t) == USING_STMT
+      /* The operand of a splice is a constant-expression, thus
+        manifestly constant-evaluated, so consteval-only types are permitted
+        here.  */
+      || TREE_CODE (t) == SPLICE_EXPR
       /* Blocks can appear in the TREE_VEC operand of OpenMP
         depend/affinity/map/to/from OMP_CLAUSEs when using iterators.  */
       || TREE_CODE (t) == BLOCK)
diff --git a/gcc/testsuite/g++.dg/reflect/expr18.C b/gcc/testsuite/g++.dg/reflect/expr18.C
new file mode 100644 (file)
index 0000000..f2ceefc
--- /dev/null
@@ -0,0 +1,25 @@
+// PR c++/125900
+// { dg-do compile { target c++26 } }
+// { dg-additional-options "-freflection" }
+
+#include <meta>
+#include <print>
+
+struct S {
+    float f;
+};
+
+int main() {
+    S s{.f = 1.5f};
+
+    constexpr auto member{^^S::f};
+    std::println("{}", s.[:member:]);
+    std::println("{}", static_cast<int>(s.[:member:]));
+
+    constexpr auto access_context{std::meta::access_context::current()};
+    template for (constexpr auto member :
+                  std::define_static_array(nonstatic_data_members_of(^^S, access_context))) {
+        std::println("{}", s.[:member:]);
+        std::println("{}", static_cast<int>(s.[:member:]));
+    }
+}