2024-03-21 20:13:15 +01: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
2024-08-17 19:49:11 +02:00
# include <vector>
2024-03-21 20:13:15 +01:00
# include "SMaterialLayer.h"
# include "ITexture.h"
# include "EDriverFeatures.h"
# include "os.h"
# include "CImage.h"
# include "CColorConverter.h"
# include "mt_opengl.h"
namespace irr
{
namespace video
{
template < class TOpenGLDriver >
class COpenGLCoreTexture : public ITexture
{
public :
struct SStatesCache
{
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 )
{
}
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-08-17 19:49:11 +02:00
COpenGLCoreTexture ( const io : : path & name , const std : : vector < IImage * > & srcImages , E_TEXTURE_TYPE type , TOpenGLDriver * driver ) :
2024-03-21 20:13:15 +01:00
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 )
{
2024-08-17 19:49:11 +02:00
_IRR_DEBUG_BREAK_IF ( srcImages . empty ( ) )
2024-03-21 20:13:15 +01:00
DriverType = Driver - > getDriverType ( ) ;
TextureType = TextureTypeIrrToGL ( Type ) ;
HasMipMaps = Driver - > getTextureCreationFlag ( ETCF_CREATE_MIP_MAPS ) ;
KeepImage = Driver - > getTextureCreationFlag ( ETCF_ALLOW_MEMORY_COPY ) ;
2024-08-17 19:49:11 +02:00
getImageValues ( srcImages [ 0 ] ) ;
2024-03-21 20:13:15 +01:00
if ( ! InternalFormat )
return ;
2024-04-12 12:18:41 +02:00
# ifdef _DEBUG
char lbuf [ 128 ] ;
snprintf_irr ( lbuf , sizeof ( lbuf ) ,
" COpenGLCoreTexture: Type = %d Size = %dx%d (%dx%d) ColorFormat = %d (%d)%s -> %#06x %#06x %#06x%s " ,
( int ) Type , Size . Width , Size . Height , OriginalSize . Width , OriginalSize . Height ,
( int ) ColorFormat , ( int ) OriginalColorFormat ,
HasMipMaps ? " +Mip " : " " ,
InternalFormat , PixelFormat , PixelType , Converter ? " (c) " : " "
) ;
os : : Printer : : log ( lbuf , ELL_DEBUG ) ;
# endif
2024-08-17 19:49:11 +02:00
const auto * tmpImages = & srcImages ;
2024-03-21 20:13:15 +01:00
if ( KeepImage | | OriginalSize ! = Size | | OriginalColorFormat ! = ColorFormat ) {
2024-08-17 19:49:11 +02:00
Images . resize ( srcImages . size ( ) ) ;
2024-03-21 20:13:15 +01:00
2024-08-17 19:49:11 +02:00
for ( size_t i = 0 ; i < srcImages . size ( ) ; + + i ) {
2024-03-21 20:13:15 +01:00
Images [ i ] = Driver - > createImage ( ColorFormat , Size ) ;
2024-08-17 19:49:11 +02:00
if ( srcImages [ i ] - > getDimension ( ) = = Size )
srcImages [ i ] - > copyTo ( Images [ i ] ) ;
2024-03-21 20:13:15 +01:00
else
2024-08-17 19:49:11 +02:00
srcImages [ i ] - > copyToScaling ( Images [ i ] ) ;
2024-03-21 20:13:15 +01:00
2024-08-17 19:49:11 +02:00
if ( srcImages [ i ] - > getMipMapsData ( ) ) {
2024-03-21 20:13:15 +01:00
if ( OriginalSize = = Size & & OriginalColorFormat = = ColorFormat ) {
2024-08-17 19:49:11 +02:00
Images [ i ] - > setMipMapsData ( srcImages [ i ] - > getMipMapsData ( ) , false ) ;
2024-03-21 20:13:15 +01:00
} 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 ;
}
GL . GenTextures ( 1 , & TextureName ) ;
const COpenGLCoreTexture * prevTexture = Driver - > getCacheHandler ( ) - > getTextureCache ( ) . get ( 0 ) ;
Driver - > getCacheHandler ( ) - > getTextureCache ( ) . set ( 0 , this ) ;
GL . TexParameteri ( TextureType , GL_TEXTURE_MIN_FILTER , GL_NEAREST ) ;
GL . TexParameteri ( TextureType , GL_TEXTURE_MAG_FILTER , GL_NEAREST ) ;
# ifdef GL_GENERATE_MIPMAP_HINT
if ( HasMipMaps ) {
if ( Driver - > getTextureCreationFlag ( ETCF_OPTIMIZED_FOR_SPEED ) )
GL . Hint ( GL_GENERATE_MIPMAP_HINT , GL_FASTEST ) ;
else if ( Driver - > getTextureCreationFlag ( ETCF_OPTIMIZED_FOR_QUALITY ) )
GL . Hint ( GL_GENERATE_MIPMAP_HINT , GL_NICEST ) ;
else
GL . Hint ( GL_GENERATE_MIPMAP_HINT , GL_DONT_CARE ) ;
}
# endif
2024-04-12 12:18:41 +02:00
TEST_GL_ERROR ( Driver ) ;
2024-08-17 19:49:11 +02:00
for ( size_t i = 0 ; i < tmpImages - > size ( ) ; + + i )
2024-03-21 20:13:15 +01:00
uploadTexture ( true , i , 0 , ( * tmpImages ) [ i ] - > getData ( ) ) ;
if ( HasMipMaps & & ! LegacyAutoGenerateMipMaps ) {
// Create mipmaps (either from image mipmaps or generate them)
2024-08-17 19:49:11 +02:00
for ( size_t i = 0 ; i < tmpImages - > size ( ) ; + + i ) {
2024-03-21 20:13:15 +01:00
void * mipmapsData = ( * tmpImages ) [ i ] - > getMipMapsData ( ) ;
regenerateMipMapLevels ( mipmapsData , i ) ;
}
}
if ( ! KeepImage ) {
2024-08-17 19:49:11 +02:00
for ( size_t i = 0 ; i < Images . size ( ) ; + + i )
2024-03-21 20:13:15 +01:00
Images [ i ] - > drop ( ) ;
Images . clear ( ) ;
}
Driver - > getCacheHandler ( ) - > getTextureCache ( ) . set ( 0 , prevTexture ) ;
2024-04-12 12:18:41 +02:00
TEST_GL_ERROR ( Driver ) ;
2024-03-21 20:13:15 +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 )
{
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 ) ;
return ;
}
2024-04-12 12:18:41 +02:00
# ifdef _DEBUG
char lbuf [ 100 ] ;
snprintf_irr ( lbuf , sizeof ( lbuf ) ,
" COpenGLCoreTexture: RTT Type = %d Size = %dx%d ColorFormat = %d -> %#06x %#06x %#06x%s " ,
( int ) Type , Size . Width , Size . Height , ( int ) ColorFormat ,
InternalFormat , PixelFormat , PixelType , Converter ? " (c) " : " "
) ;
os : : Printer : : log ( lbuf , ELL_DEBUG ) ;
# endif
2024-03-21 20:13:15 +01:00
GL . GenTextures ( 1 , & TextureName ) ;
const COpenGLCoreTexture * prevTexture = Driver - > getCacheHandler ( ) - > getTextureCache ( ) . get ( 0 ) ;
Driver - > getCacheHandler ( ) - > getTextureCache ( ) . set ( 0 , this ) ;
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 ) ;
# if defined(GL_VERSION_1_2)
GL . TexParameteri ( 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 :
GL . TexImage2D ( GL_TEXTURE_2D , 0 , InternalFormat , Size . Width , Size . Height , 0 , PixelFormat , PixelType , 0 ) ;
break ;
case ETT_CUBEMAP :
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 ) ;
break ;
}
Driver - > getCacheHandler ( ) - > getTextureCache ( ) . set ( 0 , prevTexture ) ;
2024-04-12 12:18:41 +02:00
if ( TEST_GL_ERROR ( Driver ) ) {
2024-03-21 20:13:15 +01: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 )
GL . DeleteTextures ( 1 , & TextureName ) ;
if ( LockImage )
LockImage - > drop ( ) ;
2024-08-17 19:49:11 +02:00
for ( auto * image : Images )
image - > drop ( ) ;
2024-03-21 20:13:15 +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
{
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 < u32 > 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 ;
2024-11-08 12:08:48 +01:00
# ifdef IRR_COMPILE_GL_COMMON // legacy driver
constexpr bool use_gl_impl = true ;
# else
const bool use_gl_impl = Driver - > Version . Spec ! = OpenGLSpec : : ES ;
# endif
if ( use_gl_impl ) {
2024-03-21 20:13:15 +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.
Driver - > getCacheHandler ( ) - > getTextureCache ( ) . set ( 0 , this ) ;
2024-04-12 12:18:41 +02:00
TEST_GL_ERROR ( Driver ) ;
2024-03-21 20:13:15 +01:00
GLenum tmpTextureType = TextureType ;
if ( tmpTextureType = = GL_TEXTURE_CUBE_MAP ) {
_IRR_DEBUG_BREAK_IF ( layer > 5 )
tmpTextureType = GL_TEXTURE_CUBE_MAP_POSITIVE_X + layer ;
}
GL . GetTexImage ( tmpTextureType , MipLevelStored , PixelFormat , PixelType , tmpImage - > getData ( ) ) ;
2024-04-12 12:18:41 +02:00
TEST_GL_ERROR ( Driver ) ;
2024-03-21 20:13:15 +01:00
if ( IsRenderTarget & & lockFlags = = ETLF_FLIP_Y_UP_RTT ) {
const s32 pitch = tmpImage - > getPitch ( ) ;
u8 * srcA = static_cast < u8 * > ( 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 ;
}
2024-11-08 12:08:48 +01:00
} else {
2024-03-21 20:13:15 +01:00
GLuint tmpFBO = 0 ;
Driver - > irrGlGenFramebuffers ( 1 , & tmpFBO ) ;
GLuint prevFBO = 0 ;
Driver - > getCacheHandler ( ) - > getFBO ( prevFBO ) ;
Driver - > getCacheHandler ( ) - > setFBO ( tmpFBO ) ;
2024-11-08 12:08:48 +01:00
Driver - > irrGlFramebufferTexture2D ( GL_FRAMEBUFFER , GL_COLOR_ATTACHMENT0 , GL_TEXTURE_2D , getOpenGLTextureName ( ) , 0 ) ;
2024-03-21 20:13:15 +01:00
IImage * tmpImage = Driver - > createImage ( ECF_A8R8G8B8 , Size ) ;
GL . ReadPixels ( 0 , 0 , Size . Width , Size . Height , GL_RGBA , GL_UNSIGNED_BYTE , tmpImage - > getData ( ) ) ;
2024-11-08 12:08:48 +01:00
Driver - > irrGlFramebufferTexture2D ( GL_FRAMEBUFFER , GL_COLOR_ATTACHMENT0 , GL_TEXTURE_2D , 0 , 0 ) ;
2024-03-21 20:13:15 +01:00
Driver - > getCacheHandler ( ) - > setFBO ( prevFBO ) ;
Driver - > irrGlDeleteFramebuffers ( 1 , & tmpFBO ) ;
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 ( ) ;
2024-11-08 12:08:48 +01:00
}
2024-03-21 20:13:15 +01:00
if ( ! passed ) {
LockImage - > drop ( ) ;
LockImage = 0 ;
}
}
2024-04-12 12:18:41 +02:00
TEST_GL_ERROR ( Driver ) ;
2024-03-21 20:13:15 +01:00
}
return ( LockImage ) ? getLockImageData ( MipLevelStored ) : 0 ;
}
void unlock ( ) 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 ;
}
void regenerateMipMapLevels ( void * data = 0 , u32 layer = 0 ) 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 < u8 * > ( data ) ;
u32 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 {
Driver - > irrGlGenerateMipmap ( TextureType ) ;
2024-04-12 12:18:41 +02:00
TEST_GL_ERROR ( Driver ) ;
2024-03-21 20:13:15 +01:00
}
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 ) ;
InternalFormat = 0 ;
return ;
}
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 ;
if ( width < 1 )
width = 1 ;
if ( height < 1 )
height = 1 ;
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 < u32 > 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 )
GL . TexImage2D ( tmpTextureType , level , InternalFormat , width , height , 0 , PixelFormat , PixelType , tmpData ) ;
else
GL . TexSubImage2D ( tmpTextureType , level , 0 , 0 , width , height , PixelFormat , PixelType , tmpData ) ;
2024-04-12 12:18:41 +02:00
TEST_GL_ERROR ( Driver ) ;
2024-03-21 20:13:15 +01:00
break ;
default :
break ;
}
delete tmpImage ;
} else {
u32 dataSize = 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 ) ;
2024-04-12 12:18:41 +02:00
TEST_GL_ERROR ( Driver ) ;
2024-03-21 20:13:15 +01:00
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 * , s32 , void * ) ;
bool LockReadOnly ;
IImage * LockImage ;
u32 LockLayer ;
bool KeepImage ;
2024-08-17 19:49:11 +02:00
std : : vector < IImage * > Images ;
2024-03-21 20:13:15 +01:00
u8 MipLevelStored ;
bool LegacyAutoGenerateMipMaps ;
mutable SStatesCache StatesCache ;
} ;
}
}