OpenGL: Shader loading

This commit is contained in:
Jeffrey Pfau 2015-11-01 18:41:24 -08:00
parent 42a2a5737b
commit 096f5ca27c
11 changed files with 222 additions and 5 deletions

View File

@ -298,6 +298,7 @@ void GBAConfigSetOverrideFloatValue(struct GBAConfig* config, const char* key, f
void GBAConfigMap(const struct GBAConfig* config, struct GBAOptions* opts) {
_lookupCharValue(config, "bios", &opts->bios);
_lookupCharValue(config, "shader", &opts->shader);
_lookupIntValue(config, "logLevel", &opts->logLevel);
_lookupIntValue(config, "frameskip", &opts->frameskip);
_lookupIntValue(config, "volume", &opts->volume);
@ -358,6 +359,7 @@ void GBAConfigMap(const struct GBAConfig* config, struct GBAOptions* opts) {
void GBAConfigLoadDefaults(struct GBAConfig* config, const struct GBAOptions* opts) {
ConfigurationSetValue(&config->defaultsTable, 0, "bios", opts->bios);
ConfigurationSetValue(&config->defaultsTable, 0, "shader", opts->shader);
ConfigurationSetIntValue(&config->defaultsTable, 0, "skipBios", opts->skipBios);
ConfigurationSetIntValue(&config->defaultsTable, 0, "useBios", opts->useBios);
ConfigurationSetIntValue(&config->defaultsTable, 0, "logLevel", opts->logLevel);
@ -403,5 +405,7 @@ struct Configuration* GBAConfigGetOverrides(struct GBAConfig* config) {
void GBAConfigFreeOpts(struct GBAOptions* opts) {
free(opts->bios);
free(opts->shader);
opts->bios = 0;
opts->shader = 0;
}

View File

@ -38,6 +38,7 @@ struct GBAOptions {
bool lockAspectRatio;
bool resampleVideo;
bool suspendScreensaver;
char* shader;
int volume;
bool mute;

View File

@ -6,6 +6,10 @@
#include "gles2.h"
#include "gba/video.h"
#include "util/configuration.h"
#include "util/vfs.h"
#define MAX_PASSES 8
static const char* const _vertexShader =
"attribute vec4 position;\n"
@ -352,3 +356,129 @@ void GBAGLES2ShaderDetach(struct GBAGLES2Context* context) {
}
context->shaders = 0;
}
static bool _lookupIntValue(const struct Configuration* config, const char* section, const char* key, int* out) {
const char* charValue = ConfigurationGetValue(config, section, key);
if (!charValue) {
return false;
}
char* end;
unsigned long value = strtol(charValue, &end, 10);
if (*end) {
return false;
}
*out = value;
return true;
}
bool GBAGLES2ShaderLoad(struct GBAGLES2Shader** shaders, size_t* nShaders, struct GBAGLES2ShaderMetadata* metadata, struct VDir* dir) {
struct VFile* manifest = dir->openFile(dir, "manifest.ini", O_RDONLY);
if (!manifest) {
return false;
}
bool success = false;
struct Configuration description;
ConfigurationInit(&description);
if (ConfigurationReadVFile(&description, manifest)) {
int inShaders;
success = _lookupIntValue(&description, "shader", "passes", &inShaders);
if (inShaders > MAX_PASSES || inShaders < 1) {
success = false;
}
if (success) {
if (metadata) {
metadata->name = ConfigurationGetValue(&description, "shader", "name");
if (metadata->name) {
metadata->name = strdup(metadata->name);
}
metadata->author = ConfigurationGetValue(&description, "shader", "author");
if (metadata->author) {
metadata->author = strdup(metadata->author);
}
metadata->description = ConfigurationGetValue(&description, "shader", "description");
if (metadata->description) {
metadata->description = strdup(metadata->description);
}
}
struct GBAGLES2Shader* shaderBlock = malloc(sizeof(struct GBAGLES2Shader) * inShaders);
int n;
for (n = 0; n < inShaders; ++n) {
char passName[12];
snprintf(passName, sizeof(passName), "pass.%u", n);
const char* fs = ConfigurationGetValue(&description, passName, "fragmentShader");
const char* vs = ConfigurationGetValue(&description, passName, "vertexShader");
if (fs && (fs[0] == '.' || strstr(fs, PATH_SEP))) {
success = false;
break;
}
if (vs && (vs[0] == '.' || strstr(vs, PATH_SEP))) {
success = false;
break;
}
char* fssrc = 0;
char* vssrc = 0;
if (fs) {
struct VFile* fsf = dir->openFile(dir, fs, O_RDONLY);
if (!fsf) {
success = false;
break;
}
fssrc = malloc(fsf->size(fsf));
fsf->read(fsf, fssrc, fsf->size(fsf));
fsf->close(fsf);
}
if (vs) {
struct VFile* vsf = dir->openFile(dir, vs, O_RDONLY);
if (!vsf) {
success = false;
free(fssrc);
break;
}
vssrc = malloc(vsf->size(vsf));
vsf->read(vsf, vssrc, vsf->size(vsf));
vsf->close(vsf);
}
int width = 0;
int height = 0;
_lookupIntValue(&description, passName, "width", &width);
_lookupIntValue(&description, passName, "height", &height);
GBAGLES2ShaderInit(&shaderBlock[n], vssrc, fssrc, width, height, 0, 0);
int b = 0;
_lookupIntValue(&description, passName, "blend", &b);
if (b) {
shaderBlock[n].blend = b;
}
b = 0;
_lookupIntValue(&description, passName, "filter", &b);
if (b) {
shaderBlock[n].filter = b;
}
free(fssrc);
free(vssrc);
}
if (success) {
*nShaders = inShaders;
*shaders = shaderBlock;
} else {
inShaders = n;
for (n = 0; n < inShaders; ++n) {
GBAGLES2ShaderDeinit(&shaderBlock[n]);
}
}
}
}
ConfigurationDeinit(&description);
return success;
}
void GBAGLES2ShaderFree(struct GBAGLES2Shader* shaders, size_t nShaders) {
size_t n;
for (n = 0; n < nShaders; ++n) {
GBAGLES2ShaderDeinit(&shaders[n]);
size_t u;
for (u = 0; u < shaders[n].nUniforms; ++u) {
free((void*) shaders[n].uniforms[u].name);
}
}
free(shaders);
}

View File

@ -84,6 +84,12 @@ struct GBAGLES2Context {
size_t nShaders;
};
struct GBAGLES2ShaderMetadata {
const char* name;
const char* author;
const char* description;
};
void GBAGLES2ContextCreate(struct GBAGLES2Context*);
void GBAGLES2ShaderInit(struct GBAGLES2Shader*, const char* vs, const char* fs, int width, int height, struct GBAGLES2Uniform* uniforms, size_t nUniforms);
@ -91,4 +97,8 @@ void GBAGLES2ShaderDeinit(struct GBAGLES2Shader*);
void GBAGLES2ShaderAttach(struct GBAGLES2Context*, struct GBAGLES2Shader*, size_t nShaders);
void GBAGLES2ShaderDetach(struct GBAGLES2Context*);
struct VDir;
bool GBAGLES2ShaderLoad(struct GBAGLES2Shader**, size_t* nShaders, struct GBAGLES2ShaderMetadata*, struct VDir*);
void GBAGLES2ShaderFree(struct GBAGLES2Shader*, size_t nShaders);
#endif

View File

@ -11,6 +11,7 @@
#include "MessagePainter.h"
struct GBAThread;
struct VDir;
namespace QGBA {
@ -34,6 +35,7 @@ public:
bool isFiltered() const { return m_filter; }
virtual bool isDrawing() const = 0;
virtual bool supportsShaders() const = 0;
signals:
void showCursor();
@ -48,6 +50,7 @@ public slots:
virtual void lockAspectRatio(bool lock);
virtual void filter(bool filter);
virtual void framePosted(const uint32_t*) = 0;
virtual void setShaders(struct VDir*) = 0;
void showMessage(const QString& message);

View File

@ -40,6 +40,10 @@ DisplayGL::~DisplayGL() {
delete m_painter;
}
bool DisplayGL::supportsShaders() const {
return m_painter->supportsShaders();
}
void DisplayGL::startDrawing(GBAThread* thread) {
if (m_drawThread) {
return;
@ -133,6 +137,10 @@ void DisplayGL::framePosted(const uint32_t* buffer) {
}
}
void DisplayGL::setShaders(struct VDir* shaders) {
QMetaObject::invokeMethod(m_painter, "setShaders", Q_ARG(struct VDir*, shaders));
}
void DisplayGL::resizeEvent(QResizeEvent* event) {
Display::resizeEvent(event);
resizePainter();
@ -148,7 +156,10 @@ void DisplayGL::resizePainter() {
PainterGL::PainterGL(QGLWidget* parent, QGLFormat::OpenGLVersionFlags glVersion)
: m_gl(parent)
, m_active(false)
, m_started(false)
, m_context(nullptr)
, m_shaders(nullptr)
, m_nShaders(0)
, m_messagePainter(nullptr)
{
#ifdef BUILD_GL
@ -163,6 +174,7 @@ PainterGL::PainterGL(QGLWidget* parent, QGLFormat::OpenGLVersionFlags glVersion)
gl2Backend = new GBAGLES2Context;
GBAGLES2ContextCreate(gl2Backend);
m_backend = &gl2Backend->d;
m_supportsShaders = true;
} else {
#else
{
@ -170,6 +182,7 @@ PainterGL::PainterGL(QGLWidget* parent, QGLFormat::OpenGLVersionFlags glVersion)
glBackend = new GBAGLContext;
GBAGLContextCreate(glBackend);
m_backend = &glBackend->d;
m_supportsShaders = false;
}
m_backend->swap = [](VideoBackend* v) {
PainterGL* painter = static_cast<PainterGL*>(v->user);
@ -205,21 +218,21 @@ void PainterGL::setMessagePainter(MessagePainter* messagePainter) {
void PainterGL::resize(const QSize& size) {
m_size = size;
if (m_active) {
if (m_started && !m_active) {
forceDraw();
}
}
void PainterGL::lockAspectRatio(bool lock) {
m_backend->lockAspectRatio = lock;
if (m_active) {
if (m_started && !m_active) {
forceDraw();
}
}
void PainterGL::filter(bool filter) {
m_backend->filter = filter;
if (m_active) {
if (m_started && !m_active) {
forceDraw();
}
}
@ -230,8 +243,16 @@ void PainterGL::start() {
epoxy_handle_external_wglMakeCurrent();
#endif
m_backend->init(m_backend, reinterpret_cast<WHandle>(m_gl->winId()));
#if !defined(_WIN32) || defined(USE_EPOXY)
if (m_shaders) {
GBAGLES2ShaderAttach(reinterpret_cast<GBAGLES2Context*>(m_backend), m_shaders, m_nShaders);
}
#endif
m_gl->doneCurrent();
m_active = true;
m_started = true;
}
void PainterGL::draw() {
@ -262,6 +283,7 @@ void PainterGL::forceDraw() {
void PainterGL::stop() {
m_active = false;
m_started = false;
m_gl->makeCurrent();
#if defined(_WIN32) && defined(USE_EPOXY)
epoxy_handle_external_wglMakeCurrent();
@ -331,3 +353,23 @@ void PainterGL::dequeueAll() {
}
m_mutex.unlock();
}
void PainterGL::setShaders(struct VDir* dir) {
if (!supportsShaders()) {
return;
}
#if !defined(_WIN32) || defined(USE_EPOXY)
m_gl->makeCurrent();
#if defined(_WIN32) && defined(USE_EPOXY)
epoxy_handle_external_wglMakeCurrent();
#endif
if (m_shaders) {
GBAGLES2ShaderDetach(reinterpret_cast<GBAGLES2Context*>(m_backend));
}
GBAGLES2ShaderLoad(&m_shaders, &m_nShaders, nullptr, dir);
if (m_started) {
GBAGLES2ShaderAttach(reinterpret_cast<GBAGLES2Context*>(m_backend), m_shaders, m_nShaders);
}
m_gl->doneCurrent();
#endif
}

View File

@ -22,6 +22,7 @@
struct GBAThread;
struct VideoBackend;
struct GBAGLES2Shader;
namespace QGBA {
@ -44,6 +45,7 @@ public:
~DisplayGL();
bool isDrawing() const override { return m_isDrawing; }
bool supportsShaders() const override;
public slots:
void startDrawing(GBAThread* context) override;
@ -54,6 +56,7 @@ public slots:
void lockAspectRatio(bool lock) override;
void filter(bool filter) override;
void framePosted(const uint32_t*) override;
void setShaders(struct VDir*) override;
protected:
virtual void paintEvent(QPaintEvent*) override {}
@ -80,6 +83,8 @@ public:
void setMessagePainter(MessagePainter*);
void enqueue(const uint32_t* backing);
bool supportsShaders() const { return m_supportsShaders; }
public slots:
void forceDraw();
void draw();
@ -91,6 +96,8 @@ public slots:
void lockAspectRatio(bool lock);
void filter(bool filter);
void setShaders(struct VDir*);
private:
void performDraw();
void dequeue();
@ -102,7 +109,11 @@ private:
QMutex m_mutex;
QGLWidget* m_gl;
bool m_active;
bool m_started;
GBAThread* m_context;
bool m_supportsShaders;
GBAGLES2Shader* m_shaders;
size_t m_nShaders;
VideoBackend* m_backend;
QSize m_size;
MessagePainter* m_messagePainter;

View File

@ -22,6 +22,7 @@ public:
DisplayQt(QWidget* parent = nullptr);
bool isDrawing() const override { return m_isDrawing; }
bool supportsShaders() const override { return 0; }
public slots:
void startDrawing(GBAThread* context) override;
@ -32,6 +33,7 @@ public slots:
void lockAspectRatio(bool lock) override;
void filter(bool filter) override;
void framePosted(const uint32_t*) override;
void setShaders(struct VDir*) override {}
protected:
virtual void paintEvent(QPaintEvent*) override;

View File

@ -39,6 +39,7 @@
extern "C" {
#include "platform/commandline.h"
#include "util/vfs.h"
}
using namespace QGBA;
@ -223,6 +224,14 @@ void Window::loadConfig() {
enterFullScreen();
}
if (opts->shader) {
struct VDir* shader = VDirOpen(opts->shader);
if (shader) {
m_display->setShaders(shader);
shader->close(shader);
}
}
m_inputController.setScreensaverSuspendable(opts->suspendScreensaver);
m_mruFiles = m_config->getMRU();

View File

@ -136,12 +136,16 @@ static char* _vfgets(char* stream, int size, void* user) {
}
bool ConfigurationRead(struct Configuration* configuration, const char* path) {
HashTableClear(&configuration->root);
HashTableClear(&configuration->sections);
struct VFile* vf = VFileOpen(path, O_RDONLY);
if (!vf) {
return false;
}
return ConfigurationReadVFile(configuration, vf);
}
bool ConfigurationReadVFile(struct Configuration* configuration, struct VFile* vf) {
HashTableClear(&configuration->root);
HashTableClear(&configuration->sections);
return ini_parse_stream(_vfgets, vf, _iniRead, configuration) == 0;
}

View File

@ -29,6 +29,7 @@ const char* ConfigurationGetValue(const struct Configuration*, const char* secti
void ConfigurationClearValue(struct Configuration*, const char* section, const char* key);
bool ConfigurationRead(struct Configuration*, const char* path);
bool ConfigurationReadVFile(struct Configuration*, struct VFile* vf);
bool ConfigurationWrite(const struct Configuration*, const char* path);
bool ConfigurationWriteSection(const struct Configuration*, const char* path, const char* section);