OpenGL Renderer (v0.9.11):

- Backport changes from r5154 to the 0.9.11 branch.
This commit is contained in:
rogerman 2015-04-07 22:34:29 +00:00
parent 4fe116b423
commit 0b4fd3cb5e
6 changed files with 421 additions and 736 deletions

File diff suppressed because it is too large Load Diff

View File

@ -310,16 +310,15 @@ struct OGLRenderRef
// FBO // FBO
GLuint texClearImageColorID; GLuint texClearImageColorID;
GLuint texClearImageDepthStencilID; GLuint texClearImageDepthStencilID;
GLuint rboFragColorID;
GLuint rboFragDepthStencilID;
GLuint rboMSFragColorID;
GLuint rboMSFragDepthStencilID;
GLuint fboClearImageID; GLuint fboClearImageID;
GLuint fboMSIntermediateRenderID;
GLuint rboFinalOutputColorID; GLuint fboRenderID;
GLuint rboFinalOutputDepthStencilID;
GLuint fboFinalOutputID;
// Multisampled FBO
GLuint rboMultisampleColorID;
GLuint rboMultisampleDepthStencilID;
GLuint fboMultisampleRenderID;
GLuint selectedRenderingFBO; GLuint selectedRenderingFBO;
// Shader states // Shader states
@ -327,15 +326,18 @@ struct OGLRenderRef
GLuint fragmentShaderID; GLuint fragmentShaderID;
GLuint shaderProgram; GLuint shaderProgram;
GLint uniformPolyID;
GLint uniformPolyAlpha;
GLint uniformTexScale; GLint uniformTexScale;
GLint uniformHasTexture;
GLint uniformPolygonMode; GLint uniformStateToonShadingMode;
GLint uniformToonShadingMode; GLint uniformStateEnableAlphaTest;
GLint uniformWBuffer; GLint uniformStateUseWDepth;
GLint uniformEnableAlphaTest; GLint uniformStateAlphaTestRef;
GLint uniformAlphaTestRef;
GLint uniformPolyMode;
GLint uniformPolyAlpha;
GLint uniformPolyID;
GLint uniformPolyEnableTexture;
GLuint texToonTableID; GLuint texToonTableID;
@ -347,7 +349,7 @@ struct OGLRenderRef
// Client-side Buffers // Client-side Buffers
GLfloat *color4fBuffer; GLfloat *color4fBuffer;
DS_ALIGN(16) GLushort vertIndexBuffer[OGLRENDER_VERT_INDEX_BUFFER_COUNT]; CACHE_ALIGN GLushort vertIndexBuffer[OGLRENDER_VERT_INDEX_BUFFER_COUNT];
}; };
struct GFX3D_State; struct GFX3D_State;
@ -410,12 +412,9 @@ protected:
// Textures // Textures
TexCacheItem *currTexture; TexCacheItem *currTexture;
u16 currentToonTable16[32]; CACHE_ALIGN u32 GPU_screen3D[2][GFX3D_FRAMEBUFFER_WIDTH * GFX3D_FRAMEBUFFER_HEIGHT * sizeof(u32)];
bool toonTableNeedsUpdate;
DS_ALIGN(16) u32 GPU_screen3D[2][GFX3D_FRAMEBUFFER_WIDTH * GFX3D_FRAMEBUFFER_HEIGHT * sizeof(u32)];
bool gpuScreen3DHasNewData[2]; bool gpuScreen3DHasNewData[2];
unsigned int doubleBufferIndex; size_t doubleBufferIndex;
u8 clearImageStencilValue; u8 clearImageStencilValue;
// OpenGL-specific methods // OpenGL-specific methods
@ -439,15 +438,13 @@ protected:
virtual Render3DError SetupShaderIO() = 0; virtual Render3DError SetupShaderIO() = 0;
virtual Render3DError CreateToonTable() = 0; virtual Render3DError CreateToonTable() = 0;
virtual Render3DError DestroyToonTable() = 0; virtual Render3DError DestroyToonTable() = 0;
virtual Render3DError UploadToonTable(const GLushort *toonTableBuffer) = 0; virtual Render3DError UploadToonTable(const u16 *toonTableBuffer) = 0;
virtual Render3DError CreateClearImage() = 0; virtual Render3DError UploadClearImage(const u16 *clearImageColor16Buffer, const u32 *clearImageDepthStencilBuffer) = 0;
virtual Render3DError DestroyClearImage() = 0;
virtual Render3DError UploadClearImage(const GLushort *clearImageColorBuffer, const GLint *clearImageDepthBuffer) = 0;
virtual void GetExtensionSet(std::set<std::string> *oglExtensionSet) = 0; virtual void GetExtensionSet(std::set<std::string> *oglExtensionSet) = 0;
virtual Render3DError ExpandFreeTextures() = 0; virtual Render3DError ExpandFreeTextures() = 0;
virtual Render3DError SetupVertices(const VERTLIST *vertList, const POLYLIST *polyList, const INDEXLIST *indexList, GLushort *outIndexBuffer, unsigned int *outIndexCount) = 0; virtual Render3DError SetupVertices(const VERTLIST *vertList, const POLYLIST *polyList, const INDEXLIST *indexList, GLushort *outIndexBuffer, size_t *outIndexCount) = 0;
virtual Render3DError EnableVertexAttributes(const VERTLIST *vertList, const GLushort *indexBuffer, const unsigned int vertIndexCount) = 0; virtual Render3DError EnableVertexAttributes(const VERTLIST *vertList, const GLushort *indexBuffer, const size_t vertIndexCount) = 0;
virtual Render3DError DisableVertexAttributes() = 0; virtual Render3DError DisableVertexAttributes() = 0;
virtual Render3DError SelectRenderingFramebuffer() = 0; virtual Render3DError SelectRenderingFramebuffer() = 0;
virtual Render3DError DownsampleFBO() = 0; virtual Render3DError DownsampleFBO() = 0;
@ -460,7 +457,7 @@ protected:
virtual Render3DError PostRender() = 0; virtual Render3DError PostRender() = 0;
virtual Render3DError EndRender(const u64 frameCount) = 0; virtual Render3DError EndRender(const u64 frameCount) = 0;
virtual Render3DError UpdateClearImage(const u16 *__restrict colorBuffer, const u16 *__restrict depthBuffer, const u8 clearStencil, const u8 xScroll, const u8 yScroll) = 0; virtual Render3DError UpdateClearImage(const u16 *__restrict colorBuffer, const u32 *__restrict depthStencilBuffer) = 0;
virtual Render3DError UpdateToonTable(const u16 *toonTableBuffer) = 0; virtual Render3DError UpdateToonTable(const u16 *toonTableBuffer) = 0;
virtual Render3DError ClearUsingImage() const = 0; virtual Render3DError ClearUsingImage() const = 0;
@ -468,7 +465,7 @@ protected:
virtual Render3DError SetupPolygon(const POLY *thePoly) = 0; virtual Render3DError SetupPolygon(const POLY *thePoly) = 0;
virtual Render3DError SetupTexture(const POLY *thePoly, bool enableTexturing) = 0; virtual Render3DError SetupTexture(const POLY *thePoly, bool enableTexturing) = 0;
virtual Render3DError SetupViewport(const POLY *thePoly) = 0; virtual Render3DError SetupViewport(const u32 viewportValue) = 0;
public: public:
OpenGLRenderer(); OpenGLRenderer();
@ -512,15 +509,13 @@ protected:
virtual Render3DError SetupShaderIO(); virtual Render3DError SetupShaderIO();
virtual Render3DError CreateToonTable(); virtual Render3DError CreateToonTable();
virtual Render3DError DestroyToonTable(); virtual Render3DError DestroyToonTable();
virtual Render3DError UploadToonTable(const GLushort *toonTableBuffer); virtual Render3DError UploadToonTable(const u16 *toonTableBuffer);
virtual Render3DError CreateClearImage(); virtual Render3DError UploadClearImage(const u16 *clearImageColor16Buffer, const u32 *clearImageDepthStencilBuffer);
virtual Render3DError DestroyClearImage();
virtual Render3DError UploadClearImage(const GLushort *clearImageColorBuffer, const GLint *clearImageDepthBuffer);
virtual void GetExtensionSet(std::set<std::string> *oglExtensionSet); virtual void GetExtensionSet(std::set<std::string> *oglExtensionSet);
virtual Render3DError ExpandFreeTextures(); virtual Render3DError ExpandFreeTextures();
virtual Render3DError SetupVertices(const VERTLIST *vertList, const POLYLIST *polyList, const INDEXLIST *indexList, GLushort *outIndexBuffer, unsigned int *outIndexCount); virtual Render3DError SetupVertices(const VERTLIST *vertList, const POLYLIST *polyList, const INDEXLIST *indexList, GLushort *outIndexBuffer, size_t *outIndexCount);
virtual Render3DError EnableVertexAttributes(const VERTLIST *vertList, const GLushort *indexBuffer, const unsigned int vertIndexCount); virtual Render3DError EnableVertexAttributes(const VERTLIST *vertList, const GLushort *indexBuffer, const size_t vertIndexCount);
virtual Render3DError DisableVertexAttributes(); virtual Render3DError DisableVertexAttributes();
virtual Render3DError SelectRenderingFramebuffer(); virtual Render3DError SelectRenderingFramebuffer();
virtual Render3DError DownsampleFBO(); virtual Render3DError DownsampleFBO();
@ -533,7 +528,7 @@ protected:
virtual Render3DError PostRender(); virtual Render3DError PostRender();
virtual Render3DError EndRender(const u64 frameCount); virtual Render3DError EndRender(const u64 frameCount);
virtual Render3DError UpdateClearImage(const u16 *__restrict colorBuffer, const u16 *__restrict depthBuffer, const u8 clearStencil, const u8 xScroll, const u8 yScroll); virtual Render3DError UpdateClearImage(const u16 *__restrict colorBuffer, const u32 *__restrict depthStencilBuffer);
virtual Render3DError UpdateToonTable(const u16 *toonTableBuffer); virtual Render3DError UpdateToonTable(const u16 *toonTableBuffer);
virtual Render3DError ClearUsingImage() const; virtual Render3DError ClearUsingImage() const;
@ -541,7 +536,7 @@ protected:
virtual Render3DError SetupPolygon(const POLY *thePoly); virtual Render3DError SetupPolygon(const POLY *thePoly);
virtual Render3DError SetupTexture(const POLY *thePoly, bool enableTexturing); virtual Render3DError SetupTexture(const POLY *thePoly, bool enableTexturing);
virtual Render3DError SetupViewport(const POLY *thePoly); virtual Render3DError SetupViewport(const u32 viewportValue);
public: public:
OpenGLRenderer_1_2(); OpenGLRenderer_1_2();
@ -557,12 +552,8 @@ public:
class OpenGLRenderer_1_3 : public OpenGLRenderer_1_2 class OpenGLRenderer_1_3 : public OpenGLRenderer_1_2
{ {
protected: protected:
virtual Render3DError CreateToonTable(); virtual Render3DError UploadToonTable(const u16 *toonTableBuffer);
virtual Render3DError DestroyToonTable(); virtual Render3DError UploadClearImage(const u16 *clearImageColor16Buffer, const u32 *clearImageDepthStencilBuffer);
virtual Render3DError UploadToonTable(const GLushort *toonTableBuffer);
virtual Render3DError CreateClearImage();
virtual Render3DError DestroyClearImage();
virtual Render3DError UploadClearImage(const GLushort *clearImageColorBuffer, const GLint *clearImageDepthBuffer);
}; };
class OpenGLRenderer_1_4 : public OpenGLRenderer_1_3 class OpenGLRenderer_1_4 : public OpenGLRenderer_1_3
@ -580,7 +571,7 @@ protected:
virtual void DestroyPBOs(); virtual void DestroyPBOs();
virtual Render3DError CreateVAOs(); virtual Render3DError CreateVAOs();
virtual Render3DError EnableVertexAttributes(const VERTLIST *vertList, const GLushort *indexBuffer, const unsigned int vertIndexCount); virtual Render3DError EnableVertexAttributes(const VERTLIST *vertList, const GLushort *indexBuffer, const size_t vertIndexCount);
virtual Render3DError DisableVertexAttributes(); virtual Render3DError DisableVertexAttributes();
virtual Render3DError ReadBackPixels(); virtual Render3DError ReadBackPixels();
@ -596,8 +587,8 @@ protected:
virtual Render3DError InitExtensions(); virtual Render3DError InitExtensions();
virtual Render3DError InitFinalRenderStates(const std::set<std::string> *oglExtensionSet); virtual Render3DError InitFinalRenderStates(const std::set<std::string> *oglExtensionSet);
virtual Render3DError SetupVertices(const VERTLIST *vertList, const POLYLIST *polyList, const INDEXLIST *indexList, GLushort *outIndexBuffer, unsigned int *outIndexCount); virtual Render3DError SetupVertices(const VERTLIST *vertList, const POLYLIST *polyList, const INDEXLIST *indexList, GLushort *outIndexBuffer, size_t *outIndexCount);
virtual Render3DError EnableVertexAttributes(const VERTLIST *vertList, const GLushort *indexBuffer, const unsigned int vertIndexCount); virtual Render3DError EnableVertexAttributes(const VERTLIST *vertList, const GLushort *indexBuffer, const size_t vertIndexCount);
virtual Render3DError DisableVertexAttributes(); virtual Render3DError DisableVertexAttributes();
virtual Render3DError BeginRender(const GFX3D_State *renderState); virtual Render3DError BeginRender(const GFX3D_State *renderState);

View File

@ -1,7 +1,7 @@
/* /*
Copyright (C) 2006 yopyop Copyright (C) 2006 yopyop
Copyright (C) 2006-2007 shash Copyright (C) 2006-2007 shash
Copyright (C) 2008-2013 DeSmuME team Copyright (C) 2008-2015 DeSmuME team
This file is free software: you can redistribute it and/or modify This file is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by it under the terms of the GNU General Public License as published by
@ -110,80 +110,55 @@ static const char *fragmentShader_150 = {"\
\n\ \n\
uniform sampler2D texMainRender; \n\ uniform sampler2D texMainRender; \n\
uniform sampler1D texToonTable; \n\ uniform sampler1D texToonTable; \n\
\n\
uniform int stateToonShadingMode; \n\
uniform bool stateEnableAlphaTest; \n\
uniform bool stateUseWDepth; \n\
uniform float stateAlphaTestRef; \n\
\n\
uniform int polyMode; \n\
uniform int polyID; \n\ uniform int polyID; \n\
uniform bool hasTexture; \n\ \n\
uniform int polygonMode; \n\ uniform bool polyEnableTexture; \n\
uniform int toonShadingMode; \n\
uniform bool oglWBuffer; \n\
uniform bool enableAlphaTest; \n\
uniform float alphaTestRef; \n\
\n\ \n\
out vec4 outFragColor; \n\ out vec4 outFragColor; \n\
\n\ \n\
void main() \n\ void main() \n\
{ \n\ { \n\
vec4 texColor = vec4(1.0, 1.0, 1.0, 1.0); \n\ vec4 mainTexColor = (polyEnableTexture) ? texture(texMainRender, vtxTexCoord) : vec4(1.0, 1.0, 1.0, 1.0); \n\
vec4 fragColor; \n\ vec4 tempFragColor = mainTexColor; \n\
\n\ \n\
if(hasTexture) \n\ if(polyMode == 0) \n\
{ \n\ { \n\
texColor = texture(texMainRender, vtxTexCoord); \n\ tempFragColor = vtxColor * mainTexColor; \n\
} \n\ } \n\
\n\ else if(polyMode == 1) \n\
fragColor = texColor; \n\
\n\
if(polygonMode == 0) \n\
{ \n\ { \n\
fragColor = vtxColor * texColor; \n\ tempFragColor.rgb = (polyEnableTexture) ? (mainTexColor.rgb * mainTexColor.a) + (vtxColor.rgb * (1.0 - mainTexColor.a)) : vtxColor.rgb; \n\
tempFragColor.a = vtxColor.a; \n\
} \n\ } \n\
else if(polygonMode == 1) \n\ else if(polyMode == 2) \n\
{ \n\
if (texColor.a == 0.0 || !hasTexture) \n\
{ \n\
fragColor.rgb = vtxColor.rgb; \n\
} \n\
else if (texColor.a == 1.0) \n\
{ \n\
fragColor.rgb = texColor.rgb; \n\
} \n\
else \n\
{ \n\
fragColor.rgb = texColor.rgb * (1.0-texColor.a) + vtxColor.rgb * texColor.a; \n\
} \n\
\n\
fragColor.a = vtxColor.a; \n\
} \n\
else if(polygonMode == 2) \n\
{ \n\ { \n\
vec3 toonColor = vec3(texture(texToonTable, vtxColor.r).rgb); \n\ vec3 toonColor = vec3(texture(texToonTable, vtxColor.r).rgb); \n\
\n\ tempFragColor.rgb = (stateToonShadingMode == 0) ? mainTexColor.rgb * toonColor.rgb : min((mainTexColor.rgb * vtxColor.rgb) + toonColor.rgb, 1.0); \n\
if (toonShadingMode == 0) \n\ tempFragColor.a = mainTexColor.a * vtxColor.a; \n\
{ \n\
fragColor.rgb = texColor.rgb * toonColor.rgb; \n\
} \n\ } \n\
else \n\ else if(polyMode == 3) \n\
{ \n\
fragColor.rgb = texColor.rgb * vtxColor.rgb + toonColor.rgb; \n\
} \n\
\n\
fragColor.a = texColor.a * vtxColor.a; \n\
} \n\
else if(polygonMode == 3) \n\
{ \n\ { \n\
if (polyID != 0) \n\ if (polyID != 0) \n\
{ \n\ { \n\
fragColor = vtxColor; \n\ tempFragColor = vtxColor; \n\
} \n\ } \n\
} \n\ } \n\
\n\ \n\
if (fragColor.a == 0.0 || (enableAlphaTest && fragColor.a < alphaTestRef)) \n\ if (tempFragColor.a == 0.0 || (stateEnableAlphaTest && tempFragColor.a < stateAlphaTestRef)) \n\
{ \n\ { \n\
discard; \n\ discard; \n\
} \n\ } \n\
\n\ \n\
float vertW = (vtxPosition.w == 0.0) ? 0.00000001 : vtxPosition.w; \n\ float vertW = (vtxPosition.w == 0.0) ? 0.00000001 : vtxPosition.w; \n\
gl_FragDepth = (oglWBuffer) ? vtxPosition.w/4096.0 : clamp((vtxPosition.z/vertW) * 0.5 + 0.5, 0.0, 1.0); \n\ gl_FragDepth = (stateUseWDepth) ? vtxPosition.w/4096.0 : clamp((vtxPosition.z/vertW) * 0.5 + 0.5, 0.0, 1.0); \n\
outFragColor = fragColor; \n\ outFragColor = tempFragColor; \n\
} \n\ } \n\
"}; "};
@ -252,7 +227,7 @@ Render3DError OpenGLRenderer_3_2::InitExtensions()
error = this->CreateFBOs(); error = this->CreateFBOs();
if (error != OGLERROR_NOERR) if (error != OGLERROR_NOERR)
{ {
OGLRef.fboFinalOutputID = 0; OGLRef.fboRenderID = 0;
this->isFBOSupported = false; this->isFBOSupported = false;
return error; return error;
} }
@ -283,19 +258,28 @@ Render3DError OpenGLRenderer_3_2::CreateFBOs()
OGLRenderRef &OGLRef = *this->ref; OGLRenderRef &OGLRef = *this->ref;
// Set up FBO render targets // Set up FBO render targets
this->CreateClearImage(); glGenTextures(1, &OGLRef.texClearImageColorID);
glGenTextures(1, &OGLRef.texClearImageDepthStencilID);
glGenRenderbuffers(1, &OGLRef.rboFinalOutputColorID); glBindTexture(GL_TEXTURE_2D, OGLRef.texClearImageColorID);
glGenRenderbuffers(1, &OGLRef.rboFinalOutputDepthStencilID); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, GFX3D_FRAMEBUFFER_WIDTH, GFX3D_FRAMEBUFFER_HEIGHT, 0, GL_RGBA, GL_UNSIGNED_INT_8_8_8_8_REV, NULL);
glBindRenderbuffer(GL_RENDERBUFFER, OGLRef.rboFinalOutputColorID); glBindTexture(GL_TEXTURE_2D, OGLRef.texClearImageDepthStencilID);
glRenderbufferStorage(GL_RENDERBUFFER, GL_RGBA, GFX3D_FRAMEBUFFER_WIDTH, GFX3D_FRAMEBUFFER_HEIGHT); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
glBindRenderbuffer(GL_RENDERBUFFER, OGLRef.rboFinalOutputDepthStencilID); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
glRenderbufferStorage(GL_RENDERBUFFER, GL_DEPTH24_STENCIL8, GFX3D_FRAMEBUFFER_WIDTH, GFX3D_FRAMEBUFFER_HEIGHT); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE);
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_COMPARE_MODE, GL_NONE);
glTexImage2D(GL_TEXTURE_2D, 0, GL_DEPTH24_STENCIL8, GFX3D_FRAMEBUFFER_WIDTH, GFX3D_FRAMEBUFFER_HEIGHT, 0, GL_DEPTH_STENCIL, GL_UNSIGNED_INT_24_8, NULL);
glBindTexture(GL_TEXTURE_2D, 0);
// Set up FBOs // Set up FBOs
glGenFramebuffers(1, &OGLRef.fboClearImageID); glGenFramebuffers(1, &OGLRef.fboClearImageID);
glBindFramebuffer(GL_FRAMEBUFFER, OGLRef.fboClearImageID); glBindFramebuffer(GL_FRAMEBUFFER, OGLRef.fboClearImageID);
glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, OGLRef.texClearImageColorID, 0); glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, OGLRef.texClearImageColorID, 0);
glFramebufferTexture2D(GL_FRAMEBUFFER, GL_DEPTH_STENCIL_ATTACHMENT, GL_TEXTURE_2D, OGLRef.texClearImageDepthStencilID, 0); glFramebufferTexture2D(GL_FRAMEBUFFER, GL_DEPTH_STENCIL_ATTACHMENT, GL_TEXTURE_2D, OGLRef.texClearImageDepthStencilID, 0);
@ -305,37 +289,40 @@ Render3DError OpenGLRenderer_3_2::CreateFBOs()
INFO("OpenGL: Failed to created FBOs. Some emulation features will be disabled.\n"); INFO("OpenGL: Failed to created FBOs. Some emulation features will be disabled.\n");
glBindFramebuffer(GL_FRAMEBUFFER, 0); glBindFramebuffer(GL_FRAMEBUFFER, 0);
glDeleteFramebuffers(1, &OGLRef.fboClearImageID); glDeleteFramebuffers(1, &OGLRef.fboClearImageID);
this->DestroyClearImage(); glDeleteTextures(1, &OGLRef.texClearImageColorID);
glDeleteTextures(1, &OGLRef.texClearImageDepthStencilID);
glDeleteRenderbuffers(1, &OGLRef.rboFinalOutputColorID);
glDeleteRenderbuffers(1, &OGLRef.rboFinalOutputDepthStencilID);
return OGLERROR_FBO_CREATE_ERROR; return OGLERROR_FBO_CREATE_ERROR;
} }
// Set up final output FBO // Set up final output FBO
glGenFramebuffers(1, &OGLRef.fboFinalOutputID); glGenRenderbuffers(1, &OGLRef.rboFragColorID);
glGenRenderbuffers(1, &OGLRef.rboFragDepthStencilID);
glBindRenderbuffer(GL_RENDERBUFFER, OGLRef.rboFragColorID);
glRenderbufferStorage(GL_RENDERBUFFER, GL_RGBA, GFX3D_FRAMEBUFFER_WIDTH, GFX3D_FRAMEBUFFER_HEIGHT);
glBindRenderbuffer(GL_RENDERBUFFER, OGLRef.rboFragDepthStencilID);
glRenderbufferStorage(GL_RENDERBUFFER, GL_DEPTH24_STENCIL8, GFX3D_FRAMEBUFFER_WIDTH, GFX3D_FRAMEBUFFER_HEIGHT);
glBindFramebuffer(GL_FRAMEBUFFER, OGLRef.fboFinalOutputID); glGenFramebuffers(1, &OGLRef.fboRenderID);
glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, OGLRef.rboFinalOutputColorID); glBindFramebuffer(GL_FRAMEBUFFER, OGLRef.fboRenderID);
glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_DEPTH_STENCIL_ATTACHMENT, GL_RENDERBUFFER, OGLRef.rboFinalOutputDepthStencilID); glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, OGLRef.rboFragColorID);
glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_DEPTH_STENCIL_ATTACHMENT, GL_RENDERBUFFER, OGLRef.rboFragDepthStencilID);
if (glCheckFramebufferStatus(GL_FRAMEBUFFER) != GL_FRAMEBUFFER_COMPLETE) if (glCheckFramebufferStatus(GL_FRAMEBUFFER) != GL_FRAMEBUFFER_COMPLETE)
{ {
INFO("OpenGL: Failed to created FBOs. Some emulation features will be disabled.\n"); INFO("OpenGL: Failed to created FBOs. Some emulation features will be disabled.\n");
glBindFramebuffer(GL_FRAMEBUFFER, 0); glBindFramebuffer(GL_FRAMEBUFFER, 0);
glDeleteFramebuffers(1, &OGLRef.fboClearImageID); glDeleteFramebuffers(1, &OGLRef.fboClearImageID);
this->DestroyClearImage(); glDeleteTextures(1, &OGLRef.texClearImageColorID);
glDeleteTextures(1, &OGLRef.texClearImageDepthStencilID);
glDeleteFramebuffers(1, &OGLRef.fboFinalOutputID); glDeleteFramebuffers(1, &OGLRef.fboRenderID);
glDeleteRenderbuffers(1, &OGLRef.rboFinalOutputColorID); glDeleteRenderbuffers(1, &OGLRef.rboFragColorID);
glDeleteRenderbuffers(1, &OGLRef.rboFinalOutputDepthStencilID); glDeleteRenderbuffers(1, &OGLRef.rboFragDepthStencilID);
OGLRef.fboFinalOutputID = 0; OGLRef.fboRenderID = 0;
return OGLERROR_FBO_CREATE_ERROR; return OGLERROR_FBO_CREATE_ERROR;
} }
@ -356,13 +343,13 @@ void OpenGLRenderer_3_2::DestroyFBOs()
OGLRenderRef &OGLRef = *this->ref; OGLRenderRef &OGLRef = *this->ref;
glBindFramebuffer(GL_FRAMEBUFFER, 0); glBindFramebuffer(GL_FRAMEBUFFER, 0);
glDeleteFramebuffers(1, &OGLRef.fboClearImageID); glDeleteFramebuffers(1, &OGLRef.fboClearImageID);
this->DestroyClearImage(); glDeleteTextures(1, &OGLRef.texClearImageColorID);
glDeleteTextures(1, &OGLRef.texClearImageDepthStencilID);
glDeleteFramebuffers(1, &OGLRef.fboFinalOutputID); glDeleteFramebuffers(1, &OGLRef.fboRenderID);
glDeleteRenderbuffers(1, &OGLRef.rboFinalOutputColorID); glDeleteRenderbuffers(1, &OGLRef.rboFragColorID);
glDeleteRenderbuffers(1, &OGLRef.rboFinalOutputDepthStencilID); glDeleteRenderbuffers(1, &OGLRef.rboFragDepthStencilID);
this->isFBOSupported = false; this->isFBOSupported = false;
} }
@ -388,33 +375,32 @@ Render3DError OpenGLRenderer_3_2::CreateMultisampledFBO()
OGLRenderRef &OGLRef = *this->ref; OGLRenderRef &OGLRef = *this->ref;
// Set up FBO render targets // Set up FBO render targets
glGenRenderbuffers(1, &OGLRef.rboMultisampleColorID); glGenRenderbuffers(1, &OGLRef.rboMSFragColorID);
glGenRenderbuffers(1, &OGLRef.rboMultisampleDepthStencilID); glGenRenderbuffers(1, &OGLRef.rboMSFragDepthStencilID);
glBindRenderbuffer(GL_RENDERBUFFER, OGLRef.rboMultisampleColorID); glBindRenderbuffer(GL_RENDERBUFFER, OGLRef.rboMSFragColorID);
glRenderbufferStorageMultisample(GL_RENDERBUFFER, maxSamples, GL_RGBA, GFX3D_FRAMEBUFFER_WIDTH, GFX3D_FRAMEBUFFER_HEIGHT); glRenderbufferStorageMultisample(GL_RENDERBUFFER, maxSamples, GL_RGBA, GFX3D_FRAMEBUFFER_WIDTH, GFX3D_FRAMEBUFFER_HEIGHT);
glBindRenderbuffer(GL_RENDERBUFFER, OGLRef.rboMultisampleDepthStencilID); glBindRenderbuffer(GL_RENDERBUFFER, OGLRef.rboMSFragDepthStencilID);
glRenderbufferStorageMultisample(GL_RENDERBUFFER, maxSamples, GL_DEPTH24_STENCIL8, GFX3D_FRAMEBUFFER_WIDTH, GFX3D_FRAMEBUFFER_HEIGHT); glRenderbufferStorageMultisample(GL_RENDERBUFFER, maxSamples, GL_DEPTH24_STENCIL8, GFX3D_FRAMEBUFFER_WIDTH, GFX3D_FRAMEBUFFER_HEIGHT);
// Set up multisampled rendering FBO // Set up multisampled rendering FBO
glGenFramebuffers(1, &OGLRef.fboMultisampleRenderID); glGenFramebuffers(1, &OGLRef.fboMSIntermediateRenderID);
glBindFramebuffer(GL_FRAMEBUFFER, OGLRef.fboMSIntermediateRenderID);
glBindFramebuffer(GL_FRAMEBUFFER, OGLRef.fboMultisampleRenderID); glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, OGLRef.rboMSFragColorID);
glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_RENDERBUFFER, OGLRef.rboMultisampleColorID); glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_DEPTH_STENCIL_ATTACHMENT, GL_RENDERBUFFER, OGLRef.rboMSFragDepthStencilID);
glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_DEPTH_STENCIL_ATTACHMENT, GL_RENDERBUFFER, OGLRef.rboMultisampleDepthStencilID);
if (glCheckFramebufferStatus(GL_FRAMEBUFFER) != GL_FRAMEBUFFER_COMPLETE) if (glCheckFramebufferStatus(GL_FRAMEBUFFER) != GL_FRAMEBUFFER_COMPLETE)
{ {
glBindFramebuffer(GL_FRAMEBUFFER, 0); glBindFramebuffer(GL_FRAMEBUFFER, 0);
glDeleteFramebuffers(1, &OGLRef.fboMultisampleRenderID); glDeleteFramebuffers(1, &OGLRef.fboMSIntermediateRenderID);
glDeleteRenderbuffers(1, &OGLRef.rboMultisampleColorID); glDeleteRenderbuffers(1, &OGLRef.rboMSFragColorID);
glDeleteRenderbuffers(1, &OGLRef.rboMultisampleDepthStencilID); glDeleteRenderbuffers(1, &OGLRef.rboMSFragDepthStencilID);
INFO("OpenGL: Failed to create multisampled FBO. Multisample antialiasing will be disabled.\n"); INFO("OpenGL: Failed to create multisampled FBO. Multisample antialiasing will be disabled.\n");
return OGLERROR_FBO_CREATE_ERROR; return OGLERROR_FBO_CREATE_ERROR;
} }
glBindFramebuffer(GL_FRAMEBUFFER, OGLRef.fboFinalOutputID); glBindFramebuffer(GL_FRAMEBUFFER, OGLRef.fboRenderID);
INFO("OpenGL: Successfully created multisampled FBO.\n"); INFO("OpenGL: Successfully created multisampled FBO.\n");
return OGLERROR_NOERR; return OGLERROR_NOERR;
@ -430,9 +416,9 @@ void OpenGLRenderer_3_2::DestroyMultisampledFBO()
OGLRenderRef &OGLRef = *this->ref; OGLRenderRef &OGLRef = *this->ref;
glBindFramebuffer(GL_FRAMEBUFFER, 0); glBindFramebuffer(GL_FRAMEBUFFER, 0);
glDeleteFramebuffers(1, &OGLRef.fboMultisampleRenderID); glDeleteFramebuffers(1, &OGLRef.fboMSIntermediateRenderID);
glDeleteRenderbuffers(1, &OGLRef.rboMultisampleColorID); glDeleteRenderbuffers(1, &OGLRef.rboMSFragColorID);
glDeleteRenderbuffers(1, &OGLRef.rboMultisampleDepthStencilID); glDeleteRenderbuffers(1, &OGLRef.rboMSFragDepthStencilID);
this->isMultisampledFBOSupported = false; this->isMultisampledFBOSupported = false;
} }
@ -498,14 +484,14 @@ void OpenGLRenderer_3_2::GetExtensionSet(std::set<std::string> *oglExtensionSet)
GLint extensionCount = 0; GLint extensionCount = 0;
glGetIntegerv(GL_NUM_EXTENSIONS, &extensionCount); glGetIntegerv(GL_NUM_EXTENSIONS, &extensionCount);
for (unsigned int i = 0; i < extensionCount; i++) for (size_t i = 0; i < extensionCount; i++)
{ {
std::string extensionName = std::string((const char *)glGetStringi(GL_EXTENSIONS, i)); std::string extensionName = std::string((const char *)glGetStringi(GL_EXTENSIONS, i));
oglExtensionSet->insert(extensionName); oglExtensionSet->insert(extensionName);
} }
} }
Render3DError OpenGLRenderer_3_2::EnableVertexAttributes(const VERTLIST *vertList, const GLushort *indexBuffer, const unsigned int vertIndexCount) Render3DError OpenGLRenderer_3_2::EnableVertexAttributes(const VERTLIST *vertList, const GLushort *indexBuffer, const size_t vertIndexCount)
{ {
OGLRenderRef &OGLRef = *this->ref; OGLRenderRef &OGLRef = *this->ref;
@ -525,15 +511,11 @@ Render3DError OpenGLRenderer_3_2::DisableVertexAttributes()
Render3DError OpenGLRenderer_3_2::SelectRenderingFramebuffer() Render3DError OpenGLRenderer_3_2::SelectRenderingFramebuffer()
{ {
OGLRenderRef &OGLRef = *this->ref; OGLRenderRef &OGLRef = *this->ref;
static const GLenum drawDirect[1] = {GL_COLOR_ATTACHMENT0};
if (this->isMultisampledFBOSupported) OGLRef.selectedRenderingFBO = (CommonSettings.GFX3D_Renderer_Multisample) ? OGLRef.fboMSIntermediateRenderID : OGLRef.fboRenderID;
{
OGLRef.selectedRenderingFBO = CommonSettings.GFX3D_Renderer_Multisample ? OGLRef.fboMultisampleRenderID : OGLRef.fboFinalOutputID;
glBindFramebuffer(GL_FRAMEBUFFER, OGLRef.selectedRenderingFBO); glBindFramebuffer(GL_FRAMEBUFFER, OGLRef.selectedRenderingFBO);
glDrawBuffers(1, &drawDirect[0]);
const GLenum drawBufferList = {GL_COLOR_ATTACHMENT0};
glDrawBuffers(1, &drawBufferList);
}
return OGLERROR_NOERR; return OGLERROR_NOERR;
} }
@ -542,15 +524,15 @@ Render3DError OpenGLRenderer_3_2::DownsampleFBO()
{ {
OGLRenderRef &OGLRef = *this->ref; OGLRenderRef &OGLRef = *this->ref;
if (!this->isMultisampledFBOSupported || OGLRef.selectedRenderingFBO != OGLRef.fboMultisampleRenderID) if (OGLRef.selectedRenderingFBO != OGLRef.fboMSIntermediateRenderID)
{ {
return OGLERROR_NOERR; return OGLERROR_NOERR;
} }
glBindFramebuffer(GL_READ_FRAMEBUFFER, OGLRef.selectedRenderingFBO); glBindFramebuffer(GL_READ_FRAMEBUFFER, OGLRef.selectedRenderingFBO);
glBindFramebuffer(GL_DRAW_FRAMEBUFFER, OGLRef.fboFinalOutputID); glBindFramebuffer(GL_DRAW_FRAMEBUFFER, OGLRef.fboRenderID);
glBlitFramebuffer(0, 0, GFX3D_FRAMEBUFFER_WIDTH, GFX3D_FRAMEBUFFER_HEIGHT, 0, 0, GFX3D_FRAMEBUFFER_WIDTH, GFX3D_FRAMEBUFFER_HEIGHT, GL_COLOR_BUFFER_BIT, GL_NEAREST); glBlitFramebuffer(0, 0, GFX3D_FRAMEBUFFER_WIDTH, GFX3D_FRAMEBUFFER_HEIGHT, 0, 0, GFX3D_FRAMEBUFFER_WIDTH, GFX3D_FRAMEBUFFER_HEIGHT, GL_COLOR_BUFFER_BIT, GL_NEAREST);
glBindFramebuffer(GL_FRAMEBUFFER, OGLRef.fboFinalOutputID); glBindFramebuffer(GL_FRAMEBUFFER, OGLRef.fboRenderID);
return OGLERROR_NOERR; return OGLERROR_NOERR;
} }

View File

@ -72,7 +72,7 @@ protected:
virtual Render3DError SetupShaderIO(); virtual Render3DError SetupShaderIO();
virtual void GetExtensionSet(std::set<std::string> *oglExtensionSet); virtual void GetExtensionSet(std::set<std::string> *oglExtensionSet);
virtual Render3DError EnableVertexAttributes(const VERTLIST *vertList, const GLushort *indexBuffer, const unsigned int vertIndexCount); virtual Render3DError EnableVertexAttributes(const VERTLIST *vertList, const GLushort *indexBuffer, const size_t vertIndexCount);
virtual Render3DError DisableVertexAttributes(); virtual Render3DError DisableVertexAttributes();
virtual Render3DError SelectRenderingFramebuffer(); virtual Render3DError SelectRenderingFramebuffer();
virtual Render3DError DownsampleFBO(); virtual Render3DError DownsampleFBO();

View File

@ -24,6 +24,7 @@
#include "MMU.h" #include "MMU.h"
#include "texcache.h" #include "texcache.h"
static CACHE_ALIGN u32 dsDepthToD24S8_LUT[32768] = {0};
int cur3DCore = GPU3D_NULL; int cur3DCore = GPU3D_NULL;
GPU3DInterface gpu3DNull = { GPU3DInterface gpu3DNull = {
@ -97,6 +98,23 @@ bool NDS_3D_ChangeCore(int newCore)
return true; return true;
} }
Render3D::Render3D()
{
static bool needTableInit = true;
if (needTableInit)
{
for (size_t i = 0; i < 32768; i++)
{
dsDepthToD24S8_LUT[i] = (u32)DS_DEPTH15TO24(i) << 8;
}
needTableInit = false;
}
Reset();
}
Render3DError Render3D::BeginRender(const GFX3D_State *renderState) Render3DError Render3D::BeginRender(const GFX3D_State *renderState)
{ {
return RENDER3DERROR_NOERR; return RENDER3DERROR_NOERR;
@ -122,7 +140,7 @@ Render3DError Render3D::EndRender(const u64 frameCount)
return RENDER3DERROR_NOERR; return RENDER3DERROR_NOERR;
} }
Render3DError Render3D::UpdateClearImage(const u16 *__restrict colorBuffer, const u16 *__restrict depthBuffer, const u8 clearStencil, const u8 xScroll, const u8 yScroll) Render3DError Render3D::UpdateClearImage(const u16 *__restrict colorBuffer, const u32 *__restrict depthStencilBuffer)
{ {
return RENDER3DERROR_NOERR; return RENDER3DERROR_NOERR;
} }
@ -149,7 +167,7 @@ Render3DError Render3D::ClearFramebuffer(const GFX3D_State *renderState)
clearColor.b = (renderState->clearColor >> 10) & 0x1F; clearColor.b = (renderState->clearColor >> 10) & 0x1F;
clearColor.a = (renderState->clearColor >> 16) & 0x1F; clearColor.a = (renderState->clearColor >> 16) & 0x1F;
const u8 clearStencil = (renderState->clearColor >> 24) & 0x3F; const u8 polyID = (renderState->clearColor >> 24) & 0x3F;
if (renderState->enableClearImage) if (renderState->enableClearImage)
{ {
@ -159,19 +177,39 @@ Render3DError Render3D::ClearFramebuffer(const GFX3D_State *renderState)
const u8 xScroll = scrollBits & 0xFF; const u8 xScroll = scrollBits & 0xFF;
const u8 yScroll = (scrollBits >> 8) & 0xFF; const u8 yScroll = (scrollBits >> 8) & 0xFF;
error = this->UpdateClearImage(clearColorBuffer, clearDepthBuffer, clearStencil, xScroll, yScroll); size_t dd = (GFX3D_FRAMEBUFFER_WIDTH * GFX3D_FRAMEBUFFER_HEIGHT) - GFX3D_FRAMEBUFFER_WIDTH;
for (size_t iy = 0; iy < GFX3D_FRAMEBUFFER_HEIGHT; iy++)
{
const size_t y = ((iy + yScroll) & 0xFF) << 8;
for (size_t ix = 0; ix < GFX3D_FRAMEBUFFER_WIDTH; ix++)
{
const size_t x = (ix + xScroll) & 0xFF;
const size_t adr = y + x;
this->clearImageColor16Buffer[dd] = clearColorBuffer[adr];
this->clearImageDepthStencilBuffer[dd] = dsDepthToD24S8_LUT[clearDepthBuffer[adr] & 0x7FFF] | polyID;
dd++;
}
dd -= GFX3D_FRAMEBUFFER_WIDTH * 2;
}
error = this->UpdateClearImage(this->clearImageColor16Buffer, this->clearImageDepthStencilBuffer);
if (error == RENDER3DERROR_NOERR) if (error == RENDER3DERROR_NOERR)
{ {
error = this->ClearUsingImage(); error = this->ClearUsingImage();
} }
else else
{ {
error = this->ClearUsingValues(clearColor.r, clearColor.g, clearColor.b, clearColor.a, renderState->clearDepth, clearStencil); error = this->ClearUsingValues(clearColor.r, clearColor.g, clearColor.b, clearColor.a, renderState->clearDepth, polyID);
} }
} }
else else
{ {
error = this->ClearUsingValues(clearColor.r, clearColor.g, clearColor.b, clearColor.a, renderState->clearDepth, clearStencil); error = this->ClearUsingValues(clearColor.r, clearColor.g, clearColor.b, clearColor.a, renderState->clearDepth, polyID);
} }
return error; return error;
@ -197,13 +235,16 @@ Render3DError Render3D::SetupTexture(const POLY *thePoly, bool enableTexturing)
return RENDER3DERROR_NOERR; return RENDER3DERROR_NOERR;
} }
Render3DError Render3D::SetupViewport(const POLY *thePoly) Render3DError Render3D::SetupViewport(const u32 viewportValue)
{ {
return RENDER3DERROR_NOERR; return RENDER3DERROR_NOERR;
} }
Render3DError Render3D::Reset() Render3DError Render3D::Reset()
{ {
memset(this->clearImageColor16Buffer, 0, sizeof(this->clearImageColor16Buffer));
memset(this->clearImageDepthStencilBuffer, 0, sizeof(this->clearImageDepthStencilBuffer));
return RENDER3DERROR_NOERR; return RENDER3DERROR_NOERR;
} }

View File

@ -1,6 +1,6 @@
/* /*
Copyright (C) 2006-2007 shash Copyright (C) 2006-2007 shash
Copyright (C) 2007-2013 DeSmuME team Copyright (C) 2007-2015 DeSmuME team
This file is free software: you can redistribute it and/or modify This file is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by it under the terms of the GNU General Public License as published by
@ -84,13 +84,16 @@ typedef int Render3DError;
class Render3D class Render3D
{ {
protected: protected:
CACHE_ALIGN u16 clearImageColor16Buffer[GFX3D_FRAMEBUFFER_WIDTH * GFX3D_FRAMEBUFFER_HEIGHT];
CACHE_ALIGN u32 clearImageDepthStencilBuffer[GFX3D_FRAMEBUFFER_WIDTH * GFX3D_FRAMEBUFFER_HEIGHT];
virtual Render3DError BeginRender(const GFX3D_State *renderState); virtual Render3DError BeginRender(const GFX3D_State *renderState);
virtual Render3DError PreRender(const GFX3D_State *renderState, const VERTLIST *vertList, const POLYLIST *polyList, const INDEXLIST *indexList); virtual Render3DError PreRender(const GFX3D_State *renderState, const VERTLIST *vertList, const POLYLIST *polyList, const INDEXLIST *indexList);
virtual Render3DError DoRender(const GFX3D_State *renderState, const VERTLIST *vertList, const POLYLIST *polyList, const INDEXLIST *indexList); virtual Render3DError DoRender(const GFX3D_State *renderState, const VERTLIST *vertList, const POLYLIST *polyList, const INDEXLIST *indexList);
virtual Render3DError PostRender(); virtual Render3DError PostRender();
virtual Render3DError EndRender(const u64 frameCount); virtual Render3DError EndRender(const u64 frameCount);
virtual Render3DError UpdateClearImage(const u16 *__restrict colorBuffer, const u16 *__restrict depthBuffer, const u8 clearStencil, const u8 xScroll, const u8 yScroll); virtual Render3DError UpdateClearImage(const u16 *__restrict colorBuffer, const u32 *__restrict depthStencilBuffer);
virtual Render3DError UpdateToonTable(const u16 *toonTableBuffer); virtual Render3DError UpdateToonTable(const u16 *toonTableBuffer);
virtual Render3DError ClearFramebuffer(const GFX3D_State *renderState); virtual Render3DError ClearFramebuffer(const GFX3D_State *renderState);
@ -99,9 +102,11 @@ protected:
virtual Render3DError SetupPolygon(const POLY *thePoly); virtual Render3DError SetupPolygon(const POLY *thePoly);
virtual Render3DError SetupTexture(const POLY *thePoly, bool enableTexturing); virtual Render3DError SetupTexture(const POLY *thePoly, bool enableTexturing);
virtual Render3DError SetupViewport(const POLY *thePoly); virtual Render3DError SetupViewport(const u32 viewportValue);
public: public:
Render3D();
virtual Render3DError Reset(); virtual Render3DError Reset();
virtual Render3DError Render(const GFX3D_State *renderState, const VERTLIST *vertList, const POLYLIST *polyList, const INDEXLIST *indexList, const u64 frameCount); virtual Render3DError Render(const GFX3D_State *renderState, const VERTLIST *vertList, const POLYLIST *polyList, const INDEXLIST *indexList, const u64 frameCount);
virtual Render3DError RenderFinish(); virtual Render3DError RenderFinish();