OSDN Git Service

* tree-cfg.c (find_taken_edge): Tighten conditions for
authorlaw <law@138bc75d-0d04-0410-961f-82ee72b054a4>
Mon, 19 Mar 2007 19:52:19 +0000 (19:52 +0000)
committerlaw <law@138bc75d-0d04-0410-961f-82ee72b054a4>
Mon, 19 Mar 2007 19:52:19 +0000 (19:52 +0000)
        optimizing computed gotos.

        * PR tree-optimization/30984
        * gcc.c-torture/pr30984.c: New test.

git-svn-id: svn+ssh://gcc.gnu.org/svn/gcc/trunk@123067 138bc75d-0d04-0410-961f-82ee72b054a4

gcc/ChangeLog
gcc/testsuite/ChangeLog
gcc/testsuite/gcc.c-torture/compile/pr30984.c [new file with mode: 0644]
gcc/tree-cfg.c

index fb839f5..ec45c0e 100644 (file)
@@ -1,3 +1,8 @@
+2007-03-19  Jeff Law  <law@redhat.com>
+
+       * tree-cfg.c (find_taken_edge): Tighten conditions for
+       optimizing computed gotos.
+
 2007-03-19  Michael Matz  <matz@suse.de>
 
        * builtins.c (expand_builtin_sync_operation,
index b62c6eb..72474a8 100644 (file)
@@ -1,3 +1,8 @@
+2007-03-19  Jeff Law  <law@redhat.com>
+
+       * PR tree-optimization/30984
+       * gcc.c-torture/pr30984.c: New test.
+
 2007-03-19  Andrew Pinski  <andrew_pinski@playstation.sony.com>
        Richard Guenther  <rguenther@suse.de>
 
diff --git a/gcc/testsuite/gcc.c-torture/compile/pr30984.c b/gcc/testsuite/gcc.c-torture/compile/pr30984.c
new file mode 100644 (file)
index 0000000..265a6f3
--- /dev/null
@@ -0,0 +1,7 @@
+int fs_exec(int ino)
+{
+ void *src = 0;
+ if (ino)
+   src = (void*)0xe000;
+ goto *src;
+}
index 202a69e..fa4800e 100644 (file)
@@ -2039,7 +2039,18 @@ find_taken_edge (basic_block bb, tree val)
     return find_taken_edge_switch_expr (bb, val);
 
   if (computed_goto_p (stmt))
-    return find_taken_edge_computed_goto (bb, TREE_OPERAND( val, 0));
+    {
+      /* Only optimize if the argument is a label, if the argument is
+        not a label then we can not construct a proper CFG.
+
+         It may be the case that we only need to allow the LABEL_REF to
+         appear inside an ADDR_EXPR, but we also allow the LABEL_REF to
+         appear inside a LABEL_EXPR just to be safe.  */
+      if ((TREE_CODE (val) == ADDR_EXPR || TREE_CODE (val) == LABEL_EXPR)
+         && TREE_CODE (TREE_OPERAND (val, 0)) == LABEL_DECL)
+       return find_taken_edge_computed_goto (bb, TREE_OPERAND (val, 0));
+      return NULL;
+    }
 
   gcc_unreachable ();
 }