So, annoyingly, I got this wrong.

Bootstrapped and regression tested on x86_64.

Best,
Martin


    c23: fix TBAA error for function returning structure [PR125252]
    
    To fix PR122572 we globbed pointers to structure or union types nested
    within other structure or union types to a void pointer when computing
    the TYPE_CANONICAL.  But when doing this for function return types this
    then leads to wrong aliasing decisions, because for some reason function
    derivation behaves differently than pointer and array derivation.  Instead
    of globbing to void, replace the nested structure or union type with an
    incomplete type instead.
    
            PR c/124252
    
    gcc/c/ChangeLog:
    
            * c-typeck.cc (ptr_to_tagged_member): Return type of member.
            (c_type_canonical): Use incomplete type.
    
    gcc/testsuite/ChangeLog:
    
            * gcc.dg/pr125252.c

diff --git a/gcc/c/c-typeck.cc b/gcc/c/c-typeck.cc
index 6195d179543..3ffeed73d8c 100644
--- a/gcc/c/c-typeck.cc
+++ b/gcc/c/c-typeck.cc
@@ -579,8 +579,9 @@ c_reconstruct_complex_type (tree type, tree bottom)
 
 /* Helper function for c_canonical_type.  Check whether FIELD
    contains a pointer to a structure or union with tag,
-   possibly nested in other type derivations.  */
-static bool
+   possibly nested in other type derivations, and return the
+   type of this nested structure or union.  */
+static tree
 ptr_to_tagged_member (tree field)
 {
   gcc_assert (FIELD_DECL == TREE_CODE (field));
@@ -598,17 +599,17 @@ ptr_to_tagged_member (tree field)
   if (ptr_seen
       && RECORD_OR_UNION_TYPE_P (type)
       && NULL_TREE != c_type_tag (type))
-    return true;
+    return type;
 
-  return false;
+  return NULL_TREE;
 }
 
 /* For a record or union type, make necessary adaptations so that the
    type can be used as TYPE_CANONICAL.
 
    If the TYPE contains a pointer (possibly nested in other type
-   derivations) to a structure or union as a member, create a copy
-   and change such pointers to void pointers.  Otherwise, the middle-end
+   derivations) to a structure or union as a member, create a copy and
+   change the nested type to an incomplete type.  Otherwise, the middle-end
    gets confused when recording component aliases in the case where we
    have formed equivalency classes that include types for which these
    member pointers end up pointing to other structure or unions types
@@ -635,9 +636,15 @@ c_type_canonical (tree type)
   for (tree x = TYPE_FIELDS (type); x; x = DECL_CHAIN (x))
     {
       tree f = copy_node (x);
-      if (ptr_to_tagged_member (x))
-       TREE_TYPE (f) = c_reconstruct_complex_type (TREE_TYPE (x),
-                                                   ptr_type_node);
+      if (tree m = ptr_to_tagged_member (x))
+       {
+         tree new_node = make_node (TREE_CODE (m));
+         TYPE_NAME (new_node) = TYPE_NAME (m);
+         SET_TYPE_STRUCTURAL_EQUALITY (new_node);
+         new_node = qualify_type (new_node, m);
+         TREE_TYPE (f) = c_reconstruct_complex_type (TREE_TYPE (x),
+                                                     new_node);
+       }
       *fields = f;
       fields = &DECL_CHAIN (f);
     }
diff --git a/gcc/testsuite/gcc.dg/pr125252.c b/gcc/testsuite/gcc.dg/pr125252.c
new file mode 100644
index 00000000000..4e27fa15978
--- /dev/null
+++ b/gcc/testsuite/gcc.dg/pr125252.c
@@ -0,0 +1,36 @@
+/* { dg-do run } */
+/* { dg-options "-std=gnu23 -O2" } */
+
+
+int k;
+struct S {};
+
+#define T struct S
+
+T *get_sender () {
+  k = 1;
+  return (void *)0;
+}
+
+T *get_cc () { }
+
+void test () {
+        struct {
+                T *(*get_header) (void);
+        } reply_to_map[] = {
+                { get_sender },
+                { get_cc },
+        };
+
+        for (int i = 0; i < 2; i++) {
+                asm ("" ::"r"(reply_to_map[i].get_header ()));
+        }
+}
+
+int main() {
+        test ();
+
+        if (k != 1)
+            __builtin_abort();
+}
+

Reply via email to