First public contribution.
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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
17 #include "StringPoolImplementation.h"
19 const TInt KMapGranularity=20;
21 CStringPoolNode::~CStringPoolNode()
26 CStringPoolImplementation::CStringPoolImplementation() : iStringMapList(KMapGranularity, _FOFF(TStringIdMap, iSourceTableVal)), iStringMapListReverse(KMapGranularity, _FOFF(TStringIdMap, iTargetTableVal))
30 CStringPoolImplementation::~CStringPoolImplementation()
32 // Look for non-expirable strings
34 for (i = 0; i < KHashModulo; i++ )
36 DeleteUndeletableStrings(iCIHashTable, i);
37 DeleteUndeletableStrings(iCSHashTable, i);
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++ )
49 for (TInt j = 0; j < iCIHashTable[i]->Count(); j++)
51 if (!StringUtils::IsTableEntry(iCIHashTable[i]->At(j).iVal))
55 __LOG(_L8("The following strings were leaked through not being Closed:\n"))
59 // Get the problem string
60 CStringPoolNode* theProblem =
61 reinterpret_cast<CStringPoolNode*>(
62 iCIHashTable[i]->At(j).iVal & KTokenToNode);
64 __LOG(theProblem->iDes->Des());
70 for (TInt j = 0; j < iCSHashTable[i]->Count(); j++)
72 if (!StringUtils::IsTableEntry(iCSHashTable[i]->At(j).iVal))
76 __LOG(_L8("The following strings were leaked through not being Closed:\n"))
80 // Get the problem string
81 CStringPoolNode* theProblem =
82 reinterpret_cast<CStringPoolNode*>(
83 iCSHashTable[i]->At(j).iVal & KTokenToNode);
85 __LOG(theProblem->iDes->Des());
94 __LOG(_L8("No leakages were detected\n"));
97 for (TInt ii = 0; ii < KHashModulo; ii++)
99 delete iCIHashTable[ii];
100 delete iCSHashTable[ii];
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();
113 while (--cBCounter>=0)
115 iCallBacks[cBCounter]->StringPoolClosing();
121 // Check for any undeletable string and delete them now
122 void CStringPoolImplementation::DeleteUndeletableStrings(CArrayFixSeg<RStringTokenEither>* aArray[KHashModulo], TInt i)
126 for (TInt j = 0; j < aArray[i]->Count(); ++j)
128 if (!StringUtils::IsTableEntry(aArray[i]->At(j).iVal))
130 CStringPoolNode* node= reinterpret_cast<CStringPoolNode*>(aArray[i]->At(j).iVal & KTokenToNode);
131 if (KMarkedForNoDeleted==node->iRefcount)
134 aArray[i]->Delete(j);
142 CStringPoolImplementation* CStringPoolImplementation::NewL()
144 CStringPoolImplementation* table = new (ELeave) CStringPoolImplementation();
148 void CStringPoolImplementation::CleanupHashCS(TAny* aImplementation)
150 CStringPoolImplementation* imp=reinterpret_cast<CStringPoolImplementation*>(aImplementation);
151 CleanUpHash(&imp->iRollbackHashListCS, imp->iCSHashTable);
154 void CStringPoolImplementation::CleanupHashCI(TAny* aImplementation)
156 CStringPoolImplementation* imp=reinterpret_cast<CStringPoolImplementation*>(aImplementation);
157 CleanUpHash(&imp->iRollbackHashListCI, imp->iCIHashTable);
160 void CStringPoolImplementation::CleanUpHash(RPointerArray <RStringTokenEither>* aHashCleanup, CArrayFixSeg<RStringTokenEither>* aHash[KHashModulo])
162 if (aHashCleanup->Count()>0)
164 RStringTokenEither* token=(*aHashCleanup)[0]; // Get first entry
165 for (TInt i = 0; i < KHashModulo; i++ )
169 for (TInt j = 0; j < aHash[i]->Count(); j++)
171 if (!StringUtils::IsTableEntry(aHash[i]->At(j).iVal))
173 if (aHash[i]->At(j).iVal==token->iVal)
175 CStringPoolNode* node= reinterpret_cast<CStringPoolNode*>(aHash[i]->At(j).iVal & KTokenToNode);
178 aHashCleanup->Remove(0);
189 void CStringPoolImplementation::CleanupIdMap(TAny* aImplementation)
191 CStringPoolImplementation* imp=reinterpret_cast<CStringPoolImplementation*>(aImplementation);
192 if (imp->iRollbackMapList.Count()>0)
194 TStringIdMap* map=imp->iRollbackMapList[0];
195 TInt index=imp->iStringMapList.FindInUnsignedKeyOrder(*map);
196 imp->iRollbackMapList.Remove(0);
197 if (index!=KErrNotFound)
199 imp->iStringMapList.Remove(index);
201 index=imp->iStringMapListReverse.FindInUnsignedKeyOrder(*map);
202 if (index!=KErrNotFound)
204 for (TInt count=index;count<imp->iStringMapListReverse.Count();++count)
206 if (imp->iStringMapListReverse[count].iTargetTableVal==map->iTargetTableVal && imp->iStringMapListReverse[count].iSourceTableVal==map->iSourceTableVal)
208 imp->iStringMapListReverse.Remove(index);
215 void CStringPoolImplementation::AddTableL(const TStringTable& aTable)
217 for (TInt count=0;count<iTablePtrs.Count();++count) // check for adding the same table twice
219 if (iTablePtrs[count]==&aTable)
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;
230 genericValFromIndex = StringUtils::ValFromIndexF;
232 CArrayFixSeg<RStringTokenEither>** hashTableToUse =
233 aTable.iCaseSensitive ? iCSHashTable :iCIHashTable ;
234 TInt cleanupCounter=0;
235 for (TUint i = 0; i < aTable.iCount; ++i)
237 const TStLitC8<1>* string=reinterpret_cast<const TStLitC8<1>* >(aTable.iTable[i]);
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);
244 map.iSourceTableVal=StringUtils::ValFromIndex(i, (TInt16)(iTablePtrs.Count()-1),aTable.iCaseSensitive);
245 map.iTargetTableVal=token.iVal;
247 // Put on cleanup stack
248 User::LeaveIfError(iRollbackMapList.Append(&map));
249 TCleanupItem cleanup(CleanupIdMap, this);
250 CleanupStack::PushL(cleanup);
253 User::LeaveIfError(iStringMapList.InsertInUnsignedKeyOrder(map));
255 // Check if this is a link to a dynamic string
256 if (!StringUtils::IsTableEntry(token.iVal))
258 CStringPoolNode* node = StringUtils::NodePtr(token.iVal);
259 node->iRefcount=KMarkedForNoDeleted; // Make sure this string never gets deleted
262 // Now store the reverse array
263 User::LeaveIfError(iStringMapListReverse.InsertInUnsignedKeyOrderAllowRepeats(map));
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));
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)
281 User::LeaveIfError(iRollbackHashListCS.Append(&s));
282 TCleanupItem cleanup(CleanupHashCS, this);
283 CleanupStack::PushL(cleanup);
287 User::LeaveIfError(iRollbackHashListCI.Append(&s));
288 TCleanupItem cleanup(CleanupHashCI, this);
289 CleanupStack::PushL(cleanup);
293 collisionList->AppendL(s);
296 CleanupStack::Pop(cleanupCounter);
297 iRollbackMapList.Reset();
298 iRollbackHashListCS.Reset();
299 iRollbackHashListCI.Reset();
302 // Find FirstVal given duplicate val
303 TInt32 CStringPoolImplementation::FindFirstValFromDuplicate(TInt32 aDuplicateVal) const
306 map.iSourceTableVal=aDuplicateVal;
307 TInt index=iStringMapList.FindInUnsignedKeyOrder(map);
308 if (index!=KErrNotFound)
309 return iStringMapList[index].iTargetTableVal;
316 // Find table index Val given first val & table UID
317 TInt CStringPoolImplementation::FindTableIndexFromFirstVal(TInt32 aFirstVal, TInt aTableUid) const
320 map.iTargetTableVal=aFirstVal;
321 TInt index=iStringMapListReverse.FindInUnsignedKeyOrder(map);
322 if (KErrNotFound==index)
325 for (TInt count=index;count<iStringMapListReverse.Count();++count)
327 if (iStringMapListReverse[count].iTargetTableVal==aFirstVal && StringUtils::TableUid(iStringMapListReverse[count].iSourceTableVal)==aTableUid)
329 return StringUtils::TableIndex(iStringMapListReverse[count].iSourceTableVal);
336 // Find the UId for a given table
337 TInt16 CStringPoolImplementation::TableUid(const TStringTable& aTable) const
339 for (TInt count=0; count<iTablePtrs.Count(); ++count)
341 if (iTablePtrs[count]==&aTable)
342 return (TInt16)count;
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
350 __ASSERT_DEBUG(aVal!=0, StringPoolPanic::Panic(StringPoolPanic::EIllegalUseOfNullString));
351 TInt16 tableUid=(TInt16)(aVal>>20);
352 const TStringTable* theTableRef=(iTablePtrs[tableUid]);
356 // Find the descriptor for a given table and index
357 const TDesC8& CStringPoolImplementation::TableLookup(TInt aIndex, TInt aTableUid) const
359 return *reinterpret_cast<const TStLitC8<1>*>(iTablePtrs[aTableUid]->iTable[aIndex]);
363 // Lookup with allocating
366 CStringPoolImplementation::OpenL( const TDesC8& aAttributeName,
367 TBool aCaseInsensitive)
369 // lookup the attribute
370 RStringTokenEither s(FindDes( aAttributeName , aCaseInsensitive));
373 if (!StringUtils::IsTableEntry(s.iVal))
376 CStringPoolNode* node = StringUtils::NodePtr(s.iVal);
377 if (KMarkedForNoDeleted!=node->iRefcount)
379 __LOG1(_L8("String copied (during open). Count is now %d"), node->iRefcount);
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;
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);
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 );
402 s.iVal = reinterpret_cast<TUint32>(newnode);
403 if (aCaseInsensitive)
405 collisionList->AppendL(s);
407 CleanupStack::Pop(); // newnode
412 void CStringPoolImplementation::Close(RStringTokenEither aString)
414 if (StringUtils::IsTableEntry(aString.iVal))
417 CStringPoolNode* node = StringUtils::NodePtr(aString.iVal);
418 if (KMarkedForNoDeleted == node->iRefcount) // -1 means a non-expirable string
420 if (--node->iRefcount == 0)
422 //this is the last reference of this string
423 CArrayFixSeg<RStringTokenEither>** hashTableToUse =
424 aString.iVal & 2 ? iCIHashTable : iCSHashTable;
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++)
431 if (collisionList->At(i) == aString)
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);
444 __LOG1(_L8("String closed. Count is now %d"),
446 __LOG(node->iDes->Des());
450 void CStringPoolImplementation::IncrementCount(RStringTokenEither aString)
452 if (StringUtils::IsTableEntry(aString.iVal))
454 CStringPoolNode* node = StringUtils::NodePtr(aString.iVal);
455 if (KMarkedForNoDeleted!=node->iRefcount)
457 __LOG1(_L8("String copied. Count is now %d"), node->iRefcount);
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)
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++)
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)
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++)
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')))
503 // Find the given descriptor in the hash table
506 CStringPoolImplementation::FindDes( const TDesC8& aAttributeName, TBool aCaseInsensitive)
508 CArrayFixSeg<RStringTokenEither>** hashTableToUse =
509 aCaseInsensitive ? iCIHashTable : iCSHashTable;
510 CArrayFixSeg<RStringTokenEither>* collisionList =hashTableToUse[Hash(aAttributeName)];
512 TBool (*compareFunction)(const TDesC8&, const TDesC8&);
513 if (aCaseInsensitive)
514 compareFunction = CompareCI;
516 compareFunction = CompareCS;
517 pool.iImplementation = this;
520 TInt length=aAttributeName.Length();
521 TInt count = collisionList->Count();
522 for ( TInt i = 0; i < count; i++ )
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))
532 return RStringTokenEither();
536 // Generate a hash value
538 TUint CStringPoolImplementation::Hash( const TDesC8& aDes ) const
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();
546 const TUint8* ptr=aDes.Ptr();
547 for ( TInt i = 0; i < len; i++ )
548 hash = 131*hash + (*ptr++ & KCaseInsensitive);
549 return hash % KHashModulo;
552 TInt StringUtils::ValFromIndex(TInt aIndex, TUint16 aTableId)
554 return (aTableId << 20) + (aIndex << 2) + 1;
557 TInt StringUtils::ValFromIndexF(TInt aIndex, TUint16 aTableId)
559 return (aTableId << 20) + (aIndex << 2) + 3;
562 TInt StringUtils::ValFromIndex(TInt aIndex, TUint16 aTableId, TBool aCaseSensitive)
565 return ValFromIndex(aIndex, aTableId);
567 return ValFromIndexF(aIndex, aTableId);
569 void CStringPoolImplementation::AddCallBackL( MStringPoolCloseCallBack& aCallBack)
571 User::LeaveIfError(iCallBacks.Append(&aCallBack));