// Copyright (c) 2010 The Chromium OS Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// Adapted from the javascript implementation upon WebGL by kwaters@.
#include "shader.h"
#include <stdio.h>
#include <stdlib.h>
#include "shadersrc.h"
#undef IMPORTGL_API
#undef IMPORTGL_FNPTRINIT
#include "importgl.h"
SHADERLIT sShaderLit;
SHADERFLAT sShaderFlat;
SHADERFADE sShaderFade;
Matrix4x4 sModelView;
Matrix4x4 sProjection;
static void printShaderLog(GLuint shader)
{
int infoLogSize, infoWritten;
char *infoLog;
glGetShaderiv(shader, GL_INFO_LOG_LENGTH, &infoLogSize);
infoLog = malloc(infoLogSize);
glGetShaderInfoLog(shader, infoLogSize, &infoWritten, infoLog);
fprintf(stderr, "Error: glCompileShader failed: %s\n", infoLog);
free(infoLog);
}
static GLuint createShader(const char *src, GLenum shaderType)
{
GLint bShaderCompiled;
GLuint shader = glCreateShader(shaderType);
if (shader == 0)
return 0;
glShaderSource(shader, 1, &src, NULL);
glCompileShader(shader);
glGetShaderiv(shader, GL_COMPILE_STATUS, &bShaderCompiled);
if (!bShaderCompiled)
{
printShaderLog(shader);
glDeleteShader(shader);
return 0;
}
return shader;
}
static GLuint createProgram(const char *srcVertex, const char * srcFragment)
{
GLuint program = glCreateProgram();
if (program == 0)
return 0;
GLuint shaderVertex = createShader(srcVertex, GL_VERTEX_SHADER);
if (shaderVertex == 0)
{
glDeleteProgram(program);
return 0;
}
glAttachShader(program, shaderVertex);
glDeleteShader(shaderVertex);
GLuint shaderFragment = createShader(srcFragment, GL_FRAGMENT_SHADER);
if (shaderFragment == 0)
{
glDeleteProgram(program);
return 0;
}
glAttachShader(program, shaderFragment);
glDeleteShader(shaderFragment);
glLinkProgram(program);
return program;
}
static void computeNormalMatrix(Matrix4x4 m, Matrix3x3 normal)
{
float det = m[0*4+0] * (m[1*4+1] * m[2*4+2] - m[2*4+1] * m[1*4+2]) -
m[0*4+1] * (m[1*4+0] * m[2*4+2] - m[1*4+2] * m[2*4+0]) +
m[0*4+2] * (m[1*4+0] * m[2*4+1] - m[1*4+1] * m[2*4+0]);
float invDet = 1.f / det;
normal[0*3+0] = invDet * (m[1*4+1] * m[2*4+2] - m[2*4+1] * m[1*4+2]);
normal[1*3+0] = invDet * -(m[0*4+1] * m[2*4+2] - m[0*4+2] * m[2*4+1]);
normal[2*3+0] = invDet * (m[0*4+1] * m[1*4+2] - m[0*4+2] * m[1*4+1]);
normal[0*3+1] = invDet * -(m[1*4+0] * m[2*4+2] - m[1*4+2] * m[2*4+0]);
normal[1*3+1] = invDet * (m[0*4+0] * m[2*4+2] - m[0*4+2] * m[2*4+0]);
normal[2*3+1] = invDet * -(m[0*4+0] * m[1*4+2] - m[1*4+0] * m[0*4+2]);
normal[0*3+2] = invDet * (m[1*4+0] * m[2*4+1] - m[2*4+0] * m[1*4+1]);
normal[1*3+2] = invDet * -(m[0*4+0] * m[2*4+1] - m[2*4+0] * m[0*4+1]);
normal[2*3+2] = invDet * (m[0*4+0] * m[1*4+1] - m[1*4+0] * m[0*4+1]);
}
static int getLocations()
{
int rt = 1;
#define GET_ATTRIBUTE_LOC(programName, varName) \
sShader##programName.varName = \
glGetAttribLocation(sShader##programName.program, #varName); \
if (sShader##programName.varName == -1) rt = 0
#define GET_UNIFORM_LOC(programName, varName) \
sShader##programName.varName = \
glGetUniformLocation(sShader##programName.program, #varName); \
if (sShader##programName.varName == -1) rt = 0
GET_ATTRIBUTE_LOC(Lit, pos);
GET_ATTRIBUTE_LOC(Lit, normal);
GET_ATTRIBUTE_LOC(Lit, colorIn);
GET_UNIFORM_LOC(Lit, mvp);
GET_UNIFORM_LOC(Lit, normalMatrix);
GET_UNIFORM_LOC(Lit, ambient);
GET_UNIFORM_LOC(Lit, shininess);
GET_UNIFORM_LOC(Lit, light_0_direction);
GET_UNIFORM_LOC(Lit, light_0_diffuse);
GET_UNIFORM_LOC(Lit, light_0_specular);
GET_UNIFORM_LOC(Lit, light_1_direction);
GET_UNIFORM_LOC(Lit, light_1_diffuse);
GET_UNIFORM_LOC(Lit, light_2_direction);
GET_UNIFORM_LOC(Lit, light_2_diffuse);
GET_ATTRIBUTE_LOC(Flat, pos);
GET_ATTRIBUTE_LOC(Flat, colorIn);
GET_UNIFORM_LOC(Flat, mvp);
GET_ATTRIBUTE_LOC(Fade, pos);
GET_UNIFORM_LOC(Fade, minFade);
#undef GET_ATTRIBUTE_LOC
#undef GET_UNIFORM_LOC
return rt;
}
int initShaderPrograms()
{
Matrix4x4_LoadIdentity(sModelView);
Matrix4x4_LoadIdentity(sProjection);
sShaderFlat.program = createProgram(sFlatVertexSource,
sFlatFragmentSource);
sShaderLit.program = createProgram(sLitVertexSource,
sFlatFragmentSource);
sShaderFade.program = createProgram(sFadeVertexSource,
sFlatFragmentSource);
if (sShaderFlat.program == 0 || sShaderLit.program == 0 ||
sShaderFade.program == 0)
return 0;
return getLocations();
}
void deInitShaderPrograms()
{
glDeleteProgram(sShaderFlat.program);
glDeleteProgram(sShaderLit.program);
glDeleteProgram(sShaderFade.program);
}
void bindShaderProgram(GLuint program)
{
int loc_mvp = -1;
int loc_normalMatrix = -1;
glUseProgram(program);
if (program == sShaderLit.program)
{
loc_mvp = sShaderLit.mvp;
loc_normalMatrix = sShaderLit.normalMatrix;
}
else if (program == sShaderFlat.program)
{
loc_mvp = sShaderFlat.mvp;
}
if (loc_mvp != -1)
{
Matrix4x4 mvp;
Matrix4x4_Multiply(mvp, sModelView, sProjection);
glUniformMatrix4fv(loc_mvp, 1, GL_FALSE, (GLfloat *)mvp);
}
if (loc_normalMatrix != -1)
{
Matrix3x3 normalMatrix;
computeNormalMatrix(sModelView, normalMatrix);
glUniformMatrix3fv(loc_normalMatrix, 1, GL_FALSE,
(GLfloat *)normalMatrix);
}
}