-/* Copyright (C) 1998 - 2000 by Universitaet Karlsruhe
-** All rights reserved.
-**
-** Authors: Martin Trapp, Christian Schaefer
-**
-** irgmod: ir graph modification
-*/
-
+/*
+ * Project: libFIRM
+ * File name: ir/ir/irgmod.h
+ * Purpose: Support for ir graph modification.
+ * Author: Martin Trapp, Christian Schaefer
+ * Modified by: Goetz Lindenmaier
+ * Created:
+ * CVS-ID: $Id$
+ * Copyright: (c) 1998-2003 Universität Karlsruhe
+ * Licence: This file protected by GPL - GNU GENERAL PUBLIC LICENSE.
+ */
+
+#ifdef HAVE_CONFIG_H
+# include <config.h>
+#endif
+
+# include "irvrfy.h"
+# include "irflag_t.h"
+# include "irgwalk.h"
+# include "irnode_t.h"
+# include "irgraph_t.h"
# include "irgmod.h"
-# include "iropt.h"
-
-/* ir_node * */
-/* arg_access (ir_mode *mode, long proj) */
-/* { */
-/* return new_r_Proj (current_ir_graph, current_ir_graph->start, */
-/* current_ir_graph->args, mode, proj); */
-/* } */
-
-
-/* these two functions are used in phi_merge, but defined in ircons.c */
-inline ir_node * get_r_value_internal (ir_node *block,
- int pos, ir_mode *mode);
-inline ir_node * new_r_Phi_in (ir_graph *irg, ir_node *block, ir_mode *mode,
- ir_node **in, int ins);
-
-/** This function computes the predecessors for a real Phi node, and then
- allocates and returns this node. The routine called to allocate the
- node might optimize it away and return a real value, or even a pointer
- to a deallocated Phi node on top of the obstack!
- This function is called with an in-array of proper size. **/
-static inline ir_node *
-phi_merge (ir_node *block, int pos, ir_mode *mode, ir_node **nin, int ins)
+# include "array.h"
+# include "ircons.h"
+# include "firmstat.h"
+
+/* Turns a node into a "useless" Tuple. The Tuple just forms a tuple
+ from several inputs.
+ This is useful if a node returning a tuple is removed, but the Projs
+ extracting values from the tuple are not available. */
+void
+turn_into_tuple (ir_node *node, int arity)
{
- ir_node *prevBlock, *res;
- int i;
-
- /* This loop goes to all predecessor blocks of the block the Phi node is in
- and there finds the operands of the Phi node by calling get_r_value_internal. */
- for (i = 1; i <= ins; ++i) {
- assert (block->in[i]);
- prevBlock = block->in[i]->in[0]; /* go past control flow op to prev block */
- assert (prevBlock);
- nin[i-1] = get_r_value_internal (prevBlock, pos, mode);
- }
-
- /* After collecting all predecessors into the array nin a new Phi node
- with these predecessors is created. This constructor contains an
- optimization: If all predecessors of the Phi node are identical it
- returns the only operand instead of a new Phi node. If the value
- passes two different control flow edges without being defined, and
- this is the second path treated, a pointer to the node that will be
- allocated for the first path (recurstion) is returned. We already
- know the address of this node, as it is the next node to be allocated
- and will be placed on top of the obstack. (The obstack is a _stack_!) */
- res = new_r_Phi_in (current_ir_graph, block, mode, nin, ins);
-
- /* Now we now the value "pos" and can enter it in the array with
- all known local variables. Attention: this might be a pointer to
- a node, that later will be allocated!!! See new_r_Phi_in.
- If this is called in mature, after some set_value in the same block,
- the proper value must not be overwritten:
- The call order
- get_value (makes Phi0, put's it into graph_arr)
- set_value (overwrites Phi0 in graph_arr)
- mature_block (upgrades Phi0, puts it again into graph_arr, overwriting
- the proper value.)
- fails. */
- if (!block->attr.block.graph_arr[pos]) {
- block->attr.block.graph_arr[pos] = res;
+ assert(node);
+ set_irn_op(node, op_Tuple);
+ if (get_irn_arity(node) == arity) {
+ /* keep old array */
} else {
- // printf(" value already computed by %s\n",
- // id_to_str(block->attr.block.graph_arr[pos]->op->name));
+ /* Allocate new array, don't free old in_array, it's on the obstack. */
+ ir_node *block = get_nodes_block(node);
+ node->in = NEW_ARR_D (ir_node *, current_ir_graph->obst, arity+1);
+ set_nodes_block(node, block);
}
-
- return res;
}
-
-/* this function is used in get_r_value_internal, but defined in ircons.c */
-inline ir_node * new_r_Phi0 (ir_graph *irg, ir_node *block, ir_mode *mode);
-
-
-/* This function returns the last definition of a variable. In case
- this variable was last defined in a previous block, Phi nodes are
- inserted. If the part of the firm graph containing the definition
- is not yet constructed, a dummy Phi node is returned. */
-inline ir_node *
-get_r_value_internal (ir_node *block, int pos, ir_mode *mode)
+/* Insert irnode `new' in place of irnode `old'
+ Since `new' may be bigger than `old' replace `old'
+ by an op_Id which is smaller than everything */
+INLINE void
+exchange (ir_node *old, ir_node *nw)
{
- ir_node *res;
- /* There are 4 cases to treat.
-
- 1. The block is not mature and we visit it the first time. We can not
- create a proper Phi node, therefore a Phi0, i.e., a Phi without
- predecessors is returned. This node is added to the linked list (field
- "link") of the containing block to be completed when this block is
- matured. (Comlpletion will add a new Phi and turn the Phi0 into a Id
- node.)
-
- 2. The value is already known in this block, graph_arr[pos] is set and we
- visit the block the first time. We can return the value without
- creating any new nodes.
-
- 3. The block is mature and we visit it the first time. A Phi node needs
- to be created (phi_merge). If the Phi is not needed, as all it's
- operands are the same value reaching the block through different
- paths, it's optimizes away and the value itself is returned.
-
- 4. The block is mature, and we visit it the second time. Now two
- subcases are possible:
- * The value was computed completely the last time we were here.
- This is the case if there is no loop. We can return the proper value.
- * The recursion that visited this node and set the flag did not
- return yet. We are computing a value in a loop and need to
- break the recursion without knowing the result yet.
- There is no simple check for the second subcase. Therefore we check
- for a second visit and treat all such cases as the second subcase.
- Anyways, the basic situation is the same: we reached a block
- on two paths without finding a definition of the value: No Phi
- nodes are needed on both paths.
- We return this information "Two paths, no Phi needed" by a very tricky
- implementation that relies on the fact that an obstack is a stack and
- will return a node with the same address on different allocations.
- Look also at phi_merge and get_r_phi_in to understand this.
- */
-
- /* case 4 -- already visited. */
- if (block->visit == ir_visited) return NULL;
-
- /* visited the first time */
- block->visit = ir_visited;
-
- /* Get the local valid value */
- res = block->attr.block.graph_arr[pos];
-
- /* case 2 -- If the value is actually computed, return it. */
- if (res) { return res;};
-
- if (block->attr.block.matured) { /* case 3 */
-
- /* The Phi has the same amount of ins as the corresponding block. */
- int ins = get_irn_arity(block); // ARR_LEN (block->in)-1;
- ir_node **nin;
- NEW_ARR_A (ir_node *, nin, ins);
+ ir_node *block;
+ ir_graph *irg = get_irn_irg (old);
- /* Phi merge collects the predecessors and then creates a node. */
- res = phi_merge (block, pos, mode, nin, ins);
+ assert(old != nw);
+ assert (irg);
+ assert(get_irn_op(old)->opar != oparity_dynamic);
- } else { /* case 1 */
- /* The block is not mature, we don't know how many in's are needed. A Phi
- with zero predecessors is created. Such a Phi node is called Phi0
- node. (There is also an obsolete Phi0 opcode.) The Phi0 is then added
- to the list of Phi0 nodes in this block to be matured by mature_block
- later.
- The Phi0 has to remember the pos of it's internal value. If the real Phi
- is computed, pos is used to update the array with the local values. */
+ stat_turn_into_id(old);
- res = new_r_Phi0 (current_ir_graph, block, mode);
- res->attr.phi0_pos = pos;
- res->link = block->link;
- block->link = res;
+ block = old->in[0];
+ if (!block) {
+ if (is_Block(nw)) block = nw;
+ else (block = nw->in[0]);
+ if (!block) { DDMN(old); DDMN(nw); assert(0 && "cannot find legal block for id"); }
}
- /* If we get here, the frontend missed a use-before-definition error */
- if (!res) {
- /* Error Message */
- printf("Error: no value set\n");
- assert (mode->code >= irm_f && mode->code <= irm_p);
- res = new_r_Const (current_ir_graph, block, mode,
- tarval_mode_null[mode->code]);
- }
+ old->op = op_Id;
+ old->in = NEW_ARR_D (ir_node *, irg->obst, 2);
+ old->in[0] = block;
+ old->in[1] = nw;
+}
- /* The local valid value is available now. */
- block->attr.block.graph_arr[pos] = res;
+/*--------------------------------------------------------------------*/
+/* Functionality for collect_phis */
+/*--------------------------------------------------------------------*/
- return res;
+static void
+clear_link (ir_node *n, void *env) {
+ set_irn_link(n, NULL);
}
-
-/* add an adge to a jmp node */
-void
-add_in_edge (ir_node *block, ir_node *jmp)
-{
- if (block->attr.block.matured) {
- printf("Error: Block already matured!\n");
+static void
+collect (ir_node *n, void *env) {
+ ir_node *pred;
+ if (get_irn_op(n) == op_Phi) {
+ set_irn_link(n, get_irn_link(get_nodes_block(n)));
+ set_irn_link(get_nodes_block(n), n);
}
- else {
- assert (jmp != NULL);
- ARR_APP1 (ir_node *, block->in, jmp);
+ if (get_irn_op(n) == op_Proj) {
+ pred = n;
+ while (get_irn_op(pred) == op_Proj)
+ pred = get_Proj_pred(pred);
+ set_irn_link(n, get_irn_link(pred));
+ set_irn_link(pred, n);
}
}
+void collect_phiprojs(ir_graph *irg) {
+ ir_graph *rem;
-/****************************/
-/* parameter administration */
-
-/* get a value from the parameter array from the current block by its index */
-ir_node *
-get_value (int pos, ir_mode *mode)
-{
- ++ir_visited;
- return get_r_value_internal (current_ir_graph->current_block, pos + 1, mode);
-}
+ /* Remember external state of current_ir_graph. */
+ rem = current_ir_graph;
+ current_ir_graph = irg;
+ irg_walk(get_irg_end(current_ir_graph), clear_link, collect, NULL);
-/* set a value at position pos in the parameter array from the current block */
-inline void
-set_value (int pos, ir_node *value)
-{
- current_ir_graph->current_block->attr.block.graph_arr[pos + 1] = value;
-}
-
-
-/* get the current store */
-inline ir_node *
-get_store (void)
-{
- /* GL: one could call get_value instead */
- ++ir_visited;
- return get_r_value_internal (current_ir_graph->current_block, 0, mode_M);
+ current_ir_graph = rem;
}
-/* set the current store */
-inline void
-set_store (ir_node *store)
-{
- /* GL: one could call set_value instead */
- current_ir_graph->current_block->attr.block.graph_arr[0] = store;
-}
-
-/** Finalize a Block node, when all control flows are known. */
-/** Acceptable parameters are only Block nodes. */
-void
-mature_block (ir_node *block)
-{
-
- int ins;
- ir_node *n, **nin;
- ir_node *next;
-
- assert (get_irn_opcode(block) == iro_Block);
-
- if (!get_Block_matured(block)) {
-
- /* turn the dynamic in-array into a static one. */
- ins = ARR_LEN (block->in)-1;
- NEW_ARR_A (ir_node *, nin, ins);
-
- /* GL, 7.2.2000. seems to be not complete. added this: but does'nt work.*
- memcpy (nin, block->in, sizeof (ir_node *) * ins);
- block->in = nin;
- */
-
- /* Traverse a chain of Phi nodes attached to this block and mature these, too. **/
- for (n = block->link; n; n=next) {
- ir_visited++;
- next = n->link;
- exchange (n, phi_merge (block, n->attr.phi0_pos, n->mode, nin, ins));
+/*--------------------------------------------------------------------*/
+/* Functionality for part_block */
+/*--------------------------------------------------------------------*/
+
+/**
+ * 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;
+ ir_node *proj, *pred;
+
+ /* move this node */
+ set_nodes_block(node, to_bl);
+
+ /* move its projs */
+ if (get_irn_mode(node) == mode_T) {
+ proj = get_irn_link(node);
+ while (proj) {
+ if (get_nodes_block(proj) == from_bl)
+ set_nodes_block(proj, to_bl);
+ proj = get_irn_link(proj);
}
-
- block->attr.block.matured = 1;
-
- block = optimize_in_place(block);
- ir_vrfy(block);
}
-}
-
-/* changing the current block */
-void
-switch_block (ir_node *target)
-{
- current_ir_graph->current_block = target;
-}
-
+ /* recursion ... */
+ if (get_irn_op(node) == op_Phi) return;
-void
-turn_into_tuple (ir_node *node, int arity)
-{
- assert(node);
- set_irn_op(node, op_Tuple);
- if (get_irn_arity(node) == arity) {
- /* keep old array */
- } else {
- /* allocate new array, remove old one. */
- /* !!!??? free old in_array */
- node->in = NEW_ARR_D (ir_node *, current_ir_graph->obst, arity+1);
+ arity = get_irn_arity(node);
+ for (i = 0; i < arity; i++) {
+ pred = get_irn_n(node, i);
+ if (get_nodes_block(pred) == from_bl)
+ move(pred, from_bl, to_bl);
}
}
+void part_block(ir_node *node) {
+ ir_node *new_block;
+ ir_node *old_block;
+ ir_node *phi;
+
+ /* Turn off optimizations so that blocks are not merged again. */
+ int rem_opt = get_opt_optimize();
+ set_optimize(0);
+
+ /* Transform the control flow */
+ old_block = get_nodes_block(node);
+ new_block = new_Block(get_Block_n_cfgpreds(old_block),
+ get_Block_cfgpred_arr(old_block));
+ set_irg_current_block(current_ir_graph, new_block);
+ {
+ ir_node *in[1];
+ in[0] = new_Jmp();
+ set_irn_in(old_block, 1, in);
+ irn_vrfy_irg(old_block, current_ir_graph);
+ }
+ /* move node and its predecessors to new_block */
+ move(node, old_block, new_block);
+
+ /* move Phi nodes to new_block */
+ phi = get_irn_link(old_block);
+ set_irn_link(new_block, phi);
+ set_irn_link(old_block, NULL);
+ while (phi) {
+ if(get_nodes_block(phi) == old_block); /* @@@ inlinening chokes on phis that don't
+ obey this condition. How do they get into
+ the list??? Example: InterfaceIII */
+ set_nodes_block(phi, new_block);
+ phi = get_irn_link(phi);
+ }
-/* Insert irnode `new' in place of irnode `old'
- Since `new' may be bigger than `old' replace `old'
- by an op_Id which is smaller than everything */
-inline void
-exchange (ir_node *old, ir_node *new)
-{
- ir_node *block = old->in[0];
-
- old->op = op_Id;
- old->in = NEW_ARR_D (ir_node *, current_ir_graph->obst, 2);
- old->in[0] = block;
- old->in[1] = new;
+ set_optimize(rem_opt);
}