mirror of
https://github.com/PCSX2/pcsx2.git
synced 2025-04-02 10:52:54 -04:00
git-svn-id: http://pcsx2.googlecode.com/svn/branches/pcsx2_0.9.4@186 96395faa-99c1-11dd-bbfe-3dabce05a288
516 lines
12 KiB
C++
516 lines
12 KiB
C++
/* Pcsx2 - Pc Ps2 Emulator
|
|
* Copyright (C) 2002-2007 Pcsx2 Team
|
|
*
|
|
* 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
|
|
*/
|
|
|
|
// stop compiling if NORECBUILD build (only for Visual Studio)
|
|
#if !(defined(_MSC_VER) && defined(PCSX2_NORECBUILD))
|
|
|
|
#include "PS2Etypes.h"
|
|
|
|
#if defined(_WIN32)
|
|
#include <windows.h>
|
|
#endif
|
|
|
|
#include <assert.h>
|
|
|
|
#include <vector>
|
|
#include <list>
|
|
|
|
using namespace std;
|
|
|
|
extern "C" {
|
|
#include "zlib.h"
|
|
#include "Elfheader.h"
|
|
#include "Misc.h"
|
|
#include "System.h"
|
|
#include "R5900.h"
|
|
#include "Vif.h"
|
|
#include "VU.h"
|
|
#include "VifDma.h"
|
|
#include "Memory.h"
|
|
#include "Hw.h"
|
|
|
|
#include "ix86/ix86.h"
|
|
#include "iR5900.h"
|
|
|
|
#include "Counters.h"
|
|
#include "GS.h"
|
|
|
|
extern u32 CSRw;
|
|
|
|
}
|
|
|
|
void CSRwrite(u32 value);
|
|
|
|
#ifdef PCSX2_VIRTUAL_MEM
|
|
#define PS2GS_BASE(mem) ((PS2MEM_BASE+0x12000000)+(mem&0x13ff))
|
|
#else
|
|
extern u8 g_RealGSMem[0x2000];
|
|
#define PS2GS_BASE(mem) (g_RealGSMem+(mem&0x13ff))
|
|
#endif
|
|
|
|
void gsConstWrite8(u32 mem, int mmreg)
|
|
{
|
|
switch (mem&~3) {
|
|
case 0x12001000: // GS_CSR
|
|
_eeMoveMMREGtoR(EAX, mmreg);
|
|
iFlushCall(0);
|
|
MOV32MtoR(ECX, (uptr)&CSRw);
|
|
AND32ItoR(EAX, 0xff<<(mem&3)*8);
|
|
AND32ItoR(ECX, ~(0xff<<(mem&3)*8));
|
|
OR32ItoR(EAX, ECX);
|
|
_callFunctionArg1((uptr)CSRwrite, EAX|MEM_X86TAG, 0);
|
|
break;
|
|
default:
|
|
_eeWriteConstMem8( (uptr)PS2GS_BASE(mem), mmreg );
|
|
|
|
if( CHECK_MULTIGS ) {
|
|
iFlushCall(0);
|
|
|
|
_callFunctionArg3((uptr)GSRingBufSimplePacket, MEM_CONSTTAG, MEM_CONSTTAG, mmreg,
|
|
GS_RINGTYPE_MEMWRITE8, mem&0x13ff, 0);
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
void recSetSMODE1()
|
|
{
|
|
iFlushCall(0);
|
|
AND32ItoR(EAX, 0x6000);
|
|
CMP32ItoR(EAX, 0x6000);
|
|
j8Ptr[5] = JNE8(0);
|
|
|
|
// PAL
|
|
OR32ItoM( (uptr)&Config.PsxType, 1);
|
|
j8Ptr[6] = JMP8(0);
|
|
|
|
x86SetJ8( j8Ptr[5] );
|
|
|
|
// NTSC
|
|
AND32ItoM( (uptr)&Config.PsxType, ~1 );
|
|
|
|
x86SetJ8( j8Ptr[6] );
|
|
CALLFunc((uptr)UpdateVSyncRate);
|
|
}
|
|
|
|
void recSetSMODE2()
|
|
{
|
|
TEST32ItoR(EAX, 1);
|
|
j8Ptr[5] = JZ8(0);
|
|
|
|
// Interlaced
|
|
OR32ItoM( (uptr)&Config.PsxType, 2);
|
|
j8Ptr[6] = JMP8(0);
|
|
|
|
x86SetJ8( j8Ptr[5] );
|
|
|
|
// Non-Interlaced
|
|
AND32ItoM( (uptr)&Config.PsxType, ~2 );
|
|
|
|
x86SetJ8( j8Ptr[6] );
|
|
}
|
|
|
|
void gsConstWrite16(u32 mem, int mmreg)
|
|
{
|
|
switch (mem&~3) {
|
|
case 0x12000010: // GS_SMODE1
|
|
assert( !(mem&3));
|
|
_eeMoveMMREGtoR(EAX, mmreg);
|
|
_eeWriteConstMem16( (uptr)PS2GS_BASE(mem), mmreg );
|
|
|
|
if( CHECK_MULTIGS )
|
|
_callPushArg(MEM_X86TAG|EAX, 0, X86ARG3);
|
|
|
|
recSetSMODE1();
|
|
|
|
if( CHECK_MULTIGS ) {
|
|
iFlushCall(0);
|
|
|
|
_callPushArg(MEM_CONSTTAG, mem&0x13ff, X86ARG2);
|
|
_callPushArg(MEM_CONSTTAG, GS_RINGTYPE_MEMWRITE16, X86ARG1);
|
|
CALLFunc((uptr)GSRingBufSimplePacket);
|
|
#ifndef __x86_64__
|
|
ADD32ItoR(ESP, 12);
|
|
#endif
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 0x12000020: // GS_SMODE2
|
|
assert( !(mem&3));
|
|
_eeMoveMMREGtoR(EAX, mmreg);
|
|
_eeWriteConstMem16( (uptr)PS2GS_BASE(mem), mmreg );
|
|
|
|
if( CHECK_MULTIGS )
|
|
_callPushArg(MEM_X86TAG|EAX, 0, X86ARG3);
|
|
|
|
recSetSMODE2();
|
|
|
|
if( CHECK_MULTIGS ) {
|
|
iFlushCall(0);
|
|
|
|
_callPushArg(MEM_CONSTTAG, mem&0x13ff, X86ARG2);
|
|
_callPushArg(MEM_CONSTTAG, GS_RINGTYPE_MEMWRITE16, X86ARG1);
|
|
CALLFunc((uptr)GSRingBufSimplePacket);
|
|
#ifndef __x86_64__
|
|
ADD32ItoR(ESP, 12);
|
|
#endif
|
|
}
|
|
|
|
break;
|
|
|
|
case 0x12001000: // GS_CSR
|
|
|
|
assert( !(mem&2) );
|
|
_eeMoveMMREGtoR(EAX, mmreg);
|
|
iFlushCall(0);
|
|
|
|
MOV32MtoR(ECX, (uptr)&CSRw);
|
|
AND32ItoR(EAX, 0xffff<<(mem&2)*8);
|
|
AND32ItoR(ECX, ~(0xffff<<(mem&2)*8));
|
|
OR32ItoR(EAX, ECX);
|
|
_callFunctionArg1((uptr)CSRwrite, EAX|MEM_X86TAG, 0);
|
|
break;
|
|
|
|
default:
|
|
_eeWriteConstMem16( (uptr)PS2GS_BASE(mem), mmreg );
|
|
|
|
if( CHECK_MULTIGS ) {
|
|
iFlushCall(0);
|
|
_callFunctionArg3((uptr)GSRingBufSimplePacket, MEM_CONSTTAG, MEM_CONSTTAG, mmreg,
|
|
GS_RINGTYPE_MEMWRITE16, mem&0x13ff, 0);
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
// (value&0x1f00)|0x6000
|
|
void gsConstWriteIMR(int mmreg)
|
|
{
|
|
const u32 mem = 0x12001010;
|
|
if( mmreg & MEM_XMMTAG ) {
|
|
SSE2_MOVD_XMM_to_M32((uptr)PS2GS_BASE(mem), mmreg&0xf);
|
|
AND32ItoM((uptr)PS2GS_BASE(mem), 0x1f00);
|
|
OR32ItoM((uptr)PS2GS_BASE(mem), 0x6000);
|
|
}
|
|
#ifndef __x86_64__
|
|
else if( mmreg & MEM_MMXTAG ) {
|
|
SetMMXstate();
|
|
MOVDMMXtoM((uptr)PS2GS_BASE(mem), mmreg&0xf);
|
|
AND32ItoM((uptr)PS2GS_BASE(mem), 0x1f00);
|
|
OR32ItoM((uptr)PS2GS_BASE(mem), 0x6000);
|
|
}
|
|
#endif
|
|
else if( mmreg & MEM_EECONSTTAG ) {
|
|
MOV32ItoM( (uptr)PS2GS_BASE(mem), (g_cpuConstRegs[(mmreg>>16)&0x1f].UL[0]&0x1f00)|0x6000);
|
|
}
|
|
else {
|
|
AND32ItoR(mmreg, 0x1f00);
|
|
OR32ItoR(mmreg, 0x6000);
|
|
MOV32RtoM( (uptr)PS2GS_BASE(mem), mmreg );
|
|
}
|
|
|
|
// IMR doesn't need to be updated in MTGS mode
|
|
}
|
|
|
|
void gsConstWrite32(u32 mem, int mmreg) {
|
|
|
|
switch (mem) {
|
|
|
|
case 0x12000010: // GS_SMODE1
|
|
_eeMoveMMREGtoR(EAX, mmreg);
|
|
_eeWriteConstMem32( (uptr)PS2GS_BASE(mem), mmreg );
|
|
|
|
if( CHECK_MULTIGS )
|
|
_callPushArg(MEM_X86TAG|EAX, 0, X86ARG3);
|
|
|
|
recSetSMODE1();
|
|
|
|
if( CHECK_MULTIGS ) {
|
|
iFlushCall(0);
|
|
|
|
_callPushArg(MEM_CONSTTAG, mem&0x13ff, X86ARG2);
|
|
_callPushArg(MEM_CONSTTAG, GS_RINGTYPE_MEMWRITE32, X86ARG1);
|
|
CALLFunc((uptr)GSRingBufSimplePacket);
|
|
#ifndef __x86_64__
|
|
ADD32ItoR(ESP, 12);
|
|
#endif
|
|
}
|
|
|
|
break;
|
|
|
|
case 0x12000020: // GS_SMODE2
|
|
_eeMoveMMREGtoR(EAX, mmreg);
|
|
_eeWriteConstMem32( (uptr)PS2GS_BASE(mem), mmreg );
|
|
|
|
if( CHECK_MULTIGS )
|
|
_callPushArg(MEM_X86TAG|EAX, 0, X86ARG3);
|
|
|
|
recSetSMODE2();
|
|
|
|
if( CHECK_MULTIGS ) {
|
|
iFlushCall(0);
|
|
|
|
_callPushArg(MEM_CONSTTAG, mem&0x13ff, X86ARG2);
|
|
_callPushArg(MEM_CONSTTAG, GS_RINGTYPE_MEMWRITE32, X86ARG1);
|
|
CALLFunc((uptr)GSRingBufSimplePacket);
|
|
#ifndef __x86_64__
|
|
ADD32ItoR(ESP, 12);
|
|
#endif
|
|
}
|
|
|
|
break;
|
|
|
|
case 0x12001000: // GS_CSR
|
|
iFlushCall(0);
|
|
_callFunctionArg1((uptr)CSRwrite, mmreg, 0);
|
|
break;
|
|
|
|
case 0x12001010: // GS_IMR
|
|
gsConstWriteIMR(mmreg);
|
|
break;
|
|
default:
|
|
_eeWriteConstMem32( (uptr)PS2GS_BASE(mem), mmreg );
|
|
|
|
if( CHECK_MULTIGS ) {
|
|
iFlushCall(0);
|
|
|
|
_callFunctionArg3((uptr)GSRingBufSimplePacket, MEM_CONSTTAG, MEM_CONSTTAG, mmreg,
|
|
GS_RINGTYPE_MEMWRITE32, mem&0x13ff, 0);
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
void gsConstWrite64(u32 mem, int mmreg)
|
|
{
|
|
switch (mem) {
|
|
case 0x12000010: // GS_SMODE1
|
|
_eeMoveMMREGtoR(EAX, mmreg);
|
|
_eeWriteConstMem64((uptr)PS2GS_BASE(mem), mmreg);
|
|
|
|
if( CHECK_MULTIGS )
|
|
_callPushArg(MEM_X86TAG|EAX, 0, X86ARG3);
|
|
|
|
recSetSMODE1();
|
|
|
|
if( CHECK_MULTIGS ) {
|
|
iFlushCall(0);
|
|
|
|
_callPushArg(MEM_CONSTTAG, mem&0x13ff, X86ARG2);
|
|
_callPushArg(MEM_CONSTTAG, GS_RINGTYPE_MEMWRITE32, X86ARG1);
|
|
CALLFunc((uptr)GSRingBufSimplePacket);
|
|
#ifndef __x86_64__
|
|
ADD32ItoR(ESP, 12);
|
|
#endif
|
|
}
|
|
|
|
break;
|
|
|
|
case 0x12000020: // GS_SMODE2
|
|
_eeMoveMMREGtoR(EAX, mmreg);
|
|
_eeWriteConstMem64((uptr)PS2GS_BASE(mem), mmreg);
|
|
|
|
if( CHECK_MULTIGS )
|
|
_callPushArg(MEM_X86TAG|EAX, 0, X86ARG3);
|
|
|
|
recSetSMODE2();
|
|
|
|
if( CHECK_MULTIGS ) {
|
|
iFlushCall(0);
|
|
|
|
_callPushArg(MEM_CONSTTAG, mem&0x13ff, X86ARG2);
|
|
_callPushArg(MEM_CONSTTAG, GS_RINGTYPE_MEMWRITE32, X86ARG1);
|
|
CALLFunc((uptr)GSRingBufSimplePacket);
|
|
#ifndef __x86_64__
|
|
ADD32ItoR(ESP, 12);
|
|
#endif
|
|
}
|
|
|
|
break;
|
|
|
|
case 0x12001000: // GS_CSR
|
|
iFlushCall(0);
|
|
_callFunctionArg1((uptr)CSRwrite, mmreg, 0);
|
|
break;
|
|
|
|
case 0x12001010: // GS_IMR
|
|
gsConstWriteIMR(mmreg);
|
|
break;
|
|
|
|
default:
|
|
_eeWriteConstMem64((uptr)PS2GS_BASE(mem), mmreg);
|
|
|
|
if( CHECK_MULTIGS ) {
|
|
iFlushCall(0);
|
|
|
|
_callPushArg(MEM_MEMORYTAG, (uptr)PS2GS_BASE(mem)+4, X86ARG4);
|
|
_callPushArg(MEM_MEMORYTAG, (uptr)PS2GS_BASE(mem), X86ARG3);
|
|
_callPushArg(MEM_CONSTTAG, mem&0x13ff, X86ARG2);
|
|
_callPushArg(MEM_CONSTTAG, GS_RINGTYPE_MEMWRITE64, X86ARG1);
|
|
CALLFunc((uptr)GSRingBufSimplePacket);
|
|
#ifndef __x86_64__
|
|
ADD32ItoR(ESP, 16);
|
|
#endif
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
void gsConstWrite128(u32 mem, int mmreg)
|
|
{
|
|
switch (mem) {
|
|
case 0x12000010: // GS_SMODE1
|
|
_eeMoveMMREGtoR(EAX, mmreg);
|
|
_eeWriteConstMem128( (uptr)PS2GS_BASE(mem), mmreg);
|
|
|
|
if( CHECK_MULTIGS )
|
|
_callPushArg(MEM_X86TAG|EAX, 0, X86ARG3);
|
|
|
|
recSetSMODE1();
|
|
|
|
if( CHECK_MULTIGS ) {
|
|
iFlushCall(0);
|
|
|
|
_callPushArg(MEM_CONSTTAG, mem&0x13ff, X86ARG2);
|
|
_callPushArg(MEM_CONSTTAG, GS_RINGTYPE_MEMWRITE32, X86ARG1);
|
|
CALLFunc((uptr)GSRingBufSimplePacket);
|
|
#ifndef __x86_64__
|
|
ADD32ItoR(ESP, 12);
|
|
#endif
|
|
}
|
|
|
|
break;
|
|
|
|
case 0x12000020: // GS_SMODE2
|
|
_eeMoveMMREGtoR(EAX, mmreg);
|
|
_eeWriteConstMem128( (uptr)PS2GS_BASE(mem), mmreg);
|
|
|
|
if( CHECK_MULTIGS )
|
|
_callPushArg(MEM_X86TAG|EAX, 0, X86ARG3);
|
|
|
|
recSetSMODE2();
|
|
|
|
if( CHECK_MULTIGS ) {
|
|
iFlushCall(0);
|
|
|
|
_callPushArg(MEM_CONSTTAG, mem&0x13ff, X86ARG2);
|
|
_callPushArg(MEM_CONSTTAG, GS_RINGTYPE_MEMWRITE32, X86ARG1);
|
|
CALLFunc((uptr)GSRingBufSimplePacket);
|
|
#ifndef __x86_64__
|
|
ADD32ItoR(ESP, 12);
|
|
#endif
|
|
}
|
|
|
|
break;
|
|
|
|
case 0x12001000: // GS_CSR
|
|
iFlushCall(0);
|
|
_callFunctionArg1((uptr)CSRwrite, mmreg, 0);
|
|
break;
|
|
|
|
case 0x12001010: // GS_IMR
|
|
// (value&0x1f00)|0x6000
|
|
gsConstWriteIMR(mmreg);
|
|
break;
|
|
|
|
default:
|
|
_eeWriteConstMem128( (uptr)PS2GS_BASE(mem), mmreg);
|
|
|
|
if( CHECK_MULTIGS ) {
|
|
iFlushCall(0);
|
|
|
|
_callPushArg(MEM_MEMORYTAG, (uptr)PS2GS_BASE(mem)+4, X86ARG4);
|
|
_callPushArg(MEM_MEMORYTAG, (uptr)PS2GS_BASE(mem), X86ARG3);
|
|
_callPushArg(MEM_CONSTTAG, mem&0x13ff, X86ARG2);
|
|
_callPushArg(MEM_CONSTTAG, GS_RINGTYPE_MEMWRITE64, X86ARG1);
|
|
CALLFunc((uptr)GSRingBufSimplePacket);
|
|
#ifndef __x86_64__
|
|
ADD32ItoR(ESP, 16);
|
|
#endif
|
|
|
|
_callPushArg(MEM_MEMORYTAG, (uptr)PS2GS_BASE(mem)+12, X86ARG4);
|
|
_callPushArg(MEM_MEMORYTAG, (uptr)PS2GS_BASE(mem)+8, X86ARG3);
|
|
_callPushArg(MEM_CONSTTAG, (mem&0x13ff)+8, X86ARG2);
|
|
_callPushArg(MEM_CONSTTAG, GS_RINGTYPE_MEMWRITE64, X86ARG1);
|
|
CALLFunc((uptr)GSRingBufSimplePacket);
|
|
#ifndef __x86_64__
|
|
ADD32ItoR(ESP, 16);
|
|
#endif
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
int gsConstRead8(u32 x86reg, u32 mem, u32 sign)
|
|
{
|
|
#ifdef GIF_LOG
|
|
GIF_LOG("GS read 8 %8.8lx (%8.8x), at %8.8lx\n", (uptr)PS2GS_BASE(mem), mem);
|
|
#endif
|
|
_eeReadConstMem8(x86reg, (uptr)PS2GS_BASE(mem), sign);
|
|
return 0;
|
|
}
|
|
|
|
int gsConstRead16(u32 x86reg, u32 mem, u32 sign)
|
|
{
|
|
#ifdef GIF_LOG
|
|
GIF_LOG("GS read 16 %8.8lx (%8.8x), at %8.8lx\n", (uptr)PS2GS_BASE(mem), mem);
|
|
#endif
|
|
_eeReadConstMem16(x86reg, (uptr)PS2GS_BASE(mem), sign);
|
|
return 0;
|
|
}
|
|
|
|
int gsConstRead32(u32 x86reg, u32 mem)
|
|
{
|
|
#ifdef GIF_LOG
|
|
GIF_LOG("GS read 32 %8.8lx (%8.8x), at %8.8lx\n", (uptr)PS2GS_BASE(mem), mem);
|
|
#endif
|
|
_eeReadConstMem32(x86reg, (uptr)PS2GS_BASE(mem));
|
|
return 0;
|
|
}
|
|
|
|
void gsConstRead64(u32 mem, int mmreg)
|
|
{
|
|
#ifdef GIF_LOG
|
|
GIF_LOG("GS read 64 %8.8lx (%8.8x), at %8.8lx\n", (uptr)PS2GS_BASE(mem), mem);
|
|
#endif
|
|
if( IS_XMMREG(mmreg) ) SSE_MOVLPS_M64_to_XMM(mmreg&0xff, (uptr)PS2GS_BASE(mem));
|
|
else {
|
|
#ifndef __x86_64__
|
|
MOVQMtoR(mmreg, (uptr)PS2GS_BASE(mem));
|
|
SetMMXstate();
|
|
#else
|
|
assert(0);
|
|
#endif
|
|
}
|
|
}
|
|
|
|
void gsConstRead128(u32 mem, int xmmreg)
|
|
{
|
|
#ifdef GIF_LOG
|
|
GIF_LOG("GS read 128 %8.8lx (%8.8x), at %8.8lx\n", (uptr)PS2GS_BASE(mem), mem);
|
|
#endif
|
|
_eeReadConstMem128( xmmreg, (uptr)PS2GS_BASE(mem));
|
|
}
|
|
|
|
#endif // PCSX2_NORECBUILD
|