dungeon_crawler
changeset 15:3a3236a4833c
adding shaders and renderer abstraction
author | John Tsiombikas <nuclear@member.fsf.org> |
---|---|
date | Sun, 19 Aug 2012 23:09:30 +0300 (2012-08-19) |
parents | 67ae9fcb802c |
children | 91180ee7b7d9 |
files | prototype/src/main.cc prototype/src/renderer.cc prototype/src/renderer.h prototype/src/sdr.c prototype/src/sdr.h |
diffstat | 5 files changed, 483 insertions(+), 1 deletions(-) [+] |
line diff
1.1 --- a/prototype/src/main.cc Sun Aug 19 06:30:18 2012 +0300 1.2 +++ b/prototype/src/main.cc Sun Aug 19 23:09:30 2012 +0300 1.3 @@ -7,11 +7,13 @@ 1.4 #include "camera.h" 1.5 #include "datapath.h" 1.6 #include "tileset.h" 1.7 +#include "renderer.h" 1.8 1.9 bool init(); 1.10 void cleanup(); 1.11 void idle(); 1.12 void disp(); 1.13 +void draw(); 1.14 void update(unsigned long msec); 1.15 void reshape(int x, int y); 1.16 void keyb(unsigned char key, int x, int y); 1.17 @@ -70,6 +72,10 @@ 1.18 1.19 add_data_path("data"); 1.20 1.21 + if(!init_renderer()) { 1.22 + return false; 1.23 + } 1.24 + 1.25 // load a tileset 1.26 tileset = new TileSet; 1.27 if(!tileset->load(datafile_path("default.tileset"))) { 1.28 @@ -88,6 +94,14 @@ 1.29 return true; 1.30 } 1.31 1.32 +void cleanup() 1.33 +{ 1.34 + delete level; 1.35 + delete tileset; 1.36 + 1.37 + destroy_renderer(); 1.38 +} 1.39 + 1.40 void idle() 1.41 { 1.42 glutPostRedisplay(); 1.43 @@ -103,7 +117,7 @@ 1.44 glLoadIdentity(); 1.45 cam.use_inverse(); 1.46 1.47 - level->draw(); 1.48 + render_deferred(draw); 1.49 1.50 glutSwapBuffers(); 1.51 assert(glGetError() == GL_NO_ERROR); 1.52 @@ -111,6 +125,11 @@ 1.53 usleep(10000); 1.54 } 1.55 1.56 +void draw() 1.57 +{ 1.58 + level->draw(); 1.59 +} 1.60 + 1.61 void update(unsigned long msec) 1.62 { 1.63 static unsigned long last_upd;
2.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 2.2 +++ b/prototype/src/renderer.cc Sun Aug 19 23:09:30 2012 +0300 2.3 @@ -0,0 +1,77 @@ 2.4 +#include <stdio.h> 2.5 +#include <stdlib.h> 2.6 +#include <string.h> 2.7 +#include <limits.h> 2.8 +#include "opengl.h" 2.9 +#include "renderer.h" 2.10 +#include "sdr.h" 2.11 +#include "datapath.h" 2.12 + 2.13 +#define MRT_COUNT 4 2.14 + 2.15 +static const char *mrt_sdr_file[][2] = { 2.16 + {0, "mrt0.p.glsl"}, 2.17 + {0, "mrt1.p.glsl"}, 2.18 + {0, "mrt2.p.glsl"}, 2.19 + {0, "mrt3.p.glsl"} 2.20 +}; 2.21 + 2.22 +static unsigned int mrt_prog[MRT_COUNT]; 2.23 +static unsigned int deferred_prog; 2.24 + 2.25 +bool init_renderer() 2.26 +{ 2.27 + char vsfile[PATH_MAX], psfile[PATH_MAX]; 2.28 + const char *fname; 2.29 + 2.30 + for(int i=0; i<MRT_COUNT; i++) { 2.31 + if((fname = datafile_path(mrt_sdr_file[i][0]))) { 2.32 + strcpy(vsfile, mrt_sdr_file[i][0]); 2.33 + } else { 2.34 + vsfile[0] = 0; 2.35 + } 2.36 + 2.37 + if((fname = datafile_path(mrt_sdr_file[i][1]))) { 2.38 + strcpy(psfile, mrt_sdr_file[i][1]); 2.39 + } else { 2.40 + psfile[0] = 0; 2.41 + } 2.42 + 2.43 + if(!(mrt_prog[i] = create_program_load(vsfile, psfile))) { 2.44 + fprintf(stderr, "failed to load MRT program\n"); 2.45 + return false; 2.46 + } 2.47 + } 2.48 + 2.49 + if((fname = datafile_path("deferred.v.glsl"))) { 2.50 + strcpy(vsfile, fname); 2.51 + } else { 2.52 + vsfile[0] = 0; 2.53 + } 2.54 + if((fname = datafile_path("deferred.p.glsl"))) { 2.55 + strcpy(psfile, fname); 2.56 + } else { 2.57 + psfile[0] = 0; 2.58 + } 2.59 + 2.60 + if(!(deferred_prog = create_program_load(vsfile, psfile))) { 2.61 + fprintf(stderr, "failed to load deferred shader program\n"); 2.62 + return false; 2.63 + } 2.64 + return true; 2.65 +} 2.66 + 2.67 +void destroy_renderer() 2.68 +{ 2.69 + for(int i=0; i<MRT_COUNT; i++) { 2.70 + free_program(mrt_prog[i]); 2.71 + } 2.72 + free_program(deferred_prog); 2.73 +} 2.74 + 2.75 +void render_deferred(void (*draw_func)()) 2.76 +{ 2.77 + 2.78 + 2.79 + draw_func(); 2.80 +}
3.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 3.2 +++ b/prototype/src/renderer.h Sun Aug 19 23:09:30 2012 +0300 3.3 @@ -0,0 +1,9 @@ 3.4 +#ifndef RENDERER_H_ 3.5 +#define RENDERER_H_ 3.6 + 3.7 +bool init_renderer(); 3.8 +void destroy_renderer(); 3.9 + 3.10 +void render_deferred(void (*draw_func)()); 3.11 + 3.12 +#endif // RENDERER_H_
4.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 4.2 +++ b/prototype/src/sdr.c Sun Aug 19 23:09:30 2012 +0300 4.3 @@ -0,0 +1,328 @@ 4.4 +#include <stdio.h> 4.5 +#include <stdlib.h> 4.6 +#include <string.h> 4.7 +#include <errno.h> 4.8 +#include <assert.h> 4.9 +#include <GL/glew.h> 4.10 + 4.11 +#if defined(unix) || defined(__unix__) 4.12 +#include <unistd.h> 4.13 +#include <sys/stat.h> 4.14 +#endif /* unix */ 4.15 + 4.16 +#include "sdr.h" 4.17 + 4.18 +unsigned int create_vertex_shader(const char *src) 4.19 +{ 4.20 + return create_shader(src, GL_VERTEX_SHADER); 4.21 +} 4.22 + 4.23 +unsigned int create_pixel_shader(const char *src) 4.24 +{ 4.25 + return create_shader(src, GL_FRAGMENT_SHADER); 4.26 +} 4.27 + 4.28 +unsigned int create_shader(const char *src, unsigned int sdr_type) 4.29 +{ 4.30 + unsigned int sdr; 4.31 + int success, info_len; 4.32 + char *info_str = 0; 4.33 + GLenum err; 4.34 + 4.35 + sdr = glCreateShader(sdr_type); 4.36 + assert(glGetError() == GL_NO_ERROR); 4.37 + glShaderSource(sdr, 1, &src, 0); 4.38 + err = glGetError(); 4.39 + assert(err == GL_NO_ERROR); 4.40 + glCompileShader(sdr); 4.41 + assert(glGetError() == GL_NO_ERROR); 4.42 + 4.43 + glGetShaderiv(sdr, GL_COMPILE_STATUS, &success); 4.44 + assert(glGetError() == GL_NO_ERROR); 4.45 + glGetShaderiv(sdr, GL_INFO_LOG_LENGTH, &info_len); 4.46 + assert(glGetError() == GL_NO_ERROR); 4.47 + 4.48 + if(info_len) { 4.49 + if((info_str = malloc(info_len + 1))) { 4.50 + glGetShaderInfoLog(sdr, info_len, 0, info_str); 4.51 + assert(glGetError() == GL_NO_ERROR); 4.52 + } 4.53 + } 4.54 + 4.55 + if(success) { 4.56 + fprintf(stderr, info_str ? "done: %s\n" : "done\n", info_str); 4.57 + } else { 4.58 + fprintf(stderr, info_str ? "failed: %s\n" : "failed\n", info_str); 4.59 + glDeleteShader(sdr); 4.60 + sdr = 0; 4.61 + } 4.62 + 4.63 + free(info_str); 4.64 + return sdr; 4.65 +} 4.66 + 4.67 +void free_shader(unsigned int sdr) 4.68 +{ 4.69 + glDeleteShader(sdr); 4.70 +} 4.71 + 4.72 +unsigned int load_vertex_shader(const char *fname) 4.73 +{ 4.74 + return load_shader(fname, GL_VERTEX_SHADER); 4.75 +} 4.76 + 4.77 +unsigned int load_pixel_shader(const char *fname) 4.78 +{ 4.79 + return load_shader(fname, GL_FRAGMENT_SHADER); 4.80 +} 4.81 + 4.82 +unsigned int load_shader(const char *fname, unsigned int sdr_type) 4.83 +{ 4.84 +#if defined(unix) || defined(__unix__) 4.85 + struct stat st; 4.86 +#endif 4.87 + unsigned int sdr; 4.88 + size_t filesize; 4.89 + FILE *fp; 4.90 + char *src; 4.91 + 4.92 + if(!(fp = fopen(fname, "r"))) { 4.93 + fprintf(stderr, "failed to open shader %s: %s\n", fname, strerror(errno)); 4.94 + return 0; 4.95 + } 4.96 + 4.97 +#if defined(unix) || defined(__unix__) 4.98 + fstat(fileno(fp), &st); 4.99 + filesize = st.st_size; 4.100 +#else 4.101 + fseek(fp, 0, SEEK_END); 4.102 + filesize = ftell(fp); 4.103 + fseek(fp, 0, SEEK_SET); 4.104 +#endif /* unix */ 4.105 + 4.106 + if(!(src = malloc(filesize + 1))) { 4.107 + fclose(fp); 4.108 + return 0; 4.109 + } 4.110 + fread(src, 1, filesize, fp); 4.111 + src[filesize] = 0; 4.112 + fclose(fp); 4.113 + 4.114 + fprintf(stderr, "compiling %s shader: %s... ", (sdr_type == GL_VERTEX_SHADER ? "vertex" : "pixel"), fname); 4.115 + sdr = create_shader(src, sdr_type); 4.116 + 4.117 + free(src); 4.118 + return sdr; 4.119 +} 4.120 + 4.121 + 4.122 +unsigned int get_vertex_shader(const char *fname) 4.123 +{ 4.124 + return get_shader(fname, GL_VERTEX_SHADER); 4.125 +} 4.126 + 4.127 +unsigned int get_pixel_shader(const char *fname) 4.128 +{ 4.129 + return get_shader(fname, GL_FRAGMENT_SHADER); 4.130 +} 4.131 + 4.132 +unsigned int get_shader(const char *fname, unsigned int sdr_type) 4.133 +{ 4.134 + unsigned int sdr; 4.135 + if(!(sdr = load_shader(fname, sdr_type))) { 4.136 + return 0; 4.137 + } 4.138 + return sdr; 4.139 +} 4.140 + 4.141 + 4.142 +/* ---- gpu programs ---- */ 4.143 + 4.144 +unsigned int create_program(void) 4.145 +{ 4.146 + unsigned int prog = glCreateProgram(); 4.147 + assert(glGetError() == GL_NO_ERROR); 4.148 + return prog; 4.149 +} 4.150 + 4.151 +unsigned int create_program_link(unsigned int vs, unsigned int ps) 4.152 +{ 4.153 + unsigned int prog; 4.154 + 4.155 + if(!(prog = create_program())) { 4.156 + return 0; 4.157 + } 4.158 + 4.159 + if(vs) { 4.160 + attach_shader(prog, vs); 4.161 + assert(glGetError() == GL_NO_ERROR); 4.162 + } 4.163 + if(ps) { 4.164 + attach_shader(prog, ps); 4.165 + assert(glGetError() == GL_NO_ERROR); 4.166 + } 4.167 + 4.168 + if(link_program(prog) == -1) { 4.169 + free_program(prog); 4.170 + return 0; 4.171 + } 4.172 + return prog; 4.173 +} 4.174 + 4.175 +unsigned int create_program_load(const char *vfile, const char *pfile) 4.176 +{ 4.177 + unsigned int vs = 0, ps = 0; 4.178 + 4.179 + if(vfile && !(vs = get_vertex_shader(vfile))) { 4.180 + return 0; 4.181 + } 4.182 + if(pfile && !(ps = get_pixel_shader(pfile))) { 4.183 + return 0; 4.184 + } 4.185 + return create_program_link(vs, ps); 4.186 +} 4.187 + 4.188 +void free_program(unsigned int sdr) 4.189 +{ 4.190 + glDeleteProgram(sdr); 4.191 +} 4.192 + 4.193 +void attach_shader(unsigned int prog, unsigned int sdr) 4.194 +{ 4.195 + glAttachShader(prog, sdr); 4.196 + assert(glGetError() == GL_NO_ERROR); 4.197 +} 4.198 + 4.199 +int link_program(unsigned int prog) 4.200 +{ 4.201 + int linked, info_len, retval = 0; 4.202 + char *info_str = 0; 4.203 + 4.204 + glLinkProgram(prog); 4.205 + assert(glGetError() == GL_NO_ERROR); 4.206 + glGetProgramiv(prog, GL_LINK_STATUS, &linked); 4.207 + assert(glGetError() == GL_NO_ERROR); 4.208 + glGetProgramiv(prog, GL_INFO_LOG_LENGTH, &info_len); 4.209 + assert(glGetError() == GL_NO_ERROR); 4.210 + 4.211 + if(info_len) { 4.212 + if((info_str = malloc(info_len + 1))) { 4.213 + glGetProgramInfoLog(prog, info_len, 0, info_str); 4.214 + assert(glGetError() == GL_NO_ERROR); 4.215 + } 4.216 + } 4.217 + 4.218 + if(linked) { 4.219 + fprintf(stderr, info_str ? "linking done: %s\n" : "linking done\n", info_str); 4.220 + } else { 4.221 + fprintf(stderr, info_str ? "linking failed: %s\n" : "linking failed\n", info_str); 4.222 + retval = -1; 4.223 + } 4.224 + 4.225 + free(info_str); 4.226 + return retval; 4.227 +} 4.228 + 4.229 +int bind_program(unsigned int prog) 4.230 +{ 4.231 + GLenum err; 4.232 + 4.233 + glUseProgram(prog); 4.234 + if(prog && (err = glGetError()) != GL_NO_ERROR) { 4.235 + /* maybe the program is not linked, try linking first */ 4.236 + if(err == GL_INVALID_OPERATION) { 4.237 + if(link_program(prog) == -1) { 4.238 + return -1; 4.239 + } 4.240 + glUseProgram(prog); 4.241 + return glGetError() == GL_NO_ERROR ? 0 : -1; 4.242 + } 4.243 + return -1; 4.244 + } 4.245 + return 0; 4.246 +} 4.247 + 4.248 +/* ugly but I'm not going to write the same bloody code over and over */ 4.249 +#define BEGIN_UNIFORM_CODE \ 4.250 + int loc, curr_prog; \ 4.251 + glGetIntegerv(GL_CURRENT_PROGRAM, &curr_prog); \ 4.252 + if((unsigned int)curr_prog != prog && bind_program(prog) == -1) { \ 4.253 + return -1; \ 4.254 + } \ 4.255 + if((loc = glGetUniformLocation(prog, name)) != -1) 4.256 + 4.257 +#define END_UNIFORM_CODE \ 4.258 + if((unsigned int)curr_prog != prog) { \ 4.259 + bind_program(curr_prog); \ 4.260 + } \ 4.261 + return loc == -1 ? -1 : 0 4.262 + 4.263 +int set_uniform_int(unsigned int prog, const char *name, int val) 4.264 +{ 4.265 + BEGIN_UNIFORM_CODE { 4.266 + glUniform1i(loc, val); 4.267 + } 4.268 + END_UNIFORM_CODE; 4.269 +} 4.270 + 4.271 +int set_uniform_float(unsigned int prog, const char *name, float val) 4.272 +{ 4.273 + BEGIN_UNIFORM_CODE { 4.274 + glUniform1f(loc, val); 4.275 + } 4.276 + END_UNIFORM_CODE; 4.277 +} 4.278 + 4.279 +int set_uniform_float3(unsigned int prog, const char *name, float x, float y, float z) 4.280 +{ 4.281 + BEGIN_UNIFORM_CODE { 4.282 + glUniform3f(loc, x, y, z); 4.283 + } 4.284 + END_UNIFORM_CODE; 4.285 +} 4.286 + 4.287 +int set_uniform_float4(unsigned int prog, const char *name, float x, float y, float z, float w) 4.288 +{ 4.289 + BEGIN_UNIFORM_CODE { 4.290 + glUniform4f(loc, x, y, z, w); 4.291 + } 4.292 + END_UNIFORM_CODE; 4.293 +} 4.294 + 4.295 +int set_uniform_matrix4(unsigned int prog, const char *name, float *mat) 4.296 +{ 4.297 + BEGIN_UNIFORM_CODE { 4.298 + glUniformMatrix4fv(loc, 1, GL_FALSE, mat); 4.299 + } 4.300 + END_UNIFORM_CODE; 4.301 +} 4.302 + 4.303 +int set_uniform_matrix4_transposed(unsigned int prog, const char *name, float *mat) 4.304 +{ 4.305 + BEGIN_UNIFORM_CODE { 4.306 + glUniformMatrix4fv(loc, 1, GL_TRUE, mat); 4.307 + } 4.308 + END_UNIFORM_CODE; 4.309 +} 4.310 + 4.311 +int get_attrib_loc(unsigned int prog, const char *name) 4.312 +{ 4.313 + int loc, curr_prog; 4.314 + 4.315 + glGetIntegerv(GL_CURRENT_PROGRAM, &curr_prog); 4.316 + if((unsigned int)curr_prog != prog && bind_program(prog) == -1) { 4.317 + return -1; 4.318 + } 4.319 + 4.320 + loc = glGetAttribLocation(prog, (char*)name); 4.321 + 4.322 + if((unsigned int)curr_prog != prog) { 4.323 + bind_program(curr_prog); 4.324 + } 4.325 + return loc; 4.326 +} 4.327 + 4.328 +void set_attrib_float3(int attr_loc, float x, float y, float z) 4.329 +{ 4.330 + glVertexAttrib3f(attr_loc, x, y, z); 4.331 +}
5.1 --- /dev/null Thu Jan 01 00:00:00 1970 +0000 5.2 +++ b/prototype/src/sdr.h Sun Aug 19 23:09:30 2012 +0300 5.3 @@ -0,0 +1,49 @@ 5.4 +#ifndef SDR_H_ 5.5 +#define SDR_H_ 5.6 + 5.7 +#ifdef __cplusplus 5.8 +extern "C" { 5.9 +#endif /* __cplusplus */ 5.10 + 5.11 +/* ---- shaders ---- */ 5.12 +unsigned int create_vertex_shader(const char *src); 5.13 +unsigned int create_pixel_shader(const char *src); 5.14 +unsigned int create_shader(const char *src, unsigned int sdr_type); 5.15 +void free_shader(unsigned int sdr); 5.16 + 5.17 +unsigned int load_vertex_shader(const char *fname); 5.18 +unsigned int load_pixel_shader(const char *fname); 5.19 +unsigned int load_shader(const char *src, unsigned int sdr_type); 5.20 + 5.21 +unsigned int get_vertex_shader(const char *fname); 5.22 +unsigned int get_pixel_shader(const char *fname); 5.23 +unsigned int get_shader(const char *fname, unsigned int sdr_type); 5.24 + 5.25 +int add_shader(const char *fname, unsigned int sdr); 5.26 +int remove_shader(const char *fname); 5.27 + 5.28 +/* ---- gpu programs ---- */ 5.29 +unsigned int create_program(void); 5.30 +unsigned int create_program_link(unsigned int vs, unsigned int ps); 5.31 +unsigned int create_program_load(const char *vfile, const char *pfile); 5.32 +void free_program(unsigned int sdr); 5.33 + 5.34 +void attach_shader(unsigned int prog, unsigned int sdr); 5.35 +int link_program(unsigned int prog); 5.36 +int bind_program(unsigned int prog); 5.37 + 5.38 +int set_uniform_int(unsigned int prog, const char *name, int val); 5.39 +int set_uniform_float(unsigned int prog, const char *name, float val); 5.40 +int set_uniform_float3(unsigned int prog, const char *name, float x, float y, float z); 5.41 +int set_uniform_float4(unsigned int prog, const char *name, float x, float y, float z, float w); 5.42 +int set_uniform_matrix4(unsigned int prog, const char *name, float *mat); 5.43 +int set_uniform_matrix4_transposed(unsigned int prog, const char *name, float *mat); 5.44 + 5.45 +int get_attrib_loc(unsigned int prog, const char *name); 5.46 +void set_attrib_float3(int attr_loc, float x, float y, float z); 5.47 + 5.48 +#ifdef __cplusplus 5.49 +} 5.50 +#endif /* __cplusplus */ 5.51 + 5.52 +#endif /* SDR_H_ */