From f8bd26d6c60dee1bd90fdcdcb78d24e3a8206809 Mon Sep 17 00:00:00 2001 From: cadmic Date: Mon, 14 Oct 2024 11:40:28 -0700 Subject: [PATCH] Match some S2DEX functions (#193) * Match some s2dex functions * Rename s2dex structs * Remove stray comment * Decode scissorX1/scissorY1 --- include/emulator/rsp.h | 74 +++- src/emulator/frame.c | 265 ++++++++++++++- src/emulator/rdp.c | 4 +- src/emulator/rsp.c | 754 +++++++++++++++++++++++++++++++++++++++-- 4 files changed, 1056 insertions(+), 41 deletions(-) diff --git a/include/emulator/rsp.h b/include/emulator/rsp.h index b0ff5294..6cccd464 100644 --- a/include/emulator/rsp.h +++ b/include/emulator/rsp.h @@ -35,6 +35,10 @@ #define AUDIO_SEGMENT_ADDRESS(pRSP, nOffsetRDRAM) \ (pRSP->anAudioBaseSegment[((nOffsetRDRAM) >> 24) & 0xF] + ((nOffsetRDRAM) & 0xFFFFFF)) +#define G_OBJLT_TXTRBLOCK 0x00001033 +#define G_OBJLT_TXTRTILE 0x00FC1034 +#define G_OBJLT_TLUT 0x00000030 + typedef enum RspAudioUCodeType { RUT_NOCODE = -1, RUT_ABI1 = 0, @@ -264,7 +268,7 @@ typedef struct Rsp { /* 0x39C8 */ int* dctBuf; } Rsp; // size = 0x39CC -typedef struct __anon_0x5ED4F { +typedef struct uObjBg_t { /* 0x00 */ u16 imageX; /* 0x02 */ u16 imageW; /* 0x04 */ s16 frameX; @@ -285,9 +289,9 @@ typedef struct __anon_0x5ED4F { /* 0x22 */ u16 tmemLoadTH; /* 0x24 */ u16 tmemSizeW; /* 0x26 */ u16 tmemSize; -} __anon_0x5ED4F; // size = 0x28 +} uObjBg_t; // size = 0x28 -typedef struct __anon_0x5F05A { +typedef struct uObjScaleBg_t { /* 0x00 */ u16 imageX; /* 0x02 */ u16 imageW; /* 0x04 */ s16 frameX; @@ -306,15 +310,15 @@ typedef struct __anon_0x5F05A { /* 0x1E */ u16 scaleH; /* 0x20 */ s32 imageYorig; /* 0x24 */ u8 padding[4]; -} __anon_0x5F05A; // size = 0x28 +} uObjScaleBg_t; // size = 0x28 -typedef union __anon_0x5F2FB { - __anon_0x5ED4F b; - __anon_0x5F05A s; +typedef union uObjBg { + uObjBg_t b; + uObjScaleBg_t s; s64 force_structure_alignment; -} __anon_0x5F2FB; +} uObjBg; -typedef struct __anon_0x5F429 { +typedef struct uObjSprite_t { /* 0x00 */ s16 objX; /* 0x02 */ u16 scaleW; /* 0x04 */ u16 imageW; @@ -329,16 +333,56 @@ typedef struct __anon_0x5F429 { /* 0x15 */ u8 imageSiz; /* 0x16 */ u8 imagePal; /* 0x17 */ u8 imageFlags; -} __anon_0x5F429; // size = 0x18 +} uObjSprite_t; // size = 0x18 -typedef union __anon_0x5F63B { - __anon_0x5F429 s; +typedef union uObjSprite { + uObjSprite_t s; + s64 force_structure_alignment; +} uObjSprite; + +typedef struct uObjTxtrBlock_t { + /* 0x00 */ u32 type; + /* 0x04 */ u32 image; + /* 0x08 */ u16 tmem; + /* 0x0A */ u16 tsize; + /* 0x0C */ u16 tline; + /* 0x0E */ u16 sid; + /* 0x10 */ u32 flag; + /* 0x14 */ u32 mask; +} uObjTxtrBlock_t; // size = 0x18 + +typedef struct uObjTxtrTile_t { + /* 0x00 */ u32 type; + /* 0x04 */ u32 image; + /* 0x08 */ u16 tmem; + /* 0x0A */ u16 twidth; + /* 0x0C */ u16 theight; + /* 0x0E */ u16 sid; + /* 0x10 */ u32 flag; + /* 0x14 */ u32 mask; +} uObjTxtrTile_t; // size = 0x18 + +typedef struct uObjTxtrTLUT_t { + /* 0x00 */ u32 type; + /* 0x04 */ u32 image; + /* 0x08 */ u16 phead; + /* 0x0A */ u16 pnum; + /* 0x0C */ u16 zero; + /* 0x0E */ u16 sid; + /* 0x10 */ u32 flag; + /* 0x14 */ u32 mask; +} uObjTxtrTLUT_t; // size = 0x18 + +typedef union uObjTxtr { + uObjTxtrBlock_t block; + uObjTxtrTile_t tile; + uObjTxtrTLUT_t tlut; s64 force_structure_alignment; -} __anon_0x5F63B; +} uObjTxtr; -bool rspFillObjBg(Rsp* pRSP, s32 nAddress, __anon_0x5F2FB* pBg); +bool rspFillObjBg(Rsp* pRSP, s32 nAddress, uObjBg* pBg); bool rspSetImage(struct Frame* pFrame, Rsp* pRSP, s32 nFormat, s32 nWidth, s32 nSize, s32 nImage); -bool rspFillObjBgScale(Rsp* pRSP, s32 nAddress, __anon_0x5F2FB* pBg); +bool rspFillObjBgScale(Rsp* pRSP, s32 nAddress, uObjBg* pBg); bool rspPut32(Rsp* pRSP, u32 nAddress, s32* pData); bool rspGet32(Rsp* pRSP, u32 nAddress, s32* pData); bool rspInvalidateCache(Rsp* pRSP, s32 nOffset0, s32 nOffset1); diff --git a/src/emulator/frame.c b/src/emulator/frame.c index 2f566328..d7c30902 100644 --- a/src/emulator/frame.c +++ b/src/emulator/frame.c @@ -3247,7 +3247,7 @@ bool frameHackCIMG_Zelda2_Shrink(Rdp* pRDP, Frame* pFrame, u64** ppnGBI) { u32 nCommandHi; Rsp* pRSP; s32 done; - union __anon_0x5F2FB bg; + uObjBg bg; pnGBI = *ppnGBI; for (count = 0; count < ARRAY_COUNT(GBIcode); count++) { @@ -3960,7 +3960,7 @@ static bool frameLoadTile(Frame* pFrame, FrameTexture** ppTexture, s32 iTileCode return true; } -bool frameDrawReset(Frame* pFrame, s32 nFlag) { +static inline bool frameDrawReset(Frame* pFrame, s32 nFlag) { pFrame->nFlag |= nFlag; pFrame->aDraw[0] = (FrameDrawFunc)frameDrawLine_Setup; pFrame->aDraw[1] = (FrameDrawFunc)frameDrawTriangle_Setup; @@ -4600,7 +4600,266 @@ bool frameLoadTLUT(Frame* pFrame, s32 nCount, s32 iTile) { return true; } -#pragma GLOBAL_ASM("asm/non_matchings/frame/frameLoadTMEM.s") +#define G_IM_SIZ_4b 0 +#define G_IM_SIZ_8b 1 +#define G_IM_SIZ_16b 2 +#define G_IM_SIZ_32b 3 + +bool frameLoadTMEM(Frame* pFrame, FrameLoadType eType, s32 iTile) { + // Parameters + // struct __anon_0x24C38* pFrame; // r29 + // enum __anon_0x26C3F eType; // r30 + // s32 iTile; // r31 + + // Local variables + bool bFlip; // r10 + s32 iTMEM; // r5 + s32 nSize; // r6 + s32 nStep; // r11 + s32 nDelta; // r12 + s32 iScan; // r12 + s32 nOffset; // r7 + Tile* pTile; // r1+0x8 + u8 nData8; // r30 + u16 nData16; // r30 + u32 nData32; // r30 + u32 nSum; // r1+0x8 + u64* pSource; // r4 + s32 nCount; // r6 + s32 nScanFull; // r7 + s32 nScanPart; // r8 + u8* pSource8; // r31 + u16* pSource16; // r31 + u32* pSource32; // r31 + + // TODO: what are these temps? + u32 a, b; + s32 temp; + s32 var_r0; + int i; + s32 temp2; + + frameDrawReset(pFrame, 1); + + if (gpSystem->eTypeROM == SRT_ZELDA2 && (pFrame->aBuffer[FBT_IMAGE].nAddress == 0x500 || pFrame->aBuffer[FBT_IMAGE].nAddress == 0x7DA800)) { + CopyAndConvertCFB(pFrame->aBuffer[FBT_IMAGE].pData); + } + + pSource = (u64*)((u32)pFrame->aBuffer[FBT_IMAGE].pData & ~3); + pFrame->iTileLoad = iTile; + pTile = &pFrame->aTile[iTile]; + iTMEM = pTile->nTMEM & 0x1FF; + if (eType == FLT_BLOCK) { + var_r0 = 0; + } else { + var_r0 = 2; + } + + switch (pFrame->aBuffer[FBT_IMAGE].nSize) { + case G_IM_SIZ_4b: + nSize = (pTile->nX1 + 1) >> 4; + nOffset = ((pTile->nX0 >> var_r0) >> 1) + ((pFrame->aBuffer[FBT_IMAGE].nWidth + 1) >> 1) * (pTile->nY0 >> var_r0); + break; + case G_IM_SIZ_8b: + nSize = (pTile->nX1 + 1) >> 3; + nOffset = (pTile->nX0 >> var_r0) + pFrame->aBuffer[FBT_IMAGE].nWidth * (pTile->nY0 >> var_r0); + break; + case G_IM_SIZ_16b: + nSize = (pTile->nX1 + 1) >> 2; + nOffset = ((pTile->nX0 >> var_r0) << 1) + (pFrame->aBuffer[FBT_IMAGE].nWidth << 1) * (pTile->nY0 >> var_r0); + break; + case G_IM_SIZ_32b: + nSize = (pTile->nX1 + 1) >> 1; + nOffset = ((pTile->nX0 >> var_r0) << 2) + (pFrame->aBuffer[FBT_IMAGE].nWidth << 2) * (pTile->nY0 >> var_r0); + break; + default: + return false; + } + + pSource = (u64*)((u8*)pSource + nOffset); + pFrame->nAddressLoad = (pFrame->aBuffer[FBT_IMAGE].nAddress & 0x7FFFFF) + nOffset; + nSum = 0; + if (eType == FLT_BLOCK) { + nDelta = pTile->nY1; + if (nDelta == 0) { + for (i = 0; i != nSize; i++) { + a = ((u32*)pSource)[0]; + b = ((u32*)pSource)[1]; + nSum += a; + nSum += b; + nSum ^= iTMEM; + pFrame->TMEM.data.u32[2 * iTMEM + 0] = a; + pFrame->TMEM.data.u32[2 * iTMEM + 1] = b; + pSource++; + iTMEM = (iTMEM + 1) & 0x1FF; + } + } else { + nStep = 0; + while (nSize != 0) { + while (nSize != 0 && (nStep & 0x800) == 0) { + a = ((u32*)pSource)[0]; + b = ((u32*)pSource)[1]; + nSum += a; + nSum += b; + nSum ^= iTMEM; + pFrame->TMEM.data.u32[2 * iTMEM + 0] = a; + pFrame->TMEM.data.u32[2 * iTMEM + 1] = b; + pSource++; + iTMEM = (iTMEM + 1) & 0x1FF; + nStep += nDelta; + nSize -= 1; + } + nStep -= 0x800; + if (pFrame->aBuffer[FBT_IMAGE].nSize == 3) { + bFlip = 0; + while (nSize != 0 && (nStep & 0x800) == 0) { + a = ((u32*)pSource)[0]; + b = ((u32*)pSource)[1]; + nSum += a; + nSum += b; + nSum ^= iTMEM; + pSource++; + temp = iTMEM + (bFlip ? -1 : 1); + pFrame->TMEM.data.u32[2 * temp + 0] = a; + pFrame->TMEM.data.u32[2 * temp + 1] = b; + iTMEM = (iTMEM + 1) & 0x1FF; + nStep += nDelta; + nSize -= 1; + bFlip ^= 1; + } + } else { + while (nSize != 0 && (nStep & 0x800) == 0) { + a = ((u32*)pSource)[0]; + b = ((u32*)pSource)[1]; + nSum += a; + nSum += b; + nSum ^= iTMEM; + pSource++; + pFrame->TMEM.data.u32[2 * iTMEM + 0] = a; + pFrame->TMEM.data.u32[2 * iTMEM + 1] = b; + iTMEM = (iTMEM + 1) & 0x1FF; + nStep += nDelta; + nSize -= 1; + } + } + nStep -= 0x800; + } + } + } else { + nCount = ((pTile->nY1 - pTile->nY0) + 4) >> 2; + nScanFull = pFrame->aBuffer[FBT_IMAGE].nWidth; + nScanPart = ((pTile->nX1 - pTile->nX0) + 4) >> 2; + switch (pFrame->aBuffer[FBT_IMAGE].nSize) { + case G_IM_SIZ_4b: + case G_IM_SIZ_8b: + if (nScanPart >= pTile->nSizeX * 8) { + nScanPart = pTile->nSizeX * 8; + } + iTMEM <<= 3; + for (iScan = 0; iScan < nCount; iScan++) { + pSource8 = (u8*)pSource; + i = 0; + if (iScan & 1) { + while (i != nScanPart) { + nData8 = *pSource8; + nSum += nData8; + nSum ^= iTMEM; + pFrame->TMEM.data.u8[iTMEM ^ 4] = nData8; + pSource8++; + iTMEM = (iTMEM + 1) & 0xFFF; + i++; + } + } else { + while (i != nScanPart) { + nData8 = *pSource8; + nSum += nData8; + nSum ^= iTMEM; + pFrame->TMEM.data.u8[iTMEM] = nData8; + pSource8++; + iTMEM = (iTMEM + 1) & 0xFFF; + i++; + } + } + pSource = (u64*)((u8*)pSource + nScanFull); + iTMEM += (pTile->nSizeX * 8) - nScanPart; + iTMEM &= 0xFFF; + } + break; + case G_IM_SIZ_16b: + if (nScanPart >= pTile->nSizeX * 4) { + nScanPart = pTile->nSizeX * 4; + } + iTMEM <<= 2; + for (iScan = 0; iScan < nCount; iScan++) { + pSource16 = (u16*)pSource; + i = 0; + if (iScan & 1) { + while (i != nScanPart) { + nData16 = *pSource16; + nSum += nData16; + nSum ^= iTMEM; + pFrame->TMEM.data.u16[iTMEM ^ 2] = nData16; + pSource16++; + iTMEM = (iTMEM + 1) & 0x7FF; + i++; + } + } else { + while (i != nScanPart) { + nData16 = *pSource16; + nSum += nData16; + nSum ^= iTMEM; + pFrame->TMEM.data.u16[iTMEM] = nData16; + pSource16++; + iTMEM = (iTMEM + 1) & 0x7FF; + i++; + } + } + pSource = (u64*)((u16*)pSource + nScanFull); + iTMEM = (iTMEM + ((pTile->nSizeX * 4) - nScanPart)) & 0x7FF; + } + break; + case G_IM_SIZ_32b: + if (nScanPart >= pTile->nSizeX * 4) { + nScanPart = pTile->nSizeX * 4; + } + iTMEM <<= 1; + for (iScan = 0; iScan < nCount; iScan++) { + pSource32 = (u32*)pSource; + i = 0; + if (iScan & 1) { + while (i != nScanPart) { + nData32 = *pSource32; + nSum += nData32; + nSum ^= iTMEM; + pFrame->TMEM.data.u32[iTMEM ^ 2] = nData32; + pSource32++; + iTMEM = (iTMEM + 1) & 0x3FF; + i++; + } + } else { + while (i != nScanPart) { + nData32 = *pSource32; + nSum += nData32; + nSum ^= iTMEM; + pFrame->TMEM.data.u32[iTMEM] = nData32; + pSource32++; + iTMEM = (iTMEM + 1) & 0x3FF; + i++; + } + } + pSource = (u64*)((u32*)pSource + nScanFull); + iTMEM += (pTile->nSizeX * 4) - nScanPart; + iTMEM &= 0x3FF; + } + break; + default: + return false; + } + } + + pFrame->nCodePixel = nSum; + return true; +} bool frameSetLightCount(Frame* pFrame, s32 nCount) { pFrame->nCountLight = nCount; diff --git a/src/emulator/rdp.c b/src/emulator/rdp.c index bf3f9c61..a1ccc2a6 100644 --- a/src/emulator/rdp.c +++ b/src/emulator/rdp.c @@ -361,7 +361,7 @@ bool rdpParseGBI(Rdp* pRDP, u64** ppnGBI, RspUCodeType eTypeUCode) { pFrame->aTile[iTile].nX1 = (nCommandLo >> 12) & 0xFFF; pFrame->aTile[iTile].nY1 = nCommandLo & 0xFFF; - pFrame->n2dLoadTexType = 0xFC1034; + pFrame->n2dLoadTexType = G_OBJLT_TXTRTILE; pFrame->nLastX0 = pFrame->aTile[iTile].nX0; pFrame->nLastY0 = pFrame->aTile[iTile].nY0; pFrame->nLastX1 = pFrame->aTile[iTile].nX1; @@ -379,7 +379,7 @@ bool rdpParseGBI(Rdp* pRDP, u64** ppnGBI, RspUCodeType eTypeUCode) { pFrame->aTile[iTile].nY0 = nCommandHi & 0xFFF; pFrame->aTile[iTile].nX1 = (nCommandLo >> 12) & 0xFFF; pFrame->aTile[iTile].nY1 = nCommandLo & 0xFFF; - pFrame->n2dLoadTexType = 0x1033; + pFrame->n2dLoadTexType = G_OBJLT_TXTRBLOCK; if (!frameLoadTMEM(pFrame, FLT_BLOCK, iTile)) { return false; } diff --git a/src/emulator/rsp.c b/src/emulator/rsp.c index e12b7d32..29976ecc 100644 --- a/src/emulator/rsp.c +++ b/src/emulator/rsp.c @@ -158,8 +158,23 @@ static bool nFirstTime_2648 = true; static bool nFirstTime_2757 = true; static bool nFirstTime_2796 = true; -static u16 scissorX1 = 0x500; -static u16 scissorY1 = 0x3C0; +static s32 counter; + +// TODO: initialize as 0 << 2 +static u16 scissorX0; +static u16 scissorY0; +static u16 scissorX1 = N64_FRAME_WIDTH << 2; +static u16 scissorY1 = N64_FRAME_HEIGHT << 2; + +static u8 flagBilerp; +static u32 rdpSetTimg_w0; +static u32 rdpSetTile_w0; +static u16 tmemSliceWmax; +static u16 imageSrcWsize; +static u16 flagSplit; +static u16 imagePtrX0; +static u32 imageTop; +static s16 tmemSrcLines; static s16 TMEMMASK[4] = { 0x01FF, @@ -175,19 +190,6 @@ static s16 TMEMSHIFT[4] = { 0x0040, }; -static s32 counter; -static u16 scissorX0; -static u16 scissorY0; -static u8 flagBilerp; -static u32 rdpSetTimg_w0; -static u32 rdpSetTile_w0; -static u16 tmemSliceWmax; -static u16 imageSrcWsize; -static u16 flagSplit; -static u16 imagePtrX0; -static s32 imageTop; -static u16 tmemSrcLines; - const f32 D_80136038 = 0.25f; const f32 D_8013603C = 1024.0f; const f32 D_80136040 = 0.03125f; @@ -4757,7 +4759,7 @@ static bool Matrix4by4Identity(Mtx44Ptr matrix4b4) { } #endif -static bool rspFillObjSprite(Rsp* pRSP, s32 nAddress, __anon_0x5F63B* pSprite) { +static bool rspFillObjSprite(Rsp* pRSP, s32 nAddress, uObjSprite* pSprite) { u16* pnData16; u8* pnData8; u8* pObjSprite; @@ -4787,7 +4789,7 @@ static bool rspFillObjSprite(Rsp* pRSP, s32 nAddress, __anon_0x5F63B* pSprite) { return true; } -bool rspFillObjBgScale(Rsp* pRSP, s32 nAddress, __anon_0x5F2FB* pBg) { +bool rspFillObjBgScale(Rsp* pRSP, s32 nAddress, uObjBg* pBg) { u8* pnData8; u8* pObjBg; u16* pnData16; @@ -4822,7 +4824,7 @@ bool rspFillObjBgScale(Rsp* pRSP, s32 nAddress, __anon_0x5F2FB* pBg) { return true; } -bool rspFillObjBg(Rsp* pRSP, s32 nAddress, __anon_0x5F2FB* pBg) { +bool rspFillObjBg(Rsp* pRSP, s32 nAddress, uObjBg* pBg) { u8* pnData8; u8* pObjBg; u16* pnData16; @@ -4860,6 +4862,24 @@ bool rspFillObjBg(Rsp* pRSP, s32 nAddress, __anon_0x5F2FB* pBg) { return true; } +static inline bool rspSetTile(Frame* pFrame, Tile* pTile, s32 nSize, s32 nTmem, s32 nTLUT, s32 nFormat, s32 nMaskS, + s32 nMaskT, s32 nModeS, s32 nModeT, s32 nShiftS, s32 nShiftT) { + pTile->nSize = nSize; + pTile->nTMEM = nTmem; + pTile->iTLUT = nTLUT; + pTile->nFormat = nFormat; + pTile->nMaskS = nMaskS; + pTile->nMaskT = nMaskT; + pTile->nModeS = nModeS; + pTile->nModeT = nModeT; + pTile->nShiftS = nShiftS; + pTile->nShiftT = nShiftT; + if (!frameDrawReset(pFrame, 1)) { + return false; + } + return true; +} + bool rspSetImage(Frame* pFrame, Rsp* pRSP, s32 nFormat, s32 nWidth, s32 nSize, s32 nImage) { FrameBuffer* pBuffer; s32 nAddr; @@ -4882,6 +4902,17 @@ bool rspSetImage(Frame* pFrame, Rsp* pRSP, s32 nFormat, s32 nWidth, s32 nSize, s return true; } +static inline bool rspSetTileSize(Frame* pFrame, Tile* pTile, s32 nX0, s32 nY0, s32 nX1, s32 nY1) { + pTile->nX0 = nX0; + pTile->nY0 = nY0; + pTile->nX1 = nX1; + pTile->nY1 = nY1; + if (!frameDrawReset(pFrame, 1)) { + return false; + } + return true; +} + static bool tmemLoad_B(Frame* pFrame, Rsp* pRSP, u32 imagePtr, s16 loadLines, s16 tmemSH) { FrameBuffer* pBuffer; s32 nAddr; @@ -4936,26 +4967,707 @@ static bool tmemLoad_A(Frame* pFrame, Rsp* pRSP, s32 imagePtr, s16 loadLines, s1 return true; } -#pragma GLOBAL_ASM("asm/non_matchings/rsp/tmemLoad.s") +// Similar to +// https://github.com/decompals/ultralib/blob/1616482098e51d2e1906e198bf1bde14e8fc5e90/src/gu/us2dex_emu.c#L97 +static bool tmemLoad(Frame* pFrame, Rsp* pRSP, u32* imagePtr, s16* imageRemain, s16 drawLines, s16 flagBilerp) { + s16 loadLines = drawLines + flagBilerp; + s16 iLoadable = *imageRemain - flagSplit; + + if (iLoadable >= loadLines) { + tmemLoad_B(pFrame, pRSP, *imagePtr, loadLines, tmemSliceWmax); + *imagePtr += imageSrcWsize * drawLines; + *imageRemain -= drawLines; + } else { + s16 SubSliceL2, SubSliceD2, SubSliceY2; + u32 imageTopSeg = imageTop & 0xFF000000; + + SubSliceY2 = *imageRemain; + SubSliceL2 = loadLines - SubSliceY2; + SubSliceD2 = drawLines - SubSliceY2; + + if (SubSliceL2 > 0) { + u32 imagePtr2 = imageTop + imagePtrX0; + + if (SubSliceY2 & 1) { + imagePtr2 -= imageSrcWsize; + imagePtr2 = imageTopSeg | (imagePtr2 & 0xFFFFFF); + SubSliceY2--; + SubSliceL2++; + } + tmemLoad_A(pFrame, pRSP, imagePtr2, SubSliceL2, SubSliceY2 * tmemSliceWmax, tmemSliceWmax); + } + if (flagSplit) { + u32 imagePtr1A, imagePtr1B; + s16 SubSliceY1, SubSliceL1; + s16 tmemSH_A, tmemSH_B; + + imagePtr1A = *imagePtr + iLoadable * imageSrcWsize; + imagePtr1B = imageTop; + SubSliceY1 = iLoadable; + + if (iLoadable & 1) { + imagePtr1A -= imageSrcWsize; + imagePtr1B -= imageSrcWsize; + imagePtr1B = imageTopSeg | (imagePtr1B & 0xFFFFFF); + SubSliceY1--; + SubSliceL1 = 2; + } else { + SubSliceL1 = 1; + } + tmemSH_A = (imageSrcWsize - imagePtrX0) >> 3; + tmemSH_B = tmemSliceWmax - tmemSH_A; + tmemLoad_A(pFrame, pRSP, imagePtr1B, SubSliceL1, SubSliceY1 * tmemSliceWmax + tmemSH_A, tmemSH_B); + tmemLoad_A(pFrame, pRSP, imagePtr1A, SubSliceL1, SubSliceY1 * tmemSliceWmax, tmemSH_A); + } + + if (iLoadable > 0) { + tmemLoad_A(pFrame, pRSP, *imagePtr, iLoadable, 0, tmemSliceWmax); + } else { + pFrame->aTile[7].nSize = 2; + pFrame->aTile[7].nTMEM = 0; + pFrame->aTile[7].iTLUT = 0; + pFrame->aTile[7].nSizeX = tmemSliceWmax; + pFrame->aTile[7].nFormat = 0; + pFrame->aTile[7].nMaskS = 0; + pFrame->aTile[7].nMaskT = 0; + pFrame->aTile[7].nModeS = 0; + pFrame->aTile[7].nModeT = 0; + pFrame->aTile[7].nShiftS = 0; + pFrame->aTile[7].nShiftT = 0; + + if (!frameDrawReset(pFrame, 0x1)) { + return false; + } + } + + *imageRemain -= drawLines; + if (*imageRemain > 0) { + *imagePtr += imageSrcWsize * drawLines; + } else { + *imageRemain = tmemSrcLines - SubSliceD2; + *imagePtr = imageTop + SubSliceD2 * imageSrcWsize + imagePtrX0; + } + } + + return true; +} + +static inline bool guS2DEmuSetScissor(u32 ulx, u32 uly, u32 lrx, u32 lry, u8 flag) { + scissorX0 = ulx; + scissorY0 = uly; + scissorX1 = lrx; + scissorY1 = lry; + flagBilerp = flag; + return true; +} +// Matches but data doesn't +#ifndef NON_MATCHING #pragma GLOBAL_ASM("asm/non_matchings/rsp/guS2DEmuBgRect1Cyc.s") +#else +// Similar to +// https://github.com/decompals/ultralib/blob/1616482098e51d2e1906e198bf1bde14e8fc5e90/src/gu/us2dex_emu.c#L177 +static bool guS2DEmuBgRect1Cyc(Rsp* pRSP, Frame* pFrame, uObjBg* pBG) { + s32 pad; + + s16 frameX0, frameX1, framePtrY0, frameRemain; + s16 imageX0, imageY0, imageSliceW, imageW; + s32 imageYorig; + s16 scaleW, scaleH; + + s16 imageSrcW, imageSrcH; + s16 tmemSliceLines, imageSliceLines; + s32 frameSliceLines, frameSliceCount; + u16 imageS, imageT; + u32 imagePtr; + + s16 imageISliceL0, imageIY0; + s32 frameLSliceL0; + + scaleW = pBG->s.scaleW; + scaleH = pBG->s.scaleH; + + { + s16 pixX0, pixY0, pixX1, pixY1; + s16 frameY0, frameW, frameH; + s32 frameWmax, frameHmax; + + frameWmax = (((pBG->s.imageW << 10) / scaleW) - 1) & ~3; + frameHmax = (((pBG->s.imageH << 10) / scaleH) - 1) & ~3; + + frameW = pBG->s.frameW; + frameH = pBG->s.frameH; + frameX0 = pBG->s.frameX; + frameY0 = pBG->s.frameY; + + if ((frameWmax = pBG->s.frameW - frameWmax) < 0) { + frameWmax = 0; + } + if ((frameHmax = pBG->s.frameH - frameHmax) < 0) { + frameHmax = 0; + } + + frameW -= (s16)frameWmax; + frameH -= (s16)frameHmax; + + if (pBG->s.imageFlip & 1) { + frameX0 += (s16)frameWmax; + } + + pixX0 = scissorX0 - frameX0; + pixY0 = scissorY0 - frameY0; + pixX1 = frameW - scissorX1 + frameX0; + pixY1 = frameH - scissorY1 + frameY0; + + if (pixX0 < 0) { + pixX0 = 0; + } + if (pixY0 < 0) { + pixY0 = 0; + } + if (pixX1 < 0) { + pixX1 = 0; + } + if (pixY1 < 0) { + pixY1 = 0; + } + + frameW = frameW - (pixX0 + pixX1); + frameH = frameH - (pixY0 + pixY1); + frameX0 = frameX0 + pixX0; + frameY0 = frameY0 + pixY0; + + if (frameW <= 0 || frameH <= 0) { + return true; + } + + frameX1 = frameX0 + frameW; + framePtrY0 = frameY0 >> 2; + frameRemain = frameH >> 2; + + imageSrcW = pBG->s.imageW << 3; + imageSrcH = pBG->s.imageH << 3; + + imageSliceW = (imageW = frameW * scaleW >> 7) + flagBilerp * 32; + if (pBG->s.imageFlip & 1) { + imageX0 = pBG->s.imageX + (pixX1 * scaleW >> 7); + } else { + imageX0 = pBG->s.imageX + (pixX0 * scaleW >> 7); + } + imageY0 = pBG->s.imageY + (pixY0 * scaleH >> 7); + imageYorig = pBG->s.imageYorig; + + while (imageX0 >= imageSrcW) { + imageX0 -= imageSrcW; + imageY0 += 32; + imageYorig += 32; + } + + while (imageY0 >= imageSrcH) { + imageY0 -= imageSrcH; + imageYorig -= imageSrcH; + } + } + + { + flagSplit = (imageX0 + imageSliceW >= imageSrcW); + tmemSrcLines = imageSrcH >> 5; + } + + { + s16 tmemSize, tmemMask, tmemShift; + s32 imageNumSlice; + s32 imageSliceWmax; + s32 imageLYoffset, frameLYoffset; + s32 imageLHidden, frameLHidden; + s32 frameLYslice; + // TODO: make in-function static + // static s16 TMEMSIZE[] = { 512, 512, 256, 512, 512 }; + // static s16 TMEMMASK[] = { 0x1FF, 0xFF, 0x7F, 0x3F }; + // static s16 TMEMSHIFT[] = { 0x200, 0x100, 0x80, 0x40 }; + + tmemSize = TMEMSIZE[pBG->s.imageFmt]; + tmemMask = TMEMMASK[pBG->s.imageSiz]; + tmemShift = TMEMSHIFT[pBG->s.imageSiz]; + + imageSliceWmax = ((pBG->s.frameW * scaleW) >> 7) + (flagBilerp << 5); + if (imageSliceWmax > imageSrcW) { + imageSliceWmax = imageSrcW; + } + + tmemSliceWmax = (imageSliceWmax + tmemMask) / tmemShift + 1; + tmemSliceLines = tmemSize / tmemSliceWmax; + imageSliceLines = tmemSliceLines - flagBilerp; + frameSliceLines = (imageSliceLines << 20) / scaleH; + + imageLYoffset = (imageY0 - imageYorig) << 5; + if (imageLYoffset < 0) { + imageLYoffset -= (scaleH - 1); + } + frameLYoffset = imageLYoffset / scaleH; + frameLYoffset <<= 10; + + if (frameLYoffset >= 0) { + imageNumSlice = frameLYoffset / frameSliceLines; + } else { + imageNumSlice = (frameLYoffset - frameSliceLines + 1) / frameSliceLines; + } + + frameLYslice = (frameLSliceL0 = frameSliceLines * imageNumSlice) & ~1023; + frameLHidden = frameLYoffset - frameLYslice; + imageLHidden = (frameLHidden >> 10) * scaleH; + + frameLSliceL0 = (frameLSliceL0 & 1023) + frameSliceLines - frameLHidden; + + imageT = (imageLHidden >> 5) & 31; + imageLHidden >>= 10; + imageISliceL0 = imageSliceLines - (s16)imageLHidden; + imageIY0 = imageSliceLines * imageNumSlice + (imageYorig & ~31) / 32 + imageLHidden; + if (imageIY0 < 0) { + imageIY0 += (pBG->s.imageH >> 2); + } + if (imageIY0 >= (pBG->s.imageH >> 2)) { + imageIY0 -= (pBG->s.imageH >> 2); + } + + imageTop = (u32)pBG->s.imagePtr; + imageSrcWsize = (imageSrcW / tmemShift) << 3; + imagePtrX0 = (imageX0 / tmemShift) << 3; + imagePtr = imageTop + imageSrcWsize * imageIY0 + imagePtrX0; + + imageS = imageX0 & tmemMask; + if (pBG->s.imageFlip & 1) { + imageS = -(imageS + imageW); + } + } + + { + rdpSetTimg_w0 = 0x100000 + (imageSrcWsize >> 1) - 1; + rdpSetTile_w0 = 0x100000 + (tmemSliceWmax << 9); + + pFrame->aTile[7].nSize = 2; + pFrame->aTile[7].nTMEM = 0; + pFrame->aTile[7].iTLUT = 0; + pFrame->aTile[7].nSizeX = tmemSliceWmax; + pFrame->aTile[7].nFormat = 0; + pFrame->aTile[7].nMaskS = 0; + pFrame->aTile[7].nMaskT = 0; + pFrame->aTile[7].nModeS = 0; + pFrame->aTile[7].nModeT = 0; + pFrame->aTile[7].nShiftS = 0; + pFrame->aTile[7].nShiftT = 0; + if (!frameDrawReset(pFrame, 1)) { + return false; + } + + pFrame->aTile[0].nSize = pBG->s.imageSiz; + pFrame->aTile[0].nTMEM = 0; + pFrame->aTile[0].iTLUT = pBG->s.imagePal; + pFrame->aTile[0].nSizeX = tmemSliceWmax; + pFrame->aTile[0].nFormat = pBG->s.imageFmt; + pFrame->aTile[0].nMaskS = 0xF; + pFrame->aTile[0].nMaskT = 0xF; + pFrame->aTile[0].nModeS = 1; + pFrame->aTile[0].nModeT = 1; + pFrame->aTile[0].nShiftS = 0; + pFrame->aTile[0].nShiftT = 0; + if (!frameDrawReset(pFrame, 1)) { + return false; + } + + pFrame->aTile[0].nX0 = 0; + pFrame->aTile[0].nY0 = 0; + pFrame->aTile[0].nX1 = 0; + pFrame->aTile[0].nY1 = 0; + if (!frameDrawReset(pFrame, 1)) { + return false; + } + } + + { + s16 imageRemain; + s16 imageSliceH, frameSliceH; + + imageRemain = tmemSrcLines - imageIY0; + imageSliceH = imageISliceL0; + frameSliceCount = frameLSliceL0; + + while (true) { + frameSliceH = frameSliceCount >> 10; + if (frameSliceH <= 0) { + imageRemain -= imageSliceH; + if (imageRemain > 0) { + imagePtr += imageSrcWsize * imageSliceH; + } else { + imagePtr = imageTop - imageRemain * imageSrcWsize + imagePtrX0; + imageRemain += tmemSrcLines; + } + } else { + Rectangle primitive; + s16 nS, nT; + s16 framePtrY1; + + frameSliceCount &= 1023; + frameRemain -= frameSliceH; + if (frameRemain < 0) { + frameSliceH += frameRemain; + imageSliceH += (frameRemain * scaleH >> 10) + 1; + if (imageSliceH > imageSliceLines) { + imageSliceH = imageSliceLines; + } + } + tmemLoad(pFrame, pRSP, &imagePtr, &imageRemain, imageSliceH, flagBilerp); + + framePtrY1 = framePtrY0 + frameSliceH; -#pragma GLOBAL_ASM("asm/non_matchings/rsp/rspFillObjTxtr.s") + primitive.iTile = 0; + primitive.bFlip = false; + nS = imageS - 8 * pFrame->aTile[primitive.iTile].nX0; + nT = imageT - 8 * pFrame->aTile[primitive.iTile].nY0; + + primitive.rS = nS / 32.0f; + primitive.rT = nT / 32.0f; + primitive.rDeltaS = scaleW / 1024.0f; + primitive.rDeltaT = scaleH / 1024.0f; + if (pBG->s.imageFlip & 1) { + primitive.rS *= -1.0f; + primitive.rDeltaS *= -1.0f; + } + primitive.nX0 = frameX0 << 0; + primitive.nY0 = framePtrY0 << 2; + primitive.nX1 = frameX1 << 0; + primitive.nY1 = framePtrY1 << 2; + if (!pFrame->aDraw[3](pFrame, &primitive)) { + return false; + } + + framePtrY0 = framePtrY1; + if (frameRemain <= 0) { + return true; + } + } + + frameSliceCount += frameSliceLines; + imageSliceH = imageSliceLines; + imageT = 0; + } + } +} +#endif + +bool rspFillObjTxtr(Rsp* pRSP, s32 nAddress, uObjTxtr* pTxtr, u32* pLoadType) { + u32* pnData32; + u16* pnData16; + u8* pTxtrBlock; + u32 nLoadType; + + if (!ramGetBuffer(SYSTEM_RAM(pRSP->pHost), &pTxtrBlock, nAddress, NULL)) { + return false; + } + + pnData32 = (u32*)pTxtrBlock; + pnData16 = (u16*)pTxtrBlock; + nLoadType = pnData32[0]; + + switch (nLoadType) { + case G_OBJLT_TXTRBLOCK: + pTxtr->block.type = nLoadType; + pTxtr->block.image = pnData32[1]; + pTxtr->block.sid = pnData16[7]; + pTxtr->block.flag = pnData32[4]; + pTxtr->block.mask = pnData32[5]; + pTxtr->block.tmem = pnData16[4]; + pTxtr->block.tsize = pnData16[5]; + pTxtr->block.tline = pnData16[6]; + break; + case G_OBJLT_TXTRTILE: + pTxtr->tile.type = nLoadType; + pTxtr->tile.image = pnData32[1]; + pTxtr->tile.sid = pnData16[7]; + pTxtr->tile.flag = pnData32[4]; + pTxtr->tile.mask = pnData32[5]; + pTxtr->tile.tmem = pnData16[4]; + pTxtr->tile.twidth = pnData16[5]; + pTxtr->tile.theight = pnData16[6]; + break; + case G_OBJLT_TLUT: + pTxtr->tlut.type = nLoadType; + pTxtr->tlut.image = pnData32[1]; + pTxtr->tlut.sid = pnData16[7]; + pTxtr->tlut.flag = pnData32[4]; + pTxtr->tlut.mask = pnData32[5]; + pTxtr->tlut.phead = pnData16[4]; + pTxtr->tlut.pnum = pnData16[5]; + break; + default: + return false; + } + + if (pLoadType != NULL) { + *pLoadType = nLoadType; + } + return true; +} + +static bool rspObjLoadTxtr(Rsp* pRSP, Frame* pFrame, s32 nAddress); #pragma GLOBAL_ASM("asm/non_matchings/rsp/rspObjLoadTxtr.s") +// Matches but data doesn't +#ifndef NON_MATCHING #pragma GLOBAL_ASM("asm/non_matchings/rsp/rspObjRectangle.s") +#else +static bool rspObjRectangle(Rsp* pRSP, Frame* pFrame, s32 nAddress) { + s32 pad; + u16 nSizLineBytes; + f32 fDeltaS; + f32 fDeltaT; + uObjSprite objSprite; + Tile* pTile; + Rectangle primitive; + s32 nClampSetting; + s32 nTexTrim2; + s32 nTexTrim5; + + nTexTrim2 = 0; + nTexTrim5 = 0; + pTile = &pFrame->aTile[0]; + if (!rspFillObjSprite(pRSP, nAddress, &objSprite)) { + return false; + } + + nClampSetting = pRSP->nMode2D & 1; + if (pRSP->nMode2D & 0x10) { + nTexTrim2 = -2; + nTexTrim5 = -16; + } else if (pRSP->nMode2D & 0x20) { + nTexTrim2 = -4; + nTexTrim5 = -32; + } + if (pRSP->nMode2D & 0x40) { + nTexTrim2 += 1; + nTexTrim5 += 12; + } + objSprite.s.imageW += nTexTrim5; + objSprite.s.imageH += nTexTrim5; + pFrame->nLastX1 += nTexTrim2; + if (pTile->nSize != 0) { + switch (objSprite.s.imageSiz) { + case 3: + nSizLineBytes = 2; + break; + case 2: + nSizLineBytes = 2; + break; + case 1: + nSizLineBytes = 1; + break; + default: + return false; + } + if (pFrame->n2dLoadTexType == G_OBJLT_TXTRBLOCK) { + pTile->nSizeX = objSprite.s.imageStride; + } else if (pFrame->n2dLoadTexType == G_OBJLT_TXTRTILE) { + pTile->nSizeX = (((pFrame->nLastX1 - pFrame->nLastX0 + 1) >> 5) * nSizLineBytes + 7) >> 3; + } + } else { + if (pFrame->n2dLoadTexType == G_OBJLT_TXTRBLOCK) { + pTile->nSizeX = ((objSprite.s.imageW >> 6) + 7) >> 3; + } else if (pFrame->n2dLoadTexType == G_OBJLT_TXTRTILE) { + pTile->nSizeX = (((pFrame->nLastX1 - pFrame->nLastX0) >> 6) + 7) >> 3; + } + } + + if (!rspSetTile(pFrame, pTile, objSprite.s.imageSiz, objSprite.s.imageAdrs, objSprite.s.imagePal, + objSprite.s.imageFmt, 0, 0, nClampSetting, nClampSetting, 0, 0)) { + return false; + } + if (pFrame->n2dLoadTexType == G_OBJLT_TXTRBLOCK) { + if (!rspSetTileSize(pFrame, pTile, 0, 0, ((objSprite.s.imageW >> 5) - 1) * 4, + ((objSprite.s.imageH >> 5) - 1) * 4)) { + return false; + } + } else if (pFrame->n2dLoadTexType == G_OBJLT_TXTRTILE) { + if (!rspSetTileSize(pFrame, pTile, (pFrame->nLastX0 >> 5) * 4, (pFrame->nLastY0 >> 5) * 4, + (pFrame->nLastX1 >> 5) * 4, (pFrame->nLastY1 >> 5) * 4)) { + return false; + } + } + + primitive.nX0 = objSprite.s.objX; + primitive.nY0 = objSprite.s.objY; + primitive.nX1 = + ((s32)(((objSprite.s.imageW - 1.0f) * (1024.0f / objSprite.s.scaleW)) + 8 * objSprite.s.objX) >> 3) - 2; + primitive.nY1 = + ((s32)(((objSprite.s.imageH - 1.0f) * (1024.0f / objSprite.s.scaleH)) + 8 * objSprite.s.objY) >> 3) - 2; + primitive.iTile = 0; + primitive.bFlip = false; + fDeltaS = objSprite.s.scaleW / 1024.0f; + fDeltaT = objSprite.s.scaleH / 1024.0f; + if (objSprite.s.imageFlags & 1) { + primitive.rS = (u16)(objSprite.s.imageW - 8 * pFrame->aTile[primitive.iTile].nX0) / 32.0f; + primitive.rDeltaS = -fDeltaS; + } else { + primitive.rS = (u16)(-8 * pFrame->aTile[primitive.iTile].nX0) / 32.0f; + primitive.rDeltaS = fDeltaS; + } + if (objSprite.s.imageFlags & 0x10) { + primitive.rT = (u16)(objSprite.s.imageH - 8 * pFrame->aTile[primitive.iTile].nY0) / 32.0f; + primitive.rDeltaT = -fDeltaT; + } else { + primitive.rT = (u16)(-8 * pFrame->aTile[primitive.iTile].nY0) / 32.0f; + primitive.rDeltaT = fDeltaT; + } + + if (!pFrame->aDraw[3](pFrame, &primitive)) { + return false; + } + return true; +} +#endif #pragma GLOBAL_ASM("asm/non_matchings/rsp/rspObjSprite.s") #pragma GLOBAL_ASM("asm/non_matchings/rsp/rspObjRectangleR.s") -#pragma GLOBAL_ASM("asm/non_matchings/rsp/rspBgRectCopy.s") +bool rspBgRectCopy(Rsp* pRSP, Frame* pFrame, s32 nAddress) { + uObjBg bg; + uObjBg bgScale; + u32 nOldMode1; + u32 nOldMode2; + + rspFillObjBg(pRSP, nAddress, &bg); + bgScale.s.frameH = bg.b.frameH; + bgScale.s.frameW = bg.b.frameW; + bgScale.s.frameX = bg.b.frameX; + bgScale.s.frameY = bg.b.frameY; + bgScale.s.imageFlip = bg.b.imageFlip; + bgScale.s.imageFmt = bg.b.imageFmt; + bgScale.s.imageH = bg.b.imageH; + bgScale.s.imageLoad = bg.b.imageLoad; + bgScale.s.imagePal = bg.b.imagePal; + bgScale.s.imagePtr = bg.b.imagePtr; + bgScale.s.imageSiz = bg.b.imageSiz; + bgScale.s.imageW = bg.b.imageW; + bgScale.s.imageX = bg.b.imageX; + bgScale.s.imageY = bg.b.imageY; + bgScale.s.padding[0] = 0; + bgScale.s.padding[1] = 0; + bgScale.s.padding[2] = 0; + bgScale.s.padding[3] = 0; + bgScale.s.scaleW = 0x400; + bgScale.s.scaleH = 0x400; + bgScale.s.imageYorig = 0; + + frameGetMode(pFrame, FMT_OTHER0, &nOldMode1); + frameGetMode(pFrame, FMT_OTHER1, &nOldMode2); + if ((nOldMode2 & 0x300000) == 0x200000) { + pFrame->aColor[FCT_PRIMITIVE].a = 0xFF; + } + if (!frameSetMode(pFrame, FMT_OTHER0, 0x0F0A4001)) { + return false; + } + if (!frameSetMode(pFrame, FMT_OTHER1, 0x00802CC0)) { + return false; + } + if (!guS2DEmuSetScissor(0, 0, 0x500, 0x3C0, 0)) { + return false; + } + if (!guS2DEmuBgRect1Cyc(pRSP, pFrame, &bgScale)) { + return false; + } + return true; +} +// Matches but data doesn't +#ifndef NON_MATCHING #pragma GLOBAL_ASM("asm/non_matchings/rsp/rspObjMatrix.s") +#else +static bool rspObjMatrix(Rsp* pRSP, Frame* pFrame, s32 nAddress) { + u32* pnData32; + u16* pnData16; + u8* pObjMtx; + u16 nBaseScaleX; + u16 nBaseScaleY; + s32 nA; + s32 nB; + s32 nC; + s32 nD; + s16 nX; + s16 nY; + + if (!ramGetBuffer(SYSTEM_RAM(pRSP->pHost), &pObjMtx, nAddress, NULL)) { + return false; + } + + pnData32 = (u32*)pObjMtx; + pnData16 = (u16*)pObjMtx; + + nA = pnData32[0]; + nB = pnData32[1]; + nC = pnData32[2]; + nD = pnData32[3]; + nX = pnData16[8]; + nY = pnData16[9]; + nBaseScaleX = pnData16[10]; + nBaseScaleY = pnData16[11]; + + pRSP->twoDValues.fX = (f32)nX / 4.0f; + pRSP->twoDValues.fY = -(f32)nY / 4.0f; + pRSP->twoDValues.aRotations[0][0] = nA / 65536.0f; + pRSP->twoDValues.aRotations[0][1] = nB / 65536.0f; + pRSP->twoDValues.aRotations[1][0] = nC / 65536.0f; + pRSP->twoDValues.aRotations[1][1] = nD / 65536.0f; + pRSP->twoDValues.fBaseScaleX = 1024.0f / nBaseScaleX; + pRSP->twoDValues.fBaseScaleY = 1024.0f / nBaseScaleY; + return true; +} +#endif +// Matches but data doesn't +#ifndef NON_MATCHING static bool rspSetupS2DEX(Rsp* pRSP); #pragma GLOBAL_ASM("asm/non_matchings/rsp/rspSetupS2DEX.s") +#else +static bool rspSetupS2DEX(Rsp* pRSP) { + f32 fL; + f32 fR; + f32 fB; + f32 fT; + Frame* pFrame; + f32 fScale = 2.0f; + s32 pad; + + pFrame = SYSTEM_FRAME(pRSP->pHost); + + pRSP->twoDValues.aRotations[0][0] = 1.0f; + pRSP->twoDValues.aRotations[1][0] = 0.0f; + pRSP->twoDValues.aRotations[1][1] = 1.0f; + pRSP->twoDValues.aRotations[0][1] = 0.0f; + + pRSP->twoDValues.fBaseScaleX = 1.0f; + pRSP->twoDValues.fBaseScaleY = 1.0f; + pRSP->twoDValues.fX = 0.0f; + pRSP->twoDValues.fY = 0.0f; + + fL = -pFrame->anSizeX[FS_TARGET] / 2.0f; + fR = pFrame->anSizeX[FS_TARGET] / 2.0f; + fB = -pFrame->anSizeY[FS_TARGET] / 2.0f; + fT = pFrame->anSizeY[FS_TARGET] / 2.0f; + + Matrix4by4Identity(pRSP->aMatrixOrtho); + + pRSP->aMatrixOrtho[0][0] = fScale / (fR - fL); + pRSP->aMatrixOrtho[1][1] = fScale / (fT - fB); + pRSP->aMatrixOrtho[2][2] = -1.0f; + pRSP->aMatrixOrtho[3][0] = -(fR + fL) / (fR - fL); + pRSP->aMatrixOrtho[3][1] = -(fT + fB) / (fT - fB); + pRSP->aMatrixOrtho[3][2] = -fScale / 2.0f; + pRSP->aMatrixOrtho[3][3] = 1.0f; + + return true; +} +#endif static bool rspSetGeometryMode1(Rsp* pRSP, s32 nMode) { s32 nModeFrame = 0;