os/ossrv/lowlevellibsandfws/apputils/src/StringPoolImplementation.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 2001-2009 Nokia Corporation and/or its subsidiary(-ies).
     2 // All rights reserved.
     3 // This component and the accompanying materials are made available
     4 // under the terms of "Eclipse Public License v1.0"
     5 // which accompanies this distribution, and is available
     6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 //
    15 
    16 #include <e32base.h>
    17 #include "StringPoolImplementation.h"
    18 
    19 const TInt KMapGranularity=20;
    20 
    21 CStringPoolNode::~CStringPoolNode()
    22 	{
    23 	delete iDes;
    24 	}
    25 
    26 CStringPoolImplementation::CStringPoolImplementation() : iStringMapList(KMapGranularity, _FOFF(TStringIdMap, iSourceTableVal)), iStringMapListReverse(KMapGranularity, _FOFF(TStringIdMap, iTargetTableVal))
    27 	{
    28 	}
    29 
    30 CStringPoolImplementation::~CStringPoolImplementation()
    31 	{
    32 	// Look for non-expirable strings
    33 	TInt i;
    34 	for (i = 0; i < KHashModulo; i++ )
    35 		{
    36 		DeleteUndeletableStrings(iCIHashTable, i);
    37 		DeleteUndeletableStrings(iCSHashTable, i);		
    38 		}
    39 #ifdef _DEBUG
    40 
    41 	__LOG(_L8(":Closing String Pool.\n"))
    42 		TBool leaksFound = EFalse;
    43 	// Check that the string pool is empty, or more accurately that
    44 	// everything in it is a pre-loaded string
    45 	for (i = 0; i < KHashModulo; i++ )
    46 		{
    47 		if (iCIHashTable[i])
    48 			{
    49 			for (TInt j = 0; j < iCIHashTable[i]->Count(); j++)
    50 				{
    51 				if (!StringUtils::IsTableEntry(iCIHashTable[i]->At(j).iVal))
    52 					{
    53 					if (!leaksFound)
    54 						{
    55 						__LOG(_L8("The following strings were leaked through not being Closed:\n"))
    56 						leaksFound = ETrue;
    57 						}
    58 
    59 					// Get the problem string
    60 					CStringPoolNode* theProblem = 
    61 						reinterpret_cast<CStringPoolNode*>(
    62 							iCIHashTable[i]->At(j).iVal & KTokenToNode);
    63 					if(theProblem->iDes)
    64 					    __LOG(theProblem->iDes->Des());
    65 					}
    66 				}
    67 			}
    68 		if (iCSHashTable[i])
    69 			{
    70 			for (TInt j = 0; j < iCSHashTable[i]->Count(); j++)
    71 				{
    72 				if (!StringUtils::IsTableEntry(iCSHashTable[i]->At(j).iVal))
    73 					{
    74 					if (!leaksFound)
    75 						{
    76 						__LOG(_L8("The following strings were leaked through not being Closed:\n"))
    77 						leaksFound = ETrue;
    78 						}
    79 
    80 					// Get the problem string
    81 					CStringPoolNode* theProblem = 
    82 						reinterpret_cast<CStringPoolNode*>(
    83 							iCSHashTable[i]->At(j).iVal & KTokenToNode);
    84 					if(theProblem->iDes)
    85 					    __LOG(theProblem->iDes->Des());
    86 					}
    87 				}
    88 			}
    89 		if (leaksFound)
    90 			__DEBUGGER();
    91 		}
    92 
    93 	if (!leaksFound)
    94 		__LOG(_L8("No leakages were detected\n"));
    95 
    96 #endif //_DEBUG
    97 	for (TInt ii = 0; ii < KHashModulo; ii++)
    98 		{
    99 		delete iCIHashTable[ii];
   100 		delete iCSHashTable[ii];
   101 		}
   102 		
   103 	iTablePtrs.Close();
   104 	iStringMapList.Close();
   105 	iStringMapListReverse.Close();
   106 	iRollbackMapList.Close();	
   107 	iRollbackHashListCS.Close();
   108 	iRollbackHashListCI.Close();
   109 	// Notify the external users of the StringPool that the object is getting closed
   110 	TInt cBCounter = iCallBacks.Count();
   111 	if(cBCounter>0)
   112 		{
   113 		while (--cBCounter>=0)
   114 			{
   115 			iCallBacks[cBCounter]->StringPoolClosing();
   116 			}
   117 		}
   118 	iCallBacks.Close();
   119 	}
   120 
   121 // Check for any undeletable string and delete them now
   122 void CStringPoolImplementation::DeleteUndeletableStrings(CArrayFixSeg<RStringTokenEither>* aArray[KHashModulo], TInt i)
   123 	{
   124 	if (aArray[i])
   125 		{
   126 		for (TInt j = 0; j < aArray[i]->Count(); ++j)
   127 			{
   128 			if (!StringUtils::IsTableEntry(aArray[i]->At(j).iVal))
   129 				{
   130 				CStringPoolNode* node= reinterpret_cast<CStringPoolNode*>(aArray[i]->At(j).iVal & KTokenToNode);
   131 				if (KMarkedForNoDeleted==node->iRefcount)
   132 					{
   133 					delete node;										
   134 					aArray[i]->Delete(j);
   135 					j--;
   136 					}
   137 				}
   138 			}
   139 		}	
   140 	}
   141 
   142 CStringPoolImplementation* CStringPoolImplementation::NewL()
   143 	{
   144 	CStringPoolImplementation* table = new (ELeave) CStringPoolImplementation();	
   145 	return table;
   146 	}
   147 
   148 void CStringPoolImplementation::CleanupHashCS(TAny* aImplementation)
   149 	{
   150 	CStringPoolImplementation* imp=reinterpret_cast<CStringPoolImplementation*>(aImplementation);
   151 	CleanUpHash(&imp->iRollbackHashListCS, imp->iCSHashTable);
   152 	}
   153 
   154 void CStringPoolImplementation::CleanupHashCI(TAny* aImplementation)
   155 	{
   156 	CStringPoolImplementation* imp=reinterpret_cast<CStringPoolImplementation*>(aImplementation);
   157 	CleanUpHash(&imp->iRollbackHashListCI, imp->iCIHashTable);
   158 	}
   159 
   160 void CStringPoolImplementation::CleanUpHash(RPointerArray <RStringTokenEither>* aHashCleanup, CArrayFixSeg<RStringTokenEither>* aHash[KHashModulo])
   161 	{
   162 	if (aHashCleanup->Count()>0)
   163 		{
   164 		RStringTokenEither* token=(*aHashCleanup)[0];	// Get first entry
   165 		for (TInt i = 0; i < KHashModulo; i++ )
   166 			{
   167 			if (aHash[i])
   168 				{
   169 				for (TInt j = 0; j < aHash[i]->Count(); j++)
   170 					{
   171 					if (!StringUtils::IsTableEntry(aHash[i]->At(j).iVal))
   172 						{						
   173 						if (aHash[i]->At(j).iVal==token->iVal)
   174 							{		
   175 							CStringPoolNode* node= reinterpret_cast<CStringPoolNode*>(aHash[i]->At(j).iVal & KTokenToNode);
   176 							delete node;
   177 							aHash[i]->Delete(j);
   178 							aHashCleanup->Remove(0);
   179 							break;
   180 							}
   181 						}
   182 					}
   183 				}
   184 			
   185 			}
   186 		}
   187 	}
   188 
   189 void CStringPoolImplementation::CleanupIdMap(TAny* aImplementation)
   190 	{
   191 	CStringPoolImplementation* imp=reinterpret_cast<CStringPoolImplementation*>(aImplementation);
   192 	if (imp->iRollbackMapList.Count()>0)
   193 		{
   194 		TStringIdMap* map=imp->iRollbackMapList[0];			
   195 		TInt index=imp->iStringMapList.FindInUnsignedKeyOrder(*map);
   196 		imp->iRollbackMapList.Remove(0);
   197 		if (index!=KErrNotFound)
   198 			{
   199 			imp->iStringMapList.Remove(index);			
   200 			}		
   201 		index=imp->iStringMapListReverse.FindInUnsignedKeyOrder(*map);
   202 		if (index!=KErrNotFound)
   203 			{
   204 			for (TInt count=index;count<imp->iStringMapListReverse.Count();++count)
   205 				{
   206 				if (imp->iStringMapListReverse[count].iTargetTableVal==map->iTargetTableVal && imp->iStringMapListReverse[count].iSourceTableVal==map->iSourceTableVal)
   207 					{				
   208 					imp->iStringMapListReverse.Remove(index);
   209 					}
   210 				}
   211 			}
   212 		}
   213 	}
   214 
   215 void CStringPoolImplementation::AddTableL(const TStringTable& aTable)
   216 	{
   217 	for (TInt count=0;count<iTablePtrs.Count();++count)	// check for adding the same table twice
   218 		{
   219 		if (iTablePtrs[count]==&aTable)
   220 			{
   221 			return;
   222 			}
   223 		}
   224 	User::LeaveIfError(iTablePtrs.Append(&aTable));	// Add the pointer to this table so we can keep track of the Table IDs. The table ID is the index in this array
   225 	//Is the table Case Sensitive or not?
   226 	TBool (*genericValFromIndex)(TInt, TUint16);
   227 	if (aTable.iCaseSensitive==1)
   228 		genericValFromIndex = StringUtils::ValFromIndex;
   229 	else
   230 		genericValFromIndex = StringUtils::ValFromIndexF;
   231 
   232 	CArrayFixSeg<RStringTokenEither>** hashTableToUse =
   233 	aTable.iCaseSensitive ? iCSHashTable :iCIHashTable ;	
   234 	TInt cleanupCounter=0;
   235 	for (TUint i = 0; i < aTable.iCount; ++i)
   236 		{
   237 		const TStLitC8<1>* string=reinterpret_cast<const TStLitC8<1>* >(aTable.iTable[i]);		
   238 		
   239 		// Try to find the string in memory, maybe as a dynamic string or as a member of an another table  
   240 		RStringTokenEither token=FindDes(*string, !aTable.iCaseSensitive);
   241 		if (!token.IsNull())
   242 			{
   243 			TStringIdMap map;			
   244 			map.iSourceTableVal=StringUtils::ValFromIndex(i, (TInt16)(iTablePtrs.Count()-1),aTable.iCaseSensitive);
   245 			map.iTargetTableVal=token.iVal;
   246 
   247 			// Put on cleanup stack
   248 			User::LeaveIfError(iRollbackMapList.Append(&map));
   249 			TCleanupItem cleanup(CleanupIdMap, this);
   250 			CleanupStack::PushL(cleanup);
   251 			++cleanupCounter;
   252 			
   253 			User::LeaveIfError(iStringMapList.InsertInUnsignedKeyOrder(map));
   254 
   255 			// Check if this is a link to a dynamic string
   256 			if (!StringUtils::IsTableEntry(token.iVal))
   257 				{
   258 				CStringPoolNode* node = StringUtils::NodePtr(token.iVal);
   259 				node->iRefcount=KMarkedForNoDeleted; // Make sure this string never gets deleted
   260 				}
   261 
   262 			// Now store the reverse array						
   263 			User::LeaveIfError(iStringMapListReverse.InsertInUnsignedKeyOrderAllowRepeats(map));
   264 			}
   265 		else
   266 			{
   267 			TUint8 hash = static_cast<TUint8>(Hash(*string));
   268 			CArrayFixSeg<RStringTokenEither>* collisionList = hashTableToUse[hash];
   269 			if ( !collisionList )
   270 				//HashTableToUse now is used as list of all entry with the same hash
   271 				collisionList = hashTableToUse[hash] = 
   272 				new (ELeave) CArrayFixSeg<RStringTokenEither>( 2 );
   273 			RStringTokenEither s;
   274 			s.iVal = genericValFromIndex(i, (TInt16)(iTablePtrs.Count()-1));
   275 			
   276 			__LOG2(_L8("Table entry being added with hash %d, val %d"), hash, s.iVal);
   277 			__LOG(*reinterpret_cast<const TStLitC8<1>* >(aTable.iTable[i]));
   278 			// Put on cleanup stack
   279 			if (aTable.iCaseSensitive==1)
   280 				{
   281 				User::LeaveIfError(iRollbackHashListCS.Append(&s));
   282 				TCleanupItem cleanup(CleanupHashCS, this);
   283 				CleanupStack::PushL(cleanup);
   284 				}
   285 			else
   286 				{
   287 				User::LeaveIfError(iRollbackHashListCI.Append(&s));
   288 				TCleanupItem cleanup(CleanupHashCI, this);
   289 				CleanupStack::PushL(cleanup);
   290 				}
   291 			
   292 			++cleanupCounter;
   293 			collisionList->AppendL(s);
   294 			}
   295 		}	
   296 	CleanupStack::Pop(cleanupCounter);
   297 	iRollbackMapList.Reset();
   298 	iRollbackHashListCS.Reset();
   299 	iRollbackHashListCI.Reset();
   300 	}
   301 
   302 // Find FirstVal given duplicate val
   303 TInt32 CStringPoolImplementation::FindFirstValFromDuplicate(TInt32 aDuplicateVal) const 
   304 	{
   305 	TStringIdMap map;
   306 	map.iSourceTableVal=aDuplicateVal;	
   307 	TInt index=iStringMapList.FindInUnsignedKeyOrder(map);
   308 	if (index!=KErrNotFound)		
   309 		return iStringMapList[index].iTargetTableVal;
   310 	else
   311 		return KErrNotFound;
   312 	}
   313 
   314 
   315 
   316 // Find table index Val given first val & table UID
   317 TInt CStringPoolImplementation::FindTableIndexFromFirstVal(TInt32 aFirstVal, TInt aTableUid) const
   318 	{
   319 	TStringIdMap map;
   320 	map.iTargetTableVal=aFirstVal;		
   321 	TInt index=iStringMapListReverse.FindInUnsignedKeyOrder(map);
   322 	if (KErrNotFound==index)
   323 		return KErrNotFound;
   324 
   325 	for (TInt count=index;count<iStringMapListReverse.Count();++count)
   326 		{
   327 		if (iStringMapListReverse[count].iTargetTableVal==aFirstVal && StringUtils::TableUid(iStringMapListReverse[count].iSourceTableVal)==aTableUid)
   328 			{
   329 			return StringUtils::TableIndex(iStringMapListReverse[count].iSourceTableVal);
   330 			}
   331 		}	
   332 		return KErrNotFound;
   333 	}
   334 
   335 
   336 // Find the UId for a given table
   337 TInt16 CStringPoolImplementation::TableUid(const TStringTable& aTable) const
   338 	{
   339 	for (TInt count=0; count<iTablePtrs.Count(); ++count)
   340 		{
   341 		if (iTablePtrs[count]==&aTable)
   342 			return (TInt16)count;
   343 		}
   344 	return KErrNotFound;
   345 	}
   346 
   347 // Find a reference to the table that first added the  string represented by aVal to the pool 
   348 const TStringTable& CStringPoolImplementation::TableRef(TInt32 aVal) const
   349 	{
   350 	__ASSERT_DEBUG(aVal!=0, StringPoolPanic::Panic(StringPoolPanic::EIllegalUseOfNullString));
   351 	TInt16 tableUid=(TInt16)(aVal>>20);
   352 	const TStringTable* theTableRef=(iTablePtrs[tableUid]);
   353 	return  *theTableRef;
   354 	}
   355 
   356 // Find the descriptor for a given table and index
   357 const TDesC8& CStringPoolImplementation::TableLookup(TInt aIndex, TInt aTableUid) const
   358 	{	
   359 	return *reinterpret_cast<const TStLitC8<1>*>(iTablePtrs[aTableUid]->iTable[aIndex]);
   360 	}
   361 
   362 
   363 // Lookup with allocating
   364 //
   365 RStringTokenEither 
   366 CStringPoolImplementation::OpenL( const TDesC8& aAttributeName,
   367 								  TBool aCaseInsensitive)
   368 	{
   369 	// lookup the attribute
   370 	RStringTokenEither s(FindDes( aAttributeName , aCaseInsensitive));
   371 	if (!s.IsNull())
   372 		{
   373 		if (!StringUtils::IsTableEntry(s.iVal))
   374 			{
   375 
   376 			CStringPoolNode* node = StringUtils::NodePtr(s.iVal);
   377 			if (KMarkedForNoDeleted!=node->iRefcount)
   378 				node->iRefcount++;
   379 			__LOG1(_L8("String copied (during open). Count is now %d"), node->iRefcount);
   380 			__LOG(*node->iDes);
   381 			}
   382 		return s;
   383 		}
   384 	
   385 	// create a new node at the end of the appropriate array
   386 	CStringPoolNode* newnode = new (ELeave) CStringPoolNode();
   387 	CleanupStack::PushL( newnode );
   388 	newnode->iDes = aAttributeName.AllocL();
   389 	newnode->iRefcount = 1;
   390 
   391 	TInt hash = Hash( aAttributeName );
   392 	CArrayFixSeg<RStringTokenEither>** hashTableToUse = 
   393 		aCaseInsensitive ? iCIHashTable : iCSHashTable;
   394 	__LOG2(_L8("Newly added with hash value %d, node val 0x%x\n"), hash, newnode)
   395 	__LOG(aAttributeName);
   396 
   397 	newnode->iHash = static_cast<TUint8>(hash);
   398 	CArrayFixSeg<RStringTokenEither>* collisionList = hashTableToUse[hash];
   399 	if ( !collisionList )
   400 		collisionList = hashTableToUse[hash] = new (ELeave) CArrayFixSeg<RStringTokenEither>( 2 );
   401 
   402 	s.iVal = reinterpret_cast<TUint32>(newnode);
   403 	if (aCaseInsensitive)
   404 		s.iVal += 2;
   405 	collisionList->AppendL(s);
   406 
   407 	CleanupStack::Pop(); // newnode
   408 
   409 	return s;
   410 	}
   411 
   412 void CStringPoolImplementation::Close(RStringTokenEither aString)
   413 	{
   414 	if (StringUtils::IsTableEntry(aString.iVal))
   415 		return;
   416 
   417 	CStringPoolNode* node = StringUtils::NodePtr(aString.iVal);
   418 	if (KMarkedForNoDeleted == node->iRefcount)	// -1 means a non-expirable string
   419 		return;
   420 	if (--node->iRefcount == 0)
   421 		{
   422 		//this is  the last reference of this string
   423 		CArrayFixSeg<RStringTokenEither>** hashTableToUse = 
   424 		aString.iVal & 2 ? iCIHashTable : iCSHashTable;
   425 
   426 		// Delete the node and delete the entry in the relevant collision list 
   427 		CArrayFixSeg<RStringTokenEither>* collisionList = hashTableToUse[node->iHash];
   428 		TInt count = collisionList->Count();		
   429 		for (TInt i = 0; i < count; i++)
   430 			{
   431 			if (collisionList->At(i) == aString)
   432 				{
   433 				// Log the fact that a string reference is about to die...
   434 				__LOG1(_L8("Removing string with hash value %d\n"), node->iHash)
   435 				__LOG(node->iDes->Des());
   436 				collisionList->Delete(i);
   437 				break;
   438 				}
   439 			}
   440 		delete node;
   441 		}
   442 	else
   443 		{
   444 		__LOG1(_L8("String closed. Count is now %d"), 
   445 			   node->iRefcount);
   446 		__LOG(node->iDes->Des());
   447 		}
   448 	}
   449 
   450 void CStringPoolImplementation::IncrementCount(RStringTokenEither aString)
   451 	{
   452 	if (StringUtils::IsTableEntry(aString.iVal))
   453 		return;
   454 	CStringPoolNode* node = StringUtils::NodePtr(aString.iVal);
   455 	if (KMarkedForNoDeleted!=node->iRefcount)
   456 		node->iRefcount++;
   457 	__LOG1(_L8("String copied. Count is now %d"), node->iRefcount);
   458 	__LOG(*node->iDes);
   459 	}
   460 
   461 // Very simple case-sensitive comparison. We can assume that the
   462 // strings are the same length, and we only care if the strings are
   463 // the same. (Unlike normal comparison functions that also tell you
   464 // which one is 'smaller')
   465 TBool CStringPoolImplementation::CompareCS(const TDesC8& s1, const TDesC8& s2)
   466 	{
   467 	const TUint8* ptr1 = s1.Ptr();
   468 	const TUint8* ptr2 = s2.Ptr();
   469 	const TUint8* stop = &ptr1[s1.Length()];
   470 	for (; ptr1 < stop; ptr1++,ptr2++)
   471 		{
   472 		if (*ptr1 != *ptr2)
   473 			return EFalse;
   474 		}
   475 	return ETrue;
   476 	}
   477 
   478 // Note that the hash function must generate the same hash values for
   479 // strings that differ by case. If changing the algorithm here make
   480 // sure this is still true.
   481 TBool CStringPoolImplementation::CompareCI(const TDesC8& s1, const TDesC8& s2)
   482 	{
   483 	const TUint8* ptr1 = s1.Ptr();
   484 	const TUint8* ptr2 = s2.Ptr();
   485 	const TUint8* stop = &ptr1[s1.Length()];
   486 	for (; ptr1 < stop; ptr1++,ptr2++)
   487 		{
   488 		if (*ptr1 != *ptr2)
   489 			{
   490 			// They're not exactly the same; see if they differ only
   491 			// by case. If one character is a letter, we can do a
   492 			// comparison ignoring bit 5 in both cases. If that
   493 			// matches, they are the same.
   494 			if (!((*ptr1 & KCaseInsensitive) == (*ptr2 & KCaseInsensitive) &&
   495 				  (*ptr1 >= 'A' && *ptr1 <= 'Z' || 
   496 				   *ptr1 >= 'a' && *ptr1 <= 'z')))
   497 				return EFalse;
   498 			}
   499 		}
   500 	return ETrue;
   501 	}
   502 
   503 // Find the given descriptor in the hash table
   504 //
   505 RStringTokenEither 
   506 CStringPoolImplementation::FindDes( const TDesC8& aAttributeName, TBool aCaseInsensitive) 
   507 	{
   508 	CArrayFixSeg<RStringTokenEither>** hashTableToUse = 
   509 		aCaseInsensitive ? iCIHashTable : iCSHashTable;
   510 	CArrayFixSeg<RStringTokenEither>* collisionList =hashTableToUse[Hash(aAttributeName)];
   511 	RStringPool pool;
   512 	TBool (*compareFunction)(const TDesC8&, const TDesC8&);
   513 	if (aCaseInsensitive)
   514 		compareFunction = CompareCI;
   515 	else
   516 		compareFunction = CompareCS;
   517 	pool.iImplementation = this;
   518 	if ( collisionList )
   519 		{
   520 		TInt length=aAttributeName.Length();
   521 		TInt count = collisionList->Count();
   522 		for ( TInt i = 0; i < count; i++ )
   523 			{
   524 			RStringTokenEither token = collisionList->At(i);
   525 			RStringEither s(this, token);
   526 			const TDesC8& string = s.DesC();
   527 			if ( string.Length()==length && 
   528 				 (*compareFunction)(aAttributeName, string))
   529 				return token;
   530 			}
   531 		}
   532 	return RStringTokenEither();
   533 	}
   534 
   535 
   536 // Generate a hash value
   537 //
   538 TUint CStringPoolImplementation::Hash( const TDesC8& aDes ) const
   539 	{
   540 	// We ignore bit 5, which is a crude way of making the hash case
   541 	// insensitive. This means that things that might differ only by
   542 	// case end up in the same bucket, and we can then worry about
   543 	// whether they're really the same later.
   544 	TInt len=aDes.Length();
   545 	TUint hash = 0;
   546 	const TUint8* ptr=aDes.Ptr();
   547 	for ( TInt i = 0; i < len; i++ )
   548 		hash = 131*hash + (*ptr++ & KCaseInsensitive);
   549 	return hash % KHashModulo;
   550 	}
   551 
   552 TInt StringUtils::ValFromIndex(TInt aIndex, TUint16 aTableId)
   553 	{
   554 	return (aTableId << 20) + (aIndex << 2) + 1;
   555 	}
   556 
   557 TInt StringUtils::ValFromIndexF(TInt aIndex, TUint16 aTableId)
   558 	{
   559 	return (aTableId << 20) + (aIndex << 2) + 3;
   560 	}
   561 
   562 TInt StringUtils::ValFromIndex(TInt aIndex, TUint16 aTableId, TBool aCaseSensitive)
   563 	{
   564 	if (aCaseSensitive)
   565 		return ValFromIndex(aIndex, aTableId);
   566 	else
   567 		return ValFromIndexF(aIndex, aTableId);
   568 	}
   569 void  CStringPoolImplementation::AddCallBackL( MStringPoolCloseCallBack& aCallBack)
   570 	{
   571 	User::LeaveIfError(iCallBacks.Append(&aCallBack));
   572 	}