fbs/fontandbitmapserver/sfbs/glyphatlas.cpp
author hgs
Fri, 24 Sep 2010 16:14:28 +0300
changeset 187 9f66f99ee56f
permissions -rw-r--r--
201026
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
187
hgs
parents:
diff changeset
     1
// Copyright (c) 2009-2010 Nokia Corporation and/or its subsidiary(-ies).
hgs
parents:
diff changeset
     2
// All rights reserved.
hgs
parents:
diff changeset
     3
// This component and the accompanying materials are made available
hgs
parents:
diff changeset
     4
// under the terms of "Eclipse Public License v1.0"
hgs
parents:
diff changeset
     5
// which accompanies this distribution, and is available
hgs
parents:
diff changeset
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
hgs
parents:
diff changeset
     7
//
hgs
parents:
diff changeset
     8
// Initial Contributors:
hgs
parents:
diff changeset
     9
// Nokia Corporation - initial contribution.
hgs
parents:
diff changeset
    10
//
hgs
parents:
diff changeset
    11
// Contributors:
hgs
parents:
diff changeset
    12
//
hgs
parents:
diff changeset
    13
// Description:
hgs
parents:
diff changeset
    14
//
hgs
parents:
diff changeset
    15
hgs
parents:
diff changeset
    16
#include "glyphatlas.h"
hgs
parents:
diff changeset
    17
#include "OstTraceDefinitions.h"
hgs
parents:
diff changeset
    18
#ifdef OST_TRACE_COMPILER_IN_USE
hgs
parents:
diff changeset
    19
#include "glyphatlasTraces.h"
hgs
parents:
diff changeset
    20
#endif
hgs
parents:
diff changeset
    21
hgs
parents:
diff changeset
    22
hgs
parents:
diff changeset
    23
extern void Panic(TFbsPanic aPanic);
hgs
parents:
diff changeset
    24
hgs
parents:
diff changeset
    25
static TInt16 Load16(const TUint8* aPtr);
hgs
parents:
diff changeset
    26
static void DecodeBinaryData(const TSize& aGlyphSize, const TUint8* aEncodedData, TUint8* aByteData);
hgs
parents:
diff changeset
    27
static void DecodeBinaryDataExLarge(const TSize& aGlyphSize, const TUint8* aEncodedData, TUint8* aByteData);
hgs
parents:
diff changeset
    28
static void Convert1BppTo8Bpp(TUint32 aSrcData, TUint8*& aDestDataPtr, const TUint8* aDestDataPtrLimit);
hgs
parents:
diff changeset
    29
static void CopyCharLine(TUint8*& aByteDataPtr, TInt aWidthInBytes, const TUint8* aSrcData, TInt aBitShift, TInt16 aRepeatCount);
hgs
parents:
diff changeset
    30
hgs
parents:
diff changeset
    31
// === CGlyphAtlas Functions ===
hgs
parents:
diff changeset
    32
hgs
parents:
diff changeset
    33
/**
hgs
parents:
diff changeset
    34
Glyph Atlas constructor.
hgs
parents:
diff changeset
    35
@param aMaxCacheSizeInBytes The maximum amount of specialised graphics memory 
hgs
parents:
diff changeset
    36
	that the glyph atlas should use. If this value is KGlyphAtlasNoCacheLimit,
hgs
parents:
diff changeset
    37
	then there is no limit and the atlas will use as much memory as is available
hgs
parents:
diff changeset
    38
	in the system.
hgs
parents:
diff changeset
    39
*/
hgs
parents:
diff changeset
    40
CGlyphAtlas::CGlyphAtlas(TInt aMaxCacheSizeInBytes)
hgs
parents:
diff changeset
    41
	:iLruPageList(_FOFF(CGlyphAtlasPage, iLink)),
hgs
parents:
diff changeset
    42
	 iFontEntryArray(32, _FOFF(TFontEntryMap, iFont)),
hgs
parents:
diff changeset
    43
	 iMaxCacheSizeInBytes(aMaxCacheSizeInBytes),
hgs
parents:
diff changeset
    44
	 iMaxCacheSizeHigh(aMaxCacheSizeInBytes),
hgs
parents:
diff changeset
    45
	 iGpuCacheSizeLimitIsMax(ETrue)
hgs
parents:
diff changeset
    46
	{
hgs
parents:
diff changeset
    47
    iMaxCacheSizeLow = ( KGlyphAtlasNoCacheLimit == aMaxCacheSizeInBytes )
hgs
parents:
diff changeset
    48
                     ? KGlyphAtlasLowMemCacheLimitDefault
hgs
parents:
diff changeset
    49
                     : ( aMaxCacheSizeInBytes / KGlyphAtlasLowMemCacheLimitDivisor );
hgs
parents:
diff changeset
    50
	}
hgs
parents:
diff changeset
    51
hgs
parents:
diff changeset
    52
/**
hgs
parents:
diff changeset
    53
Glyph Atlas destructor.
hgs
parents:
diff changeset
    54
Frees all the RSgImage handles, frees all the allocated system memory, and
hgs
parents:
diff changeset
    55
closes the Graphics Resource driver. 
hgs
parents:
diff changeset
    56
*/
hgs
parents:
diff changeset
    57
CGlyphAtlas::~CGlyphAtlas()
hgs
parents:
diff changeset
    58
	{
hgs
parents:
diff changeset
    59
	// cycle through all the font entries and destroy them
hgs
parents:
diff changeset
    60
	for (TInt ii = iFontEntryArray.Count()-1; ii >= 0; --ii)
hgs
parents:
diff changeset
    61
		{
hgs
parents:
diff changeset
    62
		DeleteFontEntry(iFontEntryArray[ii].iEntry);
hgs
parents:
diff changeset
    63
		}
hgs
parents:
diff changeset
    64
	iFontEntryArray.Close();
hgs
parents:
diff changeset
    65
	__ASSERT_DEBUG(iLruPageList.IsEmpty(), Panic(EFbsPanicGlyphAtlasInconsistentState));
hgs
parents:
diff changeset
    66
	
hgs
parents:
diff changeset
    67
	// there shouldn't be any remaining pages, but if there are, destroy them.
hgs
parents:
diff changeset
    68
	while (!iLruPageList.IsEmpty())
hgs
parents:
diff changeset
    69
		{
hgs
parents:
diff changeset
    70
		delete iLruPageList.First();
hgs
parents:
diff changeset
    71
		}
hgs
parents:
diff changeset
    72
	iSgDriver.Close();
hgs
parents:
diff changeset
    73
	}
hgs
parents:
diff changeset
    74
hgs
parents:
diff changeset
    75
/**
hgs
parents:
diff changeset
    76
Factory constructor method. Creates a new glyph atlas.
hgs
parents:
diff changeset
    77
hgs
parents:
diff changeset
    78
@param aMaxCacheSizeInBytes The size in bytes, to use as the upper limit
hgs
parents:
diff changeset
    79
	for the size of memory used by the glyph images in the atlas. If this
hgs
parents:
diff changeset
    80
	value is KGlyphAtlasNoCacheLimit, then there is no limit and the atlas
hgs
parents:
diff changeset
    81
	will use as much memory as is available in the system.
hgs
parents:
diff changeset
    82
hgs
parents:
diff changeset
    83
@return A pointer to the newly-constructed atlas
hgs
parents:
diff changeset
    84
hgs
parents:
diff changeset
    85
@leave KErrNoMemory if there was insufficient memory to create the atlas, 
hgs
parents:
diff changeset
    86
	or a system wide error code if its RSgDriver failed to open.
hgs
parents:
diff changeset
    87
*/
hgs
parents:
diff changeset
    88
CGlyphAtlas* CGlyphAtlas::NewL(TInt aMaxCacheSizeInBytes)
hgs
parents:
diff changeset
    89
	{
hgs
parents:
diff changeset
    90
	CGlyphAtlas* self = new (ELeave) CGlyphAtlas(aMaxCacheSizeInBytes);
hgs
parents:
diff changeset
    91
	CleanupStack::PushL(self);
hgs
parents:
diff changeset
    92
	self->ConstructL();
hgs
parents:
diff changeset
    93
	CleanupStack::Pop(); // self;
hgs
parents:
diff changeset
    94
	return self;
hgs
parents:
diff changeset
    95
	}
hgs
parents:
diff changeset
    96
hgs
parents:
diff changeset
    97
/**
hgs
parents:
diff changeset
    98
Two-phase constructor.
hgs
parents:
diff changeset
    99
@leave A system wide error code if RSgDriver failed to open.
hgs
parents:
diff changeset
   100
*/
hgs
parents:
diff changeset
   101
void CGlyphAtlas::ConstructL()
hgs
parents:
diff changeset
   102
	{
hgs
parents:
diff changeset
   103
	User::LeaveIfError(iSgDriver.Open());
hgs
parents:
diff changeset
   104
	}
hgs
parents:
diff changeset
   105
hgs
parents:
diff changeset
   106
/**
hgs
parents:
diff changeset
   107
Retrieves a glyph from the atlas.
hgs
parents:
diff changeset
   108
If the glyph is found, the glyph data passed in is populated.
hgs
parents:
diff changeset
   109
hgs
parents:
diff changeset
   110
@param[in] aFont The font the glyph belongs to.
hgs
parents:
diff changeset
   111
@param[in] aGlyphCode The glyph code for the glyph being requested.
hgs
parents:
diff changeset
   112
@param[out] aGlyphImageInfo The glyph image information if this function is successful.
hgs
parents:
diff changeset
   113
@return KErrNone if the glyph is found, KErrNotFound if not. 
hgs
parents:
diff changeset
   114
*/
hgs
parents:
diff changeset
   115
TInt CGlyphAtlas::GetGlyph(const CBitmapFont& aFont, TUint aGlyphCode, TGlyphImageInfo& aGlyphImageInfo)
hgs
parents:
diff changeset
   116
	{
hgs
parents:
diff changeset
   117
    OstTraceExt2( TRACE_NORMAL, CGLYPHATLAS_GETGLYPH, "> f=%x; gc=%04x",(TUint)&aFont, aGlyphCode);
hgs
parents:
diff changeset
   118
    
hgs
parents:
diff changeset
   119
	CGlyphAtlasFontEntry* fontEntry = FindFontEntry(aFont);
hgs
parents:
diff changeset
   120
	if (!fontEntry)
hgs
parents:
diff changeset
   121
		{
hgs
parents:
diff changeset
   122
        OstTrace0( TRACE_NORMAL, CGLYPHATLAS_GETGLYPH_END2, "< KErrNotFound");
hgs
parents:
diff changeset
   123
		return KErrNotFound;
hgs
parents:
diff changeset
   124
		}
hgs
parents:
diff changeset
   125
	TInt err = fontEntry->GetGlyph(aGlyphCode, aGlyphImageInfo);
hgs
parents:
diff changeset
   126
	
hgs
parents:
diff changeset
   127
	OstTraceExt5( TRACE_NORMAL, CGLYPHATLAS_GETGLYPH_END1, "< id=%08x%08x; w=%u; h=%u; err=%d", 
hgs
parents:
diff changeset
   128
	        (TUint)I64HIGH(aGlyphImageInfo.iImageId.iId), (TUint)I64LOW(aGlyphImageInfo.iImageId.iId),
hgs
parents:
diff changeset
   129
	        (TUint)aGlyphImageInfo.iMetrics.Width(), (TUint)aGlyphImageInfo.iMetrics.Height(), (TInt)err);
hgs
parents:
diff changeset
   130
	
hgs
parents:
diff changeset
   131
	return err;
hgs
parents:
diff changeset
   132
	}
hgs
parents:
diff changeset
   133
hgs
parents:
diff changeset
   134
/**
hgs
parents:
diff changeset
   135
Adds a glyph to the atlas from a bitmap glyph and retrieves the glyph data.
hgs
parents:
diff changeset
   136
If there is insufficient memory to create a RSgImage, then the least recently 
hgs
parents:
diff changeset
   137
used pages (and all the glyphs contained within) are removed until there is 
hgs
parents:
diff changeset
   138
enough memory to continue.
hgs
parents:
diff changeset
   139
hgs
parents:
diff changeset
   140
@param[in] aFont The font the glyph belongs to.
hgs
parents:
diff changeset
   141
@param[in] aArgs The information needed to create a glyph.
hgs
parents:
diff changeset
   142
@param[out] aGlyphImageInfo Upon return contains all the glyph image information needed to use the 
hgs
parents:
diff changeset
   143
		glyph in another process.
hgs
parents:
diff changeset
   144
@return KErrNone if the glyph was successfully added or other system-wide error.
hgs
parents:
diff changeset
   145
*/
hgs
parents:
diff changeset
   146
TInt CGlyphAtlas::AddGlyph(const CBitmapFont& aFont, const TAddGlyphArgs& aArgs, TGlyphImageInfo& aGlyphImageInfo)
hgs
parents:
diff changeset
   147
	{
hgs
parents:
diff changeset
   148
	OstTraceDefExt5( OST_TRACE_CATEGORY_DEBUG, TRACE_NORMAL, CGLYPHATLAS_ADDGLYPH, "> f=%x; bp=%08x; gc=%04x; w=%u; h=%u", 
hgs
parents:
diff changeset
   149
	        (TUint)&aFont, (TUint)aArgs.iBitmapPointer, aArgs.iGlyphCode, 
hgs
parents:
diff changeset
   150
	        aArgs.iMetrics->Width(), aArgs.iMetrics->Height());
hgs
parents:
diff changeset
   151
	
hgs
parents:
diff changeset
   152
    // Find font entry and create if none found
hgs
parents:
diff changeset
   153
	CGlyphAtlasFontEntry* fontEntry = FindFontEntry(aFont);
hgs
parents:
diff changeset
   154
	TBool isNewFont = EFalse;
hgs
parents:
diff changeset
   155
	if (!fontEntry)
hgs
parents:
diff changeset
   156
		{
hgs
parents:
diff changeset
   157
		// Create a new font.
hgs
parents:
diff changeset
   158
		fontEntry = CreateFontEntry(aFont);
hgs
parents:
diff changeset
   159
		if (!fontEntry)
hgs
parents:
diff changeset
   160
			{
hgs
parents:
diff changeset
   161
            OstTraceDef0(OST_TRACE_CATEGORY_DEBUG, TRACE_NORMAL, CGLYPHATLAS_ADDGLYPH_END2, "< KErrNoMemory");
hgs
parents:
diff changeset
   162
			return KErrNoMemory;
hgs
parents:
diff changeset
   163
			}
hgs
parents:
diff changeset
   164
		isNewFont = ETrue;
hgs
parents:
diff changeset
   165
		}
hgs
parents:
diff changeset
   166
	TInt glyphSizeInBytes = 0;
hgs
parents:
diff changeset
   167
	TInt err = fontEntry->AddGlyph(aArgs, aGlyphImageInfo, glyphSizeInBytes);
hgs
parents:
diff changeset
   168
	if (KErrNone != err)
hgs
parents:
diff changeset
   169
		{
hgs
parents:
diff changeset
   170
		if (isNewFont)
hgs
parents:
diff changeset
   171
			{
hgs
parents:
diff changeset
   172
			DeleteFontEntry(fontEntry);
hgs
parents:
diff changeset
   173
			}
hgs
parents:
diff changeset
   174
		OstTraceDef1(OST_TRACE_CATEGORY_DEBUG, TRACE_NORMAL, CGLYPHATLAS_ADDGLYPH_END3, "< err=%d", err);
hgs
parents:
diff changeset
   175
		return err;
hgs
parents:
diff changeset
   176
		}
hgs
parents:
diff changeset
   177
	iCacheSizeInBytes += glyphSizeInBytes;
hgs
parents:
diff changeset
   178
hgs
parents:
diff changeset
   179
	// If there is a cache limit and it is now exceeded, remove the least
hgs
parents:
diff changeset
   180
	// recently used pages until the cache size is within the upper limit. Do 
hgs
parents:
diff changeset
   181
	// not remove the page relating to the glyph which is being added, which is 
hgs
parents:
diff changeset
   182
	// now at the head of the LRU array.
hgs
parents:
diff changeset
   183
	if (iMaxCacheSizeInBytes != KGlyphAtlasNoCacheLimit)
hgs
parents:
diff changeset
   184
		{
hgs
parents:
diff changeset
   185
		TBool morePagesToDelete = ETrue;
hgs
parents:
diff changeset
   186
		while ((iCacheSizeInBytes > iMaxCacheSizeInBytes) && morePagesToDelete)
hgs
parents:
diff changeset
   187
			{
hgs
parents:
diff changeset
   188
			morePagesToDelete = DeleteLeastRecentlyUsedPage(EFalse);
hgs
parents:
diff changeset
   189
			}
hgs
parents:
diff changeset
   190
		}
hgs
parents:
diff changeset
   191
	
hgs
parents:
diff changeset
   192
	OstTraceDefExt2(OST_TRACE_CATEGORY_DEBUG, TRACE_NORMAL, CGLYPHATLAS_ADDGLYPH_END1, "< id=%08x%08x", 
hgs
parents:
diff changeset
   193
	            I64HIGH(aGlyphImageInfo.iImageId.iId), I64LOW(aGlyphImageInfo.iImageId.iId));
hgs
parents:
diff changeset
   194
    
hgs
parents:
diff changeset
   195
	return KErrNone;
hgs
parents:
diff changeset
   196
	}
hgs
parents:
diff changeset
   197
hgs
parents:
diff changeset
   198
/**
hgs
parents:
diff changeset
   199
Releases all glyphs associated with a particular font when it has been
hgs
parents:
diff changeset
   200
released by the font system.
hgs
parents:
diff changeset
   201
hgs
parents:
diff changeset
   202
@param aFont The font which is released.
hgs
parents:
diff changeset
   203
 */
hgs
parents:
diff changeset
   204
void CGlyphAtlas::FontReleased(const CBitmapFont& aFont)
hgs
parents:
diff changeset
   205
	{
hgs
parents:
diff changeset
   206
	OstTraceDef1(OST_TRACE_CATEGORY_DEBUG, TRACE_NORMAL, CGLYPHATLAS_FONTRELEASED, "> f=%x", (TUint)&aFont);
hgs
parents:
diff changeset
   207
	
hgs
parents:
diff changeset
   208
    TInt index = iFontEntryArray.FindInUnsignedKeyOrder(TFontEntryMap(&aFont, NULL));
hgs
parents:
diff changeset
   209
	if (KErrNotFound == index)
hgs
parents:
diff changeset
   210
		{
hgs
parents:
diff changeset
   211
        OstTraceDef0(OST_TRACE_CATEGORY_DEBUG, TRACE_NORMAL, CGLYPHATLAS_FONTRELEASED_END2, "< KErrNotFound");
hgs
parents:
diff changeset
   212
		return;
hgs
parents:
diff changeset
   213
		}
hgs
parents:
diff changeset
   214
	CGlyphAtlasFontEntry* fontEntry = iFontEntryArray[index].iEntry;
hgs
parents:
diff changeset
   215
	iCacheSizeInBytes -= fontEntry->SizeInBytes();
hgs
parents:
diff changeset
   216
	delete fontEntry;
hgs
parents:
diff changeset
   217
	iFontEntryArray.Remove(index);
hgs
parents:
diff changeset
   218
	OstTraceDef0(OST_TRACE_CATEGORY_DEBUG, TRACE_NORMAL, CGLYPHATLAS_FONTRELEASED_END3, "< KErrNone");
hgs
parents:
diff changeset
   219
	}
hgs
parents:
diff changeset
   220
hgs
parents:
diff changeset
   221
/**
hgs
parents:
diff changeset
   222
Searches the array of font entries to find the entry for the given font.
hgs
parents:
diff changeset
   223
If the font entry is found, the entry is returned. If not a NULL pointer is 
hgs
parents:
diff changeset
   224
returned.
hgs
parents:
diff changeset
   225
hgs
parents:
diff changeset
   226
@param aFont The font to match an entry with.
hgs
parents:
diff changeset
   227
@return The font entry if a match is found, NULL if not. 
hgs
parents:
diff changeset
   228
*/
hgs
parents:
diff changeset
   229
CGlyphAtlasFontEntry* CGlyphAtlas::FindFontEntry(const CBitmapFont& aFont) const
hgs
parents:
diff changeset
   230
	{
hgs
parents:
diff changeset
   231
	TFontEntryMap entryToMatch(&aFont, NULL);
hgs
parents:
diff changeset
   232
	TInt index = iFontEntryArray.FindInUnsignedKeyOrder(entryToMatch);
hgs
parents:
diff changeset
   233
	if (KErrNotFound == index)
hgs
parents:
diff changeset
   234
		{
hgs
parents:
diff changeset
   235
		return NULL;
hgs
parents:
diff changeset
   236
		}
hgs
parents:
diff changeset
   237
	return iFontEntryArray[index].iEntry;
hgs
parents:
diff changeset
   238
	}
hgs
parents:
diff changeset
   239
hgs
parents:
diff changeset
   240
/**
hgs
parents:
diff changeset
   241
Deletes the given font entry.
hgs
parents:
diff changeset
   242
The mapping from the font to the font entry is removed.
hgs
parents:
diff changeset
   243
hgs
parents:
diff changeset
   244
@param aFontEntry The entry to delete.
hgs
parents:
diff changeset
   245
*/
hgs
parents:
diff changeset
   246
void CGlyphAtlas::DeleteFontEntry(CGlyphAtlasFontEntry* aFontEntry)
hgs
parents:
diff changeset
   247
	{
hgs
parents:
diff changeset
   248
    __ASSERT_DEBUG(aFontEntry, Panic(EFbsPanicGlyphAtlasInconsistentState));
hgs
parents:
diff changeset
   249
    TInt index = iFontEntryArray.FindInUnsignedKeyOrder(TFontEntryMap(&aFontEntry->Font(), NULL));
hgs
parents:
diff changeset
   250
    __ASSERT_DEBUG(KErrNotFound != index, Panic(EFbsPanicGlyphAtlasInconsistentState));
hgs
parents:
diff changeset
   251
    if (KErrNotFound != index)
hgs
parents:
diff changeset
   252
        {
hgs
parents:
diff changeset
   253
        iFontEntryArray.Remove(index);
hgs
parents:
diff changeset
   254
        }
hgs
parents:
diff changeset
   255
    iCacheSizeInBytes -= aFontEntry->SizeInBytes();
hgs
parents:
diff changeset
   256
    delete aFontEntry;
hgs
parents:
diff changeset
   257
	}
hgs
parents:
diff changeset
   258
hgs
parents:
diff changeset
   259
/**
hgs
parents:
diff changeset
   260
Moves the given page to the front (the position of the most recently used page) 
hgs
parents:
diff changeset
   261
of the usage order list.
hgs
parents:
diff changeset
   262
hgs
parents:
diff changeset
   263
@param aPage The most recently used page.
hgs
parents:
diff changeset
   264
*/
hgs
parents:
diff changeset
   265
void CGlyphAtlas::MovePageToFront(CGlyphAtlasPage& aPage)
hgs
parents:
diff changeset
   266
	{
hgs
parents:
diff changeset
   267
	aPage.MoveToFirstInQueue(iLruPageList);
hgs
parents:
diff changeset
   268
	}
hgs
parents:
diff changeset
   269
hgs
parents:
diff changeset
   270
/**
hgs
parents:
diff changeset
   271
Creates a font entry from the given font and adds a mapping from the font 
hgs
parents:
diff changeset
   272
to the font entry.
hgs
parents:
diff changeset
   273
If successful, the font entry is returned.
hgs
parents:
diff changeset
   274
If either the creation of the font entry or the adding the mapping fails, then 
hgs
parents:
diff changeset
   275
a NULL pointer is returned.
hgs
parents:
diff changeset
   276
hgs
parents:
diff changeset
   277
@param aFont The font used to create a font entry from.
hgs
parents:
diff changeset
   278
@return A new font entry if successful, NULL if not.
hgs
parents:
diff changeset
   279
*/
hgs
parents:
diff changeset
   280
CGlyphAtlasFontEntry* CGlyphAtlas::CreateFontEntry(const CBitmapFont& aFont)
hgs
parents:
diff changeset
   281
	{
hgs
parents:
diff changeset
   282
	CGlyphAtlasFontEntry* fontEntry = new CGlyphAtlasFontEntry(aFont, *this);
hgs
parents:
diff changeset
   283
	if (!fontEntry)
hgs
parents:
diff changeset
   284
		{
hgs
parents:
diff changeset
   285
		return NULL;
hgs
parents:
diff changeset
   286
		}
hgs
parents:
diff changeset
   287
	// Add font entry to font entry array
hgs
parents:
diff changeset
   288
	TFontEntryMap fontEntryMap(&aFont, fontEntry);
hgs
parents:
diff changeset
   289
	TInt err = iFontEntryArray.InsertInUnsignedKeyOrder(fontEntryMap);
hgs
parents:
diff changeset
   290
	__ASSERT_DEBUG(KErrAlreadyExists != err, Panic(EFbsPanicGlyphAtlasInconsistentState));
hgs
parents:
diff changeset
   291
	if (KErrNone != err)
hgs
parents:
diff changeset
   292
		{
hgs
parents:
diff changeset
   293
		delete fontEntry;
hgs
parents:
diff changeset
   294
		fontEntry = NULL;
hgs
parents:
diff changeset
   295
		}
hgs
parents:
diff changeset
   296
	return fontEntry;
hgs
parents:
diff changeset
   297
	}
hgs
parents:
diff changeset
   298
hgs
parents:
diff changeset
   299
/**
hgs
parents:
diff changeset
   300
Deletes the least recently used page and removes it from the list of pages
hgs
parents:
diff changeset
   301
held by the atlas.
hgs
parents:
diff changeset
   302
hgs
parents:
diff changeset
   303
@param aAllowMruPageDeletion ETrue if the most recently used page can be deleted, 
hgs
parents:
diff changeset
   304
	EFalse otherwise.
hgs
parents:
diff changeset
   305
@return ETrue, if there are pages remaining in the atlas after the deletion, EFalse
hgs
parents:
diff changeset
   306
	otherwise. If there is only one page in the atlas and aAllowMruPageDeletion is EFalse,
hgs
parents:
diff changeset
   307
	EFalse is returned. 
hgs
parents:
diff changeset
   308
*/
hgs
parents:
diff changeset
   309
TBool CGlyphAtlas::DeleteLeastRecentlyUsedPage(TBool aAllowMruPageDeletion)
hgs
parents:
diff changeset
   310
	{
hgs
parents:
diff changeset
   311
    OstTraceDef1(OST_TRACE_CATEGORY_DEBUG, TRACE_NORMAL, CGLYPHATLAS_DELETELEASTRECENTLYUSEDPAGE, "> mru=%d", aAllowMruPageDeletion );
hgs
parents:
diff changeset
   312
    
hgs
parents:
diff changeset
   313
	CGlyphAtlasPage* lruPage = NULL;
hgs
parents:
diff changeset
   314
	if (!iLruPageList.IsEmpty())
hgs
parents:
diff changeset
   315
		{
hgs
parents:
diff changeset
   316
		lruPage = iLruPageList.Last();
hgs
parents:
diff changeset
   317
		if (!aAllowMruPageDeletion && (lruPage == iLruPageList.First()))
hgs
parents:
diff changeset
   318
			{
hgs
parents:
diff changeset
   319
			lruPage = NULL;
hgs
parents:
diff changeset
   320
			}
hgs
parents:
diff changeset
   321
		}
hgs
parents:
diff changeset
   322
	TBool canDeleteMorePages = EFalse;
hgs
parents:
diff changeset
   323
	if (lruPage)
hgs
parents:
diff changeset
   324
		{
hgs
parents:
diff changeset
   325
		iCacheSizeInBytes -= lruPage->SizeInBytes();
hgs
parents:
diff changeset
   326
		CGlyphAtlasFontEntry& fontEntry = lruPage->FontEntry();
hgs
parents:
diff changeset
   327
		fontEntry.DeletePage(lruPage);
hgs
parents:
diff changeset
   328
		lruPage = NULL;
hgs
parents:
diff changeset
   329
		if (fontEntry.IsEmpty())
hgs
parents:
diff changeset
   330
			{
hgs
parents:
diff changeset
   331
			DeleteFontEntry(&fontEntry);
hgs
parents:
diff changeset
   332
			}
hgs
parents:
diff changeset
   333
		canDeleteMorePages = !iLruPageList.IsEmpty();
hgs
parents:
diff changeset
   334
		}
hgs
parents:
diff changeset
   335
	OstTraceDefExt2(OST_TRACE_CATEGORY_DEBUG, TRACE_NORMAL, CGLYPHATLAS_DELETELEASTRECENTLYUSEDPAGE_END, "< more=%u; size=%u", (TUint)canDeleteMorePages, iCacheSizeInBytes);
hgs
parents:
diff changeset
   336
	return canDeleteMorePages;
hgs
parents:
diff changeset
   337
	}
hgs
parents:
diff changeset
   338
hgs
parents:
diff changeset
   339
/**
hgs
parents:
diff changeset
   340
Utility function that calculates the number of unique fonts associated with the atlas.
hgs
parents:
diff changeset
   341
@return Number of fonts in the atlas.
hgs
parents:
diff changeset
   342
 */
hgs
parents:
diff changeset
   343
TInt CGlyphAtlas::FontCount() const
hgs
parents:
diff changeset
   344
	{
hgs
parents:
diff changeset
   345
	return iFontEntryArray.Count();
hgs
parents:
diff changeset
   346
	}
hgs
parents:
diff changeset
   347
hgs
parents:
diff changeset
   348
/**
hgs
parents:
diff changeset
   349
Utility function that calculates the number of glyphs across all fonts stored in
hgs
parents:
diff changeset
   350
the atlas.
hgs
parents:
diff changeset
   351
@return Number of glyphs in the atlas.
hgs
parents:
diff changeset
   352
 */
hgs
parents:
diff changeset
   353
TInt CGlyphAtlas::GlyphCount() const
hgs
parents:
diff changeset
   354
	{
hgs
parents:
diff changeset
   355
	TInt glyphCount = 0;
hgs
parents:
diff changeset
   356
	for (TInt ii = iFontEntryArray.Count() - 1; ii >= 0; --ii)
hgs
parents:
diff changeset
   357
		{
hgs
parents:
diff changeset
   358
		glyphCount += iFontEntryArray[ii].iEntry->GlyphCount();
hgs
parents:
diff changeset
   359
		}
hgs
parents:
diff changeset
   360
	return glyphCount;
hgs
parents:
diff changeset
   361
	}
hgs
parents:
diff changeset
   362
hgs
parents:
diff changeset
   363
/**
hgs
parents:
diff changeset
   364
Utility function that calculates the number of glyphs for a given font in the atlas.
hgs
parents:
diff changeset
   365
@param The font to return the number of glyphs for.
hgs
parents:
diff changeset
   366
@return Number of glyphs in the atlas.
hgs
parents:
diff changeset
   367
 */
hgs
parents:
diff changeset
   368
TInt CGlyphAtlas::GlyphCount(const CBitmapFont& aFont) const
hgs
parents:
diff changeset
   369
	{
hgs
parents:
diff changeset
   370
	CGlyphAtlasFontEntry* fontEntry = FindFontEntry(aFont);
hgs
parents:
diff changeset
   371
	return (fontEntry) ? fontEntry->GlyphCount() : 0;
hgs
parents:
diff changeset
   372
	}
hgs
parents:
diff changeset
   373
hgs
parents:
diff changeset
   374
void CGlyphAtlas::GetGlyphCacheMetrics( TGlyphCacheMetrics& aGlyphCacheMetrics )
hgs
parents:
diff changeset
   375
    {
hgs
parents:
diff changeset
   376
    aGlyphCacheMetrics.iMaxCacheSizeInBytes = iMaxCacheSizeInBytes;
hgs
parents:
diff changeset
   377
    aGlyphCacheMetrics.iMaxCacheSizeHigh = iMaxCacheSizeHigh;
hgs
parents:
diff changeset
   378
    aGlyphCacheMetrics.iMaxCacheSizeLow = iMaxCacheSizeLow;
hgs
parents:
diff changeset
   379
    aGlyphCacheMetrics.iCacheSizeInBytes = iCacheSizeInBytes;
hgs
parents:
diff changeset
   380
    aGlyphCacheMetrics.iGpuCacheSizeLimitIsMax = iGpuCacheSizeLimitIsMax;
hgs
parents:
diff changeset
   381
    }
hgs
parents:
diff changeset
   382
hgs
parents:
diff changeset
   383
/**
hgs
parents:
diff changeset
   384
 Function to release the GPU cache. Called in response to the GoomMonitor's
hgs
parents:
diff changeset
   385
 requirement to reduce GPU memory use.
hgs
parents:
diff changeset
   386
hgs
parents:
diff changeset
   387
 @param aBytes. The amount of memory the GOoM framework would like us to relinquish.
hgs
parents:
diff changeset
   388
 @param aFlags. The flags conveyed from the GOoM monitor framework.
hgs
parents:
diff changeset
   389
*/
hgs
parents:
diff changeset
   390
void CGlyphAtlas::ReleaseGpuMemory( TInt /*aBytes*/, TInt /*aFlags*/ )
hgs
parents:
diff changeset
   391
    {
hgs
parents:
diff changeset
   392
    OstTraceDefExt2( OST_TRACE_CATEGORY_DEBUG, TRACE_NORMAL, CGLYPHATLAS_RELEASEGPUMEMORY, "> max=%d; size=%d", iMaxCacheSizeInBytes, iCacheSizeInBytes);
hgs
parents:
diff changeset
   393
hgs
parents:
diff changeset
   394
    if ( iCacheSizeInBytes > 0 )
hgs
parents:
diff changeset
   395
        {
hgs
parents:
diff changeset
   396
        while( DeleteLeastRecentlyUsedPage(ETrue) )
hgs
parents:
diff changeset
   397
            {
hgs
parents:
diff changeset
   398
            // Do nothing
hgs
parents:
diff changeset
   399
            }
hgs
parents:
diff changeset
   400
        }
hgs
parents:
diff changeset
   401
hgs
parents:
diff changeset
   402
    // If appropriate, reduce the cache-size limit.
hgs
parents:
diff changeset
   403
    if ( GpuCacheSizeLimitIsMax() )
hgs
parents:
diff changeset
   404
        {
hgs
parents:
diff changeset
   405
        SwitchGpuCacheSizeLimit();
hgs
parents:
diff changeset
   406
        }
hgs
parents:
diff changeset
   407
hgs
parents:
diff changeset
   408
    OstTraceDefExt2( OST_TRACE_CATEGORY_DEBUG, TRACE_NORMAL, CGLYPHATLAS_RELEASEGPUMEMORY_EXIT, "< max=%d; size=%d", iMaxCacheSizeInBytes, iCacheSizeInBytes);
hgs
parents:
diff changeset
   409
    }
hgs
parents:
diff changeset
   410
hgs
parents:
diff changeset
   411
/**
hgs
parents:
diff changeset
   412
 Function to establish GPU memory use. Called in response to the GoomMonitor's
hgs
parents:
diff changeset
   413
 notification that GPU memory may once more be utilised in the usual manner.
hgs
parents:
diff changeset
   414
hgs
parents:
diff changeset
   415
 @param aFlags. The flags conveyed from the GOoM monitor framework.
hgs
parents:
diff changeset
   416
 */
hgs
parents:
diff changeset
   417
void CGlyphAtlas::InstateGpuMemory( TInt /*aFlags*/ )
hgs
parents:
diff changeset
   418
    {
hgs
parents:
diff changeset
   419
    OstTraceDef1(OST_TRACE_CATEGORY_DEBUG, TRACE_NORMAL, CGLYPHATLAS_INSTATEGPUMEMORY, "> max=%d", iMaxCacheSizeInBytes );
hgs
parents:
diff changeset
   420
hgs
parents:
diff changeset
   421
    // If appropriate, reinstate the full cache-size limit.
hgs
parents:
diff changeset
   422
    if ( !GpuCacheSizeLimitIsMax() )
hgs
parents:
diff changeset
   423
        {
hgs
parents:
diff changeset
   424
        SwitchGpuCacheSizeLimit();
hgs
parents:
diff changeset
   425
        }
hgs
parents:
diff changeset
   426
hgs
parents:
diff changeset
   427
    OstTraceDef1(OST_TRACE_CATEGORY_DEBUG, TRACE_NORMAL, CGLYPHATLAS_INSTATEGPUMEMORY_EXIT, "< max=%d", iMaxCacheSizeInBytes );
hgs
parents:
diff changeset
   428
    }
hgs
parents:
diff changeset
   429
hgs
parents:
diff changeset
   430
/**
hgs
parents:
diff changeset
   431
 Utility function to toggle between full and reduced GPU cache-size limits.
hgs
parents:
diff changeset
   432
hgs
parents:
diff changeset
   433
 Use in conjunction with TBool GpuCacheSizeLimitIsMax()
hgs
parents:
diff changeset
   434
 */
hgs
parents:
diff changeset
   435
void CGlyphAtlas::SwitchGpuCacheSizeLimit()
hgs
parents:
diff changeset
   436
    {
hgs
parents:
diff changeset
   437
    if ( GpuCacheSizeLimitIsMax() )
hgs
parents:
diff changeset
   438
        {
hgs
parents:
diff changeset
   439
        // The short-circuit operator obviates tautological conditionals.
hgs
parents:
diff changeset
   440
        while ( (iCacheSizeInBytes >= iMaxCacheSizeLow) && DeleteLeastRecentlyUsedPage(ETrue) )
hgs
parents:
diff changeset
   441
            {
hgs
parents:
diff changeset
   442
            // Do Nothing...
hgs
parents:
diff changeset
   443
            }
hgs
parents:
diff changeset
   444
hgs
parents:
diff changeset
   445
        iMaxCacheSizeInBytes = iMaxCacheSizeLow;
hgs
parents:
diff changeset
   446
        iGpuCacheSizeLimitIsMax = EFalse;
hgs
parents:
diff changeset
   447
        }
hgs
parents:
diff changeset
   448
    else
hgs
parents:
diff changeset
   449
        {
hgs
parents:
diff changeset
   450
        iMaxCacheSizeInBytes = iMaxCacheSizeHigh;
hgs
parents:
diff changeset
   451
        iGpuCacheSizeLimitIsMax = ETrue;
hgs
parents:
diff changeset
   452
        }
hgs
parents:
diff changeset
   453
    }
hgs
parents:
diff changeset
   454
hgs
parents:
diff changeset
   455
hgs
parents:
diff changeset
   456
/**
hgs
parents:
diff changeset
   457
 Utility function to report whether the GPU cache-size limit is set to the maximum permissible
hgs
parents:
diff changeset
   458
 level, or to its reduced level.
hgs
parents:
diff changeset
   459
hgs
parents:
diff changeset
   460
 @return ETrue if the cache-size is set to the maximum permissible limit.
hgs
parents:
diff changeset
   461
         EFalse if it is set to the reduced limit.
hgs
parents:
diff changeset
   462
 */
hgs
parents:
diff changeset
   463
TBool CGlyphAtlas::GpuCacheSizeLimitIsMax() const
hgs
parents:
diff changeset
   464
    {
hgs
parents:
diff changeset
   465
    return iGpuCacheSizeLimitIsMax;
hgs
parents:
diff changeset
   466
    }
hgs
parents:
diff changeset
   467
hgs
parents:
diff changeset
   468
// === CGlyphAtlasFontEntry Functions ===
hgs
parents:
diff changeset
   469
/**
hgs
parents:
diff changeset
   470
Font entry constructor.
hgs
parents:
diff changeset
   471
hgs
parents:
diff changeset
   472
@param aFont The font to which this font entry should be associated.
hgs
parents:
diff changeset
   473
@param aGlyphAtlas The glyph atlas to which this font entry belongs.
hgs
parents:
diff changeset
   474
*/
hgs
parents:
diff changeset
   475
CGlyphAtlasFontEntry::CGlyphAtlasFontEntry(const CBitmapFont& aFont, CGlyphAtlas& aGlyphAtlas)
hgs
parents:
diff changeset
   476
	:iFont(aFont),
hgs
parents:
diff changeset
   477
	 iPageArray(32, _FOFF(TPageMap, iGlyphCode)),
hgs
parents:
diff changeset
   478
	 iAtlas(aGlyphAtlas)
hgs
parents:
diff changeset
   479
	{
hgs
parents:
diff changeset
   480
	}
hgs
parents:
diff changeset
   481
hgs
parents:
diff changeset
   482
/**
hgs
parents:
diff changeset
   483
Font entry destructor.
hgs
parents:
diff changeset
   484
Destroys the pages owned by the font entry.
hgs
parents:
diff changeset
   485
*/
hgs
parents:
diff changeset
   486
CGlyphAtlasFontEntry::~CGlyphAtlasFontEntry()
hgs
parents:
diff changeset
   487
	{
hgs
parents:
diff changeset
   488
	// cycle through all the font entries and destroy them
hgs
parents:
diff changeset
   489
	for (TInt ii = iPageArray.Count()-1; ii >= 0; --ii)
hgs
parents:
diff changeset
   490
		{
hgs
parents:
diff changeset
   491
		DeletePage(iPageArray[ii].iPage);
hgs
parents:
diff changeset
   492
		}
hgs
parents:
diff changeset
   493
	__ASSERT_DEBUG(iSizeInBytes == 0, Panic(EFbsPanicGlyphAtlasInconsistentState));
hgs
parents:
diff changeset
   494
	iPageArray.Close();
hgs
parents:
diff changeset
   495
	}
hgs
parents:
diff changeset
   496
hgs
parents:
diff changeset
   497
/**
hgs
parents:
diff changeset
   498
Deletes the given page.
hgs
parents:
diff changeset
   499
The page mapping is removed and the page is deleted.
hgs
parents:
diff changeset
   500
hgs
parents:
diff changeset
   501
@param aPage The page to delete.
hgs
parents:
diff changeset
   502
*/
hgs
parents:
diff changeset
   503
void CGlyphAtlasFontEntry::DeletePage(CGlyphAtlasPage* aPage)
hgs
parents:
diff changeset
   504
	{
hgs
parents:
diff changeset
   505
    __ASSERT_DEBUG(aPage, Panic(EFbsPanicGlyphAtlasInconsistentState));
hgs
parents:
diff changeset
   506
    __ASSERT_DEBUG(iPageArray.Count() > 0, Panic(EFbsPanicGlyphAtlasInconsistentState));
hgs
parents:
diff changeset
   507
		
hgs
parents:
diff changeset
   508
    TInt numGlyphsInPage = aPage->GlyphCount();
hgs
parents:
diff changeset
   509
hgs
parents:
diff changeset
   510
    for (TInt ii = 0; ii < numGlyphsInPage; ++ii)
hgs
parents:
diff changeset
   511
        {
hgs
parents:
diff changeset
   512
        TInt index = iPageArray.FindInUnsignedKeyOrder(TPageMap(aPage->GlyphCodeAt(ii), NULL));
hgs
parents:
diff changeset
   513
hgs
parents:
diff changeset
   514
        __ASSERT_DEBUG(KErrNotFound != index, Panic(EFbsPanicGlyphAtlasInconsistentState));
hgs
parents:
diff changeset
   515
hgs
parents:
diff changeset
   516
        if (KErrNotFound != index)
hgs
parents:
diff changeset
   517
            {
hgs
parents:
diff changeset
   518
            iPageArray.Remove(index);
hgs
parents:
diff changeset
   519
            iPageArray.GranularCompress();
hgs
parents:
diff changeset
   520
            }
hgs
parents:
diff changeset
   521
        }
hgs
parents:
diff changeset
   522
    iSizeInBytes -= aPage->SizeInBytes();
hgs
parents:
diff changeset
   523
    delete aPage;
hgs
parents:
diff changeset
   524
hgs
parents:
diff changeset
   525
	}
hgs
parents:
diff changeset
   526
hgs
parents:
diff changeset
   527
/**
hgs
parents:
diff changeset
   528
Adds a glyph to the font from a bitmap glyph and gets the glyph image info back.
hgs
parents:
diff changeset
   529
hgs
parents:
diff changeset
   530
@param[in] aArgs The information needed to create a glyph.
hgs
parents:
diff changeset
   531
@param[out] aGlyphImageInfo Upon return contains all the glyph image information needed to use the 
hgs
parents:
diff changeset
   532
		glyph in another process.
hgs
parents:
diff changeset
   533
@param[out] aSizeInBytes Upon return contains the size of the added glyph's image data in bytes.
hgs
parents:
diff changeset
   534
@return KErrNone if the glyph was successfully added or other system-wide error.
hgs
parents:
diff changeset
   535
*/
hgs
parents:
diff changeset
   536
TInt CGlyphAtlasFontEntry::AddGlyph(const CGlyphAtlas::TAddGlyphArgs& aArgs, TGlyphImageInfo& aGlyphImageInfo, TInt& aSizeInBytes)
hgs
parents:
diff changeset
   537
	{
hgs
parents:
diff changeset
   538
    CGlyphAtlasPage* newPage = new CGlyphAtlasPage(*this);
hgs
parents:
diff changeset
   539
	if (!newPage)
hgs
parents:
diff changeset
   540
		{
hgs
parents:
diff changeset
   541
		return KErrNoMemory;
hgs
parents:
diff changeset
   542
		}
hgs
parents:
diff changeset
   543
	
hgs
parents:
diff changeset
   544
	TInt err = newPage->AddGlyph(aArgs, aGlyphImageInfo, aSizeInBytes);
hgs
parents:
diff changeset
   545
	if (KErrNone != err)
hgs
parents:
diff changeset
   546
		{
hgs
parents:
diff changeset
   547
		delete newPage;
hgs
parents:
diff changeset
   548
		return err;
hgs
parents:
diff changeset
   549
		}
hgs
parents:
diff changeset
   550
hgs
parents:
diff changeset
   551
	err = iPageArray.InsertInUnsignedKeyOrder(TPageMap(aArgs.iGlyphCode, newPage));
hgs
parents:
diff changeset
   552
	__ASSERT_DEBUG(KErrAlreadyExists != err, Panic(EFbsPanicGlyphAtlasInconsistentState));
hgs
parents:
diff changeset
   553
	if (KErrNone != err)
hgs
parents:
diff changeset
   554
		{
hgs
parents:
diff changeset
   555
		delete newPage;
hgs
parents:
diff changeset
   556
		return err;
hgs
parents:
diff changeset
   557
		}
hgs
parents:
diff changeset
   558
	iSizeInBytes += aSizeInBytes;
hgs
parents:
diff changeset
   559
	iAtlas.MovePageToFront(*newPage);
hgs
parents:
diff changeset
   560
	return err;
hgs
parents:
diff changeset
   561
	}
hgs
parents:
diff changeset
   562
hgs
parents:
diff changeset
   563
/**
hgs
parents:
diff changeset
   564
Searches the array of pages to find the page containing the given glyph.
hgs
parents:
diff changeset
   565
If the page is found, the glyph image info is populated and the page is moved 
hgs
parents:
diff changeset
   566
to the front of the Glyph Atlas' LRU page array.
hgs
parents:
diff changeset
   567
hgs
parents:
diff changeset
   568
@param[in] aGlyphCode The glyph code for the glyph being requested.
hgs
parents:
diff changeset
   569
@param[out] aGlyphImageInfo Upon return contains all the glyph image information needed to use the 
hgs
parents:
diff changeset
   570
		glyph in another process.
hgs
parents:
diff changeset
   571
@return KErrNone if the glyph was found, KErrNotFound if not. 
hgs
parents:
diff changeset
   572
*/
hgs
parents:
diff changeset
   573
TInt CGlyphAtlasFontEntry::GetGlyph(TUint aGlyphCode, TGlyphImageInfo& aGlyphImageInfo)
hgs
parents:
diff changeset
   574
	{
hgs
parents:
diff changeset
   575
	TInt index = iPageArray.FindInUnsignedKeyOrder(TPageMap(aGlyphCode, NULL));
hgs
parents:
diff changeset
   576
	if (KErrNotFound == index)
hgs
parents:
diff changeset
   577
		{
hgs
parents:
diff changeset
   578
		return KErrNotFound;
hgs
parents:
diff changeset
   579
		}
hgs
parents:
diff changeset
   580
	CGlyphAtlasPage* page = iPageArray[index].iPage;
hgs
parents:
diff changeset
   581
	page->GetGlyph(aGlyphCode, aGlyphImageInfo);
hgs
parents:
diff changeset
   582
	iAtlas.MovePageToFront(*page);
hgs
parents:
diff changeset
   583
	return KErrNone;
hgs
parents:
diff changeset
   584
	}
hgs
parents:
diff changeset
   585
hgs
parents:
diff changeset
   586
/**
hgs
parents:
diff changeset
   587
Gets the font associated with the font entry.
hgs
parents:
diff changeset
   588
hgs
parents:
diff changeset
   589
@return The font associated with this entry. 
hgs
parents:
diff changeset
   590
*/
hgs
parents:
diff changeset
   591
const CBitmapFont& CGlyphAtlasFontEntry::Font() const
hgs
parents:
diff changeset
   592
	{
hgs
parents:
diff changeset
   593
	return iFont;
hgs
parents:
diff changeset
   594
	}
hgs
parents:
diff changeset
   595
hgs
parents:
diff changeset
   596
/**
hgs
parents:
diff changeset
   597
Gets the amount of memory allocated for all the image data for this font.
hgs
parents:
diff changeset
   598
hgs
parents:
diff changeset
   599
@return The size of the font's image data in bytes.
hgs
parents:
diff changeset
   600
*/
hgs
parents:
diff changeset
   601
TInt CGlyphAtlasFontEntry::SizeInBytes() const
hgs
parents:
diff changeset
   602
	{
hgs
parents:
diff changeset
   603
	return iSizeInBytes;
hgs
parents:
diff changeset
   604
	}
hgs
parents:
diff changeset
   605
hgs
parents:
diff changeset
   606
/**
hgs
parents:
diff changeset
   607
Tests whether the font entry has any pages.
hgs
parents:
diff changeset
   608
hgs
parents:
diff changeset
   609
@return ETrue if the font entry does not contain any pages, EFalse if it does..
hgs
parents:
diff changeset
   610
*/
hgs
parents:
diff changeset
   611
TBool CGlyphAtlasFontEntry::IsEmpty() const
hgs
parents:
diff changeset
   612
	{
hgs
parents:
diff changeset
   613
	if (iPageArray.Count() == 0)
hgs
parents:
diff changeset
   614
		{
hgs
parents:
diff changeset
   615
		__ASSERT_DEBUG(iSizeInBytes == 0, Panic(EFbsPanicGlyphAtlasInconsistentState));
hgs
parents:
diff changeset
   616
		return ETrue;
hgs
parents:
diff changeset
   617
		}
hgs
parents:
diff changeset
   618
	return EFalse;
hgs
parents:
diff changeset
   619
	}
hgs
parents:
diff changeset
   620
hgs
parents:
diff changeset
   621
/**
hgs
parents:
diff changeset
   622
Gets the glyph atlas the font entry belongs to.
hgs
parents:
diff changeset
   623
hgs
parents:
diff changeset
   624
@return The font entry's glyph atlas.
hgs
parents:
diff changeset
   625
*/
hgs
parents:
diff changeset
   626
CGlyphAtlas& CGlyphAtlasFontEntry::GlyphAtlas() const
hgs
parents:
diff changeset
   627
	{
hgs
parents:
diff changeset
   628
	return iAtlas;
hgs
parents:
diff changeset
   629
	}
hgs
parents:
diff changeset
   630
hgs
parents:
diff changeset
   631
/** 
hgs
parents:
diff changeset
   632
@return The number of glyphs this font entry has.
hgs
parents:
diff changeset
   633
 */
hgs
parents:
diff changeset
   634
TInt CGlyphAtlasFontEntry::GlyphCount() const
hgs
parents:
diff changeset
   635
	{
hgs
parents:
diff changeset
   636
	TInt glyphCount = 0;
hgs
parents:
diff changeset
   637
	for (TInt ii = iPageArray.Count() - 1; ii >= 0; --ii)
hgs
parents:
diff changeset
   638
		{
hgs
parents:
diff changeset
   639
		glyphCount += iPageArray[ii].iPage->GlyphCount();
hgs
parents:
diff changeset
   640
		}
hgs
parents:
diff changeset
   641
	return glyphCount;
hgs
parents:
diff changeset
   642
	}
hgs
parents:
diff changeset
   643
hgs
parents:
diff changeset
   644
hgs
parents:
diff changeset
   645
// === CGlyphAtlasPage Functions ===
hgs
parents:
diff changeset
   646
hgs
parents:
diff changeset
   647
/**
hgs
parents:
diff changeset
   648
Page constructor.
hgs
parents:
diff changeset
   649
hgs
parents:
diff changeset
   650
@param aFontEntry The font entry to which the page is associated.
hgs
parents:
diff changeset
   651
*/
hgs
parents:
diff changeset
   652
CGlyphAtlasPage::CGlyphAtlasPage(CGlyphAtlasFontEntry& aFontEntry)
hgs
parents:
diff changeset
   653
	:iFontEntry(aFontEntry)
hgs
parents:
diff changeset
   654
	{
hgs
parents:
diff changeset
   655
	}
hgs
parents:
diff changeset
   656
hgs
parents:
diff changeset
   657
/**
hgs
parents:
diff changeset
   658
Page destructor.
hgs
parents:
diff changeset
   659
Releases the RSgImage handles held by the page.
hgs
parents:
diff changeset
   660
Removes the page from the Glyph Atlas' LRU page array. 
hgs
parents:
diff changeset
   661
*/
hgs
parents:
diff changeset
   662
CGlyphAtlasPage::~CGlyphAtlasPage()
hgs
parents:
diff changeset
   663
	{
hgs
parents:
diff changeset
   664
	iLink.Deque();
hgs
parents:
diff changeset
   665
	iGlyphImage.Close();
hgs
parents:
diff changeset
   666
	}
hgs
parents:
diff changeset
   667
hgs
parents:
diff changeset
   668
/**
hgs
parents:
diff changeset
   669
Adds a glyph to the page from a bitmap glyph and gets the glyph image info back.
hgs
parents:
diff changeset
   670
An RSgImage handle is acquired for the glyph. 
hgs
parents:
diff changeset
   671
If there is not enough specialised graphics memory to create a RSgImage, then the
hgs
parents:
diff changeset
   672
least recently used pages are deleted until there there is either enough memory 
hgs
parents:
diff changeset
   673
for the creation to be successful or if there are no more pages to delete (in 
hgs
parents:
diff changeset
   674
which case an appropriate out of memory error message is returned). 
hgs
parents:
diff changeset
   675
hgs
parents:
diff changeset
   676
@param[in] aArgs The information needed to create a glyph.
hgs
parents:
diff changeset
   677
@param[out] aGlyphImageInfo Upon return contains all the glyph image information needed to use the 
hgs
parents:
diff changeset
   678
		glyph in another process.
hgs
parents:
diff changeset
   679
@param[out] aSizeInBytes Upon return contains the size of the added glyph's image data in bytes.
hgs
parents:
diff changeset
   680
@return KErrNone if the glyph was successfully added; 
hgs
parents:
diff changeset
   681
	KErrNoMemory if there is not enough system memory available;
hgs
parents:
diff changeset
   682
	KErrNoGraphicsMemory if there is not enough specialised graphics memory available.
hgs
parents:
diff changeset
   683
*/
hgs
parents:
diff changeset
   684
TInt CGlyphAtlasPage::AddGlyph(const CGlyphAtlas::TAddGlyphArgs& aArgs, TGlyphImageInfo& aGlyphImageInfo, TInt& aSizeInBytes)
hgs
parents:
diff changeset
   685
	{
hgs
parents:
diff changeset
   686
	const TSize glyphSize(aArgs.iMetrics->Width(), aArgs.iMetrics->Height());
hgs
parents:
diff changeset
   687
	// If glyph has zero size (e.g. space), set glyph data and return 
hgs
parents:
diff changeset
   688
	if (glyphSize.iWidth == 0 || glyphSize.iHeight == 0)
hgs
parents:
diff changeset
   689
		{
hgs
parents:
diff changeset
   690
		iPosX = 0;
hgs
parents:
diff changeset
   691
		iPosY = 0;
hgs
parents:
diff changeset
   692
		iMetrics = *aArgs.iMetrics;
hgs
parents:
diff changeset
   693
		iGlyphCode = aArgs.iGlyphCode;
hgs
parents:
diff changeset
   694
		iSizeInBytes = 0;
hgs
parents:
diff changeset
   695
		iNumGlyphs++;
hgs
parents:
diff changeset
   696
		aGlyphImageInfo.iImageId = KSgNullDrawableId;
hgs
parents:
diff changeset
   697
		aGlyphImageInfo.iPosX = iPosX;
hgs
parents:
diff changeset
   698
		aGlyphImageInfo.iPosY = iPosY;
hgs
parents:
diff changeset
   699
		aGlyphImageInfo.iMetrics = iMetrics;
hgs
parents:
diff changeset
   700
		return KErrNone;
hgs
parents:
diff changeset
   701
		}
hgs
parents:
diff changeset
   702
	TUint8* buf = NULL;
hgs
parents:
diff changeset
   703
	TSgImageInfo info;
hgs
parents:
diff changeset
   704
	info.iSizeInPixels = glyphSize;
hgs
parents:
diff changeset
   705
	info.iUsage = ESgUsageBitOpenVgImage;
hgs
parents:
diff changeset
   706
	info.iPixelFormat = EUidPixelFormatA_8;
hgs
parents:
diff changeset
   707
	TInt dataStride = 0;
hgs
parents:
diff changeset
   708
	const TInt KDataArraySize = 256;
hgs
parents:
diff changeset
   709
	TUint8 byteDataArray[KDataArraySize];
hgs
parents:
diff changeset
   710
	TUint8* tempBuf = NULL;
hgs
parents:
diff changeset
   711
	TUint8* byteDataBuf = NULL;
hgs
parents:
diff changeset
   712
	TGlyphBitmapType glyphBitmapType = iFontEntry.Font().GlyphBitmapType(); 
hgs
parents:
diff changeset
   713
hgs
parents:
diff changeset
   714
	switch (glyphBitmapType)
hgs
parents:
diff changeset
   715
		{
hgs
parents:
diff changeset
   716
		case EMonochromeGlyphBitmap:
hgs
parents:
diff changeset
   717
			// Decompress to 8bpp buffer
hgs
parents:
diff changeset
   718
			dataStride = glyphSize.iWidth;
hgs
parents:
diff changeset
   719
			byteDataBuf = byteDataArray;
hgs
parents:
diff changeset
   720
			// If data too big to fit in byteDataArray, allocate memory on the heap
hgs
parents:
diff changeset
   721
			if (glyphSize.iHeight * glyphSize.iWidth > KDataArraySize)
hgs
parents:
diff changeset
   722
				{
hgs
parents:
diff changeset
   723
				tempBuf = (TUint8*) User::AllocZ(dataStride * glyphSize.iHeight);
hgs
parents:
diff changeset
   724
				if (!tempBuf)
hgs
parents:
diff changeset
   725
					{
hgs
parents:
diff changeset
   726
					return KErrNoMemory;
hgs
parents:
diff changeset
   727
					}
hgs
parents:
diff changeset
   728
				byteDataBuf = tempBuf;
hgs
parents:
diff changeset
   729
				}
hgs
parents:
diff changeset
   730
			else
hgs
parents:
diff changeset
   731
				{
hgs
parents:
diff changeset
   732
				// fill array with zeros.
hgs
parents:
diff changeset
   733
				Mem::FillZ(&byteDataArray, KDataArraySize);
hgs
parents:
diff changeset
   734
				}
hgs
parents:
diff changeset
   735
			
hgs
parents:
diff changeset
   736
			if (glyphSize.iWidth >32)
hgs
parents:
diff changeset
   737
				{
hgs
parents:
diff changeset
   738
				DecodeBinaryDataExLarge(glyphSize, aArgs.iBitmapPointer, byteDataBuf);
hgs
parents:
diff changeset
   739
				}
hgs
parents:
diff changeset
   740
			else
hgs
parents:
diff changeset
   741
				{
hgs
parents:
diff changeset
   742
				DecodeBinaryData(glyphSize, aArgs.iBitmapPointer, byteDataBuf);
hgs
parents:
diff changeset
   743
				}
hgs
parents:
diff changeset
   744
			buf = byteDataBuf;
hgs
parents:
diff changeset
   745
			break;
hgs
parents:
diff changeset
   746
		case EAntiAliasedGlyphBitmap:
hgs
parents:
diff changeset
   747
			buf = const_cast<TUint8*>(aArgs.iBitmapPointer);
hgs
parents:
diff changeset
   748
			dataStride = glyphSize.iWidth;
hgs
parents:
diff changeset
   749
			break;
hgs
parents:
diff changeset
   750
		default:
hgs
parents:
diff changeset
   751
			return KErrNotSupported;
hgs
parents:
diff changeset
   752
		}
hgs
parents:
diff changeset
   753
hgs
parents:
diff changeset
   754
	TInt err = iGlyphImage.Create(info, buf, dataStride);
hgs
parents:
diff changeset
   755
hgs
parents:
diff changeset
   756
	// If RSgImage creation fails due to out of memory, delete the least
hgs
parents:
diff changeset
   757
	// recently used pages to free up memory until either creation succeeds or 
hgs
parents:
diff changeset
   758
	// there are no more pages to remove.
hgs
parents:
diff changeset
   759
	TBool morePagesToDelete = ETrue;
hgs
parents:
diff changeset
   760
	while ((KErrNoGraphicsMemory == err || KErrNoMemory == err) && morePagesToDelete)
hgs
parents:
diff changeset
   761
		{
hgs
parents:
diff changeset
   762
		// Delete least used page.  Can delete all pages if necessary as this  
hgs
parents:
diff changeset
   763
		// page has not been added to the LRU array yet.
hgs
parents:
diff changeset
   764
		morePagesToDelete = iFontEntry.GlyphAtlas().DeleteLeastRecentlyUsedPage(ETrue);
hgs
parents:
diff changeset
   765
		err = iGlyphImage.Create(info, buf, dataStride);
hgs
parents:
diff changeset
   766
		}
hgs
parents:
diff changeset
   767
hgs
parents:
diff changeset
   768
	User::Free(tempBuf);
hgs
parents:
diff changeset
   769
	
hgs
parents:
diff changeset
   770
	if (KErrNone != err)
hgs
parents:
diff changeset
   771
		{
hgs
parents:
diff changeset
   772
		return err;
hgs
parents:
diff changeset
   773
		}
hgs
parents:
diff changeset
   774
	aSizeInBytes = glyphSize.iHeight * glyphSize.iWidth;
hgs
parents:
diff changeset
   775
	iGlyphCode = aArgs.iGlyphCode;
hgs
parents:
diff changeset
   776
	iPosX = 0;
hgs
parents:
diff changeset
   777
	iPosY = 0;
hgs
parents:
diff changeset
   778
	iMetrics = *aArgs.iMetrics;
hgs
parents:
diff changeset
   779
	// As the image is stored as one byte per pixel, the size in bytes is 
hgs
parents:
diff changeset
   780
	// just the number of pixels.
hgs
parents:
diff changeset
   781
	// TODO: Replace estimating size with call to SgImage/SgDriver to get accurate size.
hgs
parents:
diff changeset
   782
	iSizeInBytes += aSizeInBytes;
hgs
parents:
diff changeset
   783
	iNumGlyphs++;
hgs
parents:
diff changeset
   784
	aGlyphImageInfo.iPosX = iPosX;
hgs
parents:
diff changeset
   785
	aGlyphImageInfo.iPosY = iPosY;
hgs
parents:
diff changeset
   786
	aGlyphImageInfo.iImageId = iGlyphImage.Id();
hgs
parents:
diff changeset
   787
	aGlyphImageInfo.iMetrics = iMetrics;
hgs
parents:
diff changeset
   788
	return err;
hgs
parents:
diff changeset
   789
	}
hgs
parents:
diff changeset
   790
hgs
parents:
diff changeset
   791
/**
hgs
parents:
diff changeset
   792
Retrieves the glyph image information for the given glyph code necessary to be 
hgs
parents:
diff changeset
   793
able to use the glyph in another process.
hgs
parents:
diff changeset
   794
hgs
parents:
diff changeset
   795
@param aGlyphCode The glyph code for the glyph being requested
hgs
parents:
diff changeset
   796
@param aGlyphImageInfo Upon return contains all the glyph image information needed to use the 
hgs
parents:
diff changeset
   797
		glyph in another process if the glyph is contained in the page.
hgs
parents:
diff changeset
   798
*/
hgs
parents:
diff changeset
   799
void CGlyphAtlasPage::GetGlyph(TUint aGlyphCode, TGlyphImageInfo& aGlyphImageInfo) const
hgs
parents:
diff changeset
   800
	{
hgs
parents:
diff changeset
   801
	__ASSERT_DEBUG(iGlyphCode == aGlyphCode, Panic(EFbsPanicGlyphAtlasInconsistentState));
hgs
parents:
diff changeset
   802
	aGlyphImageInfo.iMetrics = iMetrics;
hgs
parents:
diff changeset
   803
	aGlyphImageInfo.iPosX = iPosX;
hgs
parents:
diff changeset
   804
	aGlyphImageInfo.iPosY = iPosY;
hgs
parents:
diff changeset
   805
	aGlyphImageInfo.iImageId = iGlyphImage.Id();
hgs
parents:
diff changeset
   806
	}
hgs
parents:
diff changeset
   807
hgs
parents:
diff changeset
   808
/**
hgs
parents:
diff changeset
   809
Gets the amount of memory allocated for the image data for this page.
hgs
parents:
diff changeset
   810
hgs
parents:
diff changeset
   811
@return The size of the page's image data in bytes.
hgs
parents:
diff changeset
   812
*/
hgs
parents:
diff changeset
   813
TInt CGlyphAtlasPage::SizeInBytes() const
hgs
parents:
diff changeset
   814
	{
hgs
parents:
diff changeset
   815
	return iSizeInBytes;
hgs
parents:
diff changeset
   816
	}
hgs
parents:
diff changeset
   817
hgs
parents:
diff changeset
   818
/**
hgs
parents:
diff changeset
   819
Gets the glyph code at the given index associated with the page.
hgs
parents:
diff changeset
   820
hgs
parents:
diff changeset
   821
@param aIndex The index of the glyph code within the page.
hgs
parents:
diff changeset
   822
@return The glyph code at the given index.
hgs
parents:
diff changeset
   823
*/
hgs
parents:
diff changeset
   824
TUint CGlyphAtlasPage::GlyphCodeAt(TInt aIndex) const
hgs
parents:
diff changeset
   825
	{
hgs
parents:
diff changeset
   826
	__ASSERT_DEBUG(0 == aIndex, Panic(EFbsPanicGlyphAtlasInconsistentState));
hgs
parents:
diff changeset
   827
	return iGlyphCode;
hgs
parents:
diff changeset
   828
	}
hgs
parents:
diff changeset
   829
hgs
parents:
diff changeset
   830
/**
hgs
parents:
diff changeset
   831
Gets the number of glyphs stored in the page.
hgs
parents:
diff changeset
   832
hgs
parents:
diff changeset
   833
@return The number of glyphs in the page.
hgs
parents:
diff changeset
   834
*/
hgs
parents:
diff changeset
   835
TInt CGlyphAtlasPage::GlyphCount() const
hgs
parents:
diff changeset
   836
	{
hgs
parents:
diff changeset
   837
	return iNumGlyphs;
hgs
parents:
diff changeset
   838
	}
hgs
parents:
diff changeset
   839
hgs
parents:
diff changeset
   840
/**
hgs
parents:
diff changeset
   841
Gets the font entry which owns the page.
hgs
parents:
diff changeset
   842
hgs
parents:
diff changeset
   843
@return The font entry which owns the page. 
hgs
parents:
diff changeset
   844
*/
hgs
parents:
diff changeset
   845
CGlyphAtlasFontEntry& CGlyphAtlasPage::FontEntry() const
hgs
parents:
diff changeset
   846
	{
hgs
parents:
diff changeset
   847
	return iFontEntry;
hgs
parents:
diff changeset
   848
	}
hgs
parents:
diff changeset
   849
hgs
parents:
diff changeset
   850
hgs
parents:
diff changeset
   851
void CGlyphAtlasPage::MoveToFirstInQueue(TDblQue<CGlyphAtlasPage>& aList)
hgs
parents:
diff changeset
   852
	{
hgs
parents:
diff changeset
   853
	if(!aList.IsFirst(this))
hgs
parents:
diff changeset
   854
		{
hgs
parents:
diff changeset
   855
		iLink.Deque();
hgs
parents:
diff changeset
   856
		aList.AddFirst(*this);
hgs
parents:
diff changeset
   857
		}
hgs
parents:
diff changeset
   858
	}
hgs
parents:
diff changeset
   859
hgs
parents:
diff changeset
   860
// === Static Utility Functions ===
hgs
parents:
diff changeset
   861
hgs
parents:
diff changeset
   862
/**
hgs
parents:
diff changeset
   863
Combines 2 8-bit unsigned integers into a 16-bit integer.
hgs
parents:
diff changeset
   864
@param aPtr A pointer to a source buffer of 2 8-bit unsigned integers.
hgs
parents:
diff changeset
   865
@return The two 8-bit integers combined into a 16-bit integer.
hgs
parents:
diff changeset
   866
*/
hgs
parents:
diff changeset
   867
static TInt16 Load16(const TUint8* aPtr)
hgs
parents:
diff changeset
   868
	{
hgs
parents:
diff changeset
   869
	return TInt16(aPtr[0]+(aPtr[1]<<8)); 
hgs
parents:
diff changeset
   870
	}
hgs
parents:
diff changeset
   871
hgs
parents:
diff changeset
   872
/**
hgs
parents:
diff changeset
   873
Decodes binary data for monochrome glyph bitmap.
hgs
parents:
diff changeset
   874
hgs
parents:
diff changeset
   875
@param aGlyphSize size of glyph in pixels.
hgs
parents:
diff changeset
   876
@param aEncodedData Pointer to an encoded source buffer.
hgs
parents:
diff changeset
   877
@param aByteData Pointer to a destination buffer (8 bits per pixel).
hgs
parents:
diff changeset
   878
*/
hgs
parents:
diff changeset
   879
void DecodeBinaryData(const TSize& aGlyphSize, const TUint8* aEncodedData, TUint8* aByteData)
hgs
parents:
diff changeset
   880
	{
hgs
parents:
diff changeset
   881
	const TInt dataHeight = aGlyphSize.iHeight;
hgs
parents:
diff changeset
   882
	const TInt dataWidth = aGlyphSize.iWidth;
hgs
parents:
diff changeset
   883
	TUint32 binaryData = 0;
hgs
parents:
diff changeset
   884
	
hgs
parents:
diff changeset
   885
	// The data is encoded as follows:
hgs
parents:
diff changeset
   886
	// 1 bit for a multiple lines flag (1=yes)
hgs
parents:
diff changeset
   887
	// 4 bits for a repeat count which represents:
hgs
parents:
diff changeset
   888
	// -if the multiple line flag is 0 the number of lines whose data is repeated
hgs
parents:
diff changeset
   889
	// -if the flag is 1, the number of lines which differ from line to line.
hgs
parents:
diff changeset
   890
	// n bits representing the data at 1 bit per pixel, where:
hgs
parents:
diff changeset
   891
	// -if the multiple line flag is 0, n is the width of the glyph.
hgs
parents:
diff changeset
   892
	// -if the flag is 1, n is width of glyph multiplied by the repeat count for this block of data.
hgs
parents:
diff changeset
   893
	// This information presented in continuous packed blocks of:
hgs
parents:
diff changeset
   894
	// [data][reps][multiLineFlag]
hgs
parents:
diff changeset
   895
	TInt bitIndex = 0;
hgs
parents:
diff changeset
   896
	TInt16 repeatCount = 0;
hgs
parents:
diff changeset
   897
	TUint8* byteDataPtr = aByteData;
hgs
parents:
diff changeset
   898
	TUint8* byteDataPtrLimit = NULL;
hgs
parents:
diff changeset
   899
	for (TInt charLine = 0; charLine < dataHeight; charLine += repeatCount) // for lines in the character...
hgs
parents:
diff changeset
   900
		{
hgs
parents:
diff changeset
   901
		// Get first 5 bits of block
hgs
parents:
diff changeset
   902
		repeatCount = Load16(aEncodedData + (bitIndex >> 3));
hgs
parents:
diff changeset
   903
		repeatCount >>= bitIndex & 7;
hgs
parents:
diff changeset
   904
		// strip out multiple line flag (1st bit)
hgs
parents:
diff changeset
   905
		TInt multiLineFlag = repeatCount & 1;
hgs
parents:
diff changeset
   906
		// Get repeat count (last 4 bits)
hgs
parents:
diff changeset
   907
		repeatCount >>= 1;
hgs
parents:
diff changeset
   908
		repeatCount &= 0xf;
hgs
parents:
diff changeset
   909
		// move bit index to point to first bit of image data
hgs
parents:
diff changeset
   910
		bitIndex += 5;
hgs
parents:
diff changeset
   911
		// end pointer of destination buffer for this block of data to fill 
hgs
parents:
diff changeset
   912
		byteDataPtrLimit = aByteData + dataWidth * (charLine + repeatCount);
hgs
parents:
diff changeset
   913
		if (multiLineFlag)
hgs
parents:
diff changeset
   914
			{
hgs
parents:
diff changeset
   915
			while (byteDataPtr < byteDataPtrLimit)
hgs
parents:
diff changeset
   916
				{
hgs
parents:
diff changeset
   917
				// Pointer to beginning of data in source buffer for current scanline
hgs
parents:
diff changeset
   918
				TInt charDataOffsetPtr = TInt(aEncodedData) + (bitIndex >> 3);
hgs
parents:
diff changeset
   919
				// Pointer to beginning of current word.
hgs
parents:
diff changeset
   920
				TUint32* charDataWord = (TUint32*)(charDataOffsetPtr &~ 3);
hgs
parents:
diff changeset
   921
				// Number of bits to shift in current word to get to beginning of scanline
hgs
parents:
diff changeset
   922
				TInt bitShift = bitIndex & 7;
hgs
parents:
diff changeset
   923
				bitShift += (charDataOffsetPtr & 3) << 3;
hgs
parents:
diff changeset
   924
				// Copy scanline data into temporary buffer
hgs
parents:
diff changeset
   925
				binaryData = (*charDataWord++) >> bitShift;
hgs
parents:
diff changeset
   926
				// If data crosses a word boundary, get the rest of the data from next word.
hgs
parents:
diff changeset
   927
				if (bitShift)
hgs
parents:
diff changeset
   928
					{
hgs
parents:
diff changeset
   929
					binaryData |= (*charDataWord << (32-bitShift));
hgs
parents:
diff changeset
   930
					}
hgs
parents:
diff changeset
   931
				Convert1BppTo8Bpp(binaryData, byteDataPtr, byteDataPtr + dataWidth);
hgs
parents:
diff changeset
   932
				// Move bit index to beginning of next block
hgs
parents:
diff changeset
   933
				bitIndex += dataWidth;
hgs
parents:
diff changeset
   934
				}
hgs
parents:
diff changeset
   935
			}
hgs
parents:
diff changeset
   936
		else
hgs
parents:
diff changeset
   937
			{
hgs
parents:
diff changeset
   938
			TInt charDataOffsetPtr = TInt(aEncodedData) + (bitIndex >> 3);
hgs
parents:
diff changeset
   939
			TUint32* charDataWord = (TUint32*)(charDataOffsetPtr &~ 3);
hgs
parents:
diff changeset
   940
			TInt bitShift = bitIndex & 7;
hgs
parents:
diff changeset
   941
			bitShift += (charDataOffsetPtr & 3) << 3;
hgs
parents:
diff changeset
   942
			binaryData = (*charDataWord++) >> bitShift;
hgs
parents:
diff changeset
   943
			if (bitShift)
hgs
parents:
diff changeset
   944
				{
hgs
parents:
diff changeset
   945
				binaryData |= (*charDataWord << (32-bitShift));
hgs
parents:
diff changeset
   946
				}
hgs
parents:
diff changeset
   947
			TUint8* startByteDataPtr = byteDataPtr;
hgs
parents:
diff changeset
   948
			Convert1BppTo8Bpp(binaryData, byteDataPtr, byteDataPtr + dataWidth);
hgs
parents:
diff changeset
   949
			
hgs
parents:
diff changeset
   950
			while (byteDataPtr < byteDataPtrLimit)
hgs
parents:
diff changeset
   951
				{
hgs
parents:
diff changeset
   952
				Mem::Copy(byteDataPtr, startByteDataPtr, dataWidth);
hgs
parents:
diff changeset
   953
				byteDataPtr += dataWidth;
hgs
parents:
diff changeset
   954
				}
hgs
parents:
diff changeset
   955
			bitIndex += dataWidth;
hgs
parents:
diff changeset
   956
			}
hgs
parents:
diff changeset
   957
		}
hgs
parents:
diff changeset
   958
	}
hgs
parents:
diff changeset
   959
hgs
parents:
diff changeset
   960
/**
hgs
parents:
diff changeset
   961
Converts binary data in 1 bit per pixel format to 8 bits per pixel format, where
hgs
parents:
diff changeset
   962
0 is converted to 0x00 and 1 is converted to 0xFF.
hgs
parents:
diff changeset
   963
hgs
parents:
diff changeset
   964
@param aSrcData Pointer to a 1bpp source buffer.
hgs
parents:
diff changeset
   965
@param aDestDataPtr Pointer to a 8bpp destination buffer.
hgs
parents:
diff changeset
   966
@param aDestDataPtrLimit Pointer to the end position in destination buffer to convert to.
hgs
parents:
diff changeset
   967
*/
hgs
parents:
diff changeset
   968
void Convert1BppTo8Bpp(TUint32 aSrcData, TUint8*& aDestDataPtr, const TUint8* aDestDataPtrLimit)
hgs
parents:
diff changeset
   969
	{
hgs
parents:
diff changeset
   970
	for (; aDestDataPtr < aDestDataPtrLimit; ++aDestDataPtr, aSrcData >>= 1)
hgs
parents:
diff changeset
   971
		{
hgs
parents:
diff changeset
   972
		if (aSrcData&1)
hgs
parents:
diff changeset
   973
			{
hgs
parents:
diff changeset
   974
			*aDestDataPtr = 0xFF;
hgs
parents:
diff changeset
   975
			}
hgs
parents:
diff changeset
   976
		}
hgs
parents:
diff changeset
   977
	}
hgs
parents:
diff changeset
   978
hgs
parents:
diff changeset
   979
/**
hgs
parents:
diff changeset
   980
Decodes binary data for extra large monochrome glyph bitmap.
hgs
parents:
diff changeset
   981
hgs
parents:
diff changeset
   982
@param aGlyphSize Size of glyph in pixels.
hgs
parents:
diff changeset
   983
@param aEncodedData Pointer to an encoded source buffer.
hgs
parents:
diff changeset
   984
@param aByteData Pointer to a destination buffer (8 bits per pixel).
hgs
parents:
diff changeset
   985
*/
hgs
parents:
diff changeset
   986
void DecodeBinaryDataExLarge(const TSize& aGlyphSize, const TUint8* aEncodedData, TUint8* aByteData)
hgs
parents:
diff changeset
   987
	{
hgs
parents:
diff changeset
   988
	const TInt dataWidth = aGlyphSize.iWidth;
hgs
parents:
diff changeset
   989
	const TInt dataHeight = aGlyphSize.iHeight;
hgs
parents:
diff changeset
   990
	TInt bitIndex = 0;
hgs
parents:
diff changeset
   991
	TInt16 repeatCount = 0;
hgs
parents:
diff changeset
   992
hgs
parents:
diff changeset
   993
	for (TInt charLine = 0; charLine < dataHeight; charLine += repeatCount) // for lines in the character...
hgs
parents:
diff changeset
   994
		{
hgs
parents:
diff changeset
   995
		repeatCount = Load16(aEncodedData + (bitIndex >> 3));
hgs
parents:
diff changeset
   996
		repeatCount >>= bitIndex & 7;
hgs
parents:
diff changeset
   997
		const TInt multiLineFlag = repeatCount & 1;
hgs
parents:
diff changeset
   998
		repeatCount >>= 1;
hgs
parents:
diff changeset
   999
		repeatCount &= 0xf;
hgs
parents:
diff changeset
  1000
		bitIndex += 5;
hgs
parents:
diff changeset
  1001
		if (multiLineFlag)
hgs
parents:
diff changeset
  1002
			{
hgs
parents:
diff changeset
  1003
			for (TInt currentline = 0; currentline < repeatCount; currentline++)
hgs
parents:
diff changeset
  1004
				{
hgs
parents:
diff changeset
  1005
				CopyCharLine(aByteData, dataWidth, aEncodedData + (bitIndex >> 3), bitIndex & 7, 1);
hgs
parents:
diff changeset
  1006
				bitIndex += dataWidth;
hgs
parents:
diff changeset
  1007
				}
hgs
parents:
diff changeset
  1008
			}
hgs
parents:
diff changeset
  1009
		else
hgs
parents:
diff changeset
  1010
			{
hgs
parents:
diff changeset
  1011
			CopyCharLine(aByteData, dataWidth, aEncodedData + (bitIndex >> 3), bitIndex & 7, repeatCount);
hgs
parents:
diff changeset
  1012
			bitIndex += dataWidth;
hgs
parents:
diff changeset
  1013
			}
hgs
parents:
diff changeset
  1014
		}
hgs
parents:
diff changeset
  1015
	}
hgs
parents:
diff changeset
  1016
hgs
parents:
diff changeset
  1017
/**
hgs
parents:
diff changeset
  1018
Copies glyph image data line(s)(1 bit per pixel) to an 8 bit per pixel
hgs
parents:
diff changeset
  1019
destination buffer.
hgs
parents:
diff changeset
  1020
hgs
parents:
diff changeset
  1021
@param aByteDataPtr Pointer to a destination buffer (8bpp).
hgs
parents:
diff changeset
  1022
@param aWidthInBytes Stride of the image.
hgs
parents:
diff changeset
  1023
@param aSrcData Pointer to a source buffer (1bpp).
hgs
parents:
diff changeset
  1024
@param aBitShift Number of bits the source data pointer will be shifted. 
hgs
parents:
diff changeset
  1025
@param aRepeatCount Number of lines to copy.
hgs
parents:
diff changeset
  1026
*/
hgs
parents:
diff changeset
  1027
void CopyCharLine(TUint8*& aByteDataPtr, TInt aWidthInBytes, const TUint8* aSrcData, TInt aBitShift, TInt16 aRepeatCount)
hgs
parents:
diff changeset
  1028
	{
hgs
parents:
diff changeset
  1029
	aBitShift &= 7;
hgs
parents:
diff changeset
  1030
	TUint8* ptrLimit = aByteDataPtr + aWidthInBytes;
hgs
parents:
diff changeset
  1031
	TUint32* dataWord = (TUint32*)(TInt(aSrcData) &~ 3);
hgs
parents:
diff changeset
  1032
	aBitShift += (TInt(aSrcData) - TInt(dataWord)) << 3;
hgs
parents:
diff changeset
  1033
hgs
parents:
diff changeset
  1034
	TUint8* startByteDataPtr = aByteDataPtr;
hgs
parents:
diff changeset
  1035
	TUint32 binaryData = 0;
hgs
parents:
diff changeset
  1036
	while (aByteDataPtr < ptrLimit)
hgs
parents:
diff changeset
  1037
		{
hgs
parents:
diff changeset
  1038
		binaryData = *dataWord++;
hgs
parents:
diff changeset
  1039
		binaryData >>= aBitShift;
hgs
parents:
diff changeset
  1040
		if (aBitShift)
hgs
parents:
diff changeset
  1041
			{
hgs
parents:
diff changeset
  1042
			binaryData |= (*dataWord << (32-aBitShift));
hgs
parents:
diff changeset
  1043
			}
hgs
parents:
diff changeset
  1044
		TUint8* wordLimit = aByteDataPtr + 32;
hgs
parents:
diff changeset
  1045
		if (wordLimit > ptrLimit)
hgs
parents:
diff changeset
  1046
			{
hgs
parents:
diff changeset
  1047
			wordLimit = ptrLimit;
hgs
parents:
diff changeset
  1048
			}
hgs
parents:
diff changeset
  1049
		Convert1BppTo8Bpp(binaryData, aByteDataPtr, wordLimit);
hgs
parents:
diff changeset
  1050
		}
hgs
parents:
diff changeset
  1051
hgs
parents:
diff changeset
  1052
	while (aRepeatCount > 1)
hgs
parents:
diff changeset
  1053
		{
hgs
parents:
diff changeset
  1054
		Mem::Copy(aByteDataPtr, startByteDataPtr, aWidthInBytes);
hgs
parents:
diff changeset
  1055
		aByteDataPtr += aWidthInBytes;
hgs
parents:
diff changeset
  1056
		--aRepeatCount;
hgs
parents:
diff changeset
  1057
		}
hgs
parents:
diff changeset
  1058
	}
hgs
parents:
diff changeset
  1059
hgs
parents:
diff changeset
  1060