csgo-2018-source/materialsystem/cmaterialsystem_ps3fonts.cpp
2021-07-24 21:11:47 -07:00

799 lines
22 KiB
C++

//===== Copyright (c) Valve Corporation, All rights reserved. ======//
//
// Purpose: PS3-specific font code that for historical reasons is hidden
// in the materialsystem.
//
//==================================================================//
#include "pch_materialsystem.h"
#ifndef _PS3
#define MATSYS_INTERNAL
#endif
#include "cmaterialsystem.h"
#include "colorspace.h"
#include "materialsystem/materialsystem_config.h"
#include "materialsystem/imaterialproxyfactory.h"
#include "IHardwareConfigInternal.h"
#include "shadersystem.h"
#include "texturemanager.h"
#include "shaderlib/ShaderDLL.h"
#include "tier1/callqueue.h"
#include "vstdlib/jobthread.h"
#include "cmatnullrendercontext.h"
#include "datacache/iresourceaccesscontrol.h"
#include "filesystem/IQueuedLoader.h"
#include "filesystem/IXboxInstaller.h"
#include "cdll_int.h"
#include <cell/sysmodule.h>
#include <cell/font.h>
#include <cell/fontft.h>
#include "../common/vgui_surfacelib/vguifont.h"
#ifdef _PS3
#include "ps3_pathinfo.h"
#endif
// NOTE: This must be the last file included!!!
#include "tier0/memdbgon.h"
static const CellFontLibrary *Fontlib();
// in case I make HPS3FONT something other than just a pointer
static inline CellFont *CellFontFromHandle( HPS3FONT &h )
{
return (CellFont *)h;
}
wchar_t g_wchFallbackGlyph = L'_';
static inline bool ShouldFallBackToSystemFontForWChar( wchar_t wch, int &nFontTallAdjustment )
{
if ( ( wch == 0x00AE /*(R)*/ ) || ( wch == 0x2122 /*(TM)*/ ) )
{
nFontTallAdjustment = 1;
return true;
}
return false;
}
#define TEXTURE_PAGE_HEIGHT 128 // copied from FontTextureCache.cpp, which for some reason hardcodes this. because there's multiple implementations
// of the font cache strewn throughout various libraries all linked into the vgui dll, I didn't feel safe changing
// it to just use larger textures.
class CFallbackFont: public CellFont
{
protected:
int m_nInitError;
int m_nTall;
CellFontHorizontalLayout m_hlayout;
public:
CFallbackFont(): m_nInitError(-1), m_nTall( -1 )
{
}
void Init()
{
if( m_nInitError == CELL_OK )
{
// already initialized
return ;
}
CellFontType fontType;
fontType.type = CELL_FONT_TYPE_GOTHIC_JAPANESE_CJK_LATIN_SET;
fontType.map = CELL_FONT_MAP_UNICODE;
m_nInitError = cellFontOpenFontset( Fontlib(), &fontType, this );
if( m_nInitError == CELL_OK )
{
Warning("Cannot open fallback font, error 0x%X\n", m_nInitError );
}
}
CellFontHorizontalLayout *GetHorizontalLayout()
{
return &m_hlayout;
}
int SetTall( int nTall )
{
if( m_nInitError != CELL_OK )
{
return m_nInitError;
}
if( nTall == m_nTall )
{
return CELL_OK;
}
float pointSize = nTall * 0.79f;
cellFontSetScalePixel( this, pointSize, pointSize );
// sometimes we ask for bigger fonts than the packer can handle. Also, because of accents etc, the font may poke out of the requested tall value.
// The packer is for some reason hardcoded to a maximum height of 128.
cellFontGetHorizontalLayout( this, &m_hlayout );
if ( m_hlayout.lineHeight + m_hlayout.effectHeight > TEXTURE_PAGE_HEIGHT )
{
Warning("FAILSAFE: Fallback Font at '%d' tall is actually %d pixels high, but max texture page is %d. Shrinking to ",
nTall, (int)(m_hlayout.lineHeight + m_hlayout.effectHeight), TEXTURE_PAGE_HEIGHT );
do
{
pointSize *= 0.9;
cellFontSetScalePixel( this, nTall, nTall );
cellFontGetHorizontalLayout( this, &m_hlayout );
}
while ( ( m_hlayout.lineHeight + m_hlayout.effectHeight > TEXTURE_PAGE_HEIGHT ) && ( pointSize >= 1 ) );
Warning( "%.2f\n", pointSize );
}
m_nTall = nTall;
return CELL_OK;
}
void Shutdown()
{
if( m_nInitError == CELL_OK )
{
int nError = cellFontCloseFont( this );
if( nError != CELL_OK )
{
Warning("Cannot close fallback font, error 0x%X\n", nError );
}
}
}
};
CFallbackFont g_fallbackFont;
void CMaterialSystem::InitializePS3Fonts()
{
m_vExtantFonts.EnsureCapacity(8);
}
class CTempFontHandle
{
public:
CMaterialSystem * m_pMaterialSystem;
MaterialLock_t m_hLock;
CellFont * m_pNewFont;
public:
CTempFontHandle( CMaterialSystem * pMaterialSystem):
m_pMaterialSystem( pMaterialSystem )
{
m_hLock = pMaterialSystem->Lock();
m_pNewFont = new CellFont;
}
CellFont* DetachFont()
{
CellFont * pFont = m_pNewFont;
m_pNewFont = NULL;
return pFont;
}
~CTempFontHandle()
{
if( m_pNewFont )
{
delete m_pNewFont;
}
m_pMaterialSystem->Unlock( m_hLock );
}
};
HPS3FONT CMaterialSystem::OpenTrueTypeFont( const char *pFontname, int tall, int style )
{
CTempFontHandle hTemp( this );
static int sUniquePositiveNumber = 0;
// need absolute disk path, look in resource then platform
char fontpath[MAX_PATH] = {0};
struct fontTable_t
{
const char *pFontName;
const char *pPath;
bool m_bRestrictiveLoadIntoMemory;
bool m_bAlwaysLoadIntoMemory;
};
int nCellError;
if( !V_stricmp( pFontname, "system" ) )
{
CellFontType fontType;
fontType.type = CELL_FONT_TYPE_GOTHIC_JAPANESE_CJK_LATIN_SET;
fontType.map = CELL_FONT_MAP_UNICODE;
nCellError = cellFontOpenFontset( Fontlib(), &fontType, hTemp.m_pNewFont );
}
else if( !V_stricmp( pFontname, "japanese cjk latin" ) )
{
CellFontType fontType;
fontType.type = CELL_FONT_TYPE_GOTHIC_JAPANESE_CJK_LATIN_SET;
fontType.map = CELL_FONT_MAP_UNICODE;
nCellError = cellFontOpenFontset( Fontlib(), &fontType, hTemp.m_pNewFont );
}
else
{
// explicit mapping required, dvd searching too expensive
static fontTable_t fontToFilename[] =
{
#include "shaderapidx9/fontremaptable.h"
};
// remap typeface to diskname
char const *pDiskname = NULL;
for ( int i=0; i<ARRAYSIZE( fontToFilename ); i++ )
{
if ( !V_stricmp( pFontname, fontToFilename[i].pFontName ) )
{
pDiskname = fontToFilename[i].pPath;
break;
}
}
if ( !pDiskname )
{
// not found
DevWarning( "PS3: True Type Font: '%s' unknown.\n", pFontname );
}
else
{
Q_snprintf( fontpath, ARRAYSIZE( fontpath ), "%s/%s", g_pPS3PathInfo->GameImagePath(), pDiskname );
}
// if we failed to find the font, substitute a system font
if ( fontpath[0] == 0 )
{
CellFontType fontType;
fontType.type = CELL_FONT_TYPE_GOTHIC_JAPANESE_CJK_LATIN_SET;
fontType.map = CELL_FONT_MAP_UNICODE;
nCellError = cellFontOpenFontset( Fontlib(), &fontType, hTemp.m_pNewFont );
}
else
{
nCellError = cellFontOpenFontFile(
Fontlib(),
reinterpret_cast<uint8_t *>(fontpath),
0,
++sUniquePositiveNumber,
hTemp.m_pNewFont );
}
}
CellFont *pNewFont = NULL;
if ( nCellError != CELL_FONT_OK )
{
Warning( "could not open font '%s': %x\n", pFontname, nCellError );
return false;
}
else
{
pNewFont = hTemp.DetachFont(); // do not release the font structure
m_vExtantFonts.AddToTail( pNewFont );
}
float pointSize = tall * 0.79f;
cellFontSetScalePixel( pNewFont, pointSize, pointSize );
// sometimes we ask for bigger fonts than the packer can handle. Also, because of accents etc, the font may poke out of the requested tall value.
// The packer is for some reason hardcoded to a maximum height of 128.
CellFontHorizontalLayout hlayout;
cellFontGetHorizontalLayout( pNewFont, &hlayout );
if ( hlayout.lineHeight + hlayout.effectHeight > TEXTURE_PAGE_HEIGHT )
{
Warning("FAILSAFE: %s at '%d' tall is actually %d pixels high, but max texture page is %d. Shrinking to ",
pFontname, tall, (int)(hlayout.lineHeight + hlayout.effectHeight), TEXTURE_PAGE_HEIGHT );
do
{
pointSize *= 0.9;
cellFontSetScalePixel( pNewFont, tall, tall );
cellFontGetHorizontalLayout( pNewFont, &hlayout );
}
while ( ( hlayout.lineHeight + hlayout.effectHeight > TEXTURE_PAGE_HEIGHT ) && (pointSize >= 1) );
Warning( "%.2f\n", pointSize );
}
/*
nCellError = cellFontGetHorizontalLayout( &m_sFont, &m_HorizontalLayout );
AssertMsg2( nCellError == CELL_OK, "Could not get horizontal layout for font %s (%x)\n", m_szName.String(), nCellError );
*/
/*
nCellError = cellFontGetVerticalLayout( &m_sFont, &m_VerticalLayout );
AssertMsg2( nCellError == CELL_OK, "Could not get vertical layout for font %s (%x)\n", m_szName.String(), nCellError );
*/
return pNewFont;
}
void CMaterialSystem::CloseTrueTypeFont( HPS3FONT hFont )
{
MaterialLock_t hLock = Lock();
// find the font in the vector
CellFont *pFont = CellFontFromHandle( hFont );
cellFontCloseFont( pFont );
if ( m_vExtantFonts.FindAndFastRemove( pFont ) )
{
delete pFont;
}
else
{
AssertMsg1( false, "Tried to remove a font %x which wasn't in the material system's library\n", hFont );
}
Unlock( hLock );
}
bool CMaterialSystem::GetTrueTypeFontMetrics( HPS3FONT hFont, int nFallbackTall, wchar_t wchFirst, wchar_t wchLast, CPS3FontMetrics * RESTRICT pFontMetrics, CPS3CharMetrics * RESTRICT pCharMetrics )
{
MaterialLock_t hLock = Lock();
int numChars = wchLast - wchFirst + 1;
V_memset( pCharMetrics, 0, numChars * sizeof( CPS3CharMetrics ) );
CellFont * RESTRICT pFont = CellFontFromHandle( hFont );
int nCellError = cellFontGetHorizontalLayout( pFont, pFontMetrics );
AssertMsg1( nCellError == CELL_OK, "Could not get horizontal layout for font (%x)\n", nCellError );
// we will still need to go and fill in the max width once we've got the data for all the characters
float fMaxWidth = 0;
for ( int i = 0; i < numChars; i++ )
{
int nMetricsError = CELL_FONT_ERROR_FATAL;
int nFallbackFontTallAdjustment = 0;
if ( !ShouldFallBackToSystemFontForWChar( wchFirst + i, nFallbackFontTallAdjustment ) )
{
nMetricsError = cellFontGetCharGlyphMetrics( pFont, wchFirst + i, pCharMetrics + i );
}
if( nMetricsError != CELL_OK )
{
// try the fallback font
int nFallbackError = g_fallbackFont.SetTall( nFallbackTall + nFallbackFontTallAdjustment );
if( nFallbackError == CELL_OK )
{
nMetricsError = cellFontGetCharGlyphMetrics( &g_fallbackFont, wchFirst + i, pCharMetrics + i );
}
}
if( nMetricsError != CELL_OK )
{
nMetricsError = cellFontGetCharGlyphMetrics( pFont, g_wchFallbackGlyph, pCharMetrics + i );
}
if ( nMetricsError == CELL_OK )
{
fMaxWidth = fmax( fMaxWidth, pCharMetrics[i].Horizontal.advance + 1 );
}
else
{
memset( pCharMetrics+i, 0, sizeof(*pCharMetrics) );
}
}
pFontMetrics->fMaxWidth = fMaxWidth;
Unlock( hLock );
return nCellError == CELL_OK;
}
// encapsulates the ps3 fontlib renderer class with a RAII interface
// so it binds/unbinds automatically
class CFontRendererRAII
{
public:
CFontRendererRAII( CellFont *pBoundFont );
~CFontRendererRAII();
bool IsValid() { return m_nInitSuccess == CELL_FONT_OK; }
protected:
CellFontRenderer m_Renderer;
CellFont *m_pFont;
int m_nInitSuccess;
};
CFontRendererRAII::CFontRendererRAII( CellFont *pBoundFont ) : m_pFont(NULL)
{
CellFontRendererConfig config;
CellFontRendererConfig_initialize( &config );
// proper setup here
m_nInitSuccess = cellFontCreateRenderer( Fontlib(), &config, &m_Renderer );
if ( m_nInitSuccess != CELL_FONT_OK )
{
return ; // we have failed. we are failures. the stench of failure hangs upon us.
}
// connect the renderer to the font
m_nInitSuccess = cellFontBindRenderer( pBoundFont, &m_Renderer );
if ( m_nInitSuccess != CELL_FONT_OK )
{
return ; // we have failed. we are failures. the stench of failure hangs upon us.
}
m_pFont = pBoundFont;
}
CFontRendererRAII::~CFontRendererRAII()
{
if ( m_pFont )
{
cellFontUnbindRenderer( m_pFont );
}
cellFontDestroyRenderer( &m_Renderer );
}
// OPTIMIZATION OPPORTUNITY:
// This can be made to run on the RSX instead of the PPU, if necessary.
bool CMaterialSystem::GetTrueTypeGlyphs( HPS3FONT hFont, int nFallbackTall, int numChars, wchar_t *pWch, int *pOffsetX, int *pOffsetY, int * RESTRICT pWidth, int * RESTRICT pHeight, unsigned char *pRGBA, int *pRGBAOffset )
{
VPROF( "CMaterialSystem::GetTrueTypeGlyphs" );
MaterialLock_t hLock = Lock();
// Figure out the size of surface/texture we need to allocate
int rtWidth = 0;
int rtHeight = 0;
for ( int i = 0; i < numChars; i++ )
{
//rtWidth += pWidth[i];
rtWidth = MAX( rtWidth, pWidth[i] );
rtHeight = MAX( rtHeight, pHeight[i] );
}
/*
// per resolve() restrictions
rtWidth = AlignValue( rtWidth, 32 );
rtHeight = AlignValue( rtHeight, 32 );
*/
if ( rtWidth == 0 || rtHeight == 0 )
{
AssertMsg2( false, "Tried to draw a %dx%d glyph! What are you trying to pull?!\n", rtWidth, rtHeight );
Unlock( hLock );
return false;
}
CellFont *pFont = CellFontFromHandle( hFont );
CFontRendererRAII renderer( pFont ), *pFallbackRenderer = NULL;
CellFontRenderSurface surface;
CellFontGlyphMetrics metrics;
CellFontImageTransInfo transInfo;
CellFontHorizontalLayout hlayoutSlowWeShouldMoveThisIntoTheFontClass;
cellFontRenderSurfaceInit( &surface, NULL, rtWidth, 1, rtWidth, rtHeight ); // the surface doesn't actually use the memory pointer (go figure)
cellFontRenderSurfaceSetScissor( &surface, 0, 0, rtWidth, rtHeight );
cellFontGetHorizontalLayout( pFont, &hlayoutSlowWeShouldMoveThisIntoTheFontClass );
const int baselineY = floor(hlayoutSlowWeShouldMoveThisIntoTheFontClass.baseLineY);
// struct XUIRect { int x, int y, int w, int h };
// Draw the characters, stepping across the texture
// int xCursor = 0;
unsigned int successfulWrites = 0;
for ( int i = 0; i < numChars; i++)
{
/*
// FIXME: the drawRect params don't make much sense (should use "(xCursor+pWidth[i]), pHeight[i]", but then some characters disappear!)
XUIRect drawRect = XUIRect( xCursor + pOffsetX[i], pOffsetY[i], rtWidth, rtHeight );
cellFontRenderSurfaceSetScissor( &surface, drawRect.x, drawRect.y, drawRect.w, drawRect.h );
wchar_t text[2] = { pWch[i], 0 };
XuiDrawText( m_hDC, text, XUI_FONT_STYLE_NORMAL|XUI_FONT_STYLE_SINGLE_LINE|XUI_FONT_STYLE_NO_WORDWRAP, 0, &drawRect );
xCursor += pWidth[i];
*/
// draw one glyph to the surface
int ps3suc = CELL_FONT_ERROR_FATAL;
int nFallbackFontTallAdjustment = 0;
if ( !ShouldFallBackToSystemFontForWChar( pWch[i], nFallbackFontTallAdjustment ) )
{
ps3suc = cellFontRenderCharGlyphImage( pFont, pWch[i], &surface, 0, 0, &metrics, &transInfo );
}
if ( ps3suc != CELL_OK )
{
if( CELL_OK == g_fallbackFont.SetTall( nFallbackTall + nFallbackFontTallAdjustment ) )
{
if( !pFallbackRenderer )
pFallbackRenderer = new CFontRendererRAII( &g_fallbackFont );
ps3suc = cellFontRenderCharGlyphImage( &g_fallbackFont, pWch[i], &surface, 0, 0, &metrics, &transInfo );
}
}
if( ps3suc != CELL_OK )
{
ps3suc = cellFontRenderCharGlyphImage( pFont, g_wchFallbackGlyph, &surface, 0, 0, &metrics, &transInfo );
}
if ( ps3suc == CELL_OK )
{
// blit the glyph to the bitmap
const int ibw = transInfo.imageWidthByte;
const int iBearingY = ceilf(metrics.Horizontal.bearingY); // start the conversion early to avoid a LHS
int destPitch = pWidth[i]*4; // distance between one row and the next IN BYTES
unsigned char *pLinear = pRGBA + pRGBAOffset[i];
// Clear out all pixels to black, since the source glyph size may not match the destination buffer size perfectly
memset( pLinear, 0, pHeight[i] * destPitch );
if ( transInfo.imageHeight==0 || transInfo.imageWidth==0 )
{
// this can happen for space or control characters
// just blank out the destination box
continue; // bail out and go to next character !
}
// work out the size of the region to copy -- which is the smaller of the source or destination buffer.
// in theory these two should be identical, but the PS3 OS tends to overallocate space slightly
// in its render surface, and sometimes the VGUI font code is expecting a blank column of pixels
// where the PS3 generates none.
// also we'll try to align the font glyph to the bottom of the destination rect, rather than
// the top.
int numLeftPadPixels = ( int )floorf( metrics.Horizontal.bearingX );
numLeftPadPixels = MAX( 0, numLeftPadPixels );
int iteratorMaxColumn = MIN( pWidth[i] - numLeftPadPixels, transInfo.imageWidth );
int iteratorMaxRow = MIN( pHeight[i], transInfo.imageHeight );
unsigned char *pLinearAligned = pLinear; // work out where to start writing into the image so that chars are aligned to bottom of rect
pLinearAligned = pLinear + MAX(0, baselineY-iBearingY)*destPitch;
// for ( int row = 0 ; row < transInfo.imageHeight ; ++row )
for ( int row = 0 ; row < iteratorMaxRow; ++row )
{
uint32 * pRowStart = reinterpret_cast<uint32 *>(pLinearAligned + (row*destPitch)); // we will write the RGBA one word at a time
for ( int col = 0 ; col < iteratorMaxColumn; ++col )
{
// white characters; only the alpha mask is copied from the rendered glyph
// exception: totally absent regions are marked as just black
uint8 a = transInfo.Image[ row * ibw + col ];
*(pRowStart + col + numLeftPadPixels) = a ? 0xFFFFFF00 | a : 0 ;
}
}
#if 0
if ( FILE *f = fopen( "/app_home/dbgfonts.txt", "a+" ) )
{
fprintf( f, "% 4d %C [%dx%d]\n",
pWch[i], pWch[i],
transInfo.imageWidth, transInfo.imageHeight
);
fprintf( f, " [%d]\n",
pWidth[i]
);
fprintf( f, " %0.5f %0.5f %0.5f\n",
metrics.Horizontal.bearingX, metrics.width, metrics.Horizontal.advance
);
for ( int row = 0; row < MAX( MAX( pHeight[i], ceilf( metrics.height ) ), transInfo.imageHeight ); ++ row )
{
for ( int col = 0; col < MAX( pWidth[i], ceilf( metrics.Horizontal.advance ) ); ++ col )
{
uint8 a = transInfo.Image[ row * ibw + col ];
fprintf( f, a ? ( (a>0x20) ? "X" : "*" ) : " " );
}
fprintf( f, "|\n" );
}
fprintf( f, "-----------------------------\n" );
fclose(f);
}
#endif
#if 0 // draw a line around the outer border for debugging
for ( int row = 0 ; row < pHeight[i]; ++row )
{
*(reinterpret_cast<uint32 *>(pLinear + (row*destPitch))+0) = 0xFFFFFFFF;
*(reinterpret_cast<uint32 *>(pLinear + (row*destPitch))+pWidth[i]-1) = 0xFFFFFFFF;
}
// top and bottom row
memset( pLinear, 0xFF, pWidth[i] * 4 );
memset( (pLinear + ((pHeight[i]-1)*destPitch)), 0xFF, pWidth[i] * 4 );
#endif
successfulWrites++;
}
else
{
AssertMsg2( false, "Couldn't draw u'%x', err %x\n", (unsigned int) *(pWch+i), ps3suc );
}
}
if( pFallbackRenderer )
delete pFallbackRenderer;
Unlock( hLock );
return successfulWrites > 0;
}
static int load_libfont_module()
{
int ret;
ret = cellSysmoduleLoadModule( CELL_SYSMODULE_FONT );
if ( ret == CELL_OK ) {
ret = cellSysmoduleLoadModule( CELL_SYSMODULE_FREETYPE_TT );
if ( ret == CELL_OK ) {
ret = cellSysmoduleLoadModule( CELL_SYSMODULE_FONTFT );
if ( ret == CELL_OK ) {
return CELL_OK; // Success
}
// Error handling as follows (Unload all loads)
cellSysmoduleUnloadModule( CELL_SYSMODULE_FREETYPE_TT );
}
cellSysmoduleUnloadModule( CELL_SYSMODULE_FONT );
}
return ret; // Error end
}
static void* fonts_malloc( void*obj, uint32_t size )
{
(void)obj;
return malloc( size );
}
static void fonts_free( void*obj, void*p )
{
(void)obj;
free( p );
}
static void* fonts_realloc( void*obj, void* p, uint32_t size )
{
(void)obj;
return realloc( p, size );
}
static void* fonts_calloc( void*obj, uint32_t numb, uint32_t blockSize )
{
(void)obj;
return calloc( numb, blockSize );
}
static int s_FontLibraryRefCount = 0;
static uint32 *s_FontFileCache = NULL;
static const CellFontLibrary *s_FontLibrary = NULL;
static CellFontEntry *s_UserFontEntries = NULL;
bool CMaterialSystem::PS3InitFontLibrary( unsigned fontFileCacheSizeInBytes, unsigned maxNumFonts )
{
if ( s_FontLibraryRefCount > 0 )
{
// don't need to load the font library, just add one to refcount
++s_FontLibraryRefCount;
return true;
}
else if ( s_FontLibraryRefCount < 0 )
{
Error( "Font library refcount is %d!\n", s_FontLibraryRefCount );
}
else
{
++s_FontLibraryRefCount;
}
VPROF("CMaterialSystem::PS3InitFontLibrary");
if ( s_FontFileCache )
{
Warning("Tried to init font library twice without intervening unload\n");
return false;
}
int ret;
if ( (ret=load_libfont_module()) != CELL_OK )
{
Warning("Could not load font library: %x\n", ret);
}
maxNumFonts = maxNumFonts ? maxNumFonts : 1;
if (s_FontFileCache)
{
Assert(false); // wasn't cleaned up?!
delete[] s_FontFileCache;
}
if (s_UserFontEntries)
{
Assert(false); // wasn't cleaned up?!
delete[] s_UserFontEntries;
}
s_FontFileCache = new uint32[fontFileCacheSizeInBytes >> 2];
s_UserFontEntries = new CellFontEntry[maxNumFonts];
CellFontConfig config;
config.FileCache.buffer = s_FontFileCache;
config.FileCache.size = fontFileCacheSizeInBytes;
config.userFontEntrys = s_UserFontEntries;
config.userFontEntryMax = maxNumFonts;
ret = cellFontInit(&config);
if ( ret == CELL_OK )
{
// can override malloc/free here:
CellFontLibraryConfigFT ftConfig;
CellFontLibraryConfigFT_initialize( &ftConfig );
ftConfig.MemoryIF.Object = NULL;
ftConfig.MemoryIF.Malloc = fonts_malloc;
ftConfig.MemoryIF.Free = fonts_free;
ftConfig.MemoryIF.Realloc = fonts_realloc;
ftConfig.MemoryIF.Calloc = fonts_calloc;
ret = cellFontInitLibraryFreeType( &ftConfig, &s_FontLibrary );
if ( ret == CELL_OK )
{
g_fallbackFont.Init( );
return true;
}
else
{
Warning( "cellFontInitLibraryFreeType failed, %x\n", ret );
return false;
}
}
else
{
Warning( "CellFontInit failed, %x\n", ret );
return false;
}
}
void *CMaterialSystem::PS3GetFontLibPtr()
{
return (void*)s_FontLibrary;
}
void CMaterialSystem::PS3DumpFontLibrary()
{
if ( s_FontLibraryRefCount <= 0 )
{
Warning("Font library refcount is %d during unload!\n", s_FontLibraryRefCount);
return;
}
else if ( s_FontLibrary == NULL )
{
Warning("\t!!!FAILSAFE!!!\nPS3 font library refcount %d but was somehow unloaded!\n", s_FontLibraryRefCount);
s_FontLibraryRefCount = 0;
}
else if ( --s_FontLibraryRefCount > 0 )
{ // refcount is still greater than one, don't need to unload
return;
}
// Warning( "Font library was dumped with %d fonts left open.\n", m_vExtantFonts.Count() );
if ( s_FontLibrary )
{
g_fallbackFont.Shutdown();
cellFontEndLibrary( s_FontLibrary );
cellFontEnd();
delete[] s_FontFileCache;
s_FontFileCache = NULL;
// handled by cellFontEndLibrary: // delete s_FontLibrary;
s_FontLibrary = NULL;
delete[] s_UserFontEntries;
s_UserFontEntries = NULL;
cellSysmoduleUnloadModule( CELL_SYSMODULE_FONTFT );
cellSysmoduleUnloadModule( CELL_SYSMODULE_FREETYPE_TT );
cellSysmoduleUnloadModule( CELL_SYSMODULE_FONT );
}
}
static const CellFontLibrary *Fontlib()
{
return s_FontLibrary;
}