// 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 IRR_C_OGLCORE_TEXTURE_H_INCLUDED #define IRR_C_OGLCORE_TEXTURE_H_INCLUDED #include "IrrCompileConfig.h" #if defined(_IRR_COMPILE_WITH_OPENGL_) || defined(_IRR_COMPILE_WITH_OGLES1_) || defined(_IRR_COMPILE_WITH_OGLES2_) #include "irrArray.h" #include "SMaterialLayer.h" #include "ITexture.h" #include "EDriverFeatures.h" #include "os.h" #include "CImage.h" #include "CColorConverter.h" // Check if GL version we compile with should have the glGenerateMipmap function. #if defined(GL_VERSION_3_0) || defined(GL_ES_VERSION_2_0) #define IRR_OPENGL_HAS_glGenerateMipmap #endif namespace irr { namespace video { template class COpenGLCoreTexture : public ITexture { public: struct SStatesCache { SStatesCache() : WrapU(ETC_REPEAT), WrapV(ETC_REPEAT), WrapW(ETC_REPEAT), LODBias(0), AnisotropicFilter(0), BilinearFilter(false), TrilinearFilter(false), MipMapStatus(false), IsCached(false) { } u8 WrapU; u8 WrapV; u8 WrapW; s8 LODBias; u8 AnisotropicFilter; bool BilinearFilter; bool TrilinearFilter; bool MipMapStatus; bool IsCached; }; COpenGLCoreTexture(const io::path& name, const core::array& images, E_TEXTURE_TYPE type, TOpenGLDriver* driver) : ITexture(name, type), Driver(driver), TextureType(GL_TEXTURE_2D), TextureName(0), InternalFormat(GL_RGBA), PixelFormat(GL_RGBA), PixelType(GL_UNSIGNED_BYTE), Converter(0), LockReadOnly(false), LockImage(0), LockLayer(0), KeepImage(false), MipLevelStored(0), LegacyAutoGenerateMipMaps(false) { IRR_DEBUG_BREAK_IF(images.size() == 0) DriverType = Driver->getDriverType(); TextureType = TextureTypeIrrToGL(Type); HasMipMaps = Driver->getTextureCreationFlag(ETCF_CREATE_MIP_MAPS); KeepImage = Driver->getTextureCreationFlag(ETCF_ALLOW_MEMORY_COPY); getImageValues(images[0]); const core::array* tmpImages = &images; if (KeepImage || OriginalSize != Size || OriginalColorFormat != ColorFormat) { Images.set_used(images.size()); for (u32 i = 0; i < images.size(); ++i) { Images[i] = Driver->createImage(ColorFormat, Size); if (images[i]->getDimension() == Size) images[i]->copyTo(Images[i]); else images[i]->copyToScaling(Images[i]); if ( images[i]->getMipMapsData() ) { if ( OriginalSize == Size && OriginalColorFormat == ColorFormat ) { Images[i]->setMipMapsData( images[i]->getMipMapsData(), false, true); } else { // TODO: handle at least mipmap with changing color format os::Printer::log("COpenGLCoreTexture: Can't handle format changes for mipmap data. Mipmap data dropped", ELL_WARNING); } } } tmpImages = &Images; } glGenTextures(1, &TextureName); const COpenGLCoreTexture* prevTexture = Driver->getCacheHandler()->getTextureCache().get(0); Driver->getCacheHandler()->getTextureCache().set(0, this); glTexParameteri(TextureType, GL_TEXTURE_MIN_FILTER, GL_NEAREST); glTexParameteri(TextureType, GL_TEXTURE_MAG_FILTER, GL_NEAREST); #ifdef GL_GENERATE_MIPMAP_HINT if (HasMipMaps) { if (Driver->getTextureCreationFlag(ETCF_OPTIMIZED_FOR_SPEED)) glHint(GL_GENERATE_MIPMAP_HINT, GL_FASTEST); else if (Driver->getTextureCreationFlag(ETCF_OPTIMIZED_FOR_QUALITY)) glHint(GL_GENERATE_MIPMAP_HINT, GL_NICEST); else glHint(GL_GENERATE_MIPMAP_HINT, GL_DONT_CARE); } #endif #if !defined(IRR_OPENGL_HAS_glGenerateMipmap) && defined(GL_GENERATE_MIPMAP) if (HasMipMaps) { LegacyAutoGenerateMipMaps = Driver->getTextureCreationFlag(ETCF_AUTO_GENERATE_MIP_MAPS) && Driver->queryFeature(EVDF_MIP_MAP_AUTO_UPDATE); glTexParameteri(TextureType, GL_GENERATE_MIPMAP, LegacyAutoGenerateMipMaps ? GL_TRUE : GL_FALSE); } #endif for (u32 i = 0; i < (*tmpImages).size(); ++i) uploadTexture(true, i, 0, (*tmpImages)[i]->getData()); if (HasMipMaps && !LegacyAutoGenerateMipMaps) { // Create mipmaps (either from image mipmaps or generate them) for (u32 i = 0; i < (*tmpImages).size(); ++i) { void* mipmapsData = (*tmpImages)[i]->getMipMapsData(); regenerateMipMapLevels(mipmapsData, i); } } if (!KeepImage) { for (u32 i = 0; i < Images.size(); ++i) Images[i]->drop(); Images.clear(); } Driver->getCacheHandler()->getTextureCache().set(0, prevTexture); Driver->testGLError(__LINE__); } COpenGLCoreTexture(const io::path& name, const core::dimension2d& size, E_TEXTURE_TYPE type, ECOLOR_FORMAT format, TOpenGLDriver* driver) : ITexture(name, type), Driver(driver), TextureType(GL_TEXTURE_2D), TextureName(0), InternalFormat(GL_RGBA), PixelFormat(GL_RGBA), PixelType(GL_UNSIGNED_BYTE), Converter(0), LockReadOnly(false), LockImage(0), LockLayer(0), KeepImage(false), MipLevelStored(0), LegacyAutoGenerateMipMaps(false) { DriverType = Driver->getDriverType(); TextureType = TextureTypeIrrToGL(Type); HasMipMaps = false; IsRenderTarget = true; OriginalColorFormat = format; if (ECF_UNKNOWN == OriginalColorFormat) ColorFormat = getBestColorFormat(Driver->getColorFormat()); else ColorFormat = OriginalColorFormat; OriginalSize = size; Size = OriginalSize; Pitch = Size.Width * IImage::getBitsPerPixelFromFormat(ColorFormat) / 8; if ( !Driver->getColorFormatParameters(ColorFormat, InternalFormat, PixelFormat, PixelType, &Converter) ) { os::Printer::log("COpenGLCoreTexture: Color format is not supported", ColorFormatNames[ColorFormat < ECF_UNKNOWN?ColorFormat:ECF_UNKNOWN], ELL_ERROR); } glGenTextures(1, &TextureName); const COpenGLCoreTexture* prevTexture = Driver->getCacheHandler()->getTextureCache().get(0); Driver->getCacheHandler()->getTextureCache().set(0, this); glTexParameteri(TextureType, GL_TEXTURE_MIN_FILTER, GL_NEAREST); glTexParameteri(TextureType, GL_TEXTURE_MAG_FILTER, GL_NEAREST); glTexParameteri(TextureType, GL_TEXTURE_WRAP_S, GL_CLAMP_TO_EDGE); glTexParameteri(TextureType, GL_TEXTURE_WRAP_T, GL_CLAMP_TO_EDGE); #if defined(GL_VERSION_1_2) glTexParameteri(TextureType, GL_TEXTURE_WRAP_R, GL_CLAMP_TO_EDGE); #endif StatesCache.WrapU = ETC_CLAMP_TO_EDGE; StatesCache.WrapV = ETC_CLAMP_TO_EDGE; StatesCache.WrapW = ETC_CLAMP_TO_EDGE; switch (Type) { case ETT_2D: glTexImage2D(GL_TEXTURE_2D, 0, InternalFormat, Size.Width, Size.Height, 0, PixelFormat, PixelType, 0); break; case ETT_CUBEMAP: glTexImage2D(GL_TEXTURE_CUBE_MAP_POSITIVE_X, 0, InternalFormat, Size.Width, Size.Height, 0, PixelFormat, PixelType, 0); glTexImage2D(GL_TEXTURE_CUBE_MAP_NEGATIVE_X, 0, InternalFormat, Size.Width, Size.Height, 0, PixelFormat, PixelType, 0); glTexImage2D(GL_TEXTURE_CUBE_MAP_POSITIVE_Y, 0, InternalFormat, Size.Width, Size.Height, 0, PixelFormat, PixelType, 0); glTexImage2D(GL_TEXTURE_CUBE_MAP_NEGATIVE_Y, 0, InternalFormat, Size.Width, Size.Height, 0, PixelFormat, PixelType, 0); glTexImage2D(GL_TEXTURE_CUBE_MAP_POSITIVE_Z, 0, InternalFormat, Size.Width, Size.Height, 0, PixelFormat, PixelType, 0); glTexImage2D(GL_TEXTURE_CUBE_MAP_NEGATIVE_Z, 0, InternalFormat, Size.Width, Size.Height, 0, PixelFormat, PixelType, 0); break; } Driver->getCacheHandler()->getTextureCache().set(0, prevTexture); if ( Driver->testGLError(__LINE__) ) { char msg[256]; snprintf_irr(msg, 256, "COpenGLCoreTexture: InternalFormat:0x%04x PixelFormat:0x%04x", (int)InternalFormat, (int)PixelFormat); os::Printer::log(msg, ELL_ERROR); } } virtual ~COpenGLCoreTexture() { if (TextureName) glDeleteTextures(1, &TextureName); if (LockImage) LockImage->drop(); for (u32 i = 0; i < Images.size(); ++i) Images[i]->drop(); } virtual void* lock(E_TEXTURE_LOCK_MODE mode = ETLM_READ_WRITE, u32 mipmapLevel=0, u32 layer = 0, E_TEXTURE_LOCK_FLAGS lockFlags = ETLF_FLIP_Y_UP_RTT) IRR_OVERRIDE { if (LockImage) return getLockImageData(MipLevelStored); if (IImage::isCompressedFormat(ColorFormat)) return 0; LockReadOnly |= (mode == ETLM_READ_ONLY); LockLayer = layer; MipLevelStored = mipmapLevel; if (KeepImage) { IRR_DEBUG_BREAK_IF(LockLayer > Images.size()) if ( mipmapLevel == 0 || (Images[LockLayer] && Images[LockLayer]->getMipMapsData(mipmapLevel)) ) { LockImage = Images[LockLayer]; LockImage->grab(); } } if ( !LockImage ) { core::dimension2d lockImageSize( IImage::getMipMapsSize(Size, MipLevelStored)); // note: we save mipmap data also in the image because IImage doesn't allow saving single mipmap levels to the mipmap data LockImage = Driver->createImage(ColorFormat, lockImageSize); if (LockImage && mode != ETLM_WRITE_ONLY) { bool passed = true; #ifdef IRR_COMPILE_GL_COMMON IImage* tmpImage = LockImage; // not sure yet if the size required by glGetTexImage is always correct, if not we might have to allocate a different tmpImage and convert colors later on. Driver->getCacheHandler()->getTextureCache().set(0, this); Driver->testGLError(__LINE__); GLenum tmpTextureType = TextureType; if (tmpTextureType == GL_TEXTURE_CUBE_MAP) { IRR_DEBUG_BREAK_IF(layer > 5) tmpTextureType = GL_TEXTURE_CUBE_MAP_POSITIVE_X + layer; } glGetTexImage(tmpTextureType, MipLevelStored, PixelFormat, PixelType, tmpImage->getData()); Driver->testGLError(__LINE__); if (IsRenderTarget && lockFlags == ETLF_FLIP_Y_UP_RTT) { const s32 pitch = tmpImage->getPitch(); u8* srcA = static_cast(tmpImage->getData()); u8* srcB = srcA + (tmpImage->getDimension().Height - 1) * pitch; u8* tmpBuffer = new u8[pitch]; for (u32 i = 0; i < tmpImage->getDimension().Height; i += 2) { memcpy(tmpBuffer, srcA, pitch); memcpy(srcA, srcB, pitch); memcpy(srcB, tmpBuffer, pitch); srcA += pitch; srcB -= pitch; } delete[] tmpBuffer; } #elif (defined(IRR_COMPILE_GLES2_COMMON) || defined(IRR_COMPILE_GLES_COMMON)) // TODO: on ES2 we can likely also work with glCopyTexImage2D instead of rendering which should be faster. COpenGLCoreTexture* tmpTexture = new COpenGLCoreTexture("OGL_CORE_LOCK_TEXTURE", Size, ETT_2D, ColorFormat, Driver); GLuint tmpFBO = 0; Driver->irrGlGenFramebuffers(1, &tmpFBO); GLint prevViewportX = 0; GLint prevViewportY = 0; GLsizei prevViewportWidth = 0; GLsizei prevViewportHeight = 0; Driver->getCacheHandler()->getViewport(prevViewportX, prevViewportY, prevViewportWidth, prevViewportHeight); Driver->getCacheHandler()->setViewport(0, 0, Size.Width, Size.Height); GLuint prevFBO = 0; Driver->getCacheHandler()->getFBO(prevFBO); Driver->getCacheHandler()->setFBO(tmpFBO); Driver->irrGlFramebufferTexture2D(GL_FRAMEBUFFER, GL_COLOR_ATTACHMENT0, GL_TEXTURE_2D, tmpTexture->getOpenGLTextureName(), 0); glClear(GL_COLOR_BUFFER_BIT); Driver->draw2DImageQuad(this, layer, true); IImage* tmpImage = Driver->createImage(ECF_A8R8G8B8, Size); glReadPixels(0, 0, Size.Width, Size.Height, GL_RGBA, GL_UNSIGNED_BYTE, tmpImage->getData()); Driver->getCacheHandler()->setFBO(prevFBO); Driver->getCacheHandler()->setViewport(prevViewportX, prevViewportY, prevViewportWidth, prevViewportHeight); Driver->irrGlDeleteFramebuffers(1, &tmpFBO); delete tmpTexture; void* src = tmpImage->getData(); void* dest = LockImage->getData(); switch (ColorFormat) { case ECF_A1R5G5B5: CColorConverter::convert_A8R8G8B8toA1B5G5R5(src, tmpImage->getDimension().getArea(), dest); break; case ECF_R5G6B5: CColorConverter::convert_A8R8G8B8toR5G6B5(src, tmpImage->getDimension().getArea(), dest); break; case ECF_R8G8B8: CColorConverter::convert_A8R8G8B8toB8G8R8(src, tmpImage->getDimension().getArea(), dest); break; case ECF_A8R8G8B8: CColorConverter::convert_A8R8G8B8toA8B8G8R8(src, tmpImage->getDimension().getArea(), dest); break; default: passed = false; break; } tmpImage->drop(); #endif if (!passed) { LockImage->drop(); LockImage = 0; } } Driver->testGLError(__LINE__); } return (LockImage) ? getLockImageData(MipLevelStored) : 0; } virtual void unlock() IRR_OVERRIDE { if (!LockImage) return; if (!LockReadOnly) { const COpenGLCoreTexture* prevTexture = Driver->getCacheHandler()->getTextureCache().get(0); Driver->getCacheHandler()->getTextureCache().set(0, this); uploadTexture(false, LockLayer, MipLevelStored, getLockImageData(MipLevelStored)); Driver->getCacheHandler()->getTextureCache().set(0, prevTexture); } LockImage->drop(); LockReadOnly = false; LockImage = 0; LockLayer = 0; } virtual void regenerateMipMapLevels(void* data = 0, u32 layer = 0) IRR_OVERRIDE { if (!HasMipMaps || LegacyAutoGenerateMipMaps || (Size.Width <= 1 && Size.Height <= 1)) return; const COpenGLCoreTexture* prevTexture = Driver->getCacheHandler()->getTextureCache().get(0); Driver->getCacheHandler()->getTextureCache().set(0, this); if (data) { u32 width = Size.Width; u32 height = Size.Height; u8* tmpData = static_cast(data); size_t dataSize = 0; u32 level = 0; do { if (width > 1) width >>= 1; if (height > 1) height >>= 1; dataSize = IImage::getDataSizeFromFormat(ColorFormat, width, height); ++level; uploadTexture(true, layer, level, tmpData); tmpData += dataSize; } while (width != 1 || height != 1); } else { #ifdef IRR_OPENGL_HAS_glGenerateMipmap #if !defined(IRR_COMPILE_GLES2_COMMON) glEnable(GL_TEXTURE_2D); // Hack some ATI cards need this glEnable according to https://www.khronos.org/opengl/wiki/Common_Mistakes #endif Driver->irrGlGenerateMipmap(TextureType); #endif } Driver->getCacheHandler()->getTextureCache().set(0, prevTexture); } GLenum getOpenGLTextureType() const { return TextureType; } GLuint getOpenGLTextureName() const { return TextureName; } SStatesCache& getStatesCache() const { return StatesCache; } protected: void * getLockImageData(irr::u32 miplevel) const { if ( KeepImage && MipLevelStored > 0 && LockImage->getMipMapsData(MipLevelStored) ) { return LockImage->getMipMapsData(MipLevelStored); } return LockImage->getData(); } ECOLOR_FORMAT getBestColorFormat(ECOLOR_FORMAT format) { // We only try for to adapt "simple" formats ECOLOR_FORMAT destFormat = (format <= ECF_A8R8G8B8) ? ECF_A8R8G8B8 : format; switch (format) { case ECF_A1R5G5B5: if (!Driver->getTextureCreationFlag(ETCF_ALWAYS_32_BIT)) destFormat = ECF_A1R5G5B5; break; case ECF_R5G6B5: if (!Driver->getTextureCreationFlag(ETCF_ALWAYS_32_BIT)) destFormat = ECF_R5G6B5; break; case ECF_A8R8G8B8: if (Driver->getTextureCreationFlag(ETCF_ALWAYS_16_BIT) || Driver->getTextureCreationFlag(ETCF_OPTIMIZED_FOR_SPEED)) destFormat = ECF_A1R5G5B5; break; case ECF_R8G8B8: // Note: Using ECF_A8R8G8B8 even when ETCF_ALWAYS_32_BIT is not set as 24 bit textures fail with too many cards if (Driver->getTextureCreationFlag(ETCF_ALWAYS_16_BIT) || Driver->getTextureCreationFlag(ETCF_OPTIMIZED_FOR_SPEED)) destFormat = ECF_A1R5G5B5; default: break; } if (Driver->getTextureCreationFlag(ETCF_NO_ALPHA_CHANNEL)) { switch (destFormat) { case ECF_A1R5G5B5: destFormat = ECF_R5G6B5; break; case ECF_A8R8G8B8: destFormat = ECF_R8G8B8; break; default: break; } } return destFormat; } void getImageValues(const IImage* image) { OriginalColorFormat = image->getColorFormat(); ColorFormat = getBestColorFormat(OriginalColorFormat); if ( !Driver->getColorFormatParameters(ColorFormat, InternalFormat, PixelFormat, PixelType, &Converter) ) { os::Printer::log("getImageValues: Color format is not supported", ColorFormatNames[ColorFormat < ECF_UNKNOWN?ColorFormat:ECF_UNKNOWN], ELL_ERROR); // not quitting as it will use some alternative internal format } if (IImage::isCompressedFormat(image->getColorFormat())) { KeepImage = false; } OriginalSize = image->getDimension(); Size = OriginalSize; if (Size.Width == 0 || Size.Height == 0) { os::Printer::log("Invalid size of image for texture.", ELL_ERROR); return; } const f32 ratio = (f32)Size.Width / (f32)Size.Height; if ((Size.Width > Driver->MaxTextureSize) && (ratio >= 1.f)) { Size.Width = Driver->MaxTextureSize; Size.Height = (u32)(Driver->MaxTextureSize / ratio); } else if (Size.Height > Driver->MaxTextureSize) { Size.Height = Driver->MaxTextureSize; Size.Width = (u32)(Driver->MaxTextureSize * ratio); } bool needSquare = (!Driver->queryFeature(EVDF_TEXTURE_NSQUARE) || Type == ETT_CUBEMAP); Size = Size.getOptimalSize(!Driver->queryFeature(EVDF_TEXTURE_NPOT), needSquare, true, Driver->MaxTextureSize); Pitch = Size.Width * IImage::getBitsPerPixelFromFormat(ColorFormat) / 8; } void uploadTexture(bool initTexture, u32 layer, u32 level, void* data) { if (!data) return; u32 width = Size.Width >> level; u32 height = Size.Height >> level; GLenum tmpTextureType = TextureType; if (tmpTextureType == GL_TEXTURE_CUBE_MAP) { IRR_DEBUG_BREAK_IF(layer > 5) tmpTextureType = GL_TEXTURE_CUBE_MAP_POSITIVE_X + layer; } if (!IImage::isCompressedFormat(ColorFormat)) { CImage* tmpImage = 0; void* tmpData = data; if (Converter) { const core::dimension2d tmpImageSize(width, height); tmpImage = new CImage(ColorFormat, tmpImageSize); tmpData = tmpImage->getData(); Converter(data, tmpImageSize.getArea(), tmpData); } switch (TextureType) { case GL_TEXTURE_2D: case GL_TEXTURE_CUBE_MAP: if (initTexture) glTexImage2D(tmpTextureType, level, InternalFormat, width, height, 0, PixelFormat, PixelType, tmpData); else glTexSubImage2D(tmpTextureType, level, 0, 0, width, height, PixelFormat, PixelType, tmpData); Driver->testGLError(__LINE__); break; default: break; } delete tmpImage; } else { GLsizei dataSize = (GLsizei)IImage::getDataSizeFromFormat(ColorFormat, width, height); switch (TextureType) { case GL_TEXTURE_2D: case GL_TEXTURE_CUBE_MAP: if (initTexture) Driver->irrGlCompressedTexImage2D(tmpTextureType, level, InternalFormat, width, height, 0, dataSize, data); else Driver->irrGlCompressedTexSubImage2D(tmpTextureType, level, 0, 0, width, height, PixelFormat, dataSize, data); Driver->testGLError(__LINE__); break; default: break; } } } GLenum TextureTypeIrrToGL(E_TEXTURE_TYPE type) const { switch ( type) { case ETT_2D: return GL_TEXTURE_2D; case ETT_CUBEMAP: return GL_TEXTURE_CUBE_MAP; } os::Printer::log("COpenGLCoreTexture::TextureTypeIrrToGL unknown texture type", ELL_WARNING); return GL_TEXTURE_2D; } TOpenGLDriver* Driver; GLenum TextureType; GLuint TextureName; GLint InternalFormat; GLenum PixelFormat; GLenum PixelType; void (*Converter)(const void*, u32, void*); bool LockReadOnly; IImage* LockImage; u32 LockLayer; bool KeepImage; core::array Images; u8 MipLevelStored; bool LegacyAutoGenerateMipMaps; mutable SStatesCache StatesCache; }; } } #endif #endif