// // Lol Engine // // Copyright: (c) 2010-2013 Sam Hocevar <sam@hocevar.net> // This program is free software; you can redistribute it and/or // modify it under the terms of the Do What The Fuck You Want To // Public License, Version 2, as published by Sam Hocevar. See // http://www.wtfpl.net/ for more details. // #if defined HAVE_CONFIG_H # include "config.h" #endif #include "core.h" #include "lolgl.h" #if defined _WIN32 && defined USE_D3D9 # define FAR # define NEAR # include <d3d9.h> #endif using namespace std; #if defined USE_D3D9 extern IDirect3DDevice9 *g_d3ddevice; #elif defined _XBOX extern D3DDevice *g_d3ddevice; #endif namespace lol { // // The FramebufferData class // ------------------------- // class FramebufferData { friend class Framebuffer; ivec2 m_size; #if defined USE_D3D9 IDirect3DTexture9 *m_texture; IDirect3DSurface9 *m_surface, *m_back_surface; #elif defined _XBOX D3DTexture *m_texture; D3DSurface *m_surface, *m_back_surface; #else GLuint m_fbo, m_texture, m_depth; #endif }; // // The FramebufferFormat struct // ---------------------- // uint32_t FramebufferFormat::GetFormat() { switch (m_format) { #if defined USE_D3D9 || defined _XBOX case R_16_F: return D3DFMT_R16F; case R_32_F: return D3DFMT_R32F; case RG_16: case RG_16_I: case RG_16_UI: return D3DFMT_G16R16; case RG_16_F: return D3DFMT_G16R16F; case RG_32_F: return D3DFMT_G32R32F; case RGB_8: case RGB_8_I: case RGB_8_UI: return D3DFMT_R8G8B8; case RGBA_8: case RGBA_8_I: case RGBA_8_UI: return D3DFMT_A8R8G8B8; case RGBA_16: case RGBA_16_I: case RGBA_16_UI: return D3DFMT_A16B16G16R16; case RGBA_16_F: return D3DFMT_A16B16G16R16F; case RGBA_32_F: return D3DFMT_A32B32G32R32F; default: ASSERT(false, "not supported by DirectX"); return 0; #elif defined __CELLOS_LV2__ /* Supported drawable formats on the PS3: GL_ARGB_SCE, GL_RGB16F_ARB, * GL_RGBA16F_ARB, GL_RGB32F_ARB, GL_RGBA32F_ARB, GL_LUMINANCE32F_ARB. */ case RGB_16_F: return GL_RGB16F_ARB; case RGB_32_F: return GL_RGB32F_ARB; case RGBA_8: return GL_ARGB_SCE; case RGBA_16_F: return GL_RGBA16F_ARB; case RGBA_32_F: return GL_RGBA32F_ARB; default: ASSERT(false, "not supported by the PS3"); return 0; #elif defined HAVE_GLES_2X /* FIXME: not implemented at all */ default: ASSERT(false, "not implemented"); return 0; #elif defined __APPLE__ && defined __MACH__ case R_8: case RG_8: case RGB_8: case RGBA_8: case R_8_I: case RG_8_I: case RGB_8_I: case RGBA_8_I: return GL_BYTE; case R_8_UI: case RG_8_UI: case RGB_8_UI: case RGBA_8_UI: return GL_UNSIGNED_BYTE; case R_16: case RG_16: case RGB_16: case RGBA_16: case R_16_I: case RG_16_I: case RGB_16_I: case RGBA_16_I: return GL_SHORT; case R_16_UI: case RG_16_UI: case RGB_16_UI: case RGBA_16_UI: return GL_UNSIGNED_SHORT; case R_16_F: case RG_16_F: case RGB_16_F: case RGBA_16_F: ASSERT(false, "not supported by IOS:OGL:ES"); return 0; case R_32_I: case RG_32_I: case RGB_32_I: case RGBA_32_I: return GL_INT; case R_32_UI: case RG_32_UI: case RGB_32_UI: case RGBA_32_UI: return GL_UNSIGNED_INT; case R_32_F: case RG_32_F: case RGB_32_F: case RGBA_32_F: return GL_FLOAT; #else case R_8: return GL_R8; case R_8_I: return GL_R8I; case R_8_UI: return GL_R8UI; case R_16: return GL_R16; case R_16_I: return GL_R16I; case R_16_UI: return GL_R16UI; case R_16_F: return GL_R16F; case R_32_I: return GL_R32I; case R_32_UI: return GL_R32UI; case R_32_F: return GL_R32F; case RG_8: return GL_RG8; case RG_8_I: return GL_RG8I; case RG_8_UI: return GL_RG8UI; case RG_16: return GL_RG16; case RG_16_I: return GL_RG16I; case RG_16_UI: return GL_RG16UI; case RG_16_F: return GL_RG16F; case RG_32_I: return GL_RG32I; case RG_32_UI: return GL_RG32UI; case RG_32_F: return GL_RG32F; case RGB_8: return GL_RGB8; case RGB_8_I: return GL_RGB8I; case RGB_8_UI: return GL_RGB8UI; case RGB_16: return GL_RGB16; case RGB_16_I: return GL_RGB16I; case RGB_16_UI: return GL_RGB16UI; case RGB_16_F: return GL_RGB16F; case RGB_32_I: return GL_RGB32I; case RGB_32_UI: return GL_RGB32UI; case RGB_32_F: return GL_RGB32F; case RGBA_8: return GL_RGBA8; case RGBA_8_I: return GL_RGBA8I; case RGBA_8_UI: return GL_RGBA8UI; case RGBA_16: return GL_RGBA16; case RGBA_16_I: return GL_RGBA16I; case RGBA_16_UI: return GL_RGBA16UI; case RGBA_16_F: return GL_RGBA16F; case RGBA_32_I: return GL_RGBA32I; case RGBA_32_UI: return GL_RGBA32UI; case RGBA_32_F: return GL_RGBA32F; default: ASSERT(false, "not supported by OpenGL"); return 0; #endif }; ASSERT(false, "not implemented"); return 0; } uint32_t FramebufferFormat::GetFormatOrder() { switch (m_format) { #if defined USE_D3D9 || defined _XBOX /* FIXME: not implemented at all */ default: ASSERT(false, "not implemented"); return 0; #elif defined __CELLOS_LV2__ /* FIXME: not implemented at all */ default: ASSERT(false, "not implemented"); return 0; #elif defined HAVE_GLES_2X /* FIXME: not implemented at all */ default: ASSERT(false, "not implemented"); return 0; #elif defined __APPLE__ && defined __MACH__ case R_8: case R_8_I: case R_8_UI: case R_8_F: case R_16: case R_16_I: case R_16_UI: case R_16_F: case R_32_I: case R_32: case R_32_UI: case R_32_F: case RG_8: case RG_8_I: case RG_8_UI: case RG_8_F: case RG_16: case RG_16_I: case RG_16_UI: case RG_16_F: case RG_32: case RG_32_I: case RG_32_UI: case RG_32_F: case RGB_8: case RGB_8_I: case RGB_8_UI: case RGB_8_F: case RGB_16: case RGB_16_I: case RGB_16_UI: case RGB_16_F: case RGB_32: case RGB_32_I: case RGB_32_UI: case RGB_32_F: case RGBA_8: case RGBA_8_I: case RGBA_8_UI: case RGBA_8_F: case RGBA_16: case RGBA_16_I: case RGBA_16_UI: case RGBA_16_F: case RGBA_32: case RGBA_32_I: case RGBA_32_UI: case RGBA_32_F: #else case R_8: case R_8_I: case R_8_UI: case R_8_F: case R_16: case R_16_I: case R_16_UI: case R_16_F: case R_32_I: case R_32: case R_32_UI: case R_32_F: return GL_RED; case RG_8: case RG_8_I: case RG_8_UI: case RG_8_F: case RG_16: case RG_16_I: case RG_16_UI: case RG_16_F: case RG_32: case RG_32_I: case RG_32_UI: case RG_32_F: return GL_RG; case RGB_8: case RGB_8_I: case RGB_8_UI: case RGB_8_F: case RGB_16: case RGB_16_I: case RGB_16_UI: case RGB_16_F: case RGB_32: case RGB_32_I: case RGB_32_UI: case RGB_32_F: return (m_invert_rgb)?(GL_BGR):(GL_RGB); case RGBA_8: case RGBA_8_I: case RGBA_8_UI: case RGBA_8_F: case RGBA_16: case RGBA_16_I: case RGBA_16_UI: case RGBA_16_F: case RGBA_32: case RGBA_32_I: case RGBA_32_UI: case RGBA_32_F: return (m_invert_rgb)?(GL_BGRA):(GL_RGBA); #endif } ASSERT(false, "unknown pixel format"); return 0; } // // The Framebuffer class // ---------------------- // Framebuffer::Framebuffer(ivec2 size, FramebufferFormat fbo_format) : m_data(new FramebufferData) { m_data->m_size = size; #if defined USE_D3D9 if (FAILED(g_d3ddevice->CreateTexture(size.x, size.y, 1, D3DUSAGE_RENDERTARGET, fbo_format.GetFormat(), D3DPOOL_DEFAULT, &m_data->m_texture, nullptr))) Abort(); if (FAILED(m_data->m_texture->GetSurfaceLevel(0, &m_data->m_surface))) Abort(); #elif defined _XBOX if (FAILED(g_d3ddevice->CreateTexture(size.x, size.y, 1, 0, fbo_format.GetFormat(), D3DPOOL_DEFAULT, &m_data->m_texture, nullptr))) Abort(); if (FAILED(g_d3ddevice->CreateRenderTarget(size.x, size.y, fbo_format.GetFormat(), D3DMULTISAMPLE_NONE, 0, 0, &m_data->m_surface, nullptr))) Abort(); #else # if GL_VERSION_1_1 GLenum internal_format = fbo_format.GetFormat(); GLenum format = fbo_format.GetFormatOrder(); GLenum depth = GL_DEPTH_COMPONENT; # elif defined __CELLOS_LV2__ /* Supported drawable formats on the PS3: GL_ARGB_SCE, GL_RGB16F_ARB, * GL_RGBA16F_ARB, GL_RGB32F_ARB, GL_RGBA32F_ARB, GL_LUMINANCE32F_ARB. */ GLenum internal_format = fbo_format.GetFormat(); GLenum format = fbo_format.GetFormatOrder(); # else GLenum internal_format = fbo_format.GetFormat(); GLenum format = fbo_format.GetFormatOrder(); # endif GLenum wrapmode = GL_REPEAT; GLenum filtering = GL_NEAREST; # if GL_VERSION_1_1 || GL_ES_VERSION_2_0 glGenFramebuffers(1, &m_data->m_fbo); glBindFramebuffer(GL_FRAMEBUFFER, m_data->m_fbo); # else glGenFramebuffersOES(1, &m_data->m_fbo); glBindFramebufferOES(GL_FRAMEBUFFER_OES, m_data->m_fbo); # endif glGenTextures(1, &m_data->m_texture); glActiveTexture(GL_TEXTURE0); glBindTexture(GL_TEXTURE_2D, m_data->m_texture); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_S, (GLenum)wrapmode); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_WRAP_T, (GLenum)wrapmode); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, (GLenum)filtering); glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MAG_FILTER, (GLenum)filtering); glTexImage2D(GL_TEXTURE_2D, 0, internal_format, size.x, size.y, 0, format, GL_UNSIGNED_BYTE, nullptr); # if GL_VERSION_1_1 || GL_ES_VERSION_2_0 glFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, m_data->m_texture, 0); # else glFramebufferTexture2DOES(GL_FRAMEBUFFER_OES, GL_COLOR_ATTACHMENT0_EXT, GL_TEXTURE_2D, m_data->m_texture, 0); # endif m_data->m_depth = GL_INVALID_ENUM; # if GL_VERSION_1_1 /* FIXME: not implemented on GL ES, see * http://stackoverflow.com/q/4041682/111461 */ if (depth != GL_INVALID_ENUM) { glGenRenderbuffers(1, &m_data->m_depth); glBindRenderbuffer(GL_RENDERBUFFER, m_data->m_depth); glRenderbufferStorage(GL_RENDERBUFFER, depth, size.x, size.y); glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_DEPTH_ATTACHMENT, GL_RENDERBUFFER, m_data->m_depth); } # endif # if GL_VERSION_1_1 || GL_ES_VERSION_2_0 glCheckFramebufferStatus(GL_FRAMEBUFFER); # endif Unbind(); #endif } Framebuffer::~Framebuffer() { #if defined USE_D3D9 || defined _XBOX m_data->m_surface->Release(); m_data->m_texture->Release(); #else # if GL_VERSION_1_1 || GL_ES_VERSION_2_0 glDeleteFramebuffers(1, &m_data->m_fbo); # else glDeleteFramebuffersOES(1, &m_data->m_fbo); # endif glDeleteTextures(1, &m_data->m_texture); # if GL_VERSION_1_1 if (m_data->m_depth != GL_INVALID_ENUM) glDeleteRenderbuffers(1, &m_data->m_depth); # endif #endif delete m_data; } ShaderTexture Framebuffer::GetTexture() const { ShaderTexture ret; #if defined USE_D3D9 || defined _XBOX ret.m_flags = (uint64_t)(uintptr_t)m_data->m_texture; #else ret.m_flags = m_data->m_texture; #endif return ret; } ivec2 Framebuffer::GetSize() const { return m_data->m_size; } void Framebuffer::Bind() { #if defined USE_D3D9 || defined _XBOX if (FAILED(g_d3ddevice->GetRenderTarget(0, &m_data->m_back_surface))) Abort(); if (FAILED(g_d3ddevice->SetRenderTarget(0, m_data->m_surface))) Abort(); #else # if GL_VERSION_1_1 || GL_ES_VERSION_2_0 glBindFramebuffer(GL_FRAMEBUFFER, m_data->m_fbo); # else glBindFramebufferOES(GL_FRAMEBUFFER_OES, m_data->m_fbo); # endif //change viewport draw size Video::SetCustomSize(m_data->m_size); #endif } void Framebuffer::Unbind() { #if defined USE_D3D9 if (FAILED(g_d3ddevice->SetRenderTarget(0, m_data->m_back_surface))) Abort(); m_data->m_back_surface->Release(); #elif defined _XBOX if (FAILED(g_d3ddevice->Resolve(D3DRESOLVE_RENDERTARGET0, nullptr, m_data->m_texture, nullptr, 0, 0, nullptr, 0, 0, nullptr))) Abort(); if (FAILED(g_d3ddevice->SetRenderTarget(0, m_data->m_back_surface))) Abort(); m_data->m_back_surface->Release(); #else //Restore viewport draw size Video::RestoreSize(); # if GL_VERSION_1_1 || GL_ES_VERSION_2_0 glBindFramebuffer(GL_FRAMEBUFFER, 0); # else glBindFramebufferOES(GL_FRAMEBUFFER_OES, 0); # endif #endif } } /* namespace lol */