2019-10-05 17:28:05 +00:00
/*
* * texture . cpp
* * The base texture class
* *
* * - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
* * Copyright 2004 - 2007 Randy Heit
* * Copyright 2006 - 2018 Christoph Oelckers
* * All rights reserved .
* *
* * Redistribution and use in source and binary forms , with or without
* * modification , are permitted provided that the following conditions
* * are met :
* *
* * 1. Redistributions of source code must retain the above copyright
* * notice , this list of conditions and the following disclaimer .
* * 2. Redistributions in binary form must reproduce the above copyright
* * notice , this list of conditions and the following disclaimer in the
* * documentation and / or other materials provided with the distribution .
* * 3. The name of the author may not be used to endorse or promote products
* * derived from this software without specific prior written permission .
* *
* * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ` ` AS IS ' ' AND ANY EXPRESS OR
* * IMPLIED WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE IMPLIED WARRANTIES
* * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED .
* * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT , INDIRECT ,
* * INCIDENTAL , SPECIAL , EXEMPLARY , OR CONSEQUENTIAL DAMAGES ( INCLUDING , BUT
* * NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES ; LOSS OF USE ,
* * DATA , OR PROFITS ; OR BUSINESS INTERRUPTION ) HOWEVER CAUSED AND ON ANY
* * THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT LIABILITY , OR TORT
* * ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY OUT OF THE USE OF
* * THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE .
* * - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
* *
* *
*/
# include "bitmap.h"
# include "image.h"
2020-04-11 21:54:33 +00:00
# include "filesystem.h"
2020-05-23 21:46:44 +00:00
# include "files.h"
# include "cmdlib.h"
# include "palettecontainer.h"
2019-10-05 17:28:05 +00:00
2020-05-25 21:59:07 +00:00
FMemArena ImageArena ( 32768 ) ;
2020-05-23 21:46:44 +00:00
TArray < FImageSource * > FImageSource : : ImageForLump ;
2019-10-05 17:28:05 +00:00
int FImageSource : : NextID ;
2020-05-23 21:46:44 +00:00
static PrecacheInfo precacheInfo ;
struct PrecacheDataPaletted
{
2023-01-07 18:30:49 +00:00
PalettedPixels Pixels ;
2020-05-23 21:46:44 +00:00
int RefCount ;
int ImageID ;
2023-08-23 18:36:19 +00:00
int Frame ;
2020-05-23 21:46:44 +00:00
} ;
struct PrecacheDataRgba
{
FBitmap Pixels ;
int TransInfo ;
int RefCount ;
int ImageID ;
2023-08-23 18:36:19 +00:00
int Frame ;
2020-05-23 21:46:44 +00:00
} ;
// TMap doesn't handle this kind of data well. std::map neither. The linear search is still faster, even for a few 100 entries because it doesn't have to access the heap as often..
TArray < PrecacheDataPaletted > precacheDataPaletted ;
TArray < PrecacheDataRgba > precacheDataRgba ;
//===========================================================================
//
// the default just returns an empty texture.
//
//===========================================================================
2023-08-23 18:36:19 +00:00
PalettedPixels FImageSource : : CreatePalettedPixels ( int conversion , int frame )
2020-05-23 21:46:44 +00:00
{
2023-01-07 18:30:49 +00:00
PalettedPixels Pixels ( Width * Height ) ;
2020-05-23 21:46:44 +00:00
memset ( Pixels . Data ( ) , 0 , Width * Height ) ;
return Pixels ;
}
2023-08-23 18:36:19 +00:00
PalettedPixels FImageSource : : GetCachedPalettedPixels ( int conversion , int frame )
2020-05-23 21:46:44 +00:00
{
PalettedPixels ret ;
auto imageID = ImageID ;
// Do we have this image in the cache?
2023-08-23 18:36:19 +00:00
unsigned index = conversion ! = normal ? UINT_MAX : precacheDataPaletted . FindEx ( [ = ] ( PrecacheDataPaletted & entry ) { return entry . ImageID = = imageID & & entry . Frame = = frame ; } ) ;
2020-05-23 21:46:44 +00:00
if ( index < precacheDataPaletted . Size ( ) )
{
auto cache = & precacheDataPaletted [ index ] ;
if ( cache - > RefCount > 1 )
{
//Printf("returning reference to %s, refcount = %d\n", name.GetChars(), cache->RefCount);
ret . Pixels . Set ( cache - > Pixels . Data ( ) , cache - > Pixels . Size ( ) ) ;
cache - > RefCount - - ;
}
else if ( cache - > Pixels . Size ( ) > 0 )
{
//Printf("returning contents of %s, refcount = %d\n", name.GetChars(), cache->RefCount);
2023-01-07 18:30:49 +00:00
ret = std : : move ( cache - > Pixels ) ;
2020-05-23 21:46:44 +00:00
precacheDataPaletted . Delete ( index ) ;
}
else
{
//Printf("something bad happened for %s, refcount = %d\n", name.GetChars(), cache->RefCount);
}
}
else
{
// The image wasn't cached. Now there's two possibilities:
auto info = precacheInfo . CheckKey ( ImageID ) ;
if ( ! info | | info - > second < = 1 | | conversion ! = normal )
{
// This is either the only copy needed or some access outside the caching block. In these cases create a new one and directly return it.
//Printf("returning fresh copy of %s\n", name.GetChars());
2023-08-23 18:36:19 +00:00
return CreatePalettedPixels ( conversion , frame ) ;
2020-05-23 21:46:44 +00:00
}
else
{
//Printf("creating cached entry for %s, refcount = %d\n", name.GetChars(), info->second);
// This is the first time it gets accessed and needs to be placed in the cache.
PrecacheDataPaletted * pdp = & precacheDataPaletted [ precacheDataPaletted . Reserve ( 1 ) ] ;
pdp - > ImageID = imageID ;
pdp - > RefCount = info - > second - 1 ;
info - > second = 0 ;
2023-08-23 18:36:19 +00:00
pdp - > Pixels = CreatePalettedPixels ( normal , frame ) ;
2020-05-23 21:46:44 +00:00
ret . Pixels . Set ( pdp - > Pixels . Data ( ) , pdp - > Pixels . Size ( ) ) ;
}
}
return ret ;
}
2023-08-23 18:36:19 +00:00
TArray < uint8_t > FImageSource : : GetPalettedPixels ( int conversion , int frame )
2020-05-23 21:46:44 +00:00
{
2023-08-23 18:36:19 +00:00
auto pix = GetCachedPalettedPixels ( conversion , frame ) ;
2020-05-23 21:46:44 +00:00
if ( pix . ownsPixels ( ) )
{
// return the pixel store of the returned data directly if this was the last reference.
auto array = std : : move ( pix . PixelStore ) ;
return array ;
}
else
{
// If there are pending references, make a copy.
TArray < uint8_t > array ( pix . Pixels . Size ( ) , true ) ;
memcpy ( array . Data ( ) , pix . Pixels . Data ( ) , array . Size ( ) ) ;
return array ;
}
}
//===========================================================================
//
// FImageSource::CopyPixels
//
// this is the generic case that can handle
// any properly implemented texture for software rendering.
// Its drawback is that it is limited to the base palette which is
// why all classes that handle different palettes should subclass this
// method
//
//===========================================================================
2023-08-23 18:36:19 +00:00
int FImageSource : : CopyPixels ( FBitmap * bmp , int conversion , int frame )
2020-05-23 21:46:44 +00:00
{
if ( conversion = = luminance ) conversion = normal ; // luminance images have no use as an RGB source.
PalEntry * palette = GPalette . BaseColors ;
2023-01-07 18:30:49 +00:00
2023-08-23 18:36:19 +00:00
auto ppix = CreatePalettedPixels ( conversion , frame ) ;
2020-05-23 21:46:44 +00:00
bmp - > CopyPixelData ( 0 , 0 , ppix . Data ( ) , Width , Height , Height , 1 , 0 , palette , nullptr ) ;
return 0 ;
}
2023-08-23 18:36:19 +00:00
int FImageSource : : CopyTranslatedPixels ( FBitmap * bmp , const PalEntry * remap , int frame )
2020-05-23 21:46:44 +00:00
{
2023-08-23 18:36:19 +00:00
auto ppix = CreatePalettedPixels ( normal , frame ) ;
2020-05-23 21:46:44 +00:00
bmp - > CopyPixelData ( 0 , 0 , ppix . Data ( ) , Width , Height , Height , 1 , 0 , remap , nullptr ) ;
return 0 ;
}
2019-10-05 17:28:05 +00:00
//==========================================================================
//
//
//
//==========================================================================
2023-08-23 18:36:19 +00:00
FBitmap FImageSource : : GetCachedBitmap ( const PalEntry * remap , int conversion , int * ptrans , int frame )
2020-05-23 21:46:44 +00:00
{
FBitmap ret ;
2021-12-30 09:30:21 +00:00
2022-12-08 11:44:06 +00:00
int trans = - 1 ;
2020-05-23 21:46:44 +00:00
auto imageID = ImageID ;
2021-12-30 09:30:21 +00:00
2023-08-23 18:36:19 +00:00
if ( NumOfFrames = = 1 & & frame = = 1 )
{
frame = 0 ;
}
2020-05-23 21:46:44 +00:00
if ( remap ! = nullptr )
{
// Remapped images are never run through the cache because they would complicate matters too much for very little gain.
// Translated images are normally sprites which normally just consist of a single image and use no composition.
// Additionally, since translation requires the base palette, the really time consuming stuff will never be subjected to it.
ret . Create ( Width , Height ) ;
2023-08-23 18:36:19 +00:00
trans = CopyTranslatedPixels ( & ret , remap , frame ) ;
2020-05-23 21:46:44 +00:00
}
else
{
if ( conversion = = luminance ) conversion = normal ; // luminance has no meaning for true color.
// Do we have this image in the cache?
2023-08-23 18:36:19 +00:00
unsigned index = conversion ! = normal ? UINT_MAX : precacheDataRgba . FindEx ( [ = ] ( PrecacheDataRgba & entry ) { return entry . ImageID = = imageID & & entry . Frame = = frame ; } ) ;
2020-05-23 21:46:44 +00:00
if ( index < precacheDataRgba . Size ( ) )
{
auto cache = & precacheDataRgba [ index ] ;
2021-12-30 09:30:21 +00:00
2020-05-23 21:46:44 +00:00
trans = cache - > TransInfo ;
if ( cache - > RefCount > 1 )
{
//Printf("returning reference to %s, refcount = %d\n", name.GetChars(), cache->RefCount);
ret . Copy ( cache - > Pixels , false ) ;
cache - > RefCount - - ;
}
else if ( cache - > Pixels . GetPixels ( ) )
{
//Printf("returning contents of %s, refcount = %d\n", name.GetChars(), cache->RefCount);
ret = std : : move ( cache - > Pixels ) ;
precacheDataRgba . Delete ( index ) ;
}
else
{
// This should never happen if the function is implemented correctly
//Printf("something bad happened for %s, refcount = %d\n", name.GetChars(), cache->RefCount);
ret . Create ( Width , Height ) ;
2023-08-23 18:36:19 +00:00
trans = CopyPixels ( & ret , normal , frame ) ;
2020-05-23 21:46:44 +00:00
}
}
else
{
// The image wasn't cached. Now there's two possibilities:
auto info = precacheInfo . CheckKey ( ImageID ) ;
if ( ! info | | info - > first < = 1 | | conversion ! = normal )
{
// This is either the only copy needed or some access outside the caching block. In these cases create a new one and directly return it.
//Printf("returning fresh copy of %s\n", name.GetChars());
ret . Create ( Width , Height ) ;
2023-08-23 18:36:19 +00:00
trans = CopyPixels ( & ret , conversion , frame ) ;
2020-05-23 21:46:44 +00:00
}
else
{
//Printf("creating cached entry for %s, refcount = %d\n", name.GetChars(), info->first);
// This is the first time it gets accessed and needs to be placed in the cache.
PrecacheDataRgba * pdr = & precacheDataRgba [ precacheDataRgba . Reserve ( 1 ) ] ;
2021-12-30 09:30:21 +00:00
2020-05-23 21:46:44 +00:00
pdr - > ImageID = imageID ;
2023-08-23 18:36:19 +00:00
pdr - > Frame = frame ;
2020-05-23 21:46:44 +00:00
pdr - > RefCount = info - > first - 1 ;
info - > first = 0 ;
pdr - > Pixels . Create ( Width , Height ) ;
2023-08-23 18:36:19 +00:00
trans = pdr - > TransInfo = CopyPixels ( & pdr - > Pixels , normal , frame ) ;
2020-05-23 21:46:44 +00:00
ret . Copy ( pdr - > Pixels , false ) ;
}
}
}
if ( ptrans ) * ptrans = trans ;
return ret ;
}
//==========================================================================
//
//
//
//==========================================================================
void FImageSource : : CollectForPrecache ( PrecacheInfo & info , bool requiretruecolor )
{
auto val = info . CheckKey ( ImageID ) ;
bool tc = requiretruecolor ;
if ( val )
{
val - > first + = tc ;
val - > second + = ! tc ;
}
else
{
auto pair = std : : make_pair ( tc , ! tc ) ;
info . Insert ( ImageID , pair ) ;
}
}
void FImageSource : : BeginPrecaching ( )
{
precacheInfo . Clear ( ) ;
}
void FImageSource : : EndPrecaching ( )
{
precacheDataPaletted . Clear ( ) ;
precacheDataRgba . Clear ( ) ;
}
void FImageSource : : RegisterForPrecache ( FImageSource * img , bool requiretruecolor )
{
img - > CollectForPrecache ( precacheInfo , requiretruecolor ) ;
}
//==========================================================================
//
//
//
//==========================================================================
typedef FImageSource * ( * CreateFunc ) ( FileReader & file , int lumpnum ) ;
2019-10-05 17:28:05 +00:00
struct TexCreateInfo
{
CreateFunc TryCreate ;
2020-05-23 21:46:44 +00:00
bool checkflat ;
2019-10-05 17:28:05 +00:00
} ;
2020-05-23 22:15:38 +00:00
FImageSource * IMGZImage_TryCreate ( FileReader & , int lumpnum ) ;
2020-05-23 21:46:44 +00:00
FImageSource * PNGImage_TryCreate ( FileReader & , int lumpnum ) ;
FImageSource * JPEGImage_TryCreate ( FileReader & , int lumpnum ) ;
FImageSource * DDSImage_TryCreate ( FileReader & , int lumpnum ) ;
FImageSource * PCXImage_TryCreate ( FileReader & , int lumpnum ) ;
FImageSource * TGAImage_TryCreate ( FileReader & , int lumpnum ) ;
FImageSource * StbImage_TryCreate ( FileReader & , int lumpnum ) ;
2023-08-19 14:57:37 +00:00
FImageSource * QOIImage_TryCreate ( FileReader & , int lumpnum ) ;
2023-09-02 07:13:41 +00:00
FImageSource * WebPImage_TryCreate ( FileReader & , int lumpnum ) ;
2020-06-28 12:41:44 +00:00
FImageSource * AnmImage_TryCreate ( FileReader & , int lumpnum ) ;
2020-05-23 22:15:38 +00:00
FImageSource * RawPageImage_TryCreate ( FileReader & , int lumpnum ) ;
FImageSource * FlatImage_TryCreate ( FileReader & , int lumpnum ) ;
FImageSource * PatchImage_TryCreate ( FileReader & , int lumpnum ) ;
FImageSource * EmptyImage_TryCreate ( FileReader & , int lumpnum ) ;
FImageSource * AutomapImage_TryCreate ( FileReader & , int lumpnum ) ;
2022-06-06 09:45:02 +00:00
FImageSource * StartupPageImage_TryCreate ( FileReader & , int lumpnum ) ;
2019-10-05 17:28:05 +00:00
// Examines the lump contents to decide what type of texture to create,
// and creates the texture.
2020-05-23 21:46:44 +00:00
FImageSource * FImageSource : : GetImage ( int lumpnum , bool isflat )
2019-10-05 17:28:05 +00:00
{
static TexCreateInfo CreateInfo [ ] = {
2020-05-23 22:15:38 +00:00
{ IMGZImage_TryCreate , false } ,
2020-05-23 21:46:44 +00:00
{ PNGImage_TryCreate , false } ,
{ JPEGImage_TryCreate , false } ,
{ DDSImage_TryCreate , false } ,
{ PCXImage_TryCreate , false } ,
{ StbImage_TryCreate , false } ,
2023-08-19 14:57:37 +00:00
{ QOIImage_TryCreate , false } ,
2023-09-02 07:13:41 +00:00
{ WebPImage_TryCreate , false } ,
2020-05-23 21:46:44 +00:00
{ TGAImage_TryCreate , false } ,
2020-06-28 12:41:44 +00:00
{ AnmImage_TryCreate , false } ,
2022-06-06 09:45:02 +00:00
{ StartupPageImage_TryCreate , false } ,
2020-05-23 22:15:38 +00:00
{ RawPageImage_TryCreate , false } ,
{ FlatImage_TryCreate , true } , // flat detection is not reliable, so only consider this for real flats.
{ PatchImage_TryCreate , false } ,
{ EmptyImage_TryCreate , false } ,
{ AutomapImage_TryCreate , false } ,
2019-10-05 17:28:05 +00:00
} ;
2020-05-23 21:46:44 +00:00
if ( lumpnum = = - 1 ) return nullptr ;
unsigned size = ImageForLump . Size ( ) ;
if ( size < = ( unsigned ) lumpnum )
{
// Hires textures can be added dynamically to the end of the lump array, so this must be checked each time.
ImageForLump . Resize ( lumpnum + 1 ) ;
for ( ; size < ImageForLump . Size ( ) ; size + + ) ImageForLump [ size ] = nullptr ;
}
// An image for this lump already exists. We do not need another one.
if ( ImageForLump [ lumpnum ] ! = nullptr ) return ImageForLump [ lumpnum ] ;
auto data = fileSystem . OpenFileReader ( lumpnum ) ;
if ( ! data . isOpen ( ) )
return nullptr ;
2019-10-05 17:28:05 +00:00
2020-05-23 21:46:44 +00:00
for ( size_t i = 0 ; i < countof ( CreateInfo ) ; i + + )
2019-10-05 17:28:05 +00:00
{
2020-05-23 21:46:44 +00:00
if ( ! CreateInfo [ i ] . checkflat | | isflat )
2019-10-05 17:28:05 +00:00
{
2020-05-23 21:46:44 +00:00
auto image = CreateInfo [ i ] . TryCreate ( data , lumpnum ) ;
if ( image ! = nullptr )
{
ImageForLump [ lumpnum ] = image ;
return image ;
}
2019-10-05 17:28:05 +00:00
}
}
return nullptr ;
}