2013-08-18 16:18:06 +09:00
|
|
|
//
|
|
|
|
// SDLGLDevice.cpp
|
|
|
|
// OpenSpades
|
|
|
|
//
|
|
|
|
// Created by yvt on 7/11/13.
|
|
|
|
// Copyright (c) 2013 yvt.jp. All rights reserved.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "SDLGLDevice.h"
|
|
|
|
#include "../Imports/OpenGL.h"
|
|
|
|
#include "../Imports/SDL.h"
|
|
|
|
#include "../Core/Exception.h"
|
|
|
|
#include "../Core/Debug.h"
|
|
|
|
#include "../Core/Settings.h"
|
|
|
|
|
|
|
|
using namespace spades::draw;
|
|
|
|
|
|
|
|
#ifndef __APPLE__
|
|
|
|
#define GLEW 1
|
|
|
|
#endif
|
|
|
|
|
|
|
|
SPADES_SETTING(r_ignoreGLErrors, "1");
|
|
|
|
|
|
|
|
static uint32_t vertCount = 0;
|
|
|
|
static uint32_t drawOps = 0;
|
|
|
|
static Uint32 lastFrame = 0;
|
|
|
|
|
|
|
|
namespace spades {
|
|
|
|
namespace gui {
|
|
|
|
|
|
|
|
#define CheckError() do{ \
|
|
|
|
GLenum err; \
|
|
|
|
if(!r_ignoreGLErrors){ \
|
|
|
|
err = glGetError(); \
|
|
|
|
if(err != GL_NO_ERROR) \
|
|
|
|
ReportError(err, __LINE__, __PRETTY_FUNCTION__); \
|
|
|
|
} \
|
2013-08-26 01:46:48 +09:00
|
|
|
}while(0)
|
|
|
|
|
2013-08-27 03:56:56 +09:00
|
|
|
#define CheckErrorAlways() do{ \
|
|
|
|
GLenum err; \
|
|
|
|
err = glGetError(); \
|
|
|
|
if(err != GL_NO_ERROR) \
|
|
|
|
ReportError(err, __LINE__, __PRETTY_FUNCTION__); \
|
|
|
|
}while(0)
|
|
|
|
|
2013-08-26 01:46:48 +09:00
|
|
|
#define CheckExistence(func) do { \
|
|
|
|
if(!func) { \
|
|
|
|
ReportMissingFunc( #func ); \
|
|
|
|
}\
|
|
|
|
} while(0)
|
|
|
|
|
2013-08-27 03:56:56 +09:00
|
|
|
static std::string ErrorToString(GLenum err) {
|
|
|
|
switch(err) {
|
|
|
|
case GL_NO_ERROR:
|
|
|
|
return "No Error";
|
|
|
|
case GL_INVALID_ENUM:
|
|
|
|
return "Invalid Enum";
|
|
|
|
case GL_INVALID_VALUE:
|
|
|
|
return "Invalid Value";
|
|
|
|
case GL_INVALID_OPERATION:
|
|
|
|
return "Invalid Operation";
|
|
|
|
case GL_INVALID_FRAMEBUFFER_OPERATION:
|
|
|
|
return "Invalid Framebuffer Operation";
|
|
|
|
case GL_OUT_OF_MEMORY:
|
|
|
|
return "Out of Memory";
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
char buf[256];
|
|
|
|
sprintf(buf, "0x%08x", (unsigned int)err);
|
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
static void ReportError(GLenum err, int line, const char *func){
|
2013-08-27 03:56:56 +09:00
|
|
|
std::string msg;
|
|
|
|
msg = ErrorToString(err);
|
|
|
|
while((err = glGetError()) != GL_NO_ERROR) {
|
|
|
|
msg += ", "; msg += ErrorToString(err);
|
|
|
|
}
|
2013-08-28 02:56:01 +09:00
|
|
|
if(r_ignoreGLErrors){
|
|
|
|
SPRaise("GL error %s in %s at %s:%d\n\n"
|
|
|
|
"WARNING: r_ignoreGLErrors is enabled. "
|
|
|
|
"Information contained in this message is "
|
|
|
|
"inaccurate and non-informative.",
|
|
|
|
msg.c_str(),
|
|
|
|
func, __FILE__, line);
|
|
|
|
}else{
|
|
|
|
SPRaise("GL error %s in %s at %s:%d",
|
|
|
|
msg.c_str(),
|
|
|
|
func, __FILE__, line);
|
|
|
|
}
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
2013-08-26 01:46:48 +09:00
|
|
|
static void ReportMissingFunc(const char *func){
|
|
|
|
SPRaise("GL function %s missing",
|
|
|
|
func);
|
|
|
|
}
|
2013-08-18 16:18:06 +09:00
|
|
|
|
|
|
|
SDLGLDevice::SDLGLDevice(SDL_Surface *s):
|
|
|
|
surface(s) {
|
|
|
|
#ifndef __APPLE__
|
|
|
|
GLenum err = glewInit();
|
|
|
|
if (GLEW_OK != err){
|
|
|
|
SPRaise("GLEW error: %s", glewGetErrorString(err));
|
|
|
|
}
|
|
|
|
#endif
|
2013-08-26 01:27:44 +09:00
|
|
|
SPLog("SDLGLDevice created");
|
|
|
|
|
|
|
|
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glFrontFace);
|
2013-08-18 16:18:06 +09:00
|
|
|
glFrontFace(GL_CW);
|
2013-08-27 17:36:20 +09:00
|
|
|
|
2013-08-28 02:56:01 +09:00
|
|
|
if(r_ignoreGLErrors) {
|
|
|
|
SPLog("NOTICE: r_ignoreGLErrors is enabled. "
|
|
|
|
"OpenGL error detection might not work correctly.");
|
|
|
|
}else{
|
|
|
|
SPLog("NOTICE: r_ignoreGLErrors is disabled. "
|
|
|
|
"OpenGL error is checked for every GL call, but "
|
|
|
|
"performance may be reduced.");
|
|
|
|
}
|
|
|
|
|
2013-08-27 17:36:20 +09:00
|
|
|
// clear error state
|
|
|
|
while(glGetError() != GL_NO_ERROR);
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::DepthRange(Float near, Float far){
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glDepthRange);
|
2013-08-18 16:18:06 +09:00
|
|
|
glDepthRange(near, far);
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::Viewport(Integer x, Integer y,
|
|
|
|
Sizei width, Sizei height) {
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glViewport);
|
2013-08-18 16:18:06 +09:00
|
|
|
glViewport(x, y, width, height);
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::ClearDepth(float v) {
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glClearDepth);
|
2013-08-18 16:18:06 +09:00
|
|
|
glClearDepth(v);
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::ClearColor(float r, float g, float b, float a){
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glClearColor);
|
2013-08-18 16:18:06 +09:00
|
|
|
glClearColor(r, g, b, a);
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::Clear(Enum bits) {
|
|
|
|
GLbitfield v = 0;
|
|
|
|
if(bits & ColorBufferBit)
|
|
|
|
v |= GL_COLOR_BUFFER_BIT;
|
|
|
|
if(bits & DepthBufferBit)
|
|
|
|
v |= GL_DEPTH_BUFFER_BIT;
|
|
|
|
if(bits & StencilBufferBit)
|
|
|
|
v |= GL_STENCIL_BUFFER_BIT;
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glClear);
|
2013-08-18 16:18:06 +09:00
|
|
|
glClear(v);
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::Swap() {
|
|
|
|
//glFinish();
|
2013-08-27 03:56:56 +09:00
|
|
|
CheckErrorAlways();
|
2013-08-18 16:18:06 +09:00
|
|
|
SDL_GL_SwapBuffers();
|
|
|
|
#if 0
|
|
|
|
Uint32 t = SDL_GetTicks();
|
|
|
|
if(lastFrame == 0) t = lastFrame - 30;
|
|
|
|
double dur = (double)(t - lastFrame) / 1000.;
|
|
|
|
lastFrame = t;
|
|
|
|
|
|
|
|
printf("FPS:%.02f, Vertices: %d (%.02f/sec), Drawcalls: %d (%.02f/sec)\n", 1./dur,
|
|
|
|
vertCount, vertCount / dur,
|
|
|
|
drawOps, drawOps / dur);
|
|
|
|
|
|
|
|
//printf("%.02f,%.02f,%.02f\n", 1./dur, vertCount / dur, drawOps / dur);
|
|
|
|
#endif
|
|
|
|
vertCount = 0;
|
|
|
|
drawOps = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::Finish() {
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glFinish);
|
2013-08-18 16:18:06 +09:00
|
|
|
glFinish();
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::Flush(){
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glFlush);
|
2013-08-18 16:18:06 +09:00
|
|
|
glFlush();
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::DepthMask(bool b){
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glDepthMask);
|
2013-08-18 16:18:06 +09:00
|
|
|
glDepthMask(b?GL_TRUE:GL_FALSE);
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::ColorMask(bool r, bool g, bool b, bool a){
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glColorMask);
|
2013-08-18 16:18:06 +09:00
|
|
|
glColorMask(r?GL_TRUE:GL_FALSE,
|
|
|
|
g?GL_TRUE:GL_FALSE,
|
|
|
|
b?GL_TRUE:GL_FALSE,
|
|
|
|
a?GL_TRUE:GL_FALSE);
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::Enable(spades::draw::IGLDevice::Enum state, bool b){
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
GLenum type;
|
|
|
|
switch(state){
|
|
|
|
case DepthTest: type = GL_DEPTH_TEST; break;
|
|
|
|
case CullFace: type = GL_CULL_FACE; break;
|
|
|
|
case Blend: type = GL_BLEND; break;
|
|
|
|
case Texture2D: type = GL_TEXTURE_2D; break;
|
|
|
|
case Multisample: type = GL_MULTISAMPLE; break;
|
2013-08-23 15:10:49 +09:00
|
|
|
case FramebufferSRGB: type = GL_FRAMEBUFFER_SRGB; break;
|
2013-08-18 16:18:06 +09:00
|
|
|
default: SPInvalidEnum("state", state);
|
|
|
|
}
|
|
|
|
if(b)
|
|
|
|
glEnable(type);
|
|
|
|
else
|
|
|
|
glDisable(type);
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
IGLDevice::Integer SDLGLDevice::GetInteger(Enum type){
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
GLint v;
|
|
|
|
switch(type){
|
|
|
|
case draw::IGLDevice::FramebufferBinding:
|
|
|
|
glGetIntegerv(GL_FRAMEBUFFER_BINDING, &v);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("type", type);
|
|
|
|
}
|
|
|
|
CheckError();
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *SDLGLDevice::GetString(spades::draw::IGLDevice::Enum type) {
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
switch(type){
|
|
|
|
case Vendor:
|
|
|
|
return (const char *)glGetString(GL_VENDOR);
|
|
|
|
case Renderer:
|
|
|
|
return (const char *)glGetString(GL_RENDERER);
|
|
|
|
case Version:
|
|
|
|
return (const char *)glGetString(GL_VERSION);
|
|
|
|
case ShadingLanguageVersion:
|
|
|
|
return (const char *)glGetString(GL_SHADING_LANGUAGE_VERSION);
|
|
|
|
default: SPInvalidEnum("type", type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const char *SDLGLDevice::GetIndexedString(spades::draw::IGLDevice::Enum type, UInteger index) {
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
switch(type){
|
|
|
|
case draw::IGLDevice::Extensions:
|
|
|
|
return (const char *)glGetStringi(GL_EXTENSIONS, index);
|
|
|
|
default: SPInvalidEnum("type", type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
GLenum SDLGLDevice::parseBlendEquation(spades::draw::IGLDevice::Enum v){
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
switch(v){
|
|
|
|
case Add: return GL_FUNC_ADD;
|
|
|
|
case Subtract: return GL_FUNC_SUBTRACT;
|
|
|
|
case ReverseSubtract: return GL_FUNC_REVERSE_SUBTRACT;
|
|
|
|
case MinOp: return GL_MIN;
|
|
|
|
case MaxOp: return GL_MAX;
|
|
|
|
default: SPInvalidEnum("v", v);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
GLenum SDLGLDevice::parseBlendFunction(spades::draw::IGLDevice::Enum v){
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
switch(v){
|
|
|
|
case Zero: return GL_ZERO;
|
|
|
|
case One: return GL_ONE;
|
|
|
|
case SrcColor: return GL_SRC_COLOR;
|
|
|
|
case DestColor: return GL_DST_COLOR;
|
|
|
|
case OneMinusSrcColor: return GL_ONE_MINUS_SRC_COLOR;
|
|
|
|
case OneMinusDestColor: return GL_ONE_MINUS_DST_COLOR;
|
|
|
|
case SrcAlpha: return GL_SRC_ALPHA;
|
|
|
|
case DestAlpha: return GL_DST_ALPHA;
|
|
|
|
case OneMinusSrcAlpha: return GL_ONE_MINUS_SRC_ALPHA;
|
|
|
|
case OneMinusDestAlpha: return GL_ONE_MINUS_DST_ALPHA;
|
|
|
|
case ConstantColor: return GL_CONSTANT_COLOR;
|
|
|
|
case ConstantAlpha: return GL_CONSTANT_ALPHA;
|
|
|
|
case OneMinusConstantColor: return GL_ONE_MINUS_CONSTANT_COLOR;
|
|
|
|
case OneMinusConstantAlpha: return GL_ONE_MINUS_CONSTANT_ALPHA;
|
|
|
|
default: SPInvalidEnum("v", v);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::BlendEquation(spades::draw::IGLDevice::Enum mode) {
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glBlendEquation);
|
2013-08-18 16:18:06 +09:00
|
|
|
glBlendEquation(parseBlendEquation(mode));
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::BlendEquation(spades::draw::IGLDevice::Enum rgb,
|
|
|
|
spades::draw::IGLDevice::Enum alpha) {
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glBlendEquationSeparate);
|
2013-08-18 16:18:06 +09:00
|
|
|
glBlendEquationSeparate(parseBlendEquation(rgb),
|
|
|
|
parseBlendEquation(alpha));
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::BlendFunc(Enum src,
|
|
|
|
Enum dest) {
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glBlendFunc);
|
2013-08-18 16:18:06 +09:00
|
|
|
glBlendFunc(parseBlendFunction(src),
|
|
|
|
parseBlendFunction(dest));
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::BlendFunc(Enum srcRgb,
|
|
|
|
Enum destRgb,
|
|
|
|
Enum srcAlpha,
|
|
|
|
Enum destAlpha) {
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glBlendFuncSeparate);
|
2013-08-18 16:18:06 +09:00
|
|
|
glBlendFuncSeparate(parseBlendFunction(srcRgb),
|
|
|
|
parseBlendFunction(destRgb),
|
|
|
|
parseBlendFunction(srcAlpha),
|
|
|
|
parseBlendFunction(destAlpha));
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::BlendColor(Float r, Float g, Float b, Float a){
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glBlendColor);
|
2013-08-18 16:18:06 +09:00
|
|
|
glBlendColor(r, g, b, a);
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::LineWidth(Float w) {
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glLineWidth);
|
2013-08-18 16:18:06 +09:00
|
|
|
glLineWidth(w);
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::DepthFunc(Enum func) {
|
|
|
|
SPADES_MARK_FUNCTION();
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glDepthFunc);
|
2013-08-18 16:18:06 +09:00
|
|
|
switch(func){
|
|
|
|
case Never: glDepthFunc(GL_NEVER); break;
|
|
|
|
case Always: glDepthFunc(GL_ALWAYS); break;
|
|
|
|
case Less: glDepthFunc(GL_LESS); break;
|
|
|
|
case LessOrEqual: glDepthFunc(GL_LEQUAL); break;
|
|
|
|
case Equal: glDepthFunc(GL_EQUAL); break;
|
|
|
|
case Greater: glDepthFunc(GL_GREATER); break;
|
|
|
|
case GreaterOrEqual: glDepthFunc(GL_GEQUAL); break;
|
|
|
|
case NotEqual: glDepthFunc(GL_NOTEQUAL); break;
|
|
|
|
default: SPInvalidEnum("func", func);
|
|
|
|
}
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
IGLDevice::UInteger SDLGLDevice::GenBuffer() {
|
2013-08-25 02:37:30 +09:00
|
|
|
SPADES_MARK_FUNCTION_DEBUG();
|
2013-08-18 16:18:06 +09:00
|
|
|
GLuint i;
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glGenBuffers)
|
|
|
|
glGenBuffers(1, &i);
|
|
|
|
else if(glGenBuffersARB)
|
|
|
|
glGenBuffersARB(1, &i);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glGenBuffers");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glGenBuffers);
|
2013-08-18 16:18:06 +09:00
|
|
|
glGenBuffers(1, &i);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::DeleteBuffer(UInteger i) {
|
2013-08-25 02:37:30 +09:00
|
|
|
SPADES_MARK_FUNCTION_DEBUG();
|
2013-08-18 16:18:06 +09:00
|
|
|
GLuint v = (GLuint)i;
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glDeleteBuffers)
|
|
|
|
glDeleteBuffers(1, &v);
|
|
|
|
else if(glDeleteBuffersARB)
|
|
|
|
glDeleteBuffersARB(1, &v);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glDeleteBuffers");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glDeleteBuffers);
|
2013-08-18 16:18:06 +09:00
|
|
|
glDeleteBuffers(1, &v);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
2013-08-25 02:37:30 +09:00
|
|
|
void *SDLGLDevice::MapBuffer(Enum target,
|
|
|
|
Enum access) {
|
|
|
|
SPADES_MARK_FUNCTION_DEBUG();
|
|
|
|
GLenum acc;
|
|
|
|
switch(access){
|
|
|
|
case draw::IGLDevice::ReadOnly:
|
|
|
|
acc = GL_READ_ONLY;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::WriteOnly:
|
|
|
|
acc = GL_WRITE_ONLY;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::ReadWrite:
|
|
|
|
acc = GL_READ_WRITE;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("access", access);
|
|
|
|
}
|
2013-08-27 01:36:28 +09:00
|
|
|
void *ret;
|
|
|
|
#if GLEW
|
|
|
|
if(glMapBuffer)
|
|
|
|
ret = glMapBuffer(parseBufferTarget(target),
|
|
|
|
acc);
|
|
|
|
else if(glMapBufferARB)
|
|
|
|
ret = glMapBufferARB(parseBufferTarget(target),
|
|
|
|
acc);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glMapBuffer");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glMapBuffer);
|
2013-08-27 01:36:28 +09:00
|
|
|
ret = glMapBuffer(parseBufferTarget(target),
|
2013-08-25 02:37:30 +09:00
|
|
|
acc);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-25 02:37:30 +09:00
|
|
|
CheckError();
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::UnmapBuffer(Enum target) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glUnmapBuffer)
|
|
|
|
glUnmapBuffer(parseBufferTarget(target));
|
|
|
|
else if(glUnmapBufferARB)
|
|
|
|
glUnmapBufferARB(parseBufferTarget(target));
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glUnmapBuffer");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glUnmapBuffer);
|
2013-08-25 02:37:30 +09:00
|
|
|
glUnmapBuffer(parseBufferTarget(target));
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-25 02:37:30 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
GLenum SDLGLDevice::parseBufferTarget(spades::draw::IGLDevice::Enum v){
|
|
|
|
SPADES_MARK_FUNCTION_DEBUG();
|
|
|
|
switch(v){
|
|
|
|
case ArrayBuffer: return GL_ARRAY_BUFFER;
|
|
|
|
case ElementArrayBuffer: return GL_ELEMENT_ARRAY_BUFFER;
|
2013-08-25 02:37:30 +09:00
|
|
|
case PixelPackBuffer: return GL_PIXEL_PACK_BUFFER;
|
|
|
|
case PixelUnpackBuffer: return GL_PIXEL_UNPACK_BUFFER;
|
2013-08-18 16:18:06 +09:00
|
|
|
default: SPInvalidEnum("v", v);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::BindBuffer(Enum target, UInteger i){
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glBindBuffer)
|
|
|
|
glBindBuffer(parseBufferTarget(target), (GLuint)i);
|
|
|
|
else if(glBindBufferARB)
|
|
|
|
glBindBufferARB(parseBufferTarget(target), (GLuint)i);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glBindBuffer");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glBindBuffer);
|
2013-08-18 16:18:06 +09:00
|
|
|
glBindBuffer(parseBufferTarget(target), (GLuint)i);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::BufferData(Enum target, Sizei size,
|
|
|
|
const void *data,
|
|
|
|
Enum usage) {
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
GLenum usageVal;
|
|
|
|
switch(usage){
|
|
|
|
case StaticDraw: usageVal = GL_STATIC_DRAW; break;
|
|
|
|
case StreamDraw: usageVal = GL_STREAM_DRAW; break;
|
|
|
|
case DynamicDraw: usageVal = GL_DYNAMIC_DRAW; break;
|
|
|
|
default: SPInvalidEnum("usage", usage);
|
|
|
|
}
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glBufferData)
|
|
|
|
glBufferData(parseBufferTarget(target), (GLsizeiptr)size,
|
|
|
|
data, usageVal);
|
|
|
|
else if(glBufferDataARB)
|
|
|
|
glBufferDataARB(parseBufferTarget(target), (GLsizeiptr)size,
|
|
|
|
data, usageVal);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glBufferData");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glBufferData);
|
2013-08-18 16:18:06 +09:00
|
|
|
glBufferData(parseBufferTarget(target), (GLsizeiptr)size,
|
|
|
|
data, usageVal);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::BufferSubData(Enum target, Sizei offset,
|
|
|
|
Sizei size, const void *data) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glBufferSubData)
|
|
|
|
glBufferSubData(parseBufferTarget(target), offset,
|
|
|
|
size, data);
|
|
|
|
else if(glBufferSubDataARB)
|
|
|
|
glBufferSubDataARB(parseBufferTarget(target), offset,
|
|
|
|
size, data);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glBufferSubData");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glBufferSubData);
|
2013-08-18 16:18:06 +09:00
|
|
|
glBufferSubData(parseBufferTarget(target), offset,
|
|
|
|
size, data);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
IGLDevice::UInteger SDLGLDevice::GenTexture() {
|
|
|
|
GLuint i;
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glGenTextures);
|
2013-08-18 16:18:06 +09:00
|
|
|
glGenTextures(1, &i);
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::DeleteTexture(UInteger i) {
|
|
|
|
GLuint v = (GLuint)i;
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glDeleteTextures);
|
2013-08-18 16:18:06 +09:00
|
|
|
glDeleteTextures(1, &v);
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
GLenum SDLGLDevice::parseTextureTarget(Enum v) {
|
|
|
|
SPADES_MARK_FUNCTION_DEBUG();
|
|
|
|
switch(v){
|
|
|
|
case Texture2D: return GL_TEXTURE_2D;
|
|
|
|
case Texture3D: return GL_TEXTURE_3D;
|
|
|
|
default: SPInvalidEnum("v", v);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::ActiveTexture(UInteger stage) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glActiveTexture)
|
|
|
|
glActiveTexture(GL_TEXTURE0 + stage);
|
|
|
|
else if(glActiveTextureARB)
|
|
|
|
glActiveTextureARB(GL_TEXTURE0 + stage);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glActiveTexture");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glActiveTexture);
|
2013-08-18 16:18:06 +09:00
|
|
|
glActiveTexture(GL_TEXTURE0 + stage);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::BindTexture(Enum target, UInteger tex){
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glBindTexture);
|
2013-08-18 16:18:06 +09:00
|
|
|
glBindTexture(parseTextureTarget(target), tex);
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
GLenum SDLGLDevice::parseTextureInternalFormat(Enum v) {
|
|
|
|
SPADES_MARK_FUNCTION_DEBUG();
|
|
|
|
switch(v){
|
|
|
|
case Red: return GL_RED;
|
|
|
|
case RG: return GL_RG;
|
|
|
|
case RGB: return GL_RGB;
|
|
|
|
case RGBA: return GL_RGBA;
|
|
|
|
case DepthComponent: return GL_DEPTH_COMPONENT;
|
|
|
|
case DepthComponent24: return GL_DEPTH_COMPONENT24;
|
|
|
|
case StencilIndex: return GL_STENCIL_INDEX;
|
|
|
|
|
|
|
|
case RGB10A2: return GL_RGB10_A2;
|
|
|
|
case RGB16F: return GL_RGB16F;
|
|
|
|
case RGBA16F: return GL_RGBA16F;
|
|
|
|
case R16F: return GL_R16F;
|
|
|
|
case RGB5: return GL_RGB5;
|
|
|
|
case RGB5A1: return GL_RGB5_A1;
|
2013-08-26 01:59:10 +09:00
|
|
|
case RGB8: return GL_RGB8;
|
|
|
|
case RGBA8: return GL_RGBA8;
|
2013-08-18 16:18:06 +09:00
|
|
|
default: SPInvalidEnum("v", v);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
GLenum SDLGLDevice::parseTextureFormat(Enum v) {
|
|
|
|
SPADES_MARK_FUNCTION_DEBUG();
|
|
|
|
switch(v){
|
|
|
|
case Red: return GL_RED;
|
|
|
|
case RG: return GL_RG;
|
|
|
|
case RGB: return GL_RGB;
|
|
|
|
case RGBA: return GL_RGBA;
|
|
|
|
case BGRA: return GL_BGRA;
|
|
|
|
case DepthComponent: return GL_DEPTH_COMPONENT;
|
|
|
|
case StencilIndex: return GL_STENCIL_INDEX;
|
|
|
|
default: SPInvalidEnum("v", v);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
GLenum SDLGLDevice::parseType(Enum v) {
|
|
|
|
SPADES_MARK_FUNCTION_DEBUG();
|
|
|
|
switch(v){
|
|
|
|
case Int: return GL_INT;
|
|
|
|
case UnsignedInt: return GL_UNSIGNED_INT;
|
|
|
|
case Short: return GL_SHORT;
|
|
|
|
case UnsignedShort: return GL_UNSIGNED_SHORT;
|
|
|
|
case Byte: return GL_BYTE;
|
|
|
|
case UnsignedByte: return GL_UNSIGNED_BYTE;
|
|
|
|
case FloatType: return GL_FLOAT;
|
|
|
|
case UnsignedShort5551: return GL_UNSIGNED_SHORT_5_5_5_1;
|
|
|
|
case UnsignedShort1555Rev: return GL_UNSIGNED_SHORT_1_5_5_5_REV;
|
2013-08-25 02:37:30 +09:00
|
|
|
case UnsignedInt2101010Rev: return GL_UNSIGNED_INT_2_10_10_10_REV;
|
2013-08-18 16:18:06 +09:00
|
|
|
default: SPInvalidEnum("v", v);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::TexImage2D(Enum target, Integer level,
|
|
|
|
Enum intFmt,
|
|
|
|
Sizei width, Sizei height,
|
|
|
|
Integer border,
|
|
|
|
Enum format, Enum type,
|
|
|
|
const void *data) {
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glTexImage2D);
|
2013-08-18 16:18:06 +09:00
|
|
|
glTexImage2D(parseTextureTarget(target),
|
|
|
|
level, parseTextureInternalFormat(intFmt),
|
|
|
|
width, height, border,
|
|
|
|
parseTextureFormat(format),
|
|
|
|
parseType(type), data);
|
2013-08-27 03:56:56 +09:00
|
|
|
CheckErrorAlways();
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::TexImage3D(Enum target, Integer level,
|
|
|
|
Enum intFmt,
|
|
|
|
Sizei width, Sizei height,
|
|
|
|
Sizei depth,
|
|
|
|
Integer border,
|
|
|
|
Enum format, Enum type,
|
|
|
|
const void *data) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glTexImage3D)
|
|
|
|
glTexImage3D(parseTextureTarget(target),
|
|
|
|
level, parseTextureInternalFormat(intFmt),
|
|
|
|
width, height, depth, border,
|
|
|
|
parseTextureFormat(format),
|
|
|
|
parseType(type), data);
|
|
|
|
else if(glTexImage3DEXT)
|
|
|
|
glTexImage3DEXT(parseTextureTarget(target),
|
|
|
|
level, parseTextureInternalFormat(intFmt),
|
|
|
|
width, height, depth, border,
|
|
|
|
parseTextureFormat(format),
|
|
|
|
parseType(type), data);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glTexImage3D");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glTexImage3D);
|
2013-08-18 16:18:06 +09:00
|
|
|
glTexImage3D(parseTextureTarget(target),
|
|
|
|
level, parseTextureInternalFormat(intFmt),
|
|
|
|
width, height, depth, border,
|
|
|
|
parseTextureFormat(format),
|
|
|
|
parseType(type), data);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-27 03:56:56 +09:00
|
|
|
CheckErrorAlways();
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::TexSubImage2D(Enum target,
|
|
|
|
Integer level,
|
|
|
|
Integer x, Integer y,
|
|
|
|
Sizei width, Sizei height,
|
|
|
|
Enum format, Enum type,
|
|
|
|
const void *data) {
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glTexSubImage2D);
|
2013-08-18 16:18:06 +09:00
|
|
|
glTexSubImage2D(parseTextureTarget(target), level,
|
|
|
|
x, y, width, height,
|
|
|
|
parseTextureFormat(format),
|
|
|
|
parseType(type), data);
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::TexSubImage3D(Enum target,
|
|
|
|
Integer level,
|
|
|
|
Integer x, Integer y,
|
|
|
|
Integer z,
|
|
|
|
Sizei width, Sizei height,
|
|
|
|
Sizei depth,
|
|
|
|
Enum format, Enum type,
|
|
|
|
const void *data) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glTexSubImage3D)
|
|
|
|
glTexSubImage3D(parseTextureTarget(target), level,
|
|
|
|
x, y, z, width, height, depth,
|
|
|
|
parseTextureFormat(format),
|
|
|
|
parseType(type), data);
|
|
|
|
else if(glTexSubImage3DEXT)
|
|
|
|
glTexSubImage3DEXT(parseTextureTarget(target), level,
|
|
|
|
x, y, z, width, height, depth,
|
|
|
|
parseTextureFormat(format),
|
|
|
|
parseType(type), data);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glTexSubImage3D");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glTexSubImage3D);
|
2013-08-18 16:18:06 +09:00
|
|
|
glTexSubImage3D(parseTextureTarget(target), level,
|
|
|
|
x, y, z, width, height, depth,
|
|
|
|
parseTextureFormat(format),
|
|
|
|
parseType(type), data);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::CopyTexSubImage2D(Enum target,
|
|
|
|
Integer level,
|
|
|
|
Integer destinationX,
|
|
|
|
Integer destinationY,
|
|
|
|
Integer srcX,
|
|
|
|
Integer srcY,
|
|
|
|
Sizei width,
|
|
|
|
Sizei height) {
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glCopyTexSubImage2D);
|
2013-08-18 16:18:06 +09:00
|
|
|
glCopyTexSubImage2D(parseTextureTarget(target), level,
|
|
|
|
destinationX, destinationY,
|
|
|
|
srcX, srcY, width, height);
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::TexParamater(Enum target,Enum param, Enum val) {
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
GLenum targ = parseTextureTarget(target);
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glTexParameteri);
|
2013-08-18 16:18:06 +09:00
|
|
|
switch(param){
|
|
|
|
case TextureMinFilter:
|
|
|
|
switch(val){
|
|
|
|
case Nearest: glTexParameteri(targ, GL_TEXTURE_MIN_FILTER, GL_NEAREST); break;
|
|
|
|
case Linear: glTexParameteri(targ, GL_TEXTURE_MIN_FILTER, GL_LINEAR); break;
|
|
|
|
case NearestMipmapLinear: glTexParameteri(targ, GL_TEXTURE_MIN_FILTER, GL_NEAREST_MIPMAP_LINEAR); break;
|
|
|
|
case LinearMipmapLinear: glTexParameteri(targ, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_LINEAR); break;
|
|
|
|
case NearestMipmapNearest: glTexParameteri(targ, GL_TEXTURE_MIN_FILTER, GL_NEAREST_MIPMAP_NEAREST); break;
|
|
|
|
case LinearMipmapNearest: glTexParameteri(targ, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_NEAREST); break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("val", val);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TextureMagFilter:
|
|
|
|
switch(val){
|
|
|
|
case Nearest: glTexParameteri(targ, GL_TEXTURE_MAG_FILTER, GL_NEAREST); break;
|
|
|
|
case Linear: glTexParameteri(targ, GL_TEXTURE_MAG_FILTER, GL_LINEAR); break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("val", val);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TextureWrapS:
|
|
|
|
switch(val){
|
|
|
|
case ClampToEdge: glTexParameteri(targ, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); break;
|
|
|
|
case Repeat: glTexParameteri(targ, GL_TEXTURE_WRAP_S, GL_REPEAT); break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("val", val);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TextureWrapT:
|
|
|
|
switch(val){
|
|
|
|
case ClampToEdge: glTexParameteri(targ, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); break;
|
|
|
|
case Repeat: glTexParameteri(targ, GL_TEXTURE_WRAP_T, GL_REPEAT); break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("val", val);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TextureWrapR:
|
|
|
|
switch(val){
|
|
|
|
case ClampToEdge: glTexParameteri(targ, GL_TEXTURE_WRAP_R, GL_CLAMP_TO_EDGE); break;
|
|
|
|
case Repeat: glTexParameteri(targ, GL_TEXTURE_WRAP_R, GL_REPEAT); break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("val", val);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TextureCompareMode:
|
|
|
|
switch(val){
|
|
|
|
case draw::IGLDevice::CompareRefToTexture: glTexParameteri(targ, GL_TEXTURE_COMPARE_MODE, GL_COMPARE_REF_TO_TEXTURE); break;
|
|
|
|
case draw::IGLDevice::None: glTexParameteri(targ, GL_TEXTURE_COMPARE_MODE, GL_NONE); break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("val", val);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case TextureCompareFunc:
|
|
|
|
switch(val){
|
|
|
|
case IGLDevice::LessOrEqual: glTexParameteri(targ, GL_TEXTURE_COMPARE_FUNC, GL_LEQUAL); break;
|
|
|
|
case IGLDevice::GreaterOrEqual: glTexParameteri(targ, GL_TEXTURE_COMPARE_FUNC, GL_GEQUAL); break;
|
|
|
|
case IGLDevice::Less: glTexParameteri(targ, GL_TEXTURE_COMPARE_FUNC, GL_LESS); break;
|
|
|
|
case IGLDevice::Greater: glTexParameteri(targ, GL_TEXTURE_COMPARE_FUNC, GL_GREATER); break;
|
|
|
|
case IGLDevice::Equal: glTexParameteri(targ, GL_TEXTURE_COMPARE_FUNC, GL_EQUAL); break;
|
|
|
|
case IGLDevice::NotEqual: glTexParameteri(targ, GL_TEXTURE_COMPARE_FUNC, GL_NOTEQUAL); break;
|
|
|
|
case IGLDevice::Always: glTexParameteri(targ, GL_TEXTURE_COMPARE_FUNC, GL_ALWAYS); break;
|
|
|
|
case IGLDevice::Never: glTexParameteri(targ, GL_TEXTURE_COMPARE_FUNC, GL_NEVER); break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("val", val);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("param", param);
|
|
|
|
}
|
|
|
|
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::GenerateMipmap(spades::draw::IGLDevice::Enum target){
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glGenerateMipmap)
|
|
|
|
glGenerateMipmap(parseTextureTarget(target));
|
|
|
|
else if(glGenerateMipmapEXT)
|
|
|
|
glGenerateMipmapEXT(parseTextureTarget(target));
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glGenerateMipmap");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glGenerateMipmap);
|
2013-08-18 16:18:06 +09:00
|
|
|
glGenerateMipmap(parseTextureTarget(target));
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::VertexAttrib(UInteger index, Float x) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glVertexAttrib1f)
|
|
|
|
glVertexAttrib1f(index, x);
|
|
|
|
else if(glVertexAttrib1fARB)
|
|
|
|
glVertexAttrib1fARB(index, x);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glVertexAttrib1f");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glVertexAttrib1f);
|
2013-08-18 16:18:06 +09:00
|
|
|
glVertexAttrib1f(index, x);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::VertexAttrib(UInteger index, Float x, Float y) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glVertexAttrib2f)
|
|
|
|
glVertexAttrib2f(index, x, y);
|
|
|
|
else if(glVertexAttrib2fARB)
|
|
|
|
glVertexAttrib2fARB(index, x, y);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glVertexAttrib2f");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glVertexAttrib2f);
|
2013-08-18 16:18:06 +09:00
|
|
|
glVertexAttrib2f(index, x, y);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::VertexAttrib(UInteger index, Float x, Float y, Float z) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glVertexAttrib3f)
|
|
|
|
glVertexAttrib3f(index, x, y, z);
|
|
|
|
else if(glVertexAttrib3fARB)
|
|
|
|
glVertexAttrib3fARB(index, x, y, z);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glVertexAttrib3f");
|
|
|
|
#else
|
|
|
|
CheckExistence(glVertexAttrib2f);
|
2013-08-18 16:18:06 +09:00
|
|
|
glVertexAttrib3f(index, x, y, z);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::VertexAttrib(UInteger index, Float x, Float y, Float z, Float w) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glVertexAttrib4f)
|
|
|
|
glVertexAttrib4f(index, x, y, z, w);
|
|
|
|
else if(glVertexAttrib4fARB)
|
|
|
|
glVertexAttrib4fARB(index, x, y, z, w);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glVertexAttrib4f");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glVertexAttrib4f);
|
2013-08-18 16:18:06 +09:00
|
|
|
glVertexAttrib4f(index, x, y, z, w);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::VertexAttribPointer(UInteger index,
|
|
|
|
Integer size,
|
|
|
|
Enum type, bool normalized,
|
|
|
|
Sizei stride,
|
|
|
|
const void *data) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glVertexAttribPointer)
|
|
|
|
glVertexAttribPointer(index, size, parseType(type),
|
|
|
|
normalized, stride, data);
|
|
|
|
else if(glVertexAttribPointerARB)
|
|
|
|
glVertexAttribPointerARB(index, size, parseType(type),
|
|
|
|
normalized, stride, data);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glVertexAttribPointer");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glVertexAttribPointer);
|
2013-08-18 16:18:06 +09:00
|
|
|
glVertexAttribPointer(index, size, parseType(type),
|
|
|
|
normalized, stride, data);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::VertexAttribIPointer(UInteger index,
|
|
|
|
Integer size,
|
|
|
|
Enum type,
|
|
|
|
Sizei stride,
|
|
|
|
const void *data) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glVertexAttribIPointer)
|
|
|
|
glVertexAttribIPointer(index, size, parseType(type),
|
|
|
|
stride, data);
|
|
|
|
else if(glVertexAttribIPointerEXT)
|
|
|
|
glVertexAttribIPointerEXT(index, size, parseType(type),
|
|
|
|
stride, data);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glVertexAttribPointer");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glVertexAttribIPointer);
|
2013-08-18 16:18:06 +09:00
|
|
|
glVertexAttribIPointer(index, size, parseType(type),
|
2013-08-27 01:36:28 +09:00
|
|
|
stride, data);
|
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::EnableVertexAttribArray(UInteger index, bool b){
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glEnableVertexAttribArray){
|
|
|
|
if(b)
|
|
|
|
glEnableVertexAttribArray(index);
|
|
|
|
else
|
|
|
|
glDisableVertexAttribArray(index);
|
|
|
|
}else if(glEnableVertexAttribArrayARB){
|
|
|
|
if(b)
|
|
|
|
glEnableVertexAttribArrayARB(index);
|
|
|
|
else
|
|
|
|
glDisableVertexAttribArrayARB(index);
|
|
|
|
}else
|
|
|
|
ReportMissingFunc("glEnableVertexAttribArray");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glEnableVertexAttribArray);
|
|
|
|
CheckExistence(glDisableVertexAttribArray);
|
2013-08-18 16:18:06 +09:00
|
|
|
if(b)
|
|
|
|
glEnableVertexAttribArray(index);
|
|
|
|
else
|
|
|
|
glDisableVertexAttribArray(index);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::VertexAttribDivisor(UInteger index, UInteger divisor) {
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glVertexAttribDivisorARB);
|
2013-08-18 16:18:06 +09:00
|
|
|
glVertexAttribDivisorARB(index, divisor);
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::DrawArrays(Enum mode, Integer first, Sizei count) {
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
GLenum md;
|
|
|
|
switch(mode){
|
|
|
|
case Points: md = GL_POINTS; break;
|
|
|
|
case LineStrip: md = GL_LINE_STRIP; break;
|
|
|
|
case LineLoop: md = GL_LINE_LOOP; break;
|
|
|
|
case Lines: md = GL_LINES; break;
|
|
|
|
case TriangleStrip: md = GL_TRIANGLE_STRIP; break;
|
|
|
|
case TriangleFan: md = GL_TRIANGLE_FAN; break;
|
|
|
|
case Triangles: md = GL_TRIANGLES; break;
|
|
|
|
default: SPInvalidEnum("mode", mode);
|
|
|
|
}
|
|
|
|
vertCount += count;
|
|
|
|
drawOps++;
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glDrawArrays);
|
2013-08-18 16:18:06 +09:00
|
|
|
glDrawArrays(md, first, count);
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::DrawElements(Enum mode, Sizei count, Enum type, const void *indices) {
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
GLenum md;
|
|
|
|
switch(mode){
|
|
|
|
case Points: md = GL_POINTS; break;
|
|
|
|
case LineStrip: md = GL_LINE_STRIP; break;
|
|
|
|
case LineLoop: md = GL_LINE_LOOP; break;
|
|
|
|
case Lines: md = GL_LINES; break;
|
|
|
|
case TriangleStrip: md = GL_TRIANGLE_STRIP; break;
|
|
|
|
case TriangleFan: md = GL_TRIANGLE_FAN; break;
|
|
|
|
case Triangles: md = GL_TRIANGLES; break;
|
|
|
|
default: SPInvalidEnum("mode", mode);
|
|
|
|
}
|
|
|
|
vertCount += count;
|
|
|
|
drawOps++;
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glDrawElements);
|
2013-08-18 16:18:06 +09:00
|
|
|
glDrawElements(md, count, parseType(type), indices);
|
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::DrawArraysInstanced(Enum mode, Integer first, Sizei count,
|
|
|
|
Sizei instances) {
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
GLenum md;
|
|
|
|
switch(mode){
|
|
|
|
case Points: md = GL_POINTS; break;
|
|
|
|
case LineStrip: md = GL_LINE_STRIP; break;
|
|
|
|
case LineLoop: md = GL_LINE_LOOP; break;
|
|
|
|
case Lines: md = GL_LINES; break;
|
|
|
|
case TriangleStrip: md = GL_TRIANGLE_STRIP; break;
|
|
|
|
case TriangleFan: md = GL_TRIANGLE_FAN; break;
|
|
|
|
case Triangles: md = GL_TRIANGLES; break;
|
|
|
|
default: SPInvalidEnum("mode", mode);
|
|
|
|
}
|
|
|
|
#if GLEW
|
|
|
|
if(glDrawArraysInstanced)
|
|
|
|
glDrawArraysInstanced(md, first, count, instances);
|
|
|
|
else if(glDrawArraysInstancedARB)
|
|
|
|
glDrawArraysInstancedARB(md, first, count, instances);
|
|
|
|
else if(glDrawArraysInstancedEXT)
|
|
|
|
glDrawArraysInstancedEXT(md, first, count, instances);
|
|
|
|
else
|
2013-08-27 01:36:28 +09:00
|
|
|
ReportMissingFunc("glDrawArraysInstanced");
|
2013-08-18 16:18:06 +09:00
|
|
|
#else
|
|
|
|
glDrawArraysInstanced(md, first, count, instances);
|
|
|
|
#endif
|
|
|
|
CheckError();
|
|
|
|
vertCount += count * instances;
|
|
|
|
drawOps++;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::DrawElementsInstanced(Enum mode, Sizei count, Enum type, const void *indices,
|
|
|
|
Sizei instances) {
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
GLenum md;
|
|
|
|
switch(mode){
|
|
|
|
case Points: md = GL_POINTS; break;
|
|
|
|
case LineStrip: md = GL_LINE_STRIP; break;
|
|
|
|
case LineLoop: md = GL_LINE_LOOP; break;
|
|
|
|
case Lines: md = GL_LINES; break;
|
|
|
|
case TriangleStrip: md = GL_TRIANGLE_STRIP; break;
|
|
|
|
case TriangleFan: md = GL_TRIANGLE_FAN; break;
|
|
|
|
case Triangles: md = GL_TRIANGLES; break;
|
|
|
|
default: SPInvalidEnum("mode", mode);
|
|
|
|
}
|
|
|
|
#if GLEW
|
|
|
|
if(glDrawElementsInstanced)
|
|
|
|
glDrawElementsInstanced(md, count, parseType(type), indices, instances);
|
|
|
|
else if(glDrawElementsInstancedARB)
|
|
|
|
glDrawElementsInstancedARB(md, count, parseType(type), indices, instances);
|
|
|
|
else if(glDrawElementsInstancedEXT)
|
|
|
|
glDrawElementsInstancedEXT(md, count, parseType(type), indices, instances);
|
|
|
|
else
|
2013-08-27 01:36:28 +09:00
|
|
|
ReportMissingFunc("glDrawElementsInstanced");
|
2013-08-18 16:18:06 +09:00
|
|
|
#else
|
|
|
|
glDrawElementsInstanced(md, count, parseType(type), indices, instances);
|
|
|
|
#endif
|
|
|
|
CheckError();
|
|
|
|
vertCount += count * instances;
|
|
|
|
drawOps++;
|
|
|
|
}
|
|
|
|
|
|
|
|
IGLDevice::UInteger SDLGLDevice::CreateShader(Enum type) {
|
|
|
|
SPADES_MARK_FUNCTION();
|
2013-08-27 03:56:56 +09:00
|
|
|
IGLDevice::UInteger ret;
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glCreateShader)
|
|
|
|
switch(type){
|
|
|
|
case draw::IGLDevice::FragmentShader:
|
2013-08-27 03:56:56 +09:00
|
|
|
ret = glCreateShader(GL_FRAGMENT_SHADER);
|
|
|
|
break;
|
2013-08-27 01:36:28 +09:00
|
|
|
case draw::IGLDevice::VertexShader:
|
2013-08-27 03:56:56 +09:00
|
|
|
ret = glCreateShader(GL_VERTEX_SHADER);
|
|
|
|
break;
|
2013-08-27 01:36:28 +09:00
|
|
|
default: SPInvalidEnum("type", type);
|
|
|
|
}
|
|
|
|
else if(glCreateShaderObjectARB)
|
|
|
|
switch(type){
|
|
|
|
case draw::IGLDevice::FragmentShader:
|
2013-08-27 03:56:56 +09:00
|
|
|
ret = glCreateShaderObjectARB(GL_FRAGMENT_SHADER_ARB);
|
|
|
|
break;
|
2013-08-27 01:36:28 +09:00
|
|
|
case draw::IGLDevice::VertexShader:
|
2013-08-27 03:56:56 +09:00
|
|
|
ret = glCreateShaderObjectARB(GL_VERTEX_SHADER_ARB);
|
|
|
|
break;
|
2013-08-27 01:36:28 +09:00
|
|
|
default: SPInvalidEnum("type", type);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glCreateShader");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glCreateShader);
|
2013-08-18 16:18:06 +09:00
|
|
|
switch(type){
|
|
|
|
case draw::IGLDevice::FragmentShader:
|
2013-08-27 03:56:56 +09:00
|
|
|
ret = glCreateShader(GL_FRAGMENT_SHADER); break;
|
2013-08-18 16:18:06 +09:00
|
|
|
case draw::IGLDevice::VertexShader:
|
2013-08-27 03:56:56 +09:00
|
|
|
ret = glCreateShader(GL_VERTEX_SHADER); break;
|
2013-08-18 16:18:06 +09:00
|
|
|
default: SPInvalidEnum("type", type);
|
|
|
|
}
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-27 03:56:56 +09:00
|
|
|
return ret;
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::ShaderSource(UInteger shader, Sizei count,
|
|
|
|
const char **string, const int *len) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glShaderSource)
|
|
|
|
glShaderSource(shader, count, (const GLchar **)string, len);
|
|
|
|
else if(glShaderSourceARB)
|
|
|
|
glShaderSourceARB(shader, count, (const GLchar **)string, len);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glShaderSource");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glShaderSource);
|
2013-08-18 16:18:06 +09:00
|
|
|
glShaderSource(shader, count, (const GLchar **)string, len);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::CompileShader(UInteger i) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glCompileShader)
|
|
|
|
glCompileShader(i);
|
|
|
|
else if(glCompileShaderARB)
|
|
|
|
glCompileShaderARB(i);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glCompileShader");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glCompileShader);
|
2013-08-18 16:18:06 +09:00
|
|
|
glCompileShader(i);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::DeleteShader(UInteger i){
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glDeleteShader)
|
|
|
|
glDeleteShader(i);
|
|
|
|
else if(glDeleteObjectARB)
|
|
|
|
glDeleteObjectARB(i);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glDeleteShader");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glDeleteShader);
|
2013-08-18 16:18:06 +09:00
|
|
|
glDeleteShader(i);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
IGLDevice::Integer SDLGLDevice::GetShaderInteger(UInteger shader,
|
|
|
|
Enum param) {
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
GLint ret;
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glGetShaderiv)
|
|
|
|
switch(param){
|
|
|
|
case ShaderType:
|
|
|
|
glGetShaderiv(shader, GL_SHADER_TYPE, &ret);
|
|
|
|
break;
|
|
|
|
case DeleteStatus:
|
|
|
|
glGetShaderiv(shader, GL_DELETE_STATUS, &ret);
|
|
|
|
break;
|
|
|
|
case CompileStatus:
|
|
|
|
glGetShaderiv(shader, GL_COMPILE_STATUS, &ret);
|
|
|
|
break;
|
|
|
|
case InfoLogLength:
|
|
|
|
glGetShaderiv(shader, GL_INFO_LOG_LENGTH, &ret);
|
|
|
|
break;
|
|
|
|
case ShaderSourceLength:
|
|
|
|
glGetShaderiv(shader, GL_SHADER_SOURCE_LENGTH, &ret);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("param", param);
|
|
|
|
}
|
|
|
|
else if(glGetObjectParameterivARB)
|
|
|
|
switch(param){
|
|
|
|
case ShaderType:
|
|
|
|
SPRaise("GL_SHADER_TYPE not supported for GL_ARB_shader_objects");
|
|
|
|
case DeleteStatus:
|
|
|
|
glGetObjectParameterivARB(shader, GL_OBJECT_DELETE_STATUS_ARB, &ret);
|
|
|
|
break;
|
|
|
|
case CompileStatus:
|
|
|
|
glGetObjectParameterivARB(shader, GL_OBJECT_COMPILE_STATUS_ARB, &ret);
|
|
|
|
break;
|
|
|
|
case InfoLogLength:
|
|
|
|
glGetObjectParameterivARB(shader, GL_OBJECT_INFO_LOG_LENGTH_ARB, &ret);
|
|
|
|
break;
|
|
|
|
case ShaderSourceLength:
|
|
|
|
glGetObjectParameterivARB(shader, GL_OBJECT_SHADER_SOURCE_LENGTH_ARB, &ret);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("param", param);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glGetShaderiv");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glGetShaderiv);
|
2013-08-18 16:18:06 +09:00
|
|
|
switch(param){
|
|
|
|
case ShaderType:
|
|
|
|
glGetShaderiv(shader, GL_SHADER_TYPE, &ret);
|
|
|
|
break;
|
|
|
|
case DeleteStatus:
|
|
|
|
glGetShaderiv(shader, GL_DELETE_STATUS, &ret);
|
|
|
|
break;
|
|
|
|
case CompileStatus:
|
|
|
|
glGetShaderiv(shader, GL_COMPILE_STATUS, &ret);
|
|
|
|
break;
|
|
|
|
case InfoLogLength:
|
|
|
|
glGetShaderiv(shader, GL_INFO_LOG_LENGTH, &ret);
|
|
|
|
break;
|
|
|
|
case ShaderSourceLength:
|
|
|
|
glGetShaderiv(shader, GL_SHADER_SOURCE_LENGTH, &ret);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("param", param);
|
|
|
|
}
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::GetShaderInfoLog(UInteger shader, Sizei bufferSize,
|
|
|
|
Sizei *length, char *outString) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glGetShaderInfoLog)
|
|
|
|
glGetShaderInfoLog(shader, bufferSize, (GLsizei *)length,
|
|
|
|
(GLchar *)outString);
|
|
|
|
else if(glGetInfoLogARB)
|
|
|
|
glGetInfoLogARB(shader, bufferSize, (GLsizei *)length,
|
|
|
|
(GLchar *)outString);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glGetShaderInfoLog");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glGetShaderInfoLog);
|
2013-08-18 16:18:06 +09:00
|
|
|
glGetShaderInfoLog(shader, bufferSize, (GLsizei *)length,
|
|
|
|
(GLchar *)outString);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
IGLDevice::Integer SDLGLDevice::GetProgramInteger(UInteger shader,
|
|
|
|
Enum param) {
|
|
|
|
SPADES_MARK_FUNCTION();
|
|
|
|
GLint ret;
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glGetProgramiv)
|
|
|
|
switch(param){
|
|
|
|
case DeleteStatus:
|
|
|
|
glGetProgramiv(shader, GL_DELETE_STATUS, &ret);
|
|
|
|
break;
|
|
|
|
case LinkStatus:
|
|
|
|
glGetProgramiv(shader, GL_LINK_STATUS, &ret);
|
|
|
|
break;
|
|
|
|
case ValidateStatus:
|
|
|
|
glGetProgramiv(shader, GL_VALIDATE_STATUS, &ret);
|
|
|
|
break;
|
|
|
|
case InfoLogLength:
|
|
|
|
glGetProgramiv(shader, GL_INFO_LOG_LENGTH, &ret);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("param", param);
|
|
|
|
}
|
|
|
|
else if(glGetObjectParameterivARB)
|
|
|
|
switch(param){
|
|
|
|
case DeleteStatus:
|
|
|
|
glGetObjectParameterivARB(shader, GL_OBJECT_DELETE_STATUS_ARB, &ret);
|
|
|
|
break;
|
|
|
|
case LinkStatus:
|
|
|
|
glGetObjectParameterivARB(shader, GL_OBJECT_LINK_STATUS_ARB, &ret);
|
|
|
|
break;
|
|
|
|
case ValidateStatus:
|
|
|
|
glGetObjectParameterivARB(shader, GL_OBJECT_VALIDATE_STATUS_ARB, &ret);
|
|
|
|
break;
|
|
|
|
case InfoLogLength:
|
|
|
|
glGetObjectParameterivARB(shader, GL_OBJECT_INFO_LOG_LENGTH_ARB, &ret);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("param", param);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glGetProgramiv");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glGetProgramiv);
|
2013-08-18 16:18:06 +09:00
|
|
|
switch(param){
|
|
|
|
case DeleteStatus:
|
|
|
|
glGetProgramiv(shader, GL_DELETE_STATUS, &ret);
|
|
|
|
break;
|
|
|
|
case LinkStatus:
|
|
|
|
glGetProgramiv(shader, GL_LINK_STATUS, &ret);
|
|
|
|
break;
|
|
|
|
case ValidateStatus:
|
|
|
|
glGetProgramiv(shader, GL_VALIDATE_STATUS, &ret);
|
|
|
|
break;
|
|
|
|
case InfoLogLength:
|
|
|
|
glGetProgramiv(shader, GL_INFO_LOG_LENGTH, &ret);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("param", param);
|
|
|
|
}
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::GetProgramInfoLog(UInteger p, Sizei bufferSize,
|
|
|
|
Sizei *length, char *outString) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glGetProgramInfoLog)
|
|
|
|
glGetProgramInfoLog(p, bufferSize, (GLsizei *)length,
|
|
|
|
(GLchar *)outString);
|
|
|
|
else if(glGetInfoLogARB)
|
|
|
|
glGetInfoLogARB(p, bufferSize, (GLsizei *)length,
|
|
|
|
(GLchar *)outString);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glGetShaderInfoLog");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glGetProgramInfoLog);
|
2013-08-18 16:18:06 +09:00
|
|
|
glGetProgramInfoLog(p, bufferSize, (GLsizei *)length,
|
|
|
|
(GLchar *)outString);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
IGLDevice::UInteger SDLGLDevice::CreateProgram() {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glCreateProgram)
|
|
|
|
return glCreateProgram();
|
|
|
|
else if(glCreateProgramObjectARB)
|
|
|
|
return glCreateProgramObjectARB();
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glCreateProgram");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glCreateProgram);
|
2013-08-18 16:18:06 +09:00
|
|
|
return glCreateProgram();
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::AttachShader(UInteger program, UInteger shader){
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glAttachShader)
|
|
|
|
glAttachShader(program, shader);
|
|
|
|
else if(glAttachObjectARB)
|
|
|
|
glAttachObjectARB(program, shader);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glAttachShader");
|
|
|
|
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glAttachShader);
|
2013-08-18 16:18:06 +09:00
|
|
|
glAttachShader(program, shader);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::DetachShader(UInteger program, UInteger shader){
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glDetachShader)
|
|
|
|
glDetachShader(program, shader);
|
|
|
|
else if(glDetachObjectARB)
|
|
|
|
glDetachObjectARB(program, shader);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glDetachShader");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glDetachShader);
|
2013-08-18 16:18:06 +09:00
|
|
|
glDetachShader(program, shader);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::LinkProgram(UInteger program) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glLinkProgram)
|
|
|
|
glLinkProgram(program);
|
|
|
|
else if(glLinkProgramARB)
|
|
|
|
glLinkProgramARB(program);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glLinkProgram");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glLinkProgram);
|
2013-08-18 16:18:06 +09:00
|
|
|
glLinkProgram(program);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::UseProgram(UInteger program) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glUseProgram)
|
|
|
|
glUseProgram(program);
|
|
|
|
else if(glUseProgramObjectARB)
|
|
|
|
glUseProgramObjectARB(program);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glUseProgram");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glUseProgram);
|
2013-08-18 16:18:06 +09:00
|
|
|
glUseProgram(program);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::DeleteProgram(UInteger program) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glDeleteProgram)
|
|
|
|
glDeleteProgram(program);
|
|
|
|
else if(glDeleteObjectARB)
|
|
|
|
glDeleteObjectARB(program);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glDeleteProgram");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glDeleteProgram);
|
2013-08-18 16:18:06 +09:00
|
|
|
glDeleteProgram(program);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::ValidateProgram(UInteger program) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glValidateProgram)
|
|
|
|
glValidateProgram(program);
|
|
|
|
else if(glValidateProgramARB)
|
|
|
|
glValidateProgramARB(program);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glValidateProgram");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glValidateProgram);
|
2013-08-18 16:18:06 +09:00
|
|
|
glValidateProgram(program);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
IGLDevice::Integer SDLGLDevice::GetAttribLocation(UInteger program, const char *name){
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glGetAttribLocation)
|
|
|
|
return glGetAttribLocation(program, name);
|
|
|
|
else if(glGetAttribLocationARB)
|
|
|
|
return glGetAttribLocationARB(program, name);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glGetAttribLocation");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glGetAttribLocation);
|
2013-08-18 16:18:06 +09:00
|
|
|
return glGetAttribLocation(program, name);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::BindAttribLocation(UInteger program, UInteger index, const char *name) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glBindAttribLocation)
|
|
|
|
glBindAttribLocation(program, index, name);
|
|
|
|
else if(glBindAttribLocationARB)
|
|
|
|
glBindAttribLocationARB(program, index, name);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glBindAttribLocation");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glBindAttribLocation);
|
2013-08-18 16:18:06 +09:00
|
|
|
glBindAttribLocation(program, index, name);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
IGLDevice::Integer SDLGLDevice::GetUniformLocation(UInteger program, const char *name) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glGetUniformLocation)
|
|
|
|
return glGetUniformLocation(program, name);
|
|
|
|
else if(glGetUniformLocationARB)
|
|
|
|
return glGetUniformLocationARB(program, name);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glGetUniformLocation");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glGetUniformLocation);
|
2013-08-18 16:18:06 +09:00
|
|
|
return glGetUniformLocation(program, name);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::Uniform(Integer loc, Float x) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glUniform1f)
|
|
|
|
glUniform1f(loc, x);
|
|
|
|
else if(glUniform1fARB)
|
|
|
|
glUniform1fARB(loc, x);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glUniform1f");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glUniform1f);
|
2013-08-18 16:18:06 +09:00
|
|
|
glUniform1f(loc, x);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::Uniform(Integer loc, Float x, Float y) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glUniform2f)
|
|
|
|
glUniform2f(loc, x, y);
|
|
|
|
else if(glUniform2fARB)
|
|
|
|
glUniform2fARB(loc, x, y);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glUniform2f");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glUniform2f);
|
2013-08-18 16:18:06 +09:00
|
|
|
glUniform2f(loc, x, y);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::Uniform(Integer loc, Float x, Float y, Float z) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glUniform3f)
|
|
|
|
glUniform3f(loc, x, y, z);
|
|
|
|
else if(glUniform3fARB)
|
|
|
|
glUniform3fARB(loc, x, y, z);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glUniform3f");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glUniform3f);
|
2013-08-18 16:18:06 +09:00
|
|
|
glUniform3f(loc, x, y, z);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::Uniform(Integer loc, Float x, Float y, Float z, Float w) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glUniform4f)
|
|
|
|
glUniform4f(loc, x, y, z, w);
|
|
|
|
else if(glUniform4fARB)
|
|
|
|
glUniform4fARB(loc, x, y, z, w);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glUniform4f");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glUniform4f);
|
2013-08-18 16:18:06 +09:00
|
|
|
glUniform4f(loc, x, y, z, w);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::Uniform(Integer loc, Integer x) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glUniform1i)
|
|
|
|
glUniform1i(loc, x);
|
|
|
|
else if(glUniform1iARB)
|
|
|
|
glUniform1iARB(loc, x);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glUniform1i");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glUniform1i);
|
2013-08-18 16:18:06 +09:00
|
|
|
glUniform1i(loc, x);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::Uniform(Integer loc, Integer x, Integer y) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glUniform2i)
|
|
|
|
glUniform2i(loc, x, y);
|
|
|
|
else if(glUniform2iARB)
|
|
|
|
glUniform2iARB(loc, x, y);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glUniform2i");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glUniform2i);
|
2013-08-18 16:18:06 +09:00
|
|
|
glUniform2i(loc, x, y);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::Uniform(Integer loc, Integer x, Integer y, Integer z) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glUniform3i)
|
|
|
|
glUniform3i(loc, x, y, z);
|
|
|
|
else if(glUniform3iARB)
|
|
|
|
glUniform3iARB(loc, x, y, z);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glUniform3i");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glUniform3i);
|
2013-08-18 16:18:06 +09:00
|
|
|
glUniform3i(loc, x, y, z);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::Uniform(Integer loc, Integer x, Integer y, Integer z, Integer w) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glUniform4i)
|
|
|
|
glUniform4i(loc, x, y, z, w);
|
|
|
|
else if(glUniform4iARB)
|
|
|
|
glUniform4iARB(loc, x, y, z, w);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glUniform4i");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glUniform4i);
|
2013-08-18 16:18:06 +09:00
|
|
|
glUniform4i(loc, x, y, z, w);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::Uniform(Integer loc, bool transpose, const spades::Matrix4 &mat) {
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glUniformMatrix4fv)
|
|
|
|
glUniformMatrix4fv(loc, 1, transpose ? GL_TRUE : GL_FALSE,
|
|
|
|
mat.m);
|
|
|
|
else if(glUniformMatrix4fvARB)
|
|
|
|
glUniformMatrix4fvARB(loc, 1, transpose ? GL_TRUE : GL_FALSE,
|
|
|
|
mat.m);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glUniformMatrix4fv");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glUniformMatrix4fv);
|
2013-08-18 16:18:06 +09:00
|
|
|
glUniformMatrix4fv(loc, 1, transpose ? GL_TRUE : GL_FALSE,
|
|
|
|
mat.m);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
GLenum SDLGLDevice::parseFramebufferTarget(Enum v){
|
|
|
|
SPADES_MARK_FUNCTION_DEBUG();
|
|
|
|
switch(v){
|
|
|
|
case draw::IGLDevice::Framebuffer:
|
|
|
|
return GL_FRAMEBUFFER;
|
|
|
|
case draw::IGLDevice::ReadFramebuffer:
|
|
|
|
return GL_READ_FRAMEBUFFER;
|
|
|
|
case draw::IGLDevice::DrawFramebuffer:
|
|
|
|
return GL_DRAW_FRAMEBUFFER;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("v", v);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
IGLDevice::UInteger SDLGLDevice::GenFramebuffer() {
|
|
|
|
GLuint v;
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glGenFramebuffers)
|
|
|
|
glGenFramebuffers(1, &v);
|
|
|
|
else if(glGenFramebuffersEXT)
|
|
|
|
glGenFramebuffersEXT(1, &v);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glGenFramebuffers");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glGenFramebuffers);
|
2013-08-18 16:18:06 +09:00
|
|
|
glGenFramebuffers(1, &v);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-27 17:36:20 +09:00
|
|
|
CheckError();
|
2013-08-18 16:18:06 +09:00
|
|
|
return (IGLDevice::UInteger)v;
|
|
|
|
}
|
|
|
|
void SDLGLDevice::BindFramebuffer(Enum target,
|
|
|
|
UInteger framebuffer) {
|
2013-08-27 17:36:20 +09:00
|
|
|
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glBindFramebuffer)
|
|
|
|
glBindFramebuffer(parseFramebufferTarget(target),
|
|
|
|
framebuffer);
|
|
|
|
else if(glBindFramebufferEXT)
|
|
|
|
glBindFramebufferEXT(parseFramebufferTarget(target),
|
|
|
|
framebuffer);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glBindFramebuffer");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glBindFramebuffer);
|
2013-08-18 16:18:06 +09:00
|
|
|
glBindFramebuffer(parseFramebufferTarget(target),
|
|
|
|
framebuffer);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::DeleteFramebuffer(UInteger fb){
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glDeleteFramebuffers)
|
|
|
|
glDeleteFramebuffers(1, &fb);
|
|
|
|
else if(glDeleteFramebuffersEXT)
|
|
|
|
glDeleteFramebuffersEXT(1, &fb);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glDeleteFramebuffers");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glDeleteFramebuffers);
|
2013-08-18 16:18:06 +09:00
|
|
|
glDeleteFramebuffers(1, &fb);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
2013-08-27 17:36:20 +09:00
|
|
|
IGLDevice::Enum SDLGLDevice::CheckFramebufferStatus(spades::draw::IGLDevice::Enum target) {
|
|
|
|
GLenum ret;
|
|
|
|
#if GLEW
|
|
|
|
if(glCheckFramebufferStatus)
|
|
|
|
ret = glCheckFramebufferStatus(parseFramebufferTarget(target));
|
|
|
|
else if(glCheckFramebufferStatusEXT)
|
|
|
|
ret = glCheckFramebufferStatusEXT(parseFramebufferTarget(target));
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glCheckFramebufferStatus");
|
|
|
|
#else
|
|
|
|
CheckExistence(glCheckFramebufferStatus);
|
|
|
|
ret = glCheckFramebufferStatus(parseFramebufferTarget(target));
|
|
|
|
#endif
|
|
|
|
CheckError();
|
|
|
|
switch(ret){
|
|
|
|
case GL_FRAMEBUFFER_COMPLETE:
|
|
|
|
return FramebufferComplete;
|
|
|
|
case GL_FRAMEBUFFER_UNDEFINED:
|
|
|
|
return FramebufferUndefined;
|
|
|
|
case GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT:
|
|
|
|
return FramebufferIncompleteAttachment;
|
|
|
|
case GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT:
|
|
|
|
return FramebufferIncompleteMissingAttachment;
|
|
|
|
case GL_FRAMEBUFFER_INCOMPLETE_DRAW_BUFFER:
|
|
|
|
return FramebufferIncompleteDrawBuffer;
|
|
|
|
case GL_FRAMEBUFFER_INCOMPLETE_READ_BUFFER:
|
|
|
|
return FramebufferIncompleteReadBuffer;
|
|
|
|
case GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE:
|
|
|
|
return FramebufferIncompleteMultisample;
|
|
|
|
case GL_FRAMEBUFFER_INCOMPLETE_LAYER_TARGETS:
|
|
|
|
return FramebufferIncompleteLayerTargets;
|
|
|
|
default:
|
|
|
|
return FramebufferUnsupported;
|
|
|
|
}
|
|
|
|
}
|
2013-08-18 16:18:06 +09:00
|
|
|
void SDLGLDevice::FramebufferTexture2D(Enum target,
|
|
|
|
Enum attachment,
|
|
|
|
Enum texTarget,
|
|
|
|
UInteger texture,
|
|
|
|
Integer level) {
|
|
|
|
SPADES_MARK_FUNCTION_DEBUG();
|
|
|
|
|
|
|
|
GLenum a;
|
|
|
|
switch(attachment){
|
|
|
|
case draw::IGLDevice::ColorAttachment0:
|
|
|
|
a = GL_COLOR_ATTACHMENT0;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::ColorAttachment1:
|
|
|
|
a = GL_COLOR_ATTACHMENT1;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::ColorAttachment2:
|
|
|
|
a = GL_COLOR_ATTACHMENT2;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::ColorAttachment3:
|
|
|
|
a = GL_COLOR_ATTACHMENT3;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::ColorAttachment4:
|
|
|
|
a = GL_COLOR_ATTACHMENT4;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::ColorAttachment5:
|
|
|
|
a = GL_COLOR_ATTACHMENT5;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::ColorAttachment6:
|
|
|
|
a = GL_COLOR_ATTACHMENT6;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::ColorAttachment7:
|
|
|
|
a = GL_COLOR_ATTACHMENT7;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::DepthAttachment:
|
|
|
|
a = GL_DEPTH_ATTACHMENT;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::StencilAttachment:
|
|
|
|
a = GL_STENCIL_ATTACHMENT;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("attachment",attachment);
|
|
|
|
}
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glFramebufferTexture2D)
|
|
|
|
glFramebufferTexture2D(parseFramebufferTarget(target),
|
|
|
|
a, parseTextureTarget(texTarget),
|
|
|
|
texture, level);
|
|
|
|
else if(glFramebufferTexture2DEXT)
|
|
|
|
glFramebufferTexture2DEXT(parseFramebufferTarget(target),
|
|
|
|
a, parseTextureTarget(texTarget),
|
|
|
|
texture, level);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glFramebufferTexture2D");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glFramebufferTexture2D);
|
2013-08-18 16:18:06 +09:00
|
|
|
glFramebufferTexture2D(parseFramebufferTarget(target),
|
|
|
|
a, parseTextureTarget(texTarget),
|
|
|
|
texture, level);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-27 03:56:56 +09:00
|
|
|
CheckErrorAlways();
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::BlitFramebuffer(Integer srcX0,
|
|
|
|
Integer srcY0,
|
|
|
|
Integer srcX1,
|
|
|
|
Integer srcY1,
|
|
|
|
Integer dstX0,
|
|
|
|
Integer dstY0,
|
|
|
|
Integer dstX1,
|
|
|
|
Integer dstY1,
|
|
|
|
UInteger mask,
|
|
|
|
Enum filter){
|
|
|
|
SPADES_MARK_FUNCTION_DEBUG();
|
|
|
|
|
|
|
|
GLenum flt;
|
|
|
|
switch(filter){
|
|
|
|
case draw::IGLDevice::Linear:
|
|
|
|
flt = GL_LINEAR;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::Nearest:
|
|
|
|
flt = GL_NEAREST;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("filter", filter);
|
|
|
|
}
|
|
|
|
|
|
|
|
GLbitfield m = 0;
|
|
|
|
if(mask & ColorBufferBit)
|
|
|
|
m |= GL_COLOR_BUFFER_BIT;
|
|
|
|
if(mask & DepthBufferBit)
|
|
|
|
m |= GL_DEPTH_BUFFER_BIT;
|
|
|
|
if(mask & StencilBufferBit)
|
|
|
|
m |= GL_STENCIL_BUFFER_BIT;
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glBlitFramebuffer)
|
|
|
|
glBlitFramebuffer(srcX0, srcY0, srcX1, srcY1,
|
|
|
|
dstX0, dstY0, dstX1, dstY1,
|
|
|
|
m, flt);
|
|
|
|
else if(glBlitFramebufferEXT)
|
|
|
|
glBlitFramebufferEXT(srcX0, srcY0, srcX1, srcY1,
|
|
|
|
dstX0, dstY0, dstX1, dstY1,
|
|
|
|
m, flt);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glBlitFramebuffer");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glBlitFramebuffer);
|
2013-08-18 16:18:06 +09:00
|
|
|
glBlitFramebuffer(srcX0, srcY0, srcX1, srcY1,
|
|
|
|
dstX0, dstY0, dstX1, dstY1,
|
|
|
|
m, flt);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
|
|
|
|
GLenum SDLGLDevice::parseRenderbufferTarget(Enum e){
|
|
|
|
SPADES_MARK_FUNCTION_DEBUG();
|
|
|
|
switch(e){
|
|
|
|
case draw::IGLDevice::Renderbuffer:
|
|
|
|
return GL_RENDERBUFFER;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("e", e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
IGLDevice::UInteger SDLGLDevice::GenRenderbuffer() {
|
|
|
|
GLuint v;
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glGenRenderbuffers)
|
|
|
|
glGenRenderbuffers(1, &v);
|
|
|
|
else if(glGenRenderbuffersEXT)
|
|
|
|
glGenRenderbuffersEXT(1, &v);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glGenRenderbuffers");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glGenRenderbuffers);
|
2013-08-18 16:18:06 +09:00
|
|
|
glGenRenderbuffers(1, &v);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
void SDLGLDevice::DeleteRenderbuffer(UInteger v){
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glDeleteRenderbuffers)
|
|
|
|
glDeleteRenderbuffers(1, &v);
|
|
|
|
else if(glDeleteRenderbuffersEXT)
|
|
|
|
glDeleteRenderbuffersEXT(1, &v);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glDeleteRenderbuffers");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glDeleteRenderbuffers);
|
2013-08-18 16:18:06 +09:00
|
|
|
glDeleteRenderbuffers(1, &v);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
}
|
|
|
|
void SDLGLDevice::BindRenderbuffer(Enum target, UInteger v){
|
|
|
|
SPADES_MARK_FUNCTION_DEBUG();
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glBindRenderbuffer)
|
|
|
|
glBindRenderbuffer(parseRenderbufferTarget(target),
|
|
|
|
v);
|
|
|
|
else if(glBindRenderbufferEXT)
|
|
|
|
glBindRenderbufferEXT(parseRenderbufferTarget(target),
|
|
|
|
v);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glBindRenderbuffer");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glBindRenderbuffer);
|
2013-08-18 16:18:06 +09:00
|
|
|
glBindRenderbuffer(parseRenderbufferTarget(target),
|
|
|
|
v);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-18 16:18:06 +09:00
|
|
|
CheckError();
|
|
|
|
|
|
|
|
}
|
|
|
|
void SDLGLDevice::RenderbufferStorage(Enum target,
|
|
|
|
Enum intFormat,
|
|
|
|
Sizei width,
|
|
|
|
Sizei height){
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glRenderbufferStorage)
|
|
|
|
glRenderbufferStorage(parseRenderbufferTarget(target),
|
|
|
|
parseTextureInternalFormat(intFormat),
|
|
|
|
width, height);
|
|
|
|
else if(glRenderbufferStorageEXT)
|
|
|
|
glRenderbufferStorageEXT(parseRenderbufferTarget(target),
|
|
|
|
parseTextureInternalFormat(intFormat),
|
|
|
|
width, height);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glRenderbufferStorage");
|
|
|
|
|
|
|
|
#else
|
|
|
|
CheckExistence(glRenderbufferStorage);
|
2013-08-18 16:18:06 +09:00
|
|
|
glRenderbufferStorage(parseRenderbufferTarget(target),
|
|
|
|
parseTextureInternalFormat(intFormat),
|
|
|
|
width, height);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-27 03:56:56 +09:00
|
|
|
CheckErrorAlways();
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
void SDLGLDevice::RenderbufferStorage(Enum target,
|
|
|
|
Sizei samples,
|
|
|
|
Enum intFormat,
|
|
|
|
Sizei width, Sizei height){
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glRenderbufferStorageMultisample)
|
|
|
|
glRenderbufferStorageMultisample(parseRenderbufferTarget(target),
|
|
|
|
samples,
|
|
|
|
parseTextureInternalFormat(intFormat),
|
|
|
|
width, height);
|
|
|
|
else if(glRenderbufferStorageMultisampleEXT)
|
|
|
|
glRenderbufferStorageMultisampleEXT(parseRenderbufferTarget(target),
|
|
|
|
samples,
|
|
|
|
parseTextureInternalFormat(intFormat),
|
|
|
|
width, height);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glRenderbufferStorageMultisample");
|
|
|
|
#else
|
|
|
|
CheckExistence(glRenderbufferStorageMultisample);
|
2013-08-18 16:18:06 +09:00
|
|
|
glRenderbufferStorageMultisample(parseRenderbufferTarget(target),
|
|
|
|
samples,
|
|
|
|
parseTextureInternalFormat(intFormat),
|
|
|
|
width, height);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-27 03:56:56 +09:00
|
|
|
CheckErrorAlways();
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
void SDLGLDevice::FramebufferRenderbuffer(Enum target,
|
|
|
|
Enum attachment, Enum rbTarget, UInteger rb) {
|
2013-08-26 01:46:48 +09:00
|
|
|
|
2013-08-18 16:18:06 +09:00
|
|
|
GLenum a;
|
|
|
|
switch(attachment){
|
|
|
|
case draw::IGLDevice::ColorAttachment0:
|
|
|
|
a = GL_COLOR_ATTACHMENT0;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::ColorAttachment1:
|
|
|
|
a = GL_COLOR_ATTACHMENT1;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::ColorAttachment2:
|
|
|
|
a = GL_COLOR_ATTACHMENT2;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::ColorAttachment3:
|
|
|
|
a = GL_COLOR_ATTACHMENT3;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::ColorAttachment4:
|
|
|
|
a = GL_COLOR_ATTACHMENT4;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::ColorAttachment5:
|
|
|
|
a = GL_COLOR_ATTACHMENT5;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::ColorAttachment6:
|
|
|
|
a = GL_COLOR_ATTACHMENT6;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::ColorAttachment7:
|
|
|
|
a = GL_COLOR_ATTACHMENT7;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::DepthAttachment:
|
|
|
|
a = GL_DEPTH_ATTACHMENT;
|
|
|
|
break;
|
|
|
|
case draw::IGLDevice::StencilAttachment:
|
|
|
|
a = GL_STENCIL_ATTACHMENT;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
SPInvalidEnum("attachment",attachment);
|
|
|
|
}
|
2013-08-27 01:36:28 +09:00
|
|
|
#if GLEW
|
|
|
|
if(glFramebufferRenderbuffer)
|
|
|
|
glFramebufferRenderbuffer(parseFramebufferTarget(target),
|
|
|
|
a, parseRenderbufferTarget(rbTarget),
|
|
|
|
rb);
|
|
|
|
else if(glFramebufferRenderbufferEXT)
|
|
|
|
glFramebufferRenderbufferEXT(parseFramebufferTarget(target),
|
|
|
|
a, parseRenderbufferTarget(rbTarget),
|
|
|
|
rb);
|
|
|
|
else
|
|
|
|
ReportMissingFunc("glFramebufferRenderbuffer");
|
|
|
|
#else
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glFramebufferRenderbuffer);
|
2013-08-18 16:18:06 +09:00
|
|
|
glFramebufferRenderbuffer(parseFramebufferTarget(target),
|
|
|
|
a, parseRenderbufferTarget(rbTarget),
|
|
|
|
rb);
|
2013-08-27 01:36:28 +09:00
|
|
|
#endif
|
2013-08-27 03:56:56 +09:00
|
|
|
CheckErrorAlways();
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
void SDLGLDevice::ReadPixels(Integer x,
|
|
|
|
Integer y,
|
|
|
|
Sizei width,
|
|
|
|
Sizei height,
|
|
|
|
Enum format,
|
|
|
|
Enum type, void *data) {
|
2013-08-26 01:46:48 +09:00
|
|
|
CheckExistence(glReadPixels);
|
2013-08-18 16:18:06 +09:00
|
|
|
glReadPixels(x, y, width, height,
|
|
|
|
parseTextureFormat(format),
|
|
|
|
parseType(type), data);
|
2013-08-27 03:56:56 +09:00
|
|
|
CheckErrorAlways();
|
2013-08-18 16:18:06 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
IGLDevice::Integer SDLGLDevice::ScreenWidth() {
|
|
|
|
return surface->w;
|
|
|
|
}
|
|
|
|
|
|
|
|
IGLDevice::Integer SDLGLDevice::ScreenHeight() {
|
|
|
|
return surface->h;
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|