From 830694ee0a6a1637fccf4d91d94e1e2f86395f28 Mon Sep 17 00:00:00 2001 From: Dominic Masters Date: Sun, 19 Nov 2023 01:16:41 -0600 Subject: [PATCH] glClear is causing stack corrpution --- lib/Catch2 | 2 +- lib/SDL | 2 +- lib/freetype | 2 +- lib/glfw | 2 +- lib/glm | 2 +- lib/openal-soft | 2 +- .../display/IRenderableComponent.hpp | 26 ++- .../material/SimpleTexturedMaterial.cpp | 25 ++- .../material/SimpleTexturedMaterial.hpp | 18 +- src/dawn/display/ITexture.hpp | 109 +++++++++ src/dawn/display/RenderPipeline.cpp | 24 +- src/dawn/display/RenderTarget.hpp | 4 +- src/dawn/scene/item/SceneItemTransform.cpp | 6 + src/dawn/scene/item/SceneItemTransform.hpp | 7 + src/dawnhelloworld/scene/HelloWorldScene.cpp | 14 +- .../display/BackBufferRenderTarget.cpp | 3 +- src/dawnopengl/display/CMakeLists.txt | 1 + src/dawnopengl/display/Texture.cpp | 208 ++++++++++++++++++ src/dawnopengl/display/Texture.hpp | 43 ++++ src/dawnopengl/display/shader/Shader.hpp | 15 ++ .../display/shader/SimpleTexturedShader.cpp | 23 +- .../display/shader/SimpleTexturedShader.hpp | 2 + 22 files changed, 503 insertions(+), 37 deletions(-) create mode 100644 src/dawn/display/ITexture.hpp create mode 100644 src/dawnopengl/display/Texture.cpp create mode 100644 src/dawnopengl/display/Texture.hpp diff --git a/lib/Catch2 b/lib/Catch2 index a8cf3e67..32d9ae24 160000 --- a/lib/Catch2 +++ b/lib/Catch2 @@ -1 +1 @@ -Subproject commit a8cf3e671047da6e7bf6c3c0d9a2178afeed1b54 +Subproject commit 32d9ae24bc4a0fed9a136a73e116b4ded0ac84b7 diff --git a/lib/SDL b/lib/SDL index ac6b32bb..4722269f 160000 --- a/lib/SDL +++ b/lib/SDL @@ -1 +1 @@ -Subproject commit ac6b32bb0280f225bd117a0c1db554dc3ee8f3f2 +Subproject commit 4722269fb62315640cbdbd8916cfc937ee3603d4 diff --git a/lib/freetype b/lib/freetype index dc519d06..e969289f 160000 --- a/lib/freetype +++ b/lib/freetype @@ -1 +1 @@ -Subproject commit dc519d06ea67b3f75b2dffe238369b76158e7c8a +Subproject commit e969289f88a586c40678133abcf999362aefe565 diff --git a/lib/glfw b/lib/glfw index 3eaf1255..a87acd8c 160000 --- a/lib/glfw +++ b/lib/glfw @@ -1 +1 @@ -Subproject commit 3eaf1255b29fdf5c2895856c7be7d7185ef2b241 +Subproject commit a87acd8c1fac78011013bebb537926e18fe49f46 diff --git a/lib/glm b/lib/glm index 47585fde..586a4023 160000 --- a/lib/glm +++ b/lib/glm @@ -1 +1 @@ -Subproject commit 47585fde0c49fa77a2bf2fb1d2ead06999fd4b6e +Subproject commit 586a402397dd35d66d7a079049856d1e2cbab300 diff --git a/lib/openal-soft b/lib/openal-soft index 6420bc86..d6d572df 160000 --- a/lib/openal-soft +++ b/lib/openal-soft @@ -1 +1 @@ -Subproject commit 6420bc86cd972fd544a74bdbbc0e882d284046be +Subproject commit d6d572df66eb2b5beeb3da7f3b11a328500c33c3 diff --git a/src/dawn/component/display/IRenderableComponent.hpp b/src/dawn/component/display/IRenderableComponent.hpp index 641a74fa..863ece3b 100644 --- a/src/dawn/component/display/IRenderableComponent.hpp +++ b/src/dawn/component/display/IRenderableComponent.hpp @@ -12,7 +12,7 @@ #include "component/display/MeshRenderer.hpp" namespace Dawn { - struct IRenderPass { + class IRenderPass { public: std::shared_ptr mesh; @@ -43,14 +43,17 @@ namespace Dawn { }; template - struct RenderPass : public IRenderPass { + class RenderPass : public IRenderPass { private: std::shared_ptr shader; - const D data; + const std::unordered_map< + shadertexturebinding_t, std::shared_ptr + > &textures; std::shared_ptr mesh; const enum MeshDrawMode drawMode; const int32_t indiceStart; const int32_t indiceCount; + const D data; public: /** @@ -66,12 +69,16 @@ namespace Dawn { RenderPass( SceneComponent &self, const D d, + const std::unordered_map< + shadertexturebinding_t, std::shared_ptr + > textures, const std::shared_ptr mesh, const enum MeshDrawMode drawMode, const int32_t indiceStart, const int32_t indiceCount ) : data(d), + textures(textures), mesh(mesh), drawMode(drawMode), indiceStart(indiceStart), @@ -91,6 +98,7 @@ namespace Dawn { } void bind() override { + std::cout << "textures: " << textures.size() << "\n"; shader->bind(); } @@ -99,6 +107,10 @@ namespace Dawn { } void upload() override { + for(auto &pair : textures) { + assertNotNull(pair.second, "Texture cannot be null!"); + pair.second->bind(pair.first); + } shader->upload(); } @@ -139,17 +151,21 @@ namespace Dawn { * @return Created render pass. */ template - std::shared_ptr createRenderPass( + std::shared_ptr createRenderPass( SceneComponent &self, const D data, + const std::unordered_map< + shadertexturebinding_t, std::shared_ptr + > textures = {}, const std::shared_ptr mesh = nullptr, const enum MeshDrawMode drawMode = MeshDrawMode::TRIANGLES, int32_t indiceStart = 0, int32_t indiceCount = -1 ) { - return std::make_shared>( + return std::make_shared>( self, data, + textures, mesh, drawMode, indiceStart, diff --git a/src/dawn/component/display/material/SimpleTexturedMaterial.cpp b/src/dawn/component/display/material/SimpleTexturedMaterial.cpp index 9063181a..4adceb94 100644 --- a/src/dawn/component/display/material/SimpleTexturedMaterial.cpp +++ b/src/dawn/component/display/material/SimpleTexturedMaterial.cpp @@ -11,7 +11,16 @@ struct Color SimpleTexturedMaterial::getColor() { return this->data.color; } -void SimpleTexturedMaterial::setColor(struct Color color) { +std::shared_ptr SimpleTexturedMaterial::getTexture() { + return this->texture; +} + +void SimpleTexturedMaterial::setTexture(std::shared_ptr texture) { + this->texture = texture; +} + + +void SimpleTexturedMaterial::setColor(const struct Color color) { this->data.color = color; } @@ -22,11 +31,23 @@ std::vector> SimpleTexturedMaterial::getPasses( this->data.projection = ctx.camera->getProjection(); this->data.view = ctx.camera->getItem()->getWorldTransform(); this->data.color = COLOR_RED; + auto textures = std::unordered_map< + shadertexturebinding_t, std::shared_ptr + >(); + + if(this->texture) { + this->data.hasTexture = true; + this->data.texture = 0; + textures[this->data.texture] = this->texture; + } else { + this->data.hasTexture = false; + } return { createRenderPass( *this, - data + data, + textures ) }; } diff --git a/src/dawn/component/display/material/SimpleTexturedMaterial.hpp b/src/dawn/component/display/material/SimpleTexturedMaterial.hpp index 60aeca47..dc91d179 100644 --- a/src/dawn/component/display/material/SimpleTexturedMaterial.hpp +++ b/src/dawn/component/display/material/SimpleTexturedMaterial.hpp @@ -6,11 +6,13 @@ #pragma once #include "component/display/material/Material.hpp" #include "display/shader/SimpleTexturedShader.hpp" +#include "display/Texture.hpp" namespace Dawn { class SimpleTexturedMaterial : public Material { private: struct SimpleTexturedShaderData data; + std::shared_ptr texture; public: /** @@ -18,12 +20,26 @@ namespace Dawn { */ struct Color getColor(); + /** + * Returns the texture of this material. + * + * @return The texture of this material. + */ + std::shared_ptr getTexture(); + + /** + * Sets the texture of this material. + * + * @param texture The texture to set. + */ + void setTexture(std::shared_ptr texture); + /** * Sets the color of this material. * * @param color The color to set. */ - void setColor(struct Color color); + void setColor(const struct Color color); std::vector> getPasses( struct RenderPassContext &ctx diff --git a/src/dawn/display/ITexture.hpp b/src/dawn/display/ITexture.hpp new file mode 100644 index 00000000..8dbbe462 --- /dev/null +++ b/src/dawn/display/ITexture.hpp @@ -0,0 +1,109 @@ +// Copyright (c) 2023 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include "display/Color.hpp" + +namespace Dawn { + enum TextureFormat { + TEXTURE_FORMAT_R = 1, + TEXTURE_FORMAT_RG = 2, + TEXTURE_FORMAT_RGB = 3, + TEXTURE_FORMAT_RGBA = 4 + }; + + enum TextureWrapMode { + TEXTURE_WRAP_MODE_REPEAT = 0, + TEXTURE_WRAP_MODE_MIRRORED_REPEAT = 1, + TEXTURE_WRAP_MODE_CLAMP_TO_EDGE = 2, + TEXTURE_WRAP_MODE_CLAMP_TO_BORDER = 3 + }; + + enum TextureFilterMode { + TEXTURE_FILTER_MODE_NEAREST = 0, + TEXTURE_FILTER_MODE_LINEAR = 1 + }; + + enum TextureDataFormat { + TEXTURE_DATA_FORMAT_UNSIGNED_BYTE = 0, + TEXTURE_DATA_FORMAT_FLOAT = 1 + }; + + class ITexture { + public: + enum TextureWrapMode wrapModeX = TEXTURE_WRAP_MODE_REPEAT; + enum TextureWrapMode wrapModeY = TEXTURE_WRAP_MODE_REPEAT; + enum TextureFilterMode filterModeMin = TEXTURE_FILTER_MODE_NEAREST; + enum TextureFilterMode filterModeMag = TEXTURE_FILTER_MODE_NEAREST; + enum TextureFilterMode mipMapFilterModeMin = TEXTURE_FILTER_MODE_NEAREST; + enum TextureFilterMode mipMapFilterModeMag = TEXTURE_FILTER_MODE_NEAREST; + + /** + * Returns the width of the texture. + * + * @return Width of the texture. + */ + virtual int32_t getWidth() = 0; + + /** + * Returns the height of the texture. + * + * @return Height of the texture. + */ + virtual int32_t getHeight() = 0; + + /** + * Initializes a texture. + * + * @param width Width of the texture (in pixels). + * @param height Height of the texture (in pixels). + * @param format Data format of the texture to use. + * @param dataFormat Data format of the texture to use. + */ + virtual void setSize( + const int32_t width, + const int32_t height, + const enum TextureFormat format, + const enum TextureDataFormat dataFormat + ) = 0; + + /** + * Returns true only when the texture has been loaded, sized and put on + * the gpu for rendering. + * + * @return True if ready, otherwise false. + */ + // virtual bool_t isReady() = 0; + + /** + * Buffer pixel data onto the GPU. Pixel buffering is rather costly so + * avoid doing this too often. + * + * @param pixels Array of pixels you're trying to buffer. + */ + virtual void buffer(const struct ColorU8 pixels[]) = 0; + + /** + * Buffer pixel data onto the GPU. Pixel buffering is rather costly so + * avoid doing this too often. + * + * @param pixels Array of pixels you're trying to buffer. + */ + virtual void buffer(const uint8_t pixels[]) = 0; + + /** + * Binds the texture to the given slot (for use by the shaders). + * + * @param slot Slot to bind to. + */ + virtual void bind(const uint8_t slot) = 0; + + /** + * Disposes of the texture. + */ + virtual ~ITexture() { + } + }; +} \ No newline at end of file diff --git a/src/dawn/display/RenderPipeline.cpp b/src/dawn/display/RenderPipeline.cpp index 2b1df000..958b937d 100644 --- a/src/dawn/display/RenderPipeline.cpp +++ b/src/dawn/display/RenderPipeline.cpp @@ -68,6 +68,7 @@ void RenderPipeline::renderSceneCamera( assertNotNull(game, "Game cannot be null"); assertNotNull(scene, "Scene cannot be null"); assertNotNull(camera, "Camera cannot be null"); + assertNotNull(renderTarget, "RenderTarget cannot be null"); struct RenderPassContext ctx = { game, @@ -77,14 +78,15 @@ void RenderPipeline::renderSceneCamera( }; // Get list of renderables - std::vector> renderPasses; + std::vector> renderPasses; auto renderables = scene->findComponents(); for(auto renderable : renderables) { auto rp = renderable->getPasses(ctx); renderPasses.insert(renderPasses.end(), rp.begin(), rp.end()); } - // Sort the render passes + auto rp = renderPasses[0]; + rp->bind(); // TODO: Make clearing the buffers editable! renderTarget->bind(); @@ -92,17 +94,15 @@ void RenderPipeline::renderSceneCamera( RENDER_TARGET_CLEAR_COLOR | RENDER_TARGET_CLEAR_DEPTH ); + + rp->bind(); - std::for_each( - renderPasses.begin(), - renderPasses.end(), - [&](std::shared_ptr pass) { - pass->bind(); - pass->setData(); - pass->upload(); - pass->draw(); - } - ); + for(auto renderPass : renderPasses) { + renderPass->bind(); + renderPass->setData(); + renderPass->upload(); + renderPass->draw(); + } } RenderPipeline::~RenderPipeline() { diff --git a/src/dawn/display/RenderTarget.hpp b/src/dawn/display/RenderTarget.hpp index 59bdce25..9755e6ba 100644 --- a/src/dawn/display/RenderTarget.hpp +++ b/src/dawn/display/RenderTarget.hpp @@ -6,8 +6,8 @@ #pragma once #include "event/Event.hpp" -#define RENDER_TARGET_CLEAR_COLOR 1 << 0 -#define RENDER_TARGET_CLEAR_DEPTH 1 << 1 +#define RENDER_TARGET_CLEAR_COLOR (1 << 0) +#define RENDER_TARGET_CLEAR_DEPTH (1 << 1) namespace Dawn { class RenderTarget { diff --git a/src/dawn/scene/item/SceneItemTransform.cpp b/src/dawn/scene/item/SceneItemTransform.cpp index 7f0e4b37..f234a65f 100644 --- a/src/dawn/scene/item/SceneItemTransform.cpp +++ b/src/dawn/scene/item/SceneItemTransform.cpp @@ -123,6 +123,12 @@ glm::quat SceneItemTransform::getLocalRotation() { return this->localRotation; } +void SceneItemTransform::setLocalTransform(const glm::mat4 transform) { + this->transformLocal = transform; + this->updateLocalValuesFromLocalTransform(); + this->updateWorldTransformFromLocalTransform(); +} + void SceneItemTransform::setWorldTransform(const glm::mat4 transform) { this->transformWorld = transform; this->updateLocalTransformFromWorldTransform(); diff --git a/src/dawn/scene/item/SceneItemTransform.hpp b/src/dawn/scene/item/SceneItemTransform.hpp index b8151c7c..c360b5bb 100644 --- a/src/dawn/scene/item/SceneItemTransform.hpp +++ b/src/dawn/scene/item/SceneItemTransform.hpp @@ -106,6 +106,13 @@ namespace Dawn { * @return Local rotation of this item. */ glm::quat getLocalRotation(); + + /** + * Sets the transform of this item within local space (relative to parent) + * + * @param transform Transform of this item in local space. + */ + void setLocalTransform(const glm::mat4 transform); /** * Sets the transform of this item within world space (relative to scene diff --git a/src/dawnhelloworld/scene/HelloWorldScene.cpp b/src/dawnhelloworld/scene/HelloWorldScene.cpp index 86612b36..6b7702b8 100644 --- a/src/dawnhelloworld/scene/HelloWorldScene.cpp +++ b/src/dawnhelloworld/scene/HelloWorldScene.cpp @@ -27,11 +27,11 @@ void Dawn::helloWorldScene(Scene &s) { auto cubeMeshRenderer = cubeItem->addComponent(); cubeMeshRenderer->mesh = cubeMesh; auto cubeMaterial = cubeItem->addComponent(); - addSimpleComponent(cubeItem, [](auto &cmp, auto &events) { - events.push_back(cmp.getScene()->onUnpausedUpdate.listen([&](float_t delta) { - auto item = cmp.getItem(); - item->setLocalRotation(item->getLocalRotation() * glm::quat(glm::vec3(1, 1, 0) * delta)); - // item->setLocalScale(item->getLocalScale() + glm::vec3(1, 1, 1) * delta); - })); - }); + // addSimpleComponent(cubeItem, [](auto &cmp, auto &events) { + // events.push_back(cmp.getScene()->onUnpausedUpdate.listen([&](float_t delta) { + // auto item = cmp.getItem(); + // item->setLocalRotation(item->getLocalRotation() * glm::quat(glm::vec3(1, 1, 0) * delta)); + // // item->setLocalScale(item->getLocalScale() + glm::vec3(1, 1, 1) * delta); + // })); + // }); } \ No newline at end of file diff --git a/src/dawnopengl/display/BackBufferRenderTarget.cpp b/src/dawnopengl/display/BackBufferRenderTarget.cpp index 17628d81..149a9e3d 100644 --- a/src/dawnopengl/display/BackBufferRenderTarget.cpp +++ b/src/dawnopengl/display/BackBufferRenderTarget.cpp @@ -45,13 +45,14 @@ void BackBufferRenderTarget::setClearColor(const struct Color color) { } void BackBufferRenderTarget::clear(const int32_t clearFlags) { - glClearColor(clearColor.r, clearColor.g, clearColor.b, clearColor.a); + glClearColor(clearColor.r, clearColor.g, clearColor.b, 1.0f); assertNoGLError(); GLbitfield mask = 0; if(Flag::isOn(clearFlags, RENDER_TARGET_CLEAR_COLOR)) { mask |= GL_COLOR_BUFFER_BIT; } + if(Flag::isOn(clearFlags, RENDER_TARGET_CLEAR_DEPTH)) { mask |= GL_DEPTH_BUFFER_BIT; } diff --git a/src/dawnopengl/display/CMakeLists.txt b/src/dawnopengl/display/CMakeLists.txt index 73ca87f6..f24fd602 100644 --- a/src/dawnopengl/display/CMakeLists.txt +++ b/src/dawnopengl/display/CMakeLists.txt @@ -7,6 +7,7 @@ target_sources(${DAWN_TARGET_NAME} PRIVATE BackBufferRenderTarget.cpp + Texture.cpp ) # Subdirs diff --git a/src/dawnopengl/display/Texture.cpp b/src/dawnopengl/display/Texture.cpp new file mode 100644 index 00000000..ca367684 --- /dev/null +++ b/src/dawnopengl/display/Texture.cpp @@ -0,0 +1,208 @@ +// Copyright (c) 2022 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#include "assert/assert.hpp" +#include "assert/assertgl.hpp" +#include "Texture.hpp" + +using namespace Dawn; + +void Texture::bind(const uint8_t slot) { + assertTrue(this->id != -1, "Texture is not ready!"); + glActiveTexture(GL_TEXTURE0 + slot); + assertNoGLError(); + glBindTexture(GL_TEXTURE_2D, this->id); + assertNoGLError(); + this->updateTextureProperties(); +} + +int32_t Texture::getWidth() { + return this->width; +} + +int32_t Texture::getHeight() { + return this->height; +} + +void Texture::setSize( + const int32_t width, + const int32_t height, + const enum TextureFormat format, + const enum TextureDataFormat dataFormat +) { + if(this->id != -1) { + glDeleteTextures(1, &this->id); + assertNoGLError(); + this->id = -1; + } + + int32_t maxSize; + glGetIntegerv(GL_MAX_TEXTURE_SIZE, &maxSize); + assertTrue(width > 0 && width <= maxSize, "Width is out of bounds!"); + assertTrue(height > 0 && height <= maxSize, "Height is out of bounds!"); + + this->width = width; + this->height = height; + this->format = format; + this->dataFormat = dataFormat; + + glGenTextures(1, &this->id); + assertNoGLError(); + if(this->id <= 0) assertUnreachable("Texture generation failed!"); + + // Initialize the texture to blank + glActiveTexture(GL_TEXTURE0); + assertNoGLError(); + this->bufferRaw(NULL); +} + +// bool_t Texture::isReady() { +// return this->id != -1; +// } + +void Texture::updateTextureProperties() { + auto setWrapMode = [&](GLenum axis, enum TextureWrapMode wm) { + switch(wm) { + case TEXTURE_WRAP_MODE_REPEAT: + glTexParameteri(GL_TEXTURE_2D, axis, GL_REPEAT); + break; + + case TEXTURE_WRAP_MODE_MIRRORED_REPEAT: + glTexParameteri(GL_TEXTURE_2D, axis, GL_MIRRORED_REPEAT); + break; + + case TEXTURE_WRAP_MODE_CLAMP_TO_EDGE: + glTexParameteri(GL_TEXTURE_2D, axis, GL_CLAMP_TO_EDGE); + break; + + case TEXTURE_WRAP_MODE_CLAMP_TO_BORDER: + glTexParameteri(GL_TEXTURE_2D, axis, GL_CLAMP_TO_BORDER); + break; + + default: + assertUnreachable("Unknown wrap mode!"); + } + assertNoGLError(); + }; + + setWrapMode(GL_TEXTURE_WRAP_S, this->wrapModeX); + setWrapMode(GL_TEXTURE_WRAP_T, this->wrapModeY); + + auto setFilterMode = [&]( + GLenum minMag, + enum TextureFilterMode filter, + enum TextureFilterMode mapFilterMode + ) { + switch(filter) { + case TEXTURE_FILTER_MODE_NEAREST: { + glTexParameteri(GL_TEXTURE_2D, minMag, GL_NEAREST); + break; + } + + case TEXTURE_FILTER_MODE_LINEAR: { + glTexParameteri(GL_TEXTURE_2D, minMag, GL_LINEAR); + break; + } + + default: { + assertUnreachable("Unknown filter mode!"); + } + } + assertNoGLError(); + }; + + setFilterMode( + GL_TEXTURE_MIN_FILTER, this->filterModeMin, this->mipMapFilterModeMin + ); + setFilterMode( + GL_TEXTURE_MAG_FILTER, this->filterModeMag, this->mipMapFilterModeMag + ); +} + +void Texture::bufferRaw(const void *data) { + assertTrue(this->id != -1, "Texture is not ready!"); + + GLenum format; + switch(this->format) { + case TEXTURE_FORMAT_R: + format = GL_RED; + break; + + case TEXTURE_FORMAT_RG: + format = GL_RG; + break; + + case TEXTURE_FORMAT_RGB: + format = GL_RGB; + break; + + case TEXTURE_FORMAT_RGBA: + format = GL_RGBA; + break; + + default: + assertUnreachable("Unknown texture format!"); + } + + GLenum dataFormat; + switch(this->dataFormat) { + case TEXTURE_DATA_FORMAT_UNSIGNED_BYTE: + dataFormat = GL_UNSIGNED_BYTE; + break; + + case TEXTURE_DATA_FORMAT_FLOAT: + dataFormat = GL_FLOAT; + break; + + default: + assertUnreachable("Unknown texture data format!"); + } + + glBindTexture(GL_TEXTURE_2D, this->id); + assertNoGLError(); + glTexImage2D( + GL_TEXTURE_2D, 0, format, + this->width, this->height, + 0, format, dataFormat, data + ); + assertNoGLError(); + glGenerateMipmap(GL_TEXTURE_2D); + assertNoGLError(); +} + +void Texture::buffer(const struct ColorU8 pixels[]) { + assertTrue( + this->dataFormat == TEXTURE_DATA_FORMAT_UNSIGNED_BYTE, + "Texture data format must be unsigned byte!" + ); + this->bufferRaw((void*)pixels); +} + +void Texture::buffer(const struct Color pixels[]) { + assertTrue( + this->dataFormat == TEXTURE_DATA_FORMAT_FLOAT, + "Texture data format must be float!" + ); + assertTrue( + this->format == TEXTURE_FORMAT_RGBA, + "Texture format must be RGBA!" + ); + this->bufferRaw((void*)pixels); +} + +void Texture::buffer(const uint8_t pixels[]) { + assertTrue( + this->dataFormat == TEXTURE_DATA_FORMAT_UNSIGNED_BYTE, + "Texture data format must be unsigned byte!" + ); + this->bufferRaw((void*)pixels); +} + +Texture::~Texture() { + if(this->id != -1) { + glDeleteTextures(1, &this->id); + assertNoGLError(); + } +} \ No newline at end of file diff --git a/src/dawnopengl/display/Texture.hpp b/src/dawnopengl/display/Texture.hpp new file mode 100644 index 00000000..83b41352 --- /dev/null +++ b/src/dawnopengl/display/Texture.hpp @@ -0,0 +1,43 @@ +// Copyright (c) 2022 Dominic Masters +// +// This software is released under the MIT License. +// https://opensource.org/licenses/MIT + +#pragma once +#include "dawnopengl.hpp" +#include "display/ITexture.hpp" + +namespace Dawn { + class TextureRenderTarget; + + typedef GLuint textureslot_t; + + class Texture : public ITexture { + private: + int32_t width = -1; + int32_t height = -1; + GLuint id = -1; + enum TextureFormat format; + enum TextureDataFormat dataFormat; + + void updateTextureProperties(); + void bufferRaw(const void *data); + + public: + int32_t getWidth() override; + int32_t getHeight() override; + void setSize( + const int32_t width, + const int32_t height, + const enum TextureFormat format, + const enum TextureDataFormat dataForat + ) override; + // bool_t isReady() override; + void buffer(const struct ColorU8 pixels[]) override; + void buffer(const struct Color pixels[]); + void buffer(const uint8_t pixels[]) override; + void bind(const uint8_t slot) override; + + ~Texture(); + }; +} \ No newline at end of file diff --git a/src/dawnopengl/display/shader/Shader.hpp b/src/dawnopengl/display/shader/Shader.hpp index b75708ab..73234b0d 100644 --- a/src/dawnopengl/display/shader/Shader.hpp +++ b/src/dawnopengl/display/shader/Shader.hpp @@ -9,8 +9,11 @@ #include "assert/assert.hpp" #include "assert/assertgl.hpp" #include "display/Color.hpp" +#include "display/Texture.hpp" namespace Dawn { + typedef GLuint shadertexturebinding_t; + enum ShaderOpenGLVariant { GLSL_330_CORE }; @@ -151,6 +154,18 @@ namespace Dawn { break; } + case ShaderParameterType::BOOLEAN: { + auto boolean = (bool_t *)value; + glUniform1i(param.location, *boolean ? 1 : 0); + break; + } + + case ShaderParameterType::TEXTURE: { + textureslot_t texture = *((textureslot_t*)value); + glUniform1i(param.location, texture); + break; + } + default: { assertUnreachable("Unsupported ShaderParameterType"); } diff --git a/src/dawnopengl/display/shader/SimpleTexturedShader.cpp b/src/dawnopengl/display/shader/SimpleTexturedShader.cpp index c07114e2..efd52dfc 100644 --- a/src/dawnopengl/display/shader/SimpleTexturedShader.cpp +++ b/src/dawnopengl/display/shader/SimpleTexturedShader.cpp @@ -23,11 +23,14 @@ void SimpleTexturedShader::getStages( ShaderStageType::VERTEX, "#version 330 core\n" "layout (location = 0) in vec3 aPos;\n" + "layout (location = 1) in vec2 aTexCoord;\n" "uniform mat4 u_Projection;\n" "uniform mat4 u_View;\n" "uniform mat4 u_Model;\n" + "out vec2 o_TextCoord;\n" "void main() {\n" "gl_Position = u_Projection * u_View * u_Model * vec4(aPos, 1.0);\n" + "o_TextCoord = vec2(aTexCoord.x, aTexCoord.y);\n" "}" ); @@ -37,8 +40,14 @@ void SimpleTexturedShader::getStages( "in vec2 o_TextCoord;\n" "out vec4 o_Color;\n" "uniform vec4 u_Color;\n" + "uniform bool u_HasTexture;\n" + "uniform sampler2D u_Texture;\n" "void main() {\n" - "o_Color = u_Color;" + "if(u_HasTexture) {\n" + "o_Color = texture(u_Texture, o_TextCoord) * u_Color;\n" + "} else {\n" + "o_Color = u_Color;" + "}\n" "}\n" ); break; @@ -75,4 +84,16 @@ void SimpleTexturedShader::getStages( &rel->color, ShaderParameterType::COLOR )); + + parameters.push_back(ShaderOpenGLParameter( + "u_HasTexture", + &rel->hasTexture, + ShaderParameterType::BOOLEAN + )); + + parameters.push_back(ShaderOpenGLParameter( + "u_Texture", + &rel->texture, + ShaderParameterType::TEXTURE + )); } \ No newline at end of file diff --git a/src/dawnopengl/display/shader/SimpleTexturedShader.hpp b/src/dawnopengl/display/shader/SimpleTexturedShader.hpp index 9ed868a8..5b30df56 100644 --- a/src/dawnopengl/display/shader/SimpleTexturedShader.hpp +++ b/src/dawnopengl/display/shader/SimpleTexturedShader.hpp @@ -12,6 +12,8 @@ namespace Dawn { glm::mat4 view; glm::mat4 model; struct Color color = COLOR_WHITE; + bool hasTexture = false; + shadertexturebinding_t texture = 0; }; class SimpleTexturedShader : public Shader {