mirror of
https://github.com/Relintai/sfw.git
synced 2025-01-09 21:19:36 +01:00
172 lines
4.0 KiB
C++
172 lines
4.0 KiB
C++
#include "shader.h"
|
|
|
|
#include <vector>
|
|
#include <stdio.h>
|
|
|
|
bool Shader::bind() {
|
|
if (current_shader != this) {
|
|
glUseProgram(program);
|
|
|
|
current_shader = this;
|
|
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void Shader::unbind() {
|
|
if (current_shader == this) {
|
|
glUseProgram(0);
|
|
|
|
current_shader = NULL;
|
|
}
|
|
}
|
|
|
|
void Shader::compile() {
|
|
if (!program) {
|
|
program = glCreateProgram();
|
|
}
|
|
|
|
if (!vertex_shader) {
|
|
vertex_shader = glCreateShader(GL_VERTEX_SHADER);
|
|
}
|
|
|
|
if (!fragment_shader) {
|
|
fragment_shader = glCreateShader(GL_FRAGMENT_SHADER);
|
|
}
|
|
|
|
const GLchar **vertex_shader_source = get_vertex_shader_source();
|
|
|
|
glShaderSource(vertex_shader, 1, vertex_shader_source, NULL);
|
|
glCompileShader(vertex_shader);
|
|
|
|
GLint shader_compiled = GL_FALSE;
|
|
glGetShaderiv(vertex_shader, GL_COMPILE_STATUS, &shader_compiled);
|
|
if (shader_compiled != GL_TRUE) {
|
|
print_shader_errors(vertex_shader, "compiling Vertex Shader");
|
|
return;
|
|
}
|
|
|
|
glAttachShader(program, vertex_shader);
|
|
|
|
|
|
const GLchar **fragment_shader_source = get_fragment_shader_source();
|
|
|
|
glShaderSource(fragment_shader, 1, fragment_shader_source, NULL);
|
|
glCompileShader(fragment_shader);
|
|
|
|
shader_compiled = GL_FALSE;
|
|
glGetShaderiv(fragment_shader, GL_COMPILE_STATUS, &shader_compiled);
|
|
if (shader_compiled != GL_TRUE) {
|
|
print_shader_errors(fragment_shader, "compiling Fragment Shader");
|
|
return;
|
|
}
|
|
|
|
glAttachShader(program, fragment_shader);
|
|
|
|
glBindAttribLocation(program, ATTRIBUTE_POSITION, "a_position");
|
|
glBindAttribLocation(program, ATTRIBUTE_NORMAL, "a_normal");
|
|
glBindAttribLocation(program, ATTRIBUTE_COLOR, "a_color");
|
|
glBindAttribLocation(program, ATTRIBUTE_UV, "a_uv");
|
|
|
|
glLinkProgram(program);
|
|
|
|
GLint program_compiled = GL_FALSE;
|
|
glGetProgramiv(program, GL_LINK_STATUS, &program_compiled);
|
|
if (program_compiled != GL_TRUE) {
|
|
print_program_errors(program);
|
|
return;
|
|
}
|
|
}
|
|
void Shader::destroy() {
|
|
glDeleteShader(vertex_shader);
|
|
glDeleteShader(fragment_shader);
|
|
glDeleteProgram(program);
|
|
}
|
|
|
|
const GLchar **Shader::get_vertex_shader_source() {
|
|
return vertex_shader_source;
|
|
}
|
|
void Shader::set_vertex_shader_source(const GLchar **source) {
|
|
vertex_shader_source = source;
|
|
}
|
|
|
|
const GLchar **Shader::get_fragment_shader_source() {
|
|
return fragment_shader_source;
|
|
}
|
|
void Shader::set_fragment_shader_source(const GLchar **source) {
|
|
fragment_shader_source = source;
|
|
}
|
|
|
|
void Shader::print_shader_errors(const GLuint p_program, const char *name) {
|
|
int max_length = 5000;
|
|
std::vector<GLchar> error_log(max_length);
|
|
|
|
glGetShaderInfoLog(p_program, max_length, &max_length, &error_log[0]);
|
|
|
|
printf("Error %s!\n", name);
|
|
printf("%s!\n", &error_log[0]);
|
|
}
|
|
|
|
void Shader::print_program_errors(const GLuint p_program) {
|
|
if (glIsProgram(program)) {
|
|
int info_length = 0;
|
|
int max_length = 5000;
|
|
|
|
glGetProgramiv(p_program, GL_INFO_LOG_LENGTH, &info_length);
|
|
|
|
char *info_log = new char[max_length];
|
|
|
|
glGetProgramInfoLog(p_program, max_length, &info_length, info_log);
|
|
|
|
if (info_length > 0) {
|
|
printf("%s\n", info_log);
|
|
}
|
|
|
|
delete[] info_log;
|
|
} else {
|
|
printf("print_program_errors: Not a program!\n");
|
|
}
|
|
}
|
|
|
|
Shader::Shader() {
|
|
vertex_shader = 0;
|
|
fragment_shader = 0;
|
|
program = 0;
|
|
|
|
vertex_shader_source = NULL;
|
|
fragment_shader_source = NULL;
|
|
}
|
|
Shader::~Shader() {
|
|
destroy();
|
|
}
|
|
|
|
Shader *Shader::current_shader = NULL;
|
|
|
|
//Meyers singleton
|
|
//thread safe
|
|
ShaderCache *ShaderCache::get_singleton() {
|
|
static ShaderCache instance;
|
|
|
|
return &instance;
|
|
}
|
|
|
|
Shader *ShaderCache::get_shader(const int id) {
|
|
return shaders[id];
|
|
}
|
|
void ShaderCache::add_shader(const int id, Shader *shader) {
|
|
shaders[id] = shader;
|
|
}
|
|
|
|
ShaderCache::ShaderCache() {
|
|
}
|
|
ShaderCache::~ShaderCache() {
|
|
for (const std::pair<int, Shader *> n : shaders) {
|
|
delete n.second;
|
|
}
|
|
|
|
shaders.clear();
|
|
}
|
|
|