/* src/vm/jit/mips/codegen.h - code generation macros and definitions for MIPS Copyright (C) 1996-2005, 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, 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: Andreas Krall $Id: codegen.h 4640 2006-03-16 17:24:18Z twisti $ */ #ifndef _CODEGEN_H #define _CODEGEN_H #include "config.h" #include "vm/types.h" #include "vm/jit/jit.h" /* some defines ***************************************************************/ #define PATCHER_CALL_SIZE 2 * 4 /* size in bytes of a patcher call */ /* additional functions and macros to generate code ***************************/ #define gen_nullptr_check(objreg) \ if (checknull) { \ M_BEQZ(objreg, 0); \ codegen_add_nullpointerexception_ref(cd, mcodeptr); \ M_NOP; \ } #define gen_bound_check \ if (checkbounds) { \ M_ILD(REG_ITMP3, s1, OFFSET(java_arrayheader, size)); \ M_CMPULT(s2, REG_ITMP3, REG_ITMP3); \ M_BEQZ(REG_ITMP3, 0); \ codegen_add_arrayindexoutofboundsexception_ref(cd, mcodeptr, s2); \ M_NOP; \ } #define gen_div_check(r) \ do { \ M_BEQZ((r), 0); \ codegen_add_arithmeticexception_ref(cd, mcodeptr); \ M_NOP; \ } while (0) /* MCODECHECK(icnt) */ #define MCODECHECK(icnt) \ if ((mcodeptr + (icnt)) > cd->mcodeend) \ mcodeptr = codegen_increase(cd, (u1 *) mcodeptr) #define ALIGNCODENOP \ if ((int) ((long) mcodeptr & 7)) { \ M_NOP; \ } /* M_INTMOVE: generates an integer-move from register a to b. if a and b are the same int-register, no code will be generated. */ #define M_INTMOVE(a,b) if (a != b) { M_MOV(a, b); } /* M_FLTMOVE: generates a floating-point-move from register a to b. if a and b are the same float-register, no code will be generated */ #define M_FLTMOVE(a,b) if (a != b) { M_DMOV(a, b); } #define M_TFLTMOVE(t,a,b) \ do { \ if ((a) != (b)) \ if ((t) == TYPE_DBL) { \ M_DMOV(a,b); \ } else { \ M_FMOV(a,b); \ } \ } while (0) #define M_TFLD(t,a,b,disp) \ if ((t) == TYPE_DBL) { \ M_DLD(a,b,disp); \ } else { \ M_FLD(a,b,disp); \ } #define M_TFST(t,a,b,disp) \ if ((t)==TYPE_DBL) \ {M_DST(a,b,disp);} \ else \ {M_FST(a,b,disp);} #define M_CCFLTMOVE(t1,t2,a,b) \ if ((t1)==(t2)) \ {M_TFLTMOVE(t1,a,b);} \ else \ if ((t1)==TYPE_DBL) \ {M_CVTDF(a,b);} \ else \ {M_CVTFD(a,b);} #define M_CCFLD(t1,t2,a,b,disp) \ if ((t1)==(t2)) \ {M_DLD(a,b,disp);} \ else { \ M_DLD(REG_FTMP1,b,disp); \ if ((t1)==TYPE_DBL) \ {M_CVTDF(REG_FTMP1,a);} \ else \ {M_CVTFD(REG_FTMP1,a);} \ } #define M_CCFST(t1,t2,a,b,disp) \ if ((t1)==(t2)) \ {M_DST(a,b,disp);} \ else { \ if ((t1)==TYPE_DBL) \ {M_CVTDF(a,REG_FTMP1);} \ else \ {M_CVTFD(a,REG_FTMP1);} \ M_DST(REG_FTMP1,b,disp); \ } /* var_to_reg_xxx: this function generates code to fetch data from a pseudo-register into a real register. If the pseudo-register has actually been assigned to a real register, no code will be emitted, since following operations can use this register directly. v: pseudoregister to be fetched from tempregnum: temporary register to be used if v is actually spilled to ram return: the register number, where the operand can be found after fetching (this wil be either tempregnum or the register number allready given to v) */ #define var_to_reg_int(regnr,v,tempnr) { \ if ((v)->flags & INMEMORY) { \ COUNT_SPILLS; \ M_LLD(tempnr, REG_SP, 8 * (v)->regoff); \ regnr = tempnr; \ } else regnr = (v)->regoff; \ } #define var_to_reg_flt(regnr,v,tempnr) { \ if ((v)->flags & INMEMORY) { \ COUNT_SPILLS; \ M_DLD(tempnr, REG_SP, 8 * (v)->regoff); \ regnr = tempnr; \ } else regnr = (v)->regoff; \ } /* store_reg_to_var_xxx: This function generates the code to store the result of an operation back into a spilled pseudo-variable. If the pseudo-variable has not been spilled in the first place, this function will generate nothing. v ............ Pseudovariable tempregnum ... Number of the temporary registers as returned by reg_of_var. */ #define store_reg_to_var_int(sptr, tempregnum) { \ if ((sptr)->flags & INMEMORY) { \ COUNT_SPILLS; \ M_LST(tempregnum, REG_SP, 8 * (sptr)->regoff); \ } \ } #define store_reg_to_var_flt(sptr, tempregnum) { \ if ((sptr)->flags & INMEMORY) { \ COUNT_SPILLS; \ M_DST(tempregnum, REG_SP, 8 * (sptr)->regoff); \ } \ } #define M_COPY(from,to) \ d = reg_of_var(rd, to, REG_IFTMP); \ if ((from->regoff != to->regoff) || \ ((from->flags ^ to->flags) & INMEMORY)) { \ if (IS_FLT_DBL_TYPE(from->type)) { \ var_to_reg_flt(s1, from, d); \ M_TFLTMOVE(from->type, s1, d); \ store_reg_to_var_flt(to, d); \ } else { \ var_to_reg_int(s1, from, d); \ M_INTMOVE(s1, d); \ store_reg_to_var_int(to, d); \ } \ } #define ICONST(r,c) \ if ((c) >= -32768 && (c) <= 32767) { \ M_IADD_IMM(REG_ZERO, (c), (r)); \ } else if ((c) >= 0 && (c) <= 0xffff) { \ M_OR_IMM(REG_ZERO, (c), (r)); \ } else { \ disp = dseg_adds4(cd, (c)); \ M_ILD((r), REG_PV, disp); \ } #define LCONST(r,c) \ if ((c) >= -32768 && (c) <= 32767) { \ M_LADD_IMM(REG_ZERO, (c), (r)); \ } else if ((c) >= 0 && (c) <= 0xffff) { \ M_OR_IMM(REG_ZERO, (c), (r)); \ } else { \ disp = dseg_adds8(cd, (c)); \ M_LLD((r), REG_PV, disp); \ } /* macros to create code ******************************************************/ /* code generation macros operands: op ..... opcode fu ..... function-number rs ..... register number source 1 rt ..... register number or constant integer source 2 rd ..... register number destination imm .... immediate/offset sa ..... shift amount */ #define M_ITYPE(op,rs,rt,imm) \ *(mcodeptr++) = (((op) << 26) | ((rs) << 21) | ((rt) << 16) | ((imm) & 0xffff)) #define M_JTYPE(op,imm) \ *(mcodeptr++) = (((op) << 26) | ((off) & 0x3ffffff)) #define M_RTYPE(op,rs,rt,rd,sa,fu) \ *(mcodeptr++) = (((op) << 26) | ((rs) << 21) | ((rt) << 16) | ((rd) << 11) | ((sa) << 6) | (fu)) #define M_FP2(fu, fmt, fs, fd) M_RTYPE(0x11, fmt, 0, fs, fd, fu) #define M_FP3(fu, fmt, fs, ft, fd) M_RTYPE(0x11, fmt, ft, fs, fd, fu) #define FMT_F 16 #define FMT_D 17 #define FMT_I 20 #define FMT_L 21 /* macros for all used commands (see a MIPS-manual for description) ***********/ /* load/store macros use the form OPERATION(source/dest, base, offset) */ #define M_LDA(a,b,disp) \ do { \ s4 lo = (short) (disp); \ s4 hi = (short) (((disp) - lo) >> 16); \ if (hi == 0) { \ M_AADD_IMM(b,lo,a); \ } else { \ M_LUI(REG_ITMP3,hi); \ M_AADD_IMM(REG_ITMP3,lo,REG_ITMP3); \ M_AADD(REG_ITMP3,b,a); \ } \ } while (0) #define M_BLDS(a,b,disp) M_ITYPE(0x20,b,a,disp) /* 8 load */ #define M_BLDU(a,b,disp) M_ITYPE(0x24,b,a,disp) /* 8 load */ #define M_SLDS(a,b,disp) M_ITYPE(0x21,b,a,disp) /* 16 load */ #define M_SLDU(a,b,disp) M_ITYPE(0x25,b,a,disp) /* 16 load */ #define M_ILD_INTERN(a,b,disp) M_ITYPE(0x23,b,a,disp) /* 32 load */ #define M_LLD_INTERN(a,b,disp) M_ITYPE(0x37,b,a,disp) /* 64 load */ #define M_ILD(a,b,disp) \ do { \ s4 lo = (short) (disp); \ s4 hi = (short) (((disp) - lo) >> 16); \ if (hi == 0) { \ M_ILD_INTERN(a,b,lo); \ } else { \ M_LUI(a,hi); \ M_AADD(b,a,a); \ M_ILD_INTERN(a,a,lo); \ } \ } while (0) #define M_LLD(a,b,disp) \ do { \ s4 lo = (short) (disp); \ s4 hi = (short) (((disp) - lo) >> 16); \ if (hi == 0) { \ M_LLD_INTERN(a,b,lo); \ } else { \ M_LUI(a,hi); \ M_AADD(b,a,a); \ M_LLD_INTERN(a,a,lo); \ } \ } while (0) #define M_BST(a,b,disp) M_ITYPE(0x28,b,a,disp) /* 8 store */ #define M_SST(a,b,disp) M_ITYPE(0x29,b,a,disp) /* 16 store */ #define M_IST_INTERN(a,b,disp) M_ITYPE(0x2b,b,a,disp) /* 32 store */ #define M_LST_INTERN(a,b,disp) M_ITYPE(0x3f,b,a,disp) /* 64 store */ #define M_IST(a,b,disp) \ do { \ s4 lo = (short) (disp); \ s4 hi = (short) (((disp) - lo) >> 16); \ if (hi == 0) { \ M_IST_INTERN(a,b,lo); \ } else { \ M_LUI(REG_ITMP3, hi); \ M_AADD(b, REG_ITMP3, REG_ITMP3); \ M_IST_INTERN(a,REG_ITMP3,lo); \ } \ } while (0) #define M_LST(a,b,disp) \ do { \ s4 lo = (short) (disp); \ s4 hi = (short) (((disp) - lo) >> 16); \ if (hi == 0) { \ M_LST_INTERN(a,b,lo); \ } else { \ M_LUI(REG_ITMP3, hi); \ M_AADD(b, REG_ITMP3, REG_ITMP3); \ M_LST_INTERN(a,REG_ITMP3,lo); \ } \ } while (0) #define M_FLD_INTERN(a,b,disp) M_ITYPE(0x31,b,a,disp) /* load flt */ #define M_DLD_INTERN(a,b,disp) M_ITYPE(0x35,b,a,disp) /* load dbl */ #define M_FLD(a,b,disp) \ do { \ s4 lo = (short) (disp); \ s4 hi = (short) (((disp) - lo) >> 16); \ if (hi == 0) { \ M_FLD_INTERN(a,b,lo); \ } else { \ M_LUI(REG_ITMP3,hi); \ M_AADD(b,REG_ITMP3,REG_ITMP3); \ M_FLD_INTERN(a,REG_ITMP3,lo); \ } \ } while (0) #define M_DLD(a,b,disp) \ do { \ s4 lo = (short) (disp); \ s4 hi = (short) (((disp) - lo) >> 16); \ if (hi == 0) { \ M_DLD_INTERN(a,b,lo); \ } else { \ M_LUI(REG_ITMP3,hi); \ M_AADD(b,REG_ITMP3,REG_ITMP3); \ M_DLD_INTERN(a,REG_ITMP3,lo); \ } \ } while (0) #define M_FST_INTERN(a,b,disp) M_ITYPE(0x39,b,a,disp) /* store flt */ #define M_DST_INTERN(a,b,disp) M_ITYPE(0x3d,b,a,disp) /* store dbl */ #define M_FST(a,b,disp) \ do { \ s4 lo = (short) (disp); \ s4 hi = (short) (((disp) - lo) >> 16); \ if (hi == 0) { \ M_FST_INTERN(a,b,lo); \ } else { \ M_LUI(REG_ITMP3, hi); \ M_AADD(b, REG_ITMP3, REG_ITMP3); \ M_FST_INTERN(a,REG_ITMP3,lo); \ } \ } while (0) #define M_DST(a,b,disp) \ do { \ s4 lo = (short) (disp); \ s4 hi = (short) (((disp) - lo) >> 16); \ if (hi == 0) { \ M_DST_INTERN(a,b,lo); \ } else { \ M_LUI(REG_ITMP3, hi); \ M_AADD(b, REG_ITMP3, REG_ITMP3); \ M_DST_INTERN(a,REG_ITMP3,lo); \ } \ } while (0) #define M_BEQ(a,b,disp) M_ITYPE(0x04,a,b,disp) /* br a == b */ #define M_BNE(a,b,disp) M_ITYPE(0x05,a,b,disp) /* br a != b */ #define M_BEQZ(a,disp) M_ITYPE(0x04,a,0,disp) /* br a == 0 */ #define M_BLTZ(a,disp) M_ITYPE(0x01,a,0,disp) /* br a < 0 */ #define M_BLEZ(a,disp) M_ITYPE(0x06,a,0,disp) /* br a <= 0 */ #define M_BNEZ(a,disp) M_ITYPE(0x05,a,0,disp) /* br a != 0 */ #define M_BGEZ(a,disp) M_ITYPE(0x01,a,1,disp) /* br a >= 0 */ #define M_BGTZ(a,disp) M_ITYPE(0x07,a,0,disp) /* br a > 0 */ #define M_BEQL(a,b,disp) M_ITYPE(0x14,a,b,disp) /* br a == b */ #define M_BNEL(a,b,disp) M_ITYPE(0x15,a,b,disp) /* br a != b */ #define M_BEQZL(a,disp) M_ITYPE(0x14,a,0,disp) /* br a == 0 */ #define M_BLTZL(a,disp) M_ITYPE(0x01,a,2,disp) /* br a < 0 */ #define M_BLEZL(a,disp) M_ITYPE(0x16,a,0,disp) /* br a <= 0 */ #define M_BNEZL(a,disp) M_ITYPE(0x15,a,0,disp) /* br a != 0 */ #define M_BGEZL(a,disp) M_ITYPE(0x01,a,3,disp) /* br a >= 0 */ #define M_BGTZL(a,disp) M_ITYPE(0x17,a,0,disp) /* br a > 0 */ #define M_BR(disp) M_ITYPE(0x04,0,0,disp) /* branch */ #define M_BRS(disp) M_ITYPE(0x01,0,17,disp) /* branch sbr */ #define M_JMP(a) M_RTYPE(0,a,0,0,0,0x08) /* jump */ #define M_JSR(r,a) M_RTYPE(0,a,0,r,0,0x09) /* call */ #define M_RET(a) M_RTYPE(0,a,0,0,0,0x08) /* return */ #define M_TGE(a,b,code) M_RTYPE(0,a,b,0,code&3ff,0x30) /* trp a >= b */ #define M_TGEU(a,b,code) M_RTYPE(0,a,b,0,code&3ff,0x31) /* trp a >= b */ #define M_TLT(a,b,code) M_RTYPE(0,a,b,0,code&3ff,0x32) /* trp a < b */ #define M_TLTU(a,b,code) M_RTYPE(0,a,b,0,code&3ff,0x33) /* trp a < b */ #define M_TEQ(a,b,code) M_RTYPE(0,a,b,0,code&3ff,0x34) /* trp a == b */ #define M_TNE(a,b,code) M_RTYPE(0,a,b,0,code&3ff,0x36) /* trp a != b */ #define M_TLE(a,b,code) M_RTYPE(0,b,a,0,code&3ff,0x30) /* trp a <= b */ #define M_TLEU(a,b,code) M_RTYPE(0,b,a,0,code&3ff,0x31) /* trp a <= b */ #define M_TGT(a,b,code) M_RTYPE(0,b,a,0,code&3ff,0x32) /* trp a > b */ #define M_TGTU(a,b,code) M_RTYPE(0,b,a,0,code&3ff,0x33) /* trp a > b */ #define M_TGE_IMM(a,b) M_ITYPE(1,a,0x08,b) /* trp a >= b */ #define M_TGEU_IMM(a,b) M_ITYPE(1,a,0x09,b) /* trp a >= b */ #define M_TLT_IMM(a,b) M_ITYPE(1,a,0x0a,b) /* trp a < b */ #define M_TLTU_IMM(a,b) M_ITYPE(1,a,0x0b,b) /* trp a < b */ #define M_TEQ_IMM(a,b) M_ITYPE(1,a,0x0c,b) /* trp a == b */ #define M_TNE_IMM(a,b) M_ITYPE(1,a,0x0e,b) /* trp a != b */ #if 0 #define M_TGT_IMM(a,b) M_ITYPE(1,a,0x08,b+1) /* trp a > b */ #define M_TGTU_IMM(a,b) M_ITYPE(1,a,0x09,b+1) /* trp a > b */ #define M_TLE_IMM(a,b) M_ITYPE(1,a,0x0a,b+1) /* trp a <= b */ #define M_TLEU_IMM(a,b) M_ITYPE(1,a,0x0b,b+1) /* trp a <= b */ #endif /* arithmetic macros use the form OPERATION(source, source/immediate, dest) */ #define M_IADD(a,b,c) M_RTYPE(0,a,b,c,0,0x21) /* 32 add */ #define M_LADD(a,b,c) M_RTYPE(0,a,b,c,0,0x2d) /* 64 add */ #define M_ISUB(a,b,c) M_RTYPE(0,a,b,c,0,0x23) /* 32 sub */ #define M_LSUB(a,b,c) M_RTYPE(0,a,b,c,0,0x2f) /* 64 sub */ #define M_IMUL(a,b) M_ITYPE(0,a,b,0x18) /* 32 mul */ #define M_LMUL(a,b) M_ITYPE(0,a,b,0x1c) /* 64 mul */ #define M_IDIV(a,b) M_ITYPE(0,a,b,0x1a) /* 32 div */ #define M_LDIV(a,b) M_ITYPE(0,a,b,0x1e) /* 64 div */ #define M_MFLO(a) M_RTYPE(0,0,0,a,0,0x12) /* quotient */ #define M_MFHI(a) M_RTYPE(0,0,0,a,0,0x10) /* remainder */ #define M_IADD_IMM(a,b,c) M_ITYPE(0x09,a,c,b) /* 32 add */ #define M_LADD_IMM(a,b,c) M_ITYPE(0x19,a,c,b) /* 64 add */ #define M_ISUB_IMM(a,b,c) M_ITYPE(0x09,a,c,-(b)) /* 32 sub */ #define M_LSUB_IMM(a,b,c) M_ITYPE(0x19,a,c,-(b)) /* 64 sub */ #define M_LUI(a,imm) M_ITYPE(0x0f,0,a,imm) /* a = imm<<16*/ #define M_CMPLT(a,b,c) M_RTYPE(0,a,b,c,0,0x2a) /* c = a < b */ #define M_CMPGT(a,b,c) M_RTYPE(0,b,a,c,0,0x2a) /* c = a > b */ #define M_CMPULT(a,b,c) M_RTYPE(0,a,b,c,0,0x2b) /* c = a < b */ #define M_CMPUGT(a,b,c) M_RTYPE(0,b,a,c,0,0x2b) /* c = a > b */ #define M_CMPLT_IMM(a,b,c) M_ITYPE(0x0a,a,c,b) /* c = a < b */ #define M_CMPULT_IMM(a,b,c) M_ITYPE(0x0b,a,c,b) /* c = a < b */ #define M_AND(a,b,c) M_RTYPE(0,a,b,c,0,0x24) /* c = a & b */ #define M_OR( a,b,c) M_RTYPE(0,a,b,c,0,0x25) /* c = a | b */ #define M_XOR(a,b,c) M_RTYPE(0,a,b,c,0,0x26) /* c = a ^ b */ #define M_AND_IMM(a,b,c) M_ITYPE(0x0c,a,c,b) /* c = a & b */ #define M_OR_IMM( a,b,c) M_ITYPE(0x0d,a,c,b) /* c = a | b */ #define M_XOR_IMM(a,b,c) M_ITYPE(0x0e,a,c,b) /* c = a ^ b */ #define M_CZEXT(a,c) M_AND_IMM(a,0xffff,c) /* c = zext(a)*/ #define M_ISLL(a,b,c) M_RTYPE(0,b,a,c,0,0x04) /* c = a << b */ #define M_ISRL(a,b,c) M_RTYPE(0,b,a,c,0,0x06) /* c = a >>>b */ #define M_ISRA(a,b,c) M_RTYPE(0,b,a,c,0,0x07) /* c = a >> b */ #define M_LSLL(a,b,c) M_RTYPE(0,b,a,c,0,0x14) /* c = a << b */ #define M_LSRL(a,b,c) M_RTYPE(0,b,a,c,0,0x16) /* c = a >>>b */ #define M_LSRA(a,b,c) M_RTYPE(0,b,a,c,0,0x17) /* c = a >> b */ #define M_ISLL_IMM(a,b,c) M_RTYPE(0,0,a,c,(b)&31,0x00) /* c = a << b */ #define M_ISRL_IMM(a,b,c) M_RTYPE(0,0,a,c,(b)&31,0x02) /* c = a >>>b */ #define M_ISRA_IMM(a,b,c) M_RTYPE(0,0,a,c,(b)&31,0x03) /* c = a >> b */ #define M_LSLL_IMM(a,b,c) M_RTYPE(0,0,a,c,(b)&31,0x38+((b)>>3&4)) /*c = a << b*/ #define M_LSRL_IMM(a,b,c) M_RTYPE(0,0,a,c,(b)&31,0x3a+((b)>>3&4)) /*c = a >>>b*/ #define M_LSRA_IMM(a,b,c) M_RTYPE(0,0,a,c,(b)&31,0x3b+((b)>>3&4)) /*c = a >> b*/ #define M_MOV(a,c) M_OR(a,0,c) /* c = a */ #define M_CLR(c) M_OR(0,0,c) /* c = 0 */ #define M_NOP M_ISLL_IMM(0,0,0) /* ; */ /* floating point macros use the form OPERATION(source, source, dest) */ #define M_FADD(a,b,c) M_FP3(0x00,FMT_F,a,b,c) /* flt add */ #define M_DADD(a,b,c) M_FP3(0x00,FMT_D,a,b,c) /* dbl add */ #define M_FSUB(a,b,c) M_FP3(0x01,FMT_F,a,b,c) /* flt sub */ #define M_DSUB(a,b,c) M_FP3(0x01,FMT_D,a,b,c) /* dbl sub */ #define M_FMUL(a,b,c) M_FP3(0x02,FMT_F,a,b,c) /* flt mul */ #define M_DMUL(a,b,c) M_FP3(0x02,FMT_D,a,b,c) /* dbl mul */ #define M_FDIV(a,b,c) M_FP3(0x03,FMT_F,a,b,c) /* flt div */ #define M_DDIV(a,b,c) M_FP3(0x03,FMT_D,a,b,c) /* dbl div */ #define M_FSQRT(a,c) M_FP2(0x04,FMT_F,a,c) /* flt sqrt */ #define M_DSQRT(a,c) M_FP2(0x04,FMT_D,a,c) /* dbl sqrt */ #define M_FABS(a,c) M_FP2(0x05,FMT_F,a,c) /* flt abs */ #define M_DABS(a,c) M_FP2(0x05,FMT_D,a,c) /* dbl abs */ #define M_FMOV(a,c) M_FP2(0x06,FMT_F,a,c) /* flt mov */ #define M_DMOV(a,c) M_FP2(0x06,FMT_D,a,c) /* dbl mov */ #define M_FNEG(a,c) M_FP2(0x07,FMT_F,a,c) /* flt neg */ #define M_DNEG(a,c) M_FP2(0x07,FMT_D,a,c) /* dbl neg */ #define M_ROUNDFI(a,c) M_FP2(0x0c,FMT_F,a,c) /* flt roundi */ #define M_ROUNDDI(a,c) M_FP2(0x0c,FMT_D,a,c) /* dbl roundi */ #define M_TRUNCFI(a,c) M_FP2(0x0d,FMT_F,a,c) /* flt trunci */ #define M_TRUNCDI(a,c) M_FP2(0x0d,FMT_D,a,c) /* dbl trunci */ #define M_CEILFI(a,c) M_FP2(0x0e,FMT_F,a,c) /* flt ceili */ #define M_CEILDI(a,c) M_FP2(0x0e,FMT_D,a,c) /* dbl ceili */ #define M_FLOORFI(a,c) M_FP2(0x0f,FMT_F,a,c) /* flt trunci */ #define M_FLOORDI(a,c) M_FP2(0x0f,FMT_D,a,c) /* dbl trunci */ #define M_ROUNDFL(a,c) M_FP2(0x08,FMT_F,a,c) /* flt roundl */ #define M_ROUNDDL(a,c) M_FP2(0x08,FMT_D,a,c) /* dbl roundl */ #define M_TRUNCFL(a,c) M_FP2(0x09,FMT_F,a,c) /* flt truncl */ #define M_TRUNCDL(a,c) M_FP2(0x09,FMT_D,a,c) /* dbl truncl */ #define M_CEILFL(a,c) M_FP2(0x0a,FMT_F,a,c) /* flt ceill */ #define M_CEILDL(a,c) M_FP2(0x0a,FMT_D,a,c) /* dbl ceill */ #define M_FLOORFL(a,c) M_FP2(0x0b,FMT_F,a,c) /* flt truncl */ #define M_FLOORDL(a,c) M_FP2(0x0b,FMT_D,a,c) /* dbl truncl */ #define M_CVTDF(a,c) M_FP2(0x20,FMT_D,a,c) /* dbl2flt */ #define M_CVTIF(a,c) M_FP2(0x20,FMT_I,a,c) /* int2flt */ #define M_CVTLF(a,c) M_FP2(0x20,FMT_L,a,c) /* long2flt */ #define M_CVTFD(a,c) M_FP2(0x21,FMT_F,a,c) /* flt2dbl */ #define M_CVTID(a,c) M_FP2(0x21,FMT_I,a,c) /* int2dbl */ #define M_CVTLD(a,c) M_FP2(0x21,FMT_L,a,c) /* long2dbl */ #define M_CVTFI(a,c) M_FP2(0x24,FMT_F,a,c) /* flt2int */ #define M_CVTDI(a,c) M_FP2(0x24,FMT_D,a,c) /* dbl2int */ #define M_CVTFL(a,c) M_FP2(0x25,FMT_F,a,c) /* flt2long */ #define M_CVTDL(a,c) M_FP2(0x25,FMT_D,a,c) /* dbl2long */ #define M_MOVDI(d,i) M_FP3(0,0,d,i,0) /* i = d */ #define M_MOVDL(d,l) M_FP3(0,1,d,l,0) /* l = d */ #define M_MOVID(i,d) M_FP3(0,4,d,i,0) /* d = i */ #define M_MOVLD(l,d) M_FP3(0,5,d,l,0) /* d = l */ #define M_DMFC1(l,f) M_FP3(0,1,f,l,0) #define M_DMTC1(l,f) M_FP3(0,5,f,l,0) #define M_FCMPFF(a,b) M_FP3(0x30,FMT_F,a,b,0) /* c = a == b */ #define M_FCMPFD(a,b) M_FP3(0x30,FMT_D,a,b,0) /* c = a == b */ #define M_FCMPUNF(a,b) M_FP3(0x31,FMT_F,a,b,0) /* c = a == b */ #define M_FCMPUND(a,b) M_FP3(0x31,FMT_D,a,b,0) /* c = a == b */ #define M_FCMPEQF(a,b) M_FP3(0x32,FMT_F,a,b,0) /* c = a == b */ #define M_FCMPEQD(a,b) M_FP3(0x32,FMT_D,a,b,0) /* c = a == b */ #define M_FCMPUEQF(a,b) M_FP3(0x33,FMT_F,a,b,0) /* c = a == b */ #define M_FCMPUEQD(a,b) M_FP3(0x33,FMT_D,a,b,0) /* c = a == b */ #define M_FCMPOLTF(a,b) M_FP3(0x34,FMT_F,a,b,0) /* c = a < b */ #define M_FCMPOLTD(a,b) M_FP3(0x34,FMT_D,a,b,0) /* c = a < b */ #define M_FCMPULTF(a,b) M_FP3(0x35,FMT_F,a,b,0) /* c = a < b */ #define M_FCMPULTD(a,b) M_FP3(0x35,FMT_D,a,b,0) /* c = a < b */ #define M_FCMPOLEF(a,b) M_FP3(0x36,FMT_F,a,b,0) /* c = a <= b */ #define M_FCMPOLED(a,b) M_FP3(0x36,FMT_D,a,b,0) /* c = a <= b */ #define M_FCMPULEF(a,b) M_FP3(0x37,FMT_F,a,b,0) /* c = a <= b */ #define M_FCMPULED(a,b) M_FP3(0x37,FMT_D,a,b,0) /* c = a <= b */ #define M_FBF(disp) M_ITYPE(0x11,8,0,disp) /* br false */ #define M_FBT(disp) M_ITYPE(0x11,8,1,disp) /* br true */ #define M_FBFL(disp) M_ITYPE(0x11,8,2,disp) /* br false */ #define M_FBTL(disp) M_ITYPE(0x11,8,3,disp) /* br true */ #define M_CMOVF(a,b) M_RTYPE(0x00,a,0,b,0,1) #define M_CMOVT(a,b) M_RTYPE(0x00,a,1,b,0,1) /* * Load Address pseudo instruction: * -n32 addressing mode -> 32 bit addrs, -64 addressing mode -> 64 bit addrs */ #if SIZEOF_VOID_P == 8 #define POINTERSHIFT 3 #define M_ALD_INTERN(a,b,disp) M_LLD_INTERN(a,b,disp) #define M_ALD(a,b,disp) M_LLD(a,b,disp) #define M_AST_INTERN(a,b,disp) M_LST_INTERN(a,b,disp) #define M_AST(a,b,disp) M_LST(a,b,disp) #define M_AADD(a,b,c) M_LADD(a,b,c) #define M_AADD_IMM(a,b,c) M_LADD_IMM(a,b,c) #define M_ASUB_IMM(a,b,c) M_LSUB_IMM(a,b,c) #define M_ASLL_IMM(a,b,c) M_LSLL_IMM(a,b,c) #else /* SIZEOF_VOID_P == 8 */ #define POINTERSHIFT 2 #define M_ALD_INTERN(a,b,disp) M_ILD_INTERN(a,b,disp) #define M_ALD(a,b,disp) M_ILD(a,b,disp) #define M_AST_INTERN(a,b,disp) M_IST_INTERN(a,b,disp) #define M_AST(a,b,disp) M_IST(a,b,disp) #define M_AADD(a,b,c) M_IADD(a,b,c) #define M_AADD_IMM(a,b,c) M_IADD_IMM(a,b,c) #define M_ASUB_IMM(a,b,c) M_ISUB_IMM(a,b,c) #define M_ASLL_IMM(a,b,c) M_ISLL_IMM(a,b,c) #endif /* SIZEOF_VOID_P == 8 */ /* function gen_resolvebranch ************************************************** backpatches a branch instruction; MIPS branch instructions are very regular, so it is only necessary to overwrite some fixed bits in the instruction. parameters: ip ... pointer to instruction after branch (void*) so ... offset of instruction after branch (s4) to ... offset of branch target (s4) *******************************************************************************/ #define gen_resolvebranch(ip,so,to) \ do { \ s4 offset; \ \ offset = ((s4) (to) - (so)) >> 2; \ \ /* On the MIPS we can only branch signed 16-bit instruction words */ \ /* (signed 18-bit = 32KB = +/- 16KB). Check this! */ \ \ if ((offset < (s4) 0xffff8000) || (offset > (s4) 0x00007fff)) { \ throw_cacao_exception_exit(string_java_lang_InternalError, \ "Jump offset is out of range: %d > +/-%d", \ offset, 0x00007fff); \ } \ \ ((s4 *) (ip))[-1] |= (offset & 0x0000ffff); \ } while (0) /* function prototypes ********************************************************/ void docacheflush(u1 *p, long bytelen); #endif /* _CODEGEN_H */ /* * 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: */