2024-07-30 11:42:36 +00:00
|
|
|
// SPDX-FileCopyrightText: 2002-2024 PCSX2 Dev Team
|
|
|
|
// SPDX-License-Identifier: GPL-3.0+
|
2009-09-08 12:08:10 +00:00
|
|
|
|
2009-02-09 21:15:56 +00:00
|
|
|
/* TODO
|
|
|
|
-Fix the flags Proper as they aren't handle now..
|
|
|
|
-Add BC Table opcodes
|
|
|
|
-Add Interlock in QMFC2,QMTC2,CFC2,CTC2
|
|
|
|
-Finish instruction set
|
|
|
|
-Bug Fixes!!!
|
|
|
|
*/
|
|
|
|
|
2009-11-14 12:02:56 +00:00
|
|
|
#include "Common.h"
|
2009-02-09 21:15:56 +00:00
|
|
|
|
|
|
|
#include <cmath>
|
|
|
|
|
|
|
|
#include "R5900OpcodeTables.h"
|
|
|
|
#include "VUmicro.h"
|
2012-04-07 12:27:19 +00:00
|
|
|
#include "Vif_Dma.h"
|
2021-10-23 19:36:02 +00:00
|
|
|
#include "MTVU.h"
|
2009-02-09 21:15:56 +00:00
|
|
|
|
|
|
|
#define _Ft_ _Rt_
|
|
|
|
#define _Fs_ _Rd_
|
|
|
|
#define _Fd_ _Sa_
|
|
|
|
|
|
|
|
#define _Fsf_ ((cpuRegs.code >> 21) & 0x03)
|
|
|
|
#define _Ftf_ ((cpuRegs.code >> 23) & 0x03)
|
|
|
|
|
|
|
|
using namespace R5900;
|
|
|
|
|
|
|
|
void COP2_BC2() { Int_COP2BC2PrintTable[_Rt_]();}
|
2023-11-03 17:19:47 +00:00
|
|
|
void COP2_SPECIAL() { _vu0FinishMicro(); Int_COP2SPECIAL1PrintTable[_Funct_]();}
|
2009-02-09 21:15:56 +00:00
|
|
|
|
|
|
|
void COP2_SPECIAL2() {
|
|
|
|
Int_COP2SPECIAL2PrintTable[(cpuRegs.code & 0x3) | ((cpuRegs.code >> 4) & 0x7c)]();
|
|
|
|
}
|
|
|
|
|
|
|
|
void COP2_Unknown()
|
|
|
|
{
|
2009-03-27 06:34:51 +00:00
|
|
|
CPU_LOG("Unknown COP2 opcode called");
|
2009-02-09 21:15:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//****************************************************************************
|
2009-08-29 20:11:38 +00:00
|
|
|
|
2023-11-02 14:33:24 +00:00
|
|
|
__fi void _vu0run(bool breakOnMbit, bool addCycles, bool sync_only) {
|
2010-04-25 00:31:27 +00:00
|
|
|
|
2009-08-29 20:11:38 +00:00
|
|
|
if (!(VU0.VI[REG_VPU_STAT].UL & 1)) return;
|
|
|
|
|
2020-08-09 06:30:24 +00:00
|
|
|
//VU0 is ahead of the EE and M-Bit is already encountered, so no need to wait for it, just catch up the EE
|
2021-09-18 22:06:12 +00:00
|
|
|
if ((VU0.flags & VUFLAG_MFLAGSET) && breakOnMbit && (s32)(cpuRegs.cycle - VU0.cycle) <= 0)
|
2020-08-09 06:30:24 +00:00
|
|
|
{
|
|
|
|
cpuRegs.cycle = VU0.cycle;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-11-02 15:19:38 +00:00
|
|
|
if(!EmuConfig.Cpu.Recompiler.EnableEE)
|
|
|
|
intUpdateCPUCycles();
|
|
|
|
|
2021-09-05 20:36:41 +00:00
|
|
|
u32 startcycle = cpuRegs.cycle;
|
2023-11-02 14:33:24 +00:00
|
|
|
s32 runCycles = 0x7fffffff;
|
|
|
|
|
|
|
|
if (sync_only)
|
|
|
|
{
|
|
|
|
runCycles = (s32)(cpuRegs.cycle - VU0.cycle);
|
|
|
|
|
|
|
|
if (runCycles < 0)
|
|
|
|
return;
|
|
|
|
}
|
2010-04-25 00:31:27 +00:00
|
|
|
|
2010-02-24 07:20:33 +00:00
|
|
|
do { // Run VU until it finishes or M-Bit
|
|
|
|
CpuVU0->Execute(runCycles);
|
2009-08-29 20:11:38 +00:00
|
|
|
} while ((VU0.VI[REG_VPU_STAT].UL & 1) // E-bit Termination
|
2023-11-02 14:33:24 +00:00
|
|
|
&& !sync_only && (!breakOnMbit || (!(VU0.flags & VUFLAG_MFLAGSET) && (s32)(cpuRegs.cycle - VU0.cycle) > 0))); // M-bit Break
|
2009-02-09 21:15:56 +00:00
|
|
|
|
2010-01-25 06:42:09 +00:00
|
|
|
// Add cycles if called from EE's COP2
|
2020-08-09 06:30:24 +00:00
|
|
|
if (addCycles)
|
|
|
|
{
|
|
|
|
cpuRegs.cycle += (VU0.cycle - startcycle);
|
2021-09-05 15:37:43 +00:00
|
|
|
CpuVU1->ExecuteBlock(0); // Catch up VU1 as it's likely fallen behind
|
2021-09-25 00:51:19 +00:00
|
|
|
|
|
|
|
if(VU0.VI[REG_VPU_STAT].UL & 1)
|
|
|
|
cpuSetNextEventDelta(4);
|
2020-08-09 06:30:24 +00:00
|
|
|
}
|
2009-02-09 21:15:56 +00:00
|
|
|
}
|
|
|
|
|
2023-11-02 14:33:24 +00:00
|
|
|
void _vu0WaitMicro() { _vu0run(1, 1, 0); } // Runs VU0 Micro Until E-bit or M-Bit End
|
|
|
|
void _vu0FinishMicro() { _vu0run(0, 1, 0); } // Runs VU0 Micro Until E-Bit End
|
|
|
|
void vu0Finish() { _vu0run(0, 0, 0); } // Runs VU0 Micro Until E-Bit End (doesn't stall EE)
|
|
|
|
void vu0Sync() { _vu0run(0, 0, 1); } // Runs VU0 until it catches up
|
2009-08-29 20:11:38 +00:00
|
|
|
|
2009-02-09 21:15:56 +00:00
|
|
|
namespace R5900 {
|
|
|
|
namespace Interpreter{
|
|
|
|
namespace OpcodeImpl
|
|
|
|
{
|
|
|
|
void LQC2() {
|
2023-11-02 14:33:24 +00:00
|
|
|
vu0Sync();
|
2009-02-09 21:15:56 +00:00
|
|
|
u32 addr = cpuRegs.GPR.r[_Rs_].UL[0] + (s16)cpuRegs.code;
|
|
|
|
if (_Ft_) {
|
2010-08-25 15:32:17 +00:00
|
|
|
memRead128(addr, VU0.VF[_Ft_].UQ);
|
2009-02-09 21:15:56 +00:00
|
|
|
} else {
|
2010-08-25 15:32:17 +00:00
|
|
|
u128 val;
|
|
|
|
memRead128(addr, val);
|
2009-02-09 21:15:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Asadr.Changed
|
|
|
|
//TODO: check this
|
2009-06-06 09:24:10 +00:00
|
|
|
// HUH why ? doesn't make any sense ...
|
2009-02-09 21:15:56 +00:00
|
|
|
void SQC2() {
|
2023-11-02 14:33:24 +00:00
|
|
|
vu0Sync();
|
2009-02-09 21:15:56 +00:00
|
|
|
u32 addr = _Imm_ + cpuRegs.GPR.r[_Rs_].UL[0];
|
2010-08-25 15:32:17 +00:00
|
|
|
memWrite128(addr, VU0.VF[_Ft_].UQ);
|
2009-02-09 21:15:56 +00:00
|
|
|
}
|
|
|
|
}}}
|
|
|
|
|
|
|
|
|
|
|
|
void QMFC2() {
|
2023-11-02 15:19:38 +00:00
|
|
|
vu0Sync();
|
|
|
|
|
2009-02-09 21:15:56 +00:00
|
|
|
if (cpuRegs.code & 1) {
|
2020-08-09 06:30:24 +00:00
|
|
|
_vu0FinishMicro();
|
2009-02-09 21:15:56 +00:00
|
|
|
}
|
2023-11-02 15:19:38 +00:00
|
|
|
|
2009-02-09 21:15:56 +00:00
|
|
|
if (_Rt_ == 0) return;
|
|
|
|
cpuRegs.GPR.r[_Rt_].UD[0] = VU0.VF[_Fs_].UD[0];
|
|
|
|
cpuRegs.GPR.r[_Rt_].UD[1] = VU0.VF[_Fs_].UD[1];
|
2010-04-25 00:31:27 +00:00
|
|
|
}
|
2009-02-09 21:15:56 +00:00
|
|
|
|
|
|
|
void QMTC2() {
|
2023-11-02 15:19:38 +00:00
|
|
|
vu0Sync();
|
|
|
|
|
2009-02-09 21:15:56 +00:00
|
|
|
if (cpuRegs.code & 1) {
|
|
|
|
_vu0WaitMicro();
|
|
|
|
}
|
2023-11-02 15:19:38 +00:00
|
|
|
|
2009-02-09 21:15:56 +00:00
|
|
|
if (_Fs_ == 0) return;
|
|
|
|
VU0.VF[_Fs_].UD[0] = cpuRegs.GPR.r[_Rt_].UD[0];
|
|
|
|
VU0.VF[_Fs_].UD[1] = cpuRegs.GPR.r[_Rt_].UD[1];
|
|
|
|
}
|
|
|
|
|
2010-04-25 00:31:27 +00:00
|
|
|
void CFC2() {
|
2023-11-02 15:19:38 +00:00
|
|
|
vu0Sync();
|
|
|
|
|
2009-02-09 21:15:56 +00:00
|
|
|
if (cpuRegs.code & 1) {
|
2020-08-09 06:30:24 +00:00
|
|
|
_vu0FinishMicro();
|
2009-02-09 21:15:56 +00:00
|
|
|
}
|
2023-11-02 15:19:38 +00:00
|
|
|
|
2009-02-09 21:15:56 +00:00
|
|
|
if (_Rt_ == 0) return;
|
2022-09-14 02:20:25 +00:00
|
|
|
|
2023-03-16 20:26:42 +00:00
|
|
|
if (_Fs_ == REG_R)
|
|
|
|
cpuRegs.GPR.r[_Rt_].UL[0] = VU0.VI[REG_R].UL & 0x7FFFFF;
|
2009-02-09 21:15:56 +00:00
|
|
|
else
|
2023-03-16 20:26:42 +00:00
|
|
|
{
|
|
|
|
cpuRegs.GPR.r[_Rt_].UL[0] = VU0.VI[_Fs_].UL;
|
|
|
|
|
|
|
|
if (VU0.VI[_Fs_].UL & 0x80000000)
|
|
|
|
cpuRegs.GPR.r[_Rt_].UL[1] = 0xffffffff;
|
|
|
|
else
|
|
|
|
cpuRegs.GPR.r[_Rt_].UL[1] = 0;
|
|
|
|
}
|
|
|
|
|
2009-02-09 21:15:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void CTC2() {
|
2023-11-02 15:19:38 +00:00
|
|
|
vu0Sync();
|
|
|
|
|
2009-02-09 21:15:56 +00:00
|
|
|
if (cpuRegs.code & 1) {
|
|
|
|
_vu0WaitMicro();
|
|
|
|
}
|
2023-11-02 15:19:38 +00:00
|
|
|
|
2009-02-09 21:15:56 +00:00
|
|
|
if (_Fs_ == 0) return;
|
|
|
|
|
|
|
|
switch(_Fs_) {
|
|
|
|
case REG_MAC_FLAG: // read-only
|
|
|
|
case REG_TPC: // read-only
|
|
|
|
case REG_VPU_STAT: // read-only
|
|
|
|
break;
|
2022-07-11 16:18:17 +00:00
|
|
|
case REG_R:
|
|
|
|
VU0.VI[REG_R].UL = ((cpuRegs.GPR.r[_Rt_].UL[0] & 0x7FFFFF) | 0x3F800000);
|
|
|
|
break;
|
2009-02-09 21:15:56 +00:00
|
|
|
case REG_FBRST:
|
|
|
|
VU0.VI[REG_FBRST].UL = cpuRegs.GPR.r[_Rt_].UL[0] & 0x0C0C;
|
|
|
|
if (cpuRegs.GPR.r[_Rt_].UL[0] & 0x1) { // VU0 Force Break
|
Lots of new code maintenance stuffs:
* Completely new assertion macros: pxAssert, pxAssertMsg, and pxFail, pxAssertDev (both which default to using a message). These replace *all* wxASSERT, DevAssert, and jASSUME varieties of macros. New macros borrow the best of all assertion worlds: MSVCRT, wxASSERT, and AtlAssume. :)
* Rewrote the Console namespace as a structure called IConsoleWriter, and created several varieties of ConsoleWriters for handling different states of log and console availability (should help reduce overhead of console logging nicely).
* More improvements to the PersistentThread model, using safely interlocked "Do*" style callbacks for starting and cleaning up threads.
* Fixed console logs so that they're readable in Win32 notepad again (the log writer adds CRs to naked LFs).
* Added AppInit.cpp -- contains constructor, destructor, OnInit, and command line parsing mess.
git-svn-id: http://pcsx2.googlecode.com/svn/trunk@1950 96395faa-99c1-11dd-bbfe-3dabce05a288
2009-10-04 08:27:27 +00:00
|
|
|
Console.Error("fixme: VU0 Force Break");
|
2009-02-09 21:15:56 +00:00
|
|
|
}
|
|
|
|
if (cpuRegs.GPR.r[_Rt_].UL[0] & 0x2) { // VU0 Reset
|
Lots of new code maintenance stuffs:
* Completely new assertion macros: pxAssert, pxAssertMsg, and pxFail, pxAssertDev (both which default to using a message). These replace *all* wxASSERT, DevAssert, and jASSUME varieties of macros. New macros borrow the best of all assertion worlds: MSVCRT, wxASSERT, and AtlAssume. :)
* Rewrote the Console namespace as a structure called IConsoleWriter, and created several varieties of ConsoleWriters for handling different states of log and console availability (should help reduce overhead of console logging nicely).
* More improvements to the PersistentThread model, using safely interlocked "Do*" style callbacks for starting and cleaning up threads.
* Fixed console logs so that they're readable in Win32 notepad again (the log writer adds CRs to naked LFs).
* Added AppInit.cpp -- contains constructor, destructor, OnInit, and command line parsing mess.
git-svn-id: http://pcsx2.googlecode.com/svn/trunk@1950 96395faa-99c1-11dd-bbfe-3dabce05a288
2009-10-04 08:27:27 +00:00
|
|
|
//Console.WriteLn("fixme: VU0 Reset");
|
2009-02-09 21:15:56 +00:00
|
|
|
vu0ResetRegs();
|
|
|
|
}
|
|
|
|
if (cpuRegs.GPR.r[_Rt_].UL[0] & 0x100) { // VU1 Force Break
|
Lots of new code maintenance stuffs:
* Completely new assertion macros: pxAssert, pxAssertMsg, and pxFail, pxAssertDev (both which default to using a message). These replace *all* wxASSERT, DevAssert, and jASSUME varieties of macros. New macros borrow the best of all assertion worlds: MSVCRT, wxASSERT, and AtlAssume. :)
* Rewrote the Console namespace as a structure called IConsoleWriter, and created several varieties of ConsoleWriters for handling different states of log and console availability (should help reduce overhead of console logging nicely).
* More improvements to the PersistentThread model, using safely interlocked "Do*" style callbacks for starting and cleaning up threads.
* Fixed console logs so that they're readable in Win32 notepad again (the log writer adds CRs to naked LFs).
* Added AppInit.cpp -- contains constructor, destructor, OnInit, and command line parsing mess.
git-svn-id: http://pcsx2.googlecode.com/svn/trunk@1950 96395faa-99c1-11dd-bbfe-3dabce05a288
2009-10-04 08:27:27 +00:00
|
|
|
Console.Error("fixme: VU1 Force Break");
|
2009-02-09 21:15:56 +00:00
|
|
|
}
|
|
|
|
if (cpuRegs.GPR.r[_Rt_].UL[0] & 0x200) { // VU1 Reset
|
Lots of new code maintenance stuffs:
* Completely new assertion macros: pxAssert, pxAssertMsg, and pxFail, pxAssertDev (both which default to using a message). These replace *all* wxASSERT, DevAssert, and jASSUME varieties of macros. New macros borrow the best of all assertion worlds: MSVCRT, wxASSERT, and AtlAssume. :)
* Rewrote the Console namespace as a structure called IConsoleWriter, and created several varieties of ConsoleWriters for handling different states of log and console availability (should help reduce overhead of console logging nicely).
* More improvements to the PersistentThread model, using safely interlocked "Do*" style callbacks for starting and cleaning up threads.
* Fixed console logs so that they're readable in Win32 notepad again (the log writer adds CRs to naked LFs).
* Added AppInit.cpp -- contains constructor, destructor, OnInit, and command line parsing mess.
git-svn-id: http://pcsx2.googlecode.com/svn/trunk@1950 96395faa-99c1-11dd-bbfe-3dabce05a288
2009-10-04 08:27:27 +00:00
|
|
|
// Console.WriteLn("fixme: VU1 Reset");
|
2009-02-09 21:15:56 +00:00
|
|
|
vu1ResetRegs();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case REG_CMSAR1: // REG_CMSAR1
|
2020-12-12 03:33:24 +00:00
|
|
|
vu1Finish(true);
|
|
|
|
vu1ExecMicro(cpuRegs.GPR.r[_Rt_].US[0]); // Execute VU1 Micro SubRoutine
|
2009-02-09 21:15:56 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
VU0.VI[_Fs_].UL = cpuRegs.GPR.r[_Rt_].UL[0];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|