/* armvfp.c - ARM VFPv3 emulation unit Copyright (C) 2003 Skyeye Develop Group for help please send mail to 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 of the License, 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ /* Note: this file handles interface with arm core and vfp registers */ /* Opens debug for classic interpreter only */ //#define DEBUG #include "common/common.h" #include "core/arm/skyeye_common/armdefs.h" #include "core/arm/skyeye_common/vfp/vfp.h" #define DEBUG DBG //ARMul_State* persistent_state; /* function calls from SoftFloat lib don't have an access to ARMul_state. */ unsigned VFPInit (ARMul_State *state) { state->VFP[VFP_OFFSET(VFP_FPSID)] = VFP_FPSID_IMPLMEN<<24 | VFP_FPSID_SW<<23 | VFP_FPSID_SUBARCH<<16 | VFP_FPSID_PARTNUM<<8 | VFP_FPSID_VARIANT<<4 | VFP_FPSID_REVISION; state->VFP[VFP_OFFSET(VFP_FPEXC)] = 0; state->VFP[VFP_OFFSET(VFP_FPSCR)] = 0; //persistent_state = state; /* Reset only specify VFP_FPEXC_EN = '0' */ return 0; } unsigned VFPMRC (ARMul_State * state, unsigned type, u32 instr, u32 * value) { /* MRC ,,,,{,} */ int CoProc = BITS (8, 11); /* 10 or 11 */ int OPC_1 = BITS (21, 23); int Rt = BITS (12, 15); int CRn = BITS (16, 19); int CRm = BITS (0, 3); int OPC_2 = BITS (5, 7); /* TODO check access permission */ /* CRn/opc1 CRm/opc2 */ if (CoProc == 10 || CoProc == 11) { #define VFP_MRC_TRANS #include "core/arm/skyeye_common/vfp/vfpinstr.cpp" #undef VFP_MRC_TRANS } DEBUG("Can't identify %x, CoProc %x, OPC_1 %x, Rt %x, CRn %x, CRm %x, OPC_2 %x\n", instr, CoProc, OPC_1, Rt, CRn, CRm, OPC_2); return ARMul_CANT; } unsigned VFPMCR (ARMul_State * state, unsigned type, u32 instr, u32 value) { /* MCR ,,,,{,} */ int CoProc = BITS (8, 11); /* 10 or 11 */ int OPC_1 = BITS (21, 23); int Rt = BITS (12, 15); int CRn = BITS (16, 19); int CRm = BITS (0, 3); int OPC_2 = BITS (5, 7); /* TODO check access permission */ /* CRn/opc1 CRm/opc2 */ if (CoProc == 10 || CoProc == 11) { #define VFP_MCR_TRANS #include "core/arm/skyeye_common/vfp/vfpinstr.cpp" #undef VFP_MCR_TRANS } DEBUG("Can't identify %x, CoProc %x, OPC_1 %x, Rt %x, CRn %x, CRm %x, OPC_2 %x\n", instr, CoProc, OPC_1, Rt, CRn, CRm, OPC_2); return ARMul_CANT; } unsigned VFPMRRC (ARMul_State * state, unsigned type, u32 instr, u32 * value1, u32 * value2) { /* MCRR ,,,, */ int CoProc = BITS (8, 11); /* 10 or 11 */ int OPC_1 = BITS (4, 7); int Rt = BITS (12, 15); int Rt2 = BITS (16, 19); int CRm = BITS (0, 3); if (CoProc == 10 || CoProc == 11) { #define VFP_MRRC_TRANS #include "core/arm/skyeye_common/vfp/vfpinstr.cpp" #undef VFP_MRRC_TRANS } DEBUG("Can't identify %x, CoProc %x, OPC_1 %x, Rt %x, Rt2 %x, CRm %x\n", instr, CoProc, OPC_1, Rt, Rt2, CRm); return ARMul_CANT; } unsigned VFPMCRR (ARMul_State * state, unsigned type, u32 instr, u32 value1, u32 value2) { /* MCRR ,,,, */ int CoProc = BITS (8, 11); /* 10 or 11 */ int OPC_1 = BITS (4, 7); int Rt = BITS (12, 15); int Rt2 = BITS (16, 19); int CRm = BITS (0, 3); /* TODO check access permission */ /* CRn/opc1 CRm/opc2 */ if (CoProc == 11 || CoProc == 10) { #define VFP_MCRR_TRANS #include "core/arm/skyeye_common/vfp/vfpinstr.cpp" #undef VFP_MCRR_TRANS } DEBUG("Can't identify %x, CoProc %x, OPC_1 %x, Rt %x, Rt2 %x, CRm %x\n", instr, CoProc, OPC_1, Rt, Rt2, CRm); return ARMul_CANT; } unsigned VFPSTC (ARMul_State * state, unsigned type, u32 instr, u32 * value) { /* STC{L} ,,[],