From ada87381c5ec00ea9c36d682f35c75d9f6900198 Mon Sep 17 00:00:00 2001 From: Matt Borgerson Date: Thu, 12 Mar 2020 02:23:16 -0700 Subject: [PATCH] ui: Add xemu UI shader helpers --- ui/xemu-shaders.c | 319 ++++++++++++++++++++++++++++++++++++++++++++++ ui/xemu-shaders.h | 99 ++++++++++++++ 2 files changed, 418 insertions(+) create mode 100644 ui/xemu-shaders.c create mode 100644 ui/xemu-shaders.h diff --git a/ui/xemu-shaders.c b/ui/xemu-shaders.c new file mode 100644 index 0000000000..1f37547394 --- /dev/null +++ b/ui/xemu-shaders.c @@ -0,0 +1,319 @@ +/* + * xemu User Interface Rendering Helpers + * + * Copyright (C) 2020 Matt Borgerson + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +#include +#include +#include +#include +#include "xemu-shaders.h" +#include "ui/shader/xemu-logo-frag.h" + +#define STB_IMAGE_IMPLEMENTATION +#include "stb_image.h" + +GLuint compile_shader(GLenum type, const char *src) +{ + GLint status; + char err_buf[512]; + GLuint shader = glCreateShader(type); + if (shader == 0) { + fprintf(stderr, "ERROR: Failed to create shader\n"); + return 0; + } + glShaderSource(shader, 1, &src, NULL); + glCompileShader(shader); + glGetShaderiv(shader, GL_COMPILE_STATUS, &status); + if (status != GL_TRUE) { + glGetShaderInfoLog(shader, sizeof(err_buf), NULL, err_buf); + fprintf(stderr, "ERROR: Shader compilation failed!\n\n"); + fprintf(stderr, "[Shader Info Log]\n"); + fprintf(stderr, "%s\n", err_buf); + fprintf(stderr, "[Shader Source]\n"); + fprintf(stderr, "%s\n", src); + return 0; + } + + return shader; +} + +struct decal_shader *create_decal_shader(enum SHADER_TYPE type) +{ + // Allocate shader wrapper object + struct decal_shader *s = (struct decal_shader *)malloc(sizeof(struct decal_shader)); + assert(s != NULL); + s->flip = 0; + s->scale = 1.4; + s->smoothing = 1.0; + s->outline_dist = 1.0; + + const char *vert_src = + "#version 150 core\n" + "uniform bool in_FlipY;\n" + "uniform vec4 in_ScaleOffset;\n" + "uniform vec4 in_TexScaleOffset;\n" + "in vec2 in_Position;\n" + "in vec2 in_Texcoord;\n" + "out vec2 Texcoord;\n" + "void main() {\n" + " vec2 t = in_Texcoord;\n" + " if (in_FlipY) t.y = 1-t.y;\n" + " Texcoord = t*in_TexScaleOffset.xy + in_TexScaleOffset.zw;\n" + " gl_Position = vec4(in_Position*in_ScaleOffset.xy+in_ScaleOffset.zw, 0.0, 1.0);\n" + "}\n"; + GLuint vert = compile_shader(GL_VERTEX_SHADER, vert_src); + assert(vert != 0); + + const char *image_frag_src = + "#version 150 core\n" + "uniform sampler2D tex;\n" + "uniform vec4 in_ColorPrimary;\n" + "uniform vec4 in_ColorSecondary;\n" + "uniform vec4 in_ColorFill;\n" + "in vec2 Texcoord;\n" + "out vec4 out_Color;\n" + "void main() {\n" + " vec4 t = texture(tex, Texcoord);\n" + " out_Color.rgba = t;\n" + "}\n"; + // Simple 2-color decal shader + // - in_ColorFill is first pass + // - Red channel of the texture is used as primary color, mixed with 1-Red for + // secondary color. + // - Blue is a lazy alpha removal for now + // - Alpha channel passed through + const char *mask_frag_src = + "#version 150 core\n" + "uniform sampler2D tex;\n" + "uniform vec4 in_ColorPrimary;\n" + "uniform vec4 in_ColorSecondary;\n" + "uniform vec4 in_ColorFill;\n" + "in vec2 Texcoord;\n" + "out vec4 out_Color;\n" + "void main() {\n" + " vec4 t = texture(tex, Texcoord);\n" + " out_Color.rgba = in_ColorFill.rgba;\n" + " out_Color.rgb += mix(in_ColorSecondary.rgb, in_ColorPrimary.rgb, t.r);\n" + " out_Color.a += t.a - t.b;\n" + "}\n"; + + const char *frag_src; + if (type == SHADER_TYPE_MASK) { + frag_src = mask_frag_src; + } else if (type == SHADER_TYPE_BLIT) { + frag_src = image_frag_src; + } else if (type == SHADER_TYPE_LOGO) { + frag_src = xemu_logo_frag_src; + } else { + assert(0); + } + GLuint frag = compile_shader(GL_FRAGMENT_SHADER, frag_src); + assert(frag != 0); + + // Link vertex and fragment shaders + s->prog = glCreateProgram(); + glAttachShader(s->prog, vert); + glAttachShader(s->prog, frag); + glBindFragDataLocation(s->prog, 0, "out_Color"); + glLinkProgram(s->prog); + glUseProgram(s->prog); + + // Flag shaders for deletion when program is deleted + glDeleteShader(vert); + glDeleteShader(frag); + + s->FlipY_loc = glGetUniformLocation(s->prog, "in_FlipY"); + s->ScaleOffset_loc = glGetUniformLocation(s->prog, "in_ScaleOffset"); + s->TexScaleOffset_loc = glGetUniformLocation(s->prog, "in_TexScaleOffset"); + s->tex_loc = glGetUniformLocation(s->prog, "tex"); + s->ColorPrimary_loc = glGetUniformLocation(s->prog, "in_ColorPrimary"); + s->ColorSecondary_loc = glGetUniformLocation(s->prog, "in_ColorSecondary"); + s->ColorFill_loc = glGetUniformLocation(s->prog, "in_ColorFill"); + s->time_loc = glGetUniformLocation(s->prog, "iTime"); + s->scale_loc = glGetUniformLocation(s->prog, "scale"); + + // Create a vertex array object + glGenVertexArrays(1, &s->vao); + glBindVertexArray(s->vao); + + // Populate vertex buffer + glGenBuffers(1, &s->vbo); + glBindBuffer(GL_ARRAY_BUFFER, s->vbo); + const GLfloat verts[6][4] = { + // x y s t + { -1.0f, -1.0f, 0.0f, 0.0f }, // BL + { -1.0f, 1.0f, 0.0f, 1.0f }, // TL + { 1.0f, 1.0f, 1.0f, 1.0f }, // TR + { 1.0f, -1.0f, 1.0f, 0.0f }, // BR + }; + glBufferData(GL_ARRAY_BUFFER, sizeof(verts), verts, GL_STATIC_COPY); + + // Populate element buffer + glGenBuffers(1, &s->ebo); + glBindBuffer(GL_ELEMENT_ARRAY_BUFFER, s->ebo); + const GLint indicies[] = { 0, 1, 2, 3 }; + glBufferData(GL_ELEMENT_ARRAY_BUFFER, sizeof(indicies), indicies, GL_STATIC_DRAW); + + // Bind vertex position attribute + GLint pos_attr_loc = glGetAttribLocation(s->prog, "in_Position"); + glVertexAttribPointer(pos_attr_loc, 2, GL_FLOAT, GL_FALSE, 4*sizeof(GLfloat), (void*)0); + glEnableVertexAttribArray(pos_attr_loc); + + // Bind vertex texture coordinate attribute + GLint tex_attr_loc = glGetAttribLocation(s->prog, "in_Texcoord"); + if (tex_attr_loc >= 0) { + glVertexAttribPointer(tex_attr_loc, 2, GL_FLOAT, GL_FALSE, 4*sizeof(GLfloat), (void*)(2*sizeof(GLfloat))); + glEnableVertexAttribArray(tex_attr_loc); + } + + return s; +} + +GLuint load_texture_from_file(const char *name) +{ + GLuint tex; + glGenTextures(1, &tex); + assert(tex != 0); + glBindTexture(GL_TEXTURE_2D, tex); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAX_LEVEL, 0); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_BORDER); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_BORDER); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_LINEAR); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); + + // Flip vertically so textures are loaded according to GL convention. + stbi_set_flip_vertically_on_load(1); + + // Read file into memory + int width, height, channels = 0; + unsigned char *data = stbi_load(name, &width, &height, &channels, 4); + assert(data != NULL); + + // Upload texture data + glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, width, height, 0, GL_RGBA, GL_UNSIGNED_BYTE, data); + stbi_image_free(data); + + return tex; +} + +void render_decal( + struct decal_shader *s, + float x, float y, float w, float h, + float tex_x, float tex_y, float tex_w, float tex_h, + uint32_t primary, uint32_t secondary, uint32_t fill + ) +{ + GLint vp[4]; + glGetIntegerv(GL_VIEWPORT, vp); + float ww = vp[2], wh = vp[3]; + + x = (int)x; + y = (int)y; + w = (int)w; + h = (int)h; + tex_x = (int)tex_x; + tex_y = (int)tex_y; + tex_w = (int)tex_w; + tex_h = (int)tex_h; + + int tw_i, th_i; + glGetTexLevelParameteriv(GL_TEXTURE_2D, 0, GL_TEXTURE_WIDTH, &tw_i); + glGetTexLevelParameteriv(GL_TEXTURE_2D, 0, GL_TEXTURE_HEIGHT, &th_i); + float tw = tw_i, th = th_i; + + #define COL(color, c) (float)(((color)>>((c)*8)) & 0xff)/255.0 + glUniform1i(s->FlipY_loc, s->flip); + glUniform4f(s->ScaleOffset_loc, w/ww, h/wh, -1+((2*x+w)/ww), -1+((2*y+h)/wh)); + glUniform4f(s->TexScaleOffset_loc, tex_w/tw, tex_h/th, tex_x/tw, tex_y/th); + glUniform1i(s->tex_loc, 0); + glUniform4f(s->ColorPrimary_loc, COL(primary, 3), COL(primary, 2), COL(primary, 1), COL(primary, 0)); + glUniform4f(s->ColorSecondary_loc, COL(secondary, 3), COL(secondary, 2), COL(secondary, 1), COL(secondary, 0)); + glUniform4f(s->ColorFill_loc, COL(fill, 3), COL(fill, 2), COL(fill, 1), COL(fill, 0)); + if (s->time_loc >= 0) glUniform1f(s->time_loc, SDL_GetTicks()/1000.0f); + if (s->scale_loc >= 0) glUniform1f(s->scale_loc, s->scale); + #undef COL + glDrawElements(GL_TRIANGLE_FAN, 4, GL_UNSIGNED_INT, NULL); +} + +void render_decal_image( + struct decal_shader *s, + float x, float y, float w, float h, + float tex_x, float tex_y, float tex_w, float tex_h + ) +{ + GLint vp[4]; + glGetIntegerv(GL_VIEWPORT, vp); + float ww = vp[2], wh = vp[3]; + + int tw_i, th_i; + glGetTexLevelParameteriv(GL_TEXTURE_2D, 0, GL_TEXTURE_WIDTH, &tw_i); + glGetTexLevelParameteriv(GL_TEXTURE_2D, 0, GL_TEXTURE_HEIGHT, &th_i); + float tw = tw_i, th = th_i; + + glUniform1i(s->FlipY_loc, s->flip); + glUniform4f(s->ScaleOffset_loc, w/ww, h/wh, -1+((2*x+w)/ww), -1+((2*y+h)/wh)); + glUniform4f(s->TexScaleOffset_loc, tex_w/tw, tex_h/th, tex_x/tw, tex_y/th); + glUniform1i(s->tex_loc, 0); + glDrawElements(GL_TRIANGLE_FAN, 4, GL_UNSIGNED_INT, NULL); +} + +struct fbo *create_fbo(int width, int height) +{ + struct fbo *fbo = (struct fbo *)malloc(sizeof(struct fbo)); + assert(fbo != NULL); + + fbo->w = width; + fbo->h = height; + + // Allocate the texture + glGenTextures(1, &fbo->tex); + glBindTexture(GL_TEXTURE_2D, fbo->tex); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAX_LEVEL, 0); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_BORDER); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_BORDER); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR); + glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, GL_LINEAR); + glTexImage2D(GL_TEXTURE_2D, 0, GL_RGBA, fbo->w, fbo->h, 0, GL_RGBA, GL_UNSIGNED_BYTE, NULL); + + // Allocate the framebuffer object + glGenFramebuffers(1, &fbo->fbo); + glBindFramebuffer(GL_FRAMEBUFFER, fbo->fbo); + glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, fbo->tex, 0); + GLenum DrawBuffers[1] = {GL_COLOR_ATTACHMENT0}; + glDrawBuffers(1, DrawBuffers); + + return fbo; +} + +void render_to_default_fb(void) +{ + // Restore default framebuffer, viewport, blending funciton + glBindFramebuffer(GL_FRAMEBUFFER, main_fb); + glViewport(vp[0], vp[1], vp[2], vp[3]); + glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); +} + +GLuint render_to_fbo(struct fbo *fbo) +{ + glEnable(GL_BLEND); + glBindFramebuffer(GL_FRAMEBUFFER, fbo->fbo); + glViewport(0, 0, fbo->w, fbo->h); + glClearColor(0, 0, 0, 0.0); + glClear(GL_COLOR_BUFFER_BIT); + return fbo->tex; +} diff --git a/ui/xemu-shaders.h b/ui/xemu-shaders.h new file mode 100644 index 0000000000..5c58824f8e --- /dev/null +++ b/ui/xemu-shaders.h @@ -0,0 +1,99 @@ +/* + * xemu User Interface Rendering Helpers + * + * Copyright (C) 2020 Matt Borgerson + * + * This program is free software; you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation; either version 2 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program. If not, see . + */ + +#ifndef XEMU_SHADERS_H +#define XEMU_SHADERS_H + +#include +#include + +#include "stb_image.h" + +enum SHADER_TYPE { + SHADER_TYPE_BLIT, + SHADER_TYPE_MASK, + SHADER_TYPE_LOGO, +}; + +struct decal_shader +{ + int flip; + float scale; + float smoothing; + float outline_dist; + + // GL object handles + GLuint prog, vao, vbo, ebo; + + // Uniform locations + GLint Mat_loc; + GLint FlipY_loc; + GLint tex_loc; + GLint ScaleOffset_loc; + GLint TexScaleOffset_loc; + + GLint ColorPrimary_loc; + GLint ColorSecondary_loc; + GLint ColorFill_loc; + GLint time_loc; + GLint scale_loc; +}; + +struct fbo { + GLuint fbo; + GLuint tex; + int w, h; +}; + +#ifdef __cplusplus +extern "C" { +#endif + +extern GLuint main_fb; +extern GLint vp[4]; + +GLuint compile_shader(GLenum type, const char *src); + +struct decal_shader *create_decal_shader(enum SHADER_TYPE type); +void delete_decal_shader(struct decal_shader *s); + +GLuint load_texture_from_file(const char *name); + +struct fbo *create_fbo(int width, int height); +void render_to_default_fb(void); +GLuint render_to_fbo(struct fbo *fbo); + +void render_decal( + struct decal_shader *s, + float x, float y, float w, float h, + float tex_x, float tex_y, float tex_w, float tex_h, + uint32_t primary, uint32_t secondary, uint32_t fill + ); + +void render_decal_image( + struct decal_shader *s, + float x, float y, float w, float h, + float tex_x, float tex_y, float tex_w, float tex_h + ); + +#ifdef __cplusplus +} +#endif + +#endif