Note: Not yet caught up with trunk. git-svn-id: svn://svn.code.sf.net/p/irrlicht/code/branches/ogl-es@6114 dfc29bdd-3216-0410-991c-e03cc46cb475
		
			
				
	
	
		
			397 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			397 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright (C) 2015 Patryk Nadrowski
 | |
| // This file is part of the "Irrlicht Engine".
 | |
| // For conditions of distribution and use, see copyright notice in irrlicht.h
 | |
| 
 | |
| #ifndef __C_OGLCORE_RENDER_TARGET_H_INCLUDED__
 | |
| #define __C_OGLCORE_RENDER_TARGET_H_INCLUDED__
 | |
| 
 | |
| #include "IrrCompileConfig.h"
 | |
| 
 | |
| #if defined(_IRR_COMPILE_WITH_OPENGL_) || defined(_IRR_COMPILE_WITH_OGLES1_) || defined(_IRR_COMPILE_WITH_OGLES2_)
 | |
| 
 | |
| #include "IRenderTarget.h"
 | |
| 
 | |
| namespace irr
 | |
| {
 | |
| namespace video
 | |
| {
 | |
| 
 | |
| template <class TOpenGLDriver, class TOpenGLTexture>
 | |
| class COpenGLCoreRenderTarget : public IRenderTarget
 | |
| {
 | |
| public:
 | |
| 	COpenGLCoreRenderTarget(TOpenGLDriver* driver) : AssignedDepth(false), AssignedStencil(false), RequestTextureUpdate(false), RequestDepthStencilUpdate(false),
 | |
| 		BufferID(0), ColorAttachment(0), MultipleRenderTarget(0), Driver(driver)
 | |
| 	{
 | |
| #ifdef _DEBUG
 | |
| 		setDebugName("COpenGLCoreRenderTarget");
 | |
| #endif
 | |
| 
 | |
| 		DriverType = Driver->getDriverType();
 | |
| 
 | |
| 		Size = Driver->getScreenSize();
 | |
| 
 | |
| 		ColorAttachment = Driver->getFeature().ColorAttachment;
 | |
| 		MultipleRenderTarget = Driver->getFeature().MultipleRenderTarget;
 | |
| 
 | |
| 		if (ColorAttachment > 0)
 | |
| 			Driver->irrGlGenFramebuffers(1, &BufferID);
 | |
| 
 | |
| 		AssignedTexture.set_used(static_cast<u32>(ColorAttachment));
 | |
| 
 | |
| 		for (u32 i = 0; i < AssignedTexture.size(); ++i)
 | |
| 			AssignedTexture[i] = GL_NONE;
 | |
| 	}
 | |
| 
 | |
| 	virtual ~COpenGLCoreRenderTarget()
 | |
| 	{
 | |
| 		if (ColorAttachment > 0 && BufferID != 0)
 | |
| 			Driver->irrGlDeleteFramebuffers(1, &BufferID);
 | |
| 
 | |
| 		for (u32 i = 0; i < Texture.size(); ++i)
 | |
| 		{
 | |
| 			if (Texture[i])
 | |
| 				Texture[i]->drop();
 | |
| 		}
 | |
| 
 | |
| 		if (DepthStencil)
 | |
| 			DepthStencil->drop();
 | |
| 	}
 | |
| 
 | |
| 	virtual void setTexture(const core::array<ITexture*>& textures, ITexture* depthStencil, const core::array<E_CUBE_SURFACE>& cubeSurfaces) _IRR_OVERRIDE_
 | |
| 	{
 | |
| 		bool needSizeUpdate = false;
 | |
| 
 | |
| 		// Set color attachments.
 | |
| 		if ((Texture != textures) || (CubeSurfaces != cubeSurfaces))
 | |
| 		{
 | |
| 			needSizeUpdate = true;
 | |
| 
 | |
| 			core::array<ITexture*> prevTextures(Texture);
 | |
| 
 | |
| 			if (textures.size() > static_cast<u32>(ColorAttachment))
 | |
| 			{
 | |
| 				core::stringc message = "This GPU supports up to ";
 | |
| 				message += static_cast<u32>(ColorAttachment);
 | |
| 				message += " textures per render target.";
 | |
| 
 | |
| 				os::Printer::log(message.c_str(), ELL_WARNING);
 | |
| 			}
 | |
| 
 | |
| 			Texture.set_used(core::min_(textures.size(), static_cast<u32>(ColorAttachment)));
 | |
| 
 | |
| 			for (u32 i = 0; i < Texture.size(); ++i)
 | |
| 			{
 | |
| 				TOpenGLTexture* currentTexture = (textures[i] && textures[i]->getDriverType() == DriverType) ? static_cast<TOpenGLTexture*>(textures[i]) : 0;
 | |
| 
 | |
| 				GLuint textureID = 0;
 | |
| 
 | |
| 				if (currentTexture)
 | |
| 				{
 | |
| 					textureID = currentTexture->getOpenGLTextureName();
 | |
| 				}
 | |
| 
 | |
| 				if (textureID != 0)
 | |
| 				{
 | |
| 					Texture[i] = textures[i];
 | |
| 					Texture[i]->grab();
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					Texture[i] = 0;
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			for (u32 i = 0; i < prevTextures.size(); ++i)
 | |
| 			{
 | |
| 				if (prevTextures[i])
 | |
| 					prevTextures[i]->drop();
 | |
| 			}
 | |
| 
 | |
| 			RequestTextureUpdate = true;
 | |
| 		}
 | |
| 
 | |
| 		if (CubeSurfaces != cubeSurfaces)
 | |
| 		{
 | |
| 			CubeSurfaces = cubeSurfaces;
 | |
| 			RequestTextureUpdate = true;
 | |
| 		}
 | |
| 
 | |
| 		// Set depth and stencil attachments.
 | |
| 		if (DepthStencil != depthStencil)
 | |
| 		{
 | |
| 			if (DepthStencil)
 | |
| 			{
 | |
| 				DepthStencil->drop();
 | |
| 				DepthStencil = 0;
 | |
| 			}
 | |
| 
 | |
| 			needSizeUpdate = true;
 | |
| 			TOpenGLTexture* currentTexture = (depthStencil && depthStencil->getDriverType() == DriverType) ? static_cast<TOpenGLTexture*>(depthStencil) : 0;
 | |
| 
 | |
| 			if (currentTexture)
 | |
| 			{	
 | |
| 				if (currentTexture->getType() == ETT_2D)
 | |
| 				{
 | |
| 					GLuint textureID = currentTexture->getOpenGLTextureName();
 | |
| 
 | |
| 					const ECOLOR_FORMAT textureFormat = (textureID != 0) ? depthStencil->getColorFormat() : ECF_UNKNOWN;
 | |
| 					if (IImage::isDepthFormat(textureFormat))
 | |
| 					{
 | |
| 						DepthStencil = depthStencil;
 | |
| 						DepthStencil->grab();
 | |
| 					}
 | |
| 					else
 | |
| 					{
 | |
| 						os::Printer::log("Ignoring depth/stencil texture without depth color format.", ELL_WARNING);
 | |
| 					}
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					os::Printer::log("This driver doesn't support depth/stencil to cubemaps.", ELL_WARNING);
 | |
| 				}
 | |
| 			}
 | |
| 
 | |
| 			RequestDepthStencilUpdate = true;
 | |
| 		}
 | |
| 
 | |
| 		if (needSizeUpdate)
 | |
| 		{
 | |
| 			// Set size required for a viewport.
 | |
| 
 | |
| 			ITexture* firstTexture = getTexture();
 | |
| 
 | |
| 			if (firstTexture)
 | |
| 				Size = firstTexture->getSize();
 | |
| 			else
 | |
| 			{
 | |
| 				if (DepthStencil)
 | |
| 					Size = DepthStencil->getSize();
 | |
| 				else
 | |
| 					Size = Driver->getScreenSize();
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void update()
 | |
| 	{
 | |
| 		if (RequestTextureUpdate || RequestDepthStencilUpdate)
 | |
| 		{
 | |
| 			// Set color attachments.
 | |
| 
 | |
| 			if (RequestTextureUpdate)
 | |
| 			{
 | |
| 				// Set new color textures.
 | |
| 
 | |
| 				const u32 textureSize = core::min_(Texture.size(), AssignedTexture.size());
 | |
| 
 | |
| 				for (u32 i = 0; i < textureSize; ++i)
 | |
| 				{
 | |
| 					TOpenGLTexture* currentTexture = static_cast<TOpenGLTexture*>(Texture[i]);
 | |
| 					GLuint textureID = currentTexture ? currentTexture->getOpenGLTextureName() : 0;
 | |
| 
 | |
| 					if (textureID != 0)
 | |
| 					{
 | |
| 						AssignedTexture[i] = GL_COLOR_ATTACHMENT0 + i;
 | |
| 						GLenum textarget = currentTexture->getType() == ETT_2D ? GL_TEXTURE_2D : GL_TEXTURE_CUBE_MAP_POSITIVE_X + (int)CubeSurfaces[i];
 | |
| 						Driver->irrGlFramebufferTexture2D(GL_FRAMEBUFFER, AssignedTexture[i], textarget, textureID, 0);
 | |
| #ifdef _DEBUG
 | |
| 						Driver->testGLError(__LINE__);
 | |
| #endif
 | |
| 					}
 | |
| 					else if (AssignedTexture[i] != GL_NONE)
 | |
| 					{
 | |
| 						AssignedTexture[i] = GL_NONE;
 | |
| 						Driver->irrGlFramebufferTexture2D(GL_FRAMEBUFFER, AssignedTexture[i], GL_TEXTURE_2D, 0, 0);
 | |
| 
 | |
| 						os::Printer::log("Error: Could not set render target.", ELL_ERROR);
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				// Reset other render target channels.
 | |
| 
 | |
| 				for (u32 i = textureSize; i < AssignedTexture.size(); ++i)
 | |
| 				{
 | |
| 					if (AssignedTexture[i] != GL_NONE)
 | |
| 					{
 | |
| 						Driver->irrGlFramebufferTexture2D(GL_FRAMEBUFFER, AssignedTexture[i], GL_TEXTURE_2D, 0, 0);
 | |
| 						AssignedTexture[i] = GL_NONE;
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				RequestTextureUpdate = false;
 | |
| 			}
 | |
| 
 | |
| 			// Set depth and stencil attachments.
 | |
| 
 | |
| 			if (RequestDepthStencilUpdate)
 | |
| 			{
 | |
| 				const ECOLOR_FORMAT textureFormat = (DepthStencil) ? DepthStencil->getColorFormat() : ECF_UNKNOWN;
 | |
| 
 | |
| 				if (IImage::isDepthFormat(textureFormat))
 | |
| 				{
 | |
| 					GLuint textureID = static_cast<TOpenGLTexture*>(DepthStencil)->getOpenGLTextureName();
 | |
| 
 | |
| #ifdef _IRR_EMSCRIPTEN_PLATFORM_	// The WEBGL_depth_texture extension does not allow attaching stencil+depth separate.
 | |
| 					if (textureFormat == ECF_D24S8)
 | |
| 					{
 | |
| 						GLenum attachment = 0x821A; // GL_DEPTH_STENCIL_ATTACHMENT
 | |
| 						Driver->irrGlFramebufferTexture2D(GL_FRAMEBUFFER, attachment, GL_TEXTURE_2D, textureID, 0);
 | |
| 						AssignedStencil = true;
 | |
| 					}
 | |
| 					else
 | |
| 					{
 | |
| 						Driver->irrGlFramebufferTexture2D(GL_FRAMEBUFFER, GL_DEPTH_ATTACHMENT, GL_TEXTURE_2D, textureID, 0);
 | |
| 						AssignedStencil = false;
 | |
| 					}
 | |
| #else
 | |
| 					Driver->irrGlFramebufferTexture2D(GL_FRAMEBUFFER, GL_DEPTH_ATTACHMENT, GL_TEXTURE_2D, textureID, 0);
 | |
| 
 | |
| 					if (textureFormat == ECF_D24S8)
 | |
| 					{
 | |
| 						Driver->irrGlFramebufferTexture2D(GL_FRAMEBUFFER, GL_STENCIL_ATTACHMENT, GL_TEXTURE_2D, textureID, 0);
 | |
| 
 | |
| 						AssignedStencil = true;
 | |
| 					}
 | |
| 					else
 | |
| 					{
 | |
| 						if (AssignedStencil)
 | |
| 							Driver->irrGlFramebufferTexture2D(GL_FRAMEBUFFER, GL_STENCIL_ATTACHMENT, GL_TEXTURE_2D, 0, 0);
 | |
| 
 | |
| 						AssignedStencil = false;
 | |
| 					}
 | |
| #endif
 | |
| 					AssignedDepth = true;
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					if (AssignedDepth)
 | |
| 						Driver->irrGlFramebufferTexture2D(GL_FRAMEBUFFER, GL_DEPTH_ATTACHMENT, GL_TEXTURE_2D, 0, 0);
 | |
| 
 | |
| 					if (AssignedStencil)
 | |
| 						Driver->irrGlFramebufferTexture2D(GL_FRAMEBUFFER, GL_STENCIL_ATTACHMENT, GL_TEXTURE_2D, 0, 0);
 | |
| 
 | |
| 					AssignedDepth = false;
 | |
| 					AssignedStencil = false;
 | |
| 				}
 | |
| #ifdef _DEBUG
 | |
| 				Driver->testGLError(__LINE__);
 | |
| #endif
 | |
| 
 | |
| 				RequestDepthStencilUpdate = false;
 | |
| 			}
 | |
| 
 | |
| 			// Configure drawing operation.
 | |
| 
 | |
| 			if (ColorAttachment > 0 && BufferID != 0)
 | |
| 			{
 | |
| 				const u32 textureSize = Texture.size();
 | |
| 
 | |
| 				if (textureSize == 0)
 | |
| 					Driver->irrGlDrawBuffer(GL_NONE);
 | |
| 				else if (textureSize == 1 || MultipleRenderTarget == 0)
 | |
| 					Driver->irrGlDrawBuffer(GL_COLOR_ATTACHMENT0);
 | |
| 				else
 | |
| 				{
 | |
| 					const u32 bufferCount = core::min_(MultipleRenderTarget, core::min_(textureSize, AssignedTexture.size()));
 | |
| 
 | |
| 					Driver->irrGlDrawBuffers(bufferCount, AssignedTexture.pointer());
 | |
| 				}
 | |
| 
 | |
| #ifdef _DEBUG
 | |
| 				Driver->testGLError(__LINE__);
 | |
| #endif
 | |
| 
 | |
| 			}
 | |
| 
 | |
| #ifdef _DEBUG
 | |
| 			checkFBO(Driver);
 | |
| #endif
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	GLuint getBufferID() const
 | |
| 	{
 | |
| 		return BufferID;
 | |
| 	}
 | |
| 
 | |
| 	const core::dimension2d<u32>& getSize() const
 | |
| 	{
 | |
| 		return Size;
 | |
| 	}
 | |
| 
 | |
| 	ITexture* getTexture() const
 | |
| 	{
 | |
| 		for (u32 i = 0; i < Texture.size(); ++i)
 | |
| 		{
 | |
| 			if (Texture[i])
 | |
| 				return Texture[i];
 | |
| 		}
 | |
| 
 | |
| 		return 0;
 | |
| 	}
 | |
| 
 | |
| protected:
 | |
| 	bool checkFBO(TOpenGLDriver* driver)
 | |
| 	{
 | |
| 		if (ColorAttachment == 0)
 | |
| 			return true;
 | |
| 
 | |
| 		GLenum status = driver->irrGlCheckFramebufferStatus(GL_FRAMEBUFFER);
 | |
| 
 | |
| 		switch (status)
 | |
| 		{
 | |
| 			case GL_FRAMEBUFFER_COMPLETE:
 | |
| 				return true;
 | |
| 			case GL_FRAMEBUFFER_INCOMPLETE_READ_BUFFER:
 | |
| 				os::Printer::log("FBO has invalid read buffer", ELL_ERROR);
 | |
| 				break;
 | |
| 			case GL_FRAMEBUFFER_INCOMPLETE_DRAW_BUFFER:
 | |
| 				os::Printer::log("FBO has invalid draw buffer", ELL_ERROR);
 | |
| 				break;
 | |
| 			case GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT:
 | |
| 				os::Printer::log("FBO has one or several incomplete image attachments", ELL_ERROR);
 | |
| 				break;
 | |
| 			case GL_FRAMEBUFFER_INCOMPLETE_FORMATS:
 | |
| 				os::Printer::log("FBO has one or several image attachments with different internal formats", ELL_ERROR);
 | |
| 				break;
 | |
| 			case GL_FRAMEBUFFER_INCOMPLETE_DIMENSIONS:
 | |
| 				os::Printer::log("FBO has one or several image attachments with different dimensions", ELL_ERROR);
 | |
| 				break;
 | |
| 			case GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT:
 | |
| 				os::Printer::log("FBO missing an image attachment", ELL_ERROR);
 | |
| 				break;
 | |
| 			case GL_FRAMEBUFFER_UNSUPPORTED:
 | |
| 				os::Printer::log("FBO format unsupported", ELL_ERROR);
 | |
| 				break;
 | |
| 			default:
 | |
| 				os::Printer::log("FBO error", ELL_ERROR);
 | |
| 				break;
 | |
| 		}
 | |
| 
 | |
| 		return false;
 | |
| 	}
 | |
| 
 | |
| 	core::array<GLenum> AssignedTexture;
 | |
| 	bool AssignedDepth;
 | |
| 	bool AssignedStencil;
 | |
| 
 | |
| 	bool RequestTextureUpdate;
 | |
| 	bool RequestDepthStencilUpdate;
 | |
| 
 | |
| 	GLuint BufferID;
 | |
| 
 | |
| 	core::dimension2d<u32> Size;
 | |
| 
 | |
| 	u32 ColorAttachment;
 | |
| 	u32 MultipleRenderTarget;
 | |
| 
 | |
| 	TOpenGLDriver* Driver;
 | |
| };
 | |
| 
 | |
| }
 | |
| }
 | |
| 
 | |
| #endif
 | |
| #endif
 |