The result of build_fold_indirect_ref can be a COMPONENT_REF in
which case using DECL_SOURCE_LOCATION will crash. Look at its op1
instead.
PR c++/116741
gcc/cp/ChangeLog:
* constexpr.cc (cxx_eval_constant_expression) <case CONVERT_EXPR>: If
the result of build_fold_indirect_ref is a COMPONENT_REF, use its op1.
Check DECL_P before calling inform.
gcc/testsuite/ChangeLog:
* g++.dg/cpp26/constexpr-voidptr4.C: New test.
Reviewed-by: Jason Merrill <jason@redhat.com>
TREE_TYPE (op), TREE_TYPE (TREE_TYPE (sop)),
TREE_TYPE (type));
tree obj = build_fold_indirect_ref (sop);
- inform (DECL_SOURCE_LOCATION (obj),
- "pointed-to object declared here");
+ if (TREE_CODE (obj) == COMPONENT_REF)
+ obj = TREE_OPERAND (obj, 1);
+ if (DECL_P (obj))
+ inform (DECL_SOURCE_LOCATION (obj),
+ "pointed-to object declared here");
}
*non_constant_p = true;
return t;
--- /dev/null
+// PR c++/116741
+// { dg-do compile { target c++26 } }
+
+struct S {
+ int foo; // { dg-message "pointed-to object" }
+};
+
+struct S2 {
+ int foo; // { dg-message "pointed-to object" }
+};
+
+struct X {
+ S2 s;
+};
+
+constexpr float f1() {
+ S s;
+ void* p = &s.foo;
+ return *static_cast<float*>(p); // { dg-error "not allowed in a constant expression" }
+}
+
+constexpr float f2() {
+ X x;
+ void* p = &x.s.foo;
+ return *static_cast<float*>(p); // { dg-error "not allowed in a constant expression" }
+}
+
+constexpr auto x1 = f1();
+constexpr auto x2 = f2();