X-Git-Url: http://git.sourceforge.jp/view?a=blobdiff_plain;f=gcc%2Ftree-cfg.c;h=fa4800e00c241dc45320dfba58744b14134a1b52;hb=94ca49167953b5f6445bdcdc6a44646e5eec4c23;hp=202a69e2c8f03ce859d9d577b6494c05f9be427c;hpb=4cbf73b4bf1d81e7983204180f9c3d2a65a068e6;p=pf3gnuchains%2Fgcc-fork.git diff --git a/gcc/tree-cfg.c b/gcc/tree-cfg.c index 202a69e2c8f..fa4800e00c2 100644 --- a/gcc/tree-cfg.c +++ b/gcc/tree-cfg.c @@ -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 (); }