diff --git a/Source/Dolphin.sln b/Source/Dolphin.sln
index a92fe14e13..885744ff4a 100644
--- a/Source/Dolphin.sln
+++ b/Source/Dolphin.sln
@@ -191,6 +191,8 @@ Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "wxCore28", "..\Externals\wx
{1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE} = {1C8436C9-DBAF-42BE-83BC-CF3EC9175ABE}
EndProjectSection
EndProject
+Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "Plugin_VideoSoftware", "Plugins\Plugin_VideoSoftware\Plugin_VideoSoftware.vcproj", "{66A4E7BD-E2E8-4373-9B75-8750EB5AE683}"
+EndProject
Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution
Debug|Win32 = Debug|Win32
@@ -672,6 +674,18 @@ Global
{11F55366-12EC-4C44-A8CB-1D4E315D61ED}.Release|Win32.Build.0 = Release|Win32
{11F55366-12EC-4C44-A8CB-1D4E315D61ED}.Release|x64.ActiveCfg = Release|x64
{11F55366-12EC-4C44-A8CB-1D4E315D61ED}.Release|x64.Build.0 = Release|x64
+ {66A4E7BD-E2E8-4373-9B75-8750EB5AE683}.Debug|Win32.ActiveCfg = Debug|Win32
+ {66A4E7BD-E2E8-4373-9B75-8750EB5AE683}.Debug|Win32.Build.0 = Debug|Win32
+ {66A4E7BD-E2E8-4373-9B75-8750EB5AE683}.Debug|x64.ActiveCfg = Debug|Win32
+ {66A4E7BD-E2E8-4373-9B75-8750EB5AE683}.DebugFast|Win32.ActiveCfg = Debug|Win32
+ {66A4E7BD-E2E8-4373-9B75-8750EB5AE683}.DebugFast|Win32.Build.0 = Debug|Win32
+ {66A4E7BD-E2E8-4373-9B75-8750EB5AE683}.DebugFast|x64.ActiveCfg = Debug|Win32
+ {66A4E7BD-E2E8-4373-9B75-8750EB5AE683}.Release_JITIL|Win32.ActiveCfg = Release|Win32
+ {66A4E7BD-E2E8-4373-9B75-8750EB5AE683}.Release_JITIL|Win32.Build.0 = Release|Win32
+ {66A4E7BD-E2E8-4373-9B75-8750EB5AE683}.Release_JITIL|x64.ActiveCfg = Release|Win32
+ {66A4E7BD-E2E8-4373-9B75-8750EB5AE683}.Release|Win32.ActiveCfg = Release|Win32
+ {66A4E7BD-E2E8-4373-9B75-8750EB5AE683}.Release|Win32.Build.0 = Release|Win32
+ {66A4E7BD-E2E8-4373-9B75-8750EB5AE683}.Release|x64.ActiveCfg = Release|Win32
EndGlobalSection
GlobalSection(SolutionProperties) = preSolution
HideSolutionNode = FALSE
diff --git a/Source/Plugins/Plugin_VideoSoftware/Plugin_VideoSoftware.vcproj b/Source/Plugins/Plugin_VideoSoftware/Plugin_VideoSoftware.vcproj
new file mode 100644
index 0000000000..906dd55baf
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Plugin_VideoSoftware.vcproj
@@ -0,0 +1,481 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/Source/Plugins/Plugin_VideoSoftware/Src/BPMemLoader.cpp b/Source/Plugins/Plugin_VideoSoftware/Src/BPMemLoader.cpp
new file mode 100644
index 0000000000..f7b2a8a4cd
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Src/BPMemLoader.cpp
@@ -0,0 +1,149 @@
+// 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 "pluginspecs_video.h"
+#include "../../../Core/VideoCommon/Src/VideoCommon.h"
+#include "main.h"
+
+#include "BPMemLoader.h"
+#include "EfbCopy.h"
+#include "Rasterizer.h"
+#include "PixelEngine.h"
+#include "../../../Core/VideoCommon/Src/TextureDecoder.h"
+
+
+BPMemory bpmem;
+
+void InitBPMemory()
+{
+ memset(&bpmem, 0, sizeof(bpmem));
+ bpmem.bpMask = 0xFFFFFF;
+}
+
+void BPWritten(int address, int newvalue);
+
+void LoadBPReg(u32 value)
+{
+ //handle the mask register
+ int address = value >> 24;
+ int oldval = ((u32*)&bpmem)[address];
+ int newval = (oldval & ~bpmem.bpMask) | (value & bpmem.bpMask);
+
+ ((u32*)&bpmem)[address] = newval;
+
+ //reset the mask register
+ if (address != 0xFE)
+ bpmem.bpMask = 0xFFFFFF;
+
+ BPWritten(address, newval);
+}
+
+void BPWritten(int address, int newvalue)
+{
+ switch (address)
+ {
+ case BPMEM_SCISSORTL:
+ case BPMEM_SCISSORBR:
+ case BPMEM_SCISSOROFFSET:
+ Rasterizer::SetScissor();
+ break;
+ case BPMEM_SETDRAWDONE: // This is called when the game is done drawing (eg: like in DX: Begin(); Draw(); End();)
+ switch (bpmem.drawdone & 0xFF)
+ {
+ case 0x02:
+ PixelEngine::SetFinish(); // may generate interrupt
+ DEBUG_LOG(VIDEO, "GXSetDrawDone SetPEFinish (value: 0x%02X)", (bpmem.drawdone & 0xFFFF));
+ break;
+
+ default:
+ WARN_LOG(VIDEO, "GXSetDrawDone ??? (value 0x%02X)", (bpmem.drawdone & 0xFFFF));
+ break;
+ }
+ break;
+ case BPMEM_PE_TOKEN_ID: // Pixel Engine Token ID
+ DEBUG_LOG(VIDEO, "SetPEToken 0x%04x", (bpmem.petoken & 0xFFFF));
+ PixelEngine::SetToken(static_cast(bpmem.petokenint & 0xFFFF), false);
+ break;
+ case BPMEM_PE_TOKEN_INT_ID: // Pixel Engine Interrupt Token ID
+ DEBUG_LOG(VIDEO, "SetPEToken + INT 0x%04x", (bpmem.petokenint & 0xFFFF));
+ PixelEngine::SetToken(static_cast(bpmem.petokenint & 0xFFFF), TRUE);
+ break;
+ case BPMEM_TRIGGER_EFB_COPY:
+ EfbCopy::CopyEfb();
+ break;
+ case BPMEM_CLEARBBOX1:
+ PixelEngine::pereg.boxRight = newvalue >> 10;
+ PixelEngine::pereg.boxLeft = newvalue & 0x3ff;
+ break;
+ case BPMEM_CLEARBBOX2:
+ PixelEngine::pereg.boxBottom = newvalue >> 10;
+ PixelEngine::pereg.boxTop = newvalue & 0x3ff;
+ break;
+ case BPMEM_LOADTLUT0: // Load a Texture Look Up Table
+ case BPMEM_LOADTLUT1:
+ {
+ u32 tlutTMemAddr = (newvalue & 0x3FF) << 9;
+ u32 tlutXferCount = (newvalue & 0x1FFC00) >> 5;
+
+ u8 *ptr = 0;
+
+ // TODO - figure out a cleaner way.
+ if (g_VideoInitialize.bWii)
+ ptr = g_VideoInitialize.pGetMemoryPointer(bpmem.tlutXferSrc << 5);
+ else
+ ptr = g_VideoInitialize.pGetMemoryPointer((bpmem.tlutXferSrc & 0xFFFFF) << 5);
+
+ if (ptr)
+ memcpy_gc(texMem + tlutTMemAddr, ptr, tlutXferCount);
+ else
+ PanicAlert("Invalid palette pointer %08x %08x %08x", bpmem.tlutXferSrc, bpmem.tlutXferSrc << 5, (bpmem.tlutXferSrc & 0xFFFFF)<< 5);
+ break;
+ }
+
+ case BPMEM_TEV_REGISTER_L: // Reg 1
+ case BPMEM_TEV_REGISTER_L+2: // Reg 2
+ case BPMEM_TEV_REGISTER_L+4: // Reg 3
+ case BPMEM_TEV_REGISTER_L+6: // Reg 4
+ {
+ int regNum = (address >> 1 ) & 0x3;
+ ColReg& reg = bpmem.tevregs[regNum].low;
+ bool konst = reg.type;
+
+ Rasterizer::SetTevReg(regNum, 3, konst, reg.b); // A
+ Rasterizer::SetTevReg(regNum, 0, konst, reg.a); // R
+
+ break;
+ }
+
+ case BPMEM_TEV_REGISTER_H: // Reg 1
+ case BPMEM_TEV_REGISTER_H+2: // Reg 2
+ case BPMEM_TEV_REGISTER_H+4: // Reg 3
+ case BPMEM_TEV_REGISTER_H+6: // Reg 4
+ {
+ int regNum = (address >> 1 ) & 0x3;
+ ColReg& reg = bpmem.tevregs[regNum].high;
+ bool konst = reg.type;
+
+ Rasterizer::SetTevReg(regNum, 1, konst, reg.b); // G
+ Rasterizer::SetTevReg(regNum, 2, konst, reg.a); // B
+
+ break;
+ }
+
+ }
+}
+
diff --git a/Source/Plugins/Plugin_VideoSoftware/Src/BPMemLoader.h b/Source/Plugins/Plugin_VideoSoftware/Src/BPMemLoader.h
new file mode 100644
index 0000000000..cde13ff3cb
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Src/BPMemLoader.h
@@ -0,0 +1,28 @@
+// 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/
+
+#ifndef _BPMEMLOADER_H_
+#define _BPMEMLOADER_H_
+
+
+#include "Common.h"
+
+#include "../../../Core/VideoCommon/Src/BPMemory.h"
+
+void InitBPMemory();
+
+#endif
\ No newline at end of file
diff --git a/Source/Plugins/Plugin_VideoSoftware/Src/CPMemLoader.cpp b/Source/Plugins/Plugin_VideoSoftware/Src/CPMemLoader.cpp
new file mode 100644
index 0000000000..9843737622
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Src/CPMemLoader.cpp
@@ -0,0 +1,82 @@
+// 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 "VideoCommon.h"
+
+#include "CPMemLoader.h"
+
+
+// CP state
+u8 *cached_arraybases[16];
+
+// STATE_TO_SAVE
+u32 arraybases[16];
+u32 arraystrides[16];
+TMatrixIndexA MatrixIndexA;
+TMatrixIndexB MatrixIndexB;
+TVtxDesc g_VtxDesc;
+VAT g_VtxAttr[8];
+
+
+void LoadCPReg(u32 sub_cmd, u32 value)
+{
+ switch (sub_cmd & 0xF0)
+ {
+ case 0x30:
+ MatrixIndexA.Hex = value;
+ break;
+
+ case 0x40:
+ MatrixIndexB.Hex = value;
+ break;
+
+ case 0x50:
+ g_VtxDesc.Hex &= ~0x1FFFF; // keep the Upper bits
+ g_VtxDesc.Hex |= value;
+ break;
+
+ case 0x60:
+ g_VtxDesc.Hex &= 0x1FFFF; // keep the lower 17Bits
+ g_VtxDesc.Hex |= (u64)value << 17;
+ break;
+
+ case 0x70:
+ _assert_((sub_cmd & 0x0F) < 8);
+ g_VtxAttr[sub_cmd & 7].g0.Hex = value;
+ break;
+
+ case 0x80:
+ _assert_((sub_cmd & 0x0F) < 8);
+ g_VtxAttr[sub_cmd & 7].g1.Hex = value;
+ break;
+
+ case 0x90:
+ _assert_((sub_cmd & 0x0F) < 8);
+ g_VtxAttr[sub_cmd & 7].g2.Hex = value;
+ break;
+
+ // Pointers to vertex arrays in GC RAM
+ case 0xA0:
+ arraybases[sub_cmd & 0xF] = value;
+ cached_arraybases[sub_cmd & 0xF] = Memory_GetPtr(value);
+ break;
+
+ case 0xB0:
+ arraystrides[sub_cmd & 0xF] = value & 0xFF;
+ break;
+ }
+}
+
diff --git a/Source/Plugins/Plugin_VideoSoftware/Src/CPMemLoader.h b/Source/Plugins/Plugin_VideoSoftware/Src/CPMemLoader.h
new file mode 100644
index 0000000000..c72a5abac6
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Src/CPMemLoader.h
@@ -0,0 +1,27 @@
+// 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/
+
+#ifndef _CPMEMLOADER_H_
+#define _CPMEMLOADER_H_
+
+
+#include "Common.h"
+
+#include "../../../Core/VideoCommon/Src/CPMemory.h"
+
+
+#endif
\ No newline at end of file
diff --git a/Source/Plugins/Plugin_VideoSoftware/Src/Clipper.cpp b/Source/Plugins/Plugin_VideoSoftware/Src/Clipper.cpp
new file mode 100644
index 0000000000..f475028c24
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Src/Clipper.cpp
@@ -0,0 +1,300 @@
+// 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/
+
+/*
+Portions of this file are based off work by Markus Trenkwalder.
+Copyright (c) 2007, 2008 Markus Trenkwalder
+
+All rights reserved.
+
+Redistribution and use in source and binary forms, with or without
+modification, are permitted provided that the following conditions are met:
+
+* Redistributions of source code must retain the above copyright notice,
+ this list of conditions and the following disclaimer.
+
+* Redistributions in binary form must reproduce the above copyright notice,
+ this list of conditions and the following disclaimer in the documentation
+ and/or other materials provided with the distribution.
+
+* Neither the name of the library's copyright owner nor the names of its
+ contributors may be used to endorse or promote products derived from this
+ software without specific prior written permission.
+
+THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
+"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
+LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
+A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
+CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
+EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
+PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
+PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
+LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
+NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
+SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+*/
+
+#include "Clipper.h"
+#include "Rasterizer.h"
+#include "NativeVertexFormat.h"
+#include "XFMemLoader.h"
+#include "BPMemLoader.h"
+#include "Statistics.h"
+#include "VideoConfig.h"
+
+
+namespace Clipper
+{
+ void Init()
+ {
+ for (int i = 0; i < 18; ++i)
+ Vertices[i+3] = &ClippedVertices[i];
+ }
+
+ void SetViewOffset()
+ {
+ m_ViewOffset[0] = xfregs.viewport.xOrig - 342;
+ m_ViewOffset[1] = xfregs.viewport.yOrig - 342;
+ m_ViewOffset[2] = xfregs.viewport.farZ - xfregs.viewport.farZ;
+ }
+
+
+ enum {
+ SKIP_FLAG = -1,
+ CLIP_POS_X_BIT = 0x01,
+ CLIP_NEG_X_BIT = 0x02,
+ CLIP_POS_Y_BIT = 0x04,
+ CLIP_NEG_Y_BIT = 0x08,
+ CLIP_POS_Z_BIT = 0x10,
+ CLIP_NEG_Z_BIT = 0x20
+ };
+
+ static inline int CalcClipMask(OutputVertexData *v)
+ {
+ int cmask = 0;
+ float* pos = v->projectedPosition;
+ if (pos[3] - pos[0] < 0) cmask |= CLIP_POS_X_BIT;
+ if (pos[0] + pos[3] < 0) cmask |= CLIP_NEG_X_BIT;
+ if (pos[3] - pos[1] < 0) cmask |= CLIP_POS_Y_BIT;
+ if (pos[1] + pos[3] < 0) cmask |= CLIP_NEG_Y_BIT;
+ if (pos[3] * pos[2] > 0) cmask |= CLIP_POS_Z_BIT;
+ if (pos[2] + pos[3] < 0) cmask |= CLIP_NEG_Z_BIT;
+ return cmask;
+ }
+
+ static inline void AddInterpolatedVertex(float t, int out, int in, int& numVertices)
+ {
+ Vertices[numVertices]->Lerp(t, Vertices[out], Vertices[in]);
+ numVertices++;
+ }
+
+ #define DIFFERENT_SIGNS(x,y) ((x <= 0 && y > 0) || (x > 0 && y <= 0))
+
+ #define CLIP_DOTPROD(I, A, B, C, D) \
+ (Vertices[I]->projectedPosition[0] * A + Vertices[I]->projectedPosition[1] * B + Vertices[I]->projectedPosition[2] * C + Vertices[I]->projectedPosition[3] * D)
+
+ #define POLY_CLIP( PLANE_BIT, A, B, C, D ) \
+ { \
+ if (mask & PLANE_BIT) { \
+ int idxPrev = inlist[0]; \
+ float dpPrev = CLIP_DOTPROD(idxPrev, A, B, C, D ); \
+ int outcount = 0; \
+ int i; \
+ \
+ inlist[n] = inlist[0]; \
+ for (i = 1; i <= n; i++) { \
+ int idx = inlist[i]; \
+ float dp = CLIP_DOTPROD(idx, A, B, C, D ); \
+ if (dpPrev >= 0) { \
+ outlist[outcount++] = idxPrev; \
+ } \
+ \
+ if (DIFFERENT_SIGNS(dp, dpPrev)) { \
+ if (dp < 0) { \
+ float t = dp / (dp - dpPrev); \
+ AddInterpolatedVertex(t, idx, idxPrev, numVertices); \
+ } else { \
+ float t = dpPrev / (dpPrev - dp); \
+ AddInterpolatedVertex(t, idxPrev, idx, numVertices); \
+ } \
+ outlist[outcount++] = numVertices - 1; \
+ } \
+ \
+ idxPrev = idx; \
+ dpPrev = dp; \
+ } \
+ \
+ if (outcount < 3) \
+ continue; \
+ \
+ { \
+ int *tmp = inlist; \
+ inlist = outlist; \
+ outlist = tmp; \
+ n = outcount; \
+ } \
+ } \
+ }
+
+ void ClipTriangle(int *indices, int &numIndices)
+ {
+ int mask = 0;
+
+ mask |= CalcClipMask(Vertices[0]);
+ mask |= CalcClipMask(Vertices[1]);
+ mask |= CalcClipMask(Vertices[2]);
+
+ if (mask != 0)
+ {
+ for(int idx = 0; idx < 3; idx += 3)
+ {
+ int vlist[2][2*6+1];
+ int *inlist = vlist[0], *outlist = vlist[1];
+ int n = 3;
+ int numVertices = 3;
+
+ inlist[0] = 0;
+ inlist[1] = 1;
+ inlist[2] = 2;
+
+ // mark this triangle as unused in case it should be completely
+ // clipped
+ indices[0] = SKIP_FLAG;
+ indices[1] = SKIP_FLAG;
+ indices[2] = SKIP_FLAG;
+
+ POLY_CLIP(CLIP_POS_X_BIT, -1, 0, 0, 1);
+ POLY_CLIP(CLIP_NEG_X_BIT, 1, 0, 0, 1);
+ POLY_CLIP(CLIP_POS_Y_BIT, 0, -1, 0, 1);
+ POLY_CLIP(CLIP_NEG_Y_BIT, 0, 1, 0, 1);
+ POLY_CLIP(CLIP_POS_Z_BIT, 0, 0, 0, 1);
+ POLY_CLIP(CLIP_NEG_Z_BIT, 0, 0, 1, 1);
+
+ INCSTAT(stats.thisFrame.numTrianglesClipped);
+
+ // transform the poly in inlist into triangles
+ indices[0] = inlist[0];
+ indices[1] = inlist[1];
+ indices[2] = inlist[2];
+ for (int i = 3; i < n; ++i) {
+ indices[numIndices++] = inlist[0];
+ indices[numIndices++] = inlist[i - 1];
+ indices[numIndices++] = inlist[i];
+ }
+ }
+ }
+ }
+
+ void ProcessTriangle(OutputVertexData *v0, OutputVertexData *v1, OutputVertexData *v2)
+ {
+ if (stats.thisFrame.numDrawnObjects < g_Config.drawStart || stats.thisFrame.numDrawnObjects >= g_Config.drawEnd )
+ return;
+
+ INCSTAT(stats.thisFrame.numTrianglesIn)
+
+ bool backface;
+
+ if(!CullTest(v0, v1, v2, backface))
+ return;
+
+ int indices[21] = { 0, 1, 2, SKIP_FLAG, SKIP_FLAG, SKIP_FLAG, SKIP_FLAG, SKIP_FLAG, SKIP_FLAG,
+ SKIP_FLAG, SKIP_FLAG, SKIP_FLAG, SKIP_FLAG, SKIP_FLAG, SKIP_FLAG,
+ SKIP_FLAG, SKIP_FLAG, SKIP_FLAG, SKIP_FLAG, SKIP_FLAG, SKIP_FLAG };
+ int numIndices = 3;
+
+ if (backface)
+ {
+ Vertices[0] = v0;
+ Vertices[1] = v2;
+ Vertices[2] = v1;
+ }
+ else
+ {
+ Vertices[0] = v0;
+ Vertices[1] = v1;
+ Vertices[2] = v2;
+ }
+
+ ClipTriangle(indices, numIndices);
+
+ for(int i = 0; i+3 <= numIndices; i+=3)
+ {
+ if(indices[i] != SKIP_FLAG)
+ {
+ PerspectiveDivide(Vertices[indices[i]]);
+ PerspectiveDivide(Vertices[indices[i+1]]);
+ PerspectiveDivide(Vertices[indices[i+2]]);
+
+ Rasterizer::DrawTriangleFrontFace(Vertices[indices[i]], Vertices[indices[i+1]], Vertices[indices[i+2]]);
+ }
+ }
+ }
+
+
+ bool CullTest(OutputVertexData *v0, OutputVertexData *v1, OutputVertexData *v2, bool &backface)
+ {
+ int mask = CalcClipMask(v0);
+ mask &= CalcClipMask(v1);
+ mask &= CalcClipMask(v2);
+
+ if(mask)
+ {
+ INCSTAT(stats.thisFrame.numTrianglesRejected)
+ return false;
+ }
+
+ float x0 = v0->projectedPosition[0];
+ float x1 = v1->projectedPosition[0];
+ float x2 = v2->projectedPosition[0];
+ float y1 = v1->projectedPosition[1];
+ float y0 = v0->projectedPosition[1];
+ float y2 = v2->projectedPosition[1];
+ float w0 = v0->projectedPosition[3];
+ float w1 = v1->projectedPosition[3];
+ float w2 = v2->projectedPosition[3];
+
+ float normalZDir = (x0*w2 - x2*w0)*y1 + (x2*y0 - x0*y2)*w1 + (y2*w0 - y0*w2)*x1;
+
+ backface = normalZDir <= 0.0f;
+
+ if ((bpmem.genMode.cullmode & 1) && !backface) // cull frontfacing
+ {
+ INCSTAT(stats.thisFrame.numTrianglesCulled)
+ return false;
+ }
+
+ if ((bpmem.genMode.cullmode & 2) && backface) // cull backfacing
+ {
+ INCSTAT(stats.thisFrame.numTrianglesCulled)
+ return false;
+ }
+
+ return true;
+ }
+
+ void PerspectiveDivide(OutputVertexData *vertex)
+ {
+ float *projected = vertex->projectedPosition;
+ float *screen = vertex->screenPosition;
+
+ float wInverse = 1.0f/projected[3];
+ screen[0] = projected[0] * wInverse * xfregs.viewport.wd + m_ViewOffset[0];
+ screen[1] = projected[1] * wInverse * xfregs.viewport.ht + m_ViewOffset[1];
+ screen[2] = projected[2] * wInverse + m_ViewOffset[2];
+ }
+
+}
\ No newline at end of file
diff --git a/Source/Plugins/Plugin_VideoSoftware/Src/Clipper.h b/Source/Plugins/Plugin_VideoSoftware/Src/Clipper.h
new file mode 100644
index 0000000000..f2e6696c4c
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Src/Clipper.h
@@ -0,0 +1,46 @@
+// 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/
+
+#ifndef _CLIPPER_H_
+#define _CLIPPER_H_
+
+
+#include "Common.h"
+#include "NativeVertexFormat.h"
+
+
+namespace Clipper
+{
+ void Init();
+
+ void SetViewOffset();
+
+ void ProcessTriangle(OutputVertexData *v0, OutputVertexData *v1, OutputVertexData *v2);
+
+
+ bool CullTest(OutputVertexData *v0, OutputVertexData *v1, OutputVertexData *v2, bool &backface);
+
+ void PerspectiveDivide(OutputVertexData *vertex);
+
+ static float m_ViewOffset[3];
+
+ static OutputVertexData ClippedVertices[18];
+ static OutputVertexData *Vertices[21];
+}
+
+
+#endif
\ No newline at end of file
diff --git a/Source/Plugins/Plugin_VideoSoftware/Src/CommandProcessor.cpp b/Source/Plugins/Plugin_VideoSoftware/Src/CommandProcessor.cpp
new file mode 100644
index 0000000000..eda0214d5f
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Src/CommandProcessor.cpp
@@ -0,0 +1,443 @@
+// 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 "pluginspecs_video.h"
+
+#include "CommandProcessor.h"
+#include "OpcodeDecoder.h"
+#include "main.h"
+#include "ChunkFile.h"
+#include "MathUtil.h"
+
+
+u8* g_pVideoData; // data reader uses this as the read pointer
+
+
+namespace CommandProcessor
+{
+
+enum
+{
+ GATHER_PIPE_SIZE = 32,
+ INT_CAUSE_CP = 0x800
+};
+
+// STATE_TO_SAVE
+// variables
+
+const int commandBufferSize = 4 * 1024;
+const int commandBufferCopySize = 32;
+const int maxCommandBufferWrite = commandBufferSize - commandBufferCopySize;
+u8 commandBuffer[commandBufferSize];
+u32 readPos;
+u32 writePos;
+
+CPReg cpreg; // shared between gfx and emulator thread
+
+
+void DoState(PointerWrap &p)
+{
+ p.Do(cpreg);
+}
+
+// function
+void UpdateFifoRegister();
+void UpdateInterrupts();
+
+// does it matter that there is no synchronization between threads during writes?
+inline void WriteLow (u32& _reg, u16 lowbits) {_reg = (_reg & 0xFFFF0000) | lowbits;}
+inline void WriteHigh(u32& _reg, u16 highbits) {_reg = (_reg & 0x0000FFFF) | ((u32)highbits << 16);}
+//inline void WriteLow (volatile u32& _reg, u16 lowbits) {Common::SyncInterlockedExchange((LONG*)&_reg,(_reg & 0xFFFF0000) | lowbits);}
+//inline void WriteHigh(volatile u32& _reg, u16 highbits) {Common::SyncInterlockedExchange((LONG*)&_reg,(_reg & 0x0000FFFF) | ((u32)highbits << 16));}
+
+inline u16 ReadLow (u32 _reg) {return (u16)(_reg & 0xFFFF);}
+inline u16 ReadHigh (u32 _reg) {return (u16)(_reg >> 16);}
+
+int et_UpdateInterrupts;
+
+void UpdateInterrupts_Wrapper(u64 userdata, int cyclesLate)
+{
+ UpdateInterrupts();
+}
+
+void Init()
+{
+ cpreg.status.Hex = 0;
+ cpreg.status.CommandIdle = 1;
+ cpreg.status.ReadIdle = 1;
+
+ cpreg.ctrl.Hex = 0;
+ cpreg.clear.Hex = 0;
+
+ cpreg.bboxleft = 0;
+ cpreg.bboxtop = 0;
+ cpreg.bboxright = 0;
+ cpreg.bboxbottom = 0;
+
+ cpreg.token = 0;
+
+ et_UpdateInterrupts = g_VideoInitialize.pRegisterEvent("UpdateInterrupts", UpdateInterrupts_Wrapper);
+
+ // internal buffer position
+ readPos = 0;
+ writePos = 0;
+
+ g_pVideoData = 0;
+}
+
+void Shutdown()
+{
+#ifndef _WIN32
+ // delete fifo.sync;
+#endif
+}
+
+void Read16(u16& _rReturnValue, const u32 _Address)
+{
+ DEBUG_LOG(COMMANDPROCESSOR, "(r): 0x%08x", _Address);
+
+ u32 regAddr = (_Address & 0xFFF) >> 1;
+ if (regAddr < 0x20)
+ _rReturnValue = ((u16*)&cpreg)[regAddr];
+ else
+ _rReturnValue = 0;
+}
+
+void RunGpu()
+{
+ if (!g_VideoInitialize.bUseDualCore)
+ {
+ // We are going to do FP math on the main thread so have to save the current state
+ SaveSSEState();
+ LoadDefaultSSEState();
+
+ // run the opcode decoder
+ RunBuffer();
+
+ LoadSSEState();
+ }
+}
+
+void Write16(const u16 _Value, const u32 _Address)
+{
+ INFO_LOG(COMMANDPROCESSOR, "(write16): 0x%04x @ 0x%08x",_Value,_Address);
+
+ switch (_Address & 0xFFF)
+ {
+ case STATUS_REGISTER:
+ {
+ UCPStatusReg tmpStatus(_Value);
+
+ if (cpreg.status.Breakpoint != tmpStatus.Breakpoint)
+ INFO_LOG(COMMANDPROCESSOR,"Set breakpoint status by writing to STATUS_REGISTER");
+
+ cpreg.status.Hex = _Value;
+
+ INFO_LOG(COMMANDPROCESSOR,"\t write to STATUS_REGISTER : %04x", _Value);
+ }
+ break;
+
+ case CTRL_REGISTER:
+ {
+ cpreg.ctrl.Hex = _Value;
+
+ // clear breakpoint if BPEnable and CPIntEnable are 0
+ if (!cpreg.ctrl.BPEnable) {
+ if (!cpreg.ctrl.CPIntEnable) {
+ cpreg.status.Breakpoint = 0;
+ }
+ }
+
+ UpdateInterrupts();
+
+ DEBUG_LOG(COMMANDPROCESSOR,"\t write to CTRL_REGISTER : %04x", _Value);
+ DEBUG_LOG(COMMANDPROCESSOR, "\t GPREAD %s | CPULINK %s | BP %s || CPIntEnable %s | OvF %s | UndF %s"
+ , cpreg.ctrl.GPReadEnable ? "ON" : "OFF"
+ , cpreg.ctrl.GPLinkEnable ? "ON" : "OFF"
+ , cpreg.ctrl.BPEnable ? "ON" : "OFF"
+ , cpreg.ctrl.CPIntEnable ? "ON" : "OFF"
+ , cpreg.ctrl.FifoOverflowIntEnable ? "ON" : "OFF"
+ , cpreg.ctrl.FifoUnderflowIntEnable ? "ON" : "OFF"
+ );
+
+ }
+ break;
+
+ case CLEAR_REGISTER:
+ {
+ UCPClearReg tmpClear(_Value);
+
+ if (tmpClear.ClearFifoOverflow)
+ cpreg.status.OverflowHiWatermark = 0;
+ if (tmpClear.ClearFifoUnderflow)
+ cpreg.status.UnderflowLoWatermark = 0;
+
+ INFO_LOG(COMMANDPROCESSOR,"\t write to CLEAR_REGISTER : %04x",_Value);
+ }
+ break;
+
+ // Fifo Registers
+ case FIFO_TOKEN_REGISTER:
+ cpreg.token = _Value;
+ DEBUG_LOG(COMMANDPROCESSOR,"\t write to FIFO_TOKEN_REGISTER : %04x", _Value);
+ break;
+
+ case FIFO_BASE_LO:
+ WriteLow ((u32 &)cpreg.fifobase, _Value);
+ DEBUG_LOG(COMMANDPROCESSOR,"\t write to FIFO_BASE_LO. FIFO base is : %08x", cpreg.fifobase);
+ break;
+ case FIFO_BASE_HI:
+ WriteHigh((u32 &)cpreg.fifobase, _Value);
+ DEBUG_LOG(COMMANDPROCESSOR,"\t write to FIFO_BASE_HI. FIFO base is : %08x", cpreg.fifobase);
+ break;
+ case FIFO_END_LO:
+ WriteLow ((u32 &)cpreg.fifoend, _Value);
+ DEBUG_LOG(COMMANDPROCESSOR,"\t write to FIFO_END_LO. FIFO end is : %08x", cpreg.fifoend);
+ break;
+ case FIFO_END_HI:
+ WriteHigh((u32 &)cpreg.fifoend, _Value);
+ DEBUG_LOG(COMMANDPROCESSOR,"\t write to FIFO_END_HI. FIFO end is : %08x", cpreg.fifoend);
+ break;
+
+ case FIFO_WRITE_POINTER_LO:
+ WriteLow ((u32 &)cpreg.writeptr, _Value);
+ DEBUG_LOG(COMMANDPROCESSOR,"\t write to FIFO_WRITE_POINTER_LO. write ptr is : %08x", cpreg.writeptr);
+ break;
+ case FIFO_WRITE_POINTER_HI:
+ WriteHigh ((u32 &)cpreg.writeptr, _Value);
+ DEBUG_LOG(COMMANDPROCESSOR,"\t write to FIFO_WRITE_POINTER_HI. write ptr is : %08x", cpreg.writeptr);
+ break;
+ case FIFO_READ_POINTER_LO:
+ WriteLow ((u32 &)cpreg.readptr, _Value);
+ DEBUG_LOG(COMMANDPROCESSOR,"\t write to FIFO_READ_POINTER_LO. read ptr is : %08x", cpreg.readptr);
+ break;
+ case FIFO_READ_POINTER_HI:
+ WriteHigh ((u32 &)cpreg.readptr, _Value);
+ DEBUG_LOG(COMMANDPROCESSOR,"\t write to FIFO_READ_POINTER_HI. read ptr is : %08x", cpreg.readptr);
+ break;
+
+ case FIFO_HI_WATERMARK_LO:
+ WriteLow ((u32 &)cpreg.hiwatermark, _Value);
+ DEBUG_LOG(COMMANDPROCESSOR,"\t write to FIFO_HI_WATERMARK_LO. hiwatermark is : %08x", cpreg.hiwatermark);
+ break;
+ case FIFO_HI_WATERMARK_HI:
+ WriteHigh ((u32 &)cpreg.hiwatermark, _Value);
+ DEBUG_LOG(COMMANDPROCESSOR,"\t write to FIFO_HI_WATERMARK_HI. hiwatermark is : %08x", cpreg.hiwatermark);
+ break;
+ case FIFO_LO_WATERMARK_LO:
+ WriteLow ((u32 &)cpreg.lowatermark, _Value);
+ DEBUG_LOG(COMMANDPROCESSOR,"\t write to FIFO_LO_WATERMARK_LO. lowatermark is : %08x", cpreg.lowatermark);
+ break;
+ case FIFO_LO_WATERMARK_HI:
+ WriteHigh ((u32 &)cpreg.lowatermark, _Value);
+ DEBUG_LOG(COMMANDPROCESSOR,"\t write to FIFO_LO_WATERMARK_HI. lowatermark is : %08x", cpreg.lowatermark);
+ break;
+
+ case FIFO_BP_LO:
+ WriteLow ((u32 &)cpreg.breakpt, _Value);
+ DEBUG_LOG(COMMANDPROCESSOR,"\t write to FIFO_BP_LO. breakpt is : %08x", cpreg.breakpt);
+ break;
+ case FIFO_BP_HI:
+ WriteHigh ((u32 &)cpreg.breakpt, _Value);
+ DEBUG_LOG(COMMANDPROCESSOR,"\t write to FIFO_BP_HI. breakpt is : %08x", cpreg.breakpt);
+ break;
+
+ // Super monkey try to overwrite CPReadWriteDistance by an old saved RWD value. Which is lame for us.
+ // hack: We have to force CPU to think fifo is alway empty and on idle.
+ // When we fall here CPReadWriteDistance should be always null and the game should always want to overwrite it by 0.
+ // So, we can skip it.
+ case FIFO_RW_DISTANCE_LO:
+ WriteLow ((u32 &)cpreg.rwdistance, _Value);
+ DEBUG_LOG(COMMANDPROCESSOR,"\t write to FIFO_RW_DISTANCE_LO. rwdistance is : %08x", cpreg.rwdistance);
+ break;
+ case FIFO_RW_DISTANCE_HI:
+ WriteHigh ((u32 &)cpreg.rwdistance, _Value);
+ DEBUG_LOG(COMMANDPROCESSOR,"\t write to FIFO_RW_DISTANCE_HI. rwdistance is : %08x", cpreg.rwdistance);
+ break;
+ }
+
+ RunGpu();
+}
+
+void Read32(u32& _rReturnValue, const u32 _Address)
+{
+ _rReturnValue = 0;
+ _dbg_assert_msg_(COMMANDPROCESSOR, 0, "Read32 from CommandProccessor at 0x%08x", _Address);
+}
+
+void Write32(const u32 _Data, const u32 _Address)
+{
+ _dbg_assert_msg_(COMMANDPROCESSOR, 0, "Write32 at CommandProccessor at 0x%08x", _Address);
+}
+
+void STACKALIGN GatherPipeBursted()
+{
+ if (cpreg.ctrl.GPLinkEnable)
+ {
+ cpreg.writeptr += GATHER_PIPE_SIZE;
+ if (cpreg.writeptr >= (cpreg.fifoend & 0xFFFFFFE0))
+ cpreg.writeptr = cpreg.fifobase;
+
+ // the read/write pointers will be managed in RunGpu which will read the current fifo and
+ // send as much data as is available and the plugin can take
+ // this will have the cost of copying data to the plugin fifo buffer in the main thread
+ }
+
+ RunGpu();
+}
+
+void UpdateInterrupts()
+{
+ bool bpInt = cpreg.status.Breakpoint && cpreg.ctrl.BPEnable;
+ bool ovfInt = cpreg.status.OverflowHiWatermark && cpreg.ctrl.FifoOverflowIntEnable;
+ bool undfInt = cpreg.status.UnderflowLoWatermark && cpreg.ctrl.FifoUnderflowIntEnable;
+
+ DEBUG_LOG(COMMANDPROCESSOR, "\tUpdate Interrupts");
+ DEBUG_LOG(COMMANDPROCESSOR, "\tCPIntEnable %s | BP %s | OvF %s | UndF %s"
+ , cpreg.ctrl.CPIntEnable ? "ON" : "OFF"
+ , cpreg.ctrl.BPEnable ? "ON" : "OFF"
+ , cpreg.ctrl.FifoOverflowIntEnable ? "ON" : "OFF"
+ , cpreg.ctrl.FifoUnderflowIntEnable ? "ON" : "OFF"
+ );
+
+ if (cpreg.ctrl.CPIntEnable && (bpInt || ovfInt || undfInt))
+ {
+ DEBUG_LOG(COMMANDPROCESSOR,"Interrupt set");
+ g_VideoInitialize.pSetInterrupt(INT_CAUSE_CP, true);
+ }
+ else
+ {
+ DEBUG_LOG(COMMANDPROCESSOR,"Interrupt cleared");
+ g_VideoInitialize.pSetInterrupt(INT_CAUSE_CP, false);
+ }
+}
+
+void UpdateInterruptsFromVideoPlugin()
+{
+ g_VideoInitialize.pScheduleEvent_Threadsafe(0, et_UpdateInterrupts, 0);
+}
+
+void ReadFifo()
+{
+ bool updateInterrupts = false;
+
+ cpreg.status.ReadIdle = 0;
+
+ // update rwdistance
+ u32 writePtr = cpreg.writeptr;
+ if (cpreg.readptr <= writePtr)
+ cpreg.rwdistance = writePtr - cpreg.readptr;
+ else
+ cpreg.rwdistance = ((cpreg.fifoend & 0xFFFFFFE0) - cpreg.fifobase) - (cpreg.readptr - writePtr);
+
+ // overflow check
+ cpreg.status.OverflowHiWatermark = cpreg.rwdistance < cpreg.hiwatermark?0:1;
+ updateInterrupts |= cpreg.ctrl.FifoOverflowIntEnable && cpreg.status.OverflowHiWatermark;
+
+ // read from fifo
+ u8 *ptr = g_VideoInitialize.pGetMemoryPointer(cpreg.readptr);
+
+ u32 readptr = cpreg.readptr;
+ u32 distance = cpreg.rwdistance;
+
+ while (distance >= commandBufferCopySize && !cpreg.status.Breakpoint && writePos < maxCommandBufferWrite)
+ {
+ // check for breakpoint
+ // todo - check if this is precise enough
+ if (cpreg.ctrl.BPEnable && (cpreg.breakpt & 0xFFFFFFE0) == (readptr & 0xFFFFFFE0))
+ {
+ cpreg.status.Breakpoint = 1;
+ DEBUG_LOG(VIDEO,"Hit breakpoint at %x", readptr);
+ if (cpreg.ctrl.CPIntEnable)
+ updateInterrupts = true;
+ }
+ else
+ {
+ // copy to buffer
+ memcpy(&commandBuffer[writePos], ptr, commandBufferCopySize);
+ writePos += commandBufferCopySize;
+ ptr += commandBufferCopySize;
+ readptr += commandBufferCopySize;
+ distance -= commandBufferCopySize;
+ }
+
+ if (readptr >= (cpreg.fifoend & 0xFFFFFFE0))
+ {
+ readptr = cpreg.fifobase;
+ ptr = g_VideoInitialize.pGetMemoryPointer(readptr);
+ }
+ }
+
+ // lock read pointer until rw distance is updated?
+ cpreg.readptr = readptr;
+ cpreg.rwdistance = distance;
+
+ // underflow check
+ cpreg.status.UnderflowLoWatermark = cpreg.rwdistance > cpreg.lowatermark?0:1;
+ updateInterrupts |= cpreg.ctrl.FifoUnderflowIntEnable && cpreg.status.UnderflowLoWatermark;
+
+ cpreg.status.ReadIdle = 1;
+
+ if (updateInterrupts)
+ UpdateInterruptsFromVideoPlugin();
+}
+
+bool RunBuffer()
+{
+ // fifo is read 32 bytes at a time
+ // read fifo data to internal buffer
+ if (cpreg.ctrl.GPReadEnable)
+ ReadFifo();
+
+ g_pVideoData = &commandBuffer[readPos];
+
+ u32 availableBytes = writePos - readPos;
+ _dbg_assert_(VIDEO, writePos >= readPos);
+
+ while (OpcodeDecoder::CommandRunnable(availableBytes))
+ {
+ cpreg.status.CommandIdle = 0;
+
+ OpcodeDecoder::Run(availableBytes);
+
+ // if data was read by the opcode decoder then the video data pointer changed
+ readPos = g_pVideoData - &commandBuffer[0];
+ _dbg_assert_(VIDEO, writePos >= readPos);
+ availableBytes = writePos - readPos;
+ }
+
+ cpreg.status.CommandIdle = 1;
+
+ _dbg_assert_(VIDEO, writePos >= readPos);
+
+ bool ranDecoder = false;
+
+ // move data remaing in command buffer
+ if (readPos > 0)
+ {
+ memmove(&commandBuffer[0], &commandBuffer[readPos], availableBytes);
+ writePos -= readPos;
+ readPos = 0;
+
+ ranDecoder = true;
+ }
+
+ return ranDecoder;
+}
+
+} // end of namespace CommandProcessor
\ No newline at end of file
diff --git a/Source/Plugins/Plugin_VideoSoftware/Src/CommandProcessor.h b/Source/Plugins/Plugin_VideoSoftware/Src/CommandProcessor.h
new file mode 100644
index 0000000000..8a7b0ef52f
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Src/CommandProcessor.h
@@ -0,0 +1,153 @@
+// 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/
+
+#ifndef _COMMANDPROCESSOR_H_
+#define _COMMANDPROCESSOR_H_
+
+#include "Common.h"
+#include "pluginspecs_video.h"
+class PointerWrap;
+
+namespace CommandProcessor
+{
+ // internal hardware addresses
+ enum
+ {
+ STATUS_REGISTER = 0x00,
+ CTRL_REGISTER = 0x02,
+ CLEAR_REGISTER = 0x04,
+ FIFO_TOKEN_REGISTER = 0x0E,
+ FIFO_BOUNDING_BOX_LEFT = 0x10,
+ FIFO_BOUNDING_BOX_RIGHT = 0x12,
+ FIFO_BOUNDING_BOX_TOP = 0x14,
+ FIFO_BOUNDING_BOX_BOTTOM = 0x16,
+ FIFO_BASE_LO = 0x20,
+ FIFO_BASE_HI = 0x22,
+ FIFO_END_LO = 0x24,
+ FIFO_END_HI = 0x26,
+ FIFO_HI_WATERMARK_LO = 0x28,
+ FIFO_HI_WATERMARK_HI = 0x2a,
+ FIFO_LO_WATERMARK_LO = 0x2c,
+ FIFO_LO_WATERMARK_HI = 0x2e,
+ FIFO_RW_DISTANCE_LO = 0x30,
+ FIFO_RW_DISTANCE_HI = 0x32,
+ FIFO_WRITE_POINTER_LO = 0x34,
+ FIFO_WRITE_POINTER_HI = 0x36,
+ FIFO_READ_POINTER_LO = 0x38,
+ FIFO_READ_POINTER_HI = 0x3A,
+ FIFO_BP_LO = 0x3C,
+ FIFO_BP_HI = 0x3E
+ };
+
+ // Fifo Status Register
+ union UCPStatusReg
+ {
+ struct
+ {
+ unsigned OverflowHiWatermark : 1;
+ unsigned UnderflowLoWatermark : 1;
+ unsigned ReadIdle : 1; // done reading
+ unsigned CommandIdle : 1; // done processing commands
+ unsigned Breakpoint : 1;
+ unsigned : 11;
+ };
+ u16 Hex;
+ UCPStatusReg() {Hex = 0; }
+ UCPStatusReg(u16 _hex) {Hex = _hex; }
+ };
+
+ // Fifo Control Register
+ union UCPCtrlReg
+ {
+ struct
+ {
+ unsigned GPReadEnable : 1;
+ unsigned CPIntEnable : 1;
+ unsigned FifoOverflowIntEnable : 1;
+ unsigned FifoUnderflowIntEnable : 1;
+ unsigned GPLinkEnable : 1;
+ unsigned BPEnable : 1;
+ unsigned : 10;
+ };
+ u16 Hex;
+ UCPCtrlReg() {Hex = 0; }
+ UCPCtrlReg(u16 _hex) {Hex = _hex; }
+ };
+
+ // Fifo Control Register
+ union UCPClearReg
+ {
+ struct
+ {
+ unsigned ClearFifoOverflow : 1;
+ unsigned ClearFifoUnderflow : 1;
+ unsigned ClearMetrices : 1;
+ unsigned : 13;
+ };
+ u16 Hex;
+ UCPClearReg() {Hex = 0; }
+ UCPClearReg(u16 _hex) {Hex = _hex; }
+ };
+
+ struct CPReg
+ {
+ UCPStatusReg status; // 0x00
+ UCPCtrlReg ctrl; // 0x02
+ UCPClearReg clear; // 0x04
+ u32 unk0; // 0x06
+ u32 unk1; // 0x0a
+ u16 token; // 0x0e
+ u16 bboxleft; // 0x10
+ u16 bboxtop; // 0x12
+ u16 bboxright; // 0x14
+ u16 bboxbottom; // 0x16
+ u16 unk2; // 0x18
+ u32 fifobase; // 0x20
+ u32 fifoend; // 0x24
+ u32 hiwatermark; // 0x28
+ u32 lowatermark; // 0x2c
+ u32 rwdistance; // 0x30
+ u32 writeptr; // 0x34
+ u32 readptr; // 0x38
+ u32 breakpt; // 0x3c
+ };
+
+ extern CPReg cpreg;
+
+ // Init
+ void Init();
+ void Shutdown();
+ void DoState(PointerWrap &p);
+
+ bool RunBuffer();
+
+ // Read
+ void Read16(u16& _rReturnValue, const u32 _Address);
+ void Write16(const u16 _Data, const u32 _Address);
+ void Read32(u32& _rReturnValue, const u32 _Address);
+ void Write32(const u32 _Data, const u32 _Address);
+
+ // for CGPFIFO
+ void GatherPipeBursted();
+ void UpdateInterrupts();
+ void UpdateInterruptsFromVideoPlugin();
+
+
+} // end of namespace CommandProcessor
+
+
+#endif
\ No newline at end of file
diff --git a/Source/Plugins/Plugin_VideoSoftware/Src/DebugUtil.cpp b/Source/Plugins/Plugin_VideoSoftware/Src/DebugUtil.cpp
new file mode 100644
index 0000000000..80232e1b1d
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Src/DebugUtil.cpp
@@ -0,0 +1,177 @@
+// 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 "main.h"
+#include "DebugUtil.h"
+#include "BPMemLoader.h"
+#include "TextureSampler.h"
+#include "VideoConfig.h"
+#include "EfbInterface.h"
+#include "Statistics.h"
+#include "HwRasterizer.h"
+#include "StringUtil.h"
+#include "../../../Core/VideoCommon/Src/ImageWrite.h"
+
+namespace DebugUtil
+{
+
+u32 skipFrames = 0;
+
+bool SaveTexture(const char* filename, u32 texmap, int width, int height)
+{
+ u8 *data = new u8[width * height * 4];
+
+ GetTextureBGRA(data, texmap, width, height);
+
+ bool result = SaveTGA(filename, width, height, data);
+
+ delete []data;
+
+ return result;
+}
+
+void SaveTexture(const char* filename, u32 texmap)
+{
+ FourTexUnits& texUnit = bpmem.tex[(texmap >> 2) & 1];
+ u8 subTexmap = texmap & 3;
+
+ TexImage0& ti0 = texUnit.texImage0[subTexmap];
+
+ SaveTexture(filename, texmap, ti0.width, ti0.height);
+}
+
+void GetTextureBGRA(u8 *dst, u32 texmap, int width, int height)
+{
+ u8 sample[4];
+
+ for (int y = 0; y < height; y++)
+ for (int x = 0; x < width; x++) {
+ TextureSampler::Sample((float)x, (float)y, 0, texmap, sample);
+ // rgba to bgra
+ *(dst++) = sample[2];
+ *(dst++) = sample[1];
+ *(dst++) = sample[0];
+ *(dst++) = sample[3];
+ }
+}
+
+void DumpActiveTextures()
+{
+ for (unsigned int stageNum = 0; stageNum < bpmem.genMode.numindstages; stageNum++)
+ {
+ u32 texmap = bpmem.tevindref.getTexMap(stageNum);
+
+ SaveTexture(StringFromFormat("%s/tar%i_ind%i_map%i.tga", FULL_DUMP_TEXTURES_DIR, stats.thisFrame.numDrawnObjects, stageNum, texmap).c_str(), texmap);
+ }
+
+ for (unsigned int stageNum = 0; stageNum <= bpmem.genMode.numtevstages; stageNum++)
+ {
+ int stageNum2 = stageNum >> 1;
+ int stageOdd = stageNum&1;
+ TwoTevStageOrders &order = bpmem.tevorders[stageNum2];
+
+ int texmap = order.getTexMap(stageOdd);
+
+ SaveTexture(StringFromFormat("%s/tar%i_stage%i_map%i.tga", FULL_DUMP_TEXTURES_DIR, stats.thisFrame.numDrawnObjects, stageNum, texmap).c_str(), texmap);
+ }
+}
+
+void DumpEfb(const char* filename)
+{
+ u8 *data = new u8[EFB_WIDTH * EFB_HEIGHT * 4];
+ u8 *writePtr = data;
+ u8 sample[4];
+
+ for (int y = 0; y < EFB_HEIGHT; y++)
+ for (int x = 0; x < EFB_WIDTH; x++) {
+ EfbInterface::GetColor(x, y, sample);
+ // rgba to bgra
+ *(writePtr++) = sample[2];
+ *(writePtr++) = sample[1];
+ *(writePtr++) = sample[0];
+ *(writePtr++) = sample[3];
+ }
+
+ bool result = SaveTGA(filename, EFB_WIDTH, EFB_HEIGHT, data);
+
+ delete []data;
+}
+
+void DumpDepth(const char* filename)
+{
+ u8 *data = new u8[EFB_WIDTH * EFB_HEIGHT * 4];
+ u8 *writePtr = data;
+
+ for (int y = 0; y < EFB_HEIGHT; y++)
+ for (int x = 0; x < EFB_WIDTH; x++) {
+ u32 depth = EfbInterface::GetDepth(x, y);
+ // depth to bgra
+ *(writePtr++) = (depth >> 16) & 0xff;
+ *(writePtr++) = (depth >> 8) & 0xff;
+ *(writePtr++) = depth & 0xff;
+ *(writePtr++) = 255;
+ }
+
+ bool result = SaveTGA(filename, EFB_WIDTH, EFB_HEIGHT, data);
+
+ delete []data;
+}
+
+void OnObjectBegin()
+{
+ if (!g_SkipFrame)
+ {
+ if (g_Config.bDumpTextures)
+ DumpActiveTextures();
+
+ if (g_Config.bHwRasterizer)
+ HwRasterizer::BeginTriangles();
+ }
+}
+
+void OnObjectEnd()
+{
+ if (!g_SkipFrame)
+ {
+ if (g_Config.bDumpObjects)
+ DumpEfb(StringFromFormat("%s/object%i.tga", FULL_FRAMES_DIR, stats.thisFrame.numDrawnObjects).c_str());
+
+ if (g_Config.bHwRasterizer)
+ HwRasterizer::EndTriangles();
+
+ stats.thisFrame.numDrawnObjects++;
+ }
+}
+
+void OnFrameEnd()
+{
+ if (!g_SkipFrame)
+ {
+ if (g_Config.bDumpFrames)
+ {
+ DumpEfb(StringFromFormat("%s/frame%i_color.tga", FULL_FRAMES_DIR, stats.frameCount).c_str());
+ DumpDepth(StringFromFormat("%s/frame%i_depth.tga", FULL_FRAMES_DIR, stats.frameCount).c_str());
+ }
+ }
+
+ g_SkipFrame = skipFrames > 0;
+ skipFrames = g_SkipFrame?(skipFrames-1):g_Config.nFrameSkip;
+}
+
+}
\ No newline at end of file
diff --git a/Source/Plugins/Plugin_VideoSoftware/Src/DebugUtil.h b/Source/Plugins/Plugin_VideoSoftware/Src/DebugUtil.h
new file mode 100644
index 0000000000..f26ffcd4c4
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Src/DebugUtil.h
@@ -0,0 +1,33 @@
+// 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/
+
+#ifndef _DEBUGUTIL_H
+#define _DEBUGUTIL_H
+
+namespace DebugUtil
+{
+ void GetTextureBGRA(u8 *dst, u32 texmap, int width, int height);
+
+ void DumpActiveTextures();
+
+ void OnObjectBegin();
+ void OnObjectEnd();
+
+ void OnFrameEnd();
+}
+
+#endif
\ No newline at end of file
diff --git a/Source/Plugins/Plugin_VideoSoftware/Src/EfbCopy.cpp b/Source/Plugins/Plugin_VideoSoftware/Src/EfbCopy.cpp
new file mode 100644
index 0000000000..96ecf82ba3
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Src/EfbCopy.cpp
@@ -0,0 +1,99 @@
+// 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 "main.h"
+#include "BPMemLoader.h"
+#include "EfbCopy.h"
+#include "EfbInterface.h"
+#include "Renderer.h"
+#include "TextureEncoder.h"
+#include "Statistics.h"
+#include "VideoConfig.h"
+#include "DebugUtil.h"
+#include "HwRasterizer.h"
+
+
+namespace EfbCopy
+{
+ void CopyToXfb()
+ {
+ if (!g_Config.bHwRasterizer)
+ {
+ // copy to open gl for rendering
+ EfbInterface::UpdateColorTexture();
+ Renderer::DrawTexture(EfbInterface::efbColorTexture, EFB_WIDTH, EFB_HEIGHT);
+ }
+
+ Renderer::SwapBuffer();
+
+ }
+
+ void CopyToRam()
+ {
+ u8 *dest_ptr = g_VideoInitialize.pGetMemoryPointer(bpmem.copyTexDest << 5);
+
+ TextureEncoder::Encode(dest_ptr);
+ }
+
+ void ClearEfb()
+ {
+ u32 clearColor = (bpmem.clearcolorAR & 0xff) | Common::swap16(bpmem.clearcolorGB) << 8 | (bpmem.clearcolorAR & 0xff00) << 16;
+
+ int left = bpmem.copyTexSrcXY.x;
+ int top = bpmem.copyTexSrcXY.y;
+ int right = left + bpmem.copyTexSrcWH.x;
+ int bottom = top + bpmem.copyTexSrcWH.y;
+
+ for (u16 y = top; y <= bottom; y++)
+ {
+ for (u16 x = left; x <= right; x++)
+ {
+ EfbInterface::SetColor(x, y, (u8*)(&clearColor));
+ EfbInterface::SetDepth(x, y, bpmem.clearZValue);
+ }
+ }
+ }
+
+ void CopyEfb()
+ {
+ if (bpmem.triggerEFBCopy.copy_to_xfb)
+ DebugUtil::OnFrameEnd();
+
+ if (!g_SkipFrame)
+ {
+ if (bpmem.triggerEFBCopy.copy_to_xfb)
+ {
+ CopyToXfb();
+ g_VideoInitialize.pCopiedToXFB(false);
+
+ stats.frameCount++;
+ }
+ else
+ {
+ CopyToRam();
+ }
+
+ if (bpmem.triggerEFBCopy.clear)
+ {
+ if (g_Config.bHwRasterizer)
+ HwRasterizer::Clear();
+ else
+ ClearEfb();
+ }
+ }
+ }
+}
\ No newline at end of file
diff --git a/Source/Plugins/Plugin_VideoSoftware/Src/EfbCopy.h b/Source/Plugins/Plugin_VideoSoftware/Src/EfbCopy.h
new file mode 100644
index 0000000000..c19f9365db
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Src/EfbCopy.h
@@ -0,0 +1,32 @@
+// 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/
+
+#ifndef _EFB_COPY_H_
+#define _EFB_COPY_H_
+
+
+#include "Common.h"
+
+namespace EfbCopy
+{
+ // Copy the EFB to RAM as a texture format or XFB
+ // Clear the EFB if needed
+ void CopyEfb();
+}
+
+
+#endif
\ No newline at end of file
diff --git a/Source/Plugins/Plugin_VideoSoftware/Src/EfbInterface.cpp b/Source/Plugins/Plugin_VideoSoftware/Src/EfbInterface.cpp
new file mode 100644
index 0000000000..10153f20e5
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Src/EfbInterface.cpp
@@ -0,0 +1,545 @@
+// 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 "EfbInterface.h"
+#include "BPMemLoader.h"
+#include "../../../Core/VideoCommon/Src/LookUpTables.h"
+#include "PixelEngine.h"
+
+
+u8 efb[EFB_WIDTH*EFB_HEIGHT*6];
+
+
+namespace EfbInterface
+{
+ u8 efbColorTexture[EFB_WIDTH*EFB_HEIGHT*4];
+
+ inline u32 GetColorOffset(u16 x, u16 y)
+ {
+ return (x + y * EFB_WIDTH) * 3;
+ }
+
+ inline u32 GetDepthOffset(u16 x, u16 y)
+ {
+ return (x + y * EFB_WIDTH) * 3 + DEPTH_BUFFER_START;
+ }
+
+ void SetPixelAlphaOnly(u32 offset, u8 a)
+ {
+ switch (bpmem.zcontrol.pixel_format)
+ {
+ case PIXELFMT_RGB8_Z24:
+ case PIXELFMT_Z24:
+ case PIXELFMT_RGB565_Z16:
+ // do nothing
+ break;
+ case PIXELFMT_RGBA6_Z24:
+ {
+ u32 a32 = a;
+ u32 *dst = (u32*)&efb[offset];
+ u32 val = *dst & 0xff03ffff;
+ val |= (a32 << 16) & 0xfc0000;
+ *dst = val;
+ }
+ break;
+ default:
+ ERROR_LOG(VIDEO, "Unsupported pixel format: %i", bpmem.zcontrol.pixel_format);
+ }
+ }
+
+ void SetPixelColorOnly(u32 offset, u8 *rgb)
+ {
+ switch (bpmem.zcontrol.pixel_format)
+ {
+ case PIXELFMT_RGB8_Z24:
+ case PIXELFMT_Z24:
+ {
+ u32 src = *(u32*)rgb;
+ u32 *dst = (u32*)&efb[offset];
+ u32 val = *dst & 0xff000000;
+ val |= src & 0x00ffffff;
+ *dst = val;
+ }
+ break;
+ case PIXELFMT_RGBA6_Z24:
+ {
+ u32 src = *(u32*)rgb;
+ u32 *dst = (u32*)&efb[offset];
+ u32 val = *dst & 0xfffc0000;
+ val |= (src >> 2) & 0x3f;
+ val |= (src >> 4) & 0xfc0;
+ val |= (src >> 6) & 0x3f000;
+ *dst = val;
+ }
+ break;
+ case PIXELFMT_RGB565_Z16:
+ {
+ INFO_LOG(VIDEO, "PIXELFMT_RGB565_Z16 is not supported correctly yet");
+ u32 src = *(u32*)rgb;
+ u32 *dst = (u32*)&efb[offset];
+ u32 val = *dst & 0xff000000;
+ val |= src & 0x00ffffff;
+ *dst = val;
+ }
+ break;
+ default:
+ ERROR_LOG(VIDEO, "Unsupported pixel format: %i", bpmem.zcontrol.pixel_format);
+ }
+ }
+
+ void SetPixelAlphaColor(u32 offset, u8 *color)
+ {
+ switch (bpmem.zcontrol.pixel_format)
+ {
+ case PIXELFMT_RGB8_Z24:
+ case PIXELFMT_Z24:
+ {
+ u32 src = *(u32*)color;
+ u32 *dst = (u32*)&efb[offset];
+ u32 val = *dst & 0xff000000;
+ val |= src & 0x00ffffff;
+ *dst = val;
+ }
+ break;
+ case PIXELFMT_RGBA6_Z24:
+ {
+ u32 src = *(u32*)color;
+ u32 *dst = (u32*)&efb[offset];
+ u32 val = *dst & 0xff000000;
+ val |= (src >> 2) & 0x3f;
+ val |= (src >> 4) & 0xfc0;
+ val |= (src >> 6) & 0x3f000;
+ val |= (src >> 8) & 0xfc0000;
+ *dst = val;
+ }
+ break;
+ case PIXELFMT_RGB565_Z16:
+ {
+ INFO_LOG(VIDEO, "PIXELFMT_RGB565_Z16 is not supported correctly yet");
+ u32 src = *(u32*)color;
+ u32 *dst = (u32*)&efb[offset];
+ u32 val = *dst & 0xff000000;
+ val |= src & 0x00ffffff;
+ *dst = val;
+ }
+ break;
+ default:
+ ERROR_LOG(VIDEO, "Unsupported pixel format: %i", bpmem.zcontrol.pixel_format);
+ }
+ }
+
+ void GetPixelColor(u32 offset, u8 *color)
+ {
+ switch (bpmem.zcontrol.pixel_format)
+ {
+ case PIXELFMT_RGB8_Z24:
+ case PIXELFMT_Z24:
+ {
+ u32 src = *(u32*)&efb[offset];
+ u32 *dst = (u32*)color;
+ u32 val = 0xff000000 | (src & 0x00ffffff);
+ *dst = val;
+ }
+ break;
+ case PIXELFMT_RGBA6_Z24:
+ {
+ u32 src = *(u32*)&efb[offset];
+ color[0] = Convert6To8(src & 0x3f);
+ color[1] = Convert6To8((src >> 6) & 0x3f);
+ color[2] = Convert6To8((src >> 12) & 0x3f);
+ color[3] = Convert6To8((src >> 18) & 0x3f);
+ }
+ break;
+ case PIXELFMT_RGB565_Z16:
+ {
+ INFO_LOG(VIDEO, "PIXELFMT_RGB565_Z16 is not supported correctly yet");
+ u32 src = *(u32*)&efb[offset];
+ u32 *dst = (u32*)color;
+ u32 val = 0xff000000 | (src & 0x00ffffff);
+ *dst = val;
+ }
+ break;
+ default:
+ ERROR_LOG(VIDEO, "Unsupported pixel format: %i", bpmem.zcontrol.pixel_format);
+ }
+ }
+
+ void SetPixelDepth(u32 offset, u32 depth)
+ {
+ switch (bpmem.zcontrol.pixel_format)
+ {
+ case PIXELFMT_RGB8_Z24:
+ case PIXELFMT_RGBA6_Z24:
+ case PIXELFMT_Z24:
+ {
+ u32 *dst = (u32*)&efb[offset];
+ u32 val = *dst & 0xff000000;
+ val |= depth & 0x00ffffff;
+ *dst = val;
+ }
+ break;
+ case PIXELFMT_RGB565_Z16:
+ {
+ INFO_LOG(VIDEO, "PIXELFMT_RGB565_Z16 is not supported correctly yet");
+ u32 *dst = (u32*)&efb[offset];
+ u32 val = *dst & 0xff000000;
+ val |= depth & 0x00ffffff;
+ *dst = val;
+ }
+ break;
+ default:
+ ERROR_LOG(VIDEO, "Unsupported pixel format: %i", bpmem.zcontrol.pixel_format);
+ }
+ }
+
+ u32 GetPixelDepth(u32 offset)
+ {
+ u32 depth = 0;
+
+ switch (bpmem.zcontrol.pixel_format)
+ {
+ case PIXELFMT_RGB8_Z24:
+ case PIXELFMT_RGBA6_Z24:
+ case PIXELFMT_Z24:
+ {
+ depth = (*(u32*)&efb[offset]) & 0x00ffffff;
+ }
+ break;
+ case PIXELFMT_RGB565_Z16:
+ {
+ INFO_LOG(VIDEO, "PIXELFMT_RGB565_Z16 is not supported correctly yet");
+ depth = (*(u32*)&efb[offset]) & 0x00ffffff;
+ }
+ break;
+ default:
+ ERROR_LOG(VIDEO, "Unsupported pixel format: %i", bpmem.zcontrol.pixel_format);
+ }
+
+ return depth;
+ }
+
+ u32 GetSourceFactor(u8 *srcClr, u8 *dstClr, int mode)
+ {
+ switch (mode) {
+ case 0: // zero
+ return 0;
+ case 1: // one
+ return 0xffffffff;
+ case 2: // dstclr
+ return *(u32*)dstClr;
+ case 3: // invdstclr
+ return 0xffffffff - *(u32*)dstClr;
+ case 4: // srcalpha
+ {
+ u8 alpha = srcClr[3];
+ u32 factor = alpha << 24 | alpha << 16 | alpha << 8 | alpha;
+ return factor;
+ }
+ case 5: // invsrcalpha
+ {
+ u8 alpha = 0xff - srcClr[3];
+ u32 factor = alpha << 24 | alpha << 16 | alpha << 8 | alpha;
+ return factor;
+ }
+ case 6: // dstalpha
+ {
+ u8 alpha = dstClr[3];
+ u32 factor = alpha << 24 | alpha << 16 | alpha << 8 | alpha;
+ return factor;
+ }
+ case 7: // invdstalpha
+ {
+ u8 alpha = 0xff - dstClr[3];
+ u32 factor = alpha << 24 | alpha << 16 | alpha << 8 | alpha;
+ return factor;
+ }
+ }
+
+ return 0;
+ }
+
+ u32 GetDestinationFactor(u8 *srcClr, u8 *dstClr, int mode)
+ {
+ switch (mode) {
+ case 0: // zero
+ return 0;
+ case 1: // one
+ return 0xffffffff;
+ case 2: // srcclr
+ return *(u32*)srcClr;
+ case 3: // invsrcclr
+ return 0xffffffff - *(u32*)srcClr;
+ case 4: // srcalpha
+ {
+ u8 alpha = srcClr[3];
+ u32 factor = alpha << 24 | alpha << 16 | alpha << 8 | alpha;
+ return factor;
+ }
+ case 5: // invsrcalpha
+ {
+ u8 alpha = 0xff - srcClr[3];
+ u32 factor = alpha << 24 | alpha << 16 | alpha << 8 | alpha;
+ return factor;
+ }
+ case 6: // dstalpha
+ {
+ u8 alpha = dstClr[3] & 0xff;
+ u32 factor = alpha << 24 | alpha << 16 | alpha << 8 | alpha;
+ return factor;
+ }
+ case 7: // invdstalpha
+ {
+ u8 alpha = 0xff - dstClr[3];
+ u32 factor = alpha << 24 | alpha << 16 | alpha << 8 | alpha;
+ return factor;
+ }
+ }
+
+ return 0;
+ }
+
+ void BlendColor(u8 *srcClr, u8 *dstClr)
+ {
+ u32 srcFactor = GetSourceFactor(srcClr, dstClr, bpmem.blendmode.srcfactor);
+ u32 dstFactor = GetDestinationFactor(srcClr, dstClr, bpmem.blendmode.dstfactor);
+
+ for (int i = 0; i < 4; i++)
+ {
+ // add MSB of factors to make their range 0 -> 256
+ u32 sf = (srcFactor & 0xff);
+ sf += sf >> 7;
+
+ u32 df = (dstFactor & 0xff);
+ df += df >> 7;
+
+ u32 color = (srcClr[i] * sf + dstClr[i] * df) >> 8;
+ dstClr[i] = (color>255)?255:color;
+
+ dstFactor >>= 8;
+ srcFactor >>= 8;
+ }
+ }
+
+ void LogicBlend(u32 srcClr, u32 &dstClr, int op)
+ {
+ switch (op) {
+ case 0: // clear
+ dstClr = 0;
+ break;
+ case 1: // and
+ dstClr = srcClr & dstClr;
+ break;
+ case 2: // revand
+ dstClr = srcClr & (~dstClr);
+ break;
+ case 3: // copy
+ dstClr = srcClr;
+ break;
+ case 4: // invand
+ dstClr = (~srcClr) & dstClr;
+ break;
+ case 5: // noop
+ dstClr = dstClr;
+ break;
+ case 6: // xor
+ dstClr = srcClr ^ dstClr;
+ break;
+ case 7: // or
+ dstClr = srcClr | dstClr;
+ break;
+ case 8: // nor
+ dstClr = ~(srcClr | dstClr);
+ break;
+ case 9: // equiv
+ dstClr = ~(srcClr ^ dstClr);
+ break;
+ case 10: // inv
+ dstClr = ~dstClr;
+ break;
+ case 11: // revor
+ dstClr = srcClr | (~dstClr);
+ break;
+ case 12: // invcopy
+ dstClr = ~srcClr;
+ break;
+ case 13: // invor
+ dstClr = (~srcClr) | dstClr;
+ break;
+ case 14: // nand
+ dstClr = ~(srcClr & dstClr);
+ break;
+ case 15: // set
+ dstClr = 0xffffffff;
+ break;
+ }
+ }
+
+ void SubtractBlend(u8 *srcClr, u8 *dstClr)
+ {
+ for (int i = 0; i < 4; i++)
+ {
+ int c = (int)dstClr[i] - (int)srcClr[i];
+ dstClr[i] = (c < 0)?0:c;
+ }
+ }
+
+ void BlendTev(u16 x, u16 y, u8 *color)
+ {
+ u32 dstClr;
+ u32 offset = GetColorOffset(x, y);
+
+ u8 *dstClrPtr = (u8*)&dstClr;
+
+ GetPixelColor(offset, dstClrPtr);
+
+ if (bpmem.blendmode.blendenable)
+ {
+ if (bpmem.blendmode.subtract)
+ SubtractBlend(color, dstClrPtr);
+ else
+ BlendColor(color, dstClrPtr);
+ }
+ else if (bpmem.blendmode.logicopenable)
+ LogicBlend(*((u32*)color), dstClr, bpmem.blendmode.logicmode);
+ else
+ dstClrPtr = color;
+
+ if (bpmem.dstalpha.enable)
+ dstClrPtr[3] = bpmem.dstalpha.alpha;
+
+ if (bpmem.blendmode.colorupdate)
+ {
+ if (bpmem.blendmode.alphaupdate)
+ SetPixelAlphaColor(offset, dstClrPtr);
+ else
+ SetPixelColorOnly(offset, dstClrPtr);
+ }
+ else if (bpmem.blendmode.alphaupdate)
+ SetPixelAlphaOnly(offset, dstClrPtr[3]);
+
+ // branchless bounding box update
+ PixelEngine::pereg.boxLeft = PixelEngine::pereg.boxLeft>x?x:PixelEngine::pereg.boxLeft;
+ PixelEngine::pereg.boxRight = PixelEngine::pereg.boxRighty?y:PixelEngine::pereg.boxTop;
+ PixelEngine::pereg.boxBottom = PixelEngine::pereg.boxBottom depth;
+ break;
+ case COMPARE_NEQUAL:
+ pass = z != depth;
+ break;
+ case COMPARE_GEQUAL:
+ pass = z >= depth;
+ break;
+ case COMPARE_ALWAYS:
+ pass = true;
+ break;
+ default:
+ pass = false;
+ ERROR_LOG(VIDEO, "Bad Z compare mode %i", bpmem.zmode.func);
+ }
+
+ if (pass && bpmem.zmode.updateenable)
+ {
+ SetPixelDepth(offset, z);
+ }
+
+ return pass;
+ }
+}
\ No newline at end of file
diff --git a/Source/Plugins/Plugin_VideoSoftware/Src/EfbInterface.h b/Source/Plugins/Plugin_VideoSoftware/Src/EfbInterface.h
new file mode 100644
index 0000000000..71ecfe8cff
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Src/EfbInterface.h
@@ -0,0 +1,50 @@
+// 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/
+
+#ifndef _EFB_INTERFACE_H_
+#define _EFB_INTERFACE_H_
+
+#include "VideoCommon.h"
+
+namespace EfbInterface
+{
+ const int DEPTH_BUFFER_START = EFB_WIDTH * EFB_HEIGHT * 3;
+
+ // color order is RGBA
+
+ // does full blending of an incoming pixel
+ void BlendTev(u16 x, u16 y, u8 *color);
+
+ // compare z at location x,y
+ // writes it if it passes
+ // returns result of compare.
+ bool ZCompare(u16 x, u16 y, u32 z);
+
+ // sets the color and alpha
+ void SetColor(u16 x, u16 y, u8 *color);
+ void SetDepth(u16 x, u16 y, u32 depth);
+
+ void GetColor(u16 x, u16 y, u8 *color);
+ u32 GetDepth(u16 x, u16 y);
+
+ u8* GetPixelPointer(u16 x, u16 y, bool depth);
+
+ void UpdateColorTexture();
+ extern u8 efbColorTexture[EFB_WIDTH*EFB_HEIGHT*4]; // rgba format
+}
+
+#endif
\ No newline at end of file
diff --git a/Source/Plugins/Plugin_VideoSoftware/Src/GLUtil.cpp b/Source/Plugins/Plugin_VideoSoftware/Src/GLUtil.cpp
new file mode 100644
index 0000000000..cd85612e13
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Src/GLUtil.cpp
@@ -0,0 +1,754 @@
+// 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 "main.h"
+#include "VideoConfig.h"
+#include "IniFile.h"
+#include "svnrev.h"
+#include "Setup.h"
+
+//#include "Render.h"
+
+#if defined(_WIN32)
+#include "Win32.h"
+#else
+struct RECT
+{
+ int left, top;
+ int right, bottom;
+};
+#endif
+
+#include "GLUtil.h"
+
+// Handles OpenGL and the window
+
+// Window dimensions.
+static int s_backbuffer_width;
+static int s_backbuffer_height;
+
+#ifndef _WIN32
+GLWindow GLWin;
+#endif
+
+#if defined(_WIN32)
+static HDC hDC = NULL; // Private GDI Device Context
+static HGLRC hRC = NULL; // Permanent Rendering Context
+extern HINSTANCE g_hInstance;
+#endif
+
+void OpenGL_SwapBuffers()
+{
+#if USE_SDL
+ SDL_GL_SwapBuffers();
+#elif defined(HAVE_COCOA) && HAVE_COCOA
+ cocoaGLSwap(GLWin.cocoaCtx,GLWin.cocoaWin);
+#elif defined(_WIN32)
+ SwapBuffers(hDC);
+#elif defined(USE_WX) && USE_WX
+ GLWin.glCanvas->SwapBuffers();
+#elif defined(HAVE_X11) && HAVE_X11
+ glXSwapBuffers(GLWin.dpy, GLWin.win);
+#endif
+}
+
+u32 OpenGL_GetBackbufferWidth()
+{
+ return s_backbuffer_width;
+}
+
+u32 OpenGL_GetBackbufferHeight()
+{
+ return s_backbuffer_height;
+}
+
+void OpenGL_SetWindowText(const char *text)
+{
+#if USE_SDL
+ SDL_WM_SetCaption(text, NULL);
+#elif defined(HAVE_COCOA) && HAVE_COCOA
+ cocoaGLSetTitle(GLWin.cocoaWin, text);
+#elif defined(_WIN32)
+ // TODO convert text to unicode and change SetWindowTextA to SetWindowText
+ SetWindowTextA(EmuWindow::GetWnd(), text);
+#elif defined(USE_WX) && USE_WX
+ GLWin.frame->SetTitle(wxString::FromAscii(text));
+#elif defined(HAVE_X11) && HAVE_X11 // GLX
+ /**
+ * Tell X to ask the window manager to set the window title. (X
+ * itself doesn't provide window title functionality.)
+ */
+ XStoreName(GLWin.dpy, GLWin.win, text);
+#endif
+}
+
+// =======================================================================================
+// Draw messages on top of the screen
+// ------------------
+unsigned int Callback_PeekMessages()
+{
+#ifdef _WIN32
+ // TODO: peekmessage
+ MSG msg;
+ while (PeekMessage(&msg, 0, 0, 0, PM_REMOVE))
+ {
+ if (msg.message == WM_QUIT)
+ return FALSE;
+ TranslateMessage(&msg);
+ DispatchMessage(&msg);
+ }
+ return TRUE;
+#else
+ return FALSE;
+#endif
+}
+
+// Show the current FPS
+void UpdateFPSDisplay(const char *text)
+{
+ char temp[512];
+ sprintf(temp, "SVN R%s: SW: %s", SVN_REV_STR, text);
+ OpenGL_SetWindowText(temp);
+}
+// =========================
+
+
+// =======================================================================================
+// Create rendering window.
+// Call browser: Core.cpp:EmuThread() > main.cpp:Video_Initialize()
+// ------------------
+bool OpenGL_Create(SVideoInitialize &_VideoInitialize, int _twidth, int _theight)
+{
+ #if defined(_WIN32)
+ EmuWindow::SetSize(_twidth, _theight);
+ #endif
+ // ----------------------------
+
+ // ---------------------------------------------------------------------------------------
+ // Control window size and picture scaling
+ // ------------------
+ s_backbuffer_width = _twidth;
+ s_backbuffer_height = _theight;
+
+ g_VideoInitialize.pPeekMessages = &Callback_PeekMessages;
+ g_VideoInitialize.pUpdateFPSDisplay = &UpdateFPSDisplay;
+
+#if USE_SDL
+ //init sdl video
+ if (SDL_Init(SDL_INIT_VIDEO) < 0) {
+ //TODO : Display an error message
+ SDL_Quit();
+ return false;
+ }
+
+ //setup ogl to use double buffering
+ SDL_GL_SetAttribute(SDL_GL_DOUBLEBUFFER, 1);
+#elif defined(HAVE_COCOA) && HAVE_COCOA
+ GLWin.width = s_backbuffer_width;
+ GLWin.height = s_backbuffer_height;
+ GLWin.cocoaWin = cocoaGLCreateWindow(GLWin.width, GLWin.height);
+ GLWin.cocoaCtx = cocoaGLInit(g_Config.iMultisampleMode);
+#elif defined(USE_WX) && USE_WX
+ int args[] = {WX_GL_RGBA, WX_GL_DOUBLEBUFFER, WX_GL_DEPTH_SIZE, 16, 0};
+
+ wxSize size(_iwidth, _iheight);
+ if (!g_Config.renderToMainframe ||
+ g_VideoInitialize.pWindowHandle == NULL) {
+ GLWin.frame = new wxFrame((wxWindow *)g_VideoInitialize.pWindowHandle,
+ -1, _("Dolphin"), wxPoint(50,50), size);
+ } else {
+ GLWin.frame = new wxFrame((wxWindow *)NULL,
+ -1, _("Dolphin"), wxPoint(50,50), size);
+ }
+
+#if defined(__APPLE__)
+ GLWin.glCanvas = new wxGLCanvas(GLWin.frame, wxID_ANY, wxPoint(0,0), size, 0, wxT("Dolphin"), args, wxNullPalette);
+#else
+ GLWin.glCanvas = new wxGLCanvas(GLWin.frame, wxID_ANY, args,
+ wxPoint(0,0), size, wxSUNKEN_BORDER);
+ GLWin.glCtxt = new wxGLContext(GLWin.glCanvas);
+#endif
+
+ GLWin.frame->Show(TRUE);
+ GLWin.glCanvas->Show(TRUE);
+
+#if defined(__APPLE__)
+ GLWin.glCanvas->SetCurrent();
+#else
+ GLWin.glCanvas->SetCurrent(*GLWin.glCtxt);
+ // GLWin.glCtxt->SetCurrent(*GLWin.glCanvas);
+#endif
+
+
+#elif defined(_WIN32)
+ // ---------------------------------------------------------------------------------------
+ // Create rendering window in Windows
+ // ----------------------
+
+ // Create a separate window
+ if (!g_Config.renderToMainframe || g_VideoInitialize.pWindowHandle == NULL)
+ g_VideoInitialize.pWindowHandle = (void*)EmuWindow::Create(NULL, g_hInstance, _T("Please wait..."));
+ // Create a child window
+ else
+ g_VideoInitialize.pWindowHandle = (void*)EmuWindow::Create((HWND)g_VideoInitialize.pWindowHandle, g_hInstance, _T("Please wait..."));
+
+ // Show the window
+ EmuWindow::Show();
+
+ if (g_VideoInitialize.pWindowHandle == NULL)
+ {
+ g_VideoInitialize.pSysMessage("failed to create window");
+ return false;
+ }
+
+ GLuint PixelFormat; // Holds The Results After Searching For A Match
+ DWORD dwExStyle; // Window Extended Style
+ DWORD dwStyle; // Window Style
+
+ RECT rcdesktop;
+ GetWindowRect(GetDesktopWindow(), &rcdesktop);
+
+ if (g_Config.bFullscreen) {
+ //s_backbuffer_width = rcdesktop.right - rcdesktop.left;
+ //s_backbuffer_height = rcdesktop.bottom - rcdesktop.top;
+
+ DEVMODE dmScreenSettings;
+ memset(&dmScreenSettings,0,sizeof(dmScreenSettings));
+ dmScreenSettings.dmSize=sizeof(dmScreenSettings);
+ dmScreenSettings.dmPelsWidth = s_backbuffer_width;
+ dmScreenSettings.dmPelsHeight = s_backbuffer_height;
+ dmScreenSettings.dmBitsPerPel = 32;
+ dmScreenSettings.dmFields = DM_BITSPERPEL|DM_PELSWIDTH|DM_PELSHEIGHT;
+
+ // Try To Set Selected Mode And Get Results. NOTE: CDS_FULLSCREEN Gets Rid Of Start Bar.
+ if (ChangeDisplaySettings(&dmScreenSettings, CDS_FULLSCREEN) != DISP_CHANGE_SUCCESSFUL)
+ {
+ if (MessageBox(NULL,_T("The Requested Fullscreen Mode Is Not Supported By\nYour Video Card. Use Windowed Mode Instead?"),_T("NeHe GL"),MB_YESNO|MB_ICONEXCLAMATION)==IDYES)
+ g_Config.bFullscreen = false;
+ else
+ return false;
+ }
+ }
+ else
+ {
+ // Change to default resolution
+ ChangeDisplaySettings(NULL, 0);
+ }
+
+ if (g_Config.bFullscreen && !g_Config.renderToMainframe)
+ {
+ // Hide the cursor
+ ShowCursor(FALSE);
+ }
+ else
+ {
+ dwExStyle = WS_EX_APPWINDOW | WS_EX_WINDOWEDGE;
+ dwStyle = WS_OVERLAPPEDWINDOW;
+ }
+
+ RECT rc = {0, 0, s_backbuffer_width, s_backbuffer_height};
+ AdjustWindowRectEx(&rc, dwStyle, FALSE, dwExStyle);
+
+ int X = (rcdesktop.right-rcdesktop.left)/2 - (rc.right-rc.left)/2;
+ int Y = (rcdesktop.bottom-rcdesktop.top)/2 - (rc.bottom-rc.top)/2;
+
+ // EmuWindow::GetWnd() is either the new child window or the new separate window
+ if (g_Config.bFullscreen)
+ // We put the window at the upper left corner of the screen, so x = y = 0
+ SetWindowPos(EmuWindow::GetWnd(), NULL, 0, 0, rc.right-rc.left, rc.bottom-rc.top, SWP_NOREPOSITION | SWP_NOZORDER);
+ else
+ SetWindowPos(EmuWindow::GetWnd(), NULL, X, Y, rc.right-rc.left, rc.bottom-rc.top, SWP_NOREPOSITION | SWP_NOZORDER);
+
+ PIXELFORMATDESCRIPTOR pfd = // pfd Tells Windows How We Want Things To Be
+ {
+ sizeof(PIXELFORMATDESCRIPTOR), // Size Of This Pixel Format Descriptor
+ 1, // Version Number
+ PFD_DRAW_TO_WINDOW | // Format Must Support Window
+ PFD_SUPPORT_OPENGL | // Format Must Support OpenGL
+ PFD_DOUBLEBUFFER, // Must Support Double Buffering
+ PFD_TYPE_RGBA, // Request An RGBA Format
+ 32, // Select Our Color Depth
+ 0, 0, 0, 0, 0, 0, // Color Bits Ignored
+ 0, // 8bit Alpha Buffer
+ 0, // Shift Bit Ignored
+ 0, // No Accumulation Buffer
+ 0, 0, 0, 0, // Accumulation Bits Ignored
+ 24, // 24Bit Z-Buffer (Depth Buffer)
+ 8, // 8bit Stencil Buffer
+ 0, // No Auxiliary Buffer
+ PFD_MAIN_PLANE, // Main Drawing Layer
+ 0, // Reserved
+ 0, 0, 0 // Layer Masks Ignored
+ };
+
+ if (!(hDC=GetDC(EmuWindow::GetWnd()))) {
+ PanicAlert("(1) Can't create an OpenGL Device context. Fail.");
+ return false;
+ }
+
+ if (!(PixelFormat = ChoosePixelFormat(hDC,&pfd))) {
+ PanicAlert("(2) Can't find a suitable PixelFormat.");
+ return false;
+ }
+
+ if (!SetPixelFormat(hDC, PixelFormat, &pfd)) {
+ PanicAlert("(3) Can't set the PixelFormat.");
+ return false;
+ }
+
+ if (!(hRC = wglCreateContext(hDC))) {
+ PanicAlert("(4) Can't create an OpenGL rendering context.");
+ return false;
+ }
+ // --------------------------------------
+
+#elif defined(HAVE_X11) && HAVE_X11
+ XVisualInfo *vi;
+ Colormap cmap;
+ int dpyWidth, dpyHeight;
+ int glxMajorVersion, glxMinorVersion;
+ int vidModeMajorVersion, vidModeMinorVersion;
+ Atom wmDelete;
+
+ // attributes for a single buffered visual in RGBA format with at least
+ // 8 bits per color and a 24 bit depth buffer
+ int attrListSgl[] = {GLX_RGBA, GLX_RED_SIZE, 8,
+ GLX_GREEN_SIZE, 8,
+ GLX_BLUE_SIZE, 8,
+ GLX_DEPTH_SIZE, 24,
+ None};
+
+ // attributes for a double buffered visual in RGBA format with at least
+ // 8 bits per color and a 24 bit depth buffer
+ int attrListDbl[] = { GLX_RGBA, GLX_DOUBLEBUFFER,
+ GLX_RED_SIZE, 8,
+ GLX_GREEN_SIZE, 8,
+ GLX_BLUE_SIZE, 8,
+ GLX_DEPTH_SIZE, 24,
+ GLX_SAMPLE_BUFFERS_ARB, g_Config.iMultisampleMode, GLX_SAMPLES_ARB, 1, None };
+ GLWin.dpy = XOpenDisplay(0);
+ g_VideoInitialize.pWindowHandle = (HWND)GLWin.dpy;
+ GLWin.screen = DefaultScreen(GLWin.dpy);
+
+ // Fullscreen option.
+ GLWin.fs = g_Config.bFullscreen; //Set to setting in Options
+
+ /* get an appropriate visual */
+ vi = glXChooseVisual(GLWin.dpy, GLWin.screen, attrListDbl);
+ if (vi == NULL) {
+ vi = glXChooseVisual(GLWin.dpy, GLWin.screen, attrListSgl);
+ GLWin.doubleBuffered = False;
+ ERROR_LOG(VIDEO, "Only Singlebuffered Visual!");
+ }
+ else {
+ GLWin.doubleBuffered = True;
+ NOTICE_LOG(VIDEO, "Got Doublebuffered Visual!");
+ }
+
+ glXQueryVersion(GLWin.dpy, &glxMajorVersion, &glxMinorVersion);
+ NOTICE_LOG(VIDEO, "glX-Version %d.%d", glxMajorVersion, glxMinorVersion);
+ // Create a GLX context.
+ GLWin.ctx = glXCreateContext(GLWin.dpy, vi, 0, GL_TRUE);
+ if(!GLWin.ctx)
+ {
+ PanicAlert("Couldn't Create GLX context.Quit");
+ exit(0); // TODO: Don't bring down entire Emu
+ }
+ // Create a color map.
+ cmap = XCreateColormap(GLWin.dpy, RootWindow(GLWin.dpy, vi->screen), vi->visual, AllocNone);
+ GLWin.attr.colormap = cmap;
+ GLWin.attr.border_pixel = 0;
+ XkbSetDetectableAutoRepeat(GLWin.dpy, True, NULL);
+
+#if defined(HAVE_XXF86VM) && HAVE_XXF86VM
+ // get a connection
+ XF86VidModeQueryVersion(GLWin.dpy, &vidModeMajorVersion, &vidModeMinorVersion);
+
+ if (GLWin.fs) {
+
+ XF86VidModeModeInfo **modes = NULL;
+ int modeNum = 0;
+ int bestMode = 0;
+
+ // set best mode to current
+ bestMode = 0;
+ NOTICE_LOG(VIDEO, "XF86VidModeExtension-Version %d.%d", vidModeMajorVersion, vidModeMinorVersion);
+ XF86VidModeGetAllModeLines(GLWin.dpy, GLWin.screen, &modeNum, &modes);
+
+ if (modeNum > 0 && modes != NULL) {
+ /* save desktop-resolution before switching modes */
+ GLWin.deskMode = *modes[0];
+ /* look for mode with requested resolution */
+ for (int i = 0; i < modeNum; i++) {
+ if ((modes[i]->hdisplay == _twidth) && (modes[i]->vdisplay == _theight)) {
+ bestMode = i;
+ }
+ }
+
+ XF86VidModeSwitchToMode(GLWin.dpy, GLWin.screen, modes[bestMode]);
+ XF86VidModeSetViewPort(GLWin.dpy, GLWin.screen, 0, 0);
+ dpyWidth = modes[bestMode]->hdisplay;
+ dpyHeight = modes[bestMode]->vdisplay;
+ NOTICE_LOG(VIDEO, "Resolution %dx%d", dpyWidth, dpyHeight);
+ XFree(modes);
+
+ /* create a fullscreen window */
+ GLWin.attr.override_redirect = True;
+ GLWin.attr.event_mask = ExposureMask | KeyPressMask | ButtonPressMask | KeyReleaseMask | ButtonReleaseMask | StructureNotifyMask;
+ GLWin.win = XCreateWindow(GLWin.dpy, RootWindow(GLWin.dpy, vi->screen),
+ 0, 0, dpyWidth, dpyHeight, 0, vi->depth, InputOutput, vi->visual,
+ CWBorderPixel | CWColormap | CWEventMask | CWOverrideRedirect,
+ &GLWin.attr);
+ XWarpPointer(GLWin.dpy, None, GLWin.win, 0, 0, 0, 0, 0, 0);
+ XMapRaised(GLWin.dpy, GLWin.win);
+ XGrabKeyboard(GLWin.dpy, GLWin.win, True, GrabModeAsync, GrabModeAsync, CurrentTime);
+ XGrabPointer(GLWin.dpy, GLWin.win, True, ButtonPressMask,
+ GrabModeAsync, GrabModeAsync, GLWin.win, None, CurrentTime);
+ }
+ else {
+ ERROR_LOG(VIDEO, "Failed to start fullscreen. If you received the "
+ "\"XFree86-VidModeExtension\" extension is missing, add\n"
+ "Load \"extmod\"\n"
+ "to your X configuration file (under the Module Section)\n");
+ GLWin.fs = 0;
+ }
+ }
+#endif
+
+ if (!GLWin.fs) {
+
+ //XRootWindow(dpy,screen)
+ //int X = (rcdesktop.right-rcdesktop.left)/2 - (rc.right-rc.left)/2;
+ //int Y = (rcdesktop.bottom-rcdesktop.top)/2 - (rc.bottom-rc.top)/2;
+
+ // create a window in window mode
+ GLWin.attr.event_mask = ExposureMask | KeyPressMask | ButtonPressMask | KeyReleaseMask | ButtonReleaseMask |
+ StructureNotifyMask | ResizeRedirectMask;
+ GLWin.win = XCreateWindow(GLWin.dpy, RootWindow(GLWin.dpy, vi->screen),
+ 0, 0, _twidth, _theight, 0, vi->depth, InputOutput, vi->visual,
+ CWBorderPixel | CWColormap | CWEventMask, &GLWin.attr);
+ // only set window title and handle wm_delete_events if in windowed mode
+ wmDelete = XInternAtom(GLWin.dpy, "WM_DELETE_WINDOW", True);
+ XSetWMProtocols(GLWin.dpy, GLWin.win, &wmDelete, 1);
+ XSetStandardProperties(GLWin.dpy, GLWin.win, "GPU",
+ "GPU", None, NULL, 0, NULL);
+ XMapRaised(GLWin.dpy, GLWin.win);
+ }
+#endif
+ return true;
+}
+
+bool OpenGL_MakeCurrent()
+{
+#if USE_SDL
+ // Note: The reason for having the call to SDL_SetVideoMode in here instead
+ // of in OpenGL_Create() is that "make current" is part of the video
+ // mode setting and is not available as a separate call in SDL. We
+ // have to do "make current" here because this method runs in the CPU
+ // thread while OpenGL_Create() runs in a diferent thread and "make
+ // current" has to be done in the same thread that will be making
+ // calls to OpenGL.
+
+ // Fetch video info.
+ const SDL_VideoInfo *videoInfo = SDL_GetVideoInfo();
+ if (!videoInfo) {
+ // TODO: Display an error message.
+ SDL_Quit();
+ return false;
+ }
+ // Compute video mode flags.
+ const int videoFlags = SDL_OPENGL
+ | ( videoInfo->hw_available ? SDL_HWSURFACE : SDL_SWSURFACE )
+ | ( g_Config.bFullscreen ? SDL_FULLSCREEN : 0);
+ // Set vide mode.
+ // TODO: Can we use this field or is a separate field needed?
+ int _twidth = s_backbuffer_width;
+ int _theight = s_backbuffer_height;
+ SDL_Surface *screen = SDL_SetVideoMode(_twidth, _theight, 0, videoFlags);
+ if (!screen) {
+ //TODO : Display an error message
+ SDL_Quit();
+ return false;
+ }
+#elif defined(HAVE_COCOA) && HAVE_COCOA
+ cocoaGLMakeCurrent(GLWin.cocoaCtx,GLWin.cocoaWin);
+#elif defined(_WIN32)
+ if (!wglMakeCurrent(hDC,hRC)) {
+ PanicAlert("(5) Can't Activate The GL Rendering Context.");
+ return false;
+ }
+#elif defined(USE_WX) && USE_WX
+#if defined(__APPLE__)
+ GLWin.glCanvas->SetCurrent();
+#else
+ GLWin.glCanvas->SetCurrent(*GLWin.glCtxt);
+#endif
+ return true;
+#elif defined(HAVE_X11) && HAVE_X11
+ Window winDummy;
+ unsigned int borderDummy;
+ // connect the glx-context to the window
+ glXMakeCurrent(GLWin.dpy, GLWin.win, GLWin.ctx);
+ XGetGeometry(GLWin.dpy, GLWin.win, &winDummy, &GLWin.x, &GLWin.y,
+ &GLWin.width, &GLWin.height, &borderDummy, &GLWin.depth);
+ NOTICE_LOG(VIDEO, "GLWin Depth %d", GLWin.depth)
+ if (glXIsDirect(GLWin.dpy, GLWin.ctx)) {
+ NOTICE_LOG(VIDEO, "detected direct rendering");
+ } else {
+ ERROR_LOG(VIDEO, "no Direct Rendering possible!");
+ }
+
+ // better for pad plugin key input (thc)
+ XSelectInput(GLWin.dpy, GLWin.win, ExposureMask | KeyPressMask | ButtonPressMask | KeyReleaseMask | ButtonReleaseMask | StructureNotifyMask | EnterWindowMask | LeaveWindowMask |
+ FocusChangeMask );
+#endif
+ return true;
+}
+
+
+// =======================================================================================
+// Update window width, size and etc. Called from Render.cpp
+// ----------------
+void OpenGL_Update()
+{
+#if USE_SDL
+ SDL_Surface *surface = SDL_GetVideoSurface();
+ RECT rcWindow = {0};
+ if (!surface)
+ return;
+ s_backbuffer_width = surface->w;
+ s_backbuffer_height = surface->h;
+
+ rcWindow.right = surface->w;
+ rcWindow.bottom = surface->h;
+
+#elif defined(HAVE_COCOA) && HAVE_COCOA
+ RECT rcWindow = {0};
+ rcWindow.right = GLWin.width;
+ rcWindow.bottom = GLWin.height;
+
+#elif defined(USE_WX) && USE_WX
+ RECT rcWindow = {0};
+ rcWindow.right = GLWin.width;
+ rcWindow.bottom = GLWin.height;
+
+ // TODO fill in
+#elif defined(_WIN32)
+ RECT rcWindow;
+ if (!EmuWindow::GetParentWnd())
+ {
+ // We are not rendering to a child window - use client size.
+ GetClientRect(EmuWindow::GetWnd(), &rcWindow);
+ }
+ else
+ {
+ // We are rendering to a child window - use parent size.
+ GetWindowRect(EmuWindow::GetParentWnd(), &rcWindow);
+ }
+
+ // ---------------------------------------------------------------------------------------
+ // Get the new window width and height
+ // ------------------
+ // See below for documentation
+ // ------------------
+ int width = rcWindow.right - rcWindow.left;
+ int height = rcWindow.bottom - rcWindow.top;
+
+ // If we are rendering to a child window
+ if (EmuWindow::GetParentWnd() != 0)
+ ::MoveWindow(EmuWindow::GetWnd(), 0, 0, width, height, FALSE);
+
+ s_backbuffer_width = width;
+ s_backbuffer_height = height;
+
+#elif defined(HAVE_X11) && HAVE_X11
+ // We just check all of our events here
+ XEvent event;
+ KeySym key;
+ static RECT rcWindow;
+ static bool ShiftPressed = false;
+ static bool ControlPressed = false;
+ static int FKeyPressed = -1;
+ int num_events;
+ for (num_events = XPending(GLWin.dpy);num_events > 0;num_events--) {
+ XNextEvent(GLWin.dpy, &event);
+ switch(event.type) {
+ case KeyRelease:
+ key = XLookupKeysym((XKeyEvent*)&event, 0);
+ if(key >= XK_F1 && key <= XK_F9) {
+ g_VideoInitialize.pKeyPress(FKeyPressed, ShiftPressed, ControlPressed);
+ FKeyPressed = -1;
+ } else {
+ if(key == XK_Shift_L || key == XK_Shift_R)
+ ShiftPressed = false;
+ else if(key == XK_Control_L || key == XK_Control_R)
+ ControlPressed = false;
+ else
+ XPutBackEvent(GLWin.dpy, &event);
+ }
+ break;
+ case KeyPress:
+ key = XLookupKeysym((XKeyEvent*)&event, 0);
+ if(key >= XK_F1 && key <= XK_F9)
+ FKeyPressed = key - 0xff4e;
+ else {
+ if(key == XK_Shift_L || key == XK_Shift_R)
+ ShiftPressed = true;
+ else if(key == XK_Control_L || key == XK_Control_R)
+ ControlPressed = true;
+ else
+ XPutBackEvent(GLWin.dpy, &event);
+ }
+ break;
+ case ButtonPress:
+ case ButtonRelease:
+ XPutBackEvent(GLWin.dpy, &event);
+ break;
+ case ConfigureNotify:
+ Window winDummy;
+ unsigned int borderDummy;
+ XGetGeometry(GLWin.dpy, GLWin.win, &winDummy, &GLWin.x, &GLWin.y,
+ &GLWin.width, &GLWin.height, &borderDummy, &GLWin.depth);
+ s_backbuffer_width = GLWin.width;
+ s_backbuffer_height = GLWin.height;
+ rcWindow.left = 0;
+ rcWindow.top = 0;
+ rcWindow.right = GLWin.width;
+ rcWindow.bottom = GLWin.height;
+ break;
+ case ClientMessage: //TODO: We aren't reading this correctly, It could be anything, highest chance is that it's a close event though
+ Shutdown(); // Calling from here since returning false does nothing
+ return;
+ break;
+ default:
+ //TODO: Should we put the event back if we don't handle it?
+ // I think we handle all the needed ones, the rest shouldn't matter
+ // But to be safe, let's but them back anyway
+ //XPutBackEvent(GLWin.dpy, &event);
+ break;
+ }
+ }
+ return;
+#endif
+}
+
+
+// =======================================================================================
+// Close plugin
+// ----------------
+void OpenGL_Shutdown()
+{
+#if USE_SDL
+ SDL_Quit();
+#elif defined(HAVE_COCOA) && HAVE_COCOA
+ cocoaGLDelete(GLWin.cocoaCtx);
+#elif defined(USE_WX) && USE_WX
+ delete GLWin.glCanvas;
+ delete GLWin.frame;
+#elif defined(_WIN32)
+ if (hRC) // Do We Have A Rendering Context?
+ {
+ if (!wglMakeCurrent(NULL,NULL)) // Are We Able To Release The DC And RC Contexts?
+ {
+ // [F|RES]: if this fails i dont see the message box and
+ // cant get out of the modal state so i disable it.
+ // This function fails only if i render to main window
+ // MessageBox(NULL,"Release Of DC And RC Failed.", "SHUTDOWN ERROR", MB_OK | MB_ICONINFORMATION);
+ }
+
+ if (!wglDeleteContext(hRC)) // Are We Able To Delete The RC?
+ {
+ ERROR_LOG(VIDEO, "Release Rendering Context Failed.");
+ }
+ hRC = NULL; // Set RC To NULL
+ }
+
+ if (hDC && !ReleaseDC(EmuWindow::GetWnd(), hDC)) // Are We Able To Release The DC
+ {
+#ifndef SETUP_TIMER_WAITING // This fails
+ ERROR_LOG(VIDEO, "Release Device Context Failed.");
+#endif
+ hDC = NULL; // Set DC To NULL
+ }
+#elif defined(HAVE_X11) && HAVE_X11
+ if (GLWin.ctx)
+ {
+ if (!glXMakeCurrent(GLWin.dpy, None, NULL))
+ {
+ ERROR_LOG(VIDEO, "Could not release drawing context.\n");
+ }
+ XUnmapWindow(GLWin.dpy, GLWin.win);
+ glXDestroyContext(GLWin.dpy, GLWin.ctx);
+ XCloseDisplay(GLWin.dpy);
+ GLWin.ctx = NULL;
+ }
+#if defined(HAVE_XXF86VM) && HAVE_XXF86VM
+ /* switch back to original desktop resolution if we were in fs */
+ if (GLWin.dpy != NULL) {
+ if (GLWin.fs) {
+ XF86VidModeSwitchToMode(GLWin.dpy, GLWin.screen, &GLWin.deskMode);
+ XF86VidModeSetViewPort(GLWin.dpy, GLWin.screen, 0, 0);
+ }
+ }
+#endif
+#endif
+}
+
+GLuint OpenGL_ReportGLError(const char *function, const char *file, int line)
+{
+ GLint err = glGetError();
+ if (err != GL_NO_ERROR)
+ {
+ ERROR_LOG(VIDEO, "%s:%d: (%s) OpenGL error 0x%x - %s\n", file, line, function, err, gluErrorString(err));
+ }
+ return err;
+}
+
+void OpenGL_ReportARBProgramError()
+{
+ const GLubyte* pstr = glGetString(GL_PROGRAM_ERROR_STRING_ARB);
+ if (pstr != NULL && pstr[0] != 0)
+ {
+ GLint loc = 0;
+ glGetIntegerv(GL_PROGRAM_ERROR_POSITION_ARB, &loc);
+ ERROR_LOG(VIDEO, "program error at %d: ", loc);
+ ERROR_LOG(VIDEO, (char*)pstr);
+ ERROR_LOG(VIDEO, "");
+ }
+}
+
+bool OpenGL_ReportFBOError(const char *function, const char *file, int line)
+{
+ unsigned int fbo_status = glCheckFramebufferStatusEXT(GL_FRAMEBUFFER_EXT);
+ if (fbo_status != GL_FRAMEBUFFER_COMPLETE_EXT)
+ {
+ const char *error = "-";
+ switch (fbo_status)
+ {
+ case GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT_EXT: error = "INCOMPLETE_ATTACHMENT_EXT"; break;
+ case GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT_EXT: error = "INCOMPLETE_MISSING_ATTACHMENT_EXT"; break;
+ case GL_FRAMEBUFFER_INCOMPLETE_DIMENSIONS_EXT: error = "INCOMPLETE_DIMENSIONS_EXT"; break;
+ case GL_FRAMEBUFFER_INCOMPLETE_FORMATS_EXT: error = "INCOMPLETE_FORMATS_EXT"; break;
+ case GL_FRAMEBUFFER_INCOMPLETE_DRAW_BUFFER_EXT: error = "INCOMPLETE_DRAW_BUFFER_EXT"; break;
+ case GL_FRAMEBUFFER_INCOMPLETE_READ_BUFFER_EXT: error = "INCOMPLETE_READ_BUFFER_EXT"; break;
+ case GL_FRAMEBUFFER_UNSUPPORTED_EXT: error = "UNSUPPORTED_EXT"; break;
+ }
+ ERROR_LOG(VIDEO, "%s:%d: (%s) OpenGL FBO error - %s\n", file, line, function, error);
+ return false;
+ }
+ return true;
+}
diff --git a/Source/Plugins/Plugin_VideoSoftware/Src/GLUtil.h b/Source/Plugins/Plugin_VideoSoftware/Src/GLUtil.h
new file mode 100644
index 0000000000..fbc713ca39
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Src/GLUtil.h
@@ -0,0 +1,155 @@
+// 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/
+
+
+#ifndef _GLINIT_H_
+#define _GLINIT_H_
+
+#if defined GLTEST && GLTEST
+#include "nGLUtil.h"
+#else
+#include "Common.h"
+#include
+#include "VideoConfig.h"
+#include "pluginspecs_video.h"
+
+#ifdef _WIN32
+
+#define GLEW_STATIC
+
+#include
+#include
+#include
+#include
+
+#else // linux basic definitions
+
+#if defined(USE_WX) && USE_WX
+#include
+#include "wx/wx.h"
+#include "wx/glcanvas.h"
+#undef HAVE_X11
+#elif defined(HAVE_X11) && HAVE_X11
+#define I_NEED_OS2_H // HAXXOR
+#include
+#include
+#elif defined(USE_SDL) && USE_SDL
+#include
+#include
+#elif defined(HAVE_COCOA) && HAVE_COCOA
+#include
+#include "cocoaGL.h"
+#endif // end USE_WX
+
+#if defined(__APPLE__)
+#include
+#else
+#include
+#endif
+
+#endif // linux basic definitions
+
+#ifndef GL_DEPTH24_STENCIL8_EXT // allows FBOs to support stencils
+#define GL_DEPTH_STENCIL_EXT 0x84F9
+#define GL_UNSIGNED_INT_24_8_EXT 0x84FA
+#define GL_DEPTH24_STENCIL8_EXT 0x88F0
+#define GL_TEXTURE_STENCIL_SIZE_EXT 0x88F1
+#endif
+
+#ifndef _WIN32
+#if defined(HAVE_X11) && HAVE_X11
+#include
+#include
+#include
+#if defined(HAVE_XXF86VM) && HAVE_XXF86VM
+#include
+#endif // XXF86VM
+#endif // X11
+
+#include
+#include
+
+typedef struct {
+ int screen;
+#if defined(HAVE_COCOA) && HAVE_COCOA
+ NSWindow *cocoaWin;
+ NSOpenGLContext *cocoaCtx;
+#elif defined(HAVE_X11) && HAVE_X11
+ Window win;
+ Display *dpy;
+ GLXContext ctx;
+ XSetWindowAttributes attr;
+ Bool fs;
+ Bool doubleBuffered;
+#if defined(HAVE_XXF86VM) && HAVE_XXF86VM
+ XF86VidModeModeInfo deskMode;
+#endif // XXF86VM
+#endif // X11
+#if defined(USE_WX) && USE_WX
+ wxGLCanvas *glCanvas;
+ wxFrame *frame;
+ wxGLContext *glCtxt;
+#endif
+ int x, y;
+ unsigned int width, height;
+ unsigned int depth;
+} GLWindow;
+
+extern GLWindow GLWin;
+
+#endif
+
+// Public OpenGL util
+
+// Initialization / upkeep
+bool OpenGL_Create(SVideoInitialize &_VideoInitialize, int _width, int _height);
+void OpenGL_Shutdown();
+void OpenGL_Update();
+bool OpenGL_MakeCurrent();
+void OpenGL_SwapBuffers();
+
+// Get status
+u32 OpenGL_GetBackbufferWidth();
+u32 OpenGL_GetBackbufferHeight();
+
+// Set things
+void OpenGL_SetWindowText(const char *text);
+
+// Error reporting - use the convenient macros.
+void OpenGL_ReportARBProgramError();
+GLuint OpenGL_ReportGLError(const char *function, const char *file, int line);
+bool OpenGL_ReportFBOError(const char *function, const char *file, int line);
+
+#if 1
+#define GL_REPORT_ERROR() OpenGL_ReportGLError (__FUNCTION__, __FILE__, __LINE__)
+#define GL_REPORT_PROGRAM_ERROR() OpenGL_ReportARBProgramError()
+#define GL_REPORT_FBO_ERROR() OpenGL_ReportFBOError (__FUNCTION__, __FILE__, __LINE__)
+#else
+#define GL_REPORT_ERROR() GL_NO_ERROR
+#define GL_REPORT_PROGRAM_ERROR()
+#define GL_REPORT_FBO_ERROR()
+#endif
+
+#if defined(_DEBUG) || defined(DEBUGFAST)
+#define GL_REPORT_ERRORD() OpenGL_ReportGLError(__FUNCTION__, __FILE__, __LINE__)
+#else
+#define GL_REPORT_ERRORD() GL_NO_ERROR
+#endif
+
+#endif // GLTEST ??
+
+#endif // _GLINIT_H_
diff --git a/Source/Plugins/Plugin_VideoSoftware/Src/HwRasterizer.cpp b/Source/Plugins/Plugin_VideoSoftware/Src/HwRasterizer.cpp
new file mode 100644
index 0000000000..4adb6fcd8b
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Src/HwRasterizer.cpp
@@ -0,0 +1,191 @@
+// 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 "MemoryUtil.h"
+
+#include
+
+#include "BpMemLoader.h"
+#include "HwRasterizer.h"
+#include "GLUtil.h"
+#include "NativeVertexFormat.h"
+#include "DebugUtil.h"
+
+#define TEMP_SIZE (1024*1024*4)
+
+namespace HwRasterizer
+{
+ float efbHalfWidth;
+ float efbHalfHeight;
+ float texWidth;
+ float texHeight;
+ bool hasTexture;
+
+ u8 *temp;
+
+ void Init()
+ {
+ efbHalfWidth = EFB_WIDTH / 2.0f;
+ efbHalfHeight = 480 / 2.0f;
+
+ temp = (u8*)AllocateMemoryPages(TEMP_SIZE);
+ }
+
+ void LoadTexture()
+ {
+ FourTexUnits &texUnit = bpmem.tex[0];
+ u32 imageAddr = texUnit.texImage3[0].image_base;
+
+ TexCacheEntry &cacheEntry = textures[imageAddr];
+ cacheEntry.Update();
+
+ texWidth = (float)(bpmem.texcoords[0].s.scale_minus_1 + 1);
+ texHeight = (float)(bpmem.texcoords[0].t.scale_minus_1 + 1);
+
+ glBindTexture(GL_TEXTURE_RECTANGLE_ARB, cacheEntry.texture);
+ glTexParameteri(GL_TEXTURE_RECTANGLE_ARB, GL_TEXTURE_MAG_FILTER, texUnit.texMode0[0].mag_filter ? GL_LINEAR : GL_NEAREST);
+ glTexParameteri(GL_TEXTURE_RECTANGLE_ARB, GL_TEXTURE_MIN_FILTER, (texUnit.texMode0[0].min_filter >= 4) ? GL_LINEAR : GL_NEAREST);
+ }
+
+ void BeginTriangles()
+ {
+ // disabling depth test sometimes allows more things to be visible
+ glEnable(GL_DEPTH_TEST);
+ glEnable(GL_BLEND);
+
+ hasTexture = bpmem.tevorders[0].enable0;
+
+ if (hasTexture)
+ LoadTexture();
+ }
+
+ void EndTriangles()
+ {
+ glBindTexture(GL_TEXTURE_RECTANGLE_ARB, 0);
+
+ glDisable(GL_DEPTH_TEST);
+ glDisable(GL_BLEND);
+ }
+
+ void DrawColorVertex(OutputVertexData *v)
+ {
+ glColor3ub(v->color[0][0], v->color[0][1], v->color[0][2]);
+ glVertex3f(v->screenPosition[0] / efbHalfWidth - 1.0f, 1.0f - v->screenPosition[1] / efbHalfHeight, v->screenPosition[2]);
+ }
+
+ void DrawTextureVertex(OutputVertexData *v)
+ {
+ glTexCoord2f(v->texCoords[0][0] * texWidth, v->texCoords[0][1] * texHeight);
+ glVertex3f(v->screenPosition[0] / efbHalfWidth - 1.0f, 1.0f - v->screenPosition[1] / efbHalfHeight, v->screenPosition[2]);
+ }
+
+ void DrawTriangleFrontFace(OutputVertexData *v0, OutputVertexData *v1, OutputVertexData *v2)
+ {
+ glBegin(GL_TRIANGLES);
+ if (hasTexture)
+ {
+ DrawTextureVertex(v0);
+ DrawTextureVertex(v1);
+ DrawTextureVertex(v2);
+ }
+ else
+ {
+ DrawColorVertex(v0);
+ DrawColorVertex(v1);
+ DrawColorVertex(v2);
+ }
+ glEnd();
+ }
+
+ void Clear()
+ {
+ u8 r = (bpmem.clearcolorAR & 0x00ff);
+ u8 g = (bpmem.clearcolorGB & 0xff00) >> 8;
+ u8 b = (bpmem.clearcolorGB & 0x00ff);
+ u8 a = (bpmem.clearcolorAR & 0xff00) >> 8;
+
+ GLfloat left = (GLfloat)bpmem.copyTexSrcXY.x / efbHalfWidth - 1.0f;
+ GLfloat top = 1.0f - (GLfloat)bpmem.copyTexSrcXY.y / efbHalfHeight;
+ GLfloat right = (GLfloat)(left + bpmem.copyTexSrcWH.x + 1) / efbHalfWidth - 1.0f;
+ GLfloat bottom = 1.0f - (GLfloat)(top + bpmem.copyTexSrcWH.y + 1) / efbHalfHeight;
+ GLfloat depth = (GLfloat)bpmem.clearZValue / (GLfloat)0x00ffffff;
+
+ glBegin(GL_QUADS);
+ glColor4ub(r, g, b, a);
+ glVertex3f(left, top, depth);
+ glColor4ub(r, g, b, a);
+ glVertex3f(right, top, depth);
+ glColor4ub(r, g, b, a);
+ glVertex3f(right, bottom, depth);
+ glColor4ub(r, g, b, a);
+ glVertex3f(left, bottom, depth);
+ glEnd();
+ }
+
+ TexCacheEntry::TexCacheEntry()
+ {
+ Create();
+ }
+
+ void TexCacheEntry::Create()
+ {
+ FourTexUnits &texUnit = bpmem.tex[0];
+
+ texImage0.hex = texUnit.texImage0[0].hex;
+ texImage1.hex = texUnit.texImage1[0].hex;
+ texImage2.hex = texUnit.texImage2[0].hex;
+ texImage3.hex = texUnit.texImage3[0].hex;
+ texTlut.hex = texUnit.texTlut[0].hex;
+
+ int width = texImage0.width;
+ int height = texImage0.height;
+
+ DebugUtil::GetTextureBGRA(temp, 0, width, height);
+
+ glGenTextures(1, (GLuint *)&texture);
+ glBindTexture(GL_TEXTURE_RECTANGLE_ARB, texture);
+ glTexImage2D(GL_TEXTURE_RECTANGLE_ARB, 0, GL_RGBA8, (GLsizei)width, (GLsizei)height, 0, GL_BGRA, GL_UNSIGNED_BYTE, temp);
+ }
+
+ void TexCacheEntry::Destroy()
+ {
+ if (texture == 0)
+ return;
+
+ glDeleteTextures(1, &texture);
+ texture = 0;
+ }
+
+ void TexCacheEntry::Update()
+ {
+ FourTexUnits &texUnit = bpmem.tex[0];
+
+ // extra checks cause textures to be reloaded much more
+ if (texUnit.texImage0[0].hex != texImage0.hex ||
+ //texUnit.texImage1[0].hex != texImage1.hex ||
+ //texUnit.texImage2[0].hex != texImage2.hex ||
+ texUnit.texImage3[0].hex != texImage3.hex ||
+ texUnit.texTlut[0].hex != texTlut.hex)
+ {
+ Destroy();
+ Create();
+ }
+ }
+
+}
+
diff --git a/Source/Plugins/Plugin_VideoSoftware/Src/HwRasterizer.h b/Source/Plugins/Plugin_VideoSoftware/Src/HwRasterizer.h
new file mode 100644
index 0000000000..fd37460d7f
--- /dev/null
+++ b/Source/Plugins/Plugin_VideoSoftware/Src/HwRasterizer.h
@@ -0,0 +1,60 @@
+// 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/
+
+#ifndef _HW_RASTERIZER_H
+#define _HW_RASTERIZER_H
+
+#include