Files
dusklight/libs/TRK_MINNOW_DOLPHIN/ppc/Generic/targimpl.c
T
Pheenoh 7154ac08e1 d_a_e_fz work, doxygen revamp (#2127)
* initial freezard actor struct + setActionMode OK

* daE_FZ_Draw

* setReflectAngle

* mBoundSoundset

* daE_FZ_Execute & execute

* demoDelete

* daE_FZ_Delete & _delete

* CreateHeap

* useHeapInit

* cc_set

* mtx_set

* action WIP

* way_gake_check

* executeRollMove

* executeMove

* draw WIP

* executeDamage

* checkpoint

* create

* checkpoint

* daE_FZ_c::executeWait

* checkpoint

* daE_FZ_c::damage_check almost done

* rm asm

* rm headers

* setup_profile WIP + doxygen update

* fix merge issues

* docs fix?

* fix2

* doxygen updates

* setup g_profile_E_FZ, profile setup script WIP

* update github actions

* update progress.md
2024-04-12 00:10:30 -06:00

1173 lines
32 KiB
C

#include "TRK_MINNOW_DOLPHIN/ppc/Generic/targimpl.h"
#include "TRK_MINNOW_DOLPHIN/utils/common/MWTrace.h"
typedef struct memRange {
u8* start;
u8* end;
BOOL readable;
BOOL writeable;
} memRange;
const memRange gTRKMemMap[1] = {{(u8*)0, (u8*)-1, TRUE, TRUE}};
typedef struct StopInfo_PPC {
u32 PC;
u32 PCInstruction;
u16 exceptionID;
} StopInfo_PPC;
typedef struct TRKExceptionStatus {
StopInfo_PPC exceptionInfo;
u8 inTRK;
u8 exceptionDetected;
} TRKExceptionStatus;
typedef struct TRKStepStatus {
BOOL active; // 0x0
DSMessageStepOptions type; // 0x4
u32 count; // 0x8
u32 rangeStart; // 0xC
u32 rangeEnd; // 0x10
} TRKStepStatus;
ProcessorRestoreFlags_PPC gTRKRestoreFlags = {FALSE, FALSE};
static TRKExceptionStatus gTRKExceptionStatus = {{0, 0, 0}, TRUE, 0};
static TRKStepStatus gTRKStepStatus = {FALSE, DSSTEP_IntoCount, 0, 0};
static u16 TRK_saved_exceptionID = 0;
TRKState gTRKState;
Default_PPC gTRKSaveState;
ProcessorState_PPC gTRKCPUState;
typedef unsigned char u128[16];
u128 TRKvalue128_temp;
// Instruction macros
#define INSTR_NOP 0x60000000
#define INSTR_BLR 0x4E800020
#define INSTR_PSQ_ST(psr, offset, rDest, w, gqr) \
(0xF0000000 | (psr << 21) | (rDest << 16) | (w << 15) | (gqr << 12) | offset)
#define INSTR_PSQ_L(psr, offset, rSrc, w, gqr) \
(0xE0000000 | (psr << 21) | (rSrc << 16) | (w << 15) | (gqr << 12) | offset)
#define INSTR_STW(rSrc, offset, rDest) (0x90000000 | (rSrc << 21) | (rDest << 16) | offset)
#define INSTR_LWZ(rDest, offset, rSrc) (0x80000000 | (rDest << 21) | (rSrc << 16) | offset)
#define INSTR_STFD(fprSrc, offset, rDest) (0xD8000000 | (fprSrc << 21) | (rDest << 16) | offset)
#define INSTR_LFD(fprDest, offset, rSrc) (0xC8000000 | (fprDest << 21) | (rSrc << 16) | offset)
#define INSTR_MFSPR(rDest, spr) \
(0x7C000000 | (rDest << 21) | ((spr & 0xFE0) << 6) | ((spr & 0x1F) << 16) | 0x2A6)
#define INSTR_MTSPR(spr, rSrc) \
(0x7C000000 | (rSrc << 21) | ((spr & 0xFE0) << 6) | ((spr & 0x1F) << 16) | 0x3A6)
#define DSFetch_u32(_p_) (*((u32*)_p_))
#define DSFetch_u64(_p_) (*((u64*)_p_))
DSError TRKPPCAccessSPR(void* value, u32 spr_register_num, BOOL read);
DSError TRKPPCAccessPairedSingleRegister(void* srcDestPtr, u32 psr, BOOL read);
DSError TRKPPCAccessFPRegister(void* srcDestPtr, u32 fpr, BOOL read);
DSError TRKPPCAccessSpecialReg(void* value, u32* access_func, BOOL read);
static void TRKExceptionHandler(u16);
void TRKInterruptHandlerEnableInterrupts(void);
void WriteFPSCR(register f64*);
void ReadFPSCR(register f64*);
void __TRK_set_MSR(register u32 msr);
u32 __TRK_get_MSR();
void TRK_ppc_memcpy(register void* dest, register const void* src, register int n,
register u32 param_4, register u32 param_5);
void TRKRestoreExtended1Block();
void TRKUARTInterruptHandler();
/* 80370F20-803711C4 36B860 02A4+00 1/1 0/0 0/0 .text TRKValidMemory32 */
DSError TRKValidMemory32(const void* addr, size_t length, ValidMemoryOptions readWriteable) {
DSError err = DS_InvalidMemory; /* assume range is invalid */
const u8* start;
const u8* end;
s32 i;
/*
** Get start and end addresses for the memory range and
** verify that they are reasonable.
*/
start = (const u8*)addr;
end = ((const u8*)addr + (length - 1));
if (end < start)
return DS_InvalidMemory;
/*
** Iterate through the gTRKMemMap array to determine if the requested
** range falls within the valid ranges in the map.
*/
for (i = 0; (i < (s32)(sizeof(gTRKMemMap) / sizeof(memRange))); i++) {
/*
** If the requested range is not completely above
** the valid range AND it is not completely below
** the valid range then it must overlap somewhere.
** If the requested range overlaps with one of the
** valid ranges, do some additional checking.
**
*/
if ((start <= (const u8*)gTRKMemMap[i].end) && (end >= (const u8*)gTRKMemMap[i].start)) {
/*
** First, verify that the read/write attributes are
** acceptable. If so, then recursively check any
** part of the requested range that falls before or
** after the valid range.
*/
if (((readWriteable == VALIDMEM_Readable) && !gTRKMemMap[i].readable) ||
((readWriteable == VALIDMEM_Writeable) && !gTRKMemMap[i].writeable))
{
err = DS_InvalidMemory;
} else {
err = DS_NoError;
/*
** If a portion of the requested range falls before
** the current valid range, then recursively
** check it.
*/
if (start < (const u8*)gTRKMemMap[i].start)
err = TRKValidMemory32(start, (u32)((const u8*)gTRKMemMap[i].start - start),
readWriteable);
/*
** If a portion of the requested range falls after
** the current valid range, then recursively
** check it.
** Note: Only do this step if the previous check
** did not detect invalid access.
*/
if ((err == DS_NoError) && (end > (const u8*)gTRKMemMap[i].end))
err =
TRKValidMemory32((const u8*)gTRKMemMap[i].end,
(u32)(end - (const u8*)gTRKMemMap[i].end), readWriteable);
}
break;
}
}
return err;
}
/* 80370DD4-80370F20 36B714 014C+00 1/1 2/2 0/0 .text TRKTargetAccessMemory */
DSError TRKTargetAccessMemory(void* data, u32 start, size_t* length,
MemoryAccessOptions accessOptions, BOOL read) {
DSError error;
u32 uVar5;
void* addr;
u32 param4;
TRKExceptionStatus tempExceptionStatus = gTRKExceptionStatus;
gTRKExceptionStatus.exceptionDetected = FALSE;
addr = (void*)TRKTargetTranslate(start);
error = TRKValidMemory32(addr, *length, read == FALSE);
if (error != DS_NoError) {
*length = 0;
} else {
uVar5 = __TRK_get_MSR();
param4 = uVar5 | gTRKCPUState.Extended1.MSR & 0x10;
if (read) {
TRK_ppc_memcpy(data, addr, *length, uVar5, param4);
} else {
TRK_ppc_memcpy(addr, data, *length, param4, uVar5);
TRK_flush_cache((u32)addr, *length);
if ((void*)start != addr) {
TRK_flush_cache(start, *length);
}
}
}
if (gTRKExceptionStatus.exceptionDetected) {
*length = 0;
error = DS_CWDSException;
}
gTRKExceptionStatus = tempExceptionStatus;
return error;
}
/* 80370D88-80370DD4 36B6C8 004C+00 3/3 0/0 0/0 .text TRKTargetReadInstruction */
DSError TRKTargetReadInstruction(void* data, u32 start) {
DSError error;
size_t registersLength = 4;
error = TRKTargetAccessMemory(data, start, &registersLength, MEMACCESS_UserMemory, TRUE);
if (error == DS_NoError && registersLength != 4) {
error = DS_InvalidMemory;
}
return error;
}
/* 80370C94-80370D88 36B5D4 00F4+00 0/0 2/2 0/0 .text TRKTargetAccessDefault */
DSError TRKTargetAccessDefault(u32 firstRegister, u32 lastRegister, TRKBuffer* b,
size_t* registersLengthPtr, BOOL read) {
DSError error;
u32 count;
u32* data;
TRKExceptionStatus tempExceptionStatus;
if (lastRegister > 0x24) {
return DS_InvalidRegister;
}
tempExceptionStatus = gTRKExceptionStatus;
gTRKExceptionStatus.exceptionDetected = FALSE;
data = gTRKCPUState.Default.GPR + firstRegister;
count = (lastRegister - firstRegister) + 1;
*registersLengthPtr = count * sizeof(u32);
if (read) {
error = TRKAppendBuffer_ui32(b, data, count);
} else {
error = TRKReadBuffer_ui32(b, data, count);
}
if (gTRKExceptionStatus.exceptionDetected) {
*registersLengthPtr = 0;
error = DS_CWDSException;
}
gTRKExceptionStatus = tempExceptionStatus;
return error;
}
/* 80370788-80370C94 36B0C8 050C+00 0/0 2/2 0/0 .text TRKTargetAccessFP */
DSError TRKTargetAccessFP(u32 firstRegister, u32 lastRegister, TRKBuffer* b,
size_t* registersLengthPtr, BOOL read) {
u64 temp;
DSError error;
TRKExceptionStatus tempExceptionStatus;
u32 current;
if (lastRegister > 0x21) {
return DS_InvalidRegister;
}
tempExceptionStatus = gTRKExceptionStatus;
gTRKExceptionStatus.exceptionDetected = FALSE;
__TRK_set_MSR(__TRK_get_MSR() | 0x2000);
*registersLengthPtr = 0;
error = DS_NoError;
for (current = firstRegister; (current <= lastRegister) && (error == DS_NoError);
current++, *registersLengthPtr += sizeof(f64))
{
if (read) {
TRKPPCAccessFPRegister(&temp, current, read);
error = TRKAppendBuffer1_ui64(b, temp);
} else {
TRKReadBuffer1_ui64(b, &temp);
error = TRKPPCAccessFPRegister(&temp, current, read);
}
}
if (gTRKExceptionStatus.exceptionDetected) {
*registersLengthPtr = 0;
error = DS_CWDSException;
}
gTRKExceptionStatus = tempExceptionStatus;
return error;
}
/* 80370618-80370788 36AF58 0170+00 0/0 2/2 0/0 .text TRKTargetAccessExtended1 */
DSError TRKTargetAccessExtended1(u32 firstRegister, u32 lastRegister, TRKBuffer* b,
size_t* registersLengthPtr, BOOL read) {
TRKExceptionStatus tempExceptionStatus;
int error;
u32* data;
int count;
if (lastRegister > 0x60) {
return DS_InvalidRegister;
}
tempExceptionStatus = gTRKExceptionStatus;
gTRKExceptionStatus.exceptionDetected = FALSE;
*registersLengthPtr = 0;
if (firstRegister <= lastRegister) {
data = (u32*)&gTRKCPUState.Extended1 + firstRegister;
count = lastRegister - firstRegister + 1;
*registersLengthPtr += count * sizeof(u32);
if (read) {
error = TRKAppendBuffer_ui32(b, data, count);
} else {
if (data <= &gTRKCPUState.Extended1.TBU &&
(data + count - 1) >= &gTRKCPUState.Extended1.TBL)
{
gTRKRestoreFlags.TBR = 1;
}
if (data <= &gTRKCPUState.Extended1.DEC &&
(data + count - 1) >= &gTRKCPUState.Extended1.DEC)
{
gTRKRestoreFlags.DEC = 1;
}
error = TRKReadBuffer_ui32(b, data, count);
}
}
if (gTRKExceptionStatus.exceptionDetected) {
*registersLengthPtr = 0;
error = DS_CWDSException;
}
gTRKExceptionStatus = tempExceptionStatus;
return error;
}
/* 803701E0-80370618 36AB20 0438+00 0/0 2/2 0/0 .text TRKTargetAccessExtended2 */
DSError TRKTargetAccessExtended2(u32 firstRegister, u32 lastRegister, TRKBuffer* b,
size_t* registerStorageSize, BOOL read) {
TRKExceptionStatus savedException;
u32 i;
u32 value_buf0[1];
u32 value_buf[2];
DSError err;
u32 access_func[10];
if (lastRegister > 0x1f)
return DS_InvalidRegister;
/*
** Save any existing exception status and clear the exception flag.
** This allows detection of exceptions that occur ONLY within this
** function.
*/
savedException = gTRKExceptionStatus;
gTRKExceptionStatus.exceptionDetected = FALSE;
TRKPPCAccessSPR(value_buf0, SPR_HID2, TRUE);
value_buf0[0] |= 0xA0000000;
TRKPPCAccessSPR(value_buf0, SPR_HID2, FALSE);
value_buf0[0] = 0;
TRKPPCAccessSPR(value_buf0, SPR_GQR0, FALSE);
*registerStorageSize = 0;
err = DS_NoError;
for (i = firstRegister; (i <= lastRegister) && (err == DS_NoError); i++) {
if (read) {
err = TRKPPCAccessPairedSingleRegister((u64*)value_buf, i, read);
err = TRKAppendBuffer1_ui64(b, *(u64*)value_buf);
} else {
err = TRKReadBuffer1_ui64(b, (u64*)value_buf);
err = TRKPPCAccessPairedSingleRegister((u64*)value_buf, i, read);
}
*registerStorageSize += sizeof(u64);
}
if (gTRKExceptionStatus.exceptionDetected) {
*registerStorageSize = 0;
err = DS_CWDSException;
}
gTRKExceptionStatus = savedException;
return err;
}
/* 80370134-803701E0 36AA74 00AC+00 2/2 0/0 0/0 .text TRKPostInterruptEvent */
void TRKPostInterruptEvent(void) {
int eventType;
int local_14;
size_t registerSize;
TRKEvent event;
if (gTRKState.inputActivated) {
gTRKState.inputActivated = FALSE;
} else {
switch (gTRKCPUState.Extended1.exceptionID & 0xFFFF) {
case 0xd00:
case 0x700:
registerSize = 4;
TRKTargetReadInstruction(&local_14, gTRKCPUState.Default.PC);
if (local_14 == 0xfe00000) {
eventType = NUBEVENT_Support;
} else {
eventType = NUBEVENT_Breakpoint;
}
break;
default:
eventType = NUBEVENT_Exception;
break;
}
TRKConstructEvent(&event, eventType);
TRKPostEvent(&event);
}
}
/* 8036FFA4-80370134 36A8E4 0190+00 0/0 1/1 0/0 .text TRKTargetInterrupt */
DSError TRKTargetInterrupt(TRKEvent* event) {
DSError error = DS_NoError;
switch (event->eventType) {
case NUBEVENT_Breakpoint:
case NUBEVENT_Exception:
if (TRKTargetCheckStep() == FALSE) {
TRKTargetSetStopped(TRUE);
error = TRKDoNotifyStopped(DSMSG_NotifyStopped);
}
break;
default:
break;
}
return error;
}
/* 8036FF18-8036FFA4 36A858 008C+00 0/0 1/1 0/0 .text TRKTargetAddStopInfo */
u32* ConvertAddress(u32 addr) {
return (u32*)(addr | 0x80000000);
}
DSError TRKTargetAddStopInfo(TRKBuffer* b) {
DSError error;
CommandReply reply;
int t;
memset(&reply, 0, 0x40);
reply._00 = 0x40;
reply.commandID.b = 0x90;
reply.replyError.r = gTRKCPUState.Default.PC;
TRKTargetReadInstruction(&t, gTRKCPUState.Default.PC);
reply._0C = t;
*(u32*)reply._10 = gTRKCPUState.Extended1.exceptionID & 0xFFFF;
error = TRKAppendBuffer_ui8(b, (u8*)&reply, 0x40);
return error;
}
/* 8036FE94-8036FF18 36A7D4 0084+00 0/0 1/1 0/0 .text TRKTargetAddExceptionInfo */
void TRKTargetAddExceptionInfo(TRKBuffer* b) {
size_t local_58;
u32 local_54;
CommandReply reply;
memset(&reply, 0, 0x40);
reply._00 = 0x40;
reply.commandID.b = 0x91;
reply.replyError.r = gTRKExceptionStatus.exceptionInfo.PC;
TRKTargetReadInstruction(&local_54, gTRKExceptionStatus.exceptionInfo.PC);
reply._0C = local_54;
*(u32*)reply._10 = gTRKExceptionStatus.exceptionInfo.exceptionID;
TRKAppendBuffer_ui8(b, (u8*)&reply, 0x40);
}
inline DSError TRKTargetEnableTrace(BOOL val) {
if (val) {
gTRKCPUState.Extended1.MSR = (gTRKCPUState.Extended1.MSR | 0x400);
} else {
gTRKCPUState.Extended1.MSR = (gTRKCPUState.Extended1.MSR & ~0x400);
}
return DS_NoError;
}
BOOL TRKTargetStepDone() {
BOOL result = TRUE;
if (gTRKStepStatus.active && ((u16)gTRKCPUState.Extended1.exceptionID) == PPC_Trace) {
switch (gTRKStepStatus.type) {
case DSSTEP_IntoCount:
if (gTRKStepStatus.count > 0) {
result = FALSE;
}
break;
case DSSTEP_IntoRange:
if (gTRKCPUState.Default.PC >= gTRKStepStatus.rangeStart &&
gTRKCPUState.Default.PC <= gTRKStepStatus.rangeEnd)
{
result = FALSE;
}
break;
default:
break;
}
}
return result;
}
inline DSError TRKTargetDoStep() {
gTRKStepStatus.active = TRUE;
MWTRACE(1, "TargetDoStep()\n");
TRKTargetEnableTrace(TRUE);
if (gTRKStepStatus.type == DSSTEP_IntoCount || gTRKStepStatus.type == DSSTEP_OverCount) {
gTRKStepStatus.count--;
}
TRKTargetSetStopped(FALSE);
return DS_NoError;
}
static BOOL TRKTargetCheckStep() {
if (gTRKStepStatus.active) {
TRKTargetEnableTrace(FALSE);
if (TRKTargetStepDone()) {
gTRKStepStatus.active = FALSE;
} else {
TRKTargetDoStep();
}
}
return gTRKStepStatus.active;
}
/* 8036FDE8-8036FE94 36A728 00AC+00 0/0 1/1 0/0 .text TRKTargetSingleStep */
DSError TRKTargetSingleStep(u32 count, BOOL stepOver) {
DSError error = DS_NoError;
if (stepOver) {
error = DS_UnsupportedError;
} else {
gTRKStepStatus.count = count;
gTRKStepStatus.type = DSSTEP_IntoCount;
error = TRKTargetDoStep();
}
return error;
}
/* 8036FD30-8036FDE8 36A670 00B8+00 0/0 1/1 0/0 .text TRKTargetStepOutOfRange */
DSError TRKTargetStepOutOfRange(u32 rangeStart, u32 rangeEnd, BOOL stepOver) {
DSError error = DS_NoError;
if (stepOver) {
// Stepping over isn't supported for PowerPC
error = DS_UnsupportedError;
} else {
gTRKStepStatus.type = DSSTEP_IntoRange;
// gTRKStepStatus.active = TRUE;
gTRKStepStatus.rangeStart = rangeStart;
gTRKStepStatus.rangeEnd = rangeEnd;
error = TRKTargetDoStep();
}
return error;
}
/* 8036FD20-8036FD30 36A660 0010+00 0/0 1/1 0/0 .text TRKTargetGetPC */
u32 TRKTargetGetPC() {
return gTRKCPUState.Default.PC;
}
/* 8036FB20-8036FD20 36A460 0200+00 0/0 1/1 0/0 .text TRKTargetSupportRequest */
DSError TRKTargetSupportRequest() {
DSIOResult ioResult;
size_t* length;
MessageCommandID commandId;
DSError error;
u32 local_28;
TRKEvent event;
commandId = gTRKCPUState.Default.GPR[3];
if (commandId != DSMSG_ReadFile && commandId != DSMSG_WriteFile &&
commandId != DSMSG_OpenFile && commandId != DSMSG_CloseFile &&
commandId != DSMSG_PositionFile)
{
TRKConstructEvent(&event, 4);
TRKPostEvent(&event);
return DS_NoError;
} else if (commandId == DSMSG_OpenFile) {
error = HandleOpenFileSupportRequest(gTRKCPUState.Default.GPR[4],
gTRKCPUState.Default.GPR[5] & 0xff,
gTRKCPUState.Default.GPR[6], &ioResult);
if (ioResult == DS_IONoError && error != DS_NoError) {
ioResult = DS_IOError;
}
gTRKCPUState.Default.GPR[3] = ioResult;
} else if (commandId == DSMSG_CloseFile) {
error = HandleCloseFileSupportRequest(gTRKCPUState.Default.GPR[4], &ioResult);
if (ioResult == DS_IONoError && error != DS_NoError) {
ioResult = DS_IOError;
}
gTRKCPUState.Default.GPR[3] = ioResult;
} else if (commandId == DSMSG_PositionFile) {
local_28 = *(u32*)gTRKCPUState.Default.GPR[5];
error = HandlePositionFileSupportRequest(gTRKCPUState.Default.GPR[4], &local_28,
(u8)gTRKCPUState.Default.GPR[6], &ioResult);
if (ioResult == DS_IONoError && error != DS_NoError) {
ioResult = DS_IOError;
}
gTRKCPUState.Default.GPR[3] = ioResult;
*(u32*)gTRKCPUState.Default.GPR[5] = local_28;
} else {
length = (size_t*)gTRKCPUState.Default.GPR[5];
error = TRKSuppAccessFile(gTRKCPUState.Default.GPR[4], (u8*)gTRKCPUState.Default.GPR[6],
length, &ioResult, TRUE, commandId == DSMSG_ReadFile);
if (ioResult == DS_IONoError && error != DS_NoError) {
ioResult = DS_IOError;
}
gTRKCPUState.Default.GPR[3] = ioResult;
if (commandId == DSMSG_ReadFile) {
TRK_flush_cache(gTRKCPUState.Default.GPR[6], *length);
}
}
gTRKCPUState.Default.PC += 4;
return error;
}
/* 8036FB10-8036FB20 36A450 0010+00 0/0 3/3 0/0 .text TRKTargetStopped */
BOOL TRKTargetStopped() {
return gTRKState.isStopped;
}
/* 8036FB00-8036FB10 36A440 0010+00 0/0 1/1 0/0 .text TRKTargetSetStopped */
void TRKTargetSetStopped(unsigned int stopped) {
gTRKState.isStopped = stopped;
}
/* 8036FAE8-8036FB00 36A428 0018+00 0/0 1/1 0/0 .text TRKTargetStop */
u32 TRKTargetStop() {
gTRKState.isStopped = TRUE;
return 0;
}
DSError TRKPPCAccessSPR(void* value, u32 spr_register_num, BOOL read) {
/* Initialize instruction array with nop */
u32 access_func[10] = {INSTR_NOP, INSTR_NOP, INSTR_NOP, INSTR_NOP, INSTR_NOP,
INSTR_NOP, INSTR_NOP, INSTR_NOP, INSTR_NOP, INSTR_NOP};
/*
** Construct a small assembly function to perform the
** requested access and call it. The read/write function
** is in the form:
**
** read:
** mfspr r4, spr_register_num
** stw r4, 0(r3)
** blr
**
** write:
** lwz r4, 0(r3)
** mtspr spr_register_num, r4
** blr
**
*/
if (read) {
access_func[0] = INSTR_MFSPR(4, spr_register_num);
access_func[1] = (u32)INSTR_STW(4, 0, 3);
} else {
access_func[0] = (u32)INSTR_LWZ(4, 0, 3);
access_func[1] = INSTR_MTSPR(spr_register_num, 4);
}
return TRKPPCAccessSpecialReg(value, access_func, read);
}
DSError TRKPPCAccessPairedSingleRegister(void* srcDestPtr, u32 psr, BOOL read) {
// all nop by default
u32 instructionData[] = {INSTR_NOP, INSTR_NOP, INSTR_NOP, INSTR_NOP, INSTR_NOP,
INSTR_NOP, INSTR_NOP, INSTR_NOP, INSTR_NOP, INSTR_NOP};
if (read) {
instructionData[0] = INSTR_PSQ_ST(psr, 0, 3, 0, 0); // psq_st psr, 0(r3), 0, 0
} else {
instructionData[0] = INSTR_PSQ_L(psr, 0, 3, 0, 0); // psq_l psr, 0(r3), 0, 0
}
return TRKPPCAccessSpecialReg(srcDestPtr, instructionData, read);
}
DSError TRKPPCAccessFPRegister(void* srcDestPtr, u32 fpr, BOOL read) {
DSError error = DS_NoError;
// all nop by default
u32 instructionData1[] = {INSTR_NOP, INSTR_NOP, INSTR_NOP, INSTR_NOP, INSTR_NOP,
INSTR_NOP, INSTR_NOP, INSTR_NOP, INSTR_NOP, INSTR_NOP};
if (fpr < 0x20) {
if (read) {
instructionData1[0] = INSTR_STFD(fpr, 0, 3); // stfd fpr, 0(r3)
} else {
instructionData1[0] = INSTR_LFD(fpr, 0, 3); // lfd fpr, 0(r3)
}
error = TRKPPCAccessSpecialReg(srcDestPtr, instructionData1, read);
} else if (fpr == 0x20) {
if (read) {
ReadFPSCR(srcDestPtr);
} else {
WriteFPSCR(srcDestPtr);
}
*(u64*)srcDestPtr &= 0xFFFFFFFF;
} else if (fpr == 0x21) {
if (!read) {
*(u32*)srcDestPtr = *((u32*)(srcDestPtr) + 1);
}
error = TRKPPCAccessSPR(srcDestPtr, SPR_FPECR, read);
if (read) {
DSFetch_u64(srcDestPtr) = DSFetch_u32(srcDestPtr) & 0xffffffffLL;
}
}
return error;
}
#define DEBUG_VECTORREG_ACCESS 0
DSError TRKPPCAccessSpecialReg(void* value, u32* access_func, BOOL read) {
#if defined(__MWERKS__)
#pragma unused(read)
#elif defined(__GNUC__)
UNUSED(read);
#endif
typedef void (*asm_access_type)(void*, void*);
asm_access_type asm_access;
/*
** Construct a small assembly function to perform the
** requested access and call it. The read/write function
** is in the form:
**
** <access_func>
** blr
*/
/*
** Put blr instruction at the end of access function (it should be
** a 5-instruction array w/the last one empty).
*/
access_func[9] = INSTR_BLR;
/*
** Now that the instruction array is built, get a function pointer to it.
*/
asm_access = (asm_access_type)access_func;
#if DEBUG_VECTORREG_ACCESS
__puts("\r\nasm_access: ");
__puthex8((u32)asm_access);
__puts(" access_func: ");
__puthex8((u32)access_func);
for (i = 0; i < 10; i++) {
__puts("\r\ninst[");
__puthex2(i);
__puts("]: ");
__puthex8(access_func[i]);
__puts(" ; ");
__puthex8(*((u32*)asm_access + i));
}
__puts("\r\n");
#endif
// Flush cache
TRK_flush_cache((u32)access_func, (sizeof(access_func) * 10));
(*asm_access)((u32*)value, (void*)&TRKvalue128_temp);
return DS_NoError;
}
/* 8036FAD8-8036FAE8 36A418 0010+00 0/0 1/1 0/0 .text TRKTargetSetInputPendingPtr */
void TRKTargetSetInputPendingPtr(void* ptr) {
gTRKState.inputPendingPtr = ptr;
}
/* 8036FA14-8036FAD8 36A354 00C4+00 0/0 2/2 0/0 .text TRKTargetAccessARAM */
DSError TRKTargetAccessARAM(u32 p1, u32 p2, u32* p3, BOOL read) {
DSError err;
TRKExceptionStatus status;
err = DS_NoError;
status = gTRKExceptionStatus;
gTRKExceptionStatus.exceptionDetected = 0;
if (read) {
TRK__read_aram(p1, p2, p3);
} else {
TRK__write_aram(p1, p2, p3);
}
if (gTRKExceptionStatus.exceptionDetected) {
*p3 = 0;
err = 0x702;
}
gTRKExceptionStatus = status;
return err;
}
/* 8036F638-8036F640 -00001 0008+00 0/0 0/0 0/0 .text __TRK_get_MSR */
asm u32 __TRK_get_MSR() {
// clang-format off
nofralloc
mfmsr r3
blr
// clang-format on
}
/* 8036F640-8036F648 -00001 0008+00 0/0 0/0 0/0 .text __TRK_set_MSR */
asm void __TRK_set_MSR() {
// clang-format off
nofralloc
mtmsr r3
blr
// clang-format on
}
/* 8036F648-8036F684 369F88 003C+00 1/1 0/0 0/0 .text TRK_ppc_memcpy */
static asm void TRK_ppc_memcpy(register void* dest, register const void* src, register int n,
register u32 param_4, register u32 param_5) {
// clang-format off
#define msr r8
#define byte r9
#define count r10
nofralloc
mfmsr msr
li count, 0
top_loop:
cmpw count, n
beq out_loop
mtmsr param_5
sync
lbzx byte, count, src
mtmsr param_4
sync
stbx byte, count, dest
addi count, count, 1
b top_loop
out_loop:
mtmsr msr
sync
blr
#undef count
#undef byte
#undef msr
// clang-format on
}
/* 8036F684-8036F818 369FC4 0194+00 0/0 1/1 0/0 .text TRKInterruptHandler */
asm void TRKInterruptHandler() {
// clang-format off
nofralloc
mtsrr0 r2
mtsrr1 r4
mfsprg r4, 3
mfcr r2
mtsprg 3, r2
lis r2, gTRKState@h
ori r2, r2, gTRKState@l
lwz r2, TRKState_PPC.MSR(r2)
ori r2, r2, 0x8002
xori r2, r2, 0x8002
sync
mtmsr r2
sync
lis r2, TRK_saved_exceptionID@h
ori r2, r2, TRK_saved_exceptionID@l
sth r3, 0(r2)
cmpwi r3, 0x500
bne L_802CF694
lis r2, gTRKCPUState@h
ori r2, r2, gTRKCPUState@l
mflr r3
stw r3, ProcessorState_PPC.transport_handler_saved_ra(r2)
bl TRKUARTInterruptHandler
lis r2, gTRKCPUState@h
ori r2, r2, gTRKCPUState@l
lwz r3, ProcessorState_PPC.transport_handler_saved_ra(r2)
mtlr r3
lis r2, gTRKState@h
ori r2, r2, gTRKState@l
lwz r2, TRKState_PPC.inputPendingPtr(r2)
lbz r2, TRKState_PPC.GPR[0](r2)
cmpwi r2, 0
beq L_802CF678
lis r2, gTRKExceptionStatus@h
ori r2, r2, gTRKExceptionStatus@l
lbz r2, TRKExceptionStatus.inTRK(r2)
cmpwi r2, 1
beq L_802CF678
lis r2, gTRKState@h
ori r2, r2, gTRKState@l
li r3, 1
stb r3, TRKState_PPC.inputActivated(r2)
b L_802CF694
L_802CF678:
lis r2, gTRKSaveState@h
ori r2, r2, gTRKSaveState@l
lwz r3, Default_PPC.CR(r2)
mtcrf 0xff, r3
lwz r3, Default_PPC.GPR[3](r2)
lwz r2, Default_PPC.GPR[2](r2)
rfi
L_802CF694:
lis r2, TRK_saved_exceptionID@h
ori r2, r2, TRK_saved_exceptionID@l
lhz r3, 0(r2)
lis r2, gTRKExceptionStatus@h
ori r2, r2, gTRKExceptionStatus@l
lbz r2, TRKExceptionStatus.inTRK(r2)
cmpwi r2, 0
bne TRKExceptionHandler
lis r2, gTRKCPUState@h
ori r2, r2, gTRKCPUState@l
stw r0, ProcessorState_PPC.Default.GPR[0](r2)
stw r1, ProcessorState_PPC.Default.GPR[1](r2)
mfsprg r0, 1
stw r0, ProcessorState_PPC.Default.GPR[2](r2)
sth r3, ProcessorState_PPC.Extended1.exceptionID(r2)
sth r3, (ProcessorState_PPC.Extended1.exceptionID + 2)(r2)
mfsprg r0, 2
stw r0, ProcessorState_PPC.Default.GPR[3](r2)
stmw r4, ProcessorState_PPC.Default.GPR[4](r2)
mfsrr0 r27
mflr r28
mfsprg r29, 3
mfctr r30
mfxer r31
stmw r27, ProcessorState_PPC.Default.PC(r2)
bl TRKSaveExtended1Block
lis r2, gTRKExceptionStatus@h
ori r2, r2, gTRKExceptionStatus@l
li r3, 1
stb r3, TRKExceptionStatus.inTRK(r2)
lis r2, gTRKState@h
ori r2, r2, gTRKState@l
lwz r0, TRKState_PPC.MSR(r2)
sync
mtmsr r0
sync
lwz r0, TRKState_PPC.LR(r2)
mtlr r0
lwz r0, TRKState_PPC.CTR(r2)
mtctr r0
lwz r0, TRKState_PPC.XER(r2)
mtxer r0
lwz r0, TRKState_PPC.DSISR(r2)
mtdsisr r0
lwz r0, TRKState_PPC.DAR(r2)
mtdar r0
lmw r3, TRKState_PPC.GPR[3](r2)
lwz r0, TRKState_PPC.GPR[0](r2)
lwz r1, TRKState_PPC.GPR[1](r2)
lwz r2, TRKState_PPC.GPR[2](r2)
b TRKPostInterruptEvent
// clang-format on
}
/* 8036F818-8036F8B4 36A158 009C+00 1/1 0/0 0/0 .text TRKExceptionHandler */
static asm void TRKExceptionHandler(u16) {
// clang-format off
nofralloc
lis r2, gTRKExceptionStatus@h
ori r2, r2, gTRKExceptionStatus@l
sth r3, TRKExceptionStatus.exceptionInfo.exceptionID(r2)
mfsrr0 r3
stw r3, TRKExceptionStatus.exceptionInfo.PC(r2)
lhz r3, TRKExceptionStatus.exceptionInfo.exceptionID(r2)
cmpwi r3, 0x200
beq LAB_00010ba4
cmpwi r3, 0x300
beq LAB_00010ba4
cmpwi r3, 0x400
beq LAB_00010ba4
cmpwi r3, 0x600
beq LAB_00010ba4
cmpwi r3, 0x700
beq LAB_00010ba4
cmpwi r3, 0x800
beq LAB_00010ba4
cmpwi r3, 0x1000
beq LAB_00010ba4
cmpwi r3, 0x1100
beq LAB_00010ba4
cmpwi r3, 0x1200
beq LAB_00010ba4
cmpwi r3, 0x1300
beq LAB_00010ba4
b LAB_00010bb0
LAB_00010ba4:
mfsrr0 r3
addi r3, r3, 0x4
mtsrr0 r3
LAB_00010bb0:
lis r2, gTRKExceptionStatus@h
ori r2, r2, gTRKExceptionStatus@l
li r3, 0x1
stb r3, TRKExceptionStatus.exceptionDetected(r2)
mfsprg r3, 3
mtcrf 0xff, r3
mfsprg r2, 1
mfsprg r3, 2
rfi
// clang-format on
}
/* 8036F8B4-8036F978 36A1F4 00C4+00 0/0 1/1 0/0 .text TRKSwapAndGo */
asm void TRKSwapAndGo() {
// clang-format off
nofralloc
lis r3, gTRKState@h
ori r3, r3, gTRKState@l
stmw r0, TRKState_PPC.GPR[0](r3)
mfmsr r0
stw r0, TRKState_PPC.MSR(r3)
mflr r0
stw r0, TRKState_PPC.LR(r3)
mfctr r0
stw r0, TRKState_PPC.CTR(r3)
mfxer r0
stw r0, TRKState_PPC.XER(r3)
mfdsisr r0
stw r0, TRKState_PPC.DSISR(r3)
mfdar r0
stw r0, TRKState_PPC.DAR(r3)
li r1, -0x7ffe
nor r1, r1, r1
mfmsr r3
and r3, r3, r1
mtmsr r3
lis r2, gTRKState@h
ori r2, r2, gTRKState@l
lwz r2, TRKState_PPC.inputPendingPtr(r2)
lbz r2, TRKState_PPC.GPR[0](r2)
cmpwi r2, 0
beq L_802CF930
lis r2, gTRKState@h
ori r2, r2, gTRKState@l
li r3, 1
stb r3, TRKState_PPC.inputActivated(r2)
b TRKInterruptHandlerEnableInterrupts
L_802CF930:
lis r2, gTRKExceptionStatus@h
ori r2, r2, gTRKExceptionStatus@l
li r3, 0
stb r3, 0xc(r2)
bl TRKRestoreExtended1Block
lis r2, gTRKCPUState@h
ori r2, r2, gTRKCPUState@l
lmw r27, ProcessorState_PPC.Default.PC(r2)
mtsrr0 r27
mtlr r28
mtcrf 0xff, r29
mtctr r30
mtxer r31
lmw r3, ProcessorState_PPC.Default.GPR[3](r2)
lwz r0, ProcessorState_PPC.Default.GPR[0](r2)
lwz r1, ProcessorState_PPC.Default.GPR[1](r2)
lwz r2, ProcessorState_PPC.Default.GPR[2](r2)
rfi
// clang-format on
}
/* 8036F978-8036F9CC 36A2B8 0054+00 1/1 0/0 0/0 .text TRKInterruptHandlerEnableInterrupts
*/
asm void TRKInterruptHandlerEnableInterrupts(void) {
// clang-format off
nofralloc;
lis r2, gTRKState@h
ori r2, r2, gTRKState@l
lwz r0, TRKState_PPC.MSR(r2)
sync
mtmsr r0
sync
lwz r0, TRKState_PPC.LR(r2)
mtlr r0
lwz r0, TRKState_PPC.CTR(r2)
mtctr r0
lwz r0, TRKState_PPC.XER(r2)
mtxer r0
lwz r0, TRKState_PPC.DSISR(r2)
mtdsisr r0
lwz r0, TRKState_PPC.DAR(r2)
mtdar r0
lmw r3, TRKState_PPC.GPR[3](r2)
lwz r0, TRKState_PPC.GPR[0](r2)
lwz r1, TRKState_PPC.GPR[1](r2)
lwz r2, TRKState_PPC.GPR[2](r2)
b TRKPostInterruptEvent
// clang-format on
}
/* 8036F9CC-8036F9F0 36A30C 0024+00 1/1 0/0 0/0 .text ReadFPSCR */
asm void ReadFPSCR(register f64*) {
// clang-format off
nofralloc
stwu r1, -0x40(r1)
stfd f31, 0x10(r1)
psq_st f31, 0x20(r1),0,0
mffs f31
stfd f31, 0x0(r3)
psq_l f31, 0x20(r1),0,0
lfd f31, 0x10(r1)
addi r1, r1, 0x40
blr
// clang-format on
}
/* 8036F9F0-8036FA14 36A330 0024+00 1/1 0/0 0/0 .text WriteFPSCR */
asm void WriteFPSCR(register f64*) {
// clang-format off
nofralloc
stwu r1, -0x40(r1)
stfd f31, 0x10(r1)
psq_st f31, 0x20(r1), 0, 0
lfd f31, 0(r3)
mtfsf 0xff, f31
psq_l f31, 0x20(r1), 0, 0
lfd f31, 0x10(r1)
addi r1, r1, 0x40
blr
// clang-format on
}