2018-06-13 19:15:16 +00:00
|
|
|
|
/*
|
|
|
|
|
* glsurface.cpp
|
|
|
|
|
* A 32-bit rendering surface that can quickly blit 8-bit paletted buffers implemented in OpenGL.
|
|
|
|
|
*
|
|
|
|
|
* Copyright <EFBFBD> 2018, Alex Dawson. All rights reserved.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#include "glsurface.h"
|
|
|
|
|
#include "glad/glad.h"
|
|
|
|
|
|
|
|
|
|
#include "baselayer.h"
|
|
|
|
|
#include "build.h"
|
2019-09-16 19:08:42 +00:00
|
|
|
|
#include "../../glbackend/glbackend.h"
|
2018-06-13 19:15:16 +00:00
|
|
|
|
|
|
|
|
|
static void* buffer;
|
2019-09-18 18:44:21 +00:00
|
|
|
|
static FHardwareTexture* bufferTexture;
|
2018-06-13 19:15:16 +00:00
|
|
|
|
static vec2_t bufferRes;
|
|
|
|
|
|
2019-09-18 18:44:21 +00:00
|
|
|
|
static FHardwareTexture* paletteTexture;
|
2018-06-13 19:15:16 +00:00
|
|
|
|
|
|
|
|
|
static GLuint shaderProgramID = 0;
|
|
|
|
|
static GLint texSamplerLoc = -1;
|
|
|
|
|
static GLint paletteSamplerLoc = -1;
|
|
|
|
|
|
|
|
|
|
static GLuint compileShader(GLenum shaderType, const char* const source)
|
|
|
|
|
{
|
|
|
|
|
GLuint shaderID = glCreateShader(shaderType);
|
|
|
|
|
if (shaderID == 0)
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
const char* const sources[1] = {source};
|
|
|
|
|
glShaderSource(shaderID,
|
|
|
|
|
1,
|
|
|
|
|
sources,
|
|
|
|
|
NULL);
|
|
|
|
|
glCompileShader(shaderID);
|
|
|
|
|
|
|
|
|
|
GLint compileStatus;
|
|
|
|
|
glGetShaderiv(shaderID, GL_COMPILE_STATUS, &compileStatus);
|
|
|
|
|
if (!compileStatus)
|
|
|
|
|
{
|
|
|
|
|
GLint logLength;
|
|
|
|
|
glGetShaderiv(shaderID, GL_INFO_LOG_LENGTH, &logLength);
|
2018-07-30 23:01:48 +00:00
|
|
|
|
OSD_Printf("Compile Status: %u\n", compileStatus);
|
2018-06-13 19:15:16 +00:00
|
|
|
|
if (logLength > 0)
|
|
|
|
|
{
|
2018-10-25 23:28:56 +00:00
|
|
|
|
char *infoLog = (char*) Xmalloc(logLength);
|
2018-06-13 19:15:16 +00:00
|
|
|
|
glGetShaderInfoLog(shaderID, logLength, &logLength, infoLog);
|
|
|
|
|
OSD_Printf("Log:\n%s\n", infoLog);
|
2019-06-25 11:29:08 +00:00
|
|
|
|
Xfree(infoLog);
|
2018-06-13 19:15:16 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return shaderID;
|
|
|
|
|
}
|
|
|
|
|
|
2018-07-14 21:36:44 +00:00
|
|
|
|
bool glsurface_initialize(vec2_t bufferResolution)
|
2018-06-13 19:15:16 +00:00
|
|
|
|
{
|
|
|
|
|
if (buffer)
|
|
|
|
|
glsurface_destroy();
|
|
|
|
|
|
|
|
|
|
glPixelStorei(GL_UNPACK_ALIGNMENT, 1);
|
|
|
|
|
|
2018-07-14 21:36:44 +00:00
|
|
|
|
bufferRes = bufferResolution;
|
2018-06-24 00:55:17 +00:00
|
|
|
|
buffer = Xaligned_alloc(16, bufferRes.x * bufferRes.y);
|
2018-06-13 19:15:16 +00:00
|
|
|
|
|
2019-09-18 18:44:21 +00:00
|
|
|
|
bufferTexture = GLInterface.NewTexture();
|
|
|
|
|
bufferTexture->CreateTexture(bufferRes.x, bufferRes.y, true, false);
|
2018-06-13 19:15:16 +00:00
|
|
|
|
|
2019-09-18 18:44:21 +00:00
|
|
|
|
GLInterface.Init();
|
2018-06-25 14:53:46 +00:00
|
|
|
|
glsurface_setPalette(curpalettefaded);
|
|
|
|
|
|
2018-06-13 19:15:16 +00:00
|
|
|
|
const char* const VERTEX_SHADER_CODE =
|
|
|
|
|
"#version 110\n\
|
|
|
|
|
\n\
|
|
|
|
|
attribute vec4 i_vertPos;\n\
|
|
|
|
|
attribute vec2 i_texCoord;\n\
|
|
|
|
|
\n\
|
|
|
|
|
varying vec2 v_texCoord;\n\
|
|
|
|
|
\n\
|
|
|
|
|
void main()\n\
|
|
|
|
|
{\n\
|
|
|
|
|
gl_Position = i_vertPos;\n\
|
|
|
|
|
v_texCoord = i_texCoord;\n\
|
|
|
|
|
}\n";
|
|
|
|
|
const char* const FRAGMENT_SHADER_CODE =
|
|
|
|
|
"#version 110\n\
|
|
|
|
|
\n\
|
|
|
|
|
//s_texture points to an indexed color texture\n\
|
|
|
|
|
uniform sampler2D s_texture;\n\
|
|
|
|
|
//s_palette is the palette texture\n\
|
|
|
|
|
uniform sampler2D s_palette;\n\
|
|
|
|
|
\n\
|
|
|
|
|
varying vec2 v_texCoord;\n\
|
|
|
|
|
\n\
|
|
|
|
|
const float c_paletteScale = 255.0/256.0;\n\
|
|
|
|
|
const float c_paletteOffset = 0.5/256.0;\n\
|
|
|
|
|
\n\
|
|
|
|
|
void main()\n\
|
|
|
|
|
{\n\
|
|
|
|
|
vec4 color = texture2D(s_texture, v_texCoord.xy);\n\
|
|
|
|
|
color.r = c_paletteOffset + c_paletteScale*color.r;\n\
|
2018-06-25 14:53:46 +00:00
|
|
|
|
color.rgb = texture2D(s_palette, color.rg).rgb;\n\
|
2018-06-13 19:15:16 +00:00
|
|
|
|
\n\
|
|
|
|
|
// DEBUG \n\
|
|
|
|
|
//color = texture2D(s_palette, v_texCoord.xy);\n\
|
|
|
|
|
//color = texture2D(s_texture, v_texCoord.xy);\n\
|
|
|
|
|
\n\
|
|
|
|
|
gl_FragColor = color;\n\
|
|
|
|
|
}\n";
|
|
|
|
|
|
|
|
|
|
shaderProgramID = glCreateProgram();
|
|
|
|
|
GLuint vertexShaderID = compileShader(GL_VERTEX_SHADER, VERTEX_SHADER_CODE);
|
|
|
|
|
GLuint fragmentShaderID = compileShader(GL_FRAGMENT_SHADER, FRAGMENT_SHADER_CODE);
|
|
|
|
|
glAttachShader(shaderProgramID, vertexShaderID);
|
|
|
|
|
glAttachShader(shaderProgramID, fragmentShaderID);
|
|
|
|
|
glBindAttribLocation(shaderProgramID, 0, "i_vertPos");
|
|
|
|
|
glBindAttribLocation(shaderProgramID, 1, "i_texCoord");
|
|
|
|
|
glLinkProgram(shaderProgramID);
|
|
|
|
|
glDetachShader(shaderProgramID, vertexShaderID);
|
|
|
|
|
glDeleteShader(vertexShaderID);
|
|
|
|
|
glDetachShader(shaderProgramID, fragmentShaderID);
|
|
|
|
|
glDeleteShader(fragmentShaderID);
|
|
|
|
|
glUseProgram(shaderProgramID);
|
|
|
|
|
|
|
|
|
|
texSamplerLoc = glGetUniformLocation(shaderProgramID, "s_texture");
|
|
|
|
|
paletteSamplerLoc = glGetUniformLocation(shaderProgramID, "s_palette");
|
|
|
|
|
|
|
|
|
|
glUniform1i(texSamplerLoc, 0);
|
|
|
|
|
glUniform1i(paletteSamplerLoc, 1);
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void glsurface_destroy()
|
|
|
|
|
{
|
|
|
|
|
if (!buffer)
|
|
|
|
|
return;
|
|
|
|
|
|
2018-06-24 00:55:17 +00:00
|
|
|
|
ALIGNED_FREE_AND_NULL(buffer);
|
2018-06-13 19:15:16 +00:00
|
|
|
|
|
2019-09-18 18:44:21 +00:00
|
|
|
|
delete bufferTexture;
|
|
|
|
|
bufferTexture = nullptr;
|
|
|
|
|
delete paletteTexture;
|
|
|
|
|
paletteTexture = nullptr;
|
2018-06-13 19:15:16 +00:00
|
|
|
|
|
2018-07-14 21:36:34 +00:00
|
|
|
|
glUseProgram(0);
|
2018-06-13 19:15:16 +00:00
|
|
|
|
glDeleteProgram(shaderProgramID);
|
|
|
|
|
shaderProgramID = 0;
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-25 14:53:46 +00:00
|
|
|
|
void glsurface_setPalette(void* pPalette)
|
2018-06-13 19:15:16 +00:00
|
|
|
|
{
|
|
|
|
|
if (!buffer)
|
|
|
|
|
return;
|
|
|
|
|
if (!pPalette)
|
|
|
|
|
return;
|
|
|
|
|
|
2019-09-18 18:44:21 +00:00
|
|
|
|
if (!paletteTexture)
|
|
|
|
|
{
|
|
|
|
|
paletteTexture = GLInterface.NewTexture();
|
|
|
|
|
paletteTexture->CreateTexture(256, 1, false, false);
|
|
|
|
|
}
|
|
|
|
|
paletteTexture->LoadTexture(palette);
|
|
|
|
|
GLInterface.BindTexture(1, paletteTexture, Sampler2DNoFilter);
|
2018-06-13 19:15:16 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void* glsurface_getBuffer()
|
|
|
|
|
{
|
|
|
|
|
return buffer;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
vec2_t glsurface_getBufferResolution()
|
|
|
|
|
{
|
|
|
|
|
return bufferRes;
|
|
|
|
|
}
|
|
|
|
|
|
2018-06-25 14:53:46 +00:00
|
|
|
|
void glsurface_blitBuffer()
|
2018-06-13 19:15:16 +00:00
|
|
|
|
{
|
|
|
|
|
if (!buffer)
|
|
|
|
|
return;
|
|
|
|
|
|
2019-09-18 18:44:21 +00:00
|
|
|
|
bufferTexture->LoadTexture((uint8_t*)buffer);
|
|
|
|
|
GLInterface.BindTexture(0, bufferTexture, Sampler2DNoFilter);
|
2018-06-13 19:15:16 +00:00
|
|
|
|
|
2019-09-16 19:08:42 +00:00
|
|
|
|
auto data = GLInterface.AllocVertices(4);
|
|
|
|
|
auto vt = data.second;
|
|
|
|
|
|
|
|
|
|
vt[0].Set(-1.0f, 1.0f, 0.0f, 0.0f, 0.0f); //top-left
|
|
|
|
|
vt[1].Set(-1.0f, -1.0f, 0.0f, 0.0f, 1.0f); //bottom-left
|
|
|
|
|
vt[2].Set(1.0f, 1.0f, 0.0f, 1.0f, 0.0f); //top-right
|
|
|
|
|
vt[3].Set(1.0f, -1.0f, 0.0f, 1.0f, 1.0f); //bottom-right
|
|
|
|
|
GLInterface.Draw(DT_TRIANGLE_STRIP, data.first, 4);
|
2018-06-13 19:15:16 +00:00
|
|
|
|
}
|