// Copyright (C) 2003-2009 Dolphin 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 // the Free Software Foundation, version 2.0. // 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 SVN repository and contact information can be found at // http://code.google.com/p/dolphin-emu/ #include "Common.h" #include "DataReader.h" #include "OpcodeDecoder.h" #include "BPMemLoader.h" #include "CPMemLoader.h" #include "XFMemLoader.h" #include "SWVertexLoader.h" #include "SWStatistics.h" #include "DebugUtil.h" #include "SWCommandProcessor.h" #include "CPMemLoader.h" #include "SWVideoConfig.h" #include "HW/Memmap.h" typedef void (*DecodingFunction)(u32); namespace OpcodeDecoder { static DecodingFunction currentFunction = NULL; static u32 minCommandSize; static u16 streamSize; static u16 streamAddress; static bool readOpcode; static SWVertexLoader vertexLoader; static bool inObjectStream; static u8 lastPrimCmd; void DecodePrimitiveStream(u32 iBufferSize) { u32 vertexSize = vertexLoader.GetVertexSize(); bool skipPrimitives = g_bSkipCurrentFrame || swstats.thisFrame.numDrawnObjects < g_SWVideoConfig.drawStart || swstats.thisFrame.numDrawnObjects >= g_SWVideoConfig.drawEnd; if (skipPrimitives) { while (streamSize > 0 && iBufferSize >= vertexSize) { g_pVideoData += vertexSize; iBufferSize -= vertexSize; streamSize--; } } else { while (streamSize > 0 && iBufferSize >= vertexSize) { vertexLoader.LoadVertex(); iBufferSize -= vertexSize; streamSize--; } } if (streamSize == 0) { // return to normal command processing ResetDecoding(); } } void ReadXFData(u32 iBufferSize) { _assert_msg_(VIDEO, iBufferSize >= (u32)(streamSize * 4), "Underflow during standard opcode decoding"); u32 pData[16]; for (int i = 0; i < streamSize; i++) pData[i] = DataReadU32(); SWLoadXFReg(streamSize, streamAddress, pData); // return to normal command processing ResetDecoding(); } void ExecuteDisplayList(u32 addr, u32 count) { u8 *videoDataSave = g_pVideoData; u8 *dlStart = Memory::GetPointer(addr); g_pVideoData = dlStart; while (OpcodeDecoder::CommandRunnable(count)) { OpcodeDecoder::Run(count); // if data was read by the opcode decoder then the video data pointer changed u32 readCount = (u32)(g_pVideoData - dlStart); dlStart = g_pVideoData; _assert_msg_(VIDEO, count >= readCount, "Display list underrun"); count -= readCount; } g_pVideoData = videoDataSave; } void DecodeStandard(u32 bufferSize) { _assert_msg_(VIDEO, CommandRunnable(bufferSize), "Underflow during standard opcode decoding"); int Cmd = DataReadU8(); if (Cmd == GX_NOP) return; // Causes a SIGBUS error on Android // XXX: Investigate #ifndef ANDROID // check if switching in or out of an object // only used for debuggging if (inObjectStream && (Cmd & 0x87) != lastPrimCmd) { inObjectStream = false; DebugUtil::OnObjectEnd(); } if (Cmd & 0x80 && !inObjectStream) { inObjectStream = true; lastPrimCmd = Cmd & 0x87; DebugUtil::OnObjectBegin(); } #endif switch(Cmd) { case GX_NOP: break; case GX_LOAD_CP_REG: //0x08 { u32 SubCmd = DataReadU8(); u32 Value = DataReadU32(); SWLoadCPReg(SubCmd, Value); } break; case GX_LOAD_XF_REG: { u32 Cmd2 = DataReadU32(); streamSize = ((Cmd2 >> 16) & 15) + 1; streamAddress = Cmd2 & 0xFFFF; currentFunction = ReadXFData; minCommandSize = streamSize * 4; readOpcode = false; } break; case GX_LOAD_INDX_A: //used for position matrices SWLoadIndexedXF(DataReadU32(), 0xC); break; case GX_LOAD_INDX_B: //used for normal matrices SWLoadIndexedXF(DataReadU32(), 0xD); break; case GX_LOAD_INDX_C: //used for postmatrices SWLoadIndexedXF(DataReadU32(), 0xE); break; case GX_LOAD_INDX_D: //used for lights SWLoadIndexedXF(DataReadU32(), 0xF); break; case GX_CMD_CALL_DL: { u32 dwAddr = DataReadU32(); u32 dwCount = DataReadU32(); ExecuteDisplayList(dwAddr, dwCount); } break; case 0x44: // zelda 4 swords calls it and checks the metrics registers after that break; case GX_CMD_INVL_VC:// Invalidate (vertex cache?) DEBUG_LOG(VIDEO, "Invalidate (vertex cache?)"); break; case GX_LOAD_BP_REG: //0x61 { u32 cmd = DataReadU32(); SWLoadBPReg(cmd); } break; // draw primitives default: if (Cmd & 0x80) { u8 vatIndex = Cmd & GX_VAT_MASK; u8 primitiveType = (Cmd & GX_PRIMITIVE_MASK) >> GX_PRIMITIVE_SHIFT; vertexLoader.SetFormat(vatIndex, primitiveType); // switch to primitive processing streamSize = DataReadU16(); currentFunction = DecodePrimitiveStream; minCommandSize = vertexLoader.GetVertexSize(); readOpcode = false; INCSTAT(swstats.thisFrame.numPrimatives); DEBUG_LOG(VIDEO, "Draw begin"); } else { PanicAlert("GFX: Unknown Opcode (0x%x).\n", Cmd); break; } break; } } void Init() { inObjectStream = false; lastPrimCmd = 0; ResetDecoding(); } void ResetDecoding() { currentFunction = DecodeStandard; minCommandSize = 1; readOpcode = true; } bool CommandRunnable(u32 iBufferSize) { if (iBufferSize < minCommandSize) return false; if (readOpcode) { u8 Cmd = DataPeek8(0); u32 minSize = 1; switch(Cmd) { case GX_LOAD_CP_REG: //0x08 minSize = 6; break; case GX_LOAD_XF_REG: minSize = 5; break; case GX_LOAD_INDX_A: //used for position matrices minSize = 5; break; case GX_LOAD_INDX_B: //used for normal matrices minSize = 5; break; case GX_LOAD_INDX_C: //used for postmatrices minSize = 5; break; case GX_LOAD_INDX_D: //used for lights minSize = 5; break; case GX_CMD_CALL_DL: minSize = 9; break; case GX_LOAD_BP_REG: //0x61 minSize = 5; break; // draw primitives default: if (Cmd & 0x80) minSize = 3; break; } return (iBufferSize >= minSize); } return true; } void Run(u32 iBufferSize) { currentFunction(iBufferSize); } }