/* src/vm/reorder.c - basic block reordering Copyright (C) 2006 R. Grafl, A. Krall, C. Kruegel, C. Oates, R. Obermaisser, M. Platter, M. Probst, S. Ring, E. Steiner, C. Thalinger, D. Thuernbeck, P. Tomsich, C. Ullrich, J. Wenninger, J. Wenninger, Institut f. Computersprachen - TU Wien This file is part of CACAO. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. Contact: cacao@cacaojvm.org Authors: Christian Thalinger Changes: */ #include "config.h" #include #include "vm/types.h" #include "mm/memory.h" #include "vm/jit/jit.h" /* reorder_place_next_unplaced_block ******************************************* Find next unplaced basic block in the array and place it. *******************************************************************************/ static basicblock *reorder_place_next_unplaced_block(jitdata *jd, u1 *blocks, basicblock *bptr) { basicblock *tbptr; s4 i; for (i = 0; i < jd->basicblockcount; i++) { if (blocks[i] == false) { tbptr = &jd->basicblocks[i]; /* place the block */ blocks[tbptr->nr] = true; /* change the basic block order */ bptr->next = tbptr; return tbptr; } } return NULL; } /* reorder ********************************************************************* Reorders basic blocks in respect to their execution frequency. *******************************************************************************/ bool reorder(jitdata *jd) { methodinfo *m; codeinfo *pcode; basicblock *bptr; basicblock *tbptr; /* taken basic block pointer */ basicblock *ntbptr; /* not-taken basic block pointer */ basicblock *pbptr; /* predecessor basic block pointer */ u4 freq; u4 tfreq; u4 ntfreq; u4 pfreq; instruction *iptr; u1 *blocks; /* flag array */ s4 placed; /* number of placed basic blocks */ s4 i; /* get required compiler data */ m = jd->m; #if !defined(NDEBUG) method_println(m); #endif /* get the codeinfo of the previous method */ pcode = m->code; /* XXX debug */ if (jd->basicblockcount > 8) return true; /* allocate flag array for blocks which are placed */ blocks = DMNEW(u1, jd->basicblockcount); MZERO(blocks, u1, jd->basicblockcount); /* Get the entry block and iterate over all basic blocks until we have placed all blocks. */ bptr = jd->basicblocks; placed = 0; /* the first block is always placed as the first one */ blocks[0] = true; placed++; while (placed <= jd->basicblockcount + 1) { /* get last instruction of basic block */ iptr = bptr->iinstr + bptr->icount - 1; printf("L%03d, ", bptr->nr); switch (bptr->type) { case BBTYPE_STD: printf("STD"); break; case BBTYPE_EXH: printf("EXH"); break; case BBTYPE_SBR: printf("SBR"); break; } printf(", predecessors: %d, successors: %d, frequency: %d -> ", bptr->predecessorcount, bptr->successorcount, pcode->bbfrequency[bptr->nr]); switch (iptr->opc) { case ICMD_RETURN: case ICMD_IRETURN: case ICMD_LRETURN: case ICMD_FRETURN: case ICMD_DRETURN: case ICMD_ARETURN: case ICMD_ATHROW: puts("return"); tbptr = reorder_place_next_unplaced_block(jd, blocks, bptr); placed++; /* all blocks placed? return. */ if (tbptr == NULL) continue; /* set last placed block */ bptr = tbptr; break; case ICMD_IFEQ: case ICMD_IFNE: case ICMD_IFLT: case ICMD_IFGE: case ICMD_IFGT: case ICMD_IFLE: case ICMD_IF_ICMPEQ: case ICMD_IF_ICMPNE: case ICMD_IF_ICMPLT: case ICMD_IF_ICMPGE: case ICMD_IF_ICMPGT: case ICMD_IF_ICMPLE: case ICMD_IF_ACMPEQ: case ICMD_IF_ACMPNE: case ICMD_IFNULL: case ICMD_IFNONNULL: case ICMD_IF_LEQ: case ICMD_IF_LNE: case ICMD_IF_LLT: case ICMD_IF_LGE: case ICMD_IF_LGT: case ICMD_IF_LLE: case ICMD_IF_LCMPEQ: case ICMD_IF_LCMPNE: case ICMD_IF_LCMPLT: case ICMD_IF_LCMPGE: case ICMD_IF_LCMPGT: case ICMD_IF_LCMPLE: tbptr = iptr->dst.block; ntbptr = bptr->next; printf("cond. L%03d\n", tbptr->nr); tfreq = pcode->bbfrequency[tbptr->nr]; ntfreq = pcode->bbfrequency[ntbptr->nr]; /* check which branch is more frequently */ if ((blocks[tbptr->nr] == false) && (tfreq > ntfreq)) { /* If we place the taken block, we need to change the conditional instruction (opcode and target). */ iptr->opc = jit_complement_condition(iptr->opc); iptr->dst.block = ntbptr; /* change the basic block order */ bptr->next = tbptr; /* place taken block */ blocks[tbptr->nr] = true; placed++; /* set last placed block */ bptr = tbptr; } else if (blocks[ntbptr->nr] == false) { /* place not-taken block */ blocks[ntbptr->nr] = true; placed++; /* set last placed block */ bptr = ntbptr; } else { tbptr = reorder_place_next_unplaced_block(jd, blocks, bptr); placed++; /* all blocks placed? return. */ if (tbptr == NULL) continue; /* set last placed block */ bptr = tbptr; } break; case ICMD_GOTO: tbptr = iptr->dst.block; printf("L%03d\n", tbptr->nr); /* If next block is already placed, search for another one. */ if (blocks[tbptr->nr] == true) { tbptr = reorder_place_next_unplaced_block(jd, blocks, bptr); placed++; /* all block placed? return. */ if (tbptr == NULL) continue; } else if (tbptr->predecessorcount > 1) { /* Check if the target block has other predecessors. And if the other predecessors have a higher frequency, don't place it. */ freq = pcode->bbfrequency[bptr->nr]; for (i = 0; i < tbptr->predecessorcount; i++) { pbptr = tbptr->predecessors[i]; /* skip the current basic block */ if (pbptr->nr != bptr->nr) { pfreq = pcode->bbfrequency[pbptr->nr]; /* Other predecessor has a higher frequency? Search for another block to place. */ if (pfreq > freq) { tbptr = reorder_place_next_unplaced_block(jd, blocks, bptr); placed++; /* all block placed? return. */ if (tbptr == NULL) break; goto goto_continue; } } } goto_continue: /* place block */ blocks[tbptr->nr] = true; placed++; } /* set last placed block */ bptr = tbptr; break; default: /* No control flow instruction at the end of the basic block (fall-through). Just place the block. */ puts("next"); tbptr = bptr->next; /* If next block is already placed, search for another one. */ if (blocks[tbptr->nr] == true) { tbptr = reorder_place_next_unplaced_block(jd, blocks, bptr); placed++; /* all block placed? return. */ if (tbptr == NULL) continue; } else { /* place block */ blocks[tbptr->nr] = true; placed++; } /* set last placed block */ bptr = tbptr; break; } } /* close the basic block chain with the last dummy basic block */ bptr->next = &jd->basicblocks[jd->basicblockcount]; puts(""); for (bptr = jd->basicblocks; bptr != NULL; bptr = bptr->next) { printf("L%03d\n", bptr->nr); } /* everything's ok */ return true; } /* * These are local overrides for various environment variables in Emacs. * Please do not remove this and leave it at the end of the file, where * Emacs will automagically detect them. * --------------------------------------------------------------------- * Local variables: * mode: c * indent-tabs-mode: t * c-basic-offset: 4 * tab-width: 4 * End: * vim:noexpandtab:sw=4:ts=4: */