]> git.ipfire.org Git - thirdparty/gcc.git/commitdiff
middle-end/66279 - gimplification clobbers shared asm constraints
authorRichard Biener <rguenther@suse.de>
Fri, 28 Feb 2025 08:58:36 +0000 (09:58 +0100)
committerRichard Biener <rguenth@gcc.gnu.org>
Wed, 2 Jul 2025 10:48:26 +0000 (12:48 +0200)
When the C++ frontend clones a CTOR we do not copy ASM_EXPR constraints
fully as walk_tree does not recurse to TREE_PURPOSE of TREE_LIST nodes.
At this point doing that seems too dangerous so the following instead
avoids gimplification of ASM_EXPRs to clobber the shared constraints
and unshares it there, like it also unshares TREE_VALUE when it
re-writes a "+" output constraint to separate "=" output and matching
input constraint.

PR middle-end/66279
* gimplify.cc (gimplify_asm_expr): Copy TREE_PURPOSE before
rewriting it for "+" processing.

* g++.dg/pr66279.C: New testcase.

(cherry picked from commit 95f5d6cc17e7d6b689674756c62b6b5e1284afd0)

gcc/gimplify.cc
gcc/testsuite/g++.dg/pr66279.C [new file with mode: 0644]

index ecbe58d2b0567b298501da6bf328a6a7909d38e9..7d65377e9dc6c28add916bfdaa5ab53caa48969d 100644 (file)
@@ -6659,6 +6659,7 @@ gimplify_asm_expr (tree *expr_p, gimple_seq *pre_p, gimple_seq *post_p)
          /* Turn the in/out constraint into an output constraint.  */
          char *p = xstrdup (constraint);
          p[0] = '=';
+         TREE_PURPOSE (link) = unshare_expr (TREE_PURPOSE (link));
          TREE_VALUE (TREE_PURPOSE (link)) = build_string (constraint_len, p);
 
          /* And add a matching input constraint.  */
diff --git a/gcc/testsuite/g++.dg/pr66279.C b/gcc/testsuite/g++.dg/pr66279.C
new file mode 100644 (file)
index 0000000..c878044
--- /dev/null
@@ -0,0 +1,23 @@
+// { dg-do run }
+
+struct A {};
+
+struct B : public virtual A
+{
+  B();
+};
+
+B::B()
+{
+  unsigned int x = 42;
+
+  __asm__ __volatile__ ("" : "+r"(x));
+
+  if (x != 42)
+    __builtin_abort ();
+}
+
+int main()
+{
+  B b;
+}