2008-07-12 17:40:22 +00:00
|
|
|
// Copyright (C) 2003-2008 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/
|
|
|
|
|
2008-08-14 21:34:39 +00:00
|
|
|
|
|
|
|
// Preliminary non-working code.
|
|
|
|
|
2008-07-12 17:40:22 +00:00
|
|
|
#include "Globals.h"
|
2008-10-22 21:02:37 +00:00
|
|
|
#include "GLUtil.h"
|
2008-10-16 08:11:03 +00:00
|
|
|
#include "MemoryUtil.h"
|
2008-08-14 21:34:39 +00:00
|
|
|
#include "Render.h"
|
|
|
|
#include "TextureMngr.h"
|
2008-10-15 14:07:03 +00:00
|
|
|
#include "VertexShaderManager.h"
|
2008-08-14 21:34:39 +00:00
|
|
|
#include "XFBConvert.h"
|
2008-07-12 17:40:22 +00:00
|
|
|
|
2008-08-14 21:34:39 +00:00
|
|
|
enum {
|
|
|
|
XFB_WIDTH = 640,
|
2008-10-16 22:06:06 +00:00
|
|
|
XFB_HEIGHT = 480, // 528 is max height.
|
2008-10-15 21:29:44 +00:00
|
|
|
XFB_BUF_HEIGHT = 538, //480,
|
|
|
|
// TODO: figure out what to do with PAL
|
2008-08-14 21:34:39 +00:00
|
|
|
};
|
2008-07-12 17:40:22 +00:00
|
|
|
|
2008-08-14 21:34:39 +00:00
|
|
|
static GLuint xfb_texture;
|
2008-10-15 14:07:03 +00:00
|
|
|
static u8 *xfb_buffer = 0;
|
|
|
|
static u8 *efb_buffer = 0;
|
|
|
|
static GLuint s_xfbFrameBuffer = 0;
|
|
|
|
static GLuint s_xfbRenderBuffer = 0;
|
2008-07-12 17:40:22 +00:00
|
|
|
|
2008-08-14 21:34:39 +00:00
|
|
|
void XFB_Init()
|
2008-07-12 17:40:22 +00:00
|
|
|
{
|
2008-10-15 14:07:03 +00:00
|
|
|
// used to render XFB
|
2008-10-15 21:29:44 +00:00
|
|
|
xfb_buffer = new u8[XFB_WIDTH * XFB_BUF_HEIGHT * 4];
|
|
|
|
memset(xfb_buffer, 0, XFB_WIDTH * XFB_BUF_HEIGHT * 4);
|
2008-08-14 21:34:39 +00:00
|
|
|
glGenTextures(1, &xfb_texture);
|
2008-10-05 05:39:59 +00:00
|
|
|
glBindTexture(GL_TEXTURE_RECTANGLE_ARB, xfb_texture);
|
|
|
|
glTexImage2D(GL_TEXTURE_RECTANGLE_ARB, 0, 4, XFB_WIDTH, XFB_HEIGHT, 0, GL_BGRA, GL_UNSIGNED_BYTE, xfb_buffer);
|
2008-10-15 14:07:03 +00:00
|
|
|
|
|
|
|
// used to render EFB
|
|
|
|
glGenFramebuffersEXT( 1, &s_xfbFrameBuffer);
|
|
|
|
glGenRenderbuffersEXT(1, &s_xfbRenderBuffer);
|
|
|
|
glBindRenderbufferEXT(GL_RENDERBUFFER_EXT, s_xfbRenderBuffer);
|
|
|
|
glRenderbufferStorageEXT(GL_RENDERBUFFER_EXT, GL_RGBA, nBackbufferWidth, nBackbufferHeight);
|
|
|
|
|
2008-10-16 08:11:03 +00:00
|
|
|
// Ensure efb_buffer is aligned.
|
|
|
|
efb_buffer = (u8 *)AllocateMemoryPages(nBackbufferWidth * nBackbufferHeight * 4);
|
2008-07-12 17:40:22 +00:00
|
|
|
}
|
2008-08-07 21:29:15 +00:00
|
|
|
|
2008-10-16 08:11:03 +00:00
|
|
|
void XFB_Shutdown()
|
|
|
|
{
|
|
|
|
glDeleteFramebuffersEXT(1, &s_xfbFrameBuffer);
|
|
|
|
|
|
|
|
glDeleteTextures(1, &xfb_texture);
|
|
|
|
xfb_texture = 0;
|
|
|
|
delete [] xfb_buffer;
|
|
|
|
xfb_buffer = 0;
|
|
|
|
FreeMemoryPages(efb_buffer, nBackbufferWidth * nBackbufferHeight * 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-10-15 14:07:03 +00:00
|
|
|
void XFB_Write(u8 *xfb_in_ram, const TRectangle& sourceRc, u32 dstWd, u32 dstHt, float yScale)
|
2008-07-12 17:40:22 +00:00
|
|
|
{
|
2008-10-15 14:07:03 +00:00
|
|
|
Renderer::SetRenderMode(Renderer::RM_Normal);
|
|
|
|
|
2008-08-14 21:34:39 +00:00
|
|
|
Renderer::ResetGLState();
|
2008-10-15 14:07:03 +00:00
|
|
|
|
|
|
|
// switch to XFB frame buffer
|
|
|
|
Renderer::SetFramebuffer(s_xfbFrameBuffer);
|
|
|
|
glBindRenderbufferEXT(GL_RENDERBUFFER_EXT, s_xfbRenderBuffer);
|
|
|
|
glFramebufferRenderbufferEXT(GL_FRAMEBUFFER_EXT, GL_COLOR_ATTACHMENT0_EXT, GL_RENDERBUFFER_EXT, s_xfbRenderBuffer);
|
|
|
|
GL_REPORT_ERRORD();
|
|
|
|
|
|
|
|
glViewport(nXoff, nYoff, nBackbufferWidth, nBackbufferHeight);
|
|
|
|
|
|
|
|
glActiveTexture(GL_TEXTURE0);
|
|
|
|
glBindTexture(GL_TEXTURE_RECTANGLE_ARB, Renderer::GetRenderTarget());
|
|
|
|
TextureMngr::EnableTexRECT(0);
|
|
|
|
for (int i = 1; i < 8; ++i)
|
|
|
|
TextureMngr::DisableStage(i);
|
|
|
|
GL_REPORT_ERRORD();
|
|
|
|
|
|
|
|
float scaledTextureY = nBackbufferHeight - (nBackbufferHeight / yScale);
|
|
|
|
|
|
|
|
glBegin(GL_QUADS);
|
2008-10-15 23:58:13 +00:00
|
|
|
glTexCoord2f(0, nBackbufferHeight); glVertex2f(-1,-1);
|
|
|
|
glTexCoord2f(0, scaledTextureY); glVertex2f(-1,1);
|
|
|
|
glTexCoord2f(nBackbufferWidth, scaledTextureY); glVertex2f(1,1);
|
|
|
|
glTexCoord2f(nBackbufferWidth, nBackbufferHeight); glVertex2f(1,-1);
|
2008-10-15 14:07:03 +00:00
|
|
|
glEnd();
|
|
|
|
GL_REPORT_ERRORD();
|
|
|
|
|
|
|
|
int width = sourceRc.right - sourceRc.left;
|
|
|
|
int height = sourceRc.bottom - sourceRc.top;
|
|
|
|
glReadPixels(sourceRc.left, sourceRc.top, width, height, GL_RGBA, GL_UNSIGNED_BYTE, efb_buffer);
|
|
|
|
GL_REPORT_ERRORD();
|
|
|
|
|
|
|
|
Renderer::SetFramebuffer(0);
|
|
|
|
Renderer::RestoreGLState();
|
|
|
|
VertexShaderMngr::SetViewportChanged();
|
|
|
|
|
|
|
|
glBindTexture(GL_TEXTURE_RECTANGLE_ARB, 0);
|
|
|
|
TextureMngr::DisableStage(0);
|
|
|
|
|
|
|
|
Renderer::RestoreGLState();
|
|
|
|
GL_REPORT_ERRORD();
|
|
|
|
|
|
|
|
// TODO - use shader for conversion
|
|
|
|
ConvertToXFB((u32 *)xfb_in_ram, efb_buffer, dstWd, dstHt);
|
|
|
|
}
|
|
|
|
|
|
|
|
void XFB_Draw(u8 *xfb_in_ram, u32 width, u32 height, s32 yOffset)
|
|
|
|
{
|
|
|
|
OpenGL_Update(); // just updates the render window position and the backbuffer size
|
|
|
|
|
|
|
|
Renderer::SetRenderMode(Renderer::RM_Normal);
|
|
|
|
|
|
|
|
// render to the real buffer now
|
2008-10-15 23:58:13 +00:00
|
|
|
glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, 0); // switch to the backbuffer
|
2008-10-15 14:07:03 +00:00
|
|
|
glViewport(nXoff, nYoff, nBackbufferWidth, nBackbufferHeight);
|
|
|
|
|
|
|
|
Renderer::ResetGLState();
|
|
|
|
|
|
|
|
// TODO - use shader for conversion
|
|
|
|
ConvertFromXFB((u32 *)xfb_buffer, xfb_in_ram, width, height);
|
|
|
|
|
|
|
|
glActiveTexture(GL_TEXTURE0);
|
|
|
|
glBindTexture(GL_TEXTURE_RECTANGLE_ARB, xfb_texture);
|
|
|
|
glTexImage2D(GL_TEXTURE_RECTANGLE_ARB, 0, 4, width, height, 0, GL_BGRA, GL_UNSIGNED_BYTE, xfb_buffer);
|
2008-08-14 21:34:39 +00:00
|
|
|
TextureMngr::EnableTexRECT(0);
|
|
|
|
for (int i = 1; i < 8; ++i)
|
|
|
|
TextureMngr::DisableStage(i);
|
|
|
|
|
2008-10-15 14:07:03 +00:00
|
|
|
GL_REPORT_ERRORD();
|
|
|
|
|
2008-08-14 21:34:39 +00:00
|
|
|
glBegin(GL_QUADS);
|
2008-10-15 23:58:13 +00:00
|
|
|
glTexCoord2f(width, height + yOffset); glVertex2f(1,-1);
|
|
|
|
glTexCoord2f(width, 0 + yOffset); glVertex2f(1,1);
|
|
|
|
glTexCoord2f(0, 0 + yOffset); glVertex2f(-1,1);
|
|
|
|
glTexCoord2f(0, height + yOffset); glVertex2f(-1,-1);
|
2008-08-14 21:34:39 +00:00
|
|
|
glEnd();
|
|
|
|
|
2008-10-15 14:07:03 +00:00
|
|
|
glBindTexture(GL_TEXTURE_RECTANGLE_ARB, 0);
|
|
|
|
TextureMngr::DisableStage(0);
|
|
|
|
|
|
|
|
Renderer::SwapBuffers();
|
|
|
|
|
2008-08-14 21:34:39 +00:00
|
|
|
Renderer::RestoreGLState();
|
2008-10-15 14:07:03 +00:00
|
|
|
GL_REPORT_ERRORD();
|
2008-07-12 17:40:22 +00:00
|
|
|
}
|