2012-11-01 16:19:01 +01:00
// Copyright (c) 2012- PPSSPP Project.
// 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
2012-11-04 23:01:49 +01:00
// the Free Software Foundation, version 2.0 or later versions.
2012-11-01 16:19:01 +01:00
// 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 2.0 for more details.
// A copy of the GPL 2.0 should have been included with the program.
// If not, see http://www.gnu.org/licenses/
// Official git repository and contact information can be found at
// https://github.com/hrydgard/ppsspp and http://www.ppsspp.org/.
# pragma once
# include "../../Globals.h"
# include "../CPU.h"
enum
{
MIPS_REG_ZERO = 0 ,
MIPS_REG_COMPILER_SCRATCH = 1 ,
MIPS_REG_V0 = 2 ,
MIPS_REG_V1 = 3 ,
MIPS_REG_A0 = 4 ,
MIPS_REG_A1 = 5 ,
MIPS_REG_A2 = 6 ,
MIPS_REG_A3 = 7 ,
MIPS_REG_A4 = 8 , // Seems to be N32 register calling convention - there are 8 args instead of 4.
MIPS_REG_A5 = 9 ,
MIPS_REG_S0 = 16 ,
MIPS_REG_S1 = 17 ,
MIPS_REG_S2 = 18 ,
MIPS_REG_S3 = 19 ,
MIPS_REG_S4 = 20 ,
MIPS_REG_S5 = 21 ,
MIPS_REG_S6 = 22 ,
MIPS_REG_S7 = 23 ,
MIPS_REG_K0 = 26 ,
MIPS_REG_K1 = 27 ,
MIPS_REG_GP = 28 ,
MIPS_REG_SP = 29 ,
MIPS_REG_FP = 30 ,
MIPS_REG_RA = 31 ,
2012-11-07 15:44:48 +01:00
// ID for mipscall "callback" is stored here - from JPCSP
MIPS_REG_CALL_ID = MIPS_REG_S0 ,
2012-11-01 16:19:01 +01:00
} ;
enum
{
VFPU_CTRL_SPREFIX ,
VFPU_CTRL_TPREFIX ,
VFPU_CTRL_DPREFIX ,
VFPU_CTRL_CC ,
VFPU_CTRL_INF4 ,
VFPU_CTRL_RSV5 ,
VFPU_CTRL_RSV6 ,
VFPU_CTRL_REV ,
VFPU_CTRL_RCX0 ,
VFPU_CTRL_RCX1 ,
VFPU_CTRL_RCX2 ,
VFPU_CTRL_RCX3 ,
VFPU_CTRL_RCX4 ,
VFPU_CTRL_RCX5 ,
VFPU_CTRL_RCX6 ,
VFPU_CTRL_RCX7 ,
2012-11-22 20:14:24 +01:00
VFPU_CTRL_MAX ,
2012-11-01 16:19:01 +01:00
//unknown....
} ;
// George Marsaglia-style random number generator.
class GMRng {
public :
2013-02-10 16:36:06 +01:00
GMRng ( ) {
m_w = 0x23E866ED ;
m_z = 0x80FD5AF2 ;
}
2012-11-01 16:19:01 +01:00
void Init ( int seed ) {
m_w = seed ^ ( seed < < 16 ) ;
if ( ! m_w ) m_w = 1337 ;
m_z = ~ seed ;
if ( ! m_z ) m_z = 31337 ;
}
u32 R32 ( ) {
m_z = 36969 * ( m_z & 65535 ) + ( m_z > > 16 ) ;
m_w = 18000 * ( m_w & 65535 ) + ( m_w > > 16 ) ;
return ( m_z < < 16 ) + m_w ;
}
2013-02-10 16:36:06 +01:00
float F ( ) {
return ( float ) R32 ( ) / ( float ) ( 0xFFFFFFFF ) ;
}
2012-11-01 16:19:01 +01:00
2013-02-03 20:31:46 -08:00
void DoState ( PointerWrap & p ) ;
2012-12-27 20:33:10 -08:00
2012-11-01 16:19:01 +01:00
private :
u32 m_w ;
u32 m_z ;
} ;
2013-01-08 14:20:06 +01:00
class MIPSState
2012-11-01 16:19:01 +01:00
{
public :
MIPSState ( ) ;
~ MIPSState ( ) ;
void Reset ( ) ;
2012-12-27 20:33:10 -08:00
void DoState ( PointerWrap & p ) ;
2012-11-01 16:19:01 +01:00
2013-01-09 11:20:48 +01:00
// MUST start with r!
2012-11-01 16:19:01 +01:00
u32 r [ 32 ] ;
float f [ 32 ] ;
float v [ 128 ] ;
u32 vfpuCtrl [ 16 ] ;
u32 pc ;
u32 nextPC ;
2013-01-17 01:14:49 -08:00
int downcount ; // This really doesn't belong here, it belongs in CoreTiming. But you gotta do what you gotta do, this needs to be reachable in the ARM JIT.
2013-01-12 00:44:18 +01:00
2012-11-01 16:19:01 +01:00
u32 hi ;
u32 lo ;
u32 fcr0 ;
u32 fcr31 ; //fpu control register
2013-01-07 22:33:09 +01:00
u32 fpcond ; // cache the cond flag of fcr31 (& 1 << 23)
2012-11-01 16:19:01 +01:00
bool inDelaySlot ;
2012-11-07 17:34:25 +01:00
int llBit ; // ll/sc
2012-11-01 16:19:01 +01:00
2013-01-12 00:44:18 +01:00
GMRng rng ; // VFPU hardware random number generator. Probably not the right type.
2012-11-01 16:19:01 +01:00
// Debug stuff
u32 debugCount ; // can be used to count basic blocks before crashes, etc.
void WriteFCR ( int reg , int value ) ;
u32 ReadFCR ( int reg ) ;
2013-02-15 22:56:38 +01:00
u8 VfpuWriteMask ( ) const {
return ( vfpuCtrl [ VFPU_CTRL_DPREFIX ] > > 8 ) & 0xF ;
}
bool VfpuWriteMask ( int i ) const {
return ( vfpuCtrl [ VFPU_CTRL_DPREFIX ] > > ( 8 + i ) ) & 1 ;
}
2012-11-01 16:19:01 +01:00
void Irq ( ) ;
void SWI ( ) ;
void Abort ( ) ;
void SingleStep ( ) ;
2012-11-19 14:16:37 +01:00
int RunLoopUntil ( u64 globalTicks ) ;
2012-11-01 16:19:01 +01:00
} ;
class MIPSDebugInterface ;
//The one we are compiling or running currently
extern MIPSState * currentMIPS ;
extern MIPSDebugInterface * currentDebugMIPS ;
extern MIPSState mipsr4k ;
void MIPS_Init ( ) ;
int MIPS_SingleStep ( ) ;
void MIPS_Shutdown ( ) ;
void MIPS_Irq ( ) ;
void MIPS_SWI ( ) ;