taisei/src/taiseigl.c
2018-01-04 19:14:31 +02:00

187 lines
4.4 KiB
C

/*
* This software is licensed under the terms of the MIT-License
* See COPYING for further information.
* ---
* Copyright (c) 2011-2018, Lukas Weber <laochailan@web.de>.
* Copyright (c) 2012-2018, Andrei Alexeyev <akari@alienslab.net>.
*/
#include "taisei.h"
#include "global.h"
#define TAISEIGL_NO_EXT_ABSTRACTION
#include "taiseigl.h"
#undef TAISEIGL_NO_EXT_ABSTRACTION
struct glext_s glext;
#ifndef LINK_TO_LIBGL
#define GLDEF(glname,tsname,typename) typename tsname;
GLDEFS
#undef GLDEF
#endif
typedef void (*tsglproc_ptr)(void);
#ifndef LINK_TO_LIBGL
static tsglproc_ptr get_proc_address(const char *name) {
void *addr = SDL_GL_GetProcAddress(name);
if(!addr) {
log_warn("SDL_GL_GetProcAddress(\"%s\") failed: %s", name, SDL_GetError());
}
// shut up a stupid warning about conversion between data pointers and function pointers
// yes, such a conversion is not allowed by the standard, but when several major APIs
// (POSIX and SDL to name a few) require casting void* to a function pointer, that says something.
return *(tsglproc_ptr*)&addr;
}
#endif
static void get_gl_version(char *major, char *minor) {
// the glGetIntegerv way only works in >=3.0 contexts, so...
const char *vstr = (const char*)glGetString(GL_VERSION);
if(!vstr) {
*major = 1;
*minor = 1;
return;
}
const char *dot = strchr(vstr, '.');
*major = atoi(vstr);
*minor = atoi(dot+1);
}
static bool extension_supported(const char *ext) {
const char *overrides = getenv("TAISEI_GL_EXT_OVERRIDES");
if(overrides) {
char buf[strlen(overrides)+1], *save, *arg, *e;
strcpy(buf, overrides);
arg = buf;
while((e = strtok_r(arg, " ", &save))) {
bool r = true;
if(*e == '-') {
++e;
r = false;
}
if(!strcmp(e, ext)) {
return r;
}
arg = NULL;
}
}
return SDL_GL_ExtensionSupported(ext);
}
static void check_glext_draw_instanced(void) {
if((glext.draw_instanced = (
(glext.ARB_draw_instanced = extension_supported("GL_ARB_draw_instanced")) &&
(glext.DrawArraysInstanced = tsglDrawArraysInstancedARB)
))) {
log_debug("Using GL_ARB_draw_instanced");
return;
} else {
glext.ARB_draw_instanced = false;
}
if((glext.draw_instanced = (
(glext.EXT_draw_instanced = extension_supported("GL_EXT_draw_instanced")) &&
(glext.DrawArraysInstanced = tsglDrawArraysInstancedARB)
))) {
log_debug("Using GL_EXT_draw_instanced");
return;
} else {
glext.EXT_draw_instanced = false;
}
log_warn(
"glDrawArraysInstanced is not supported. "
"Your video driver is probably bad, or very old, or both. "
"Expect terrible performance."
);
}
static void check_glext_debug_output(void) {
if((glext.debug_output = (
extension_supported("GL_ARB_debug_output") &&
(glext.DebugMessageCallback = tsglDebugMessageCallbackARB) &&
(glext.DebugMessageControl = tsglDebugMessageControlARB)
))) {
log_debug("Using GL_ARB_debug_output");
return;
}
if((glext.debug_output = (
extension_supported("GL_KHR_debug") &&
(glext.DebugMessageCallback = tsglDebugMessageCallback) &&
(glext.DebugMessageControl = tsglDebugMessageControl)
))) {
log_debug("Using GL_KHR_debug");
return;
}
}
void check_gl_extensions(void) {
memset(&glext, 0, sizeof(glext));
get_gl_version(&glext.version.major, &glext.version.minor);
const char *glslv = (const char*)glGetString(GL_SHADING_LANGUAGE_VERSION);
if(!glslv) {
glslv = "None";
}
log_info("OpenGL version: %s", (const char*)glGetString(GL_VERSION));
log_info("OpenGL vendor: %s", (const char*)glGetString(GL_VENDOR));
log_info("OpenGL renderer: %s", (const char*)glGetString(GL_RENDERER));
log_info("GLSL version: %s", glslv);
log_debug("Supported extensions: %s", (const char*)glGetString(GL_EXTENSIONS));
check_glext_draw_instanced();
check_glext_debug_output();
}
void load_gl_library(void) {
#ifndef LINK_TO_LIBGL
char *lib = getenv("TAISEI_LIBGL");
if(lib && !*lib) {
lib = NULL;
}
if(SDL_GL_LoadLibrary(lib) < 0) {
log_fatal("SDL_GL_LoadLibrary() failed: %s", SDL_GetError());
}
#endif
}
void unload_gl_library(void) {
#ifndef LINK_TO_LIBGL
SDL_GL_UnloadLibrary();
#endif
}
void load_gl_functions(void) {
#ifndef LINK_TO_LIBGL
#define GLDEF(glname,tsname,typename) tsname = (typename)get_proc_address(#glname);
GLDEFS
#undef GLDEF
#endif
}
void gluPerspective(GLdouble fovY, GLdouble aspect, GLdouble zNear, GLdouble zFar) {
GLdouble fW, fH;
fH = tan(fovY / 360 * M_PI) * zNear;
fW = fH * aspect;
glFrustum(-fW, fW, -fH, fH, zNear, zFar);
}