2012-03-24 22:39:19 +00:00
|
|
|
#include <set>
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
|
|
|
|
#include "base/logging.h"
|
|
|
|
#include "file/vfs.h"
|
2013-10-13 22:11:25 +00:00
|
|
|
#include "file/zip_read.h"
|
2012-11-26 10:27:35 +00:00
|
|
|
#include "glsl_program.h"
|
2012-03-24 22:39:19 +00:00
|
|
|
|
|
|
|
static std::set<GLSLProgram *> active_programs;
|
|
|
|
|
2013-10-13 09:56:42 +00:00
|
|
|
bool CompileShader(const char *source, GLuint shader, const char *filename, std::string *error_message) {
|
2012-10-30 12:20:55 +00:00
|
|
|
glShaderSource(shader, 1, &source, NULL);
|
|
|
|
glCompileShader(shader);
|
|
|
|
GLint success;
|
|
|
|
glGetShaderiv(shader, GL_COMPILE_STATUS, &success);
|
|
|
|
if (!success) {
|
2012-03-24 22:39:19 +00:00
|
|
|
#define MAX_INFO_LOG_SIZE 2048
|
2012-10-30 12:20:55 +00:00
|
|
|
GLchar infoLog[MAX_INFO_LOG_SIZE];
|
|
|
|
GLsizei len;
|
|
|
|
glGetShaderInfoLog(shader, MAX_INFO_LOG_SIZE, &len, infoLog);
|
|
|
|
infoLog[len] = '\0';
|
|
|
|
ELOG("Error in shader compilation of %s!\n", filename);
|
|
|
|
ELOG("Info log: %s\n", infoLog);
|
|
|
|
ELOG("Shader source:\n%s\n", (const char *)source);
|
2013-10-13 09:56:42 +00:00
|
|
|
if (error_message)
|
|
|
|
*error_message = infoLog;
|
2012-10-30 12:20:55 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
2012-03-24 22:39:19 +00:00
|
|
|
}
|
|
|
|
|
2013-10-13 09:56:42 +00:00
|
|
|
GLSLProgram *glsl_create(const char *vshader, const char *fshader, std::string *error_message) {
|
2012-10-30 12:20:55 +00:00
|
|
|
GLSLProgram *program = new GLSLProgram();
|
|
|
|
program->program_ = 0;
|
|
|
|
program->vsh_ = 0;
|
|
|
|
program->fsh_ = 0;
|
|
|
|
program->vshader_source = 0;
|
|
|
|
program->fshader_source = 0;
|
|
|
|
strcpy(program->name, vshader + strlen(vshader) - 15);
|
|
|
|
strcpy(program->vshader_filename, vshader);
|
|
|
|
strcpy(program->fshader_filename, fshader);
|
2013-10-13 09:56:42 +00:00
|
|
|
if (glsl_recompile(program, error_message)) {
|
2012-10-30 12:20:55 +00:00
|
|
|
active_programs.insert(program);
|
2013-10-13 09:56:42 +00:00
|
|
|
} else {
|
|
|
|
ELOG("Failed compiling GLSL program: %s %s", vshader, fshader);
|
|
|
|
delete program;
|
|
|
|
return 0;
|
2012-10-30 12:20:55 +00:00
|
|
|
}
|
|
|
|
return program;
|
2012-08-27 21:28:52 +00:00
|
|
|
}
|
|
|
|
|
2013-10-13 09:56:42 +00:00
|
|
|
GLSLProgram *glsl_create_source(const char *vshader_src, const char *fshader_src, std::string *error_message) {
|
2012-10-30 12:20:55 +00:00
|
|
|
GLSLProgram *program = new GLSLProgram();
|
|
|
|
program->program_ = 0;
|
|
|
|
program->vsh_ = 0;
|
|
|
|
program->fsh_ = 0;
|
|
|
|
program->vshader_source = vshader_src;
|
|
|
|
program->fshader_source = fshader_src;
|
|
|
|
strcpy(program->name, "[srcshader]");
|
|
|
|
strcpy(program->vshader_filename, "");
|
|
|
|
strcpy(program->fshader_filename, "");
|
2013-10-13 09:56:42 +00:00
|
|
|
if (glsl_recompile(program, error_message)) {
|
2012-10-30 12:20:55 +00:00
|
|
|
active_programs.insert(program);
|
2013-10-13 09:56:42 +00:00
|
|
|
} else {
|
|
|
|
ELOG("Failed compiling GLSL program from source strings");
|
|
|
|
delete program;
|
|
|
|
return 0;
|
2012-10-30 12:20:55 +00:00
|
|
|
}
|
|
|
|
return program;
|
2012-03-24 22:39:19 +00:00
|
|
|
}
|
|
|
|
|
2015-01-17 19:59:27 +00:00
|
|
|
// Not wanting to change ReadLocalFile semantics.
|
2016-10-11 16:48:49 +00:00
|
|
|
// TODO: Use C++11 unique_ptr, remove delete[]
|
2015-01-17 19:59:27 +00:00
|
|
|
struct AutoCharArrayBuf {
|
|
|
|
AutoCharArrayBuf(char *buf = nullptr) : buf_(buf) {
|
|
|
|
}
|
|
|
|
~AutoCharArrayBuf() {
|
|
|
|
delete [] buf_;
|
|
|
|
buf_ = nullptr;
|
|
|
|
}
|
|
|
|
void reset(char *buf) {
|
|
|
|
if (buf_) {
|
|
|
|
delete[] buf_;
|
|
|
|
}
|
|
|
|
buf_ = buf;
|
|
|
|
}
|
|
|
|
operator char *() {
|
|
|
|
return buf_;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
char *buf_;
|
|
|
|
};
|
|
|
|
|
2013-10-13 09:56:42 +00:00
|
|
|
bool glsl_recompile(GLSLProgram *program, std::string *error_message) {
|
2012-10-30 12:20:55 +00:00
|
|
|
struct stat vs, fs;
|
2015-01-17 19:59:27 +00:00
|
|
|
AutoCharArrayBuf vsh_src, fsh_src;
|
2013-10-13 22:11:25 +00:00
|
|
|
|
|
|
|
if (strlen(program->vshader_filename) > 0 && 0 == stat(program->vshader_filename, &vs)) {
|
2012-10-30 12:20:55 +00:00
|
|
|
program->vshader_mtime = vs.st_mtime;
|
2013-10-13 22:11:25 +00:00
|
|
|
if (!program->vshader_source) {
|
|
|
|
size_t sz;
|
2015-01-17 19:59:27 +00:00
|
|
|
vsh_src.reset((char *)ReadLocalFile(program->vshader_filename, &sz));
|
2013-10-13 22:11:25 +00:00
|
|
|
}
|
|
|
|
} else {
|
2012-10-30 12:20:55 +00:00
|
|
|
program->vshader_mtime = 0;
|
2013-10-13 22:11:25 +00:00
|
|
|
}
|
2012-10-30 12:20:55 +00:00
|
|
|
|
2013-10-13 22:11:25 +00:00
|
|
|
if (strlen(program->fshader_filename) > 0 && 0 == stat(program->fshader_filename, &fs)) {
|
2012-10-30 12:20:55 +00:00
|
|
|
program->fshader_mtime = fs.st_mtime;
|
2013-10-13 22:11:25 +00:00
|
|
|
if (!program->fshader_source) {
|
|
|
|
size_t sz;
|
2015-01-17 19:59:27 +00:00
|
|
|
fsh_src.reset((char *)ReadLocalFile(program->fshader_filename, &sz));
|
2013-10-13 22:11:25 +00:00
|
|
|
}
|
|
|
|
} else {
|
2012-10-30 12:20:55 +00:00
|
|
|
program->fshader_mtime = 0;
|
2013-10-13 22:11:25 +00:00
|
|
|
}
|
2012-10-30 12:20:55 +00:00
|
|
|
|
2013-10-13 22:11:25 +00:00
|
|
|
if (!program->vshader_source && !vsh_src) {
|
2012-10-30 12:20:55 +00:00
|
|
|
size_t sz;
|
2015-01-17 19:59:27 +00:00
|
|
|
vsh_src.reset((char *)VFSReadFile(program->vshader_filename, &sz));
|
2013-10-13 22:11:25 +00:00
|
|
|
}
|
|
|
|
if (!program->vshader_source && !vsh_src) {
|
|
|
|
ELOG("File missing: %s", program->vshader_filename);
|
|
|
|
if (error_message) {
|
|
|
|
*error_message = std::string("File missing: ") + program->vshader_filename;
|
2012-10-30 12:20:55 +00:00
|
|
|
}
|
2013-10-13 22:11:25 +00:00
|
|
|
return false;
|
2012-10-30 12:20:55 +00:00
|
|
|
}
|
2013-10-13 22:11:25 +00:00
|
|
|
if (!program->fshader_source && !fsh_src) {
|
2012-10-30 12:20:55 +00:00
|
|
|
size_t sz;
|
2015-01-17 19:59:27 +00:00
|
|
|
fsh_src.reset((char *)VFSReadFile(program->fshader_filename, &sz));
|
2013-10-13 22:11:25 +00:00
|
|
|
}
|
|
|
|
if (!program->fshader_source && !fsh_src) {
|
|
|
|
ELOG("File missing: %s", program->fshader_filename);
|
|
|
|
if (error_message) {
|
|
|
|
*error_message = std::string("File missing: ") + program->fshader_filename;
|
2012-10-30 12:20:55 +00:00
|
|
|
}
|
2013-10-13 22:11:25 +00:00
|
|
|
return false;
|
2012-10-30 12:20:55 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
GLuint vsh = glCreateShader(GL_VERTEX_SHADER);
|
|
|
|
const GLchar *vsh_str = program->vshader_source ? program->vshader_source : (const GLchar *)(vsh_src);
|
2013-10-13 09:56:42 +00:00
|
|
|
if (!CompileShader(vsh_str, vsh, program->vshader_filename, error_message)) {
|
2012-10-30 12:20:55 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
const GLchar *fsh_str = program->fshader_source ? program->fshader_source : (const GLchar *)(fsh_src);
|
|
|
|
GLuint fsh = glCreateShader(GL_FRAGMENT_SHADER);
|
2013-10-13 09:56:42 +00:00
|
|
|
if (!CompileShader(fsh_str, fsh, program->fshader_filename, error_message)) {
|
2012-10-30 12:20:55 +00:00
|
|
|
glDeleteShader(vsh);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
GLuint prog = glCreateProgram();
|
|
|
|
glAttachShader(prog, vsh);
|
|
|
|
glAttachShader(prog, fsh);
|
|
|
|
|
|
|
|
glLinkProgram(prog);
|
|
|
|
|
|
|
|
GLint linkStatus;
|
|
|
|
glGetProgramiv(prog, GL_LINK_STATUS, &linkStatus);
|
2013-07-26 20:21:09 +00:00
|
|
|
if (linkStatus == GL_FALSE) {
|
2012-10-30 12:20:55 +00:00
|
|
|
GLint bufLength = 0;
|
|
|
|
glGetProgramiv(prog, GL_INFO_LOG_LENGTH, &bufLength);
|
|
|
|
if (bufLength) {
|
2013-10-13 09:56:42 +00:00
|
|
|
char* buf = new char[bufLength + 1]; // safety
|
2012-10-30 12:20:55 +00:00
|
|
|
glGetProgramInfoLog(prog, bufLength, NULL, buf);
|
2013-07-26 20:21:09 +00:00
|
|
|
ILOG("vsh: %i fsh: %i", vsh, fsh);
|
2013-10-13 09:56:42 +00:00
|
|
|
ELOG("Could not link shader program (linkstatus=%i):\n %s \n", linkStatus, buf);
|
|
|
|
if (error_message) {
|
|
|
|
*error_message = buf;
|
|
|
|
}
|
|
|
|
delete [] buf;
|
2012-10-30 12:20:55 +00:00
|
|
|
} else {
|
2013-07-26 20:21:09 +00:00
|
|
|
ILOG("vsh: %i fsh: %i", vsh, fsh);
|
2013-10-13 09:56:42 +00:00
|
|
|
ELOG("Could not link shader program (linkstatus=%i). No OpenGL error log was available.", linkStatus);
|
|
|
|
if (error_message) {
|
|
|
|
*error_message = "(no error message available)";
|
|
|
|
}
|
2012-10-30 12:20:55 +00:00
|
|
|
}
|
|
|
|
glDeleteShader(vsh);
|
|
|
|
glDeleteShader(fsh);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Destroy the old program, if any.
|
|
|
|
if (program->program_) {
|
|
|
|
glDeleteProgram(program->program_);
|
|
|
|
}
|
|
|
|
|
|
|
|
program->program_ = prog;
|
|
|
|
program->vsh_ = vsh;
|
2015-01-25 04:33:38 +00:00
|
|
|
program->fsh_ = fsh;
|
2012-10-30 12:20:55 +00:00
|
|
|
|
|
|
|
program->sampler0 = glGetUniformLocation(program->program_, "sampler0");
|
|
|
|
program->sampler1 = glGetUniformLocation(program->program_, "sampler1");
|
|
|
|
|
|
|
|
program->a_position = glGetAttribLocation(program->program_, "a_position");
|
|
|
|
program->a_color = glGetAttribLocation(program->program_, "a_color");
|
|
|
|
program->a_normal = glGetAttribLocation(program->program_, "a_normal");
|
|
|
|
program->a_texcoord0 = glGetAttribLocation(program->program_, "a_texcoord0");
|
|
|
|
program->a_texcoord1 = glGetAttribLocation(program->program_, "a_texcoord1");
|
|
|
|
|
|
|
|
program->u_worldviewproj = glGetUniformLocation(program->program_, "u_worldviewproj");
|
|
|
|
program->u_world = glGetUniformLocation(program->program_, "u_world");
|
|
|
|
program->u_viewproj = glGetUniformLocation(program->program_, "u_viewproj");
|
|
|
|
program->u_fog = glGetUniformLocation(program->program_, "u_fog");
|
|
|
|
program->u_sundir = glGetUniformLocation(program->program_, "u_sundir");
|
|
|
|
program->u_camerapos = glGetUniformLocation(program->program_, "u_camerapos");
|
|
|
|
|
|
|
|
//ILOG("Shader compilation success: %s %s",
|
|
|
|
// program->vshader_filename,
|
|
|
|
// program->fshader_filename);
|
|
|
|
return true;
|
2012-03-24 22:39:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int glsl_attrib_loc(const GLSLProgram *program, const char *name) {
|
2012-10-30 12:20:55 +00:00
|
|
|
return glGetAttribLocation(program->program_, name);
|
2012-03-24 22:39:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
int glsl_uniform_loc(const GLSLProgram *program, const char *name) {
|
2012-10-30 12:20:55 +00:00
|
|
|
return glGetUniformLocation(program->program_, name);
|
2012-03-24 22:39:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void glsl_destroy(GLSLProgram *program) {
|
2013-07-27 18:07:02 +00:00
|
|
|
if (program) {
|
|
|
|
glDeleteShader(program->vsh_);
|
|
|
|
glDeleteShader(program->fsh_);
|
|
|
|
glDeleteProgram(program->program_);
|
|
|
|
active_programs.erase(program);
|
|
|
|
} else {
|
|
|
|
ELOG("Deleting null GLSL program!");
|
|
|
|
}
|
2012-10-30 12:20:55 +00:00
|
|
|
delete program;
|
2012-03-24 22:39:19 +00:00
|
|
|
}
|
|
|
|
|
2017-02-15 11:32:48 +00:00
|
|
|
static const GLSLProgram *curProgram;
|
|
|
|
|
2012-03-24 22:39:19 +00:00
|
|
|
void glsl_bind(const GLSLProgram *program) {
|
2014-04-08 14:08:38 +00:00
|
|
|
if (program)
|
|
|
|
glUseProgram(program->program_);
|
|
|
|
else
|
|
|
|
glUseProgram(0);
|
2017-02-15 11:32:48 +00:00
|
|
|
curProgram = program;
|
2012-03-24 22:39:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void glsl_unbind() {
|
2012-10-30 12:20:55 +00:00
|
|
|
glUseProgram(0);
|
2017-02-15 11:32:48 +00:00
|
|
|
curProgram = nullptr;
|
2012-03-24 22:39:19 +00:00
|
|
|
}
|
2017-02-15 11:32:48 +00:00
|
|
|
|
|
|
|
const GLSLProgram *glsl_get_program() {
|
|
|
|
return curProgram;
|
2017-03-18 14:20:36 +00:00
|
|
|
}
|