This commit is contained in:
2024-10-04 16:41:01 -05:00
parent be27408cf4
commit c1345218a3
31 changed files with 1652 additions and 14 deletions

View File

@ -13,3 +13,11 @@ add_subdirectory(glad)
# GLFW # GLFW
FetchContent_Declare(glfw URL https://github.com/glfw/glfw/releases/download/3.4/glfw-3.4.zip) FetchContent_Declare(glfw URL https://github.com/glfw/glfw/releases/download/3.4/glfw-3.4.zip)
FetchContent_MakeAvailable(glfw) FetchContent_MakeAvailable(glfw)
# GLM
FetchContent_Declare(
cglm
GIT_REPOSITORY https://github.com/recp/cglm
GIT_TAG 1796cc5ce298235b615dc7a4750b8c3ba56a05dd
)
FetchContent_MakeAvailable(cglm)

View File

@ -16,6 +16,7 @@ if(DAWN_TARGET STREQUAL "linux-x64-terminal")
add_subdirectory(dawntermlinux) add_subdirectory(dawntermlinux)
elseif(DAWN_TARGET STREQUAL "linux-x64-glfw") elseif(DAWN_TARGET STREQUAL "linux-x64-glfw")
add_subdirectory(dawnlinux) add_subdirectory(dawnlinux)
add_subdirectory(dawnterm)
add_subdirectory(dawnglfw) add_subdirectory(dawnglfw)
add_subdirectory(dawnopengl) add_subdirectory(dawnopengl)
else() else()

View File

@ -97,6 +97,19 @@ void assertTrueImplement(
(value & flag) == flag, __VA_ARGS__ \ (value & flag) == flag, __VA_ARGS__ \
) )
/**
* Asserts that the given string is not null, and has a length that is less
* the maximum buffer size, including the NULL terminator.
*
* @param str String to check.
* @param bufferSize Maximum buffer size.
* @param message Message (sprintf format) to send to the logger.
* @param args Optional TParam args for the sprintf message to accept.
*/
#define assertStringValid(str, bufferSize, ...) assertTrue( \
((str != NULL) && ((strlen(str)+1) < bufferSize)), __VA_ARGS__ \
)
/** /**
* Asserts that the current code is deprecated and should not be used anymore. * Asserts that the current code is deprecated and should not be used anymore.
* @deprecated * @deprecated

View File

@ -18,4 +18,5 @@
typedef bool bool_t; typedef bool bool_t;
typedef char char_t; typedef char char_t;
typedef unsigned char uchar_t; typedef unsigned char uchar_t;
typedef uint_fast32_t flag_t;

37
src/dawn/util/math.h Normal file
View File

@ -0,0 +1,37 @@
/**
* Copyright (c) 2023 Dominic Masters
*
* This software is released under the MIT License.
* https://opensource.org/licenses/MIT
*/
#pragma once
#include "dawn.h"
/**
* Returns the minimum of two values.
*
* @param a First value.
* @param b Second value.
* @return The minimum of the two values.
*/
#define mathMin(a, b) ((a) < (b) ? (a) : (b))
/**
* Returns the maximum of two values.
*
* @param a First value.
* @param b Second value.
* @return The maximum of the two values.
*/
#define mathMax(a, b) ((a) > (b) ? (a) : (b))
/**
* Clamps a value between two values.
*
* @param x Value to clamp.
* @param a Minimum value.
* @param b Maximum value.
* @return The clamped value.
*/
#define mathClamp(x, a, b) mathMin(mathMax(x, a), b)

View File

@ -7,6 +7,7 @@
target_link_libraries(${DAWN_TARGET_NAME} target_link_libraries(${DAWN_TARGET_NAME}
PUBLIC PUBLIC
glfw glfw
glad
) )
# Includes # Includes

View File

@ -69,6 +69,9 @@ int32_t dawnGlfwStart() {
((float_t)fbWidth) / ((float_t)windowWidth) ((float_t)fbWidth) / ((float_t)windowWidth)
); );
// Init game
gameInit();
// Set the input binds // Set the input binds
dawnGlfwInputBind(INPUT_BIND_UP, GLFW_KEY_W); dawnGlfwInputBind(INPUT_BIND_UP, GLFW_KEY_W);
dawnGlfwInputBind(INPUT_BIND_UP, GLFW_KEY_UP); dawnGlfwInputBind(INPUT_BIND_UP, GLFW_KEY_UP);
@ -85,7 +88,6 @@ int32_t dawnGlfwStart() {
dawnGlfwInputBind(INPUT_BIND_CANCEL, GLFW_KEY_Q); dawnGlfwInputBind(INPUT_BIND_CANCEL, GLFW_KEY_Q);
dawnGlfwInputBind(INPUT_BIND_CANCEL, GLFW_KEY_BACKSPACE); dawnGlfwInputBind(INPUT_BIND_CANCEL, GLFW_KEY_BACKSPACE);
// Event callbacks // Event callbacks
glfwSetFramebufferSizeCallback(DAWN_GLFW.window, &dawnGlfwOnResize); glfwSetFramebufferSizeCallback(DAWN_GLFW.window, &dawnGlfwOnResize);

View File

@ -0,0 +1,10 @@
/**
* Copyright (c) 2024 Dominic Masters
*
* This software is released under the MIT License.
* https://opensource.org/licenses/MIT
*/
#pragma once
#include "dawn.h"
#include <glad/glad.h>

View File

@ -8,6 +8,6 @@
#include "linuxhost.h" #include "linuxhost.h"
#include "dawnglfw.h" #include "dawnglfw.h"
int32_t linuxHostBeginUpdating() { int32_t linuxHostHandoff() {
return dawnGlfwStart(); return dawnGlfwStart();
} }

View File

@ -11,4 +11,4 @@
/** /**
* Linux Host Compatibility * Linux Host Compatibility
*/ */
int32_t linuxHostBeginUpdating(); int32_t linuxHostHandoff();

View File

@ -6,12 +6,8 @@
*/ */
#include "main.h" #include "main.h"
#include "rpg/world/map.h"
#include "game.h"
#include "linuxhost.h" #include "linuxhost.h"
int32_t main() { int32_t main() {
gameInit(); return linuxHostHandoff();
return linuxHostBeginUpdating();
} }

View File

@ -7,7 +7,7 @@
target_link_libraries(${DAWN_TARGET_NAME} target_link_libraries(${DAWN_TARGET_NAME}
PUBLIC PUBLIC
font8x8 font8x8
glad cglm
) )
# Includes # Includes

View File

@ -6,5 +6,10 @@
*/ */
#pragma once #pragma once
#include "dawn.h" #include "dawnopenglimpl.h"
#include <glad/glad.h> #include <cglm/cglm.h>
typedef vec2 vec2_t;
typedef vec3 vec3_t;
typedef vec4 vec4_t;
typedef mat4 mat4_t;

View File

@ -4,10 +4,16 @@
# https://opensource.org/licenses/MIT # https://opensource.org/licenses/MIT
# Subdirs # Subdirs
add_subdirectory(shaders)
# Sources # Sources
target_sources(${DAWN_TARGET_NAME} target_sources(${DAWN_TARGET_NAME}
PRIVATE PRIVATE
display.c display.c
backbuffer.c backbuffer.c
shader.c
mesh.c
font.c
texture.c
textureframebuffer.c
) )

View File

@ -0,0 +1,59 @@
/**
* Copyright (c) 2024 Dominic Masters
*
* This software is released under the MIT License.
* https://opensource.org/licenses/MIT
*/
#pragma once
#include "dawn.h"
typedef float_t color3f_t[3];
typedef float_t color4f_t[4];
typedef color4f_t color_t;
#define COLOR3F(r,g,b) ((color3f_t){ r, g, b })
#define COLOR3F_RED COLOR3F(1, 0, 0)
#define COLOR3F_GREEN COLOR3F(0, 1, 0)
#define COLOR3F_BLUE COLOR3F(0, 0, 1)
#define COLOR3F_BLACK COLOR3F(0, 0, 0)
#define COLOR3F_WHITE COLOR3F(1, 1, 1)
#define COLOR3F_MAGENTA COLOR3F(1, 0, 1)
#define COLOR3F_YELLOW COLOR3F(1, 1, 0)
#define COLOR3F_CYAN COLOR3F(0, 1, 1)
#define COLOR3F_GRAY COLOR3F(0.5f, 0.5f, 0.5f)
#define COLOR3F_DARKGRAY COLOR3F(0.25f, 0.25f, 0.25f)
#define COLOR3F_LIGHTGRAY COLOR3F(0.75f, 0.75f, 0.75f)
#define COLOR3F_ORANGE COLOR3F(1, 0.5f, 0)
#define COLOR3F_PURPLE COLOR3F(0.5f, 0, 1)
#define COLOR3F_PINK COLOR3F(1, 0, 0.5f)
#define COLOR3F_BROWN COLOR3F(0.5f, 0.25f, 0)
#define COLOR3F_GOLD COLOR3F(1, 0.75f, 0)
#define COLOR3F_SILVER COLOR3F(0.75f, 0.75f, 0.75f)
#define COLOR3F_BRONZE COLOR3F(0.75f, 0.5f, 0.25f)
#define COLOR3F_CORNFLOWERBLUE COLOR3F(0.4f, 0.6f, 0.9f)
#define COLOR4F(r,g,b,a) ((color4f_t){ r, g, b, a })
#define COLOR4F_RED COLOR4F(1, 0, 0, 1)
#define COLOR4F_GREEN COLOR4F(0, 1, 0, 1)
#define COLOR4F_BLUE COLOR4F(0, 0, 1, 1)
#define COLOR4F_BLACK COLOR4F(0, 0, 0, 1)
#define COLOR4F_WHITE COLOR4F(1, 1, 1, 1)
#define COLOR4F_MAGENTA COLOR4F(1, 0, 1, 1)
#define COLOR4F_YELLOW COLOR4F(1, 1, 0, 1)
#define COLOR4F_CYAN COLOR4F(0, 1, 1, 1)
#define COLOR4F_GRAY COLOR4F(0.5f, 0.5f, 0.5f, 1)
#define COLOR4F_DARKGRAY COLOR4F(0.25f, 0.25f, 0.25f, 1)
#define COLOR4F_LIGHTGRAY COLOR4F(0.75f, 0.75f, 0.75f, 1)
#define COLOR4F_ORANGE COLOR4F(1, 0.5f, 0, 1)
#define COLOR4F_PURPLE COLOR4F(0.5f, 0, 1, 1)
#define COLOR4F_PINK COLOR4F(1, 0, 0.5f, 1)
#define COLOR4F_BROWN COLOR4F(0.5f, 0.25f, 0, 1)
#define COLOR4F_GOLD COLOR4F(1, 0.75f, 0, 1)
#define COLOR4F_SILVER COLOR4F(0.75f, 0.75f, 0.75f, 1)
#define COLOR4F_BRONZE COLOR4F(0.75f, 0.5f, 0.25f, 1)
#define COLOR4F_CORNFLOWERBLUE COLOR4F(0.4f, 0.6f, 0.9f, 1)
#define COLOR4F_TRANSPARENT_BLACK COLOR4F(0, 0, 0, 0)
#define COLOR4F_TRANSPARENT_WHITE COLOR4F(1, 1, 1, 0)
#define COLOR4F_TRANSPARENT COLOR4F_TRANSPARENT_BLACK

View File

@ -7,12 +7,30 @@
#include "display/display.h" #include "display/display.h"
#include "assert/assert.h" #include "assert/assert.h"
#include "display/shaders/simpletexturedshader.h"
#include "display/backbuffer.h"
#include "display/font.h"
#include "display/mesh.h"
#include "display/frame.h"
void displayInit() { void displayInit() {
simpleTexturedShaderInit();
fontInit();
frameInit();
} }
void displayUpdate() { void displayUpdate() {
frameUpdate();
backBufferBind();
backBufferClear();
simpleTexturedShaderBind();
} }
void displayDispose() { void displayDispose() {
simpleTexturedShaderDispose();
fontDispose();
} }

View File

@ -0,0 +1,30 @@
/**
* Copyright (c) 2024 Dominic Masters
*
* This software is released under the MIT License.
* https://opensource.org/licenses/MIT
*/
#include "font.h"
#include <font8x8.h>
font_t FONT;
void fontInit() {
memset(&FONT, 0, sizeof(FONT));
// font is uint8_t[8] * count of chars
size_t charCount = sizeof(font8x8_basic) / 8;
int32_t totalPixels = charCount * 8 * 8;
textureInit(
&FONT.texture,
totalPixels / 8, 8,
TEXTURE_FORMAT_R,
TEXTURE_DATA_FORMAT_UNSIGNED_BYTE
);
}
void fontDispose() {
textureDispose(&FONT.texture);
}

View File

@ -0,0 +1,25 @@
/**
* Copyright (c) 2024 Dominic Masters
*
* This software is released under the MIT License.
* https://opensource.org/licenses/MIT
*/
#pragma once
#include "texture.h"
typedef struct {
texture_t texture;
} font_t;
extern font_t FONT;
/**
* Initializes the render font.
*/
void fontInit();
/**
* Disposes of the render font.
*/
void fontDispose();

View File

@ -0,0 +1,309 @@
/**
* Copyright (c) 2023 Dominic Masters
*
* This software is released under the MIT License.
* https://opensource.org/licenses/MIT
*/
#include "mesh.h"
#include "assert/assert.h"
#include "assert/assertgl.h"
int32_t MESH_ACTIVE_COUNT = 0;
void meshInit(mesh_t *mesh) {
assertNotNull(mesh, "meshInit: Mesh cannot be null.");
mesh->vertexBuffer = -1;
mesh->indexBuffer = -1;
mesh->vertexArray = -1;
mesh->verticeCount = -1;
mesh->indiceCount = -1;
MESH_ACTIVE_COUNT++;
}
void meshCreateBuffers(
mesh_t *mesh,
const int32_t verticeCount,
const int32_t indiceCount
) {
assertNotNull(mesh, "meshCreateBuffers: Mesh cannot be null.");
assertTrue(
verticeCount > 0,
"meshCreateBuffers: Vertice count must be greater than 0."
);
assertTrue(
indiceCount > 0,
"meshCreateBuffers: Indice count must be greater than 0."
);
// Dispose old buffers (if present).
meshDestroyBuffers(mesh);
mesh->verticeCount = verticeCount;
mesh->indiceCount = indiceCount;
size_t sizePos = sizeof(vec3_t) * verticeCount;
size_t sizeInds = sizeof(meshindice_t) * indiceCount;
size_t sizeCoords = sizeof(vec2_t) * verticeCount;
// Generate vertex array, I don't think I need to do this tbh.
glGenVertexArrays(1, &mesh->vertexArray);
assertNoGLError();
glBindVertexArray(mesh->vertexArray);
assertNoGLError();
// Create some buffers, one for the vertex data, one for the indices
GLuint buffer[2];
glGenBuffers(2, buffer);
assertNoGLError();
mesh->vertexBuffer = buffer[0];
assertTrue(
mesh->vertexBuffer >= 0,
"meshCreateBuffers: Failed to create vertex buffer"
);
mesh->indexBuffer = buffer[1];
assertTrue(
mesh->indexBuffer >= 0,
"meshCreateBuffers: Failed to create index buffer"
);
// Buffer an empty set of data then buffer each component
glBindBuffer(GL_ARRAY_BUFFER, mesh->vertexBuffer);
assertNoGLError();
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, mesh->indexBuffer);
assertNoGLError();
glBufferData(GL_ARRAY_BUFFER, sizePos+sizeCoords, 0, GL_DYNAMIC_DRAW);
assertNoGLError();
glBufferData(GL_ELEMENT_ARRAY_BUFFER, sizeInds, 0, GL_DYNAMIC_DRAW);
assertNoGLError();
// Setup the attrib pointers
size_t offset = 0;
glVertexAttribPointer(
0, sizeof(vec3_t) / sizeof(float_t),
GL_FLOAT, GL_FALSE,
0, (void *)offset
);
assertNoGLError();
glEnableVertexAttribArray(0);
assertNoGLError();
offset += sizePos;
glVertexAttribPointer(
1, sizeof(vec2_t) / sizeof(float_t),
GL_FLOAT, GL_FALSE,
0, (void *)offset
);
assertNoGLError();
glEnableVertexAttribArray(1);
assertNoGLError();
}
void meshDestroyBuffers(mesh_t *mesh) {
assertNotNull(mesh, "meshDestroyBuffers: Mesh cannot be null.");
glBindBuffer(GL_ARRAY_BUFFER, 0);
assertNoGLError();
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, 0);
assertNoGLError();
if(mesh->vertexBuffer != -1) {
glDeleteBuffers(1, &mesh->vertexBuffer);
assertNoGLError();
mesh->vertexBuffer = -1;
mesh->verticeCount = -1;
}
if(mesh->indexBuffer != -1) {
glDeleteBuffers(1, &mesh->indexBuffer);
assertNoGLError();
mesh->indexBuffer = -1;
mesh->indiceCount = -1;
}
if(mesh->vertexArray != -1) {
glDeleteVertexArrays(1, &mesh->vertexArray);
assertNoGLError();
mesh->vertexArray = -1;
}
}
void meshBufferPositions(
const mesh_t *mesh,
const vec3_t *positions,
const size_t position,
const size_t count
) {
assertNotNull(mesh, "meshBufferPositions: Mesh cannot be null");
assertNotNull(
positions,
"meshBufferPositions: Positions cannot be null"
);
assertTrue(
position >= 0 && position < mesh->verticeCount,
"meshBufferPositions: Position must be within range"
);
assertTrue(
position+count <= mesh->verticeCount,
"meshBufferPositions: Position + count must be within range"
);
assertTrue(
count > 0,
"meshBufferPositions: Count must be greater than zero"
);
size_t offsetPositions = sizeof(vec3_t) * position;
glBindBuffer(GL_ARRAY_BUFFER, mesh->vertexBuffer);
assertNoGLError();
glBufferSubData(
GL_ARRAY_BUFFER,
offsetPositions,
sizeof(vec3_t) * count,
(void*)positions
);
assertNoGLError();
}
void meshBufferCoordinates(
const mesh_t *mesh,
const vec2_t *coordinates,
const size_t position,
const size_t count
) {
assertNotNull(mesh, "meshBufferCoordinates: Mesh cannot be null");
assertNotNull(
coordinates,
"meshBufferCoordinates: Coordinates cannot be null"
);
assertTrue(
position >= 0 && position < mesh->verticeCount,
"meshBufferCoordinates: Position must be within range"
);
assertTrue(
position+count <= mesh->verticeCount,
"meshBufferCoordinates: Position + count must be within range"
);
assertTrue(
count > 0,
"meshBufferCoordinates: Count must be greater than zero"
);
size_t offsetCoordinates = (
(sizeof(vec3_t) * mesh->verticeCount) +
(sizeof(vec2_t) * position)
);
glBindBuffer(GL_ARRAY_BUFFER, mesh->vertexBuffer);
assertNoGLError();
glBufferSubData(
GL_ARRAY_BUFFER,
offsetCoordinates,
sizeof(vec2_t) * count,
(void*)coordinates
);
assertNoGLError();
}
void meshBufferIndices(
const mesh_t *mesh,
const meshindice_t *indices,
const size_t position,
const size_t count
) {
assertNotNull(
mesh,
"meshBufferIndices: Mesh cannot be null"
);
assertNotNull(
indices,
"meshBufferIndices: Indices cannot be null"
);
assertTrue(
position >= 0 && position < mesh->indiceCount,
"meshBufferIndices: Position must be within range"
);
assertTrue(
position+count <= mesh->indiceCount,
"meshBufferIndices: Position + count must be within range"
);
assertTrue(
count > 0,
"meshBufferIndices: Count must be greater than zero"
);
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, mesh->indexBuffer);
assertNoGLError();
glBufferSubData(
GL_ELEMENT_ARRAY_BUFFER,
sizeof(meshindice_t) * position,
sizeof(meshindice_t) * count,
(void*)indices
);
assertNoGLError();
}
void meshDraw(
const mesh_t *mesh,
const meshdrawmode_t mode,
const int32_t indiceStart,
const int32_t indiceCount
) {
assertNotNull(mesh, "meshDraw: Mesh cannot be null");
// Don't draw empty meshes.
if(indiceCount == 0 || mesh->vertexBuffer == -1 || mesh->indexBuffer == -1) {
return;
}
assertTrue(
indiceCount > 0 || indiceCount == -1,
"meshDraw: Invalid indiceCount"
);
// Re-Bind the buffers
assertNoGLError();
glBindVertexArray(mesh->vertexArray);
assertNoGLError();
glBindBuffer(GL_ARRAY_BUFFER, mesh->vertexBuffer);
assertNoGLError();
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, mesh->indexBuffer);
assertNoGLError();
// Re-Calculate the attrib pointers. Will need to revist this later.
size_t offset = 0;
glVertexAttribPointer(
0, sizeof(vec3_t) / sizeof(float_t),
GL_FLOAT, GL_FALSE,
0, (void*)offset
);
assertNoGLError();
glEnableVertexAttribArray(0);
assertNoGLError();
offset += sizeof(vec3_t) * mesh->verticeCount;
glVertexAttribPointer(
1, sizeof(vec2_t) / sizeof(float_t),
GL_FLOAT, GL_FALSE,
0, (void*)offset
);
assertNoGLError();
glEnableVertexAttribArray(1);
assertNoGLError();
// Render the elements.
glDrawElements(
mode,
indiceCount == -1 ? (mesh->indiceCount - indiceStart) : indiceCount,
GL_UNSIGNED_INT,
(void*)(sizeof(meshindice_t) * indiceStart)
);
assertNoGLError();
}
void meshDispose(mesh_t *mesh) {
meshDestroyBuffers(mesh);
MESH_ACTIVE_COUNT--;
}

View File

@ -0,0 +1,134 @@
/**
* Copyright (c) 2023 Dominic Masters
*
* This software is released under the MIT License.
* https://opensource.org/licenses/MIT
*/
#pragma once
#include "dawn.h"
#include "dawnopengl.h"
extern int32_t MESH_ACTIVE_COUNT;
/** Indice that references a specific vertice */
typedef int32_t meshindice_t;
typedef enum {
/** Draw the mesh as a list of points */
MESH_DRAW_MODE_POINTS = GL_POINTS,
/** Draw the mesh as a list of lines */
MESH_DRAW_MODE_LINES = GL_LINES,
/** Draw the mesh as a list of triangles */
MESH_DRAW_MODE_TRIANGLES = GL_TRIANGLES
} meshdrawmode_t;
typedef struct {
/** Pointer to the vertex buffer on the GPU */
GLuint vertexBuffer;
/** Pointer to the index buffer on the GPU */
GLuint indexBuffer;
/** Pointer to the vertex buffer on the GPU */
GLuint vertexArray;
/** How many vertices are in the mesh */
int32_t verticeCount;
/** How many indices are in the mesh */
int32_t indiceCount;
} mesh_t;
/**
* Initializes a mesh to a default state. Mesh's must be initialized and
* destroyed.
*
* @param mesh Mesh to initialize.
*/
void meshInit(mesh_t *mesh);
/**
* Create a new set of buffers for the mesh to use.
*
* @param mesh Mesh to create buffers for.
* @param verticeCount How many Vertices will this buffer support.
* @param indiceCount How many Indices will this buffer support.
*/
void meshCreateBuffers(
mesh_t *mesh,
const int32_t verticeCount,
const int32_t indiceCount
);
/**
* Cleanup the buffers on a given mesh. This is useful if you intend to expand
* the count of vertices your mesh supports.
*
* @param mesh Mesh to destroy buffers for.
*/
void meshDestroyBuffers(mesh_t *mesh);
/**
* Write vertice coordinates to the mesh.
*
* @param mesh Mesh to write to.
* @param positions Array of coordinates to write.
* @param position Position, within the buffer, to write to.
* @param count How many coordinates are in the array.
*/
void meshBufferPositions(
const mesh_t *mesh,
const vec3_t *positions,
const size_t position,
const size_t count
);
/**
* Write texture coordinates to the mesh.
*
* @param mesh Mesh to write to.
* @param coordinates Array of coordinates to write.
* @param position Position, within the buffer, to write to.
* @param count How many coordinates are in the array.
*/
void meshBufferCoordinates(
const mesh_t *mesh,
const vec2_t *coordinates,
const size_t position,
const size_t count
);
/**
* Write indices to the mesh.
*
* @param mesh Mesh to write to.
* @param indices Array of indices to write.
* @param position Position, within the buffer, to write to.
* @param count How many indices are in the array.
*/
void meshBufferIndices(
const mesh_t *mesh,
const meshindice_t *indices,
const size_t position,
const size_t count
);
/**
* Draw a mesh. Meshes are drawn by their indices.
*
* @param mesh Mesh to draw.
* @param mode Which drawing mode to use to draw the mesh.
* @param indiceStart Start indice (index) to draw.
* @param indiceCount Count of indices to draw. Use -1 to draw all.
*/
void meshDraw(
const mesh_t *mesh,
const meshdrawmode_t mode,
const int32_t indiceStart,
const int32_t indiceCount
);
/**
* Cleanup a mesh. This will destroy the buffers and free the mesh.
*
* @param mesh Mesh to cleanup.
*/
void meshDispose(mesh_t *mesh);

View File

@ -0,0 +1,208 @@
/**
* Copyright (c) 2023 Dominic Masters
*
* This software is released under the MIT License.
* https://opensource.org/licenses/MIT
*/
#include "shader.h"
#include "assert/assert.h"
#include "assert/assertgl.h"
int32_t SHADER_ACTIVE_COUNT;
void shaderInit(shader_t *shader) {
assertNotNull(shader, "shaderInit: shader cannot be null.");
shader->shaderVertex = -1;
shader->shaderFrag = -1;
shader->shaderProgram = -1;
SHADER_ACTIVE_COUNT++;
}
void shaderCompile(
shader_t *shader,
const char_t *vertex,
const char_t *fragment
) {
assertNotNull(shader, "shaderCompile: shader cannot be null.");
assertStringValid(vertex, 2048, "shaderGetParameterByName: name invalid.");
assertStringValid(fragment, 2048, "shaderGetParameterByName: name invalid.");
assertTrue(shader->shaderVertex == -1, "shaderCompile: vertex not -1.");
assertTrue(shader->shaderFrag == -1, "shaderCompile: fragment not -1.");
assertTrue(shader->shaderProgram == -1, "shaderCompile: program not -1.");
GLint isSuccess;
int32_t maxLength;
char_t error[2048];
// Load the vertex shader first
shader->shaderVertex = glCreateShader(GL_VERTEX_SHADER);
assertNoGLError();
glShaderSource(shader->shaderVertex, 1, &vertex, NULL);
assertNoGLError();
glCompileShader(shader->shaderVertex);
assertNoGLError();
// Validate
glGetShaderiv(shader->shaderVertex, GL_COMPILE_STATUS, &isSuccess);
assertNoGLError();
if(!isSuccess) {
glGetShaderiv(shader->shaderVertex, GL_INFO_LOG_LENGTH, &maxLength);
glGetShaderInfoLog(shader->shaderVertex, maxLength, &maxLength, error);
sprintf(
error,
"shaderCompile: Failed to compile vert shader:\n%s",
error
);
assertUnreachable(error);
}
assertNoGLError();
// Now load the Frag shader
shader->shaderFrag = glCreateShader(GL_FRAGMENT_SHADER);
assertNoGLError();
glShaderSource(shader->shaderFrag, 1, &fragment, NULL);
assertNoGLError();
glCompileShader(shader->shaderFrag);
assertNoGLError();
glGetShaderiv(shader->shaderFrag, GL_COMPILE_STATUS, &isSuccess);
assertNoGLError();
if(!isSuccess) {
glGetShaderiv(shader->shaderFrag, GL_INFO_LOG_LENGTH, &maxLength);
glGetShaderInfoLog(shader->shaderFrag, maxLength, &maxLength, error);
glDeleteShader(shader->shaderVertex);
sprintf(
error,
"shaderCompile: Failed to compile frag shader:\n%s",
error
);
assertUnreachable(error);
}
assertNoGLError();
// Now create the shader program.
shader->shaderProgram = glCreateProgram();
assertNoGLError();
glAttachShader(shader->shaderProgram, shader->shaderVertex);
assertNoGLError();
glAttachShader(shader->shaderProgram, shader->shaderFrag);
assertNoGLError();
//Bind, Verify & Use the shader program
glLinkProgram(shader->shaderProgram);
assertNoGLError();
glGetProgramiv(shader->shaderProgram, GL_LINK_STATUS, &isSuccess);
assertNoGLError();
if(!isSuccess) {
glGetProgramiv(shader->shaderProgram, GL_INFO_LOG_LENGTH, &maxLength);
assertNoGLError();
glGetProgramInfoLog(shader->shaderProgram, maxLength, &maxLength, error);
assertNoGLError();
glDeleteShader(shader->shaderVertex);
assertNoGLError();
glDeleteShader(shader->shaderFrag);
assertNoGLError();
sprintf(
error,
"shaderCompile: Error compiling shader program:\n%s",
error
);
assertUnreachable(error);
}
}
shaderparameter_t shaderGetParameterByName(
const shader_t *shader,
const char_t *name
) {
assertNotNull(shader, "shaderGetParameterByName: shader cannot be null.");
assertStringValid(name, 128, "shaderGetParameterByName: name invalid.");
shaderparameter_t param = glGetUniformLocation(shader->shaderProgram, name);
assertNoGLError();
assertTrue(param != -1, "shaderGetParameterByName: param not found.");
return param;
}
void shaderBind(const shader_t *shader) {
assertNotNull(shader, "shaderBind: Shader cannot be null.");
assertTrue(
shader->shaderProgram != -1,
"shaderBind: Cannot bind a program that is not ready"
);
glUseProgram(shader->shaderProgram);
assertNoGLError();
}
void shaderSetMatrix(const shaderparameter_t param, const mat4_t matrix) {
glUniformMatrix4fv(param, 1, GL_FALSE, matrix[0]);
assertNoGLError();
}
void shaderSetBoolean(const shaderparameter_t param, const bool_t value) {
glUniform1i(param, value);
assertNoGLError();
}
void shaderSetInteger(const shaderparameter_t param, const int32_t value) {
glUniform1i(param, value);
assertNoGLError();
}
void shaderSetFloat(const shaderparameter_t param, const float_t value) {
glUniform1f(param, value);
assertNoGLError();
}
void shaderSetVector2(const shaderparameter_t param, const vec2_t vector) {
glUniform2f(param, vector[0], vector[1]);
assertNoGLError();
}
void shaderSetVector3(const shaderparameter_t param, const vec3_t vector) {
glUniform3f(param, vector[0], vector[1], vector[2]);
assertNoGLError();
}
void shaderSetVector4(const shaderparameter_t param, const vec4_t vector) {
glUniform4f(param, vector[0], vector[1], vector[2], vector[3]);
assertNoGLError();
}
void shaderSetColor3f(const shaderparameter_t param, const color3f_t color) {
glUniform3f(param, color[0], color[1], color[2]);
assertNoGLError();
}
void shaderSetColor4f(const shaderparameter_t param, const color4f_t color) {
glUniform4f(param, color[0], color[1], color[2], color[3]);
assertNoGLError();
}
void shaderSetTexture(const shaderparameter_t param, const textureslot_t slot) {
glUniform1i(param, slot);
assertNoGLError();
}
void shaderDispose(const shader_t *shader) {
if(shader->shaderProgram != -1) {
glDeleteProgram(shader->shaderProgram);
assertNoGLError();
}
if(shader->shaderVertex != -1) {
glDeleteShader(shader->shaderVertex);
assertNoGLError();
}
if(shader->shaderFrag != -1) {
glDeleteShader(shader->shaderFrag);
assertNoGLError();
}
SHADER_ACTIVE_COUNT--;
}

View File

@ -0,0 +1,152 @@
/**
* Copyright (c) 2023 Dominic Masters
*
* This software is released under the MIT License.
* https://opensource.org/licenses/MIT
*/
#pragma once
#include "dawn.h"
#include "dawnopengl.h"
#include "display/colorgl.h"
#include "display/texture.h"
typedef GLuint shaderparameter_t;
extern int32_t SHADER_ACTIVE_COUNT;
typedef struct {
/** Pointer to an uploaded vertex shader program */
GLuint shaderVertex;
/** Pointer to an uploaded fragment shader program */
GLuint shaderFrag;
/** Pointer to an uploaded shader program linked */
GLuint shaderProgram;
} shader_t;
/**
* Initializes a shader object.
*
* @param shader The shader object to initialize.
*/
void shaderInit(shader_t *shader);
/**
* Compiles a shader object.
*
* @param shader The shader object to compile.
* @param vertex The vertex shader source.
* @param fragment The fragment shader source.
*/
void shaderCompile(
shader_t *shader,
const char_t *vertex,
const char_t *fragment
);
/**
* Returns a shader parameter by name.
*
* @param shader Shader to get the parameter from.
* @param name Name of the parameter to get.
* @return The shader parameter.
*/
shaderparameter_t shaderGetParameterByName(
const shader_t *shader,
const char_t *name
);
/**
* Attaches the supplied shader as the current shader.
*
* @param shader
*/
void shaderBind(const shader_t *shader);
/**
* Sets a matrix parameter on the shader.
*
* @param param Parameter to set.
* @param matrix Matrix value to set.
*/
void shaderSetMatrix(const shaderparameter_t param, const mat4_t matrix);
/**
* Sets a boolean parameter on the shader.
*
* @param param Parameter to set.
* @param value Boolean value to set.
*/
void shaderSetBoolean(const shaderparameter_t param, const bool_t value);
/**
* Sets an integer parameter on the shader.
*
* @param param Parameter to set.
* @param value Integer value to set.
*/
void shaderSetInteger(const shaderparameter_t param, const int32_t value);
/**
* Sets a float parameter on the shader.
*
* @param param Parameter to set.
* @param value Float value to set.
*/
void shaderSetFloat(const shaderparameter_t param, const float_t value);
/**
* Sets a vector parameter on the shader.
*
* @param param Parameter to set.
* @param vector Vector value to set.
*/
void shaderSetVector2(const shaderparameter_t param, const vec2_t vector);
/**
* Sets a vector parameter on the shader.
*
* @param param Parameter to set.
* @param vector Vector value to set.
*/
void shaderSetVector3(const shaderparameter_t param, const vec3_t vector);
/**
* Sets a vector parameter on the shader.
*
* @param param Parameter to set.
* @param vector Vector value to set.
*/
void shaderSetVector4(const shaderparameter_t param, const vec4_t vector);
/**
* Sets a color parameter on the shader.
*
* @param param Parameter to set.
* @param color Color value to set.
*/
void shaderSetColor3f(const shaderparameter_t param, const color3f_t color);
/**
* Sets a color parameter on the shader.
*
* @param param Parameter to set.
* @param color Color value to set.
*/
void shaderSetColor4f(const shaderparameter_t param, const color4f_t color);
/**
* Sets a texture parameter on the shader.
*
* @param param Parameter to set.
* @param slot Slot to bind the texture to.
*/
void shaderSetTexture(const shaderparameter_t param, const textureslot_t slot);
/**
* Disposes of a shader object.
*
* @param shader Shader to destroy.
*/
void shaderDispose(const shader_t *shader);

View File

@ -0,0 +1,10 @@
# Copyright (c) 2022 Dominic Masters
#
# This software is released under the MIT License.
# https://opensource.org/licenses/MIT
# Sources
target_sources(${DAWN_TARGET_NAME}
PRIVATE
simpletexturedshader.c
)

View File

@ -0,0 +1,114 @@
/**
* Copyright (c) 2023 Dominic Masters
*
* This software is released under the MIT License.
* https://opensource.org/licenses/MIT
*/
#include "simpletexturedshader.h"
shader_t SIMPLE_TEXTURED_SHADER;
void simpleTexturedShaderInit() {
shaderInit(&SIMPLE_TEXTURED_SHADER);
shaderCompile(&SIMPLE_TEXTURED_SHADER,
// Vertex Shader
"#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"
"}",
// Fragment Shader
"#version 330 core\n"
"in vec2 o_TextCoord;\n"
"out vec4 o_Color;\n"
"uniform vec4 u_Color;\n"
"uniform bool u_HasTexture;\n"
"uniform sampler2D u_Text;\n"
"void main() {\n"
"if(u_HasTexture) {\n"
"o_Color = texture(u_Text, o_TextCoord) * u_Color;\n"
"} else {\n"
"o_Color = u_Color;"
"}\n"
"}\n"
);
shaderBind(&SIMPLE_TEXTURED_SHADER);
simpleTexturedShaderSetColor(COLOR4F_WHITE);
simpleTexturedShaderSetTexture(NULL);
mat4 projection;
glm_mat4_identity(projection);
glm_perspective(45.0f, 16.0f/9.0f, 0.01f, 1000.0f, projection);
simpleTexturedShaderSetProjection(projection);
mat4 view;
glm_mat4_identity(view);
glm_lookat((vec3_t){ 5,5,5 }, (vec3_t){ 0, 0, 0 }, (vec3_t){ 0, 1, 0 }, view);
simpleTexturedShaderSetView(view);
mat4 model;
glm_mat4_identity(model);
glm_translate(model, (vec3_t){ 0, 0, 0 });
simpleTexturedShaderSetModel(model);
}
void simpleTexturedShaderBind() {
shaderBind(&SIMPLE_TEXTURED_SHADER);
}
void simpleTexturedShaderSetModel(const mat4_t model) {
shaderparameter_t param = shaderGetParameterByName(
&SIMPLE_TEXTURED_SHADER, "u_Model"
);
shaderSetMatrix(param, model);
}
void simpleTexturedShaderSetView(const mat4_t view) {
shaderparameter_t param = shaderGetParameterByName(
&SIMPLE_TEXTURED_SHADER, "u_View"
);
shaderSetMatrix(param, view);
}
void simpleTexturedShaderSetProjection(const mat4_t projection) {
shaderparameter_t param = shaderGetParameterByName(
&SIMPLE_TEXTURED_SHADER, "u_Projection"
);
shaderSetMatrix(param, projection);
}
void simpleTexturedShaderSetColor(const color4f_t color) {
shaderparameter_t param = shaderGetParameterByName(
&SIMPLE_TEXTURED_SHADER, "u_Color"
);
shaderSetVector4(param, color);
}
void simpleTexturedShaderSetTexture(const texture_t *texture) {
shaderparameter_t paramHas = shaderGetParameterByName(
&SIMPLE_TEXTURED_SHADER, "u_HasTexture"
);
if(texture == NULL) {
shaderSetBoolean(paramHas, false);
return;
}
shaderparameter_t param = shaderGetParameterByName(
&SIMPLE_TEXTURED_SHADER, "u_Text"
);
textureBind(texture, 0);
shaderSetTexture(param, 0);
shaderSetBoolean(paramHas, true);
}
void simpleTexturedShaderDispose() {
shaderDispose(&SIMPLE_TEXTURED_SHADER);
}

View File

@ -0,0 +1,57 @@
/**
* Copyright (c) 2023 Dominic Masters
*
* This software is released under the MIT License.
* https://opensource.org/licenses/MIT
*/
#pragma once
#include "display/shader.h"
extern shader_t SIMPLE_TEXTURED_SHADER;
/**
* Initialize the simple textured shader. This is a simple shader for rendering
* textured objects without lighting or anything fancy.
*/
void simpleTexturedShaderInit();
/**
* Binds the simple textured shader.
*/
void simpleTexturedShaderBind();
/**
* Sets the model matrix for the shader.
* @param model The model matrix to set.
*/
void simpleTexturedShaderSetModel(const mat4_t model);
/**
* Sets the color for the shader.
* @param color The color to set.
*/
void simpleTexturedShaderSetColor(const color4f_t color);
/**
* Sets the projection matrix for the shader.
* @param projection The projection matrix to set.
*/
void simpleTexturedShaderSetProjection(const mat4_t projection);
/**
* Sets the view matrix for the shader.
* @param view The view matrix to set.
*/
void simpleTexturedShaderSetView(const mat4_t view);
/**
* Sets the texture for the shader.
* @param texture The texture to set.
*/
void simpleTexturedShaderSetTexture(const texture_t *texture);
/**
* Destroys the simple textured shader.
*/
void simpleTexturedShaderDispose();

View File

@ -0,0 +1,199 @@
/**
* Copyright (c) 2023 Dominic Masters
*
* This software is released under the MIT License.
* https://opensource.org/licenses/MIT
*/
#include "texture.h"
#include "assert/assert.h"
#include "assert/assertgl.h"
#include "util/math.h"
int32_t TEXTURE_ACTIVE_COUNT;
void textureInit(
texture_t *texture,
int32_t width,
int32_t height,
textureformat_t format,
texturedataformat_t dataFormat
) {
assertNotNull(texture, "textureInit: texture cannot be null.");
assertTrue(width > 0, "textureInit: width must be greater than 0.");
assertTrue(height > 0, "textureInit: height must be greater than 0.");
texture->width = width;
texture->height = height;
texture->format = format;
texture->dataFormat = dataFormat;
texture->wrapModeX = TEXTURE_WRAP_MODE_REPEAT;
texture->wrapModeY = TEXTURE_WRAP_MODE_REPEAT;
texture->filterModeMin = TEXTURE_FILTER_MODE_LINEAR;
texture->filterModeMag = TEXTURE_FILTER_MODE_LINEAR;
int32_t maxSize;
glGetIntegerv(GL_MAX_TEXTURE_SIZE, &maxSize);
assertTrue(
width > 0 && width <= maxSize, "textureInit: Width is out of bounds!"
);
assertTrue(
height > 0 && height <= maxSize, "textureInit: Height is out of bounds!"
);
glGenTextures(1, &texture->id);
assertNoGLError();
assertTrue(texture->id > 0, "textureInit: Failed to generate texture ID!");
TEXTURE_ACTIVE_COUNT++;
switch(texture->dataFormat) {
case TEXTURE_DATA_FORMAT_FLOAT:
texture->elementSize = sizeof(float_t);
break;
case TEXTURE_DATA_FORMAT_UNSIGNED_BYTE:
texture->elementSize = sizeof(uint8_t);
break;
default:
assertUnreachable("textureInit: Unknown data format.");
break;
}
switch(texture->format) {
case TEXTURE_FORMAT_RGBA:
texture->elementSize *= 4;
break;
case TEXTURE_FORMAT_RGB:
texture->elementSize *= 3;
break;
case TEXTURE_FORMAT_RG:
texture->elementSize *= 2;
break;
case TEXTURE_FORMAT_R:
break;
default:
assertUnreachable("textureInit: Unknown texture format.");
break;
}
textureBind(texture, 0);
textureBuffer(texture, NULL);
}
void textureBind(const texture_t *texture, const textureslot_t slot) {
glActiveTexture(GL_TEXTURE0 + slot);
assertNoGLError();
glBindTexture(GL_TEXTURE_2D, texture->id);
assertNoGLError();
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, texture->wrapModeX);
assertNoGLError();
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, texture->wrapModeY);
assertNoGLError();
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, texture->filterModeMin);
assertNoGLError();
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, texture->filterModeMag);
assertNoGLError();
}
void textureBuffer(const texture_t *texture, const void *data) {
assertNotNull(texture, "textureBuffer: texture cannot be null.");
textureBind(texture, 0);
glTexImage2D(
GL_TEXTURE_2D,
0,
texture->format,
texture->width,
texture->height,
0,
texture->format,
texture->dataFormat,
data
);
assertNoGLError();
// glGenerateMipmap(GL_TEXTURE_2D);
// assertNoGLError();
}
void textureBufferSub(
const texture_t *texture,
const void *data,
const size_t offset,
const size_t length
) {
assertNotNull(texture, "textureBufferSub: Texture cannot be null.");
assertNotNull(data, "textureBufferSub: Data cannot be null.");
assertTrue(offset >= 0, "textureBufferSub: Offset must be greater than 0.");
assertTrue(length > 0, "textureBufferSub: Length must be greater than 0.");
assertTrue(
offset + length <= texture->width * texture->height,
"textureBufferSub: Buffer out of bounds."
);
if(offset == 0 && length == texture->width * texture->height) {
textureBuffer(texture, data);
return;
}
// Bind the texture.
textureBind(texture, 0);
// Buffer initial row only.
int32_t xOffset = offset % texture->width;
int32_t yOffset = offset / texture->width;
size_t n = mathMin(length, texture->width - xOffset);
size_t buffered = 0;
glTexSubImage2D(
GL_TEXTURE_2D,
0,
xOffset, yOffset,
n, 1,
texture->format, texture->dataFormat,
data
);
assertNoGLError();
buffered += n;
yOffset += 1;
// Buffer the creamy middle (inefficient?)
int32_t rows = 0;
size_t initialBuffered = buffered;
while(buffered + texture->width < length) {
rows += 1;
buffered += texture->width;
}
if(rows > 0) {
glTexSubImage2D(
GL_TEXTURE_2D,
0,
0, yOffset,
texture->width, rows,
texture->format, texture->dataFormat,
data + (initialBuffered * texture->elementSize)
);
assertNoGLError();
yOffset += rows;
}
// Buffer last row only.
if(buffered != length) {
glTexSubImage2D(
GL_TEXTURE_2D,
0,
0, yOffset,
length - buffered, 1,
texture->format, texture->dataFormat,
data + (buffered * texture->elementSize)
);
assertNoGLError();
}
}
void textureDispose(const texture_t *texture) {
glDeleteTextures(1, &texture->id);
assertNoGLError();
TEXTURE_ACTIVE_COUNT--;
}

View File

@ -0,0 +1,106 @@
/**
* Copyright (c) 2024 Dominic Masters
*
* This software is released under the MIT License.
* https://opensource.org/licenses/MIT
*/
#pragma once
#include "dawn.h"
#include "dawnopengl.h"
extern int32_t TEXTURE_ACTIVE_COUNT;
typedef GLuint textureslot_t;
typedef enum {
TEXTURE_FORMAT_R = GL_RED,
TEXTURE_FORMAT_RG = GL_RG,
TEXTURE_FORMAT_RGB = GL_RGB,
TEXTURE_FORMAT_RGBA = GL_RGBA
} textureformat_t;
typedef enum {
TEXTURE_WRAP_MODE_REPEAT = GL_REPEAT,
TEXTURE_WRAP_MODE_MIRRORED_REPEAT = GL_MIRRORED_REPEAT,
TEXTURE_WRAP_MODE_CLAMP_TO_EDGE = GL_CLAMP_TO_EDGE,
} texturewrapmode_t;
typedef enum {
TEXTURE_FILTER_MODE_NEAREST = GL_NEAREST,
TEXTURE_FILTER_MODE_LINEAR = GL_LINEAR
} texturefiltermode_t;
typedef enum {
TEXTURE_DATA_FORMAT_UNSIGNED_BYTE = GL_UNSIGNED_BYTE,
TEXTURE_DATA_FORMAT_FLOAT = GL_FLOAT
} texturedataformat_t;
typedef struct {
GLuint id;
int32_t width;
int32_t height;
textureformat_t format;
texturedataformat_t dataFormat;
texturewrapmode_t wrapModeX;
texturewrapmode_t wrapModeY;
texturefiltermode_t filterModeMin;
texturefiltermode_t filterModeMag;
size_t elementSize;
} texture_t;
/**
* Initializes a texture to be read to be used.
*
* @param texture Texture to initialize.
* @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.
*/
void textureInit(
texture_t *texture,
int32_t width,
int32_t height,
textureformat_t format,
texturedataformat_t dataFormat
);
/**
* Binds the texture to the given slot (for use by the shaders).
*
* @param texture Texture to bind.
* @param slot Slot to bind to.
*/
void textureBind(const texture_t *texture, const textureslot_t slot);
/**
* Buffers pixels on to the texture. Set data to NULL to fill with empty pixels
* (black by default).
*
* @param texture Texture to buffer.
* @param data Data to buffer.
*/
void textureBuffer(const texture_t *texture, const void *data);
/**
* Buffer sub data to this texture.
*
* @param texture Texture to buffer on to.
* @param data Data to buffer.
* @param offset Offset position to buffer on to.
* @param length Length of the data you are buffering.
*/
void textureBufferSub(
const texture_t *texture,
const void *data,
const size_t offset,
const size_t length
);
/**
* Unloads a previously initialized texture.
*
* @param texture Texture to destroy.
*/
void textureDispose(const texture_t *texture);

View File

@ -0,0 +1,85 @@
/**
* Copyright (c) 2024 Dominic Masters
*
* This software is released under the MIT License.
* https://opensource.org/licenses/MIT
*/
#include "textureframebuffer.h"
#include "assert/assert.h"
#include "assert/assertgl.h"
void textureFrameBufferInit(
textureframebuffer_t *framebuffer,
const int32_t width,
const int32_t height
) {
assertNotNull(framebuffer, "Framebuffer cannot be NULL");
assertTrue(width > 0, "Width must be greater than 0");
assertTrue(height > 0, "Height must be greater than 0");
textureInit(
&framebuffer->texture,
width, height,
TEXTURE_FORMAT_RGBA,
TEXTURE_DATA_FORMAT_FLOAT
);
// Create Frame Buffer
glGenFramebuffers(1, &framebuffer->fboId);
assertNoGLError();
glBindFramebuffer(GL_FRAMEBUFFER, framebuffer->fboId);
assertNoGLError();
glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0,
GL_TEXTURE_2D, framebuffer->texture.id, 0
);
assertNoGLError();
// Create Render Buffer
glGenRenderbuffers(1, &framebuffer->rboId);
assertNoGLError();
glBindRenderbuffer(GL_RENDERBUFFER, framebuffer->rboId);
assertNoGLError();
glRenderbufferStorage(
GL_RENDERBUFFER, GL_DEPTH24_STENCIL8,
framebuffer->texture.width, framebuffer->texture.height
);
assertNoGLError();
glBindRenderbuffer(GL_RENDERBUFFER, 0);
assertNoGLError();
glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_DEPTH_STENCIL_ATTACHMENT,
GL_RENDERBUFFER, framebuffer->rboId
);
assertNoGLError();
// Validate things went correct.
if(glCheckFramebufferStatus(GL_FRAMEBUFFER) != GL_FRAMEBUFFER_COMPLETE) {
assertUnreachable("Framebuffer is not complete!");
}
}
void textureFrameBufferBind(const textureframebuffer_t *framebuffer) {
assertNotNull(framebuffer, "Framebuffer cannot be NULL");
glBindFramebuffer(GL_FRAMEBUFFER, framebuffer->fboId);
assertNoGLError();
glViewport(0, 0, framebuffer->texture.width, framebuffer->texture.height);
assertNoGLError();
}
void textureFrameBufferClear(const textureframebuffer_t *framebuffer) {
assertNotNull(framebuffer, "Framebuffer cannot be NULL");
glClearColor(0,0,0,1);
assertNoGLError();
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
assertNoGLError();
}
void textureFrameBufferDispose(textureframebuffer_t *framebuffer) {
assertNotNull(framebuffer, "Framebuffer cannot be NULL");
glDeleteFramebuffers(1, &framebuffer->fboId);
glDeleteRenderbuffers(1, &framebuffer->rboId);
textureDispose(&framebuffer->texture);
}

View File

@ -0,0 +1,50 @@
/**
* Copyright (c) 2024 Dominic Masters
*
* This software is released under the MIT License.
* https://opensource.org/licenses/MIT
*/
#pragma once
#include "display/texture.h"
#include "display/color.h"
typedef struct {
texture_t texture;
GLuint fboId;
GLuint rboId;
} textureframebuffer_t;
/**
* Initializes a texture framebuffer.
*
* @param framebuffer Framebuffer to initialize.
* @param width Width of the framebuffer (in pixels).
* @param height Height of the framebuffer (in pixels).
*/
void textureFrameBufferInit(
textureframebuffer_t *framebuffer,
const int32_t width,
const int32_t height
);
/**
* Binds the framebuffer to the current context.
*
* @param framebuffer Framebuffer to bind.
*/
void textureFrameBufferBind(const textureframebuffer_t *framebuffer);
/**
* Clears the framebuffer.
*
* @param framebuffer Framebuffer to clear.
*/
void textureFrameBufferClear(const textureframebuffer_t *framebuffer);
/**
* Disposes of the framebuffer.
*
* @param framebuffer Framebuffer to dispose.
*/
void textureFrameBufferDispose(textureframebuffer_t *framebuffer);

View File

@ -48,7 +48,9 @@ int32_t keyboardCharacterRead() {
} }
} }
int32_t linuxHostBeginUpdating() { int32_t linuxHostHandoff() {
gameInit();
while(true) { while(true) {
terminalModeSetCONIO(); terminalModeSetCONIO();

View File

@ -13,4 +13,4 @@ extern int32_t LINUX_TERM_HOST_KEY_PRESS;
/** /**
* Begins updating the host. * Begins updating the host.
*/ */
int32_t linuxHostBeginUpdating(); int32_t linuxHostHandoff();