]> git.ipfire.org Git - thirdparty/gcc.git/commitdiff
tree-sra: Do not propagate bit-field accesses across assignments (PR 117217)
authorMartin Jambor <mjambor@suse.cz>
Mon, 9 Feb 2026 09:31:10 +0000 (10:31 +0100)
committerMartin Jambor <jamborm@gcc.gnu.org>
Mon, 9 Feb 2026 09:31:28 +0000 (10:31 +0100)
On master at least, this is a bug which no longer reproduces with the
provided test-cases because after r15-5747-gfd62fdc5e1b3c4:(Jakub
Jelinek: c++: Small initial fixes for zeroing of padding bits
[PR117256]), the input to SRA looks very different and SRA does not do
anything.

However, before that commit, SRA sees the following input and if it
encountered something similar now, it could still misbehave in the
same way:

  D.2908.i = 0;
  D.2908.b = 0;
  e ={v} {CLOBBER(bob)};
  e.b = MEM[(const struct B &)&D.2908];
  D.2908 ={v} {CLOBBER(eos)};

(Where the "e" in "e.b" is actually a MEM_REF of the union type into
&e so that is why the "data" field is missing.)

Field D.2908.b is a SRA candidate of boolean type and has size 1 bit
because its decl has size 1 bit even though its type has size 8 bits.

The SRA access representing the store to D.2908.b is then propagated
across the assignment to e and in the process
build_user_friendly_ref_for_offset tries to find a nice expression for
it to possibly use in warnings.  It finds types_compatible_p
e.data.a.b which however has size 8 bits and so the verifier screams
when it discovers the discrepancy from the copied-over size of 1 bit.

This patch avoids the situation by refusing to propagate
non-byte-sized accesses across assignments.

gcc/ChangeLog:

2026-02-06  Martin Jambor  <mjambor@suse.cz>

PR tree-optimization/117217
* tree-sra.cc (propagate_subaccesses_from_rhs): Do not propagate
bit-field children.
(propagate_subaccesses_from_lhs): Likewise.

gcc/testsuite/ChangeLog:

2026-02-06  Martin Jambor  <mjambor@suse.cz>

PR tree-optimization/117217
* g++.dg/torture/pr117217-1.C: New test.
* g++.dg/torture/pr117217-2.C: Likewise.

gcc/testsuite/g++.dg/torture/pr117217-1.C [new file with mode: 0644]
gcc/testsuite/g++.dg/torture/pr117217-2.C [new file with mode: 0644]
gcc/tree-sra.cc

diff --git a/gcc/testsuite/g++.dg/torture/pr117217-1.C b/gcc/testsuite/g++.dg/torture/pr117217-1.C
new file mode 100644 (file)
index 0000000..887ab69
--- /dev/null
@@ -0,0 +1,24 @@
+struct [[gnu::packed]] A {
+  int i;
+  bool b;
+};
+
+struct [[gnu::packed]] B {
+  int i;
+  bool b : 1;
+};
+
+struct E {
+  union Data {
+    A a;
+    B b;
+    Data(const B &b) : b(b) {}
+  } data;
+};
+
+extern B copy;
+
+int main() {
+  E e{{B()}};
+  copy = e.data.b; // NEEDED FOR ICE
+}
diff --git a/gcc/testsuite/g++.dg/torture/pr117217-2.C b/gcc/testsuite/g++.dg/torture/pr117217-2.C
new file mode 100644 (file)
index 0000000..1939021
--- /dev/null
@@ -0,0 +1,24 @@
+struct a {
+  int b;
+  long c;
+  long d;
+  bool f;
+};
+struct g {
+  int b;
+  long c;
+  long d;
+  bool : 1;
+} h;
+struct l {
+  union i {
+    a j;
+    g k;
+    i(g m) : k(m) {}
+  } data;
+};
+int main() {
+  l e{g()};
+  h = e.data.k;
+  return 0;
+}
index edb03fe237b8394d5f29a468236851413ab76917..65ae5c99a0babe959a638626f46f226ebfaf068f 100644 (file)
@@ -3265,6 +3265,7 @@ propagate_subaccesses_from_rhs (struct access *lacc, struct access *racc)
        }
 
       if (rchild->grp_unscalarizable_region
+         || (rchild->size % BITS_PER_UNIT) != 0
          || !budget_for_propagation_access (lacc->base))
        {
          if (!lacc->grp_write && access_or_its_child_written (rchild))
@@ -3324,6 +3325,7 @@ propagate_subaccesses_from_lhs (struct access *lacc, struct access *racc)
       HOST_WIDE_INT norm_offset = lchild->offset + norm_delta;
 
       if (lchild->grp_unscalarizable_region
+         || (lchild->size % BITS_PER_UNIT) != 0
          || child_would_conflict_in_acc (racc, norm_offset, lchild->size,
                                          &matching_acc)
          || !budget_for_propagation_access (racc->base))