📄 hdefs.c
字号:
/*---------------------------------------------------------------*//*--- ---*//*--- This file (host-ppc/hdefs.c) is ---*//*--- Copyright (C) OpenWorks LLP. All rights reserved. ---*//*--- ---*//*---------------------------------------------------------------*//* This file is part of LibVEX, a library for dynamic binary instrumentation and translation. Copyright (C) 2004-2006 OpenWorks LLP. All rights reserved. This library is made available under a dual licensing scheme. If you link LibVEX against other code all of which is itself licensed under the GNU General Public License, version 2 dated June 1991 ("GPL v2"), then you may use LibVEX under the terms of the GPL v2, as appearing in the file LICENSE.GPL. If the file LICENSE.GPL is missing, you can obtain a copy of the GPL v2 from the Free Software Foundation Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. For any other uses of LibVEX, you must first obtain a commercial license from OpenWorks LLP. Please contact info@open-works.co.uk for information about commercial licensing. This software is provided by OpenWorks LLP "as is" and any express or implied warranties, including, but not limited to, the implied warranties of merchantability and fitness for a particular purpose are disclaimed. In no event shall OpenWorks LLP be liable for any direct, indirect, incidental, special, exemplary, or consequential damages (including, but not limited to, procurement of substitute goods or services; loss of use, data, or profits; or business interruption) however caused and on any theory of liability, whether in contract, strict liability, or tort (including negligence or otherwise) arising in any way out of the use of this software, even if advised of the possibility of such damage. Neither the names of the U.S. Department of Energy nor the University of California nor the names of its contributors may be used to endorse or promote products derived from this software without prior written permission.*/#include "libvex_basictypes.h"#include "libvex.h"#include "libvex_trc_values.h"#include "main/vex_util.h"#include "host-generic/h_generic_regs.h"#include "host-ppc/hdefs.h"/* --------- Registers. --------- */void ppHRegPPC ( HReg reg ) { Int r; static HChar* ireg32_names[32] = { "%r0", "%r1", "%r2", "%r3", "%r4", "%r5", "%r6", "%r7", "%r8", "%r9", "%r10", "%r11", "%r12", "%r13", "%r14", "%r15", "%r16", "%r17", "%r18", "%r19", "%r20", "%r21", "%r22", "%r23", "%r24", "%r25", "%r26", "%r27", "%r28", "%r29", "%r30", "%r31" }; /* Be generic for all virtual regs. */ if (hregIsVirtual(reg)) { ppHReg(reg); return; } /* But specific for real regs. */ switch (hregClass(reg)) { case HRcInt64: r = hregNumber(reg); vassert(r >= 0 && r < 32); vex_printf("%s", ireg32_names[r]); return; case HRcInt32: r = hregNumber(reg); vassert(r >= 0 && r < 32); vex_printf("%s", ireg32_names[r]); return; case HRcFlt64: r = hregNumber(reg); vassert(r >= 0 && r < 32); vex_printf("%%fr%d", r); return; case HRcVec128: r = hregNumber(reg); vassert(r >= 0 && r < 32); vex_printf("%%v%d", r); return; default: vpanic("ppHRegPPC"); }}#define MkHRegGPR(_n, _mode64) \ mkHReg(_n, _mode64 ? HRcInt64 : HRcInt32, False)HReg hregPPC_GPR0 ( Bool mode64 ) { return MkHRegGPR( 0, mode64); }HReg hregPPC_GPR1 ( Bool mode64 ) { return MkHRegGPR( 1, mode64); }HReg hregPPC_GPR2 ( Bool mode64 ) { return MkHRegGPR( 2, mode64); }HReg hregPPC_GPR3 ( Bool mode64 ) { return MkHRegGPR( 3, mode64); }HReg hregPPC_GPR4 ( Bool mode64 ) { return MkHRegGPR( 4, mode64); }HReg hregPPC_GPR5 ( Bool mode64 ) { return MkHRegGPR( 5, mode64); }HReg hregPPC_GPR6 ( Bool mode64 ) { return MkHRegGPR( 6, mode64); }HReg hregPPC_GPR7 ( Bool mode64 ) { return MkHRegGPR( 7, mode64); }HReg hregPPC_GPR8 ( Bool mode64 ) { return MkHRegGPR( 8, mode64); }HReg hregPPC_GPR9 ( Bool mode64 ) { return MkHRegGPR( 9, mode64); }HReg hregPPC_GPR10 ( Bool mode64 ) { return MkHRegGPR(10, mode64); }HReg hregPPC_GPR11 ( Bool mode64 ) { return MkHRegGPR(11, mode64); }HReg hregPPC_GPR12 ( Bool mode64 ) { return MkHRegGPR(12, mode64); }HReg hregPPC_GPR13 ( Bool mode64 ) { return MkHRegGPR(13, mode64); }HReg hregPPC_GPR14 ( Bool mode64 ) { return MkHRegGPR(14, mode64); }HReg hregPPC_GPR15 ( Bool mode64 ) { return MkHRegGPR(15, mode64); }HReg hregPPC_GPR16 ( Bool mode64 ) { return MkHRegGPR(16, mode64); }HReg hregPPC_GPR17 ( Bool mode64 ) { return MkHRegGPR(17, mode64); }HReg hregPPC_GPR18 ( Bool mode64 ) { return MkHRegGPR(18, mode64); }HReg hregPPC_GPR19 ( Bool mode64 ) { return MkHRegGPR(19, mode64); }HReg hregPPC_GPR20 ( Bool mode64 ) { return MkHRegGPR(20, mode64); }HReg hregPPC_GPR21 ( Bool mode64 ) { return MkHRegGPR(21, mode64); }HReg hregPPC_GPR22 ( Bool mode64 ) { return MkHRegGPR(22, mode64); }HReg hregPPC_GPR23 ( Bool mode64 ) { return MkHRegGPR(23, mode64); }HReg hregPPC_GPR24 ( Bool mode64 ) { return MkHRegGPR(24, mode64); }HReg hregPPC_GPR25 ( Bool mode64 ) { return MkHRegGPR(25, mode64); }HReg hregPPC_GPR26 ( Bool mode64 ) { return MkHRegGPR(26, mode64); }HReg hregPPC_GPR27 ( Bool mode64 ) { return MkHRegGPR(27, mode64); }HReg hregPPC_GPR28 ( Bool mode64 ) { return MkHRegGPR(28, mode64); }HReg hregPPC_GPR29 ( Bool mode64 ) { return MkHRegGPR(29, mode64); }HReg hregPPC_GPR30 ( Bool mode64 ) { return MkHRegGPR(30, mode64); }HReg hregPPC_GPR31 ( Bool mode64 ) { return MkHRegGPR(31, mode64); }#undef MK_INT_HREGHReg hregPPC_FPR0 ( void ) { return mkHReg( 0, HRcFlt64, False); }HReg hregPPC_FPR1 ( void ) { return mkHReg( 1, HRcFlt64, False); }HReg hregPPC_FPR2 ( void ) { return mkHReg( 2, HRcFlt64, False); }HReg hregPPC_FPR3 ( void ) { return mkHReg( 3, HRcFlt64, False); }HReg hregPPC_FPR4 ( void ) { return mkHReg( 4, HRcFlt64, False); }HReg hregPPC_FPR5 ( void ) { return mkHReg( 5, HRcFlt64, False); }HReg hregPPC_FPR6 ( void ) { return mkHReg( 6, HRcFlt64, False); }HReg hregPPC_FPR7 ( void ) { return mkHReg( 7, HRcFlt64, False); }HReg hregPPC_FPR8 ( void ) { return mkHReg( 8, HRcFlt64, False); }HReg hregPPC_FPR9 ( void ) { return mkHReg( 9, HRcFlt64, False); }HReg hregPPC_FPR10 ( void ) { return mkHReg(10, HRcFlt64, False); }HReg hregPPC_FPR11 ( void ) { return mkHReg(11, HRcFlt64, False); }HReg hregPPC_FPR12 ( void ) { return mkHReg(12, HRcFlt64, False); }HReg hregPPC_FPR13 ( void ) { return mkHReg(13, HRcFlt64, False); }HReg hregPPC_FPR14 ( void ) { return mkHReg(14, HRcFlt64, False); }HReg hregPPC_FPR15 ( void ) { return mkHReg(15, HRcFlt64, False); }HReg hregPPC_FPR16 ( void ) { return mkHReg(16, HRcFlt64, False); }HReg hregPPC_FPR17 ( void ) { return mkHReg(17, HRcFlt64, False); }HReg hregPPC_FPR18 ( void ) { return mkHReg(18, HRcFlt64, False); }HReg hregPPC_FPR19 ( void ) { return mkHReg(19, HRcFlt64, False); }HReg hregPPC_FPR20 ( void ) { return mkHReg(20, HRcFlt64, False); }HReg hregPPC_FPR21 ( void ) { return mkHReg(21, HRcFlt64, False); }HReg hregPPC_FPR22 ( void ) { return mkHReg(22, HRcFlt64, False); }HReg hregPPC_FPR23 ( void ) { return mkHReg(23, HRcFlt64, False); }HReg hregPPC_FPR24 ( void ) { return mkHReg(24, HRcFlt64, False); }HReg hregPPC_FPR25 ( void ) { return mkHReg(25, HRcFlt64, False); }HReg hregPPC_FPR26 ( void ) { return mkHReg(26, HRcFlt64, False); }HReg hregPPC_FPR27 ( void ) { return mkHReg(27, HRcFlt64, False); }HReg hregPPC_FPR28 ( void ) { return mkHReg(28, HRcFlt64, False); }HReg hregPPC_FPR29 ( void ) { return mkHReg(29, HRcFlt64, False); }HReg hregPPC_FPR30 ( void ) { return mkHReg(30, HRcFlt64, False); }HReg hregPPC_FPR31 ( void ) { return mkHReg(31, HRcFlt64, False); }HReg hregPPC_VR0 ( void ) { return mkHReg( 0, HRcVec128, False); }HReg hregPPC_VR1 ( void ) { return mkHReg( 1, HRcVec128, False); }HReg hregPPC_VR2 ( void ) { return mkHReg( 2, HRcVec128, False); }HReg hregPPC_VR3 ( void ) { return mkHReg( 3, HRcVec128, False); }HReg hregPPC_VR4 ( void ) { return mkHReg( 4, HRcVec128, False); }HReg hregPPC_VR5 ( void ) { return mkHReg( 5, HRcVec128, False); }HReg hregPPC_VR6 ( void ) { return mkHReg( 6, HRcVec128, False); }HReg hregPPC_VR7 ( void ) { return mkHReg( 7, HRcVec128, False); }HReg hregPPC_VR8 ( void ) { return mkHReg( 8, HRcVec128, False); }HReg hregPPC_VR9 ( void ) { return mkHReg( 9, HRcVec128, False); }HReg hregPPC_VR10 ( void ) { return mkHReg(10, HRcVec128, False); }HReg hregPPC_VR11 ( void ) { return mkHReg(11, HRcVec128, False); }HReg hregPPC_VR12 ( void ) { return mkHReg(12, HRcVec128, False); }HReg hregPPC_VR13 ( void ) { return mkHReg(13, HRcVec128, False); }HReg hregPPC_VR14 ( void ) { return mkHReg(14, HRcVec128, False); }HReg hregPPC_VR15 ( void ) { return mkHReg(15, HRcVec128, False); }HReg hregPPC_VR16 ( void ) { return mkHReg(16, HRcVec128, False); }HReg hregPPC_VR17 ( void ) { return mkHReg(17, HRcVec128, False); }HReg hregPPC_VR18 ( void ) { return mkHReg(18, HRcVec128, False); }HReg hregPPC_VR19 ( void ) { return mkHReg(19, HRcVec128, False); }HReg hregPPC_VR20 ( void ) { return mkHReg(20, HRcVec128, False); }HReg hregPPC_VR21 ( void ) { return mkHReg(21, HRcVec128, False); }HReg hregPPC_VR22 ( void ) { return mkHReg(22, HRcVec128, False); }HReg hregPPC_VR23 ( void ) { return mkHReg(23, HRcVec128, False); }HReg hregPPC_VR24 ( void ) { return mkHReg(24, HRcVec128, False); }HReg hregPPC_VR25 ( void ) { return mkHReg(25, HRcVec128, False); }HReg hregPPC_VR26 ( void ) { return mkHReg(26, HRcVec128, False); }HReg hregPPC_VR27 ( void ) { return mkHReg(27, HRcVec128, False); }HReg hregPPC_VR28 ( void ) { return mkHReg(28, HRcVec128, False); }HReg hregPPC_VR29 ( void ) { return mkHReg(29, HRcVec128, False); }HReg hregPPC_VR30 ( void ) { return mkHReg(30, HRcVec128, False); }HReg hregPPC_VR31 ( void ) { return mkHReg(31, HRcVec128, False); }void getAllocableRegs_PPC ( Int* nregs, HReg** arr, Bool mode64 ){ UInt i=0; if (mode64) *nregs = (32-9) + (32-24) + (32-24); else *nregs = (32-7) + (32-24) + (32-24); *arr = LibVEX_Alloc(*nregs * sizeof(HReg)); // GPR0 = scratch reg where poss. - some ops interpret as value zero // GPR1 = stack pointer // GPR2 = TOC pointer (*arr)[i++] = hregPPC_GPR3(mode64); (*arr)[i++] = hregPPC_GPR4(mode64); (*arr)[i++] = hregPPC_GPR5(mode64); (*arr)[i++] = hregPPC_GPR6(mode64); (*arr)[i++] = hregPPC_GPR7(mode64); (*arr)[i++] = hregPPC_GPR8(mode64); (*arr)[i++] = hregPPC_GPR9(mode64); (*arr)[i++] = hregPPC_GPR10(mode64); if (!mode64) { /* in mode64: r11 used for calls by ptr / env ptr for some langs r12 used for exception handling and global linkage code */ (*arr)[i++] = hregPPC_GPR11(mode64); (*arr)[i++] = hregPPC_GPR12(mode64); } // GPR13 = thread specific pointer // GPR14 and above are callee save. Yay. (*arr)[i++] = hregPPC_GPR14(mode64); (*arr)[i++] = hregPPC_GPR15(mode64); (*arr)[i++] = hregPPC_GPR16(mode64); (*arr)[i++] = hregPPC_GPR17(mode64); (*arr)[i++] = hregPPC_GPR18(mode64); (*arr)[i++] = hregPPC_GPR19(mode64); (*arr)[i++] = hregPPC_GPR20(mode64); (*arr)[i++] = hregPPC_GPR21(mode64); (*arr)[i++] = hregPPC_GPR22(mode64); (*arr)[i++] = hregPPC_GPR23(mode64); (*arr)[i++] = hregPPC_GPR24(mode64); (*arr)[i++] = hregPPC_GPR25(mode64); (*arr)[i++] = hregPPC_GPR26(mode64); (*arr)[i++] = hregPPC_GPR27(mode64); (*arr)[i++] = hregPPC_GPR28(mode64); // GPR29 is reserved for the dispatcher // GPR30 is reserved as AltiVec spill reg temporary // GPR31 is reserved for the GuestStatePtr /* Don't waste the reg-allocs's time trawling through zillions of FP registers - they mostly will never be used. We'll tolerate the occasional extra spill instead. */ /* For both ppc32-linux and ppc64-linux, f14-f31 are callee save. So use them. */ (*arr)[i++] = hregPPC_FPR14(); (*arr)[i++] = hregPPC_FPR15(); (*arr)[i++] = hregPPC_FPR16(); (*arr)[i++] = hregPPC_FPR17(); (*arr)[i++] = hregPPC_FPR18(); (*arr)[i++] = hregPPC_FPR19(); (*arr)[i++] = hregPPC_FPR20(); (*arr)[i++] = hregPPC_FPR21(); /* Same deal re Altivec */ /* For both ppc32-linux and ppc64-linux, v20-v31 are callee save. So use them. */ /* NB, vr29 is used as a scratch temporary -- do not allocate */ (*arr)[i++] = hregPPC_VR20(); (*arr)[i++] = hregPPC_VR21(); (*arr)[i++] = hregPPC_VR22(); (*arr)[i++] = hregPPC_VR23(); (*arr)[i++] = hregPPC_VR24(); (*arr)[i++] = hregPPC_VR25(); (*arr)[i++] = hregPPC_VR26(); (*arr)[i++] = hregPPC_VR27(); vassert(i == *nregs);}/* --------- Condition codes, Intel encoding. --------- */HChar* showPPCCondCode ( PPCCondCode cond ){ if (cond.test == Pct_ALWAYS) return "always"; switch (cond.flag) { case Pcf_7SO: return (cond.test == Pct_TRUE) ? "cr7.so=1" : "cr7.so=0"; case Pcf_7EQ: return (cond.test == Pct_TRUE) ? "cr7.eq=1" : "cr7.eq=0"; case Pcf_7GT: return (cond.test == Pct_TRUE) ? "cr7.gt=1" : "cr7.gt=0"; case Pcf_7LT: return (cond.test == Pct_TRUE) ? "cr7.lt=1" : "cr7.lt=0"; default: vpanic("ppPPCCondCode"); }}/* construct condition code */PPCCondCode mk_PPCCondCode ( PPCCondTest test, PPCCondFlag flag ){ PPCCondCode cc; cc.flag = flag; cc.test = test; return cc;}/* false->true, true->false */PPCCondTest invertCondTest ( PPCCondTest ct ){ vassert(ct != Pct_ALWAYS); return (ct == Pct_TRUE) ? Pct_FALSE : Pct_TRUE;}/* --------- PPCAMode: memory address expressions. --------- */PPCAMode* PPCAMode_IR ( Int idx, HReg base ) { PPCAMode* am = LibVEX_Alloc(sizeof(PPCAMode)); vassert(idx >= -0x8000 && idx < 0x8000); am->tag = Pam_IR; am->Pam.IR.base = base; am->Pam.IR.index = idx; return am;}PPCAMode* PPCAMode_RR ( HReg idx, HReg base ) { PPCAMode* am = LibVEX_Alloc(sizeof(PPCAMode)); am->tag = Pam_RR; am->Pam.RR.base = base; am->Pam.RR.index = idx; return am;}PPCAMode* dopyPPCAMode ( PPCAMode* am ) { switch (am->tag) { case Pam_IR: return PPCAMode_IR( am->Pam.IR.index, am->Pam.IR.base ); case Pam_RR: return PPCAMode_RR( am->Pam.RR.index, am->Pam.RR.base ); default: vpanic("dopyPPCAMode"); }}void ppPPCAMode ( PPCAMode* am ) { switch (am->tag) { case Pam_IR: if (am->Pam.IR.index == 0) vex_printf("0("); else vex_printf("%d(", (Int)am->Pam.IR.index); ppHRegPPC(am->Pam.IR.base); vex_printf(")"); return; case Pam_RR: ppHRegPPC(am->Pam.RR.base); vex_printf(","); ppHRegPPC(am->Pam.RR.index); return; default: vpanic("ppPPCAMode"); }}static void addRegUsage_PPCAMode ( HRegUsage* u, PPCAMode* am ) { switch (am->tag) { case Pam_IR: addHRegUse(u, HRmRead, am->Pam.IR.base); return; case Pam_RR: addHRegUse(u, HRmRead, am->Pam.RR.base); addHRegUse(u, HRmRead, am->Pam.RR.index); return; default: vpanic("addRegUsage_PPCAMode"); }}static void mapRegs_PPCAMode ( HRegRemap* m, PPCAMode* am ) { switch (am->tag) { case Pam_IR: am->Pam.IR.base = lookupHRegRemap(m, am->Pam.IR.base); return; case Pam_RR: am->Pam.RR.base = lookupHRegRemap(m, am->Pam.RR.base); am->Pam.RR.index = lookupHRegRemap(m, am->Pam.RR.index); return; default:
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -