X-Git-Url: http://nsz.repo.hu/git/?a=blobdiff_plain;ds=sidebyside;f=ir%2Flower%2Flower_dw.c;h=78e48d3e540c6698763fe53985efbefabe1ce194;hb=0ac0b440ce2d239c5e7e7db56b8273559d9a7741;hp=aacd53cfef1e50dadb3c37e8424fcebad238ca01;hpb=ab182d4b9ed44239ab3ff1d08b2f8e14a3699ccf;p=libfirm diff --git a/ir/lower/lower_dw.c b/ir/lower/lower_dw.c index aacd53cfe..78e48d3e5 100644 --- a/ir/lower/lower_dw.c +++ b/ir/lower/lower_dw.c @@ -1,38 +1,26 @@ /* - * Copyright (C) 1995-2008 University of Karlsruhe. All right reserved. - * * This file is part of libFirm. - * - * This file may be distributed and/or modified under the terms of the - * GNU General Public License version 2 as published by the Free Software - * Foundation and appearing in the file LICENSE.GPL included in the - * packaging of this file. - * - * Licensees holding valid libFirm Professional Edition licenses may use - * this file in accordance with the libFirm Commercial License. - * Agreement provided with the Software. - * - * This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE - * WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR - * PURPOSE. + * Copyright (C) 2012 University of Karlsruhe. */ /** * @file - * @brief Lower Double word operations, ie 64bit -> 32bit, 32bit -> 16bit etc. + * @brief Lower double word operations, i.e. 64bit -> 32bit, 32bit -> 16bit etc. * @date 8.10.2004 * @author Michael Beck - * @version $Id$ */ #include "config.h" #include #include +#include #include +#include "be.h" #include "error.h" #include "lowering.h" #include "irnode_t.h" +#include "irnodeset.h" #include "irgraph_t.h" #include "irmode_t.h" #include "iropt_t.h" @@ -45,6 +33,7 @@ #include "irgwalk.h" #include "ircons.h" #include "irflag.h" +#include "iroptimize.h" #include "irtools.h" #include "debug.h" #include "set.h" @@ -53,6 +42,7 @@ #include "irdump.h" #include "array_t.h" #include "irpass_t.h" +#include "lower_dw.h" /** A map from (op, imode, omode) to Intrinsic functions entities. */ static set *intrinsic_fkt; @@ -63,8 +53,14 @@ static set *conv_types; /** A map from a method type to its lowered type. */ static pmap *lowered_type; +/** A map from a builtin type to its lower and higher type. */ +static pmap *lowered_builtin_type_high; +static pmap *lowered_builtin_type_low; + /** The types for the binop and unop intrinsics. */ -static ir_type *binop_tp_u, *binop_tp_s, *unop_tp_u, *unop_tp_s, *shiftop_tp_u, *shiftop_tp_s, *tp_s, *tp_u; +static ir_type *binop_tp_u, *binop_tp_s, *unop_tp_u, *unop_tp_s, *tp_s, *tp_u; + +static ir_nodeset_t created_mux_nodes; /** the debug handle */ DEBUG_ONLY(static firm_dbg_module_t *dbg = NULL;) @@ -88,15 +84,6 @@ typedef struct conv_tp_entry { ir_type *mtd; /**< the associated method type of this (imode, omode) pair */ } conv_tp_entry_t; -/** - * Every double word node will be replaced, - * we need some store to hold the replacement: - */ -typedef struct node_entry_t { - ir_node *low_word; /**< the low word */ - ir_node *high_word; /**< the high word */ -} node_entry_t; - enum lower_flags { MUST_BE_LOWERED = 1, /**< graph must be lowered */ CF_CHANGED = 2, /**< control flow was changed */ @@ -105,26 +92,32 @@ enum lower_flags { /** * The lower environment. */ -typedef struct lower_env_t { - node_entry_t **entries; /**< entries per node */ - struct obstack obst; /**< an obstack holding the temporary data */ - ir_type *l_mtp; /**< lowered method type of the current method */ - tarval *tv_mode_bytes; /**< a tarval containing the number of bytes in the lowered modes */ - tarval *tv_mode_bits; /**< a tarval containing the number of bits in the lowered modes */ - pdeq *waitq; /**< a wait queue of all nodes that must be handled later */ - pmap *proj_2_block; /**< a map from ProjX to its destination blocks */ - ident *first_id; /**< .l for little and .h for big endian */ - ident *next_id; /**< .h for little and .l for big endian */ - const lwrdw_param_t *params; /**< transformation parameter */ - unsigned flags; /**< some flags */ - int n_entries; /**< number of entries */ - ir_type *value_param_tp; /**< the old value param type */ -} lower_env_t; +typedef struct lower_dw_env_t { + lower64_entry_t **entries; /**< entries per node */ + ir_graph *irg; + struct obstack obst; /**< an obstack holding the temporary data */ + ir_tarval *tv_mode_bytes; /**< a tarval containing the number of bytes in the lowered modes */ + pdeq *waitq; /**< a wait queue of all nodes that must be handled later */ + ir_node **lowered_phis; /**< list of lowered phis */ + ir_mode *high_signed; /**< doubleword signed type */ + ir_mode *high_unsigned; /**< doubleword unsigned type */ + ir_mode *low_signed; /**< word signed type */ + ir_mode *low_unsigned; /**< word unsigned type */ + ident *first_id; /**< .l for little and .h for big endian */ + ident *next_id; /**< .h for little and .l for big endian */ + const lwrdw_param_t *params; /**< transformation parameter */ + unsigned flags; /**< some flags */ + unsigned n_entries; /**< number of entries */ +} lower_dw_env_t; + +static lower_dw_env_t *env; + +static void lower_node(ir_node *node); /** * Create a method type for a Conv emulation from imode to omode. */ -static ir_type *get_conv_type(ir_mode *imode, ir_mode *omode, lower_env_t *env) +static ir_type *get_conv_type(ir_mode *imode, ir_mode *omode) { conv_tp_entry_t key, *entry; ir_type *mtd; @@ -133,13 +126,13 @@ static ir_type *get_conv_type(ir_mode *imode, ir_mode *omode, lower_env_t *env) key.omode = omode; key.mtd = NULL; - entry = set_insert(conv_types, &key, sizeof(key), HASH_PTR(imode) ^ HASH_PTR(omode)); + entry = set_insert(conv_tp_entry_t, conv_types, &key, sizeof(key), hash_ptr(imode) ^ hash_ptr(omode)); if (! entry->mtd) { int n_param = 1, n_res = 1; - if (imode == env->params->high_signed || imode == env->params->high_unsigned) + if (imode == env->high_signed || imode == env->high_unsigned) n_param = 2; - if (omode == env->params->high_signed || omode == env->params->high_unsigned) + if (omode == env->high_signed || omode == env->high_unsigned) n_res = 2; /* create a new one */ @@ -147,34 +140,44 @@ static ir_type *get_conv_type(ir_mode *imode, ir_mode *omode, lower_env_t *env) /* set param types and result types */ n_param = 0; - if (imode == env->params->high_signed) { - set_method_param_type(mtd, n_param++, tp_u); - set_method_param_type(mtd, n_param++, tp_s); - } else if (imode == env->params->high_unsigned) { + if (imode == env->high_signed) { + if (env->params->little_endian) { + set_method_param_type(mtd, n_param++, tp_u); + set_method_param_type(mtd, n_param++, tp_s); + } else { + set_method_param_type(mtd, n_param++, tp_s); + set_method_param_type(mtd, n_param++, tp_u); + } + } else if (imode == env->high_unsigned) { set_method_param_type(mtd, n_param++, tp_u); set_method_param_type(mtd, n_param++, tp_u); } else { ir_type *tp = get_type_for_mode(imode); set_method_param_type(mtd, n_param++, tp); - } /* if */ + } n_res = 0; - if (omode == env->params->high_signed) { - set_method_res_type(mtd, n_res++, tp_u); - set_method_res_type(mtd, n_res++, tp_s); - } else if (omode == env->params->high_unsigned) { + if (omode == env->high_signed) { + if (env->params->little_endian) { + set_method_res_type(mtd, n_res++, tp_u); + set_method_res_type(mtd, n_res++, tp_s); + } else { + set_method_res_type(mtd, n_res++, tp_s); + set_method_res_type(mtd, n_res++, tp_u); + } + } else if (omode == env->high_unsigned) { set_method_res_type(mtd, n_res++, tp_u); set_method_res_type(mtd, n_res++, tp_u); } else { ir_type *tp = get_type_for_mode(omode); set_method_res_type(mtd, n_res++, tp); - } /* if */ + } entry->mtd = mtd; } else { mtd = entry->mtd; - } /* if */ + } return mtd; -} /* get_conv_type */ +} /** * Add an additional control flow input to a block. @@ -183,25 +186,29 @@ static ir_type *get_conv_type(ir_mode *imode, ir_mode *omode, lower_env_t *env) */ static void add_block_cf_input_nr(ir_node *block, int nr, ir_node *cf) { - int i, arity = get_irn_arity(block); - ir_node **in, *phi; + int i, arity = get_Block_n_cfgpreds(block); + ir_node **in; assert(nr < arity); NEW_ARR_A(ir_node *, in, arity + 1); for (i = 0; i < arity; ++i) - in[i] = get_irn_n(block, i); + in[i] = get_Block_cfgpred(block, i); in[i] = cf; set_irn_in(block, i + 1, in); - for (phi = get_Block_phis(block); phi != NULL; phi = get_Phi_next(phi)) { + foreach_out_edge(block, edge) { + ir_node *phi = get_edge_src_irn(edge); + if (!is_Phi(phi)) + continue; + for (i = 0; i < arity; ++i) in[i] = get_irn_n(phi, i); in[i] = in[nr]; set_irn_in(phi, i + 1, in); - } /* for */ -} /* add_block_cf_input_nr */ + } +} /** * Add an additional control flow input to a block. @@ -210,18 +217,18 @@ static void add_block_cf_input_nr(ir_node *block, int nr, ir_node *cf) */ static void add_block_cf_input(ir_node *block, ir_node *tmpl, ir_node *cf) { - int i, arity = get_irn_arity(block); + int i, arity = get_Block_n_cfgpreds(block); int nr = 0; for (i = 0; i < arity; ++i) { - if (get_irn_n(block, i) == tmpl) { + if (get_Block_cfgpred(block, i) == tmpl) { nr = i; break; - } /* if */ - } /* for */ + } + } assert(i < arity); add_block_cf_input_nr(block, nr, cf); -} /* add_block_cf_input */ +} /** * Return the "operational" mode of a Firm node. @@ -233,8 +240,6 @@ static ir_mode *get_irn_op_mode(ir_node *node) return get_Load_mode(node); case iro_Store: return get_irn_mode(get_Store_value(node)); - case iro_DivMod: - return get_irn_mode(get_DivMod_left(node)); case iro_Div: return get_irn_mode(get_Div_left(node)); case iro_Mod: @@ -243,112 +248,121 @@ static ir_mode *get_irn_op_mode(ir_node *node) return get_irn_mode(get_Cmp_left(node)); default: return get_irn_mode(node); - } /* switch */ -} /* get_irn_op_mode */ + } +} /** - * Walker, prepare the node links. + * Walker, prepare the node links and determine which nodes need to be lowered + * at all. */ -static void prepare_links(ir_node *node, void *env) +static void prepare_links(ir_node *node) { - lower_env_t *lenv = env; - ir_mode *mode = get_irn_op_mode(node); - node_entry_t *link; - int i, idx; + ir_mode *mode = get_irn_op_mode(node); + lower64_entry_t *link; - if (mode == lenv->params->high_signed || - mode == lenv->params->high_unsigned) { + if (mode == env->high_signed || mode == env->high_unsigned) { + unsigned idx = get_irn_idx(node); /* ok, found a node that will be lowered */ - link = OALLOCZ(&lenv->obst, node_entry_t); + link = OALLOCZ(&env->obst, lower64_entry_t); - idx = get_irn_idx(node); - if (idx >= lenv->n_entries) { + if (idx >= env->n_entries) { /* enlarge: this happens only for Rotl nodes which is RARELY */ - int old = lenv->n_entries; - int n_idx = idx + (idx >> 3); + unsigned old = env->n_entries; + unsigned n_idx = idx + (idx >> 3); - ARR_RESIZE(node_entry_t *, lenv->entries, n_idx); - memset(&lenv->entries[old], 0, (n_idx - old) * sizeof(lenv->entries[0])); - lenv->n_entries = n_idx; + ARR_RESIZE(lower64_entry_t *, env->entries, n_idx); + memset(&env->entries[old], 0, (n_idx - old) * sizeof(env->entries[0])); + env->n_entries = n_idx; } - lenv->entries[idx] = link; - lenv->flags |= MUST_BE_LOWERED; + env->entries[idx] = link; + env->flags |= MUST_BE_LOWERED; } else if (is_Conv(node)) { /* Conv nodes have two modes */ ir_node *pred = get_Conv_op(node); mode = get_irn_mode(pred); - if (mode == lenv->params->high_signed || - mode == lenv->params->high_unsigned) { + if (mode == env->high_signed || mode == env->high_unsigned) { /* must lower this node either but don't need a link */ - lenv->flags |= MUST_BE_LOWERED; - } /* if */ + env->flags |= MUST_BE_LOWERED; + } return; - } /* if */ + } else if (is_Call(node)) { + /* Special case: If the result of the Call is never used, we won't + * find a Proj with a mode that potentially triggers MUST_BE_LOWERED + * to be set. Thus, if we see a call, we check its result types and + * decide whether MUST_BE_LOWERED has to be set. + */ + ir_type *tp = get_Call_type(node); + size_t n_res, i; + + n_res = get_method_n_ress(tp); + for (i = 0; i < n_res; ++i) { + ir_type *rtp = get_method_res_type(tp, i); + + if (is_Primitive_type(rtp)) { + ir_mode *rmode = get_type_mode(rtp); + + if (rmode == env->high_signed || rmode == env->high_unsigned) { + env->flags |= MUST_BE_LOWERED; + } + } + } + } +} - if (is_Proj(node)) { - /* link all Proj nodes to its predecessor: - Note that Tuple Proj's and its Projs are linked either. */ - ir_node *pred = get_Proj_pred(node); +lower64_entry_t *get_node_entry(ir_node *node) +{ + unsigned idx = get_irn_idx(node); + assert(idx < env->n_entries); + return env->entries[idx]; +} - set_irn_link(node, get_irn_link(pred)); - set_irn_link(pred, node); - } else if (is_Phi(node)) { - /* link all Phi nodes to its block */ - ir_node *block = get_nodes_block(node); - add_Block_phi(block, node); - } else if (is_Block(node)) { - /* fill the Proj -> Block map */ - for (i = get_Block_n_cfgpreds(node) - 1; i >= 0; --i) { - ir_node *pred = get_Block_cfgpred(node, i); - - if (is_Proj(pred)) - pmap_insert(lenv->proj_2_block, pred, node); - } /* for */ - } /* if */ -} /* prepare_links */ +void ir_set_dw_lowered(ir_node *old, ir_node *new_low, ir_node *new_high) +{ + lower64_entry_t *entry = get_node_entry(old); + entry->low_word = new_low; + entry->high_word = new_high; +} + +ir_mode *ir_get_low_unsigned_mode(void) +{ + return env->low_unsigned; +} /** * Translate a Constant: create two. */ -static void lower_Const(ir_node *node, ir_mode *mode, lower_env_t *env) +static void lower_Const(ir_node *node, ir_mode *mode) { - tarval *tv, *tv_l, *tv_h; - ir_node *low, *high; - dbg_info *dbg = get_irn_dbg_info(node); - int idx; - ir_graph *irg = current_ir_graph; - ir_mode *low_mode = env->params->low_unsigned; - - tv = get_Const_tarval(node); - - tv_l = tarval_convert_to(tv, low_mode); - low = new_rd_Const(dbg, irg, tv_l); - - tv_h = tarval_convert_to(tarval_shrs(tv, env->tv_mode_bits), mode); - high = new_rd_Const(dbg, irg, tv_h); - - idx = get_irn_idx(node); - assert(idx < env->n_entries); - env->entries[idx]->low_word = low; - env->entries[idx]->high_word = high; -} /* lower_Const */ + ir_graph *irg = get_irn_irg(node); + dbg_info *dbg = get_irn_dbg_info(node); + ir_mode *low_mode = env->low_unsigned; + ir_tarval *tv = get_Const_tarval(node); + ir_tarval *tv_l = tarval_convert_to(tv, low_mode); + ir_node *res_low = new_rd_Const(dbg, irg, tv_l); + ir_tarval *tv_shrs = tarval_shrs_unsigned(tv, get_mode_size_bits(low_mode)); + ir_tarval *tv_h = tarval_convert_to(tv_shrs, mode); + ir_node *res_high = new_rd_Const(dbg, irg, tv_h); + + ir_set_dw_lowered(node, res_low, res_high); +} /** * Translate a Load: create two. */ -static void lower_Load(ir_node *node, ir_mode *mode, lower_env_t *env) +static void lower_Load(ir_node *node, ir_mode *mode) { - ir_mode *low_mode = env->params->low_unsigned; - ir_graph *irg = current_ir_graph; + ir_mode *low_mode = env->low_unsigned; + ir_graph *irg = get_irn_irg(node); ir_node *adr = get_Load_ptr(node); ir_node *mem = get_Load_mem(node); - ir_node *low, *high, *proj; + ir_node *low; + ir_node *high; + ir_node *proj_m; dbg_info *dbg; ir_node *block = get_nodes_block(node); - int idx; ir_cons_flags volatility = get_Load_volatility(node) == volatility_is_volatile - ? cons_volatile : 0; + ? cons_volatile : cons_none; if (env->params->little_endian) { low = adr; @@ -356,21 +370,18 @@ static void lower_Load(ir_node *node, ir_mode *mode, lower_env_t *env) } else { low = new_r_Add(block, adr, new_r_Const(irg, env->tv_mode_bytes), get_irn_mode(adr)); high = adr; - } /* if */ + } /* create two loads */ - dbg = get_irn_dbg_info(node); - low = new_rd_Load(dbg, block, mem, low, low_mode, volatility); - proj = new_r_Proj(low, mode_M, pn_Load_M); - high = new_rd_Load(dbg, block, proj, high, mode, volatility); - - idx = get_irn_idx(node); - assert(idx < env->n_entries); - env->entries[idx]->low_word = low; - env->entries[idx]->high_word = high; - - for (proj = get_irn_link(node); proj; proj = get_irn_link(proj)) { - idx = get_irn_idx(proj); + dbg = get_irn_dbg_info(node); + low = new_rd_Load(dbg, block, mem, low, low_mode, volatility); + proj_m = new_r_Proj(low, mode_M, pn_Load_M); + high = new_rd_Load(dbg, block, proj_m, high, mode, volatility); + + foreach_out_edge_safe(node, edge) { + ir_node *proj = get_edge_src_irn(edge); + if (!is_Proj(proj)) + continue; switch (get_Proj_proj(proj)) { case pn_Load_M: /* Memory result. */ @@ -381,46 +392,45 @@ static void lower_Load(ir_node *node, ir_mode *mode, lower_env_t *env) /* put it to the first one */ set_Proj_pred(proj, low); break; - case pn_Load_res: /* Result of load operation. */ - assert(idx < env->n_entries); - env->entries[idx]->low_word = new_r_Proj(low, low_mode, pn_Load_res); - env->entries[idx]->high_word = new_r_Proj(high, mode, pn_Load_res); + case pn_Load_res: { /* Result of load operation. */ + ir_node *res_low = new_r_Proj(low, low_mode, pn_Load_res); + ir_node *res_high = new_r_Proj(high, mode, pn_Load_res); + ir_set_dw_lowered(proj, res_low, res_high); break; + } default: assert(0 && "unexpected Proj number"); - } /* switch */ - /* mark this proj: we have handled it already, otherwise we might fall into - * out new nodes. */ + } + /* mark this proj: we have handled it already, otherwise we might fall + * into out new nodes. */ mark_irn_visited(proj); - } /* for */ -} /* lower_Load */ + } +} /** * Translate a Store: create two. */ -static void lower_Store(ir_node *node, ir_mode *mode, lower_env_t *env) -{ - ir_graph *irg; - ir_node *block, *adr, *mem; - ir_node *low, *high, *irn, *proj; - dbg_info *dbg; - int idx; - node_entry_t *entry; - ir_cons_flags volatility = get_Store_volatility(node) == volatility_is_volatile - ? cons_volatile : 0; +static void lower_Store(ir_node *node, ir_mode *mode) +{ + ir_graph *irg; + ir_node *block, *adr, *mem; + ir_node *low, *high, *proj_m; + dbg_info *dbg; + ir_node *value = get_Store_value(node); + const lower64_entry_t *entry = get_node_entry(value); + ir_cons_flags volatility = get_Store_volatility(node) == volatility_is_volatile + ? cons_volatile : cons_none; (void) mode; - irn = get_Store_value(node); - entry = env->entries[get_irn_idx(irn)]; assert(entry); if (! entry->low_word) { /* not ready yet, wait */ pdeq_putr(env->waitq, node); return; - } /* if */ + } - irg = current_ir_graph; + irg = get_irn_irg(node); adr = get_Store_ptr(node); mem = get_Store_mem(node); block = get_nodes_block(node); @@ -431,21 +441,18 @@ static void lower_Store(ir_node *node, ir_mode *mode, lower_env_t *env) } else { low = new_r_Add(block, adr, new_r_Const(irg, env->tv_mode_bytes), get_irn_mode(adr)); high = adr; - } /* if */ + } /* create two Stores */ - dbg = get_irn_dbg_info(node); - low = new_rd_Store(dbg, block, mem, low, entry->low_word, volatility); - proj = new_r_Proj(low, mode_M, pn_Store_M); - high = new_rd_Store(dbg, block, proj, high, entry->high_word, volatility); - - idx = get_irn_idx(node); - assert(idx < env->n_entries); - env->entries[idx]->low_word = low; - env->entries[idx]->high_word = high; - - for (proj = get_irn_link(node); proj; proj = get_irn_link(proj)) { - idx = get_irn_idx(proj); + dbg = get_irn_dbg_info(node); + low = new_rd_Store(dbg, block, mem, low, entry->low_word, volatility); + proj_m = new_r_Proj(low, mode_M, pn_Store_M); + high = new_rd_Store(dbg, block, proj_m, high, entry->high_word, volatility); + + foreach_out_edge_safe(node, edge) { + ir_node *proj = get_edge_src_irn(edge); + if (!is_Proj(proj)) + continue; switch (get_Proj_proj(proj)) { case pn_Store_M: /* Memory result. */ @@ -458,12 +465,12 @@ static void lower_Store(ir_node *node, ir_mode *mode, lower_env_t *env) break; default: assert(0 && "unexpected Proj number"); - } /* switch */ + } /* mark this proj: we have handled it already, otherwise we might fall into * out new nodes. */ mark_irn_visited(proj); - } /* for */ -} /* lower_Store */ + } +} /** * Return a node containing the address of the intrinsic emulation function. @@ -475,8 +482,7 @@ static void lower_Store(ir_node *node, ir_mode *mode, lower_env_t *env) * @param env the lower environment */ static ir_node *get_intrinsic_address(ir_type *method, ir_op *op, - ir_mode *imode, ir_mode *omode, - lower_env_t *env) + ir_mode *imode, ir_mode *omode) { symconst_symbol sym; ir_entity *ent; @@ -487,8 +493,8 @@ static ir_node *get_intrinsic_address(ir_type *method, ir_op *op, key.omode = omode; key.ent = NULL; - entry = set_insert(intrinsic_fkt, &key, sizeof(key), - HASH_PTR(op) ^ HASH_PTR(imode) ^ (HASH_PTR(omode) << 8)); + entry = set_insert(op_mode_entry_t, intrinsic_fkt, &key, sizeof(key), + hash_ptr(op) ^ hash_ptr(imode) ^ (hash_ptr(omode) << 8)); if (! entry->ent) { /* create a new one */ ent = env->params->create_intrinsic(method, op, imode, omode, env->params->ctx); @@ -497,1118 +503,1189 @@ static ir_node *get_intrinsic_address(ir_type *method, ir_op *op, entry->ent = ent; } else { ent = entry->ent; - } /* if */ + } sym.entity_p = ent; - return new_r_SymConst(current_ir_graph, mode_P_code, sym, symconst_addr_ent); -} /* get_intrinsic_address */ + return new_r_SymConst(env->irg, mode_P_code, sym, symconst_addr_ent); +} /** * Translate a Div. * * Create an intrinsic Call. */ -static void lower_Div(ir_node *node, ir_mode *mode, lower_env_t *env) +static void lower_Div(ir_node *node, ir_mode *mode) { - ir_node *block, *irn, *call, *proj; + ir_node *left = get_Div_left(node); + ir_node *right = get_Div_right(node); + ir_node *block = get_nodes_block(node); + dbg_info *dbgi = get_irn_dbg_info(node); + ir_type *mtp = mode_is_signed(mode) ? binop_tp_s : binop_tp_u; + ir_mode *opmode = get_irn_op_mode(node); + ir_node *addr = get_intrinsic_address(mtp, get_irn_op(node), opmode, opmode); ir_node *in[4]; - ir_mode *opmode; - dbg_info *dbg; - ir_type *mtp; - int idx; - node_entry_t *entry; - - irn = get_Div_left(node); - entry = env->entries[get_irn_idx(irn)]; - assert(entry); - - if (! entry->low_word) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; - } /* if */ - - in[0] = entry->low_word; - in[1] = entry->high_word; - - irn = get_Div_right(node); - entry = env->entries[get_irn_idx(irn)]; - assert(entry); - - if (! entry->low_word) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; - } /* if */ - - in[2] = entry->low_word; - in[3] = entry->high_word; - - dbg = get_irn_dbg_info(node); - block = get_nodes_block(node); + ir_node *call; + ir_node *resproj; - mtp = mode_is_signed(mode) ? binop_tp_s : binop_tp_u; - opmode = get_irn_op_mode(node); - irn = get_intrinsic_address(mtp, get_irn_op(node), opmode, opmode, env); - call = new_rd_Call(dbg, block, get_Div_mem(node), irn, 4, in, mtp); + if (env->params->little_endian) { + in[0] = get_lowered_low(left); + in[1] = get_lowered_high(left); + in[2] = get_lowered_low(right); + in[3] = get_lowered_high(right); + } else { + in[0] = get_lowered_high(left); + in[1] = get_lowered_low(left); + in[2] = get_lowered_high(right); + in[3] = get_lowered_low(right); + } + call = new_rd_Call(dbgi, block, get_Div_mem(node), addr, 4, in, mtp); + resproj = new_r_Proj(call, mode_T, pn_Call_T_result); set_irn_pinned(call, get_irn_pinned(node)); - irn = new_r_Proj(call, mode_T, pn_Call_T_result); - for (proj = get_irn_link(node); proj; proj = get_irn_link(proj)) { + foreach_out_edge_safe(node, edge) { + ir_node *proj = get_edge_src_irn(edge); + if (!is_Proj(proj)) + continue; + switch (get_Proj_proj(proj)) { case pn_Div_M: /* Memory result. */ /* reroute to the call */ set_Proj_pred(proj, call); set_Proj_proj(proj, pn_Call_M); break; - case pn_Div_X_except: /* Execution result if exception occurred. */ - /* reroute to the call */ + case pn_Div_X_regular: + set_Proj_pred(proj, call); + set_Proj_proj(proj, pn_Call_X_regular); + break; + case pn_Div_X_except: set_Proj_pred(proj, call); set_Proj_proj(proj, pn_Call_X_except); break; - case pn_Div_res: /* Result of computation. */ - idx = get_irn_idx(proj); - assert(idx < env->n_entries); - env->entries[idx]->low_word = new_r_Proj(irn, env->params->low_unsigned, 0); - env->entries[idx]->high_word = new_r_Proj(irn, mode, 1); + case pn_Div_res: + if (env->params->little_endian) { + ir_node *res_low = new_r_Proj(resproj, env->low_unsigned, 0); + ir_node *res_high = new_r_Proj(resproj, mode, 1); + ir_set_dw_lowered(proj, res_low, res_high); + } else { + ir_node *res_low = new_r_Proj(resproj, env->low_unsigned, 1); + ir_node *res_high = new_r_Proj(resproj, mode, 0); + ir_set_dw_lowered(proj, res_low, res_high); + } break; default: assert(0 && "unexpected Proj number"); - } /* switch */ + } /* mark this proj: we have handled it already, otherwise we might fall into * out new nodes. */ mark_irn_visited(proj); - } /* for */ -} /* lower_Div */ + } +} /** * Translate a Mod. * * Create an intrinsic Call. */ -static void lower_Mod(ir_node *node, ir_mode *mode, lower_env_t *env) +static void lower_Mod(ir_node *node, ir_mode *mode) { - ir_node *block, *proj, *irn, *call; + ir_node *left = get_Mod_left(node); + ir_node *right = get_Mod_right(node); + dbg_info *dbgi = get_irn_dbg_info(node); + ir_node *block = get_nodes_block(node); + ir_type *mtp = mode_is_signed(mode) ? binop_tp_s : binop_tp_u; + ir_mode *opmode = get_irn_op_mode(node); + ir_node *addr = get_intrinsic_address(mtp, get_irn_op(node), opmode, opmode); ir_node *in[4]; - ir_mode *opmode; - dbg_info *dbg; - ir_type *mtp; - int idx; - node_entry_t *entry; - - irn = get_Mod_left(node); - entry = env->entries[get_irn_idx(irn)]; - assert(entry); - - if (! entry->low_word) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; - } /* if */ - - in[0] = entry->low_word; - in[1] = entry->high_word; - - irn = get_Mod_right(node); - entry = env->entries[get_irn_idx(irn)]; - assert(entry); - - if (! entry->low_word) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; - } /* if */ - - in[2] = entry->low_word; - in[3] = entry->high_word; - - dbg = get_irn_dbg_info(node); - block = get_nodes_block(node); + ir_node *call; + ir_node *resproj; - mtp = mode_is_signed(mode) ? binop_tp_s : binop_tp_u; - opmode = get_irn_op_mode(node); - irn = get_intrinsic_address(mtp, get_irn_op(node), opmode, opmode, env); - call = new_rd_Call(dbg, block, get_Mod_mem(node), irn, 4, in, mtp); + if (env->params->little_endian) { + in[0] = get_lowered_low(left); + in[1] = get_lowered_high(left); + in[2] = get_lowered_low(right); + in[3] = get_lowered_high(right); + } else { + in[0] = get_lowered_high(left); + in[1] = get_lowered_low(left); + in[2] = get_lowered_high(right); + in[3] = get_lowered_low(right); + } + call = new_rd_Call(dbgi, block, get_Mod_mem(node), addr, 4, in, mtp); + resproj = new_r_Proj(call, mode_T, pn_Call_T_result); set_irn_pinned(call, get_irn_pinned(node)); - irn = new_r_Proj(call, mode_T, pn_Call_T_result); - for (proj = get_irn_link(node); proj; proj = get_irn_link(proj)) { + foreach_out_edge_safe(node, edge) { + ir_node *proj = get_edge_src_irn(edge); + if (!is_Proj(proj)) + continue; + switch (get_Proj_proj(proj)) { case pn_Mod_M: /* Memory result. */ /* reroute to the call */ set_Proj_pred(proj, call); set_Proj_proj(proj, pn_Call_M); break; - case pn_Mod_X_except: /* Execution result if exception occurred. */ - /* reroute to the call */ + case pn_Div_X_regular: + set_Proj_pred(proj, call); + set_Proj_proj(proj, pn_Call_X_regular); + break; + case pn_Mod_X_except: set_Proj_pred(proj, call); set_Proj_proj(proj, pn_Call_X_except); break; - case pn_Mod_res: /* Result of computation. */ - idx = get_irn_idx(proj); - assert(idx < env->n_entries); - env->entries[idx]->low_word = new_r_Proj(irn, env->params->low_unsigned, 0); - env->entries[idx]->high_word = new_r_Proj(irn, mode, 1); + case pn_Mod_res: + if (env->params->little_endian) { + ir_node *res_low = new_r_Proj(resproj, env->low_unsigned, 0); + ir_node *res_high = new_r_Proj(resproj, mode, 1); + ir_set_dw_lowered(proj, res_low, res_high); + } else { + ir_node *res_low = new_r_Proj(resproj, env->low_unsigned, 1); + ir_node *res_high = new_r_Proj(resproj, mode, 0); + ir_set_dw_lowered(proj, res_low, res_high); + } break; default: assert(0 && "unexpected Proj number"); - } /* switch */ - /* mark this proj: we have handled it already, otherwise we might fall into - * out new nodes. */ + } + /* mark this proj: we have handled it already, otherwise we might fall + * into out new nodes. */ mark_irn_visited(proj); - } /* for */ -} /* lower_Mod */ + } +} /** - * Translate a DivMod. + * Translate a binop. * - * Create two intrinsic Calls. + * Create an intrinsic Call. */ -static void lower_DivMod(ir_node *node, ir_mode *mode, lower_env_t *env) +static void lower_binop(ir_node *node, ir_mode *mode) { - ir_node *block, *proj, *irn, *mem, *callDiv, *callMod; - ir_node *resDiv = NULL; - ir_node *resMod = NULL; + ir_node *left = get_binop_left(node); + ir_node *right = get_binop_right(node); + dbg_info *dbgi = get_irn_dbg_info(node); + ir_node *block = get_nodes_block(node); + ir_graph *irg = get_irn_irg(block); + ir_type *mtp = mode_is_signed(mode) ? binop_tp_s : binop_tp_u; + ir_node *addr = get_intrinsic_address(mtp, get_irn_op(node), mode, mode); ir_node *in[4]; - ir_mode *opmode; - dbg_info *dbg; - ir_type *mtp; - int idx; - node_entry_t *entry; - unsigned flags = 0; + ir_node *call; + ir_node *resproj; - /* check if both results are needed */ - for (proj = get_irn_link(node); proj; proj = get_irn_link(proj)) { - switch (get_Proj_proj(proj)) { - case pn_DivMod_res_div: flags |= 1; break; - case pn_DivMod_res_mod: flags |= 2; break; - default: break; - } /* switch */ - } /* for */ - - irn = get_DivMod_left(node); - entry = env->entries[get_irn_idx(irn)]; - assert(entry); + if (env->params->little_endian) { + in[0] = get_lowered_low(left); + in[1] = get_lowered_high(left); + in[2] = get_lowered_low(right); + in[3] = get_lowered_high(right); + } else { + in[0] = get_lowered_high(left); + in[1] = get_lowered_low(left); + in[2] = get_lowered_high(right); + in[3] = get_lowered_low(right); + } + call = new_rd_Call(dbgi, block, get_irg_no_mem(irg), addr, 4, in, mtp); + resproj = new_r_Proj(call, mode_T, pn_Call_T_result); + set_irn_pinned(call, get_irn_pinned(node)); - if (! entry->low_word) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; - } /* if */ + if (env->params->little_endian) { + ir_node *res_low = new_r_Proj(resproj, env->low_unsigned, 0); + ir_node *res_high = new_r_Proj(resproj, mode, 1); + ir_set_dw_lowered(node, res_low, res_high); + } else { + ir_node *res_low = new_r_Proj(resproj, env->low_unsigned, 1); + ir_node *res_high = new_r_Proj(resproj, mode, 0); + ir_set_dw_lowered(node, res_low, res_high); + } +} - in[0] = entry->low_word; - in[1] = entry->high_word; +static ir_node *create_conv(ir_node *block, ir_node *node, ir_mode *dest_mode) +{ + if (get_irn_mode(node) == dest_mode) + return node; + return new_r_Conv(block, node, dest_mode); +} - irn = get_DivMod_right(node); - entry = env->entries[get_irn_idx(irn)]; - assert(entry); +/** + * Moves node and all predecessors of node from from_bl to to_bl. + * Does not move predecessors of Phi nodes (or block nodes). + */ +static void move(ir_node *node, ir_node *from_bl, ir_node *to_bl) +{ + int i, arity; + + /* move this node */ + set_nodes_block(node, to_bl); + + /* move its Projs */ + if (get_irn_mode(node) == mode_T) { + foreach_out_edge(node, edge) { + ir_node *proj = get_edge_src_irn(edge); + if (!is_Proj(proj)) + continue; + move(proj, from_bl, to_bl); + } + } - if (! entry->low_word) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); + /* We must not move predecessors of Phi nodes, even if they are in + * from_bl. (because these are values from an earlier loop iteration + * which are not predecessors of node here) + */ + if (is_Phi(node)) return; - } /* if */ - - in[2] = entry->low_word; - in[3] = entry->high_word; - - dbg = get_irn_dbg_info(node); - block = get_nodes_block(node); - mem = get_DivMod_mem(node); - - callDiv = callMod = NULL; - mtp = mode_is_signed(mode) ? binop_tp_s : binop_tp_u; - if (flags & 1) { - opmode = get_irn_op_mode(node); - irn = get_intrinsic_address(mtp, op_Div, opmode, opmode, env); - callDiv = new_rd_Call(dbg, block, mem, irn, 4, in, mtp); - set_irn_pinned(callDiv, get_irn_pinned(node)); - resDiv = new_r_Proj(callDiv, mode_T, pn_Call_T_result); - } /* if */ - if (flags & 2) { - if (flags & 1) - mem = new_r_Proj(callDiv, mode_M, pn_Call_M); - opmode = get_irn_op_mode(node); - irn = get_intrinsic_address(mtp, op_Mod, opmode, opmode, env); - callMod = new_rd_Call(dbg, block, mem, irn, 4, in, mtp); - set_irn_pinned(callMod, get_irn_pinned(node)); - resMod = new_r_Proj(callMod, mode_T, pn_Call_T_result); - } /* if */ - - for (proj = get_irn_link(node); proj; proj = get_irn_link(proj)) { - switch (get_Proj_proj(proj)) { - case pn_DivMod_M: /* Memory result. */ - /* reroute to the first call */ - set_Proj_pred(proj, callDiv ? callDiv : (callMod ? callMod : mem)); - set_Proj_proj(proj, pn_Call_M); - break; - case pn_DivMod_X_except: /* Execution result if exception occurred. */ - /* reroute to the first call */ - set_Proj_pred(proj, callDiv ? callDiv : (callMod ? callMod : mem)); - set_Proj_proj(proj, pn_Call_X_except); - break; - case pn_DivMod_res_div: /* Result of Div. */ - idx = get_irn_idx(proj); - assert(idx < env->n_entries); - env->entries[idx]->low_word = new_r_Proj(resDiv, env->params->low_unsigned, 0); - env->entries[idx]->high_word = new_r_Proj(resDiv, mode, 1); - break; - case pn_DivMod_res_mod: /* Result of Mod. */ - idx = get_irn_idx(proj); - env->entries[idx]->low_word = new_r_Proj(resMod, env->params->low_unsigned, 0); - env->entries[idx]->high_word = new_r_Proj(resMod, mode, 1); - break; - default: - assert(0 && "unexpected Proj number"); - } /* switch */ - /* mark this proj: we have handled it already, otherwise we might fall into - * out new nodes. */ - mark_irn_visited(proj); - } /* for */ -} /* lower_DivMod */ + /* recursion ... */ + arity = get_irn_arity(node); + for (i = 0; i < arity; i++) { + ir_node *pred = get_irn_n(node, i); + ir_mode *pred_mode = get_irn_mode(pred); + if (get_nodes_block(pred) == from_bl) + move(pred, from_bl, to_bl); + if (pred_mode == env->high_signed || pred_mode == env->high_unsigned) { + ir_node *pred_low = get_lowered_low(pred); + ir_node *pred_high = get_lowered_high(pred); + if (get_nodes_block(pred_low) == from_bl) + move(pred_low, from_bl, to_bl); + if (pred_high != NULL && get_nodes_block(pred_high) == from_bl) + move(pred_high, from_bl, to_bl); + } + } +} /** - * Translate a Binop. - * - * Create an intrinsic Call. + * We need a custom version of part_block_edges because during transformation + * not all data-dependencies are explicit yet if a lowered nodes users are not + * lowered yet. + * We can fix this by modifying move to look for such implicit dependencies. + * Additionally we have to keep the proj_2_block map updated */ -static void lower_Binop(ir_node *node, ir_mode *mode, lower_env_t *env) +static ir_node *part_block_dw(ir_node *node) { - ir_node *block, *irn; - ir_node *in[4]; - dbg_info *dbg; - ir_type *mtp; - int idx; - ir_graph *irg; - node_entry_t *entry; - - irn = get_binop_left(node); - entry = env->entries[get_irn_idx(irn)]; - assert(entry); - - if (! entry->low_word) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; - } /* if */ + ir_graph *irg = get_irn_irg(node); + ir_node *old_block = get_nodes_block(node); + int n_cfgpreds = get_Block_n_cfgpreds(old_block); + ir_node **cfgpreds = get_Block_cfgpred_arr(old_block); + ir_node *new_block = new_r_Block(irg, n_cfgpreds, cfgpreds); + + /* old_block has no predecessors anymore for now */ + set_irn_in(old_block, 0, NULL); + + /* move node and its predecessors to new_block */ + move(node, old_block, new_block); + + /* move Phi nodes to new_block */ + foreach_out_edge_safe(old_block, edge) { + ir_node *phi = get_edge_src_irn(edge); + if (!is_Phi(phi)) + continue; + set_nodes_block(phi, new_block); + } + return old_block; +} - in[0] = entry->low_word; - in[1] = entry->high_word; +typedef ir_node* (*new_rd_shr_func)(dbg_info *dbgi, ir_node *block, + ir_node *left, ir_node *right, + ir_mode *mode); - irn = get_binop_right(node); - entry = env->entries[get_irn_idx(irn)]; - assert(entry); +static void lower_shr_helper(ir_node *node, ir_mode *mode, + new_rd_shr_func new_rd_shrs) +{ + ir_node *right = get_binop_right(node); + ir_node *left = get_binop_left(node); + ir_mode *shr_mode = get_irn_mode(node); + unsigned modulo_shift = get_mode_modulo_shift(shr_mode); + ir_mode *low_unsigned = env->low_unsigned; + unsigned modulo_shift2 = get_mode_modulo_shift(mode); + ir_graph *irg = get_irn_irg(node); + ir_node *left_low = get_lowered_low(left); + ir_node *left_high = get_lowered_high(left); + dbg_info *dbgi = get_irn_dbg_info(node); + ir_node *lower_block; + ir_node *block; + ir_node *cnst; + ir_node *andn; + ir_node *cmp; + ir_node *cond; + ir_node *proj_true; + ir_node *proj_false; + ir_node *phi_low; + ir_node *phi_high; + ir_node *lower_in[2]; + ir_node *phi_low_in[2]; + ir_node *phi_high_in[2]; + + /* this version is optimized for modulo shift architectures + * (and can't handle anything else) */ + if (modulo_shift != get_mode_size_bits(shr_mode) + || modulo_shift2<<1 != modulo_shift) { + panic("Shr lowering only implemented for modulo shift shr operations"); + } + if (!is_po2(modulo_shift) || !is_po2(modulo_shift2)) { + panic("Shr lowering only implemented for power-of-2 modes"); + } + /* without 2-complement the -x instead of (bit_width-x) trick won't work */ + if (get_mode_arithmetic(shr_mode) != irma_twos_complement) { + panic("Shr lowering only implemented for two-complement modes"); + } - if (! entry->low_word) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; - } /* if */ + block = get_nodes_block(node); - in[2] = entry->low_word; - in[3] = entry->high_word; + /* if the right operand is a 64bit value, we're only interested in the + * lower word */ + if (get_irn_mode(right) == env->high_unsigned) { + right = get_lowered_low(right); + } else { + /* shift should never have signed mode on the right */ + assert(get_irn_mode(right) != env->high_signed); + right = create_conv(block, right, low_unsigned); + } - dbg = get_irn_dbg_info(node); + lower_block = part_block_dw(node); + env->flags |= CF_CHANGED; block = get_nodes_block(node); - irg = current_ir_graph; - mtp = mode_is_signed(mode) ? binop_tp_s : binop_tp_u; - irn = get_intrinsic_address(mtp, get_irn_op(node), mode, mode, env); - irn = new_rd_Call(dbg, block, get_irg_no_mem(current_ir_graph), - irn, 4, in, mtp); - set_irn_pinned(irn, get_irn_pinned(node)); - irn = new_r_Proj(irn, mode_T, pn_Call_T_result); + /* add a Cmp to test if highest bit is set <=> whether we shift more + * than half the word width */ + cnst = new_r_Const_long(irg, low_unsigned, modulo_shift2); + andn = new_r_And(block, right, cnst, low_unsigned); + cnst = new_r_Const(irg, get_mode_null(low_unsigned)); + cmp = new_rd_Cmp(dbgi, block, andn, cnst, ir_relation_equal); + cond = new_rd_Cond(dbgi, block, cmp); + proj_true = new_r_Proj(cond, mode_X, pn_Cond_true); + proj_false = new_r_Proj(cond, mode_X, pn_Cond_false); + + /* the true block => shift_width < 1word */ + { + /* In theory the low value (for 64bit shifts) is: + * Or(High << (32-x)), Low >> x) + * In practice High << 32-x will fail when x is zero (since we have + * modulo shift and 32 will be 0). So instead we use: + * Or(High<<1<<~x, Low >> x) + */ + ir_node *in[1] = { proj_true }; + ir_node *block_true = new_r_Block(irg, ARRAY_SIZE(in), in); + ir_node *res_high = new_rd_shrs(dbgi, block_true, left_high, + right, mode); + ir_node *shift_low = new_rd_Shr(dbgi, block_true, left_low, right, + low_unsigned); + ir_node *not_shiftval = new_rd_Not(dbgi, block_true, right, + low_unsigned); + ir_node *conv = create_conv(block_true, left_high, + low_unsigned); + ir_node *one = new_r_Const(irg, get_mode_one(low_unsigned)); + ir_node *carry0 = new_rd_Shl(dbgi, block_true, conv, one, + low_unsigned); + ir_node *carry1 = new_rd_Shl(dbgi, block_true, carry0, + not_shiftval, low_unsigned); + ir_node *res_low = new_rd_Or(dbgi, block_true, shift_low, carry1, + low_unsigned); + lower_in[0] = new_r_Jmp(block_true); + phi_low_in[0] = res_low; + phi_high_in[0] = res_high; + } - idx = get_irn_idx(node); - assert(idx < env->n_entries); - env->entries[idx]->low_word = new_r_Proj(irn, env->params->low_unsigned, 0); - env->entries[idx]->high_word = new_r_Proj(irn, mode, 1); -} /* lower_Binop */ + /* false block => shift_width > 1word */ + { + ir_node *in[1] = { proj_false }; + ir_node *block_false = new_r_Block(irg, ARRAY_SIZE(in), in); + ir_node *conv = create_conv(block_false, left_high, low_unsigned); + ir_node *res_low = new_rd_shrs(dbgi, block_false, conv, right, + low_unsigned); + int cnsti = modulo_shift2-1; + ir_node *cnst2 = new_r_Const_long(irg, low_unsigned, cnsti); + ir_node *res_high; + if (new_rd_shrs == new_rd_Shrs) { + res_high = new_rd_shrs(dbgi, block_false, left_high, cnst2, mode); + } else { + res_high = new_r_Const(irg, get_mode_null(mode)); + } + lower_in[1] = new_r_Jmp(block_false); + phi_low_in[1] = res_low; + phi_high_in[1] = res_high; + } -/** - * Translate a Shiftop. - * - * Create an intrinsic Call. - */ -static void lower_Shiftop(ir_node *node, ir_mode *mode, lower_env_t *env) -{ - ir_node *block, *irn; - ir_node *in[3]; - dbg_info *dbg; - ir_type *mtp; - int idx; - ir_graph *irg; - node_entry_t *entry; + /* patch lower block */ + set_irn_in(lower_block, ARRAY_SIZE(lower_in), lower_in); + phi_low = new_r_Phi(lower_block, ARRAY_SIZE(phi_low_in), phi_low_in, + low_unsigned); + phi_high = new_r_Phi(lower_block, ARRAY_SIZE(phi_high_in), phi_high_in, + mode); + ir_set_dw_lowered(node, phi_low, phi_high); +} - irn = get_binop_left(node); - entry = env->entries[get_irn_idx(irn)]; - assert(entry); +static void lower_Shr(ir_node *node, ir_mode *mode) +{ + lower_shr_helper(node, mode, new_rd_Shr); +} - if (! entry->low_word) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; - } /* if */ +static void lower_Shrs(ir_node *node, ir_mode *mode) +{ + lower_shr_helper(node, mode, new_rd_Shrs); +} - in[0] = entry->low_word; - in[1] = entry->high_word; +static void lower_Shl(ir_node *node, ir_mode *mode) +{ + ir_node *right = get_binop_right(node); + ir_node *left = get_binop_left(node); + ir_mode *shr_mode = get_irn_mode(node); + unsigned modulo_shift = get_mode_modulo_shift(shr_mode); + ir_mode *low_unsigned = env->low_unsigned; + unsigned modulo_shift2 = get_mode_modulo_shift(mode); + ir_graph *irg = get_irn_irg(node); + ir_node *left_low = get_lowered_low(left); + ir_node *left_high = get_lowered_high(left); + dbg_info *dbgi = get_irn_dbg_info(node); + ir_node *lower_block = get_nodes_block(node); + ir_node *block; + ir_node *cnst; + ir_node *andn; + ir_node *cmp; + ir_node *cond; + ir_node *proj_true; + ir_node *proj_false; + ir_node *phi_low; + ir_node *phi_high; + ir_node *lower_in[2]; + ir_node *phi_low_in[2]; + ir_node *phi_high_in[2]; + + /* this version is optimized for modulo shift architectures + * (and can't handle anything else) */ + if (modulo_shift != get_mode_size_bits(shr_mode) + || modulo_shift2<<1 != modulo_shift) { + panic("Shl lowering only implemented for modulo shift shl operations"); + } + if (!is_po2(modulo_shift) || !is_po2(modulo_shift2)) { + panic("Shl lowering only implemented for power-of-2 modes"); + } + /* without 2-complement the -x instead of (bit_width-x) trick won't work */ + if (get_mode_arithmetic(shr_mode) != irma_twos_complement) { + panic("Shl lowering only implemented for two-complement modes"); + } - /* The shift count is always mode_Iu in firm, so there is no need for lowering */ - in[2] = get_binop_right(node); - assert(get_irn_mode(in[2]) != env->params->high_signed - && get_irn_mode(in[2]) != env->params->high_unsigned); + /* if the right operand is a 64bit value, we're only interested in the + * lower word */ + if (get_irn_mode(right) == env->high_unsigned) { + right = get_lowered_low(right); + } else { + /* shift should never have signed mode on the right */ + assert(get_irn_mode(right) != env->high_signed); + right = create_conv(lower_block, right, low_unsigned); + } - dbg = get_irn_dbg_info(node); + part_block_dw(node); + env->flags |= CF_CHANGED; block = get_nodes_block(node); - irg = current_ir_graph; - mtp = mode_is_signed(mode) ? shiftop_tp_s : shiftop_tp_u; - irn = get_intrinsic_address(mtp, get_irn_op(node), mode, mode, env); - irn = new_rd_Call(dbg, block, get_irg_no_mem(current_ir_graph), - irn, 3, in, mtp); - set_irn_pinned(irn, get_irn_pinned(node)); - irn = new_r_Proj(irn, mode_T, pn_Call_T_result); + /* add a Cmp to test if highest bit is set <=> whether we shift more + * than half the word width */ + cnst = new_r_Const_long(irg, low_unsigned, modulo_shift2); + andn = new_r_And(block, right, cnst, low_unsigned); + cnst = new_r_Const(irg, get_mode_null(low_unsigned)); + cmp = new_rd_Cmp(dbgi, block, andn, cnst, ir_relation_equal); + cond = new_rd_Cond(dbgi, block, cmp); + proj_true = new_r_Proj(cond, mode_X, pn_Cond_true); + proj_false = new_r_Proj(cond, mode_X, pn_Cond_false); + + /* the true block => shift_width < 1word */ + { + ir_node *in[1] = { proj_true }; + ir_node *block_true = new_r_Block(irg, ARRAY_SIZE(in), in); + + ir_node *res_low = new_rd_Shl(dbgi, block_true, left_low, + right, low_unsigned); + ir_node *shift_high = new_rd_Shl(dbgi, block_true, left_high, right, + mode); + ir_node *not_shiftval = new_rd_Not(dbgi, block_true, right, + low_unsigned); + ir_node *conv = create_conv(block_true, left_low, mode); + ir_node *one = new_r_Const(irg, get_mode_one(low_unsigned)); + ir_node *carry0 = new_rd_Shr(dbgi, block_true, conv, one, mode); + ir_node *carry1 = new_rd_Shr(dbgi, block_true, carry0, + not_shiftval, mode); + ir_node *res_high = new_rd_Or(dbgi, block_true, shift_high, carry1, + mode); + lower_in[0] = new_r_Jmp(block_true); + phi_low_in[0] = res_low; + phi_high_in[0] = res_high; + } + + /* false block => shift_width > 1word */ + { + ir_node *in[1] = { proj_false }; + ir_node *block_false = new_r_Block(irg, ARRAY_SIZE(in), in); + ir_node *res_low = new_r_Const(irg, get_mode_null(low_unsigned)); + ir_node *conv = create_conv(block_false, left_low, mode); + ir_node *res_high = new_rd_Shl(dbgi, block_false, conv, right, mode); + lower_in[1] = new_r_Jmp(block_false); + phi_low_in[1] = res_low; + phi_high_in[1] = res_high; + } - idx = get_irn_idx(node); - assert(idx < env->n_entries); - env->entries[idx]->low_word = new_r_Proj(irn, env->params->low_unsigned, 0); - env->entries[idx]->high_word = new_r_Proj(irn, mode, 1); -} /* lower_Shiftop */ + /* patch lower block */ + set_irn_in(lower_block, ARRAY_SIZE(lower_in), lower_in); + phi_low = new_r_Phi(lower_block, ARRAY_SIZE(phi_low_in), phi_low_in, + low_unsigned); + phi_high = new_r_Phi(lower_block, ARRAY_SIZE(phi_high_in), phi_high_in, + mode); + ir_set_dw_lowered(node, phi_low, phi_high); +} /** - * Translate a Shr and handle special cases. + * Rebuild Rotl nodes into Or(Shl, Shr) and prepare all nodes. */ -static void lower_Shr(ir_node *node, ir_mode *mode, lower_env_t *env) -{ - ir_node *right = get_Shr_right(node); - ir_graph *irg = current_ir_graph; - - if (get_mode_arithmetic(mode) == irma_twos_complement && is_Const(right)) { - tarval *tv = get_Const_tarval(right); - - if (tarval_is_long(tv) && - get_tarval_long(tv) >= (long)get_mode_size_bits(mode)) { - ir_node *block = get_nodes_block(node); - ir_node *left = get_Shr_left(node); - ir_mode *low_unsigned = env->params->low_unsigned; - ir_node *c; - long shf_cnt = get_tarval_long(tv) - get_mode_size_bits(mode); - int idx = get_irn_idx(left); - - left = env->entries[idx]->high_word; - if (left == NULL) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; - } - - idx = get_irn_idx(node); - /* convert high word into low_unsigned mode if necessary */ - if (get_irn_mode(left) != low_unsigned) - left = new_r_Conv(block, left, low_unsigned); - - if (shf_cnt > 0) { - ir_mode *low_unsigned = env->params->low_unsigned; - c = new_r_Const_long(irg, low_unsigned, shf_cnt); - env->entries[idx]->low_word = new_r_Shr(block, left, c, - low_unsigned); - } else { - env->entries[idx]->low_word = left; - } /* if */ - env->entries[idx]->high_word = new_r_Const(irg, get_mode_null(mode)); +static void prepare_links_and_handle_rotl(ir_node *node, void *data) +{ + (void) data; + if (is_Rotl(node)) { + ir_mode *mode = get_irn_op_mode(node); + ir_node *right; + ir_node *left, *shl, *shr, *ornode, *block, *sub, *c; + ir_mode *omode, *rmode; + ir_graph *irg; + dbg_info *dbg; + optimization_state_t state; + if (mode != env->high_signed && mode != env->high_unsigned) { + prepare_links(node); return; - } /* if */ - } /* if */ - lower_Shiftop(node, mode, env); -} /* lower_Shr */ + } -/** - * Translate a Shl and handle special cases. - */ -static void lower_Shl(ir_node *node, ir_mode *mode, lower_env_t *env) -{ - ir_node *right = get_Shl_right(node); - ir_graph *irg = current_ir_graph; - - if (get_mode_arithmetic(mode) == irma_twos_complement && is_Const(right)) { - tarval *tv = get_Const_tarval(right); - - if (tarval_is_long(tv) && - get_tarval_long(tv) >= (long)get_mode_size_bits(mode)) { - ir_mode *mode_l; - ir_node *block = get_nodes_block(node); - ir_node *left = get_Shl_left(node); - ir_node *c; - long shf_cnt = get_tarval_long(tv) - get_mode_size_bits(mode); - int idx = get_irn_idx(left); - - left = env->entries[idx]->low_word; - if (left == NULL) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; - } - - left = new_r_Conv(block, left, mode); - idx = get_irn_idx(node); - - mode_l = env->params->low_unsigned; - if (shf_cnt > 0) { - c = new_r_Const_long(irg, mode_l, shf_cnt); - env->entries[idx]->high_word = new_r_Shl(block, left, c, mode); - } else { - env->entries[idx]->high_word = left; - } /* if */ - env->entries[idx]->low_word = new_r_Const(irg, get_mode_null(mode_l)); - - return; - } /* if */ - } /* if */ - lower_Shiftop(node, mode, env); -} /* lower_Shl */ - -/** - * Translate a Shrs and handle special cases. - */ -static void lower_Shrs(ir_node *node, ir_mode *mode, lower_env_t *env) -{ - ir_node *right = get_Shrs_right(node); - ir_graph *irg = current_ir_graph; - - if (get_mode_arithmetic(mode) == irma_twos_complement && is_Const(right)) { - tarval *tv = get_Const_tarval(right); - - if (tarval_is_long(tv) && - get_tarval_long(tv) >= (long)get_mode_size_bits(mode)) { - ir_node *block = get_nodes_block(node); - ir_node *left = get_Shrs_left(node); - ir_mode *low_unsigned = env->params->low_unsigned; - long shf_cnt = get_tarval_long(tv) - get_mode_size_bits(mode); - int idx = get_irn_idx(left); - ir_node *left_unsigned = left; - ir_node *low; - ir_node *c; - - left = env->entries[idx]->high_word; - if (left == NULL) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; - } - - idx = get_irn_idx(node); - /* convert high word into low_unsigned mode if necessary */ - if (get_irn_mode(left_unsigned) != low_unsigned) - left_unsigned = new_r_Conv(block, left, low_unsigned); - - if (shf_cnt > 0) { - c = new_r_Const_long(irg, low_unsigned, shf_cnt); - low = new_r_Shrs(block, left_unsigned, c, low_unsigned); - } else { - low = left_unsigned; - } /* if */ - /* low word is expected to have low_unsigned */ - env->entries[idx]->low_word = new_r_Conv(block, low, low_unsigned); - - c = new_r_Const_long(irg, low_unsigned, - get_mode_size_bits(mode) - 1); - env->entries[idx]->high_word = new_r_Shrs(block, left, c, mode); - - return; - } /* if */ - } /* if */ - lower_Shiftop(node, mode, env); -} /* lower_Shrs */ - -/** - * Rebuild Rotl nodes into Or(Shl, Shr) and prepare all nodes. - */ -static void prepare_links_and_handle_rotl(ir_node *node, void *env) -{ - lower_env_t *lenv = env; - - if (is_Rotl(node)) { - ir_mode *mode = get_irn_op_mode(node); - if (mode == lenv->params->high_signed || - mode == lenv->params->high_unsigned) { - ir_node *right = get_Rotl_right(node); - ir_node *left, *shl, *shr, *or, *block, *sub, *c; - ir_mode *omode, *rmode; - dbg_info *dbg; - optimization_state_t state; - - if (get_mode_arithmetic(mode) == irma_twos_complement && is_Const(right)) { - tarval *tv = get_Const_tarval(right); - - if (tarval_is_long(tv) && - get_tarval_long(tv) == (long)get_mode_size_bits(mode)) { - /* will be optimized in lower_Rotl() */ - return; - } - } - - /* replace the Rotl(x,y) by an Or(Shl(x,y), Shr(x,64-y)) and lower those */ - dbg = get_irn_dbg_info(node); - omode = get_irn_mode(node); - left = get_Rotl_left(node); - block = get_nodes_block(node); - shl = new_rd_Shl(dbg, block, left, right, omode); - rmode = get_irn_mode(right); - c = new_Const_long(rmode, get_mode_size_bits(omode)); - sub = new_rd_Sub(dbg, block, c, right, rmode); - shr = new_rd_Shr(dbg, block, left, sub, omode); - - /* optimization must be switched off here, or we will get the Rotl back */ - save_optimization_state(&state); - set_opt_algebraic_simplification(0); - or = new_rd_Or(dbg, block, shl, shr, omode); - restore_optimization_state(&state); - - exchange(node, or); - - /* do lowering on the new nodes */ - prepare_links(shl, env); - prepare_links(c, env); - prepare_links(sub, env); - prepare_links(shr, env); - prepare_links(or, env); - } - } else { - prepare_links(node, env); + /* replace the Rotl(x,y) by an Or(Shl(x,y), Shr(x,64-y)) */ + right = get_Rotl_right(node); + irg = get_irn_irg(node); + dbg = get_irn_dbg_info(node); + omode = get_irn_mode(node); + left = get_Rotl_left(node); + block = get_nodes_block(node); + shl = new_rd_Shl(dbg, block, left, right, omode); + rmode = get_irn_mode(right); + c = new_r_Const_long(irg, rmode, get_mode_size_bits(omode)); + sub = new_rd_Sub(dbg, block, c, right, rmode); + shr = new_rd_Shr(dbg, block, left, sub, omode); + + /* switch optimization off here, or we will get the Rotl back */ + save_optimization_state(&state); + set_opt_algebraic_simplification(0); + ornode = new_rd_Or(dbg, block, shl, shr, omode); + restore_optimization_state(&state); + + exchange(node, ornode); + + /* do lowering on the new nodes */ + prepare_links(shl); + prepare_links(c); + prepare_links(sub); + prepare_links(shr); + prepare_links(ornode); + return; } -} - -/** - * Translate a special case Rotl(x, sizeof(w)). - */ -static void lower_Rotl(ir_node *node, ir_mode *mode, lower_env_t *env) -{ - ir_node *right = get_Rotl_right(node); - ir_node *left = get_Rotl_left(node); - ir_node *h, *l; - int idx = get_irn_idx(left); - (void) right; - (void) mode; - - assert(get_mode_arithmetic(mode) == irma_twos_complement && - is_Const(right) && tarval_is_long(get_Const_tarval(right)) && - get_tarval_long(get_Const_tarval(right)) == (long)get_mode_size_bits(mode)); - l = env->entries[idx]->low_word; - h = env->entries[idx]->high_word; - idx = get_irn_idx(node); - - env->entries[idx]->low_word = h; - env->entries[idx]->high_word = l; -} /* lower_Rotl */ + prepare_links(node); +} /** * Translate an Unop. * * Create an intrinsic Call. */ -static void lower_Unop(ir_node *node, ir_mode *mode, lower_env_t *env) +static void lower_unop(ir_node *node, ir_mode *mode) { - ir_node *block, *irn; + ir_node *op = get_unop_op(node); + dbg_info *dbgi = get_irn_dbg_info(node); + ir_node *block = get_nodes_block(node); + ir_graph *irg = get_irn_irg(block); + ir_type *mtp = mode_is_signed(mode) ? unop_tp_s : unop_tp_u; + ir_op *irop = get_irn_op(node); + ir_node *addr = get_intrinsic_address(mtp, irop, mode, mode); + ir_node *nomem = get_irg_no_mem(irg); ir_node *in[2]; - dbg_info *dbg; - ir_type *mtp; - int idx; - node_entry_t *entry; - - irn = get_unop_op(node); - entry = env->entries[get_irn_idx(irn)]; - assert(entry); - - if (! entry->low_word) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; - } /* if */ - - in[0] = entry->low_word; - in[1] = entry->high_word; - - dbg = get_irn_dbg_info(node); - block = get_nodes_block(node); + ir_node *call; + ir_node *resproj; - mtp = mode_is_signed(mode) ? unop_tp_s : unop_tp_u; - irn = get_intrinsic_address(mtp, get_irn_op(node), mode, mode, env); - irn = new_rd_Call(dbg, block, get_irg_no_mem(current_ir_graph), - irn, 2, in, mtp); - set_irn_pinned(irn, get_irn_pinned(node)); - irn = new_r_Proj(irn, mode_T, pn_Call_T_result); + if (env->params->little_endian) { + in[0] = get_lowered_low(op); + in[1] = get_lowered_high(op); + } else { + in[0] = get_lowered_high(op); + in[1] = get_lowered_low(op); + } + call = new_rd_Call(dbgi, block, nomem, addr, 2, in, mtp); + resproj = new_r_Proj(call, mode_T, pn_Call_T_result); + set_irn_pinned(call, get_irn_pinned(node)); - idx = get_irn_idx(node); - assert(idx < env->n_entries); - env->entries[idx]->low_word = new_r_Proj(irn, env->params->low_unsigned, 0); - env->entries[idx]->high_word = new_r_Proj(irn, mode, 1); -} /* lower_Unop */ + if (env->params->little_endian) { + ir_node *res_low = new_r_Proj(resproj, env->low_unsigned, 0); + ir_node *res_high = new_r_Proj(resproj, mode, 1); + ir_set_dw_lowered(node, res_low, res_high); + } else { + ir_node *res_low = new_r_Proj(resproj, env->low_unsigned, 1); + ir_node *res_high = new_r_Proj(resproj, mode, 0); + ir_set_dw_lowered(node, res_low, res_high); + } +} /** - * Translate a logical Binop. + * Translate a logical binop. * - * Create two logical Binops. + * Create two logical binops. */ -static void lower_Binop_logical(ir_node *node, ir_mode *mode, lower_env_t *env, - ir_node *(*constr_rd)(dbg_info *db, ir_node *block, ir_node *op1, ir_node *op2, ir_mode *mode) ) { - ir_node *block, *irn; - ir_node *lop_l, *lop_h, *rop_l, *rop_h; - dbg_info *dbg; - int idx; - ir_graph *irg; - node_entry_t *entry; - - irn = get_binop_left(node); - entry = env->entries[get_irn_idx(irn)]; - assert(entry); - - if (! entry->low_word) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; - } /* if */ - - lop_l = entry->low_word; - lop_h = entry->high_word; - - irn = get_binop_right(node); - entry = env->entries[get_irn_idx(irn)]; - assert(entry); - - if (! entry->low_word) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; - } /* if */ - - rop_l = entry->low_word; - rop_h = entry->high_word; - - dbg = get_irn_dbg_info(node); - block = get_nodes_block(node); +static void lower_binop_logical(ir_node *node, ir_mode *mode, + ir_node *(*constr_rd)(dbg_info *db, ir_node *block, ir_node *op1, ir_node *op2, ir_mode *mode) ) +{ + ir_node *left = get_binop_left(node); + ir_node *right = get_binop_right(node); + const lower64_entry_t *left_entry = get_node_entry(left); + const lower64_entry_t *right_entry = get_node_entry(right); + dbg_info *dbgi = get_irn_dbg_info(node); + ir_node *block = get_nodes_block(node); + ir_node *res_low + = constr_rd(dbgi, block, left_entry->low_word, right_entry->low_word, + env->low_unsigned); + ir_node *res_high + = constr_rd(dbgi, block, left_entry->high_word, right_entry->high_word, + mode); + ir_set_dw_lowered(node, res_low, res_high); +} - idx = get_irn_idx(node); - assert(idx < env->n_entries); - irg = current_ir_graph; - env->entries[idx]->low_word = constr_rd(dbg, block, lop_l, rop_l, env->params->low_unsigned); - env->entries[idx]->high_word = constr_rd(dbg, block, lop_h, rop_h, mode); -} /* lower_Binop_logical */ +static void lower_And(ir_node *node, ir_mode *mode) +{ + lower_binop_logical(node, mode, new_rd_And); +} -/** create a logical operation transformation */ -#define lower_logical(op) \ -static void lower_##op(ir_node *node, ir_mode *mode, lower_env_t *env) { \ - lower_Binop_logical(node, mode, env, new_rd_##op); \ +static void lower_Or(ir_node *node, ir_mode *mode) +{ + lower_binop_logical(node, mode, new_rd_Or); } -lower_logical(And) -lower_logical(Or) -lower_logical(Eor) +static void lower_Eor(ir_node *node, ir_mode *mode) +{ + lower_binop_logical(node, mode, new_rd_Eor); +} /** * Translate a Not. * * Create two logical Nots. */ -static void lower_Not(ir_node *node, ir_mode *mode, lower_env_t *env) +static void lower_Not(ir_node *node, ir_mode *mode) { - ir_node *block, *irn; - ir_node *op_l, *op_h; - dbg_info *dbg; - int idx; - node_entry_t *entry; - - irn = get_Not_op(node); - entry = env->entries[get_irn_idx(irn)]; - assert(entry); + ir_node *op = get_Not_op(node); + const lower64_entry_t *op_entry = get_node_entry(op); + dbg_info *dbgi = get_irn_dbg_info(node); + ir_node *block = get_nodes_block(node); + ir_node *res_low + = new_rd_Not(dbgi, block, op_entry->low_word, env->low_unsigned); + ir_node *res_high + = new_rd_Not(dbgi, block, op_entry->high_word, mode); + ir_set_dw_lowered(node, res_low, res_high); +} - if (! entry->low_word) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); +static void lower_Proj(ir_node *node, ir_mode *op_mode) +{ + ir_mode *mode = get_irn_mode(node); + ir_node *pred; + (void)op_mode; + if (mode != env->high_signed && mode != env->high_unsigned) return; - } /* if */ + /* skip tuples */ + pred = get_Proj_pred(node); + if (is_Tuple(pred)) { + long pn = get_Proj_proj(node); + ir_node *op = get_irn_n(pred, pn); + const lower64_entry_t *entry = get_node_entry(op); + ir_set_dw_lowered(node, entry->low_word, entry->high_word); + } +} - op_l = entry->low_word; - op_h = entry->high_word; +static bool is_equality_cmp(const ir_node *node) +{ + ir_relation relation = get_Cmp_relation(node); + ir_node *left = get_Cmp_left(node); + ir_node *right = get_Cmp_right(node); + ir_mode *mode = get_irn_mode(left); - dbg = get_irn_dbg_info(node); - block = get_nodes_block(node); + /* this probably makes no sense if unordered is involved */ + assert(!mode_is_float(mode)); - idx = get_irn_idx(node); - assert(idx < env->n_entries); - env->entries[idx]->low_word = new_rd_Not(dbg, block, op_l, env->params->low_unsigned); - env->entries[idx]->high_word = new_rd_Not(dbg, block, op_h, mode); -} /* lower_Not */ + if (relation == ir_relation_equal || relation == ir_relation_less_greater) + return true; + + if (!is_Const(right) || !is_Const_null(right)) + return false; + if (mode_is_signed(mode)) { + return relation == ir_relation_less_greater; + } else { + return relation == ir_relation_greater; + } +} + +static ir_node *get_cfop_destination(const ir_node *cfop) +{ + const ir_edge_t *first = get_irn_out_edge_first(cfop); + /* we should only have 1 destination */ + assert(get_irn_n_edges(cfop) == 1); + return get_edge_src_irn(first); +} + +static void lower_Switch(ir_node *node, ir_mode *high_mode) +{ + ir_node *selector = get_Switch_selector(node); + ir_mode *mode = get_irn_mode(selector); + (void)high_mode; + if (mode == env->high_signed || mode == env->high_unsigned) { + /* we can't really handle Switch with 64bit offsets */ + panic("Switch with 64bit jumptable not supported"); + } + lower_node(selector); +} /** * Translate a Cond. */ -static void lower_Cond(ir_node *node, ir_mode *mode, lower_env_t *env) +static void lower_Cond(ir_node *node, ir_mode *high_mode) { - ir_node *cmp, *left, *right, *block; + ir_node *left, *right, *block; ir_node *sel = get_Cond_selector(node); - ir_mode *m = get_irn_mode(sel); - int idx; - (void) mode; - - if (m == mode_b) { - node_entry_t *lentry, *rentry; - ir_node *proj, *projT = NULL, *projF = NULL; - ir_node *new_bl, *cmpH, *cmpL, *irn; - ir_node *projHF, *projHT; - ir_node *dst_blk; - pn_Cmp pnc; - ir_graph *irg; - dbg_info *dbg; - - if (!is_Proj(sel)) - return; - - cmp = get_Proj_pred(sel); - if (!is_Cmp(cmp)) - return; - - left = get_Cmp_left(cmp); - idx = get_irn_idx(left); - lentry = env->entries[idx]; + ir_mode *cmp_mode; + const lower64_entry_t *lentry, *rentry; + ir_node *projT = NULL, *projF = NULL; + ir_node *new_bl, *irn; + ir_node *projHF, *projHT; + ir_node *dst_blk; + ir_relation relation; + ir_graph *irg; + dbg_info *dbg; - if (! lentry) { - /* a normal Cmp */ - return; - } /* if */ + (void) high_mode; - right = get_Cmp_right(cmp); - idx = get_irn_idx(right); - rentry = env->entries[idx]; - assert(rentry); + if (!is_Cmp(sel)) { + lower_node(sel); + return; + } - if (! lentry->low_word || !rentry->low_word) { - /* not yet ready */ - pdeq_putr(env->waitq, node); - return; - } /* if */ + left = get_Cmp_left(sel); + cmp_mode = get_irn_mode(left); + if (cmp_mode != env->high_signed && cmp_mode != env->high_unsigned) { + lower_node(sel); + return; + } - /* all right, build the code */ - for (proj = get_irn_link(node); proj; proj = get_irn_link(proj)) { - long proj_nr = get_Proj_proj(proj); + right = get_Cmp_right(sel); + lower_node(left); + lower_node(right); + lentry = get_node_entry(left); + rentry = get_node_entry(right); + + /* all right, build the code */ + foreach_out_edge_safe(node, edge) { + ir_node *proj = get_edge_src_irn(edge); + long proj_nr; + if (!is_Proj(proj)) + continue; + proj_nr = get_Proj_proj(proj); - if (proj_nr == pn_Cond_true) { - assert(projT == NULL && "more than one Proj(true)"); - projT = proj; - } else { - assert(proj_nr == pn_Cond_false); - assert(projF == NULL && "more than one Proj(false)"); - projF = proj; - } /* if */ - mark_irn_visited(proj); - } /* for */ - assert(projT && projF); - - /* create a new high compare */ - block = get_nodes_block(node); - irg = get_Block_irg(block); - dbg = get_irn_dbg_info(cmp); - pnc = get_Proj_proj(sel); - - if (is_Const(right) && is_Const_null(right)) { - if (pnc == pn_Cmp_Eq || pnc == pn_Cmp_Lg) { - /* x ==/!= 0 ==> or(low,high) ==/!= 0 */ - ir_mode *mode = env->params->low_unsigned; - ir_node *low = new_r_Conv(block, lentry->low_word, mode); - ir_node *high = new_r_Conv(block, lentry->high_word, mode); - ir_node *or = new_rd_Or(dbg, block, low, high, mode); - ir_node *cmp = new_rd_Cmp(dbg, block, or, new_Const_long(mode, 0)); - - ir_node *proj = new_r_Proj(cmp, mode_b, pnc); - set_Cond_selector(node, proj); - return; - } + if (proj_nr == pn_Cond_true) { + assert(projT == NULL && "more than one Proj(true)"); + projT = proj; + } else { + assert(proj_nr == pn_Cond_false); + assert(projF == NULL && "more than one Proj(false)"); + projF = proj; } + mark_irn_visited(proj); + } + assert(projT && projF); + + /* create a new high compare */ + block = get_nodes_block(node); + irg = get_Block_irg(block); + dbg = get_irn_dbg_info(sel); + relation = get_Cmp_relation(sel); + + if (is_equality_cmp(sel)) { + /* x ==/!= y ==> or(x_low^y_low,x_high^y_high) ==/!= 0 */ + ir_mode *mode = env->low_unsigned; + ir_node *low_left = new_rd_Conv(dbg, block, lentry->low_word, mode); + ir_node *high_left = new_rd_Conv(dbg, block, lentry->high_word, mode); + ir_node *low_right = new_rd_Conv(dbg, block, rentry->low_word, mode); + ir_node *high_right = new_rd_Conv(dbg, block, rentry->high_word, mode); + ir_node *xor_low = new_rd_Eor(dbg, block, low_left, low_right, mode); + ir_node *xor_high = new_rd_Eor(dbg, block, high_left, high_right, mode); + ir_node *ornode = new_rd_Or(dbg, block, xor_low, xor_high, mode); + ir_node *cmp = new_rd_Cmp(dbg, block, ornode, new_r_Const(irg, get_mode_null(mode)), relation); + set_Cond_selector(node, cmp); + return; + } - cmpH = new_rd_Cmp(dbg, block, lentry->high_word, rentry->high_word); - - if (pnc == pn_Cmp_Eq) { - /* simple case:a == b <==> a_h == b_h && a_l == b_l */ - pmap_entry *entry = pmap_find(env->proj_2_block, projF); - - assert(entry); - dst_blk = entry->value; - - irn = new_r_Proj(cmpH, mode_b, pn_Cmp_Eq); - dbg = get_irn_dbg_info(node); - irn = new_rd_Cond(dbg, block, irn); + if (relation == ir_relation_equal) { + ir_node *proj; + /* simple case:a == b <==> a_h == b_h && a_l == b_l */ + dst_blk = get_cfop_destination(projF); - projHF = new_r_Proj(irn, mode_X, pn_Cond_false); - mark_irn_visited(projHF); - exchange(projF, projHF); + irn = new_rd_Cmp(dbg, block, lentry->high_word, rentry->high_word, + ir_relation_equal); + dbg = get_irn_dbg_info(node); + irn = new_rd_Cond(dbg, block, irn); - projHT = new_r_Proj(irn, mode_X, pn_Cond_true); - mark_irn_visited(projHT); + projHF = new_r_Proj(irn, mode_X, pn_Cond_false); + mark_irn_visited(projHF); + exchange(projF, projHF); - new_bl = new_r_Block(irg, 1, &projHT); + projHT = new_r_Proj(irn, mode_X, pn_Cond_true); + mark_irn_visited(projHT); - dbg = get_irn_dbg_info(cmp); - cmpL = new_rd_Cmp(dbg, new_bl, lentry->low_word, rentry->low_word); - irn = new_r_Proj(cmpL, mode_b, pn_Cmp_Eq); - dbg = get_irn_dbg_info(node); - irn = new_rd_Cond(dbg, new_bl, irn); + new_bl = new_r_Block(irg, 1, &projHT); - proj = new_r_Proj(irn, mode_X, pn_Cond_false); - mark_irn_visited(proj); - add_block_cf_input(dst_blk, projHF, proj); + dbg = get_irn_dbg_info(sel); + irn = new_rd_Cmp(dbg, new_bl, lentry->low_word, rentry->low_word, + ir_relation_equal); + dbg = get_irn_dbg_info(node); + irn = new_rd_Cond(dbg, new_bl, irn); - proj = new_r_Proj(irn, mode_X, pn_Cond_true); - mark_irn_visited(proj); - exchange(projT, proj); - } else if (pnc == pn_Cmp_Lg) { - /* simple case:a != b <==> a_h != b_h || a_l != b_l */ - pmap_entry *entry = pmap_find(env->proj_2_block, projT); + proj = new_r_Proj(irn, mode_X, pn_Cond_false); + mark_irn_visited(proj); + add_block_cf_input(dst_blk, projHF, proj); - assert(entry); - dst_blk = entry->value; + proj = new_r_Proj(irn, mode_X, pn_Cond_true); + mark_irn_visited(proj); + exchange(projT, proj); + } else if (relation == ir_relation_less_greater) { + ir_node *proj; + /* simple case:a != b <==> a_h != b_h || a_l != b_l */ + dst_blk = get_cfop_destination(projT); - irn = new_r_Proj(cmpH, mode_b, pn_Cmp_Lg); - dbg = get_irn_dbg_info(node); - irn = new_rd_Cond(dbg, block, irn); + irn = new_rd_Cmp(dbg, block, lentry->high_word, rentry->high_word, + ir_relation_less_greater); + dbg = get_irn_dbg_info(node); + irn = new_rd_Cond(dbg, block, irn); - projHT = new_r_Proj(irn, mode_X, pn_Cond_true); - mark_irn_visited(projHT); - exchange(projT, projHT); + projHT = new_r_Proj(irn, mode_X, pn_Cond_true); + mark_irn_visited(projHT); + exchange(projT, projHT); - projHF = new_r_Proj(irn, mode_X, pn_Cond_false); - mark_irn_visited(projHF); + projHF = new_r_Proj(irn, mode_X, pn_Cond_false); + mark_irn_visited(projHF); - new_bl = new_r_Block(irg, 1, &projHF); + new_bl = new_r_Block(irg, 1, &projHF); - dbg = get_irn_dbg_info(cmp); - cmpL = new_rd_Cmp(dbg, new_bl, lentry->low_word, rentry->low_word); - irn = new_r_Proj(cmpL, mode_b, pn_Cmp_Lg); - dbg = get_irn_dbg_info(node); - irn = new_rd_Cond(dbg, new_bl, irn); + dbg = get_irn_dbg_info(sel); + irn = new_rd_Cmp(dbg, new_bl, lentry->low_word, rentry->low_word, + ir_relation_less_greater); + dbg = get_irn_dbg_info(node); + irn = new_rd_Cond(dbg, new_bl, irn); - proj = new_r_Proj(irn, mode_X, pn_Cond_true); - mark_irn_visited(proj); - add_block_cf_input(dst_blk, projHT, proj); + proj = new_r_Proj(irn, mode_X, pn_Cond_true); + mark_irn_visited(proj); + add_block_cf_input(dst_blk, projHT, proj); - proj = new_r_Proj(irn, mode_X, pn_Cond_false); - mark_irn_visited(proj); - exchange(projF, proj); - } else { - /* a rel b <==> a_h REL b_h || (a_h == b_h && a_l rel b_l) */ - ir_node *dstT, *dstF, *newbl_eq, *newbl_l; - pmap_entry *entry; + proj = new_r_Proj(irn, mode_X, pn_Cond_false); + mark_irn_visited(proj); + exchange(projF, proj); + } else { + ir_node *proj; + /* a rel b <==> a_h REL b_h || (a_h == b_h && a_l rel b_l) */ + ir_node *dstT, *dstF, *newbl_eq, *newbl_l; + ir_node *projEqF; - entry = pmap_find(env->proj_2_block, projT); - assert(entry); - dstT = entry->value; + dstT = get_cfop_destination(projT); + dstF = get_cfop_destination(projF); - entry = pmap_find(env->proj_2_block, projF); - assert(entry); - dstF = entry->value; + irn = new_rd_Cmp(dbg, block, lentry->high_word, rentry->high_word, + relation & ~ir_relation_equal); + dbg = get_irn_dbg_info(node); + irn = new_rd_Cond(dbg, block, irn); - irn = new_r_Proj(cmpH, mode_b, pnc & ~pn_Cmp_Eq); - dbg = get_irn_dbg_info(node); - irn = new_rd_Cond(dbg, block, irn); + projHT = new_r_Proj(irn, mode_X, pn_Cond_true); + mark_irn_visited(projHT); - projHT = new_r_Proj(irn, mode_X, pn_Cond_true); - mark_irn_visited(projHT); - exchange(projT, projHT); - projT = projHT; + projHF = new_r_Proj(irn, mode_X, pn_Cond_false); + mark_irn_visited(projHF); - projHF = new_r_Proj(irn, mode_X, pn_Cond_false); - mark_irn_visited(projHF); + newbl_eq = new_r_Block(irg, 1, &projHF); - newbl_eq = new_r_Block(irg, 1, &projHF); + irn = new_rd_Cmp(dbg, block, lentry->high_word, rentry->high_word, + ir_relation_equal); + irn = new_rd_Cond(dbg, newbl_eq, irn); - irn = new_r_Proj(cmpH, mode_b, pn_Cmp_Eq); - irn = new_rd_Cond(dbg, newbl_eq, irn); + projEqF = new_r_Proj(irn, mode_X, pn_Cond_false); + mark_irn_visited(projEqF); - proj = new_r_Proj(irn, mode_X, pn_Cond_false); - mark_irn_visited(proj); - exchange(projF, proj); - projF = proj; + proj = new_r_Proj(irn, mode_X, pn_Cond_true); + mark_irn_visited(proj); - proj = new_r_Proj(irn, mode_X, pn_Cond_true); - mark_irn_visited(proj); + newbl_l = new_r_Block(irg, 1, &proj); - newbl_l = new_r_Block(irg, 1, &proj); + dbg = get_irn_dbg_info(sel); + irn = new_rd_Cmp(dbg, newbl_l, lentry->low_word, rentry->low_word, + relation); + dbg = get_irn_dbg_info(node); + irn = new_rd_Cond(dbg, newbl_l, irn); - dbg = get_irn_dbg_info(cmp); - cmpL = new_rd_Cmp(dbg, newbl_l, lentry->low_word, rentry->low_word); - irn = new_r_Proj(cmpL, mode_b, pnc); - dbg = get_irn_dbg_info(node); - irn = new_rd_Cond(dbg, newbl_l, irn); + proj = new_r_Proj(irn, mode_X, pn_Cond_true); + mark_irn_visited(proj); + add_block_cf_input(dstT, projT, proj); - proj = new_r_Proj(irn, mode_X, pn_Cond_true); - mark_irn_visited(proj); - add_block_cf_input(dstT, projT, proj); + proj = new_r_Proj(irn, mode_X, pn_Cond_false); + mark_irn_visited(proj); + add_block_cf_input(dstF, projF, proj); - proj = new_r_Proj(irn, mode_X, pn_Cond_false); - mark_irn_visited(proj); - add_block_cf_input(dstF, projF, proj); - } /* if */ + exchange(projT, projHT); + exchange(projF, projEqF); + } - /* we have changed the control flow */ - env->flags |= CF_CHANGED; - } else { - idx = get_irn_idx(sel); - - if (env->entries[idx]) { - /* - Bad, a jump-table with double-word index. - This should not happen, but if it does we handle - it like a Conv were between (in other words, ignore - the high part. - */ - - if (! env->entries[idx]->low_word) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; - } /* if */ - set_Cond_selector(node, env->entries[idx]->low_word); - } /* if */ - } /* if */ -} /* lower_Cond */ + /* we have changed the control flow */ + env->flags |= CF_CHANGED; +} /** * Translate a Conv to higher_signed */ -static void lower_Conv_to_Ll(ir_node *node, lower_env_t *env) +static void lower_Conv_to_Ll(ir_node *node) { ir_mode *omode = get_irn_mode(node); ir_node *op = get_Conv_op(node); ir_mode *imode = get_irn_mode(op); - int idx = get_irn_idx(node); - ir_graph *irg = current_ir_graph; + ir_graph *irg = get_irn_irg(node); ir_node *block = get_nodes_block(node); dbg_info *dbg = get_irn_dbg_info(node); - node_entry_t *entry = env->entries[idx]; - ir_mode *low_unsigned = env->params->low_unsigned; - ir_mode *low_signed - = mode_is_signed(omode) ? env->params->low_signed : low_unsigned; + ir_node *res_low; + ir_node *res_high; - assert(idx < env->n_entries); + ir_mode *low_unsigned = env->low_unsigned; + ir_mode *low_signed + = mode_is_signed(omode) ? env->low_signed : low_unsigned; if (mode_is_int(imode) || mode_is_reference(imode)) { - if (imode == env->params->high_signed - || imode == env->params->high_unsigned) { + if (imode == env->high_signed || imode == env->high_unsigned) { /* a Conv from Lu to Ls or Ls to Lu */ - int op_idx = get_irn_idx(op); - node_entry_t *op_entry = env->entries[op_idx]; - - if (! op_entry->low_word) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; - } - entry->low_word = op_entry->low_word; - entry->high_word = new_rd_Conv(dbg, block, op_entry->high_word, - low_signed); + const lower64_entry_t *op_entry = get_node_entry(op); + res_low = op_entry->low_word; + res_high = new_rd_Conv(dbg, block, op_entry->high_word, low_signed); } else { /* simple case: create a high word */ if (imode != low_unsigned) op = new_rd_Conv(dbg, block, op, low_unsigned); - entry->low_word = op; + res_low = op; if (mode_is_signed(imode)) { int c = get_mode_size_bits(low_signed) - 1; - ir_node *cnst = new_Const_long(low_unsigned, c); + ir_node *cnst = new_r_Const_long(irg, low_unsigned, c); if (get_irn_mode(op) != low_signed) op = new_rd_Conv(dbg, block, op, low_signed); - entry->high_word = new_rd_Shrs(dbg, block, op, cnst, - low_signed); + res_high = new_rd_Shrs(dbg, block, op, cnst, low_signed); } else { - entry->high_word = new_Const(get_mode_null(low_signed)); + res_high = new_r_Const(irg, get_mode_null(low_signed)); } } } else if (imode == mode_b) { - entry->low_word = new_rd_Conv(dbg, block, op, low_unsigned); - entry->high_word = new_Const(get_mode_null(low_signed)); + res_low = new_rd_Conv(dbg, block, op, low_unsigned); + res_high = new_r_Const(irg, get_mode_null(low_signed)); } else { ir_node *irn, *call; - ir_type *mtp = get_conv_type(imode, omode, env); + ir_type *mtp = get_conv_type(imode, omode); - irn = get_intrinsic_address(mtp, get_irn_op(node), imode, omode, env); + irn = get_intrinsic_address(mtp, get_irn_op(node), imode, omode); call = new_rd_Call(dbg, block, get_irg_no_mem(irg), irn, 1, &op, mtp); set_irn_pinned(call, get_irn_pinned(node)); irn = new_r_Proj(call, mode_T, pn_Call_T_result); - entry->low_word = new_r_Proj(irn, low_unsigned, 0); - entry->high_word = new_r_Proj(irn, low_signed, 1); + if (env->params->little_endian) { + res_low = new_r_Proj(irn, low_unsigned, 0); + res_high = new_r_Proj(irn, low_signed, 1); + } else { + res_low = new_r_Proj(irn, low_unsigned, 1); + res_high = new_r_Proj(irn, low_signed, 0); + } } + ir_set_dw_lowered(node, res_low, res_high); } /** * Translate a Conv from higher_unsigned */ -static void lower_Conv_from_Ll(ir_node *node, lower_env_t *env) +static void lower_Conv_from_Ll(ir_node *node) { - ir_node *op = get_Conv_op(node); - ir_mode *omode = get_irn_mode(node); - ir_node *block = get_nodes_block(node); - dbg_info *dbg = get_irn_dbg_info(node); - int idx = get_irn_idx(op); - ir_graph *irg = current_ir_graph; - node_entry_t *entry = env->entries[idx]; - - assert(idx < env->n_entries); - - if (! entry->low_word) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; - } + ir_node *op = get_Conv_op(node); + ir_mode *omode = get_irn_mode(node); + ir_node *block = get_nodes_block(node); + dbg_info *dbg = get_irn_dbg_info(node); + ir_graph *irg = get_irn_irg(node); + const lower64_entry_t *entry = get_node_entry(op); if (mode_is_int(omode) || mode_is_reference(omode)) { op = entry->low_word; /* simple case: create a high word */ - if (omode != env->params->low_unsigned) + if (omode != env->low_unsigned) op = new_rd_Conv(dbg, block, op, omode); set_Conv_op(node, op); } else if (omode == mode_b) { /* llu ? true : false <=> (low|high) ? true : false */ - ir_mode *mode = env->params->low_unsigned; - ir_node *or = new_rd_Or(dbg, block, entry->low_word, entry->high_word, - mode); - set_Conv_op(node, or); + ir_mode *mode = env->low_unsigned; + ir_node *ornode = new_rd_Or(dbg, block, entry->low_word, + entry->high_word, mode); + set_Conv_op(node, ornode); } else { ir_node *irn, *call, *in[2]; ir_mode *imode = get_irn_mode(op); - ir_type *mtp = get_conv_type(imode, omode, env); + ir_type *mtp = get_conv_type(imode, omode); + ir_node *res; - irn = get_intrinsic_address(mtp, get_irn_op(node), imode, omode, env); - in[0] = entry->low_word; - in[1] = entry->high_word; + irn = get_intrinsic_address(mtp, get_irn_op(node), imode, omode); + if (env->params->little_endian) { + in[0] = entry->low_word; + in[1] = entry->high_word; + } else { + in[0] = entry->high_word; + in[1] = entry->low_word; + } call = new_rd_Call(dbg, block, get_irg_no_mem(irg), irn, 2, in, mtp); set_irn_pinned(call, get_irn_pinned(node)); irn = new_r_Proj(call, mode_T, pn_Call_T_result); + res = new_r_Proj(irn, omode, 0); + + exchange(node, res); + } +} + +/** + * lower Cmp + */ +static void lower_Cmp(ir_node *cmp, ir_mode *m) +{ + ir_node *l = get_Cmp_left(cmp); + ir_mode *cmp_mode = get_irn_mode(l); + ir_node *r, *low, *high, *t, *res; + ir_relation relation; + ir_node *block; + dbg_info *dbg; + const lower64_entry_t *lentry; + const lower64_entry_t *rentry; + (void) m; + + if (cmp_mode != env->high_signed && cmp_mode != env->high_unsigned) + return; - exchange(node, new_r_Proj(irn, omode, 0)); + r = get_Cmp_right(cmp); + lentry = get_node_entry(l); + rentry = get_node_entry(r); + relation = get_Cmp_relation(cmp); + block = get_nodes_block(cmp); + dbg = get_irn_dbg_info(cmp); + + /* easy case for x ==/!= 0 (see lower_Cond for details) */ + if (is_equality_cmp(cmp)) { + ir_graph *irg = get_irn_irg(cmp); + ir_mode *mode = env->low_unsigned; + ir_node *low_left = new_rd_Conv(dbg, block, lentry->low_word, mode); + ir_node *high_left = new_rd_Conv(dbg, block, lentry->high_word, mode); + ir_node *low_right = new_rd_Conv(dbg, block, rentry->low_word, mode); + ir_node *high_right = new_rd_Conv(dbg, block, rentry->high_word, mode); + ir_node *xor_low = new_rd_Eor(dbg, block, low_left, low_right, mode); + ir_node *xor_high = new_rd_Eor(dbg, block, high_left, high_right, mode); + ir_node *ornode = new_rd_Or(dbg, block, xor_low, xor_high, mode); + ir_node *new_cmp = new_rd_Cmp(dbg, block, ornode, new_r_Const(irg, get_mode_null(mode)), relation); + exchange(cmp, new_cmp); + return; + } + + if (relation == ir_relation_equal) { + /* simple case:a == b <==> a_h == b_h && a_l == b_l */ + low = new_rd_Cmp(dbg, block, lentry->low_word, rentry->low_word, + relation); + high = new_rd_Cmp(dbg, block, lentry->high_word, rentry->high_word, + relation); + res = new_rd_And(dbg, block, low, high, mode_b); + } else if (relation == ir_relation_less_greater) { + /* simple case:a != b <==> a_h != b_h || a_l != b_l */ + low = new_rd_Cmp(dbg, block, lentry->low_word, rentry->low_word, + relation); + high = new_rd_Cmp(dbg, block, lentry->high_word, rentry->high_word, + relation); + res = new_rd_Or(dbg, block, low, high, mode_b); + } else { + /* a rel b <==> a_h REL b_h || (a_h == b_h && a_l rel b_l) */ + ir_node *high1 = new_rd_Cmp(dbg, block, lentry->high_word, + rentry->high_word, relation & ~ir_relation_equal); + low = new_rd_Cmp(dbg, block, lentry->low_word, rentry->low_word, + relation); + high = new_rd_Cmp(dbg, block, lentry->high_word, rentry->high_word, + ir_relation_equal); + t = new_rd_And(dbg, block, low, high, mode_b); + res = new_rd_Or(dbg, block, high1, t, mode_b); } + exchange(cmp, res); } /** * Translate a Conv. */ -static void lower_Conv(ir_node *node, ir_mode *mode, lower_env_t *env) +static void lower_Conv(ir_node *node, ir_mode *mode) { mode = get_irn_mode(node); - if (mode == env->params->high_signed - || mode == env->params->high_unsigned) { - lower_Conv_to_Ll(node, env); + if (mode == env->high_signed || mode == env->high_unsigned) { + lower_Conv_to_Ll(node); } else { - ir_mode *mode = get_irn_mode(get_Conv_op(node)); + ir_mode *op_mode = get_irn_mode(get_Conv_op(node)); + + if (op_mode == env->high_signed || op_mode == env->high_unsigned) { + lower_Conv_from_Ll(node); + } + } +} + +static void fix_parameter_entities(ir_graph *irg, ir_type *orig_mtp) +{ + size_t orig_n_params = get_method_n_params(orig_mtp); + ir_entity **parameter_entities; + + parameter_entities = ALLOCANZ(ir_entity*, orig_n_params); + + ir_type *frame_type = get_irg_frame_type(irg); + size_t n = get_compound_n_members(frame_type); + size_t i; + size_t n_param; + + /* collect parameter entities */ + for (i = 0; i < n; ++i) { + ir_entity *entity = get_compound_member(frame_type, i); + size_t p; + if (!is_parameter_entity(entity)) + continue; + p = get_entity_parameter_number(entity); + if (p == IR_VA_START_PARAMETER_NUMBER) + continue; + assert(p < orig_n_params); + assert(parameter_entities[p] == NULL); + parameter_entities[p] = entity; + } + + /* adjust indices */ + n_param = 0; + for (i = 0; i < orig_n_params; ++i, ++n_param) { + ir_entity *entity = parameter_entities[i]; + ir_type *tp; - if (mode == env->params->high_signed - || mode == env->params->high_unsigned) { - lower_Conv_from_Ll(node, env); + if (entity != NULL) + set_entity_parameter_number(entity, n_param); + + tp = get_method_param_type(orig_mtp, i); + if (is_Primitive_type(tp)) { + ir_mode *mode = get_type_mode(tp); + if (mode == env->high_signed || mode == env->high_unsigned) { + ++n_param; + /* note that we do not change the type of the parameter + * entities, as calling convention fixup later still needs to + * know which is/was a lowered doubleword. + * So we just mark/remember it for later */ + if (entity != NULL) { + assert(entity->attr.parameter.doubleword_low_mode == NULL); + entity->attr.parameter.doubleword_low_mode + = env->low_unsigned; + } + } } } } @@ -1616,840 +1693,1388 @@ static void lower_Conv(ir_node *node, ir_mode *mode, lower_env_t *env) /** * Lower the method type. * + * @param env the lower environment * @param mtp the method type to lower - * @param ent the lower environment * * @return the lowered type */ -static ir_type *lower_mtp(ir_type *mtp, lower_env_t *env) +static ir_type *lower_mtp(ir_type *mtp) { - pmap_entry *entry; - ident *lid; - ir_type *res, *value_type; - - if (is_lowered_type(mtp)) + ir_type *res; + size_t i; + size_t orig_n_params; + size_t orig_n_res; + size_t n_param; + size_t n_res; + bool must_be_lowered; + + res = pmap_get(ir_type, lowered_type, mtp); + if (res != NULL) + return res; + if (type_visited(mtp)) return mtp; + mark_type_visited(mtp); - entry = pmap_find(lowered_type, mtp); - if (! entry) { - int i, n, r, n_param, n_res; + orig_n_params = get_method_n_params(mtp); + orig_n_res = get_method_n_ress(mtp); + n_param = orig_n_params; + n_res = orig_n_res; + must_be_lowered = false; - /* count new number of params */ - n_param = n = get_method_n_params(mtp); - for (i = n_param - 1; i >= 0; --i) { - ir_type *tp = get_method_param_type(mtp, i); + /* count new number of params */ + for (i = orig_n_params; i > 0;) { + ir_type *tp = get_method_param_type(mtp, --i); - if (is_Primitive_type(tp)) { - ir_mode *mode = get_type_mode(tp); + if (is_Primitive_type(tp)) { + ir_mode *mode = get_type_mode(tp); - if (mode == env->params->high_signed || - mode == env->params->high_unsigned) - ++n_param; - } /* if */ - } /* for */ + if (mode == env->high_signed || mode == env->high_unsigned) { + ++n_param; + must_be_lowered = true; + } + } + } - /* count new number of results */ - n_res = r = get_method_n_ress(mtp); - for (i = n_res - 1; i >= 0; --i) { - ir_type *tp = get_method_res_type(mtp, i); + /* count new number of results */ + for (i = orig_n_res; i > 0;) { + ir_type *tp = get_method_res_type(mtp, --i); - if (is_Primitive_type(tp)) { - ir_mode *mode = get_type_mode(tp); + if (is_Primitive_type(tp)) { + ir_mode *mode = get_type_mode(tp); - if (mode == env->params->high_signed || - mode == env->params->high_unsigned) - ++n_res; - } /* if */ - } /* for */ + if (mode == env->high_signed || mode == env->high_unsigned) { + ++n_res; + must_be_lowered = true; + } + } + } + if (!must_be_lowered) { + set_type_link(mtp, NULL); + return mtp; + } - res = new_type_method(n_param, n_res); + res = new_d_type_method(n_param, n_res, get_type_dbg_info(mtp)); - /* set param types and result types */ - for (i = n_param = 0; i < n; ++i) { - ir_type *tp = get_method_param_type(mtp, i); + /* set param types and result types */ + for (i = n_param = 0; i < orig_n_params; ++i) { + ir_type *tp = get_method_param_type(mtp, i); - if (is_Primitive_type(tp)) { - ir_mode *mode = get_type_mode(tp); + if (is_Primitive_type(tp)) { + ir_mode *mode = get_type_mode(tp); - if (mode == env->params->high_signed) { + if (mode == env->high_signed) { + if (env->params->little_endian) { set_method_param_type(res, n_param++, tp_u); set_method_param_type(res, n_param++, tp_s); - } else if (mode == env->params->high_unsigned) { - set_method_param_type(res, n_param++, tp_u); - set_method_param_type(res, n_param++, tp_u); } else { - set_method_param_type(res, n_param++, tp); - } /* if */ + set_method_param_type(res, n_param++, tp_s); + set_method_param_type(res, n_param++, tp_u); + } + } else if (mode == env->high_unsigned) { + set_method_param_type(res, n_param++, tp_u); + set_method_param_type(res, n_param++, tp_u); } else { - set_method_param_type(res, n_param++, tp); - } /* if */ - } /* for */ - for (i = n_res = 0; i < r; ++i) { - ir_type *tp = get_method_res_type(mtp, i); + set_method_param_type(res, n_param, tp); + ++n_param; + } + } else { + set_method_param_type(res, n_param, tp); + ++n_param; + } + } + for (i = n_res = 0; i < orig_n_res; ++i) { + ir_type *tp = get_method_res_type(mtp, i); - if (is_Primitive_type(tp)) { - ir_mode *mode = get_type_mode(tp); + if (is_Primitive_type(tp)) { + ir_mode *mode = get_type_mode(tp); - if (mode == env->params->high_signed) { + if (mode == env->high_signed) { + if (env->params->little_endian) { set_method_res_type(res, n_res++, tp_u); set_method_res_type(res, n_res++, tp_s); - } else if (mode == env->params->high_unsigned) { - set_method_res_type(res, n_res++, tp_u); - set_method_res_type(res, n_res++, tp_u); } else { - set_method_res_type(res, n_res++, tp); - } /* if */ + set_method_res_type(res, n_res++, tp_s); + set_method_res_type(res, n_res++, tp_u); + } + } else if (mode == env->high_unsigned) { + set_method_res_type(res, n_res++, tp_u); + set_method_res_type(res, n_res++, tp_u); } else { set_method_res_type(res, n_res++, tp); - } /* if */ - } /* for */ - set_lowered_type(mtp, res); - pmap_insert(lowered_type, mtp, res); - - value_type = get_method_value_param_type(mtp); - if (value_type != NULL) { - /* this creates a new value parameter type */ - (void)get_method_value_param_ent(res, 0); - - /* set new param positions */ - for (i = n_param = 0; i < n; ++i) { - ir_type *tp = get_method_param_type(mtp, i); - ident *id = get_method_param_ident(mtp, i); - ir_entity *ent = get_method_value_param_ent(mtp, i); - - set_entity_link(ent, INT_TO_PTR(n_param)); - if (is_Primitive_type(tp)) { - ir_mode *mode = get_type_mode(tp); - - if (mode == env->params->high_signed || mode == env->params->high_unsigned) { - if (id != NULL) { - lid = id_mangle(id, env->first_id); - set_method_param_ident(res, n_param, lid); - set_entity_ident(get_method_value_param_ent(res, n_param), lid); - lid = id_mangle(id, env->next_id); - set_method_param_ident(res, n_param + 1, lid); - set_entity_ident(get_method_value_param_ent(res, n_param + 1), lid); - } /* if */ - n_param += 2; - continue; - } /* if */ - } /* if */ - if (id != NULL) { - set_method_param_ident(res, n_param, id); - set_entity_ident(get_method_value_param_ent(res, n_param), id); - } /* if */ - ++n_param; - } /* for */ + } + } else { + set_method_res_type(res, n_res++, tp); + } + } - set_lowered_type(value_type, get_method_value_param_type(res)); - } /* if */ - } else { - res = entry->value; - } /* if */ + set_method_variadicity(res, get_method_variadicity(mtp)); + set_method_calling_convention(res, get_method_calling_convention(mtp)); + set_method_additional_properties(res, get_method_additional_properties(mtp)); + + set_higher_type(res, mtp); + set_type_link(res, mtp); + + mark_type_visited(res); + pmap_insert(lowered_type, mtp, res); return res; -} /* lower_mtp */ +} /** * Translate a Return. */ -static void lower_Return(ir_node *node, ir_mode *mode, lower_env_t *env) +static void lower_Return(ir_node *node, ir_mode *mode) { - ir_graph *irg = current_ir_graph; - ir_entity *ent = get_irg_entity(irg); - ir_type *mtp = get_entity_type(ent); - ir_node **in; - int i, j, n, idx; - int need_conv = 0; + ir_node **in; + size_t i, j, n; + int need_conv = 0; (void) mode; /* check if this return must be lowered */ for (i = 0, n = get_Return_n_ress(node); i < n; ++i) { - ir_node *pred = get_Return_res(node, i); - ir_mode *mode = get_irn_op_mode(pred); - - if (mode == env->params->high_signed || - mode == env->params->high_unsigned) { - idx = get_irn_idx(pred); - if (! env->entries[idx]->low_word) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; - } /* if */ + ir_node *pred = get_Return_res(node, i); + ir_mode *rmode = get_irn_op_mode(pred); + + if (rmode == env->high_signed || rmode == env->high_unsigned) need_conv = 1; - } /* if */ - } /* for */ + } if (! need_conv) return; - ent = get_irg_entity(irg); - mtp = get_entity_type(ent); - - mtp = lower_mtp(mtp, env); - set_entity_type(ent, mtp); + ir_graph *irg = get_irn_irg(node); + ir_entity *ent = get_irg_entity(irg); + ir_type *mtp = get_entity_type(ent); /* create a new in array */ NEW_ARR_A(ir_node *, in, get_method_n_ress(mtp) + 1); - in[0] = get_Return_mem(node); - - for (j = i = 0, n = get_Return_n_ress(node); i < n; ++i) { - ir_node *pred = get_Return_res(node, i); - - idx = get_irn_idx(pred); - assert(idx < env->n_entries); + j = 0; + in[j++] = get_Return_mem(node); - if (env->entries[idx]) { - in[++j] = env->entries[idx]->low_word; - in[++j] = env->entries[idx]->high_word; + for (i = 0, n = get_Return_n_ress(node); i < n; ++i) { + ir_node *pred = get_Return_res(node, i); + ir_mode *pred_mode = get_irn_mode(pred); + + if (pred_mode == env->high_signed || pred_mode == env->high_unsigned) { + const lower64_entry_t *entry = get_node_entry(pred); + if (env->params->little_endian) { + in[j++] = entry->low_word; + in[j++] = entry->high_word; + } else { + in[j++] = entry->high_word; + in[j++] = entry->low_word; + } } else { - in[++j] = pred; - } /* if */ - } /* for */ + in[j++] = pred; + } + } + assert(j == get_method_n_ress(mtp)+1); - set_irn_in(node, j+1, in); -} /* lower_Return */ + set_irn_in(node, j, in); +} /** * Translate the parameters. */ -static void lower_Start(ir_node *node, ir_mode *mode, lower_env_t *env) +static void lower_Start(ir_node *node, ir_mode *high_mode) { - ir_graph *irg = get_irn_irg(node); - ir_entity *ent = get_irg_entity(irg); - ir_type *tp = get_entity_type(ent); - ir_type *mtp; + ir_graph *irg = get_irn_irg(node); + ir_entity *ent = get_irg_entity(irg); + ir_type *mtp = get_entity_type(ent); + ir_type *orig_mtp = (ir_type*)get_type_link(mtp); + ir_node *args; long *new_projs; - int i, j, n_params, rem; - ir_node *proj, *args; - (void) mode; - - if (is_lowered_type(tp)) { - mtp = get_associated_type(tp); - } else { - mtp = tp; - } /* if */ - assert(! is_lowered_type(mtp)); + size_t i, j, n_params; + (void) high_mode; - n_params = get_method_n_params(mtp); - if (n_params <= 0) + /* if type link is NULL then the type was not lowered, hence no changes + * at Start necessary */ + if (orig_mtp == NULL) return; + n_params = get_method_n_params(orig_mtp); + NEW_ARR_A(long, new_projs, n_params); - /* first check if we have parameters that must be fixed */ + /* Calculate mapping of proj numbers in new_projs */ for (i = j = 0; i < n_params; ++i, ++j) { - ir_type *tp = get_method_param_type(mtp, i); + ir_type *ptp = get_method_param_type(orig_mtp, i); new_projs[i] = j; - if (is_Primitive_type(tp)) { - ir_mode *mode = get_type_mode(tp); - - if (mode == env->params->high_signed || - mode == env->params->high_unsigned) + if (is_Primitive_type(ptp)) { + ir_mode *amode = get_type_mode(ptp); + if (amode == env->high_signed || amode == env->high_unsigned) ++j; - } /* if */ - } /* for */ - if (i == j) + } + } + + /* find args Proj */ + args = NULL; + foreach_out_edge(node, edge) { + ir_node *proj = get_edge_src_irn(edge); + if (!is_Proj(proj)) + continue; + if (get_Proj_proj(proj) == pn_Start_T_args) { + args = proj; + break; + } + } + if (args == NULL) return; - mtp = lower_mtp(mtp, env); - set_entity_type(ent, mtp); - - /* switch off optimization for new Proj nodes or they might be CSE'ed - with not patched one's */ - rem = get_optimize(); - set_optimize(0); - - /* ok, fix all Proj's and create new ones */ - args = get_irg_args(irg); - for (proj = get_irn_link(node); proj; proj = get_irn_link(proj)) { - ir_node *pred = get_Proj_pred(proj); - long proj_nr; - int idx; - ir_mode *mode; + /* fix all Proj's and create new ones */ + foreach_out_edge_safe(args, edge) { + ir_node *proj = get_edge_src_irn(edge); + ir_mode *mode = get_irn_mode(proj); + ir_mode *mode_l = env->low_unsigned; + ir_node *pred; + long proj_nr; + ir_mode *mode_h; + ir_node *res_low; + ir_node *res_high; + int old_cse; dbg_info *dbg; - /* do not visit this node again */ - mark_irn_visited(proj); - - if (pred != args) + if (!is_Proj(proj)) continue; - + pred = get_Proj_pred(proj); proj_nr = get_Proj_proj(proj); - set_Proj_proj(proj, new_projs[proj_nr]); - idx = get_irn_idx(proj); - if (env->entries[idx]) { - ir_mode *low_mode = env->params->low_unsigned; - - mode = get_irn_mode(proj); + if (mode == env->high_signed) { + mode_h = env->low_signed; + } else if (mode == env->high_unsigned) { + mode_h = env->low_unsigned; + } else { + long new_pn = new_projs[proj_nr]; + set_Proj_proj(proj, new_pn); + continue; + } - if (mode == env->params->high_signed) { - mode = env->params->low_signed; - } else { - mode = env->params->low_unsigned; - } /* if */ - - dbg = get_irn_dbg_info(proj); - env->entries[idx]->low_word = - new_rd_Proj(dbg, args, low_mode, new_projs[proj_nr]); - env->entries[idx]->high_word = - new_rd_Proj(dbg, args, mode, new_projs[proj_nr] + 1); - } /* if */ - } /* for */ - set_optimize(rem); -} /* lower_Start */ + /* Switch off CSE or we might get an already existing Proj. */ + old_cse = get_opt_cse(); + set_opt_cse(0); + dbg = get_irn_dbg_info(proj); + if (env->params->little_endian) { + res_low = new_rd_Proj(dbg, pred, mode_l, new_projs[proj_nr]); + res_high = new_rd_Proj(dbg, pred, mode_h, new_projs[proj_nr] + 1); + } else { + res_high = new_rd_Proj(dbg, pred, mode_h, new_projs[proj_nr]); + res_low = new_rd_Proj(dbg, pred, mode_l, new_projs[proj_nr] + 1); + } + set_opt_cse(old_cse); + ir_set_dw_lowered(proj, res_low, res_high); + } +} /** * Translate a Call. */ -static void lower_Call(ir_node *node, ir_mode *mode, lower_env_t *env) +static void lower_Call(ir_node *node, ir_mode *mode) { ir_type *tp = get_Call_type(node); - ir_type *call_tp; - ir_node **in, *proj, *results; - int n_params, n_res, need_lower = 0; - int i, j; + ir_node **in; + size_t n_params, n_res; + bool need_lower = false; + size_t i, j; + size_t p; long *res_numbers = NULL; + ir_node *resproj; (void) mode; - if (is_lowered_type(tp)) { - call_tp = get_associated_type(tp); - } else { - call_tp = tp; - } /* if */ - - assert(! is_lowered_type(call_tp)); - - n_params = get_method_n_params(call_tp); - for (i = 0; i < n_params; ++i) { - ir_type *tp = get_method_param_type(call_tp, i); - - if (is_Primitive_type(tp)) { - ir_mode *mode = get_type_mode(tp); + n_params = get_method_n_params(tp); + for (p = 0; p < n_params; ++p) { + ir_type *ptp = get_method_param_type(tp, p); - if (mode == env->params->high_signed || - mode == env->params->high_unsigned) { - need_lower = 1; + if (is_Primitive_type(ptp)) { + ir_mode *pmode = get_type_mode(ptp); + if (pmode == env->high_signed || pmode == env->high_unsigned) { + need_lower = true; break; - } /* if */ - } /* if */ - } /* for */ - n_res = get_method_n_ress(call_tp); + } + } + } + n_res = get_method_n_ress(tp); if (n_res > 0) { NEW_ARR_A(long, res_numbers, n_res); for (i = j = 0; i < n_res; ++i, ++j) { - ir_type *tp = get_method_res_type(call_tp, i); + ir_type *ptp = get_method_res_type(tp, i); res_numbers[i] = j; - if (is_Primitive_type(tp)) { - ir_mode *mode = get_type_mode(tp); - - if (mode == env->params->high_signed || - mode == env->params->high_unsigned) { - need_lower = 1; + if (is_Primitive_type(ptp)) { + ir_mode *rmode = get_type_mode(ptp); + if (rmode == env->high_signed || rmode == env->high_unsigned) { + need_lower = true; ++j; - } /* if */ - } /* if */ - } /* for */ - } /* if */ + } + } + } + } if (! need_lower) return; /* let's lower it */ - call_tp = lower_mtp(call_tp, env); - set_Call_type(node, call_tp); + tp = lower_mtp(tp); + set_Call_type(node, tp); - NEW_ARR_A(ir_node *, in, get_method_n_params(call_tp) + 2); + NEW_ARR_A(ir_node *, in, get_method_n_params(tp) + 2); in[0] = get_Call_mem(node); in[1] = get_Call_ptr(node); for (j = 2, i = 0; i < n_params; ++i) { - ir_node *pred = get_Call_param(node, i); - int idx = get_irn_idx(pred); - - if (env->entries[idx]) { - if (! env->entries[idx]->low_word) { - /* not ready yet, wait */ - pdeq_putr(env->waitq, node); - return; + ir_node *pred = get_Call_param(node, i); + ir_mode *pred_mode = get_irn_mode(pred); + + if (pred_mode == env->high_signed || pred_mode == env->high_unsigned) { + const lower64_entry_t *pred_entry = get_node_entry(pred); + if (env->params->little_endian) { + in[j++] = pred_entry->low_word; + in[j++] = pred_entry->high_word; + } else { + in[j++] = pred_entry->high_word; + in[j++] = pred_entry->low_word; } - in[j++] = env->entries[idx]->low_word; - in[j++] = env->entries[idx]->high_word; } else { in[j++] = pred; - } /* if */ - } /* for */ + } + } set_irn_in(node, j, in); + /* find results T */ + resproj = NULL; + foreach_out_edge(node, edge) { + ir_node *proj = get_edge_src_irn(edge); + if (!is_Proj(proj)) + continue; + if (get_Proj_proj(proj) == pn_Call_T_result) { + resproj = proj; + break; + } + } + if (resproj == NULL) + return; + /* fix the results */ - results = NULL; - for (proj = get_irn_link(node); proj; proj = get_irn_link(proj)) { - long proj_nr = get_Proj_proj(proj); + foreach_out_edge_safe(resproj, edge) { + ir_node *proj = get_edge_src_irn(edge); + ir_mode *proj_mode = get_irn_mode(proj); + ir_mode *mode_l = env->low_unsigned; + ir_node *pred; + long proj_nr; + ir_mode *mode_h; + ir_node *res_low; + ir_node *res_high; + dbg_info *dbg; - if (proj_nr == pn_Call_T_result && get_Proj_pred(proj) == node) { - /* found the result proj */ - results = proj; - break; - } /* if */ - } /* for */ - - if (results) { /* there are results */ - int rem = get_optimize(); - - /* switch off optimization for new Proj nodes or they might be CSE'ed - with not patched one's */ - set_optimize(0); - for (i = j = 0, proj = get_irn_link(results); proj; proj = get_irn_link(proj), ++i, ++j) { - if (get_Proj_pred(proj) == results) { - long proj_nr = get_Proj_proj(proj); - int idx; - - /* found a result */ - set_Proj_proj(proj, res_numbers[proj_nr]); - idx = get_irn_idx(proj); - if (env->entries[idx]) { - ir_mode *mode = get_irn_mode(proj); - ir_mode *low_mode = env->params->low_unsigned; - dbg_info *dbg; - - if (mode == env->params->high_signed) { - mode = env->params->low_signed; - } else { - mode = env->params->low_unsigned; - } /* if */ - - dbg = get_irn_dbg_info(proj); - env->entries[idx]->low_word = - new_rd_Proj(dbg, results, low_mode, res_numbers[proj_nr]); - env->entries[idx]->high_word = - new_rd_Proj(dbg, results, mode, res_numbers[proj_nr] + 1); - } /* if */ - mark_irn_visited(proj); - } /* if */ - } /* for */ - set_optimize(rem); - } -} /* lower_Call */ + if (!is_Proj(proj)) + continue; + pred = get_Proj_pred(proj); + proj_nr = get_Proj_proj(proj); + + if (proj_mode == env->high_signed) { + mode_h = env->low_signed; + } else if (proj_mode == env->high_unsigned) { + mode_h = env->low_unsigned; + } else { + long new_nr = res_numbers[proj_nr]; + set_Proj_proj(proj, new_nr); + continue; + } + + dbg = get_irn_dbg_info(proj); + if (env->params->little_endian) { + res_low = new_rd_Proj(dbg, pred, mode_l, res_numbers[proj_nr]); + res_high = new_rd_Proj(dbg, pred, mode_h, res_numbers[proj_nr] + 1); + } else { + res_high = new_rd_Proj(dbg, pred, mode_h, res_numbers[proj_nr]); + res_low = new_rd_Proj(dbg, pred, mode_l, res_numbers[proj_nr] + 1); + } + ir_set_dw_lowered(proj, res_low, res_high); + } +} /** * Translate an Unknown into two. */ -static void lower_Unknown(ir_node *node, ir_mode *mode, lower_env_t *env) +static void lower_Unknown(ir_node *node, ir_mode *mode) { - int idx = get_irn_idx(node); - ir_graph *irg = get_irn_irg(node); - ir_mode *low_mode = env->params->low_unsigned; + ir_mode *low_mode = env->low_unsigned; + ir_graph *irg = get_irn_irg(node); + ir_node *res_low = new_r_Unknown(irg, low_mode); + ir_node *res_high = new_r_Unknown(irg, mode); + ir_set_dw_lowered(node, res_low, res_high); +} - env->entries[idx]->low_word = new_r_Unknown(irg, low_mode); - env->entries[idx]->high_word = new_r_Unknown(irg, mode); -} /* lower_Unknown */ +/** + * Translate a Bad into two. + */ +static void lower_Bad(ir_node *node, ir_mode *mode) +{ + ir_mode *low_mode = env->low_unsigned; + ir_graph *irg = get_irn_irg(node); + ir_node *res_low = new_r_Bad(irg, low_mode); + ir_node *res_high = new_r_Bad(irg, mode); + ir_set_dw_lowered(node, res_low, res_high); +} /** * Translate a Phi. * * First step: just create two templates */ -static void lower_Phi(ir_node *phi, ir_mode *mode, lower_env_t *env) +static void lower_Phi(ir_node *phi) { - ir_mode *mode_l = env->params->low_unsigned; - ir_graph *irg = get_irn_irg(phi); - ir_node *block, *unk_l, *unk_h, *phi_l, *phi_h; - ir_node **inl, **inh; + ir_mode *mode = get_irn_mode(phi); + int i; + int arity; + ir_node **in_l; + ir_node **in_h; + ir_node *unk_l; + ir_node *unk_h; + ir_node *phi_l; + ir_node *phi_h; dbg_info *dbg; - int idx, i, arity = get_Phi_n_preds(phi); - int enq = 0; - - idx = get_irn_idx(phi); - if (env->entries[idx]->low_word) { - /* Phi nodes already build, check for inputs */ - ir_node *phil = env->entries[idx]->low_word; - ir_node *phih = env->entries[idx]->high_word; + ir_node *block; + ir_graph *irg; + ir_mode *mode_l; + ir_mode *mode_h; - for (i = 0; i < arity; ++i) { - ir_node *pred = get_Phi_pred(phi, i); - int idx = get_irn_idx(pred); + /* enqueue predecessors */ + arity = get_Phi_n_preds(phi); + for (i = 0; i < arity; ++i) { + ir_node *pred = get_Phi_pred(phi, i); + pdeq_putr(env->waitq, pred); + } - if (env->entries[idx]->low_word) { - set_Phi_pred(phil, i, env->entries[idx]->low_word); - set_Phi_pred(phih, i, env->entries[idx]->high_word); - } else { - /* still not ready */ - pdeq_putr(env->waitq, phi); - return; - } /* if */ - } /* for */ - } /* if */ + if (mode != env->high_signed && mode != env->high_unsigned) + return; /* first create a new in array */ - NEW_ARR_A(ir_node *, inl, arity); - NEW_ARR_A(ir_node *, inh, arity); - unk_l = new_r_Unknown(irg, mode_l); - unk_h = new_r_Unknown(irg, mode); - + NEW_ARR_A(ir_node *, in_l, arity); + NEW_ARR_A(ir_node *, in_h, arity); + irg = get_irn_irg(phi); + mode_l = env->low_unsigned; + mode_h = mode == env->high_signed ? env->low_signed : env->low_unsigned; + unk_l = new_r_Dummy(irg, mode_l); + unk_h = new_r_Dummy(irg, mode_h); for (i = 0; i < arity; ++i) { - ir_node *pred = get_Phi_pred(phi, i); - int idx = get_irn_idx(pred); - - if (env->entries[idx]->low_word) { - inl[i] = env->entries[idx]->low_word; - inh[i] = env->entries[idx]->high_word; - } else { - inl[i] = unk_l; - inh[i] = unk_h; - enq = 1; - } /* if */ - } /* for */ + in_l[i] = unk_l; + in_h[i] = unk_h; + } dbg = get_irn_dbg_info(phi); block = get_nodes_block(phi); + phi_l = new_rd_Phi(dbg, block, arity, in_l, mode_l); + phi_h = new_rd_Phi(dbg, block, arity, in_h, mode_h); - idx = get_irn_idx(phi); - assert(idx < env->n_entries); - env->entries[idx]->low_word = phi_l = new_rd_Phi(dbg, block, arity, inl, mode_l); - env->entries[idx]->high_word = phi_h = new_rd_Phi(dbg, block, arity, inh, mode); - - /* Don't forget to link the new Phi nodes into the block. - * Beware that some Phis might be optimized away. */ - if (is_Phi(phi_l)) - add_Block_phi(block, phi_l); - if (is_Phi(phi_h)) - add_Block_phi(block, phi_h); - - if (enq) { - /* not yet finished */ - pdeq_putr(env->waitq, phi); - } /* if */ -} /* lower_Phi */ + ir_set_dw_lowered(phi, phi_l, phi_h); + + /* remember that we need to fixup the predecessors later */ + ARR_APP1(ir_node*, env->lowered_phis, phi); +} + +static void fixup_phi(ir_node *phi) +{ + const lower64_entry_t *entry = get_node_entry(phi); + ir_node *phi_l = entry->low_word; + ir_node *phi_h = entry->high_word; + int arity = get_Phi_n_preds(phi); + int i; + + /* exchange phi predecessors which are lowered by now */ + for (i = 0; i < arity; ++i) { + ir_node *pred = get_Phi_pred(phi, i); + const lower64_entry_t *pred_entry = get_node_entry(pred); + + set_Phi_pred(phi_l, i, pred_entry->low_word); + set_Phi_pred(phi_h, i, pred_entry->high_word); + } +} /** * Translate a Mux. */ -static void lower_Mux(ir_node *mux, ir_mode *mode, lower_env_t *env) +static void lower_Mux(ir_node *mux, ir_mode *mode) { - ir_node *block, *val; - ir_node *true_l, *true_h, *false_l, *false_h, *sel; - dbg_info *dbg; - int idx; - - val = get_Mux_true(mux); - idx = get_irn_idx(val); - if (env->entries[idx]->low_word) { - /* Values already build */ - true_l = env->entries[idx]->low_word; - true_h = env->entries[idx]->high_word; - } else { - /* still not ready */ - pdeq_putr(env->waitq, mux); - return; - } /* if */ - - val = get_Mux_false(mux); - idx = get_irn_idx(val); - if (env->entries[idx]->low_word) { - /* Values already build */ - false_l = env->entries[idx]->low_word; - false_h = env->entries[idx]->high_word; - } else { - /* still not ready */ - pdeq_putr(env->waitq, mux); + ir_node *truen = get_Mux_true(mux); + ir_node *falsen = get_Mux_false(mux); + ir_node *sel = get_Mux_sel(mux); + const lower64_entry_t *true_entry = get_node_entry(truen); + const lower64_entry_t *false_entry = get_node_entry(falsen); + ir_node *true_l = true_entry->low_word; + ir_node *true_h = true_entry->high_word; + ir_node *false_l = false_entry->low_word; + ir_node *false_h = false_entry->high_word; + dbg_info *dbgi = get_irn_dbg_info(mux); + ir_node *block = get_nodes_block(mux); + ir_node *res_low + = new_rd_Mux(dbgi, block, sel, false_l, true_l, env->low_unsigned); + ir_node *res_high + = new_rd_Mux(dbgi, block, sel, false_h, true_h, mode); + ir_set_dw_lowered(mux, res_low, res_high); +} + +/** + * Translate an ASM node. + */ +static void lower_ASM(ir_node *asmn, ir_mode *mode) +{ + ir_mode *high_signed = env->high_signed; + ir_mode *high_unsigned = env->high_unsigned; + int n_outs = get_ASM_n_output_constraints(asmn); + ir_asm_constraint *output_constraints = get_ASM_output_constraints(asmn); + ir_asm_constraint *input_constraints = get_ASM_input_constraints(asmn); + unsigned n_64bit_outs = 0; + + (void)mode; + + for (int i = get_irn_arity(asmn) - 1; i >= 0; --i) { + ir_node *op = get_irn_n(asmn, i); + ir_mode *op_mode = get_irn_mode(op); + if (op_mode == high_signed || op_mode == high_unsigned) { + panic("lowering ASM 64bit input unimplemented"); + } + } + + for (int o = 0; o < n_outs; ++o) { + const ir_asm_constraint *constraint = &output_constraints[o]; + if (constraint->mode == high_signed || constraint->mode == high_unsigned) { + const char *constr = get_id_str(constraint->constraint); + ++n_64bit_outs; + /* TODO: How to do this architecture neutral? This is very + * i386 specific... */ + if (constr[0] != '=' || constr[1] != 'A') { + panic("lowering ASM 64bit output only supports '=A' currently"); + } + } + } + + if (n_64bit_outs == 0) return; - } /* if */ + dbg_info *dbgi = get_irn_dbg_info(asmn); + ir_node *block = get_nodes_block(asmn); + ir_node *mem = get_ASM_mem(asmn); + int new_n_outs = 0; + int n_clobber = get_ASM_n_clobbers(asmn); + long *proj_map = ALLOCAN(long, n_outs); + ident **clobbers = get_ASM_clobbers(asmn); + ident *asm_text = get_ASM_text(asmn); + ir_asm_constraint *new_outputs + = ALLOCAN(ir_asm_constraint, n_outs+n_64bit_outs); + ir_node *new_asm; + + for (int o = 0; o < n_outs; ++o) { + const ir_asm_constraint *constraint = &output_constraints[o]; + if (constraint->mode == high_signed || constraint->mode == high_unsigned) { + new_outputs[new_n_outs].pos = constraint->pos; + new_outputs[new_n_outs].constraint = new_id_from_str("=a"); + new_outputs[new_n_outs].mode = env->low_unsigned; + proj_map[o] = new_n_outs; + ++new_n_outs; + new_outputs[new_n_outs].pos = constraint->pos; + new_outputs[new_n_outs].constraint = new_id_from_str("=d"); + if (constraint->mode == high_signed) + new_outputs[new_n_outs].mode = env->low_signed; + else + new_outputs[new_n_outs].mode = env->low_unsigned; + ++new_n_outs; + } else { + new_outputs[new_n_outs] = *constraint; + proj_map[o] = new_n_outs; + ++new_n_outs; + } + } + assert(new_n_outs == n_outs+(int)n_64bit_outs); - sel = get_Mux_sel(mux); + int n_inputs = get_ASM_n_inputs(asmn); + ir_node **new_ins = ALLOCAN(ir_node*, n_inputs); + for (int i = 0; i < n_inputs; ++i) + new_ins[i] = get_ASM_input(asmn, i); - dbg = get_irn_dbg_info(mux); - block = get_nodes_block(mux); + new_asm = new_rd_ASM(dbgi, block, mem, n_inputs, new_ins, input_constraints, + new_n_outs, new_outputs, n_clobber, clobbers, + asm_text); - idx = get_irn_idx(mux); - assert(idx < env->n_entries); - env->entries[idx]->low_word = new_rd_Mux(dbg, block, sel, false_l, true_l, env->params->low_unsigned); - env->entries[idx]->high_word = new_rd_Mux(dbg, block, sel, false_h, true_h, mode); -} /* lower_Mux */ + foreach_out_edge_safe(asmn, edge) { + ir_node *proj = get_edge_src_irn(edge); + ir_mode *proj_mode = get_irn_mode(proj); + long pn; + + if (!is_Proj(proj)) + continue; + pn = get_Proj_proj(proj); + + if (pn < n_outs) + pn = proj_map[pn]; + else + pn = new_n_outs + pn - n_outs; + + if (proj_mode == high_signed || proj_mode == high_unsigned) { + ir_mode *high_mode + = proj_mode == high_signed ? env->low_signed : env->low_unsigned; + ir_node *np_low = new_r_Proj(new_asm, env->low_unsigned, pn); + ir_node *np_high = new_r_Proj(new_asm, high_mode, pn+1); + ir_set_dw_lowered(proj, np_low, np_high); + } else { + ir_node *np = new_r_Proj(new_asm, proj_mode, pn); + exchange(proj, np); + } + } +} /** - * Translate an ASM node. + * Lower the builtin type to its higher part. + * + * @param mtp the builtin type to lower + * + * @return the lowered type */ -static void lower_ASM(ir_node *asmn, ir_mode *mode, lower_env_t *env) +static ir_type *lower_Builtin_type_high(ir_type *mtp) { - ir_mode *his = env->params->high_signed; - ir_mode *hiu = env->params->high_unsigned; - int i; - ir_node *n; + ir_type *res; + size_t i; + size_t n_params; + size_t n_results; + bool must_be_lowered; - (void)mode; + res = pmap_get(ir_type, lowered_builtin_type_high, mtp); + if (res != NULL) + return res; - for (i = get_irn_arity(asmn) - 1; i >= 0; --i) { - ir_mode *op_mode = get_irn_mode(get_irn_n(asmn, i)); - if (op_mode == his || op_mode == hiu) { - panic("lowering ASM unimplemented"); - } /* if */ - } /* for */ + n_params = get_method_n_params(mtp); + n_results = get_method_n_ress(mtp); + must_be_lowered = false; - for (n = asmn;;) { - ir_mode *proj_mode; + /* check for double word parameter */ + for (i = n_params; i > 0;) { + ir_type *tp = get_method_param_type(mtp, --i); - n = get_irn_link(n); - if (n == NULL) - break; + if (is_Primitive_type(tp)) { + ir_mode *mode = get_type_mode(tp); + + if (mode == env->high_signed || mode == env->high_unsigned) { + must_be_lowered = true; + break; + } + } + } + + if (!must_be_lowered) { + set_type_link(mtp, NULL); + return mtp; + } + + res = new_d_type_method(n_params, n_results, get_type_dbg_info(mtp)); + + /* set param types and result types */ + for (i = 0; i < n_params; ++i) { + ir_type *tp = get_method_param_type(mtp, i); + + if (is_Primitive_type(tp)) { + ir_mode *mode = get_type_mode(tp); + + if (mode == env->high_signed) { + if (env->params->little_endian) { + set_method_param_type(res, i, tp_u); + } else { + set_method_param_type(res, i, tp_s); + } + } else if (mode == env->high_unsigned) { + set_method_param_type(res, i, tp_u); + } else { + set_method_param_type(res, i, tp); + } + } else { + set_method_param_type(res, i, tp); + } + } + for (i = n_results = 0; i < n_results; ++i) { + ir_type *tp = get_method_res_type(mtp, i); + + if (is_Primitive_type(tp)) { + ir_mode *mode = get_type_mode(tp); + + if (mode == env->high_signed) { + if (env->params->little_endian) { + set_method_res_type(res, i, tp_u); + } else { + set_method_res_type(res, i, tp_s); + } + } else if (mode == env->high_unsigned) { + set_method_res_type(res, i, tp_u); + } else { + set_method_res_type(res, i, tp); + } + } else { + set_method_res_type(res, i, tp); + } + } + + set_method_variadicity(res, get_method_variadicity(mtp)); + set_method_calling_convention(res, get_method_calling_convention(mtp)); + set_method_additional_properties(res, get_method_additional_properties(mtp)); + + pmap_insert(lowered_builtin_type_high, mtp, res); + return res; +} + +/** + * Lower the builtin type to its lower part. + * + * @param mtp the builtin type to lower + * + * @return the lowered type + */ +static ir_type *lower_Builtin_type_low(ir_type *mtp) +{ + ir_type *res; + size_t i; + size_t n_params; + size_t n_results; + bool must_be_lowered; + + res = pmap_get(ir_type, lowered_builtin_type_low, mtp); + if (res != NULL) + return res; + + n_params = get_method_n_params(mtp); + n_results = get_method_n_ress(mtp); + must_be_lowered = false; + + /* check for double word parameter */ + for (i = n_params; i > 0;) { + ir_type *tp = get_method_param_type(mtp, --i); + + if (is_Primitive_type(tp)) { + ir_mode *mode = get_type_mode(tp); + + if (mode == env->high_signed || mode == env->high_unsigned) { + must_be_lowered = true; + break; + } + } + } + + if (!must_be_lowered) { + set_type_link(mtp, NULL); + return mtp; + } + + res = new_d_type_method(n_params, n_results, get_type_dbg_info(mtp)); + + /* set param types and result types */ + for (i = 0; i < n_params; ++i) { + ir_type *tp = get_method_param_type(mtp, i); + + if (is_Primitive_type(tp)) { + ir_mode *mode = get_type_mode(tp); + + if (mode == env->high_signed) { + if (env->params->little_endian) { + set_method_param_type(res, i, tp_s); + } else { + set_method_param_type(res, i, tp_u); + } + } else if (mode == env->high_unsigned) { + set_method_param_type(res, i, tp_u); + } else { + set_method_param_type(res, i, tp); + } + } else { + set_method_param_type(res, i, tp); + } + } + for (i = 0; i < n_results; ++i) { + ir_type *tp = get_method_res_type(mtp, i); + + if (is_Primitive_type(tp)) { + ir_mode *mode = get_type_mode(tp); + + if (mode == env->high_signed) { + if (env->params->little_endian) { + set_method_res_type(res, i, tp_s); + } else { + set_method_res_type(res, i, tp_u); + } + } else if (mode == env->high_unsigned) { + set_method_res_type(res, i, tp_u); + } else { + set_method_res_type(res, i, tp); + } + } else { + set_method_res_type(res, i, tp); + } + } + + set_method_variadicity(res, get_method_variadicity(mtp)); + set_method_calling_convention(res, get_method_calling_convention(mtp)); + set_method_additional_properties(res, get_method_additional_properties(mtp)); + + pmap_insert(lowered_builtin_type_low, mtp, res); + return res; +} + +/** + * lowers a builtin which reduces a 64bit value to a simple summary value + * (popcount, ffs, ...) + */ +static void lower_reduce_builtin(ir_node *builtin, ir_mode *mode) +{ + ir_builtin_kind kind = get_Builtin_kind(builtin); + ir_node *operand = get_Builtin_param(builtin, 0); + ir_mode *operand_mode = get_irn_mode(operand); + if (operand_mode != env->high_signed && operand_mode != env->high_unsigned) + return; - proj_mode = get_irn_mode(n); - if (proj_mode == his || proj_mode == hiu) { - panic("lowering ASM unimplemented"); - } /* if */ - } /* for */ -} /* lower_ASM */ + { + arch_allow_ifconv_func allow_ifconv = be_get_backend_param()->allow_ifconv; + int arity = get_irn_arity(builtin); + dbg_info *dbgi = get_irn_dbg_info(builtin); + ir_graph *irg = get_irn_irg(builtin); + ir_type *type = get_Builtin_type(builtin); + ir_type *lowered_type_high = lower_Builtin_type_high(type); + ir_type *lowered_type_low = lower_Builtin_type_low(type); + ir_type *result_type = get_method_res_type(lowered_type_low, 0); + ir_mode *result_mode = get_type_mode(result_type); + ir_node *block = get_nodes_block(builtin); + ir_node *mem = get_Builtin_mem(builtin); + const lower64_entry_t *entry = get_node_entry(operand); + ir_mode *high_mode = get_irn_mode(entry->high_word); + ir_node *in_high[1] = {entry->high_word}; + ir_node *in_low[1] = {entry->low_word}; + ir_node *res; + + assert(is_NoMem(mem)); + assert(arity == 2); + + switch (kind) { + case ir_bk_ffs: { + ir_node *number_of_bits = new_r_Const_long(irg, result_mode, get_mode_size_bits(env->low_unsigned)); + ir_node *zero_high = new_rd_Const(dbgi, irg, get_mode_null(high_mode)); + ir_node *zero_unsigned = new_rd_Const(dbgi, irg, get_mode_null(env->low_unsigned)); + ir_node *zero_result = new_rd_Const(dbgi, irg, get_mode_null(result_mode)); + ir_node *cmp_low = new_rd_Cmp(dbgi, block, entry->low_word, zero_unsigned, ir_relation_equal); + ir_node *cmp_high = new_rd_Cmp(dbgi, block, entry->high_word, zero_high, ir_relation_equal); + ir_node *ffs_high = new_rd_Builtin(dbgi, block, mem, 1, in_high, kind, lowered_type_high); + ir_node *high_proj = new_r_Proj(ffs_high, result_mode, pn_Builtin_max+1); + ir_node *high = new_rd_Add(dbgi, block, high_proj, number_of_bits, result_mode); + ir_node *ffs_low = new_rd_Builtin(dbgi, block, mem, 1, in_low, kind, lowered_type_low); + ir_node *low = new_r_Proj(ffs_low, result_mode, pn_Builtin_max+1); + ir_node *mux_high = new_rd_Mux(dbgi, block, cmp_high, high, zero_result, result_mode); + + if (! allow_ifconv(cmp_high, high, zero_result)) + ir_nodeset_insert(&created_mux_nodes, mux_high); + + res = new_rd_Mux(dbgi, block, cmp_low, low, mux_high, result_mode); + + if (! allow_ifconv(cmp_low, low, mux_high)) + ir_nodeset_insert(&created_mux_nodes, res); + break; + } + case ir_bk_clz: { + ir_node *zero = new_rd_Const(dbgi, irg, get_mode_null(high_mode)); + ir_node *cmp_high = new_rd_Cmp(dbgi, block, entry->high_word, zero, ir_relation_equal); + ir_node *clz_high = new_rd_Builtin(dbgi, block, mem, 1, in_high, kind, lowered_type_high); + ir_node *high = new_r_Proj(clz_high, result_mode, pn_Builtin_max+1); + ir_node *clz_low = new_rd_Builtin(dbgi, block, mem, 1, in_low, kind, lowered_type_low); + ir_node *low_proj = new_r_Proj(clz_low, result_mode, pn_Builtin_max+1); + ir_node *number_of_bits = new_r_Const_long(irg, result_mode, get_mode_size_bits(mode)); + ir_node *low = new_rd_Add(dbgi, block, low_proj, number_of_bits, result_mode); + + res = new_rd_Mux(dbgi, block, cmp_high, high, low, result_mode); + + if (! allow_ifconv(cmp_high, high, low)) + ir_nodeset_insert(&created_mux_nodes, res); + break; + } + case ir_bk_ctz: { + ir_node *zero_unsigned = new_rd_Const(dbgi, irg, get_mode_null(env->low_unsigned)); + ir_node *cmp_low = new_rd_Cmp(dbgi, block, entry->low_word, zero_unsigned, ir_relation_equal); + ir_node *ffs_high = new_rd_Builtin(dbgi, block, mem, 1, in_high, kind, lowered_type_high); + ir_node *high_proj = new_r_Proj(ffs_high, result_mode, pn_Builtin_max+1); + ir_node *number_of_bits = new_r_Const_long(irg, result_mode, get_mode_size_bits(env->low_unsigned)); + ir_node *high = new_rd_Add(dbgi, block, high_proj, number_of_bits, result_mode); + ir_node *ffs_low = new_rd_Builtin(dbgi, block, mem, 1, in_low, kind, lowered_type_low); + ir_node *low = new_r_Proj(ffs_low, result_mode, pn_Builtin_max+1); + + res = new_rd_Mux(dbgi, block, cmp_low, low, high, result_mode); + + if (! allow_ifconv(cmp_low, low, high)) + ir_nodeset_insert(&created_mux_nodes, res); + break; + } + case ir_bk_popcount: { + ir_node *popcount_high = new_rd_Builtin(dbgi, block, mem, 1, in_high, kind, lowered_type_high); + ir_node *popcount_low = new_rd_Builtin(dbgi, block, mem, 1, in_low, kind, lowered_type_low); + ir_node *high = new_r_Proj(popcount_high, result_mode, pn_Builtin_max+1); + ir_node *low = new_r_Proj(popcount_low, result_mode, pn_Builtin_max+1); + + res = new_rd_Add(dbgi, block, high, low, result_mode); + break; + } + case ir_bk_parity: { + ir_node *parity_high; + ir_node *parity_low; + ir_node *high; + ir_node *low; + + assert(arity == 2); + + parity_high = new_rd_Builtin(dbgi, block, mem, 1, in_high, kind, lowered_type_high); + high = new_r_Proj(parity_high, result_mode, pn_Builtin_max+1); + parity_low = new_rd_Builtin(dbgi, block, mem, 1, in_low, kind, lowered_type_low); + low = new_r_Proj(parity_low, result_mode, pn_Builtin_max+1); + res = new_rd_Eor(dbgi, block, high, low, result_mode); + break; + } + default: + panic("unexpected builtin"); + } + + ir_node *const in[] = { + [pn_Builtin_M] = mem, + [pn_Builtin_max + 1] = res, + }; + turn_into_tuple(builtin, ARRAY_SIZE(in), in); + } +} /** - * Translate a Sel node. + * lowers builtins performing arithmetic (bswap) */ -static void lower_Sel(ir_node *sel, ir_mode *mode, lower_env_t *env) +static void lower_arithmetic_builtin(ir_node *builtin, ir_mode *mode) { + ir_builtin_kind kind = get_Builtin_kind(builtin); + ir_node *operand = get_Builtin_param(builtin, 0); + ir_mode *operand_mode = get_irn_mode(operand); (void) mode; + if (operand_mode != env->high_signed && operand_mode != env->high_unsigned) + return; + + { + dbg_info *dbgi = get_irn_dbg_info(builtin); + ir_type *type = get_Builtin_type(builtin); + ir_type *lowered_type_high = lower_Builtin_type_high(type); + ir_type *lowered_type_low = lower_Builtin_type_low(type); + ir_node *block = get_nodes_block(builtin); + ir_node *mem = get_Builtin_mem(builtin); + const lower64_entry_t *entry = get_node_entry(operand); + ir_mode *mode_high = get_irn_mode(entry->high_word); + ir_node *res_high; + ir_node *res_low; + + switch (kind) { + case ir_bk_bswap: { + ir_node *in_high[1] = { entry->high_word }; + ir_node *in_low[1] = { entry->low_word }; + ir_node *swap_high = new_rd_Builtin(dbgi, block, mem, 1, in_high, kind, lowered_type_high); + ir_node *swap_low = new_rd_Builtin(dbgi, block, mem, 1, in_low, kind, lowered_type_low); + ir_node *high = new_r_Proj(swap_high, mode_high, pn_Builtin_max+1); + ir_node *low = new_r_Proj(swap_low, env->low_unsigned, pn_Builtin_max+1); + if (mode_high == env->low_signed) { + res_high = new_rd_Conv(dbgi, block, low, env->low_signed); + res_low = new_rd_Conv(dbgi, block, high, env->low_unsigned); + } else { + res_high = low; + res_low = high; + } + break; + } + default: + panic("unexpected builtin"); + } - /* we must only lower value parameter Sels if we change the - value parameter type. */ - if (env->value_param_tp != NULL) { - ir_entity *ent = get_Sel_entity(sel); - if (get_entity_owner(ent) == env->value_param_tp) { - int pos = PTR_TO_INT(get_entity_link(ent)); + /* search result Proj */ + foreach_out_edge_safe(builtin, edge) { + ir_node *proj = get_edge_src_irn(edge); + if (!is_Proj(proj)) + continue; - ent = get_method_value_param_ent(env->l_mtp, pos); - set_Sel_entity(sel, ent); - } /* if */ - } /* if */ -} /* lower_Sel */ + if (get_Proj_proj(proj) == pn_Builtin_max+1) { + ir_set_dw_lowered(proj, res_low, res_high); + } + } + } +} + +/** + * Lower double word builtins. + */ +static void lower_Builtin(ir_node *builtin, ir_mode *mode) +{ + ir_builtin_kind kind = get_Builtin_kind(builtin); + + switch (kind) { + case ir_bk_trap: + case ir_bk_debugbreak: + case ir_bk_return_address: + case ir_bk_frame_address: + case ir_bk_prefetch: + case ir_bk_inport: + case ir_bk_outport: + case ir_bk_inner_trampoline: + /* Nothing to do. */ + return; + case ir_bk_bswap: + lower_arithmetic_builtin(builtin, mode); + return; + case ir_bk_ffs: + case ir_bk_clz: + case ir_bk_ctz: + case ir_bk_popcount: + case ir_bk_parity: + lower_reduce_builtin(builtin, mode); + return; + } + panic("unknown builtin"); +} /** * check for opcodes that must always be lowered. */ -static int always_lower(ir_opcode code) +static bool always_lower(unsigned code) { switch (code) { case iro_ASM: + case iro_Builtin: case iro_Proj: case iro_Start: case iro_Call: case iro_Return: case iro_Cond: + case iro_Switch: case iro_Conv: case iro_Sel: - return 1; + return true; default: - return 0; - } /* switch */ -} /* always_lower */ - -/** - * lower boolean Proj(Cmp) - */ -static ir_node *lower_boolean_Proj_Cmp(ir_node *proj, ir_node *cmp, lower_env_t *env) -{ - int lidx, ridx; - ir_node *l, *r, *low, *high, *t, *res; - pn_Cmp pnc; - ir_node *blk; - dbg_info *db; - - l = get_Cmp_left(cmp); - lidx = get_irn_idx(l); - if (! env->entries[lidx]->low_word) { - /* still not ready */ - return NULL; - } /* if */ - - r = get_Cmp_right(cmp); - ridx = get_irn_idx(r); - if (! env->entries[ridx]->low_word) { - /* still not ready */ - return NULL; - } /* if */ - - pnc = get_Proj_proj(proj); - blk = get_nodes_block(cmp); - db = get_irn_dbg_info(cmp); - low = new_rd_Cmp(db, blk, env->entries[lidx]->low_word, env->entries[ridx]->low_word); - high = new_rd_Cmp(db, blk, env->entries[lidx]->high_word, env->entries[ridx]->high_word); - - if (pnc == pn_Cmp_Eq) { - /* simple case:a == b <==> a_h == b_h && a_l == b_l */ - res = new_rd_And(db, blk, - new_r_Proj(low, mode_b, pnc), - new_r_Proj(high, mode_b, pnc), - mode_b); - } else if (pnc == pn_Cmp_Lg) { - /* simple case:a != b <==> a_h != b_h || a_l != b_l */ - res = new_rd_Or(db, blk, - new_r_Proj(low, mode_b, pnc), - new_r_Proj(high, mode_b, pnc), - mode_b); - } else { - /* a rel b <==> a_h REL b_h || (a_h == b_h && a_l rel b_l) */ - t = new_rd_And(db, blk, - new_r_Proj(low, mode_b, pnc), - new_r_Proj(high, mode_b, pn_Cmp_Eq), - mode_b); - res = new_rd_Or(db, blk, - new_r_Proj(high, mode_b, pnc & ~pn_Cmp_Eq), - t, - mode_b); - } /* if */ - return res; -} /* lower_boolean_Proj_Cmp */ - -/** - * The type of a lower function. - * - * @param node the node to be lowered - * @param mode the low mode for the destination node - * @param env the lower environment - */ -typedef void (*lower_func)(ir_node *node, ir_mode *mode, lower_env_t *env); - -/** - * Lower a node. - */ -static void lower_ops(ir_node *node, void *env) -{ - lower_env_t *lenv = env; - node_entry_t *entry; - int idx = get_irn_idx(node); - ir_mode *mode = get_irn_mode(node); - - if (mode == mode_b || is_Mux(node) || is_Conv(node)) { - int i; - - for (i = get_irn_arity(node) - 1; i >= 0; --i) { - ir_node *proj = get_irn_n(node, i); - - if (is_Proj(proj)) { - ir_node *cmp = get_Proj_pred(proj); - - if (is_Cmp(cmp)) { - ir_node *arg = get_Cmp_left(cmp); - - mode = get_irn_mode(arg); - if (mode == lenv->params->high_signed || - mode == lenv->params->high_unsigned) { - ir_node *res = lower_boolean_Proj_Cmp(proj, cmp, lenv); - - if (res == NULL) { - /* could not lower because predecessors not ready */ - waitq_put(lenv->waitq, node); - return; - } /* if */ - set_irn_n(node, i, res); - } /* if */ - } /* if */ - } /* if */ - } /* for */ - } /* if */ - - entry = idx < lenv->n_entries ? lenv->entries[idx] : NULL; - if (entry || always_lower(get_irn_opcode(node))) { - ir_op *op = get_irn_op(node); - lower_func func = (lower_func)op->ops.generic; - - if (func) { - mode = get_irn_op_mode(node); - - if (mode == lenv->params->high_signed) - mode = lenv->params->low_signed; - else - mode = lenv->params->low_unsigned; - - DB((dbg, LEVEL_1, " %+F\n", node)); - func(node, mode, lenv); - } /* if */ - } /* if */ -} /* lower_ops */ - -#define IDENT(s) new_id_from_chars(s, sizeof(s)-1) + return false; + } +} /** * Compare two op_mode_entry_t's. */ static int cmp_op_mode(const void *elt, const void *key, size_t size) { - const op_mode_entry_t *e1 = elt; - const op_mode_entry_t *e2 = key; + const op_mode_entry_t *e1 = (const op_mode_entry_t*)elt; + const op_mode_entry_t *e2 = (const op_mode_entry_t*)key; (void) size; - return (e1->op - e2->op) | (e1->imode - e2->imode) | (e1->omode - e2->omode); -} /* cmp_op_mode */ + return (e1->op != e2->op) | (e1->imode != e2->imode) | (e1->omode != e2->omode); +} /** * Compare two conv_tp_entry_t's. */ static int cmp_conv_tp(const void *elt, const void *key, size_t size) { - const conv_tp_entry_t *e1 = elt; - const conv_tp_entry_t *e2 = key; + const conv_tp_entry_t *e1 = (const conv_tp_entry_t*)elt; + const conv_tp_entry_t *e2 = (const conv_tp_entry_t*)key; (void) size; - return (e1->imode - e2->imode) | (e1->omode - e2->omode); -} /* cmp_conv_tp */ + return (e1->imode != e2->imode) | (e1->omode != e2->omode); +} /** * Enter a lowering function into an ir_op. */ -static void enter_lower_func(ir_op *op, lower_func func) +void ir_register_dw_lower_function(ir_op *op, lower_dw_func func) { op->ops.generic = (op_func)func; -} /* enter_lower_func */ +} -/** - * Returns non-zero if a method type must be lowered. - * - * @param mtp the method type - */ -static int mtp_must_to_lowered(ir_type *mtp, lower_env_t *env) +/* Determine which modes need to be lowered */ +static void setup_modes(void) { - int i, n_params; - - n_params = get_method_n_params(mtp); - if (n_params <= 0) - return 0; + unsigned size_bits = env->params->doubleword_size; + ir_mode *doubleword_signed = NULL; + ir_mode *doubleword_unsigned = NULL; + size_t n_modes = ir_get_n_modes(); + ir_mode_arithmetic arithmetic; + unsigned modulo_shift; + size_t i; + + /* search for doubleword modes... */ + for (i = 0; i < n_modes; ++i) { + ir_mode *mode = ir_get_mode(i); + if (!mode_is_int(mode)) + continue; + if (get_mode_size_bits(mode) != size_bits) + continue; + if (mode_is_signed(mode)) { + if (doubleword_signed != NULL) { + /* sigh - the lowerer should really just lower all mode with + * size_bits it finds. Unfortunately this required a bigger + * rewrite. */ + panic("multiple double word signed modes found"); + } + doubleword_signed = mode; + } else { + if (doubleword_unsigned != NULL) { + /* sigh - the lowerer should really just lower all mode with + * size_bits it finds. Unfortunately this required a bigger + * rewrite. */ + panic("multiple double word unsigned modes found"); + } + doubleword_unsigned = mode; + } + } + if (doubleword_signed == NULL || doubleword_unsigned == NULL) { + panic("Couldn't find doubleword modes"); + } - /* first check if we have parameters that must be fixed */ - for (i = 0; i < n_params; ++i) { - ir_type *tp = get_method_param_type(mtp, i); + arithmetic = get_mode_arithmetic(doubleword_signed); + modulo_shift = get_mode_modulo_shift(doubleword_signed); + + assert(get_mode_size_bits(doubleword_unsigned) == size_bits); + assert(size_bits % 2 == 0); + assert(get_mode_sign(doubleword_signed) == 1); + assert(get_mode_sign(doubleword_unsigned) == 0); + assert(get_mode_sort(doubleword_signed) == irms_int_number); + assert(get_mode_sort(doubleword_unsigned) == irms_int_number); + assert(get_mode_arithmetic(doubleword_unsigned) == arithmetic); + assert(get_mode_modulo_shift(doubleword_unsigned) == modulo_shift); + + /* try to guess a sensible modulo shift for the new mode. + * (This is IMO another indication that this should really be a node + * attribute instead of a mode thing) */ + if (modulo_shift == size_bits) { + modulo_shift = modulo_shift / 2; + } else if (modulo_shift == 0) { + /* fine */ + } else { + panic("Don't know what new modulo shift to use for lowered doubleword mode"); + } + size_bits /= 2; + + /* produce lowered modes */ + env->high_signed = doubleword_signed; + env->high_unsigned = doubleword_unsigned; + env->low_signed = new_int_mode("WS", arithmetic, size_bits, 1, + modulo_shift); + env->low_unsigned = new_int_mode("WU", arithmetic, size_bits, 0, + modulo_shift); +} - if (is_Primitive_type(tp)) { - ir_mode *mode = get_type_mode(tp); +static void enqueue_preds(ir_node *node) +{ + int arity = get_irn_arity(node); + int i; - if (mode == env->params->high_signed || - mode == env->params->high_unsigned) - return 1; - } /* if */ - } /* for */ - return 0; -} /* mtp_must_to_lowered */ + for (i = 0; i < arity; ++i) { + ir_node *pred = get_irn_n(node, i); + pdeq_putr(env->waitq, pred); + } +} -/* - * Do the lowering. - */ -void lower_dw_ops(const lwrdw_param_t *param) +static void lower_node(ir_node *node) { - lower_env_t lenv; - int i; - ir_graph *rem; + int arity; + int i; + lower_dw_func func; + ir_op *op; + ir_mode *mode; + unsigned idx; + lower64_entry_t *entry; + + if (irn_visited_else_mark(node)) + return; - if (! param) + /* cycles are always broken at Phi and Block nodes. So we don't need special + * magic in all the other lower functions */ + if (is_Block(node)) { + enqueue_preds(node); return; + } else if (is_Phi(node)) { + lower_Phi(node); + return; + } + + /* depth-first: descend into operands */ + if (!is_Block(node)) { + ir_node *block = get_nodes_block(node); + lower_node(block); + } + + if (!is_Cond(node)) { + arity = get_irn_arity(node); + for (i = 0; i < arity; ++i) { + ir_node *pred = get_irn_n(node, i); + lower_node(pred); + } + } - if (! param->enable) + op = get_irn_op(node); + func = (lower_dw_func) op->ops.generic; + if (func == NULL) return; + idx = get_irn_idx(node); + entry = idx < env->n_entries ? env->entries[idx] : NULL; + if (entry != NULL || always_lower(get_irn_opcode(node))) { + mode = get_irn_op_mode(node); + if (mode == env->high_signed) { + mode = env->low_signed; + } else { + mode = env->low_unsigned; + } + DB((dbg, LEVEL_1, " %+F\n", node)); + func(node, mode); + } +} + +static void clear_node_and_phi_links(ir_node *node, void *data) +{ + (void) data; + if (get_irn_mode(node) == mode_T) { + set_irn_link(node, node); + } else { + set_irn_link(node, NULL); + } + if (is_Block(node)) + set_Block_phis(node, NULL); + else if (is_Phi(node)) + set_Phi_next(node, NULL); +} + +static void lower_irg(ir_graph *irg) +{ + ir_entity *ent; + ir_type *mtp; + ir_type *lowered_mtp; + unsigned n_idx; + + obstack_init(&env->obst); + + /* just here for debugging */ + current_ir_graph = irg; + assure_edges(irg); + + n_idx = get_irg_last_idx(irg); + n_idx = n_idx + (n_idx >> 2); /* add 25% */ + env->n_entries = n_idx; + env->entries = NEW_ARR_FZ(lower64_entry_t*, n_idx); + + env->irg = irg; + env->flags = 0; + + ent = get_irg_entity(irg); + mtp = get_entity_type(ent); + lowered_mtp = lower_mtp(mtp); + + if (lowered_mtp != mtp) { + set_entity_type(ent, lowered_mtp); + env->flags |= MUST_BE_LOWERED; + + fix_parameter_entities(irg, mtp); + } + + /* first step: link all nodes and allocate data */ + ir_reserve_resources(irg, IR_RESOURCE_PHI_LIST | IR_RESOURCE_IRN_LINK); + visit_all_identities(irg, clear_node_and_phi_links, NULL); + irg_walk_graph(irg, NULL, prepare_links_and_handle_rotl, env); + + if (env->flags & MUST_BE_LOWERED) { + size_t i; + ir_reserve_resources(irg, IR_RESOURCE_IRN_VISITED); + inc_irg_visited(irg); + + assert(pdeq_empty(env->waitq)); + pdeq_putr(env->waitq, get_irg_end(irg)); + + env->lowered_phis = NEW_ARR_F(ir_node*, 0); + while (!pdeq_empty(env->waitq)) { + ir_node *node = (ir_node*)pdeq_getl(env->waitq); + lower_node(node); + } + + /* we need to fixup phis */ + for (i = 0; i < ARR_LEN(env->lowered_phis); ++i) { + ir_node *phi = env->lowered_phis[i]; + fixup_phi(phi); + } + DEL_ARR_F(env->lowered_phis); + + + ir_free_resources(irg, IR_RESOURCE_IRN_VISITED); + + if (env->flags & CF_CHANGED) { + /* control flow changed, dominance info is invalid */ + clear_irg_properties(irg, IR_GRAPH_PROPERTY_CONSISTENT_DOMINANCE); + } + edges_deactivate(irg); + } + + ir_free_resources(irg, IR_RESOURCE_PHI_LIST | IR_RESOURCE_IRN_LINK); + + DEL_ARR_F(env->entries); + obstack_free(&env->obst, NULL); +} + +static const lwrdw_param_t *param; + +void ir_prepare_dw_lowering(const lwrdw_param_t *new_param) +{ + assert(new_param != NULL); FIRM_DBG_REGISTER(dbg, "firm.lower.dw"); - assert(2 * get_mode_size_bits(param->low_signed) == get_mode_size_bits(param->high_signed)); - assert(2 * get_mode_size_bits(param->low_unsigned) == get_mode_size_bits(param->high_unsigned)); - assert(get_mode_size_bits(param->low_signed) == get_mode_size_bits(param->low_unsigned)); + param = new_param; + + ir_clear_opcodes_generic_func(); + ir_register_dw_lower_function(op_ASM, lower_ASM); + ir_register_dw_lower_function(op_Add, lower_binop); + ir_register_dw_lower_function(op_And, lower_And); + ir_register_dw_lower_function(op_Bad, lower_Bad); + ir_register_dw_lower_function(op_Builtin, lower_Builtin); + ir_register_dw_lower_function(op_Call, lower_Call); + ir_register_dw_lower_function(op_Cmp, lower_Cmp); + ir_register_dw_lower_function(op_Cond, lower_Cond); + ir_register_dw_lower_function(op_Const, lower_Const); + ir_register_dw_lower_function(op_Conv, lower_Conv); + ir_register_dw_lower_function(op_Div, lower_Div); + ir_register_dw_lower_function(op_Eor, lower_Eor); + ir_register_dw_lower_function(op_Load, lower_Load); + ir_register_dw_lower_function(op_Minus, lower_unop); + ir_register_dw_lower_function(op_Mod, lower_Mod); + ir_register_dw_lower_function(op_Mul, lower_binop); + ir_register_dw_lower_function(op_Mux, lower_Mux); + ir_register_dw_lower_function(op_Not, lower_Not); + ir_register_dw_lower_function(op_Or, lower_Or); + ir_register_dw_lower_function(op_Proj, lower_Proj); + ir_register_dw_lower_function(op_Return, lower_Return); + ir_register_dw_lower_function(op_Shl, lower_Shl); + ir_register_dw_lower_function(op_Shr, lower_Shr); + ir_register_dw_lower_function(op_Shrs, lower_Shrs); + ir_register_dw_lower_function(op_Start, lower_Start); + ir_register_dw_lower_function(op_Store, lower_Store); + ir_register_dw_lower_function(op_Sub, lower_binop); + ir_register_dw_lower_function(op_Switch, lower_Switch); + ir_register_dw_lower_function(op_Unknown, lower_Unknown); +} + +/** + * Callback to lower only the Mux nodes we created. + */ +static int lower_mux_cb(ir_node *mux) +{ + return ir_nodeset_contains(&created_mux_nodes, mux); +} + +/* + * Do the lowering. + */ +void ir_lower_dw_ops(void) +{ + lower_dw_env_t lenv; + size_t i, n; + + memset(&lenv, 0, sizeof(lenv)); + lenv.params = param; + env = &lenv; + + setup_modes(); /* create the necessary maps */ if (! intrinsic_fkt) @@ -2458,12 +3083,16 @@ void lower_dw_ops(const lwrdw_param_t *param) conv_types = new_set(cmp_conv_tp, 16); if (! lowered_type) lowered_type = pmap_create(); + if (! lowered_builtin_type_low) + lowered_builtin_type_low = pmap_create(); + if (! lowered_builtin_type_high) + lowered_builtin_type_high = pmap_create(); /* create a primitive unsigned and signed type */ if (! tp_u) - tp_u = get_type_for_mode(param->low_unsigned); + tp_u = get_type_for_mode(lenv.low_unsigned); if (! tp_s) - tp_s = get_type_for_mode(param->low_signed); + tp_s = get_type_for_mode(lenv.low_signed); /* create method types for the created binop calls */ if (! binop_tp_u) { @@ -2474,198 +3103,72 @@ void lower_dw_ops(const lwrdw_param_t *param) set_method_param_type(binop_tp_u, 3, tp_u); set_method_res_type(binop_tp_u, 0, tp_u); set_method_res_type(binop_tp_u, 1, tp_u); - } /* if */ + } if (! binop_tp_s) { binop_tp_s = new_type_method(4, 2); - set_method_param_type(binop_tp_s, 0, tp_u); - set_method_param_type(binop_tp_s, 1, tp_s); - set_method_param_type(binop_tp_s, 2, tp_u); - set_method_param_type(binop_tp_s, 3, tp_s); - set_method_res_type(binop_tp_s, 0, tp_u); - set_method_res_type(binop_tp_s, 1, tp_s); - } /* if */ - if (! shiftop_tp_u) { - shiftop_tp_u = new_type_method(3, 2); - set_method_param_type(shiftop_tp_u, 0, tp_u); - set_method_param_type(shiftop_tp_u, 1, tp_u); - set_method_param_type(shiftop_tp_u, 2, tp_u); - set_method_res_type(shiftop_tp_u, 0, tp_u); - set_method_res_type(shiftop_tp_u, 1, tp_u); - } /* if */ - if (! shiftop_tp_s) { - shiftop_tp_s = new_type_method(3, 2); - set_method_param_type(shiftop_tp_s, 0, tp_u); - set_method_param_type(shiftop_tp_s, 1, tp_s); - set_method_param_type(shiftop_tp_s, 2, tp_u); - set_method_res_type(shiftop_tp_s, 0, tp_u); - set_method_res_type(shiftop_tp_s, 1, tp_s); - } /* if */ + if (env->params->little_endian) { + set_method_param_type(binop_tp_s, 0, tp_u); + set_method_param_type(binop_tp_s, 1, tp_s); + set_method_param_type(binop_tp_s, 2, tp_u); + set_method_param_type(binop_tp_s, 3, tp_s); + set_method_res_type(binop_tp_s, 0, tp_u); + set_method_res_type(binop_tp_s, 1, tp_s); + } else { + set_method_param_type(binop_tp_s, 0, tp_s); + set_method_param_type(binop_tp_s, 1, tp_u); + set_method_param_type(binop_tp_s, 2, tp_s); + set_method_param_type(binop_tp_s, 3, tp_u); + set_method_res_type(binop_tp_s, 0, tp_s); + set_method_res_type(binop_tp_s, 1, tp_u); + } + } if (! unop_tp_u) { unop_tp_u = new_type_method(2, 2); set_method_param_type(unop_tp_u, 0, tp_u); set_method_param_type(unop_tp_u, 1, tp_u); set_method_res_type(unop_tp_u, 0, tp_u); set_method_res_type(unop_tp_u, 1, tp_u); - } /* if */ + } if (! unop_tp_s) { unop_tp_s = new_type_method(2, 2); - set_method_param_type(unop_tp_s, 0, tp_u); - set_method_param_type(unop_tp_s, 1, tp_s); - set_method_res_type(unop_tp_s, 0, tp_u); - set_method_res_type(unop_tp_s, 1, tp_s); - } /* if */ - - lenv.tv_mode_bytes = new_tarval_from_long(get_mode_size_bytes(param->low_unsigned), param->low_unsigned); - lenv.tv_mode_bits = new_tarval_from_long(get_mode_size_bits(param->low_unsigned), param->low_unsigned); + if (env->params->little_endian) { + set_method_param_type(unop_tp_s, 0, tp_u); + set_method_param_type(unop_tp_s, 1, tp_s); + set_method_res_type(unop_tp_s, 0, tp_u); + set_method_res_type(unop_tp_s, 1, tp_s); + } else { + set_method_param_type(unop_tp_s, 0, tp_s); + set_method_param_type(unop_tp_s, 1, tp_u); + set_method_res_type(unop_tp_s, 0, tp_s); + set_method_res_type(unop_tp_s, 1, tp_u); + } + } + + lenv.tv_mode_bytes = new_tarval_from_long(param->doubleword_size/(2*8), lenv.low_unsigned); lenv.waitq = new_pdeq(); - lenv.params = param; lenv.first_id = new_id_from_chars(param->little_endian ? ".l" : ".h", 2); lenv.next_id = new_id_from_chars(param->little_endian ? ".h" : ".l", 2); - /* first clear the generic function pointer for all ops */ - clear_irp_opcodes_generic_func(); - -#define LOWER2(op, fkt) enter_lower_func(op_##op, fkt) -#define LOWER(op) LOWER2(op, lower_##op) -#define LOWER_BIN(op) LOWER2(op, lower_Binop) -#define LOWER_UN(op) LOWER2(op, lower_Unop) - - /* the table of all operations that must be lowered follows */ - LOWER(ASM); - LOWER(Load); - LOWER(Store); - LOWER(Const); - LOWER(And); - LOWER(Or); - LOWER(Eor); - LOWER(Not); - LOWER(Cond); - LOWER(Return); - LOWER(Call); - LOWER(Unknown); - LOWER(Phi); - LOWER(Mux); - LOWER(Start); - - LOWER_BIN(Add); - LOWER_BIN(Sub); - LOWER_BIN(Mul); - LOWER(Shl); - LOWER(Shr); - LOWER(Shrs); - LOWER(Rotl); - LOWER(DivMod); - LOWER(Div); - LOWER(Mod); - LOWER(Sel); - LOWER_UN(Abs); - LOWER_UN(Minus); - - LOWER(Conv); - -#undef LOWER_UN -#undef LOWER_BIN -#undef LOWER -#undef LOWER2 - + irp_reserve_resources(irp, IRP_RESOURCE_TYPE_LINK | IRP_RESOURCE_TYPE_VISITED); + inc_master_type_visited(); /* transform all graphs */ - rem = current_ir_graph; - for (i = get_irp_n_irgs() - 1; i >= 0; --i) { - ir_graph *irg = get_irp_irg(i); - ir_entity *ent; - ir_type *mtp; - int n_idx; - - obstack_init(&lenv.obst); - - n_idx = get_irg_last_idx(irg); - n_idx = n_idx + (n_idx >> 2); /* add 25% */ - lenv.n_entries = n_idx; - lenv.entries = NEW_ARR_F(node_entry_t *, n_idx); - memset(lenv.entries, 0, n_idx * sizeof(lenv.entries[0])); - - lenv.l_mtp = NULL; - lenv.flags = 0; - lenv.proj_2_block = pmap_create(); - lenv.value_param_tp = NULL; - ir_reserve_resources(irg, IR_RESOURCE_PHI_LIST | IR_RESOURCE_IRN_LINK); - - ent = get_irg_entity(irg); - mtp = get_entity_type(ent); - - if (mtp_must_to_lowered(mtp, &lenv)) { - ir_type *ltp = lower_mtp(mtp, &lenv); - lenv.flags |= MUST_BE_LOWERED; - set_entity_type(ent, ltp); - lenv.l_mtp = ltp; - lenv.value_param_tp = get_method_value_param_type(mtp); - } /* if */ - - /* first step: link all nodes and allocate data */ - irg_walk_graph(irg, firm_clear_node_and_phi_links, prepare_links_and_handle_rotl, &lenv); - - if (lenv.flags & MUST_BE_LOWERED) { - DB((dbg, LEVEL_1, "Lowering graph %+F\n", irg)); - - /* must do some work */ - irg_walk_graph(irg, NULL, lower_ops, &lenv); - - /* last step: all waiting nodes */ - DB((dbg, LEVEL_1, "finishing waiting nodes:\n")); - current_ir_graph = irg; - while (! pdeq_empty(lenv.waitq)) { - ir_node *node = pdeq_getl(lenv.waitq); - - lower_ops(node, &lenv); - } /* while */ - - ir_free_resources(irg, IR_RESOURCE_PHI_LIST | IR_RESOURCE_IRN_LINK); - - /* outs are invalid, we changed the graph */ - set_irg_outs_inconsistent(irg); - - if (lenv.flags & CF_CHANGED) { - /* control flow changed, dominance info is invalid */ - set_irg_doms_inconsistent(irg); - set_irg_extblk_inconsistent(irg); - set_irg_loopinfo_inconsistent(irg); - } /* if */ - } else { - ir_free_resources(irg, IR_RESOURCE_PHI_LIST | IR_RESOURCE_IRN_LINK); - } /* if */ - pmap_destroy(lenv.proj_2_block); - DEL_ARR_F(lenv.entries); - obstack_free(&lenv.obst, NULL); - } /* for */ - del_pdeq(lenv.waitq); - current_ir_graph = rem; -} /* lower_dw_ops */ + for (i = 0, n = get_irp_n_irgs(); i < n; ++i) { + ir_graph *irg = get_irp_irg(i); -struct pass_t { - ir_prog_pass_t pass; - const lwrdw_param_t *param; -}; + ir_nodeset_init(&created_mux_nodes); -/** - * Creates a wrapper around lower_dw_ops(). - */ -static int pass_wrapper(ir_prog *irp, void *context) -{ - struct pass_t *pass = context; + lower_irg(irg); - (void)irp; - lower_dw_ops(pass->param); - return 0; -} + if (ir_nodeset_size(&created_mux_nodes) > 0) + lower_mux(irg, lower_mux_cb); -ir_prog_pass_t *lower_dw_ops_pass(const char *name, const lwrdw_param_t *param) -{ - struct pass_t *pass = XMALLOCZ(struct pass_t); + ir_nodeset_destroy(&created_mux_nodes); + } + irp_free_resources(irp, IRP_RESOURCE_TYPE_LINK | IRP_RESOURCE_TYPE_VISITED); + del_pdeq(lenv.waitq); - pass->param = param; - return def_prog_pass_constructor( - &pass->pass, name ? name : "lower_dw", pass_wrapper); -} /* lower_dw_ops_pass */ + env = NULL; +} /* Default implementation. */ ir_entity *def_create_intrinsic_fkt(ir_type *method, const ir_op *op, @@ -2682,10 +3185,11 @@ ir_entity *def_create_intrinsic_fkt(ir_type *method, const ir_op *op, } else { snprintf(buf, sizeof(buf), "__l%s%s%s", get_op_name(op), get_mode_name(imode), get_mode_name(omode)); - } /* if */ + } id = new_id_from_str(buf); ent = new_entity(get_glob_type(), id, method); set_entity_ld_ident(ent, get_entity_ident(ent)); + set_entity_visibility(ent, ir_visibility_external); return ent; -} /* def_create_intrinsic_fkt */ +}