Added texture support.
This commit is contained in:
53
src/engine/display/primitives/quad.c
Normal file
53
src/engine/display/primitives/quad.c
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
/**
|
||||||
|
* Copyright (c) 2021 Dominic Masters
|
||||||
|
*
|
||||||
|
* This software is released under the MIT License.
|
||||||
|
* https://opensource.org/licenses/MIT
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "quad.h"
|
||||||
|
|
||||||
|
void quadBuffer(primitive_t *primitive,
|
||||||
|
float x0, float y0, float u0, float v0,
|
||||||
|
float x1, float y1, float u1, float v1,
|
||||||
|
int32_t verticeStart, int32_t indiceStart
|
||||||
|
) {
|
||||||
|
vertice_t *vertices = malloc(sizeof(vertice_t) * 4);
|
||||||
|
indice_t *indices = malloc(sizeof(indice_t) * 6);
|
||||||
|
|
||||||
|
vertices[0].x = x0, vertices[0].y = y0, vertices[0].z = 0;
|
||||||
|
vertices[0].u = u0, vertices[0].v = v0;
|
||||||
|
|
||||||
|
vertices[1].x = x1, vertices[1].y = y0, vertices[1].z = 0;
|
||||||
|
vertices[1].u = u1, vertices[1].v = v0;
|
||||||
|
|
||||||
|
vertices[2].x = x0, vertices[2].y = y1, vertices[2].z = 0;
|
||||||
|
vertices[2].u = u0, vertices[2].v = v1;
|
||||||
|
|
||||||
|
vertices[3].x = x1, vertices[3].y = y1, vertices[3].z = 0;
|
||||||
|
vertices[3].u = u1, vertices[3].v = v1;
|
||||||
|
|
||||||
|
indices[0] = (indice_t)0, indices[1] = (indice_t)1, indices[2] = (indice_t)2;
|
||||||
|
indices[3] = (indice_t)1, indices[4] = (indice_t)2, indices[5] = (indice_t)3;
|
||||||
|
|
||||||
|
primitiveBufferVertices(primitive, verticeStart, 4, vertices);
|
||||||
|
primitiveBufferIndices(primitive, indiceStart, 6, indices);
|
||||||
|
|
||||||
|
free(vertices);
|
||||||
|
free(indices);
|
||||||
|
}
|
||||||
|
|
||||||
|
primitive_t * quadCreate(
|
||||||
|
float x0, float y0, float u0, float v0,
|
||||||
|
float x1, float y1, float u1, float v1
|
||||||
|
) {
|
||||||
|
primitive_t *primitive = primitiveCreate(4, 6);
|
||||||
|
|
||||||
|
quadBuffer(primitive,
|
||||||
|
x0, y0, u0, v0,
|
||||||
|
x1, y1, u1, v1,
|
||||||
|
0, 0
|
||||||
|
);
|
||||||
|
|
||||||
|
return primitive;
|
||||||
|
}
|
19
src/engine/display/primitives/quad.h
Normal file
19
src/engine/display/primitives/quad.h
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
// Copyright (c) 2021 Dominic Masters
|
||||||
|
//
|
||||||
|
// This software is released under the MIT License.
|
||||||
|
// https://opensource.org/licenses/MIT
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
#include <stdint.h>
|
||||||
|
#include "../primitive.h"
|
||||||
|
|
||||||
|
void quadBuffer(primitive_t *primitive,
|
||||||
|
float x0, float y0, float u0, float v0,
|
||||||
|
float x1, float y1, float u1, float v1,
|
||||||
|
int32_t verticeStart, int32_t indiceStart
|
||||||
|
);
|
||||||
|
|
||||||
|
primitive_t * quadCreate(
|
||||||
|
float x0, float y0, float u0, float v0,
|
||||||
|
float x1, float y1, float u1, float v1
|
||||||
|
);
|
@ -7,8 +7,6 @@
|
|||||||
|
|
||||||
#include "render.h"
|
#include "render.h"
|
||||||
|
|
||||||
primitive_t *primitive;
|
|
||||||
|
|
||||||
render_t * renderInit(char *name) {
|
render_t * renderInit(char *name) {
|
||||||
// Initialize the renderer
|
// Initialize the renderer
|
||||||
render_t *render = malloc(sizeof(render_t));
|
render_t *render = malloc(sizeof(render_t));
|
||||||
@ -16,42 +14,21 @@ render_t * renderInit(char *name) {
|
|||||||
// render->width = WINDOW_WIDTH_DEFAULT;
|
// render->width = WINDOW_WIDTH_DEFAULT;
|
||||||
// render->height = WINDOW_HEIGHT_DEFAULT;
|
// render->height = WINDOW_HEIGHT_DEFAULT;
|
||||||
|
|
||||||
//GLEnable Things
|
// Enable GL things.
|
||||||
glEnable(GL_BLEND);
|
glEnable(GL_BLEND);
|
||||||
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
glEnable(GL_TEXTURE_2D);
|
||||||
glEnable(GL_DEPTH_TEST);
|
glEnable(GL_DEPTH_TEST);
|
||||||
|
|
||||||
|
// Setup the alpha blend function.
|
||||||
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
||||||
glDepthMask(GL_TRUE);
|
glDepthMask(GL_TRUE);
|
||||||
glDepthFunc(GL_LEQUAL);
|
glDepthFunc(GL_LEQUAL);
|
||||||
|
|
||||||
// Test
|
|
||||||
for(int i = 0; i < 100; i++) {
|
|
||||||
vertice_t *vertices = malloc(sizeof(vertice_t) * 4);
|
|
||||||
indice_t *indices = malloc(sizeof(indice_t) * 6);
|
|
||||||
|
|
||||||
vertices[0].x = -1, vertices[0].y = -1, vertices[0].z = 0;
|
|
||||||
vertices[1].x = 1, vertices[1].y = -1, vertices[1].z = 0;
|
|
||||||
vertices[2].x = -1, vertices[2].y = 1, vertices[2].z = 0;
|
|
||||||
vertices[3].x = 1, vertices[3].y = 1, vertices[3].z = 0;
|
|
||||||
|
|
||||||
indices[0] = (indice_t)0;
|
|
||||||
indices[1] = (indice_t)1;
|
|
||||||
indices[2] = (indice_t)2;
|
|
||||||
|
|
||||||
indices[3] = (indice_t)1;
|
|
||||||
indices[4] = (indice_t)2;
|
|
||||||
indices[5] = (indice_t)3;
|
|
||||||
|
|
||||||
primitive = primitiveCreate(4, 6);
|
|
||||||
primitiveBufferVertices(primitive, 0, 4, vertices);
|
|
||||||
primitiveBufferIndices(primitive, 0, 6, indices);
|
|
||||||
}
|
|
||||||
|
|
||||||
return render;
|
return render;
|
||||||
}
|
}
|
||||||
|
|
||||||
void renderFrame(render_t *render) {
|
void renderFrame(render_t *render) {
|
||||||
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
|
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
|
||||||
primitiveDraw(primitive, 0, 6);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool renderDispose(render_t *render) {
|
bool renderDispose(render_t *render) {
|
||||||
|
@ -77,6 +77,7 @@ shader_t * shaderCompile(char *vertexShaderSource, char* fragmentShaderSource) {
|
|||||||
|
|
||||||
shader->uniProj = glGetUniformLocation(shader->shaderProgram, SHADER_UNI_PROJ);
|
shader->uniProj = glGetUniformLocation(shader->shaderProgram, SHADER_UNI_PROJ);
|
||||||
shader->uniView = glGetUniformLocation(shader->shaderProgram, SHADER_UNI_VIEW);
|
shader->uniView = glGetUniformLocation(shader->shaderProgram, SHADER_UNI_VIEW);
|
||||||
|
shader->uniText = glGetUniformLocation(shader->shaderProgram, SHADER_UNI_TEXT);
|
||||||
|
|
||||||
// Bind the shader
|
// Bind the shader
|
||||||
shaderUse(shader);
|
shaderUse(shader);
|
||||||
@ -100,4 +101,10 @@ void shaderUse(shader_t *shader) {
|
|||||||
void shaderUseCamera(shader_t *shader, camera_t *camera) {
|
void shaderUseCamera(shader_t *shader, camera_t *camera) {
|
||||||
glUniformMatrix4fv(shader->uniView, 1, GL_FALSE, (float*)camera->view);
|
glUniformMatrix4fv(shader->uniView, 1, GL_FALSE, (float*)camera->view);
|
||||||
glUniformMatrix4fv(shader->uniProj, 1, GL_FALSE, (float*)camera->projection);
|
glUniformMatrix4fv(shader->uniProj, 1, GL_FALSE, (float*)camera->projection);
|
||||||
|
}
|
||||||
|
void shaderUseTexture(shader_t *shader, texture_t *texture) {
|
||||||
|
// OpenGL requires us to set the active texure.
|
||||||
|
glActiveTexture(GL_TEXTURE0);
|
||||||
|
glBindTexture(GL_TEXTURE_2D, texture->id);
|
||||||
|
glUniform1i(shader->uniText, 0);
|
||||||
}
|
}
|
@ -12,9 +12,11 @@
|
|||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
#include <malloc.h>
|
#include <malloc.h>
|
||||||
#include "camera.h"
|
#include "camera.h"
|
||||||
|
#include "texture.h"
|
||||||
|
|
||||||
#define SHADER_UNI_VIEW "u_View"
|
#define SHADER_UNI_VIEW "u_View"
|
||||||
#define SHADER_UNI_PROJ "u_Proj"
|
#define SHADER_UNI_PROJ "u_Proj"
|
||||||
|
#define SHADER_UNI_TEXT "u_Text"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Structure containing information about an OpenGL Shader. For simplicity sake
|
* Structure containing information about an OpenGL Shader. For simplicity sake
|
||||||
@ -35,6 +37,9 @@ typedef struct {
|
|||||||
|
|
||||||
/** Matrix for the projection matrix */
|
/** Matrix for the projection matrix */
|
||||||
GLint uniProj;
|
GLint uniProj;
|
||||||
|
|
||||||
|
/** Uniform for the current texture */
|
||||||
|
GLint uniText;
|
||||||
} shader_t;
|
} shader_t;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -61,4 +66,16 @@ bool shaderDipose(shader_t *shader);
|
|||||||
void shaderUse(shader_t *shader);
|
void shaderUse(shader_t *shader);
|
||||||
|
|
||||||
|
|
||||||
void shaderUseCamera(shader_t *shader, camera_t *camera);
|
/**
|
||||||
|
* Attaches a camera to the shader.
|
||||||
|
* @param shader Shader to attach to.
|
||||||
|
* @param camera Camera to attach.
|
||||||
|
*/
|
||||||
|
void shaderUseCamera(shader_t *shader, camera_t *camera);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Attaches a texture to the shader.
|
||||||
|
* @param shader Shader to attach to.
|
||||||
|
* @param texture Texture to attach.
|
||||||
|
*/
|
||||||
|
void shaderUseTexture(shader_t *shader, texture_t *texture);
|
61
src/engine/display/texture.c
Normal file
61
src/engine/display/texture.c
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
/**
|
||||||
|
* Copyright (c) 2021 Dominic Masters
|
||||||
|
*
|
||||||
|
* This software is released under the MIT License.
|
||||||
|
* https://opensource.org/licenses/MIT
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "texture.h"
|
||||||
|
|
||||||
|
texture_t * textureCreate(int32_t width, int32_t height, pixel_t *pixels) {
|
||||||
|
texture_t *texture = malloc(sizeof(texture_t));
|
||||||
|
if(texture == NULL) return NULL;
|
||||||
|
|
||||||
|
texture->width = width;
|
||||||
|
texture->height = height;
|
||||||
|
|
||||||
|
// Generate a texture ID and bind.
|
||||||
|
glGenTextures(1, &texture->id);
|
||||||
|
glActiveTexture(GL_TEXTURE0);
|
||||||
|
glBindTexture(GL_TEXTURE_2D, texture->id);
|
||||||
|
|
||||||
|
// Setup our preferred texture params
|
||||||
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_REPEAT);
|
||||||
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_REPEAT);
|
||||||
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_NEAREST);
|
||||||
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_NEAREST);
|
||||||
|
|
||||||
|
// Start by buffering all transparent black pixels.
|
||||||
|
if(pixels == NULL) {
|
||||||
|
pixels = calloc(width * height, sizeof(pixel_t));
|
||||||
|
|
||||||
|
glTexImage2D(
|
||||||
|
GL_TEXTURE_2D, 0, GL_RGBA, width, height, 0,
|
||||||
|
GL_RGBA, GL_UNSIGNED_BYTE, pixels
|
||||||
|
);
|
||||||
|
|
||||||
|
free(pixels);
|
||||||
|
} else {
|
||||||
|
glTexImage2D(
|
||||||
|
GL_TEXTURE_2D, 0, GL_RGBA, width, height, 0,
|
||||||
|
GL_RGBA, GL_UNSIGNED_BYTE, pixels
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return texture;
|
||||||
|
}
|
||||||
|
|
||||||
|
void textureBufferPixels(texture_t *texture,
|
||||||
|
int32_t x, int32_t y, int32_t width, int32_t height, pixel_t *pixels
|
||||||
|
) {
|
||||||
|
glBindTexture(GL_TEXTURE_2D, texture->id);
|
||||||
|
glTexSubImage2D(GL_TEXTURE_2D, 0,
|
||||||
|
x, y, width, height,
|
||||||
|
GL_RGBA, GL_UNSIGNED_BYTE, pixels
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
void textureDispose(texture_t *texture) {
|
||||||
|
glDeleteTextures(1, &texture->id);
|
||||||
|
free(texture);
|
||||||
|
}
|
57
src/engine/display/texture.h
Normal file
57
src/engine/display/texture.h
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
// Copyright (c) 2021 Dominic Masters
|
||||||
|
//
|
||||||
|
// This software is released under the MIT License.
|
||||||
|
// https://opensource.org/licenses/MIT
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <glad/glad.h>
|
||||||
|
#include <malloc.h>
|
||||||
|
#include <memory.h>
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Structure detailing information about a texture.
|
||||||
|
* Because we plan to upload the pixels of a texture into the GPU, we don't
|
||||||
|
* store the pixels in memory because we don't need to!
|
||||||
|
*/
|
||||||
|
typedef struct {
|
||||||
|
int32_t width;
|
||||||
|
int32_t height;
|
||||||
|
GLuint id;
|
||||||
|
} texture_t;
|
||||||
|
|
||||||
|
/** Information about a single pixel. */
|
||||||
|
typedef struct {
|
||||||
|
uint8_t r, g, b, a;
|
||||||
|
} pixel_t;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a new texture that can be written in to.
|
||||||
|
*
|
||||||
|
* @param width Width of the texture (in pixels).
|
||||||
|
* @param height Height of the texture (in pixels).
|
||||||
|
* @param pixels Default pixel array, set to NULL to set all pixel data to 0.
|
||||||
|
* @return The newly created texture instance.
|
||||||
|
*/
|
||||||
|
texture_t * textureCreate(int32_t width, int32_t height, pixel_t *pixels);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Buffer pixel data onto the GPU. Pixel buffering is rather costly so avoid
|
||||||
|
* doing this too often.
|
||||||
|
*
|
||||||
|
* @param texture Texture to buffer in to.
|
||||||
|
* @param x X coordinate in texture space to render to.
|
||||||
|
* @param y Y coordinate in texture space to render to.
|
||||||
|
* @param width Width of the pixel region you're buffering.
|
||||||
|
* @param height Height of the pixel region you're buffering.
|
||||||
|
* @param pixels Array of pixels to buffer onto the GPU.
|
||||||
|
*/
|
||||||
|
void textureBufferPixels(texture_t *texture,
|
||||||
|
int32_t x, int32_t y, int32_t width, int32_t height, pixel_t *pixels
|
||||||
|
);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Clean a previously created texture.
|
||||||
|
* @param texture Texture to clean up.
|
||||||
|
*/
|
||||||
|
void textureDispose(texture_t *texture);
|
@ -7,9 +7,16 @@
|
|||||||
|
|
||||||
#include "engine.h"
|
#include "engine.h"
|
||||||
|
|
||||||
|
#include "display/primitives/quad.h"
|
||||||
|
#include "display/texture.h"
|
||||||
|
#include "display/shader.h"
|
||||||
|
|
||||||
camera_t *camera;
|
camera_t *camera;
|
||||||
shader_t *shader;
|
shader_t *shader;
|
||||||
|
|
||||||
|
primitive_t *primitive;
|
||||||
|
texture_t *texture;
|
||||||
|
|
||||||
engine_t * engineInit(platform_t *platform, char *name, uint32_t inputCount) {
|
engine_t * engineInit(platform_t *platform, char *name, uint32_t inputCount) {
|
||||||
// Create the engine instance.
|
// Create the engine instance.
|
||||||
engine_t *engine = malloc(sizeof(engine_t));
|
engine_t *engine = malloc(sizeof(engine_t));
|
||||||
@ -43,12 +50,30 @@ engine_t * engineInit(platform_t *platform, char *name, uint32_t inputCount) {
|
|||||||
cameraPerspective(camera, 45.0f, 1920.0f/1080.0f, 3.0f, 100.0f);
|
cameraPerspective(camera, 45.0f, 1920.0f/1080.0f, 3.0f, 100.0f);
|
||||||
shaderUseCamera(shader, camera);
|
shaderUseCamera(shader, camera);
|
||||||
|
|
||||||
|
// Test
|
||||||
|
primitive = quadCreate(
|
||||||
|
-1, -1, 0, 0,
|
||||||
|
1, 1, 1, 1
|
||||||
|
);
|
||||||
|
|
||||||
|
texture = textureCreate(1, 1, NULL);
|
||||||
|
pixel_t *pixels = malloc(sizeof(pixel_t) * 1 * 1);
|
||||||
|
for(int32_t i = 0; i < 1*1; i++) {
|
||||||
|
pixels[i].r = 0xFF, pixels[i].g = 0x00, pixels[i].b = 0xFF;
|
||||||
|
pixels[i].a = 0xFF;
|
||||||
|
}
|
||||||
|
textureBufferPixels(texture, 0, 0, 1, 1, pixels);
|
||||||
|
shaderUseTexture(shader, texture);
|
||||||
|
|
||||||
return engine;
|
return engine;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32_t engineUpdate(engine_t *engine) {
|
uint32_t engineUpdate(engine_t *engine) {
|
||||||
shaderUse(shader);
|
shaderUse(shader);
|
||||||
renderFrame(engine->render);
|
renderFrame(engine->render);
|
||||||
|
|
||||||
|
primitiveDraw(primitive, 0, 6);
|
||||||
|
|
||||||
inputUpdate(engine->input);
|
inputUpdate(engine->input);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -29,9 +29,7 @@ input_t * inputInit(uint32_t inputBindCount, uint32_t inputSourceCount) {
|
|||||||
input->times = malloc(sizeof(float) * inputBindCount);
|
input->times = malloc(sizeof(float) * inputBindCount);
|
||||||
|
|
||||||
// Create the buffer, zero all the values out.
|
// Create the buffer, zero all the values out.
|
||||||
size_t size = sizeof(inputval_t) * inputSourceCount;
|
input->buffer = calloc(inputSourceCount, sizeof(inputval_t));
|
||||||
input->buffer = malloc(size);
|
|
||||||
memset(input->buffer, 0, size);
|
|
||||||
|
|
||||||
// Pass off the input manager
|
// Pass off the input manager
|
||||||
return input;
|
return input;
|
||||||
|
@ -7,6 +7,7 @@
|
|||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include <malloc.h>
|
#include <malloc.h>
|
||||||
|
#include <memory.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include "../util/list/list.h"
|
#include "../util/list/list.h"
|
||||||
#include "../platform.h"
|
#include "../platform.h"
|
||||||
|
Reference in New Issue
Block a user