2023-10-03 20:37:00 +02:00
// 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
# pragma once
# include "irrArray.h"
# include "SMaterialLayer.h"
# include "ITexture.h"
# include "EDriverFeatures.h"
# include "os.h"
# include "CImage.h"
# include "CColorConverter.h"
2024-01-16 20:17:33 +01:00
# include "mt_opengl.h"
2023-10-03 20:37:00 +02:00
namespace irr
{
namespace video
{
template < class TOpenGLDriver >
class COpenGLCoreTexture : public ITexture
{
public :
struct SStatesCache
{
2024-03-20 19:35:52 +01:00
SStatesCache ( ) :
WrapU ( ETC_REPEAT ) , WrapV ( ETC_REPEAT ) , WrapW ( ETC_REPEAT ) ,
LODBias ( 0 ) , AnisotropicFilter ( 0 ) , MinFilter ( video : : ETMINF_NEAREST_MIPMAP_NEAREST ) ,
MagFilter ( video : : ETMAGF_NEAREST ) , MipMapStatus ( false ) , IsCached ( false )
2023-10-03 20:37:00 +02:00
{
}
u8 WrapU ;
u8 WrapV ;
u8 WrapW ;
s8 LODBias ;
u8 AnisotropicFilter ;
video : : E_TEXTURE_MIN_FILTER MinFilter ;
video : : E_TEXTURE_MAG_FILTER MagFilter ;
bool MipMapStatus ;
bool IsCached ;
} ;
2024-03-20 19:35:52 +01:00
COpenGLCoreTexture ( const io : : path & name , const core : : array < IImage * > & 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 )
2023-10-03 20:37:00 +02:00
{
_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 ] ) ;
2024-02-21 19:20:01 +01:00
if ( ! InternalFormat )
return ;
2023-10-03 20:37:00 +02:00
2024-03-20 19:35:52 +01:00
const core : : array < IImage * > * tmpImages = & images ;
2023-10-03 20:37:00 +02:00
2024-03-20 19:35:52 +01:00
if ( KeepImage | | OriginalSize ! = Size | | OriginalColorFormat ! = ColorFormat ) {
2023-10-03 20:37:00 +02:00
Images . set_used ( images . size ( ) ) ;
2024-03-20 19:35:52 +01:00
for ( u32 i = 0 ; i < images . size ( ) ; + + i ) {
2023-10-03 20:37:00 +02:00
Images [ i ] = Driver - > createImage ( ColorFormat , Size ) ;
if ( images [ i ] - > getDimension ( ) = = Size )
images [ i ] - > copyTo ( Images [ i ] ) ;
else
images [ i ] - > copyToScaling ( Images [ i ] ) ;
2024-03-20 19:35:52 +01:00
if ( images [ i ] - > getMipMapsData ( ) ) {
if ( OriginalSize = = Size & & OriginalColorFormat = = ColorFormat ) {
Images [ i ] - > setMipMapsData ( images [ i ] - > getMipMapsData ( ) , false ) ;
} else {
2023-10-03 20:37:00 +02:00
// 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 ;
}
2024-01-16 20:17:33 +01:00
GL . GenTextures ( 1 , & TextureName ) ;
2023-10-03 20:37:00 +02:00
2024-03-20 19:35:52 +01:00
const COpenGLCoreTexture * prevTexture = Driver - > getCacheHandler ( ) - > getTextureCache ( ) . get ( 0 ) ;
2023-10-03 20:37:00 +02:00
Driver - > getCacheHandler ( ) - > getTextureCache ( ) . set ( 0 , this ) ;
2024-01-16 20:17:33 +01:00
GL . TexParameteri ( TextureType , GL_TEXTURE_MIN_FILTER , GL_NEAREST ) ;
GL . TexParameteri ( TextureType , GL_TEXTURE_MAG_FILTER , GL_NEAREST ) ;
2023-10-03 20:37:00 +02:00
# ifdef GL_GENERATE_MIPMAP_HINT
2024-03-20 19:35:52 +01:00
if ( HasMipMaps ) {
2023-10-03 20:37:00 +02:00
if ( Driver - > getTextureCreationFlag ( ETCF_OPTIMIZED_FOR_SPEED ) )
2024-01-16 20:17:33 +01:00
GL . Hint ( GL_GENERATE_MIPMAP_HINT , GL_FASTEST ) ;
2023-10-03 20:37:00 +02:00
else if ( Driver - > getTextureCreationFlag ( ETCF_OPTIMIZED_FOR_QUALITY ) )
2024-01-16 20:17:33 +01:00
GL . Hint ( GL_GENERATE_MIPMAP_HINT , GL_NICEST ) ;
2023-10-03 20:37:00 +02:00
else
2024-01-16 20:17:33 +01:00
GL . Hint ( GL_GENERATE_MIPMAP_HINT , GL_DONT_CARE ) ;
2023-10-03 20:37:00 +02:00
}
# endif
for ( u32 i = 0 ; i < ( * tmpImages ) . size ( ) ; + + i )
uploadTexture ( true , i , 0 , ( * tmpImages ) [ i ] - > getData ( ) ) ;
2024-03-20 19:35:52 +01:00
if ( HasMipMaps & & ! LegacyAutoGenerateMipMaps ) {
2023-10-03 20:37:00 +02:00
// Create mipmaps (either from image mipmaps or generate them)
2024-03-20 19:35:52 +01:00
for ( u32 i = 0 ; i < ( * tmpImages ) . size ( ) ; + + i ) {
void * mipmapsData = ( * tmpImages ) [ i ] - > getMipMapsData ( ) ;
2023-10-03 20:37:00 +02:00
regenerateMipMapLevels ( mipmapsData , i ) ;
}
}
2024-03-20 19:35:52 +01:00
if ( ! KeepImage ) {
2023-10-03 20:37:00 +02:00
for ( u32 i = 0 ; i < Images . size ( ) ; + + i )
Images [ i ] - > drop ( ) ;
Images . clear ( ) ;
}
Driver - > getCacheHandler ( ) - > getTextureCache ( ) . set ( 0 , prevTexture ) ;
Driver - > testGLError ( __LINE__ ) ;
}
2024-03-20 19:35:52 +01:00
COpenGLCoreTexture ( const io : : path & name , const core : : dimension2d < u32 > & 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 )
2023-10-03 20:37:00 +02:00
{
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 ;
2024-03-20 19:35:52 +01:00
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 ) ;
2024-02-21 19:20:01 +01:00
return ;
2023-10-03 20:37:00 +02:00
}
2024-01-16 20:17:33 +01:00
GL . GenTextures ( 1 , & TextureName ) ;
2023-10-03 20:37:00 +02:00
2024-03-20 19:35:52 +01:00
const COpenGLCoreTexture * prevTexture = Driver - > getCacheHandler ( ) - > getTextureCache ( ) . get ( 0 ) ;
2023-10-03 20:37:00 +02:00
Driver - > getCacheHandler ( ) - > getTextureCache ( ) . set ( 0 , this ) ;
2024-01-16 20:17:33 +01:00
GL . TexParameteri ( TextureType , GL_TEXTURE_MIN_FILTER , GL_NEAREST ) ;
GL . TexParameteri ( TextureType , GL_TEXTURE_MAG_FILTER , GL_NEAREST ) ;
GL . TexParameteri ( TextureType , GL_TEXTURE_WRAP_S , GL_CLAMP_TO_EDGE ) ;
GL . TexParameteri ( TextureType , GL_TEXTURE_WRAP_T , GL_CLAMP_TO_EDGE ) ;
2023-10-03 20:37:00 +02:00
# if defined(GL_VERSION_1_2)
2024-01-16 20:17:33 +01:00
GL . TexParameteri ( TextureType , GL_TEXTURE_WRAP_R , GL_CLAMP_TO_EDGE ) ;
2023-10-03 20:37:00 +02:00
# endif
StatesCache . WrapU = ETC_CLAMP_TO_EDGE ;
StatesCache . WrapV = ETC_CLAMP_TO_EDGE ;
StatesCache . WrapW = ETC_CLAMP_TO_EDGE ;
2024-03-20 19:35:52 +01:00
switch ( Type ) {
2023-10-03 20:37:00 +02:00
case ETT_2D :
2024-01-16 20:17:33 +01:00
GL . TexImage2D ( GL_TEXTURE_2D , 0 , InternalFormat , Size . Width , Size . Height , 0 , PixelFormat , PixelType , 0 ) ;
2023-10-03 20:37:00 +02:00
break ;
case ETT_CUBEMAP :
2024-01-16 20:17:33 +01:00
GL . TexImage2D ( GL_TEXTURE_CUBE_MAP_POSITIVE_X , 0 , InternalFormat , Size . Width , Size . Height , 0 , PixelFormat , PixelType , 0 ) ;
GL . TexImage2D ( GL_TEXTURE_CUBE_MAP_NEGATIVE_X , 0 , InternalFormat , Size . Width , Size . Height , 0 , PixelFormat , PixelType , 0 ) ;
GL . TexImage2D ( GL_TEXTURE_CUBE_MAP_POSITIVE_Y , 0 , InternalFormat , Size . Width , Size . Height , 0 , PixelFormat , PixelType , 0 ) ;
GL . TexImage2D ( GL_TEXTURE_CUBE_MAP_NEGATIVE_Y , 0 , InternalFormat , Size . Width , Size . Height , 0 , PixelFormat , PixelType , 0 ) ;
GL . TexImage2D ( GL_TEXTURE_CUBE_MAP_POSITIVE_Z , 0 , InternalFormat , Size . Width , Size . Height , 0 , PixelFormat , PixelType , 0 ) ;
GL . TexImage2D ( GL_TEXTURE_CUBE_MAP_NEGATIVE_Z , 0 , InternalFormat , Size . Width , Size . Height , 0 , PixelFormat , PixelType , 0 ) ;
2023-10-03 20:37:00 +02:00
break ;
}
Driver - > getCacheHandler ( ) - > getTextureCache ( ) . set ( 0 , prevTexture ) ;
2024-03-20 19:35:52 +01:00
if ( Driver - > testGLError ( __LINE__ ) ) {
2023-10-03 20:37:00 +02:00
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 )
2024-01-16 20:17:33 +01:00
GL . DeleteTextures ( 1 , & TextureName ) ;
2023-10-03 20:37:00 +02:00
if ( LockImage )
LockImage - > drop ( ) ;
for ( u32 i = 0 ; i < Images . size ( ) ; + + i )
Images [ i ] - > drop ( ) ;
}
2024-03-20 19:35:52 +01:00
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 ) override
2023-10-03 20:37:00 +02:00
{
if ( LockImage )
return getLockImageData ( MipLevelStored ) ;
if ( IImage : : isCompressedFormat ( ColorFormat ) )
return 0 ;
LockReadOnly | = ( mode = = ETLM_READ_ONLY ) ;
LockLayer = layer ;
MipLevelStored = mipmapLevel ;
2024-03-20 19:35:52 +01:00
if ( KeepImage ) {
2023-10-03 20:37:00 +02:00
_IRR_DEBUG_BREAK_IF ( LockLayer > Images . size ( ) )
2024-03-20 19:35:52 +01:00
if ( mipmapLevel = = 0 | | ( Images [ LockLayer ] & & Images [ LockLayer ] - > getMipMapsData ( mipmapLevel ) ) ) {
2023-10-03 20:37:00 +02:00
LockImage = Images [ LockLayer ] ;
LockImage - > grab ( ) ;
}
}
2024-03-20 19:35:52 +01:00
if ( ! LockImage ) {
core : : dimension2d < u32 > lockImageSize ( IImage : : getMipMapsSize ( Size , MipLevelStored ) ) ;
2023-10-03 20:37:00 +02:00
// 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 ) ;
2024-03-20 19:35:52 +01:00
if ( LockImage & & mode ! = ETLM_WRITE_ONLY ) {
2023-10-03 20:37:00 +02:00
bool passed = true ;
# ifdef IRR_COMPILE_GL_COMMON
2024-03-20 19:35:52 +01:00
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.
2023-10-03 20:37:00 +02:00
Driver - > getCacheHandler ( ) - > getTextureCache ( ) . set ( 0 , this ) ;
Driver - > testGLError ( __LINE__ ) ;
GLenum tmpTextureType = TextureType ;
2024-03-20 19:35:52 +01:00
if ( tmpTextureType = = GL_TEXTURE_CUBE_MAP ) {
2023-10-03 20:37:00 +02:00
_IRR_DEBUG_BREAK_IF ( layer > 5 )
tmpTextureType = GL_TEXTURE_CUBE_MAP_POSITIVE_X + layer ;
}
2024-01-16 20:17:33 +01:00
GL . GetTexImage ( tmpTextureType , MipLevelStored , PixelFormat , PixelType , tmpImage - > getData ( ) ) ;
2023-10-03 20:37:00 +02:00
Driver - > testGLError ( __LINE__ ) ;
2024-03-20 19:35:52 +01:00
if ( IsRenderTarget & & lockFlags = = ETLF_FLIP_Y_UP_RTT ) {
2023-10-03 20:37:00 +02:00
const s32 pitch = tmpImage - > getPitch ( ) ;
2024-03-20 19:35:52 +01:00
u8 * srcA = static_cast < u8 * > ( tmpImage - > getData ( ) ) ;
u8 * srcB = srcA + ( tmpImage - > getDimension ( ) . Height - 1 ) * pitch ;
2023-10-03 20:37:00 +02:00
2024-03-20 19:35:52 +01:00
u8 * tmpBuffer = new u8 [ pitch ] ;
2023-10-03 20:37:00 +02:00
2024-03-20 19:35:52 +01:00
for ( u32 i = 0 ; i < tmpImage - > getDimension ( ) . Height ; i + = 2 ) {
2023-10-03 20:37:00 +02:00
memcpy ( tmpBuffer , srcA , pitch ) ;
memcpy ( srcA , srcB , pitch ) ;
memcpy ( srcB , tmpBuffer , pitch ) ;
srcA + = pitch ;
srcB - = pitch ;
}
delete [ ] tmpBuffer ;
}
2024-03-20 19:35:52 +01:00
# 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 ) ;
2023-10-03 20:37:00 +02:00
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 ) ;
2024-01-16 20:17:33 +01:00
GL . Clear ( GL_COLOR_BUFFER_BIT ) ;
2023-10-03 20:37:00 +02:00
Driver - > draw2DImage ( this , layer , true ) ;
2024-03-20 19:35:52 +01:00
IImage * tmpImage = Driver - > createImage ( ECF_A8R8G8B8 , Size ) ;
2024-01-16 20:17:33 +01:00
GL . ReadPixels ( 0 , 0 , Size . Width , Size . Height , GL_RGBA , GL_UNSIGNED_BYTE , tmpImage - > getData ( ) ) ;
2023-10-03 20:37:00 +02:00
Driver - > getCacheHandler ( ) - > setFBO ( prevFBO ) ;
Driver - > getCacheHandler ( ) - > setViewport ( prevViewportX , prevViewportY , prevViewportWidth , prevViewportHeight ) ;
Driver - > irrGlDeleteFramebuffers ( 1 , & tmpFBO ) ;
delete tmpTexture ;
2024-03-20 19:35:52 +01:00
void * src = tmpImage - > getData ( ) ;
void * dest = LockImage - > getData ( ) ;
2023-10-03 20:37:00 +02:00
2024-03-20 19:35:52 +01:00
switch ( ColorFormat ) {
2023-10-03 20:37:00 +02:00
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
2024-03-20 19:35:52 +01:00
if ( ! passed ) {
2023-10-03 20:37:00 +02:00
LockImage - > drop ( ) ;
LockImage = 0 ;
}
}
Driver - > testGLError ( __LINE__ ) ;
}
return ( LockImage ) ? getLockImageData ( MipLevelStored ) : 0 ;
}
void unlock ( ) override
{
if ( ! LockImage )
return ;
2024-03-20 19:35:52 +01:00
if ( ! LockReadOnly ) {
const COpenGLCoreTexture * prevTexture = Driver - > getCacheHandler ( ) - > getTextureCache ( ) . get ( 0 ) ;
2023-10-03 20:37:00 +02:00
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 ;
}
2024-03-20 19:35:52 +01:00
void regenerateMipMapLevels ( void * data = 0 , u32 layer = 0 ) override
2023-10-03 20:37:00 +02:00
{
if ( ! HasMipMaps | | LegacyAutoGenerateMipMaps | | ( Size . Width < = 1 & & Size . Height < = 1 ) )
return ;
2024-03-20 19:35:52 +01:00
const COpenGLCoreTexture * prevTexture = Driver - > getCacheHandler ( ) - > getTextureCache ( ) . get ( 0 ) ;
2023-10-03 20:37:00 +02:00
Driver - > getCacheHandler ( ) - > getTextureCache ( ) . set ( 0 , this ) ;
2024-03-20 19:35:52 +01:00
if ( data ) {
2023-10-03 20:37:00 +02:00
u32 width = Size . Width ;
u32 height = Size . Height ;
2024-03-20 19:35:52 +01:00
u8 * tmpData = static_cast < u8 * > ( data ) ;
2023-10-03 20:37:00 +02:00
u32 dataSize = 0 ;
u32 level = 0 ;
2024-03-20 19:35:52 +01:00
do {
2023-10-03 20:37:00 +02:00
if ( width > 1 )
width > > = 1 ;
if ( height > 1 )
height > > = 1 ;
dataSize = IImage : : getDataSizeFromFormat ( ColorFormat , width , height ) ;
+ + level ;
uploadTexture ( true , layer , level , tmpData ) ;
tmpData + = dataSize ;
2024-03-20 19:35:52 +01:00
} while ( width ! = 1 | | height ! = 1 ) ;
} else {
2023-10-03 20:37:00 +02:00
Driver - > irrGlGenerateMipmap ( TextureType ) ;
}
Driver - > getCacheHandler ( ) - > getTextureCache ( ) . set ( 0 , prevTexture ) ;
}
GLenum getOpenGLTextureType ( ) const
{
return TextureType ;
}
GLuint getOpenGLTextureName ( ) const
{
return TextureName ;
}
2024-03-20 19:35:52 +01:00
SStatesCache & getStatesCache ( ) const
2023-10-03 20:37:00 +02:00
{
return StatesCache ;
}
protected :
2024-03-20 19:35:52 +01:00
void * getLockImageData ( irr : : u32 miplevel ) const
2023-10-03 20:37:00 +02:00
{
2024-03-20 19:35:52 +01:00
if ( KeepImage & & MipLevelStored > 0 & & LockImage - > getMipMapsData ( MipLevelStored ) ) {
2023-10-03 20:37:00 +02:00
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 ;
2024-03-20 19:35:52 +01:00
switch ( format ) {
2023-10-03 20:37:00 +02:00
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 ) | |
2024-03-20 19:35:52 +01:00
Driver - > getTextureCreationFlag ( ETCF_OPTIMIZED_FOR_SPEED ) )
2023-10-03 20:37:00 +02:00
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 ;
}
2024-03-20 19:35:52 +01:00
if ( Driver - > getTextureCreationFlag ( ETCF_NO_ALPHA_CHANNEL ) ) {
switch ( destFormat ) {
2023-10-03 20:37:00 +02:00
case ECF_A1R5G5B5 :
destFormat = ECF_R5G6B5 ;
break ;
case ECF_A8R8G8B8 :
destFormat = ECF_R8G8B8 ;
break ;
default :
break ;
}
}
return destFormat ;
}
2024-03-20 19:35:52 +01:00
void getImageValues ( const IImage * image )
2023-10-03 20:37:00 +02:00
{
OriginalColorFormat = image - > getColorFormat ( ) ;
ColorFormat = getBestColorFormat ( OriginalColorFormat ) ;
2024-03-20 19:35:52 +01:00
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 ) ;
2024-02-21 19:20:01 +01:00
InternalFormat = 0 ;
return ;
2023-10-03 20:37:00 +02:00
}
2024-03-20 19:35:52 +01:00
if ( IImage : : isCompressedFormat ( image - > getColorFormat ( ) ) ) {
2023-10-03 20:37:00 +02:00
KeepImage = false ;
}
OriginalSize = image - > getDimension ( ) ;
Size = OriginalSize ;
2024-03-20 19:35:52 +01:00
if ( Size . Width = = 0 | | Size . Height = = 0 ) {
2023-10-03 20:37:00 +02:00
os : : Printer : : log ( " Invalid size of image for texture. " , ELL_ERROR ) ;
return ;
}
const f32 ratio = ( f32 ) Size . Width / ( f32 ) Size . Height ;
2024-03-20 19:35:52 +01:00
if ( ( Size . Width > Driver - > MaxTextureSize ) & & ( ratio > = 1.f ) ) {
2023-10-03 20:37:00 +02:00
Size . Width = Driver - > MaxTextureSize ;
Size . Height = ( u32 ) ( Driver - > MaxTextureSize / ratio ) ;
2024-03-20 19:35:52 +01:00
} else if ( Size . Height > Driver - > MaxTextureSize ) {
2023-10-03 20:37:00 +02:00
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 ;
}
2024-03-20 19:35:52 +01:00
void uploadTexture ( bool initTexture , u32 layer , u32 level , void * data )
2023-10-03 20:37:00 +02:00
{
if ( ! data )
return ;
u32 width = Size . Width > > level ;
u32 height = Size . Height > > level ;
2024-01-05 12:07:30 +01:00
if ( width < 1 )
width = 1 ;
if ( height < 1 )
height = 1 ;
2023-10-03 20:37:00 +02:00
GLenum tmpTextureType = TextureType ;
2024-03-20 19:35:52 +01:00
if ( tmpTextureType = = GL_TEXTURE_CUBE_MAP ) {
2023-10-03 20:37:00 +02:00
_IRR_DEBUG_BREAK_IF ( layer > 5 )
tmpTextureType = GL_TEXTURE_CUBE_MAP_POSITIVE_X + layer ;
}
2024-03-20 19:35:52 +01:00
if ( ! IImage : : isCompressedFormat ( ColorFormat ) ) {
CImage * tmpImage = 0 ;
void * tmpData = data ;
2023-10-03 20:37:00 +02:00
2024-03-20 19:35:52 +01:00
if ( Converter ) {
2023-10-03 20:37:00 +02:00
const core : : dimension2d < u32 > tmpImageSize ( width , height ) ;
tmpImage = new CImage ( ColorFormat , tmpImageSize ) ;
tmpData = tmpImage - > getData ( ) ;
Converter ( data , tmpImageSize . getArea ( ) , tmpData ) ;
}
2024-03-20 19:35:52 +01:00
switch ( TextureType ) {
2023-10-03 20:37:00 +02:00
case GL_TEXTURE_2D :
case GL_TEXTURE_CUBE_MAP :
if ( initTexture )
2024-01-16 20:17:33 +01:00
GL . TexImage2D ( tmpTextureType , level , InternalFormat , width , height , 0 , PixelFormat , PixelType , tmpData ) ;
2023-10-03 20:37:00 +02:00
else
2024-01-16 20:17:33 +01:00
GL . TexSubImage2D ( tmpTextureType , level , 0 , 0 , width , height , PixelFormat , PixelType , tmpData ) ;
2023-10-03 20:37:00 +02:00
Driver - > testGLError ( __LINE__ ) ;
break ;
default :
break ;
}
delete tmpImage ;
2024-03-20 19:35:52 +01:00
} else {
2023-10-03 20:37:00 +02:00
u32 dataSize = IImage : : getDataSizeFromFormat ( ColorFormat , width , height ) ;
2024-03-20 19:35:52 +01:00
switch ( TextureType ) {
2023-10-03 20:37:00 +02:00
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
{
2024-03-20 19:35:52 +01:00
switch ( type ) {
2023-10-03 20:37:00 +02:00
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 ;
}
2024-03-20 19:35:52 +01:00
TOpenGLDriver * Driver ;
2023-10-03 20:37:00 +02:00
GLenum TextureType ;
GLuint TextureName ;
GLint InternalFormat ;
GLenum PixelFormat ;
GLenum PixelType ;
2024-03-20 19:35:52 +01:00
void ( * Converter ) ( const void * , s32 , void * ) ;
2023-10-03 20:37:00 +02:00
bool LockReadOnly ;
2024-03-20 19:35:52 +01:00
IImage * LockImage ;
2023-10-03 20:37:00 +02:00
u32 LockLayer ;
bool KeepImage ;
2024-03-20 19:35:52 +01:00
core : : array < IImage * > Images ;
2023-10-03 20:37:00 +02:00
u8 MipLevelStored ;
bool LegacyAutoGenerateMipMaps ;
mutable SStatesCache StatesCache ;
} ;
}
}