2016-02-07 17:32:38 -05:00
|
|
|
//
|
|
|
|
// Shader.cpp
|
|
|
|
// Clock Signal
|
|
|
|
//
|
|
|
|
// Created by Thomas Harte on 07/02/2016.
|
|
|
|
// Copyright © 2016 Thomas Harte. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "Shader.hpp"
|
2016-02-07 19:21:22 -05:00
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
|
|
|
|
using namespace OpenGL;
|
|
|
|
|
2016-05-01 11:07:49 -04:00
|
|
|
namespace {
|
|
|
|
Shader *bound_shader = nullptr;
|
|
|
|
}
|
|
|
|
|
2016-02-07 19:21:22 -05:00
|
|
|
GLuint Shader::compile_shader(const char *source, GLenum type)
|
|
|
|
{
|
|
|
|
GLuint shader = glCreateShader(type);
|
|
|
|
glShaderSource(shader, 1, &source, NULL);
|
|
|
|
glCompileShader(shader);
|
|
|
|
|
|
|
|
#if defined(DEBUG)
|
|
|
|
GLint isCompiled = 0;
|
|
|
|
glGetShaderiv(shader, GL_COMPILE_STATUS, &isCompiled);
|
|
|
|
if(isCompiled == GL_FALSE)
|
|
|
|
{
|
|
|
|
GLint logLength;
|
|
|
|
glGetShaderiv(shader, GL_INFO_LOG_LENGTH, &logLength);
|
2016-04-19 08:05:43 -04:00
|
|
|
if(logLength > 0) {
|
2016-02-07 19:21:22 -05:00
|
|
|
GLchar *log = (GLchar *)malloc((size_t)logLength);
|
|
|
|
glGetShaderInfoLog(shader, logLength, &logLength, log);
|
|
|
|
printf("Compile log:\n%s\n", log);
|
|
|
|
free(log);
|
|
|
|
}
|
2016-04-19 08:05:43 -04:00
|
|
|
|
|
|
|
throw (type == GL_VERTEX_SHADER) ? VertexShaderCompilationError : FragmentShaderCompilationError;
|
2016-02-07 19:21:22 -05:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return shader;
|
|
|
|
}
|
|
|
|
|
2016-04-19 07:23:15 -04:00
|
|
|
Shader::Shader(const char *vertex_shader, const char *fragment_shader, const AttributeBinding *attribute_bindings)
|
2016-02-07 19:21:22 -05:00
|
|
|
{
|
|
|
|
_shader_program = glCreateProgram();
|
|
|
|
GLuint vertex = compile_shader(vertex_shader, GL_VERTEX_SHADER);
|
|
|
|
GLuint fragment = compile_shader(fragment_shader, GL_FRAGMENT_SHADER);
|
|
|
|
|
|
|
|
glAttachShader(_shader_program, vertex);
|
|
|
|
glAttachShader(_shader_program, fragment);
|
2016-04-19 07:23:15 -04:00
|
|
|
|
|
|
|
if(attribute_bindings)
|
|
|
|
{
|
|
|
|
while(attribute_bindings->name)
|
|
|
|
{
|
|
|
|
glBindAttribLocation(_shader_program, attribute_bindings->index, attribute_bindings->name);
|
|
|
|
attribute_bindings++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-07 19:21:22 -05:00
|
|
|
glLinkProgram(_shader_program);
|
2016-04-18 21:32:48 -04:00
|
|
|
|
|
|
|
#if defined(DEBUG)
|
|
|
|
GLint didLink = 0;
|
|
|
|
glGetProgramiv(_shader_program, GL_LINK_STATUS, &didLink);
|
|
|
|
if(didLink == GL_FALSE)
|
|
|
|
{
|
|
|
|
GLint logLength;
|
|
|
|
glGetProgramiv(_shader_program, GL_INFO_LOG_LENGTH, &logLength);
|
2016-04-24 22:32:24 -04:00
|
|
|
if(logLength > 0) {
|
2016-04-18 21:32:48 -04:00
|
|
|
GLchar *log = (GLchar *)malloc((size_t)logLength);
|
|
|
|
glGetProgramInfoLog(_shader_program, logLength, &logLength, log);
|
|
|
|
printf("Link log:\n%s\n", log);
|
|
|
|
free(log);
|
|
|
|
}
|
2016-04-19 08:05:43 -04:00
|
|
|
throw ProgramLinkageError;
|
2016-04-18 21:32:48 -04:00
|
|
|
}
|
|
|
|
#endif
|
2016-02-07 19:21:22 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
Shader::~Shader()
|
|
|
|
{
|
2016-05-01 11:07:49 -04:00
|
|
|
if(bound_shader == this) Shader::unbind();
|
2016-04-24 20:34:25 -04:00
|
|
|
glDeleteProgram(_shader_program);
|
2016-02-07 19:21:22 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void Shader::bind()
|
|
|
|
{
|
2016-05-01 11:07:49 -04:00
|
|
|
if(bound_shader != this)
|
|
|
|
{
|
|
|
|
glUseProgram(_shader_program);
|
|
|
|
bound_shader = this;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Shader::unbind()
|
|
|
|
{
|
|
|
|
bound_shader = nullptr;
|
|
|
|
glUseProgram(0);
|
2016-02-07 19:21:22 -05:00
|
|
|
}
|
|
|
|
|
2016-04-19 07:23:15 -04:00
|
|
|
GLint Shader::get_attrib_location(const GLchar *name)
|
2016-02-07 19:21:22 -05:00
|
|
|
{
|
|
|
|
return glGetAttribLocation(_shader_program, name);
|
|
|
|
}
|
|
|
|
|
2016-04-19 07:23:15 -04:00
|
|
|
GLint Shader::get_uniform_location(const GLchar *name)
|
2016-02-07 19:21:22 -05:00
|
|
|
{
|
|
|
|
return glGetUniformLocation(_shader_program, name);
|
|
|
|
}
|