From 096f5ca27c52d1d0aa14b5641791c9b24ac88f3a Mon Sep 17 00:00:00 2001 From: Jeffrey Pfau Date: Sun, 1 Nov 2015 18:41:24 -0800 Subject: [PATCH] OpenGL: Shader loading --- src/gba/context/config.c | 4 ++ src/gba/context/config.h | 1 + src/platform/opengl/gles2.c | 130 ++++++++++++++++++++++++++++++++++ src/platform/opengl/gles2.h | 10 +++ src/platform/qt/Display.h | 3 + src/platform/qt/DisplayGL.cpp | 48 ++++++++++++- src/platform/qt/DisplayGL.h | 11 +++ src/platform/qt/DisplayQt.h | 2 + src/platform/qt/Window.cpp | 9 +++ src/util/configuration.c | 8 ++- src/util/configuration.h | 1 + 11 files changed, 222 insertions(+), 5 deletions(-) diff --git a/src/gba/context/config.c b/src/gba/context/config.c index ee4eccb09..c0c87ea67 100644 --- a/src/gba/context/config.c +++ b/src/gba/context/config.c @@ -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; } diff --git a/src/gba/context/config.h b/src/gba/context/config.h index 659a02fa2..f1ed84bab 100644 --- a/src/gba/context/config.h +++ b/src/gba/context/config.h @@ -38,6 +38,7 @@ struct GBAOptions { bool lockAspectRatio; bool resampleVideo; bool suspendScreensaver; + char* shader; int volume; bool mute; diff --git a/src/platform/opengl/gles2.c b/src/platform/opengl/gles2.c index 865e812ea..32cb29efa 100644 --- a/src/platform/opengl/gles2.c +++ b/src/platform/opengl/gles2.c @@ -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); +} diff --git a/src/platform/opengl/gles2.h b/src/platform/opengl/gles2.h index 2d17ee438..8f90efae7 100644 --- a/src/platform/opengl/gles2.h +++ b/src/platform/opengl/gles2.h @@ -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 diff --git a/src/platform/qt/Display.h b/src/platform/qt/Display.h index 7b3605ab2..e6b8650f5 100644 --- a/src/platform/qt/Display.h +++ b/src/platform/qt/Display.h @@ -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); diff --git a/src/platform/qt/DisplayGL.cpp b/src/platform/qt/DisplayGL.cpp index f2bfbb38f..b8e9610f0 100644 --- a/src/platform/qt/DisplayGL.cpp +++ b/src/platform/qt/DisplayGL.cpp @@ -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(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(m_gl->winId())); + +#if !defined(_WIN32) || defined(USE_EPOXY) + if (m_shaders) { + GBAGLES2ShaderAttach(reinterpret_cast(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(m_backend)); + } + GBAGLES2ShaderLoad(&m_shaders, &m_nShaders, nullptr, dir); + if (m_started) { + GBAGLES2ShaderAttach(reinterpret_cast(m_backend), m_shaders, m_nShaders); + } + m_gl->doneCurrent(); +#endif +} diff --git a/src/platform/qt/DisplayGL.h b/src/platform/qt/DisplayGL.h index 1b7226c27..bdcca5ec2 100644 --- a/src/platform/qt/DisplayGL.h +++ b/src/platform/qt/DisplayGL.h @@ -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; diff --git a/src/platform/qt/DisplayQt.h b/src/platform/qt/DisplayQt.h index 1d5acc5b4..843c50c6f 100644 --- a/src/platform/qt/DisplayQt.h +++ b/src/platform/qt/DisplayQt.h @@ -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; diff --git a/src/platform/qt/Window.cpp b/src/platform/qt/Window.cpp index c514cbd7b..c09301ace 100644 --- a/src/platform/qt/Window.cpp +++ b/src/platform/qt/Window.cpp @@ -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(); diff --git a/src/util/configuration.c b/src/util/configuration.c index a2c8bdaaa..c4c180bdb 100644 --- a/src/util/configuration.c +++ b/src/util/configuration.c @@ -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; } diff --git a/src/util/configuration.h b/src/util/configuration.h index 14a0e7014..8f85883eb 100644 --- a/src/util/configuration.h +++ b/src/util/configuration.h @@ -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);