#include "shader.h" #include #include 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 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 n : shaders) { delete n.second; } shaders.clear(); }