kerneltest/e32test/buffer/t_hashtab.cpp
changeset 0 a41df078684a
equal deleted inserted replaced
-1:000000000000 0:a41df078684a
       
     1 // Copyright (c) 2005-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 the License "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 // e32test/buffer/t_hashtab.cpp
       
    15 // 
       
    16 //
       
    17 
       
    18 #include <e32test.h>
       
    19 #include <e32hashtab.h>
       
    20 #include <hal.h>
       
    21 
       
    22 #if defined(__VC32__) || defined(__CW32__)
       
    23 typedef unsigned short wch;
       
    24 #else
       
    25 typedef wchar_t wch;
       
    26 #endif
       
    27 
       
    28 
       
    29 RTest test(_L("T_HASHTAB"));
       
    30 TInt NanoTickPeriod;
       
    31 
       
    32 typedef TBuf8<80> TTestName;
       
    33 typedef RHashSet<TInt> TIntSet;
       
    34 typedef RHashSet<TTestName> TNameSet;
       
    35 typedef RPtrHashSet<TDesC8> TStringSet8;
       
    36 typedef RPtrHashSet<TDesC16> TStringSet16;
       
    37 typedef RPtrHashMap<TDesC8,TDesC8> TStringMap8;
       
    38 typedef RPtrHashMap<TDesC16,TDesC16> TStringMap16;
       
    39 
       
    40 #define INTSET(x)	TIntSet x
       
    41 #define NAMESET(x)	TNameSet x(&HashTestName, &TestNameIdentity)
       
    42 #define STRSET8(x)	TStringSet8 x
       
    43 #define STRSET16(x)	TStringSet16 x
       
    44 #define STRMAP8(x)	TStringMap8 x
       
    45 #define STRMAP16(x)	TStringMap16 x
       
    46 
       
    47 RPointerArray<TDesC8> DesC8Array;
       
    48 RPointerArray<TDesC16> DesC16Array;
       
    49 
       
    50 class HashTest : public RHashTableBase
       
    51 	{
       
    52 public:
       
    53 	HashTest() : RHashTableBase(0,0,0,0) {}
       
    54 	using RHashTableBase::ConsistencyCheck;
       
    55 	using RHashTableBase::Count;
       
    56 	};
       
    57 
       
    58 void CCheck(RHashTableBase& aHT)
       
    59 	{
       
    60 	TUint32 ci[1024];
       
    61 	TUint32 dc;
       
    62 	TUint32 cmp;
       
    63 	HashTest* h = (HashTest*)&aHT;
       
    64 	h->ConsistencyCheck(&dc, &cmp, 1024, ci);
       
    65 	test.Printf(_L("Count = %d, Deleted = %d, Cmp = %d\n"), h->Count(), dc, cmp);
       
    66 	TInt i;
       
    67 	for (i=0; i<1024; ++i)
       
    68 		{
       
    69 		if (ci[i])
       
    70 			test.Printf(_L("%d chains of length %d\n"), ci[i], i);
       
    71 		}
       
    72 	}
       
    73 
       
    74 const char* Numbers[] =
       
    75 	{
       
    76 	"zero",
       
    77 	"one",
       
    78 	"two",
       
    79 	"three",
       
    80 	"four",
       
    81 	"five",
       
    82 	"six",
       
    83 	"seven",
       
    84 	"eight",
       
    85 	"nine",
       
    86 	"ten",
       
    87 	"eleven",
       
    88 	"twelve",
       
    89 	"thirteen",
       
    90 	"fourteen",
       
    91 	"fifteen",
       
    92 	"sixteen",
       
    93 	"seventeen",
       
    94 	"eighteen",
       
    95 	"nineteen",
       
    96 	"twenty",
       
    97 	"thirty",
       
    98 	"forty",
       
    99 	"fifty",
       
   100 	"sixty",
       
   101 	"seventy",
       
   102 	"eighty",
       
   103 	"ninety",
       
   104 	"hundred",
       
   105 	"thousand"
       
   106 	};
       
   107 
       
   108 TTestName NumberInWords(const TInt& aNum)
       
   109 	{
       
   110 	TInt a = aNum;
       
   111 	TTestName n;
       
   112 	if (a<20)
       
   113 		{
       
   114 		n.Copy((const TUint8*)Numbers[a]);
       
   115 		return n;
       
   116 		}
       
   117 	if (a<100)
       
   118 		{
       
   119 		TInt tens = a/10;
       
   120 		TInt units = a%10;
       
   121 		n.Copy((const TUint8*)Numbers[tens-2+20]);
       
   122 		if (units)
       
   123 			{
       
   124 			n.Append(' ');
       
   125 			n.Append(TPtrC8((const TUint8*)Numbers[units]));
       
   126 			}
       
   127 		return n;
       
   128 		}
       
   129 	if (a<1000)
       
   130 		{
       
   131 		TInt h = a/100;
       
   132 		n.Copy((const TUint8*)Numbers[h]);
       
   133 		n.Append(' ');
       
   134 		n.Append(TPtrC8((const TUint8*)Numbers[28]));
       
   135 		a%=100;
       
   136 		if (a)
       
   137 			{
       
   138 			TTestName n2(NumberInWords(a));
       
   139 			n.Append(_L8(" and "));
       
   140 			n+=n2;
       
   141 			}
       
   142 		return n;
       
   143 		}
       
   144 	TInt t = a/1000;
       
   145 	n = NumberInWords(t);
       
   146 	n.Append(' ');
       
   147 	n.Append(TPtrC8((const TUint8*)Numbers[29]));
       
   148 	a%=1000;
       
   149 	if (a)
       
   150 		{
       
   151 		TTestName n2(NumberInWords(a));
       
   152 		if (a<100)
       
   153 			n.Append(_L8(" and "));
       
   154 		else
       
   155 			n.Append(' ');
       
   156 		n+=n2;
       
   157 		}
       
   158 	return n;
       
   159 	}
       
   160 
       
   161 void PrintNumberInWords(TInt a)
       
   162 	{
       
   163 	TBuf<256> buf;
       
   164 	TTestName n(NumberInWords(a));
       
   165 	buf.Copy(n);
       
   166 	test.Printf(_L("%d: %S\n"), a, &buf);
       
   167 	}
       
   168 
       
   169 TUint32 HashTestName(const TTestName& aN)
       
   170 	{
       
   171 	return DefaultHash::Des8(aN);
       
   172 	}
       
   173 
       
   174 TBool TestNameIdentity(const TTestName& aA, const TTestName& aB)
       
   175 	{
       
   176 	return aA == aB;
       
   177 	}
       
   178 
       
   179 /******************************************************************************
       
   180  * Utility functions for RHashSet<T>
       
   181  ******************************************************************************/
       
   182 template <class T>
       
   183 void Union(RHashSet<T>& aD, const RHashSet<T>& aS)
       
   184 	{
       
   185 	TInt c = aS.Count();
       
   186 	TInt c2 = c;
       
   187 	TInt d0 = aD.Count();
       
   188 	TInt d1 = d0;
       
   189 	typename RHashSet<T>::TIter iS(aS);
       
   190 	FOREVER
       
   191 		{
       
   192 		const T* p = iS.Next();
       
   193 		if (!p)
       
   194 			break;
       
   195 		--c2;
       
   196 		TInt r = aD.Insert(*p);
       
   197 		test(r==KErrNone);
       
   198 		++d1;
       
   199 		}
       
   200 	test(d1 == aD.Count());
       
   201 	test(c2 == 0);
       
   202 	}
       
   203 
       
   204 template <class T>
       
   205 void Subtract(RHashSet<T>& aD, const RHashSet<T>& aS)
       
   206 	{
       
   207 	TInt c = aS.Count();
       
   208 	TInt c2 = c;
       
   209 	TInt d0 = aD.Count();
       
   210 	TInt d1 = d0;
       
   211 	TInt nfd = 0;
       
   212 	typename RHashSet<T>::TIter iS(aS);
       
   213 	FOREVER
       
   214 		{
       
   215 		const T* p = iS.Next();
       
   216 		if (!p)
       
   217 			break;
       
   218 		--c2;
       
   219 		TInt r = aD.Remove(*p);
       
   220 		test(r==KErrNone || r==KErrNotFound);
       
   221 		if (r==KErrNotFound)
       
   222 			++nfd;
       
   223 		else
       
   224 			--d1;
       
   225 		}
       
   226 	test(d1 == aD.Count());
       
   227 	test(c2 == 0);
       
   228 	test( (d0-d1) + nfd == c);
       
   229 	}
       
   230 
       
   231 template <class T>
       
   232 void Intersect(RHashSet<T>& aD, const RHashSet<T>& aS)
       
   233 	{
       
   234 	typename RHashSet<T>::TIter iD(aD);
       
   235 	FOREVER
       
   236 		{
       
   237 		const T* p = iD.Next();
       
   238 		if (!p)
       
   239 			break;
       
   240 		if (!aS.Find(*p))
       
   241 			iD.RemoveCurrent();
       
   242 		}
       
   243 	}
       
   244 
       
   245 template <class T>
       
   246 void CheckIdentical(const RHashSet<T>& aA, const RHashSet<T>& aB)
       
   247 	{
       
   248 	test(aA.Count()==aB.Count());
       
   249 	TInt c = aA.Count();
       
   250 	typename RHashSet<T>::TIter iA(aA);
       
   251 	FOREVER
       
   252 		{
       
   253 		const T* p = iA.Next();
       
   254 		if (!p)
       
   255 			break;
       
   256 		--c;
       
   257 		test(aB.Find(*p)!=0);
       
   258 		}
       
   259 	test(c==0);
       
   260 	c = aA.Count();
       
   261 	typename RHashSet<T>::TIter iB(aB);
       
   262 	FOREVER
       
   263 		{
       
   264 		const T* p = iB.Next();
       
   265 		if (!p)
       
   266 			break;
       
   267 		--c;
       
   268 		test(aA.Find(*p)!=0);
       
   269 		}
       
   270 	test(c==0);
       
   271 	}
       
   272 
       
   273 
       
   274 /******************************************************************************
       
   275  * Utility functions for RPtrHashSet<T>
       
   276  ******************************************************************************/
       
   277 template <class T>
       
   278 void Union(RPtrHashSet<T>& aD, const RPtrHashSet<T>& aS)
       
   279 	{
       
   280 	TInt c = aS.Count();
       
   281 	TInt c2 = c;
       
   282 	TInt d0 = aD.Count();
       
   283 	TInt d1 = d0;
       
   284 	typename RPtrHashSet<T>::TIter iS(aS);
       
   285 	FOREVER
       
   286 		{
       
   287 		const T* p = iS.Next();
       
   288 		if (!p)
       
   289 			break;
       
   290 		--c2;
       
   291 		TInt r = aD.Insert(p);
       
   292 		test(r==KErrNone);
       
   293 		++d1;
       
   294 		}
       
   295 	test(d1 == aD.Count());
       
   296 	test(c2 == 0);
       
   297 	}
       
   298 
       
   299 template <class T>
       
   300 void Subtract(RPtrHashSet<T>& aD, const RPtrHashSet<T>& aS)
       
   301 	{
       
   302 	TInt c = aS.Count();
       
   303 	TInt c2 = c;
       
   304 	TInt d0 = aD.Count();
       
   305 	TInt d1 = d0;
       
   306 	TInt nfd = 0;
       
   307 	typename RPtrHashSet<T>::TIter iS(aS);
       
   308 	FOREVER
       
   309 		{
       
   310 		const T* p = iS.Next();
       
   311 		if (!p)
       
   312 			break;
       
   313 		--c2;
       
   314 		TInt r = aD.Remove(p);
       
   315 		test(r==KErrNone || r==KErrNotFound);
       
   316 		if (r==KErrNotFound)
       
   317 			++nfd;
       
   318 		else
       
   319 			--d1;
       
   320 		}
       
   321 	test(d1 == aD.Count());
       
   322 	test(c2 == 0);
       
   323 	test( (d0-d1) + nfd == c);
       
   324 	}
       
   325 
       
   326 template <class T>
       
   327 void Intersect(RPtrHashSet<T>& aD, const RPtrHashSet<T>& aS)
       
   328 	{
       
   329 	typename RPtrHashSet<T>::TIter iD(aD);
       
   330 	FOREVER
       
   331 		{
       
   332 		const T* p = iD.Next();
       
   333 		if (!p)
       
   334 			break;
       
   335 		if (!aS.Find(*p))
       
   336 			iD.RemoveCurrent();
       
   337 		}
       
   338 	}
       
   339 
       
   340 template <class T>
       
   341 void CheckIdentical(const RPtrHashSet<T>& aA, const RPtrHashSet<T>& aB)
       
   342 	{
       
   343 	test(aA.Count()==aB.Count());
       
   344 	TInt c = aA.Count();
       
   345 	typename RPtrHashSet<T>::TIter iA(aA);
       
   346 	FOREVER
       
   347 		{
       
   348 		const T* p = iA.Next();
       
   349 		if (!p)
       
   350 			break;
       
   351 		--c;
       
   352 		test(aB.Find(*p)!=0);
       
   353 		}
       
   354 	test(c==0);
       
   355 	c = aA.Count();
       
   356 	typename RPtrHashSet<T>::TIter iB(aB);
       
   357 	FOREVER
       
   358 		{
       
   359 		const T* p = iB.Next();
       
   360 		if (!p)
       
   361 			break;
       
   362 		--c;
       
   363 		test(aA.Find(*p)!=0);
       
   364 		}
       
   365 	test(c==0);
       
   366 	}
       
   367 
       
   368 
       
   369 /******************************************************************************
       
   370  * Utility functions for RHashMap<K,V>
       
   371  ******************************************************************************/
       
   372 template <class K, class V>
       
   373 void UnionTransform(RHashMap<K,V>& aD, const RHashSet<K>& aS, V (*aTransform)(const K&) )
       
   374 	{
       
   375 	TInt c = aS.Count();
       
   376 	TInt c2 = c;
       
   377 	TInt d0 = aD.Count();
       
   378 	TInt d1 = d0;
       
   379 	typename RHashSet<K>::TIter iS(aS);
       
   380 	FOREVER
       
   381 		{
       
   382 		const K* p = iS.Next();
       
   383 		if (!p)
       
   384 			break;
       
   385 		--c2;
       
   386 		TInt r = aD.Insert(*p, (*aTransform)(*p) );
       
   387 		test(r==KErrNone);
       
   388 		++d1;
       
   389 		}
       
   390 	test(d1 == aD.Count());
       
   391 	test(c2 == 0);
       
   392 	}
       
   393 
       
   394 
       
   395 template <class K, class V>
       
   396 void UnionMap(RHashSet<V>& aD, const RHashSet<K>& aS, const RHashMap<K,V>& aM)
       
   397 	{
       
   398 	TInt c = aS.Count();
       
   399 	TInt c2 = c;
       
   400 	TInt d0 = aD.Count();
       
   401 	TInt d1 = d0;
       
   402 	typename RHashSet<K>::TIter iS(aS);
       
   403 	FOREVER
       
   404 		{
       
   405 		const K* p = iS.Next();
       
   406 		if (!p)
       
   407 			break;
       
   408 		--c2;
       
   409 		const V* q = aM.Find(*p);
       
   410 		if (!q)
       
   411 			continue;
       
   412 		TInt r = aD.Insert(*q);
       
   413 		test(r==KErrNone);
       
   414 		++d1;
       
   415 		}
       
   416 	test(d1 == aD.Count());
       
   417 	test(c2 == 0);
       
   418 	}
       
   419 
       
   420 
       
   421 template <class K, class V>
       
   422 void UnionKeys(RHashSet<K>& aD, const RHashMap<K,V>& aM)
       
   423 	{
       
   424 	typename RHashMap<K,V>::TIter iM(aM);
       
   425 	FOREVER
       
   426 		{
       
   427 		const K* p = iM.NextKey();
       
   428 		if (!p)
       
   429 			break;
       
   430 		aD.Insert(*p);
       
   431 		}
       
   432 	}
       
   433 
       
   434 
       
   435 template <class K, class V>
       
   436 void UnionValues(RHashSet<V>& aD, const RHashMap<K,V>& aM)
       
   437 	{
       
   438 	typename RHashMap<K,V>::TIter iM(aM);
       
   439 	FOREVER
       
   440 		{
       
   441 		const K* p = iM.NextKey();
       
   442 		if (!p)
       
   443 			break;
       
   444 		const V* q = iM.CurrentValue();
       
   445 		aD.Insert(*q);
       
   446 		}
       
   447 	}
       
   448 
       
   449 
       
   450 template <class K, class V>
       
   451 void UnionTransform(RHashMap<K,V>& aD, const RHashMap<K,V>& aS, V (*aTransform)(const V&) )
       
   452 	{
       
   453 	TInt c = aS.Count();
       
   454 	TInt c2 = c;
       
   455 	TInt d0 = aD.Count();
       
   456 	TInt d1 = d0;
       
   457 	typename RHashMap<K,V>::TIter iS(aS);
       
   458 	FOREVER
       
   459 		{
       
   460 		const K* p = iS.NextKey();
       
   461 		if (!p)
       
   462 			break;
       
   463 		--c2;
       
   464 		TInt r = aD.Insert(*p, (*aTransform)(*iS.CurrentValue()) );
       
   465 		test(r==KErrNone);
       
   466 		++d1;
       
   467 		}
       
   468 	test(d1 == aD.Count());
       
   469 	test(c2 == 0);
       
   470 	}
       
   471 
       
   472 
       
   473 template <class K, class V>
       
   474 void UnionInverse(RHashMap<V,K>& aD, const RHashMap<K,V>& aS)
       
   475 	{
       
   476 	TInt c = aS.Count();
       
   477 	TInt c2 = c;
       
   478 	TInt d0 = aD.Count();
       
   479 	TInt d1 = d0;
       
   480 	typename RHashMap<K,V>::TIter iS(aS);
       
   481 	FOREVER
       
   482 		{
       
   483 		const K* p = iS.NextKey();
       
   484 		if (!p)
       
   485 			break;
       
   486 		--c2;
       
   487 		const V* q = iS.CurrentValue();
       
   488 		TInt r = aD.Insert(*q, *p);
       
   489 		test(r==KErrNone);
       
   490 		++d1;
       
   491 		}
       
   492 	test(d1 == aD.Count());
       
   493 	test(c2 == 0);
       
   494 	}
       
   495 
       
   496 template <class K, class V>
       
   497 void SetMap(RHashMap<V,K>& aM, const V& aV)
       
   498 	{
       
   499 	typename RHashMap<K,V>::TIter iM(aM);
       
   500 	FOREVER
       
   501 		{
       
   502 		const K* p = iM.NextKey();
       
   503 		if (!p)
       
   504 			break;
       
   505 		V* q = iM.CurrentValue();
       
   506 		*q = aV;
       
   507 		test(*q == aV);
       
   508 		}
       
   509 	}
       
   510 
       
   511 
       
   512 /******************************************************************************
       
   513  * Utility functions for RPtrHashMap<K,V>
       
   514  ******************************************************************************/
       
   515 template <class K, class V>
       
   516 void UnionTransform(RPtrHashMap<K,V>& aD, const RPtrHashMap<K,V>& aS, V* (*aTransform)(const V*) )
       
   517 	{
       
   518 	TInt c = aS.Count();
       
   519 	TInt c2 = c;
       
   520 	TInt d0 = aD.Count();
       
   521 	TInt d1 = d0;
       
   522 	typename RPtrHashMap<K,V>::TIter iS(aS);
       
   523 	FOREVER
       
   524 		{
       
   525 		const K* p = iS.NextKey();
       
   526 		if (!p)
       
   527 			break;
       
   528 		--c2;
       
   529 		TInt r = aD.Insert(p, (*aTransform)(iS.CurrentValue()) );
       
   530 		test(r==KErrNone);
       
   531 		++d1;
       
   532 		}
       
   533 	test(d1 == aD.Count());
       
   534 	test(c2 == 0);
       
   535 	}
       
   536 
       
   537 
       
   538 template <class A, class B>
       
   539 void UnionInverse(RPtrHashMap<B,A>& aD, const RPtrHashMap<A,B>& a1)
       
   540 	{
       
   541 	typename RPtrHashMap<A,B>::TIter iter(a1);
       
   542 	const A* pA;
       
   543 	while ( (pA=iter.NextKey()) != 0 )
       
   544 		{
       
   545 		const B* pB = iter.CurrentValue();
       
   546 		TInt r = aD.Insert(pB, pA);
       
   547 		test(r==KErrNone);
       
   548 		}
       
   549 	}
       
   550 
       
   551 
       
   552 template <class A, class B, class C>
       
   553 void UnionCompose(RPtrHashMap<A,C>& aD, const RPtrHashMap<A,B>& a1, const RPtrHashMap<B,C>& a2)
       
   554 	{
       
   555 	typename RPtrHashMap<A,B>::TIter iter(a1);
       
   556 	const A* pA;
       
   557 	while ( (pA=iter.NextKey()) != 0 )
       
   558 		{
       
   559 		const B* pB = iter.CurrentValue();
       
   560 		const C* pC = a2.Find(*pB);
       
   561 		if (pC)
       
   562 			{
       
   563 			TInt r = aD.Insert(pA, pC);
       
   564 			test(r==KErrNone);
       
   565 			}
       
   566 		}
       
   567 	}
       
   568 
       
   569 
       
   570 template <class K, class V>
       
   571 void CheckIdenticalMaps(const RPtrHashMap<K,V>& aA, const RPtrHashMap<K,V>& aB)
       
   572 	{
       
   573 	test(aA.Count()==aB.Count());
       
   574 	TInt c = aA.Count();
       
   575 	const K* p;
       
   576 	const V* q;
       
   577 	typename RPtrHashMap<K,V>::TIter iA(aA);
       
   578 	while( (p=iA.NextKey()) != 0)
       
   579 		{
       
   580 		--c;
       
   581 		q = aB.Find(*p);
       
   582 		test(q && q==iA.CurrentValue());
       
   583 		}
       
   584 	test(c==0);
       
   585 	c = aB.Count();
       
   586 	typename RPtrHashMap<K,V>::TIter iB(aB);
       
   587 	while( (p=iB.NextKey()) != 0)
       
   588 		{
       
   589 		--c;
       
   590 		q = aA.Find(*p);
       
   591 		test(q && q==iB.CurrentValue());
       
   592 		}
       
   593 	test(c==0);
       
   594 	}
       
   595 
       
   596 
       
   597 
       
   598 
       
   599 
       
   600 /******************************************************************************
       
   601  * Tests of TIntSet
       
   602  ******************************************************************************/
       
   603 void Populate(TIntSet& aH, TInt aS, TInt aE, TInt aM)
       
   604 	{
       
   605 	TInt x = aS;
       
   606 	for (; x<=aE; x+=aM)
       
   607 		aH.Insert(x);
       
   608 	}
       
   609 
       
   610 void PrimeSieve(TIntSet& aH, TInt aStart, TInt aEnd)
       
   611 	{
       
   612 	TInt x;
       
   613 	TInt e = (aEnd&1) ? aEnd : aEnd-1;
       
   614 	TInt s = (aStart<2) ? 2 : aStart|1;
       
   615 	for (x=s; x<=e; ++x)
       
   616 		{
       
   617 //		test.Printf(_L("add %d\n"),x);
       
   618 		aH.Insert(x);
       
   619 		}
       
   620 	TInt m=2;
       
   621 	FOREVER
       
   622 		{
       
   623 		for (; m*m<=e && !aH.Find(m); ++m)
       
   624 			{}
       
   625 		if (m*m > e)
       
   626 			break;
       
   627 		for (x=m*2; x<=e; x+=m)
       
   628 			{
       
   629 //			test.Printf(_L("remove %d\n"),x);
       
   630 			aH.Remove(x);
       
   631 			}
       
   632 		++m;
       
   633 		}
       
   634 	}
       
   635 
       
   636 TBool IsPrime(TInt x)
       
   637 	{
       
   638 	if (x==2)
       
   639 		return ETrue;
       
   640 	if (!(x&1))
       
   641 		return EFalse;
       
   642 	TInt d;
       
   643 	for (d=3; d*d<=x; d+=2)
       
   644 		{
       
   645 		if (x%d==0)
       
   646 			return EFalse;
       
   647 		}
       
   648 	return ETrue;
       
   649 	}
       
   650 
       
   651 void CheckSieveOutput(TIntSet& aH)
       
   652 	{
       
   653 	RHashSet<TInt>::TIter iter(aH);
       
   654 	TInt min = KMaxTInt;
       
   655 	TInt max = KMinTInt;
       
   656 	TInt c = 0;
       
   657 	test.Printf(_L("%d elements:\n"),aH.Count());
       
   658 	FOREVER
       
   659 		{
       
   660 		const TInt* p = iter.Next();
       
   661 		if (!p)
       
   662 			break;
       
   663 		++c;
       
   664 		if (*p>max) max=*p;
       
   665 		if (*p<min) min=*p;
       
   666 //		test.Printf(_L("%d\n"),*p);
       
   667 		test(IsPrime(*p));
       
   668 		}
       
   669 	test(c==aH.Count());
       
   670 	TInt x;
       
   671 	if (min==2)
       
   672 		{
       
   673 		test(aH.Find(2)!=0);
       
   674 		min++;
       
   675 		}
       
   676 	for (x=min; x<=max; x+=2)
       
   677 		{
       
   678 		if (IsPrime(x))
       
   679 			test(aH.Find(x)!=0);
       
   680 		}
       
   681 	for (x=min; x<=max; x+=2)
       
   682 		{
       
   683 		if (IsPrime(x))
       
   684 			test(aH.Find(x)==&aH.FindL(x));
       
   685 		else
       
   686 			{
       
   687 			TRAPD(rr,aH.FindL(x));
       
   688 			test(rr==KErrNotFound);
       
   689 			}
       
   690 		}
       
   691 	}
       
   692 
       
   693 TUint32 CheckSmallHashSet(const TIntSet& aA)
       
   694 	{
       
   695 	TUint32 m = 0;
       
   696 	RHashSet<TInt>::TIter iter(aA);
       
   697 	FOREVER
       
   698 		{
       
   699 		const TInt* p = iter.Next();
       
   700 		if (!p)
       
   701 			break;
       
   702 		m |= (1<<*p);
       
   703 		}
       
   704 	return m;
       
   705 	}
       
   706 
       
   707 void AddToSmallHashSet(TIntSet& aA, TUint32 aMask)
       
   708 	{
       
   709 	CheckSmallHashSet(aA);
       
   710 	TInt i;
       
   711 	TInt r;
       
   712 	for (i=0; i<32; ++i)
       
   713 		{
       
   714 		if (aMask & (1<<i))
       
   715 			{
       
   716 			r = aA.Insert(i);
       
   717 			test(r==KErrNone);
       
   718 			}
       
   719 		}
       
   720 	}
       
   721 
       
   722 void RemoveFromSmallHashSet(TIntSet& aA, TUint32 aMask)
       
   723 	{
       
   724 	TUint32 m = CheckSmallHashSet(aA);
       
   725 	TInt i;
       
   726 	TInt r;
       
   727 	for (i=0; i<32; ++i)
       
   728 		{
       
   729 		if (aMask & (1<<i))
       
   730 			{
       
   731 			r = aA.Remove(i);
       
   732 			if (m & (1<<i))
       
   733 				test(r==KErrNone);
       
   734 			else
       
   735 				test(r==KErrNotFound);
       
   736 			}
       
   737 		}
       
   738 	}
       
   739 
       
   740 void TestHashSet()
       
   741 	{
       
   742 	test.Next(_L("Test RHashSet"));
       
   743 
       
   744 	INTSET(hs);
       
   745 	CCheck(hs);		// check consistency for empty table
       
   746 	INTSET(hs2);
       
   747 	INTSET(hs3);
       
   748 	PrimeSieve(hs, 1, 100);
       
   749 	CheckSieveOutput(hs);
       
   750 	test(hs.Reserve(1000)==KErrNone);
       
   751 	CCheck(hs);
       
   752 	CheckSieveOutput(hs);	// check that Reserve() preserves existing entries
       
   753 
       
   754 	INTSET(m1);
       
   755 	INTSET(m2);
       
   756 	INTSET(m3);
       
   757 	INTSET(m5);
       
   758 	INTSET(m7);
       
   759 	Populate(m1,2,100,1);
       
   760 	Populate(m2,4,100,2);
       
   761 	Populate(m3,6,100,3);
       
   762 	Populate(m5,10,100,5);
       
   763 	Populate(m7,14,100,7);
       
   764 	Union(hs3, m1);
       
   765 	Subtract(hs3, m2);
       
   766 	Subtract(hs3, m3);
       
   767 	Subtract(hs3, m5);
       
   768 	Subtract(hs3, m7);
       
   769 	CheckSieveOutput(hs3);
       
   770 	CheckIdentical(hs,hs3);
       
   771 	hs3.Close();
       
   772 	INTSET(cm2);
       
   773 	INTSET(cm3);
       
   774 	INTSET(cm5);
       
   775 	INTSET(cm7);
       
   776 	Union(cm2, m1);
       
   777 	Subtract(cm2, m2);
       
   778 	Union(cm3, m1);
       
   779 	Subtract(cm3, m3);
       
   780 	Union(cm5, m1);
       
   781 	Subtract(cm5, m5);
       
   782 	Union(cm7, m1);
       
   783 	Subtract(cm7, m7);
       
   784 	Union(hs3, m1);
       
   785 	Intersect(hs3, cm7);
       
   786 	Intersect(hs3, cm3);
       
   787 	Intersect(hs3, cm5);
       
   788 	Intersect(hs3, cm2);
       
   789 	CheckSieveOutput(hs3);
       
   790 	CheckIdentical(hs,hs3);
       
   791 	hs3.Close();
       
   792 
       
   793 
       
   794 	cm2.Close();
       
   795 	cm3.Close();
       
   796 	cm5.Close();
       
   797 	cm7.Close();
       
   798 	m1.Close();
       
   799 	m2.Close();
       
   800 	m3.Close();
       
   801 	m5.Close();
       
   802 	m7.Close();
       
   803 
       
   804 	PrimeSieve(hs2, 1, 1000);
       
   805 	CheckSieveOutput(hs2);
       
   806 	test(hs2.Reserve(1000)==KErrNone);
       
   807 	CCheck(hs2);
       
   808 	CheckSieveOutput(hs2);	// check that Reserve() preserves existing entries
       
   809 
       
   810 	PrimeSieve(hs, 100, 997);
       
   811 	CheckSieveOutput(hs);
       
   812 	CCheck(hs);
       
   813 	CheckIdentical(hs,hs2);
       
   814 
       
   815 	hs2.Close();
       
   816 	Union(hs2, hs);
       
   817 	CheckIdentical(hs,hs2);
       
   818 	CheckSieveOutput(hs2);
       
   819 
       
   820 	hs2.Close();
       
   821 	hs.Close();
       
   822 
       
   823 	test(CheckSmallHashSet(hs)==0);
       
   824 	AddToSmallHashSet(hs, 1);
       
   825 	test(CheckSmallHashSet(hs)==1);
       
   826 	AddToSmallHashSet(hs, 4);
       
   827 	test(CheckSmallHashSet(hs)==5);
       
   828 	AddToSmallHashSet(hs, 0x80000001);
       
   829 	test(CheckSmallHashSet(hs)==0x80000005);
       
   830 	AddToSmallHashSet(hs, 0x317217f8);
       
   831 	test(CheckSmallHashSet(hs)==0xb17217fd);
       
   832 	RemoveFromSmallHashSet(hs, 0x00000007);
       
   833 	test(CheckSmallHashSet(hs)==0xb17217f8);
       
   834 	RemoveFromSmallHashSet(hs, 0xffffffff);
       
   835 	test(CheckSmallHashSet(hs)==0);
       
   836 	hs.Close();
       
   837 	}
       
   838 
       
   839 void TestHashIter()
       
   840 	{
       
   841 	test.Next(_L("Test iterators"));
       
   842 
       
   843 	RHashSet<TInt> hs;		// empty
       
   844 	RHashSet<TInt>::TIter iter(hs);
       
   845 	test(iter.Next() == 0);
       
   846 	test(iter.Current() == 0);
       
   847 	iter.RemoveCurrent();
       
   848 	test(iter.Next() == 0);
       
   849 	test(iter.Current() == 0);
       
   850 
       
   851 	test(hs.Insert(1) == KErrNone);
       
   852 	test(hs.Insert(2) == KErrNone);
       
   853 	test(hs.Insert(3) == KErrNone);
       
   854 	iter.Reset();
       
   855 	TUint mask = 14;
       
   856 	TInt i;
       
   857 	for (i=0; i<3; ++i)
       
   858 		{
       
   859 		const TInt* p = iter.Next();
       
   860 		test(p!=0);
       
   861 		TInt x = *p;
       
   862 		test (x>=1 && x<=3 && (mask&(1<<x))!=0);
       
   863 		mask &= ~(1<<x);
       
   864 		}
       
   865 	test(iter.Next() == 0);
       
   866 	test(mask==0);
       
   867 	test(CheckSmallHashSet(hs)==0x0e);
       
   868 	test(hs.Insert(4) == KErrNone);
       
   869 	test(hs.Insert(5) == KErrNone);
       
   870 	test(hs.Insert(6) == KErrNone);
       
   871 	test(hs.Insert(7) == KErrNone);
       
   872 	test(CheckSmallHashSet(hs)==0xfe);
       
   873 	iter.Reset();
       
   874 	while(iter.Next())
       
   875 		{
       
   876 		if ((*iter.Current() & 1) == 0)
       
   877 			iter.RemoveCurrent();
       
   878 		}
       
   879 	test(CheckSmallHashSet(hs)==0xaa);
       
   880 	iter.Reset();
       
   881 	while(iter.Next())
       
   882 		{
       
   883 		iter.RemoveCurrent();
       
   884 		}
       
   885 	test(CheckSmallHashSet(hs)==0);
       
   886 	iter.Reset();
       
   887 	test(iter.Next() == 0);
       
   888 	test(iter.Current() == 0);
       
   889 	RHashSet<TInt> empty;
       
   890 	CheckIdentical(hs, empty);
       
   891 #ifdef _DEBUG
       
   892 	__UHEAP_FAILNEXT(1);
       
   893 	test(empty.Insert(1)==KErrNoMemory);
       
   894 	test(empty.Insert(1)==KErrNone);
       
   895 	empty.Close();
       
   896 	__UHEAP_FAILNEXT(1);
       
   897 	test(hs.Insert(1)==KErrNone);
       
   898 	hs.Close();
       
   899 	__UHEAP_FAILNEXT(1);
       
   900 	test(hs.Insert(1)==KErrNoMemory);
       
   901 #endif
       
   902 	hs.Close();
       
   903 	}
       
   904 
       
   905 void Print(const char* aTitle, const RHashMap<TInt,TTestName>& aM)
       
   906 	{
       
   907 	TBuf<256> buf;
       
   908 	buf.Copy(TPtrC8((const TUint8*)aTitle));
       
   909 	test.Printf(_L("%S\n"), &buf);
       
   910 	RHashMap<TInt,TTestName>::TIter iter(aM);
       
   911 	FOREVER
       
   912 		{
       
   913 		const TInt* p = iter.NextKey();
       
   914 		if (!p) break;
       
   915 		buf.Copy(*iter.CurrentValue());
       
   916 		test.Printf(_L("%d: %S\n"), *p, &buf);
       
   917 		}
       
   918 	}
       
   919 
       
   920 void Print(const char* aTitle, const RHashSet<TTestName>& aS)
       
   921 	{
       
   922 	TBuf<256> buf;
       
   923 	buf.Copy(TPtrC8((const TUint8*)aTitle));
       
   924 	test.Printf(_L("%S\n"), &buf);
       
   925 	RHashSet<TTestName>::TIter iter(aS);
       
   926 	FOREVER
       
   927 		{
       
   928 		if (!iter.Next())
       
   929 			break;
       
   930 		buf.Copy(*iter.Current());
       
   931 		test.Printf(_L("%S\n"), &buf);
       
   932 		}
       
   933 	}
       
   934 
       
   935 void TestHashMap()
       
   936 	{
       
   937 	test.Next(_L("Test RHashMap"));
       
   938 
       
   939 	RHashMap<TInt,TInt> ht;
       
   940 	CCheck(ht);		// check consistency for empty table
       
   941 	TInt x;
       
   942 	for (x=0; x<200; x++)
       
   943 		{
       
   944 		TInt r = ht.Insert(x*x, x);
       
   945 		test(r==KErrNone);
       
   946 		}
       
   947 	test(ht.Count()==200);
       
   948 
       
   949 	TInt z = 0;
       
   950 	TInt y;
       
   951 	for (x=0; x<40000; x++)
       
   952 		{
       
   953 		const TInt* e = ht.Find(x);
       
   954 		if (e)
       
   955 			{
       
   956 			const TInt& ee = ht.FindL(x);
       
   957 			test(&ee==e);
       
   958 			y = *e;
       
   959 //			test.Printf(_L("Find(%d) -> %d\n"), x, y);
       
   960 			test(x == z*z);
       
   961 			test(y == z);
       
   962 			++z;
       
   963 			}
       
   964 		else
       
   965 			{
       
   966 			TRAPD(rr, ht.FindL(x));
       
   967 			test(rr==KErrNotFound);
       
   968 			}
       
   969 		}
       
   970 	CCheck(ht);
       
   971 
       
   972 	for (x=0; x<200; x++)
       
   973 		{
       
   974 		TInt r = ht.Insert(x*x*x, x);
       
   975 		test(r==KErrNone);
       
   976 		}
       
   977 	test(ht.Count()==200*2-6);
       
   978 	CCheck(ht);
       
   979 
       
   980 	TInt sq = 0;
       
   981 	TInt cb = 0;
       
   982 	for (x=0; x<8000000; x++)
       
   983 		{
       
   984 		const TInt* e = ht.Find(x);
       
   985 		if (e)
       
   986 			{
       
   987 			const TInt& ee = ht.FindL(x);
       
   988 			test(&ee==e);
       
   989 			y = *e;
       
   990 //			test.Printf(_L("Find(%d) -> %d\n"), x, y);
       
   991 			if (x == cb*cb*cb)
       
   992 				{
       
   993 				test(y==cb);
       
   994 				++cb;
       
   995 				if (x == sq*sq)
       
   996 					++sq;
       
   997 				}
       
   998 			else if (x == sq*sq)
       
   999 				{
       
  1000 				test(y==sq);
       
  1001 				++sq;
       
  1002 				}
       
  1003 			}
       
  1004 		}
       
  1005 
       
  1006 	for (x=0; x<200; x++)
       
  1007 		{
       
  1008 		TInt r = ht.Remove(x*x);
       
  1009 		test(r==KErrNone);
       
  1010 		}
       
  1011 	test(ht.Count()==200-6);
       
  1012 	CCheck(ht);
       
  1013 
       
  1014 	cb = 2;
       
  1015 	for (x=2; x<8000000; x++)
       
  1016 		{
       
  1017 		const TInt* e = ht.Find(x);
       
  1018 		if (e)
       
  1019 			{
       
  1020 			const TInt& ee = ht.FindL(x);
       
  1021 			test(&ee==e);
       
  1022 			y = *e;
       
  1023 //			test.Printf(_L("Find(%d) -> %d\n"), x, y);
       
  1024 			if (cb == 4 || cb == 9 || cb == 16 || cb == 25) ++cb;
       
  1025 			test(x == cb*cb*cb);
       
  1026 			test(y == cb);
       
  1027 			++cb;
       
  1028 			}
       
  1029 		}
       
  1030 
       
  1031 	SetMap<TInt,TInt>(ht, 17);
       
  1032 
       
  1033 	ht.Close();
       
  1034 
       
  1035 	PrintNumberInWords(2000);
       
  1036 	PrintNumberInWords(2001);
       
  1037 	PrintNumberInWords(131072);
       
  1038 	PrintNumberInWords(111111);
       
  1039 	PrintNumberInWords(524288);
       
  1040 
       
  1041 	INTSET(all);
       
  1042 	Populate(all,0,1000,1);
       
  1043 	RHashMap<TInt,TTestName> to_words(&DefaultHash::Integer, &DefaultIdentity::Integer);
       
  1044 	UnionTransform<TInt,TTestName>(to_words, all, &NumberInWords);
       
  1045 	RHashMap<TTestName,TInt> from_words(&HashTestName, &TestNameIdentity);
       
  1046 	UnionInverse<TInt,TTestName>(from_words, to_words);
       
  1047 //	Print("TO WORDS:", to_words);
       
  1048 	INTSET(primes);
       
  1049 	PrimeSieve(primes, 1, 100);
       
  1050 //	Print("Primes 1-100", primes);
       
  1051 	RHashMap<TInt,TTestName> prime_map(&DefaultHash::Integer, &DefaultIdentity::Integer);
       
  1052 	UnionTransform<TInt,TTestName>(prime_map, primes, &NumberInWords);
       
  1053 //	Print("Prime map 1-100", prime_map);
       
  1054 	INTSET(pmkeys);
       
  1055 	UnionKeys<TInt,TTestName>(pmkeys, prime_map);
       
  1056 	NAMESET(pmval);
       
  1057 	NAMESET(pmval2);
       
  1058 	UnionValues<TInt,TTestName>(pmval, prime_map);
       
  1059 	CheckIdentical(pmkeys, primes);
       
  1060 	INTSET(pr2);
       
  1061 	UnionMap<TTestName,TInt>(pr2, pmval, from_words);
       
  1062 	CheckIdentical(pr2, primes);
       
  1063 	pr2.Close();
       
  1064 	Union(pmval2, pmval);
       
  1065 //	Print("pmval",pmval);
       
  1066 //	Print("pmval2",pmval2);
       
  1067 	CheckIdentical(pmval2, pmval);
       
  1068 	UnionMap<TTestName,TInt>(pr2, pmval2, from_words);
       
  1069 	CheckIdentical(pr2, primes);
       
  1070 
       
  1071 	pr2.Close();
       
  1072 	pmval.Close();
       
  1073 	pmval2.Close();
       
  1074 	pmkeys.Close();
       
  1075 	prime_map.Close();
       
  1076 	primes.Close();
       
  1077 	all.Close();
       
  1078 
       
  1079 	INTSET(m);
       
  1080 	Populate(all,2,1000,1);
       
  1081 
       
  1082 	NAMESET(pr3);
       
  1083 	NAMESET(nm);
       
  1084 	UnionMap<TInt,TTestName>(pr3, all, to_words);
       
  1085 	all.Close();
       
  1086 	CCheck(pr3);
       
  1087 
       
  1088 	Populate(m,4,1000,2);
       
  1089 	UnionMap<TInt,TTestName>(nm, m, to_words);
       
  1090 	m.Close();
       
  1091 	Subtract(pr3, nm);
       
  1092 	nm.Close();
       
  1093 
       
  1094 	Populate(m,6,1000,3);
       
  1095 	UnionMap<TInt,TTestName>(nm, m, to_words);
       
  1096 	m.Close();
       
  1097 	Subtract(pr3, nm);
       
  1098 	nm.Close();
       
  1099 
       
  1100 	Populate(m,10,1000,5);
       
  1101 	UnionMap<TInt,TTestName>(nm, m, to_words);
       
  1102 	m.Close();
       
  1103 	Subtract(pr3, nm);
       
  1104 	nm.Close();
       
  1105 
       
  1106 	Populate(m,14,1000,7);
       
  1107 	UnionMap<TInt,TTestName>(nm, m, to_words);
       
  1108 	m.Close();
       
  1109 	Subtract(pr3, nm);
       
  1110 	nm.Close();
       
  1111 
       
  1112 	Populate(m,22,1000,11);
       
  1113 	UnionMap<TInt,TTestName>(nm, m, to_words);
       
  1114 	m.Close();
       
  1115 	Subtract(pr3, nm);
       
  1116 	nm.Close();
       
  1117 
       
  1118 	Populate(m,26,1000,13);
       
  1119 	UnionMap<TInt,TTestName>(nm, m, to_words);
       
  1120 	m.Close();
       
  1121 	Subtract(pr3, nm);
       
  1122 	nm.Close();
       
  1123 
       
  1124 	Populate(m,34,1000,17);
       
  1125 	UnionMap<TInt,TTestName>(nm, m, to_words);
       
  1126 	m.Close();
       
  1127 	Subtract(pr3, nm);
       
  1128 	nm.Close();
       
  1129 
       
  1130 	Populate(m,38,1000,19);
       
  1131 	UnionMap<TInt,TTestName>(nm, m, to_words);
       
  1132 	m.Close();
       
  1133 	Subtract(pr3, nm);
       
  1134 	nm.Close();
       
  1135 
       
  1136 	Populate(m,46,1000,23);
       
  1137 	UnionMap<TInt,TTestName>(nm, m, to_words);
       
  1138 	m.Close();
       
  1139 	Subtract(pr3, nm);
       
  1140 	nm.Close();
       
  1141 
       
  1142 	Populate(m,58,1000,29);
       
  1143 	UnionMap<TInt,TTestName>(nm, m, to_words);
       
  1144 	m.Close();
       
  1145 	Subtract(pr3, nm);
       
  1146 	nm.Close();
       
  1147 
       
  1148 	Populate(m,62,1000,31);
       
  1149 	UnionMap<TInt,TTestName>(nm, m, to_words);
       
  1150 	m.Close();
       
  1151 	Subtract(pr3, nm);
       
  1152 	nm.Close();
       
  1153 
       
  1154 //	Print("pr3",pr3);
       
  1155 	PrimeSieve(primes, 1, 1000);
       
  1156 	UnionMap<TInt,TTestName>(nm, primes, to_words);
       
  1157 	CheckIdentical(nm, pr3);
       
  1158 	CCheck(pr3);
       
  1159 	UnionMap<TTestName,TInt>(m, pr3, from_words);
       
  1160 	CheckIdentical(m, primes);
       
  1161 
       
  1162 	m.Close();
       
  1163 	nm.Close();
       
  1164 	primes.Close();
       
  1165 	pr3.Close();
       
  1166 	from_words.Close();
       
  1167 	to_words.Close();
       
  1168 	}
       
  1169 
       
  1170 void PopulateArray8(TInt aMax)
       
  1171 	{
       
  1172 	TInt i;
       
  1173 	for (i=0; i<aMax; ++i)
       
  1174 		{
       
  1175 		TTestName n(NumberInWords(i));
       
  1176 		HBufC8* p = n.Alloc();
       
  1177 		test(p!=0);
       
  1178 		TInt r = DesC8Array.Append(p);
       
  1179 		test(r==KErrNone);
       
  1180 		}
       
  1181 	}
       
  1182 
       
  1183 void PopulateArray16(TInt aMax)
       
  1184 	{
       
  1185 	TInt i;
       
  1186 	for (i=0; i<aMax; ++i)
       
  1187 		{
       
  1188 		TTestName n(NumberInWords(i));
       
  1189 		TBuf<128> n16;
       
  1190 		n16.Copy(n);
       
  1191 		HBufC16* p = n16.Alloc();
       
  1192 		test(p!=0);
       
  1193 		TInt r = DesC16Array.Append(p);
       
  1194 		test(r==KErrNone);
       
  1195 		}
       
  1196 	}
       
  1197 
       
  1198 TUint32 istrh8(const TDesC8* const & a)
       
  1199 	{
       
  1200 	return DefaultHash::Des8(*a);
       
  1201 	}
       
  1202 
       
  1203 TBool istrid8(const TDesC8* const & aA, const TDesC8* const & aB)
       
  1204 	{
       
  1205 	return *aA == *aB;
       
  1206 	}
       
  1207 
       
  1208 TUint32 istrh16(const TDesC16* const & a)
       
  1209 	{
       
  1210 	return DefaultHash::Des16(*a);
       
  1211 	}
       
  1212 
       
  1213 TBool istrid16(const TDesC16* const & aA, const TDesC16* const & aB)
       
  1214 	{
       
  1215 	return *aA == *aB;
       
  1216 	}
       
  1217 
       
  1218 void TestPtrHashSet()
       
  1219 	{
       
  1220 	test.Next(_L("Test RPtrHashSet"));
       
  1221 
       
  1222 	STRSET8(s);
       
  1223 	CCheck(s);		// check consistency for empty table
       
  1224 	RHashMap<const TDesC8*, TInt> hm(&istrh8, &istrid8);
       
  1225 	RPtrHashSet<TDesC8>::TIter iter(s);
       
  1226 	const TDesC8* q;
       
  1227 	TInt i;
       
  1228 	for (i=0; i<4096; ++i)
       
  1229 		{
       
  1230 		TInt r = s.Insert(DesC8Array[i]);
       
  1231 		test(r==KErrNone);
       
  1232 		r = hm.Insert(DesC8Array[i], i);
       
  1233 		test(r==KErrNone);
       
  1234 		}
       
  1235 	CCheck(s);
       
  1236 	for (i=0; i<4100; ++i)
       
  1237 		{
       
  1238 		TTestName n(NumberInWords(i));
       
  1239 		const TDesC8* p = s.Find(n);
       
  1240 		if (i<4096)
       
  1241 			{
       
  1242 			const TDesC8& pp = s.FindL(n);
       
  1243 			test(p && *p==n && p==DesC8Array[i]);
       
  1244 			test(&pp == p);
       
  1245 			}
       
  1246 		else
       
  1247 			{
       
  1248 			test(!p);
       
  1249 			TRAPD(rr,s.FindL(n));
       
  1250 			test(rr==KErrNotFound);
       
  1251 			}
       
  1252 		}
       
  1253 	while((q=iter.Next())!=0)
       
  1254 		{
       
  1255 		const TInt* qi = hm.Find(q);
       
  1256 		test(qi!=0 && *qi>=0 && *qi<4096);
       
  1257 		test(DesC8Array[*qi]==q);
       
  1258 		}
       
  1259 	for (i=0; i<4100; i+=2)
       
  1260 		{
       
  1261 		TTestName n(NumberInWords(i));
       
  1262 		TInt r = s.Remove(&n);
       
  1263 		if (i<4096)
       
  1264 			test(r==KErrNone);
       
  1265 		else
       
  1266 			test(r==KErrNotFound);
       
  1267 		}
       
  1268 	test(s.Count()==2048);
       
  1269 	CCheck(s);
       
  1270 	for (i=0; i<4100; ++i)
       
  1271 		{
       
  1272 		TTestName n(NumberInWords(i));
       
  1273 		const TDesC8* p = s.Find(n);
       
  1274 		if (i<4096 && (i&1))
       
  1275 			test(p && *p==n && p==DesC8Array[i]);
       
  1276 		else
       
  1277 			test(!p);
       
  1278 		}
       
  1279 	iter.Reset();
       
  1280 	while((q=iter.Next())!=0)
       
  1281 		{
       
  1282 		const TInt* qi = hm.Find(q);
       
  1283 		test(qi!=0 && *qi>=0 && *qi<4096 && (*qi&1));
       
  1284 		test(DesC8Array[*qi]==q);
       
  1285 		}
       
  1286 	for (i=2; i<4096; i+=4)
       
  1287 		{
       
  1288 		TInt r = s.Insert(DesC8Array[i]);
       
  1289 		test(r==KErrNone);
       
  1290 		}
       
  1291 	test(s.Count()==3072);
       
  1292 	CCheck(s);
       
  1293 	for (i=0; i<4100; ++i)
       
  1294 		{
       
  1295 		TTestName n(NumberInWords(i));
       
  1296 		const TDesC8* p = s.Find(n);
       
  1297 		if (i<4096 && (i&3))
       
  1298 			test(p && *p==n && p==DesC8Array[i]);
       
  1299 		else
       
  1300 			test(!p);
       
  1301 		}
       
  1302 	iter.Reset();
       
  1303 	while((q=iter.Next())!=0)
       
  1304 		{
       
  1305 		const TInt* qi = hm.Find(q);
       
  1306 		test(qi!=0 && *qi>=0 && *qi<4096 && (*qi&3));
       
  1307 		test(DesC8Array[*qi]==q);
       
  1308 		}
       
  1309 	s.Close();
       
  1310 
       
  1311 	// test ResetAndDestroy
       
  1312 	for (i=0; i<16; ++i)
       
  1313 		{
       
  1314 		TInt r = s.Insert(DesC8Array[i]->Alloc());
       
  1315 		test(r==KErrNone);
       
  1316 		}
       
  1317 	iter.Reset();
       
  1318 	while((q=iter.Next())!=0)
       
  1319 		{
       
  1320 		const TInt* qi = hm.Find(q);
       
  1321 		test(qi!=0 && *qi>=0 && *qi<16);
       
  1322 		test(*DesC8Array[*qi]==*q);
       
  1323 		test(DesC8Array[*qi]!=q);
       
  1324 		}
       
  1325 	s.ResetAndDestroy();
       
  1326 	hm.Close();
       
  1327 
       
  1328 
       
  1329 	STRSET16(S);
       
  1330 	RHashMap<const TDesC16*, TInt> HM(&istrh16, &istrid16);
       
  1331 	RPtrHashSet<TDesC16>::TIter ITER(S);
       
  1332 	const TDesC16* Q;
       
  1333 	for (i=0; i<4096; ++i)
       
  1334 		{
       
  1335 		TInt r = S.Insert(DesC16Array[i]);
       
  1336 		test(r==KErrNone);
       
  1337 		r = HM.Insert(DesC16Array[i], i);
       
  1338 		test(r==KErrNone);
       
  1339 		}
       
  1340 	CCheck(S);
       
  1341 	for (i=0; i<4100; ++i)
       
  1342 		{
       
  1343 		TTestName n(NumberInWords(i));
       
  1344 		TBuf<80> buf;
       
  1345 		buf.Copy(n);
       
  1346 		const TDesC16* p = S.Find(buf);
       
  1347 		if (i<4096)
       
  1348 			test(p && *p==buf && p==DesC16Array[i]);
       
  1349 		else
       
  1350 			test(!p);
       
  1351 		}
       
  1352 	while((Q=ITER.Next())!=0)
       
  1353 		{
       
  1354 		const TInt* qi = HM.Find(Q);
       
  1355 		test(qi!=0 && *qi>=0 && *qi<4096);
       
  1356 		test(DesC16Array[*qi]==Q);
       
  1357 		}
       
  1358 	for (i=0; i<4100; i+=2)
       
  1359 		{
       
  1360 		TTestName n(NumberInWords(i));
       
  1361 		TBuf<80> buf;
       
  1362 		buf.Copy(n);
       
  1363 		TInt r = S.Remove(&buf);
       
  1364 		if (i<4096)
       
  1365 			test(r==KErrNone);
       
  1366 		else
       
  1367 			test(r==KErrNotFound);
       
  1368 		}
       
  1369 	test(S.Count()==2048);
       
  1370 	CCheck(S);
       
  1371 	for (i=0; i<4100; ++i)
       
  1372 		{
       
  1373 		TTestName n(NumberInWords(i));
       
  1374 		TBuf<80> buf;
       
  1375 		buf.Copy(n);
       
  1376 		const TDesC16* p = S.Find(buf);
       
  1377 		if (i<4096 && (i&1))
       
  1378 			test(p && *p==buf && p==DesC16Array[i]);
       
  1379 		else
       
  1380 			test(!p);
       
  1381 		}
       
  1382 	ITER.Reset();
       
  1383 	while((Q=ITER.Next())!=0)
       
  1384 		{
       
  1385 		const TInt* qi = HM.Find(Q);
       
  1386 		test(qi!=0 && *qi>=0 && *qi<4096 && (*qi&1));
       
  1387 		test(DesC16Array[*qi]==Q);
       
  1388 		}
       
  1389 	for (i=2; i<4096; i+=4)
       
  1390 		{
       
  1391 		TInt r = S.Insert(DesC16Array[i]);
       
  1392 		test(r==KErrNone);
       
  1393 		}
       
  1394 	test(S.Count()==3072);
       
  1395 	CCheck(S);
       
  1396 	for (i=0; i<4100; ++i)
       
  1397 		{
       
  1398 		TTestName n(NumberInWords(i));
       
  1399 		TBuf<80> buf;
       
  1400 		buf.Copy(n);
       
  1401 		const TDesC16* p = S.Find(buf);
       
  1402 		if (i<4096 && (i&3))
       
  1403 			test(p && *p==buf && p==DesC16Array[i]);
       
  1404 		else
       
  1405 			test(!p);
       
  1406 		}
       
  1407 	ITER.Reset();
       
  1408 	while((Q=ITER.Next())!=0)
       
  1409 		{
       
  1410 		const TInt* qi = HM.Find(Q);
       
  1411 		test(qi!=0 && *qi>=0 && *qi<4096 && (*qi&3));
       
  1412 		test(DesC16Array[*qi]==Q);
       
  1413 		}
       
  1414 	S.Close();
       
  1415 
       
  1416 	// test ResetAndDestroy
       
  1417 	for (i=0; i<16; ++i)
       
  1418 		{
       
  1419 		TInt r = S.Insert(DesC16Array[i]->Alloc());
       
  1420 		test(r==KErrNone);
       
  1421 		}
       
  1422 	ITER.Reset();
       
  1423 	while((Q=ITER.Next())!=0)
       
  1424 		{
       
  1425 		const TInt* qi = HM.Find(Q);
       
  1426 		test(qi!=0 && *qi>=0 && *qi<16);
       
  1427 		test(*DesC16Array[*qi]==*Q);
       
  1428 		test(DesC16Array[*qi]!=Q);
       
  1429 		}
       
  1430 	S.ResetAndDestroy();
       
  1431 	HM.Close();
       
  1432 	}
       
  1433 
       
  1434 void TestPtrHashMap()
       
  1435 	{
       
  1436 	test.Next(_L("Test RPtrHashMap"));
       
  1437 
       
  1438 	STRMAP8(map);
       
  1439 	CCheck(map);		// check consistency for empty table
       
  1440 	STRMAP8(id);
       
  1441 	TInt i;
       
  1442 	for (i=0; i<4096; ++i)
       
  1443 		{
       
  1444 		TInt r = map.Insert(DesC8Array[i], DesC8Array[(i+1)&0xfff]);
       
  1445 		test(r==KErrNone);
       
  1446 		r = id.Insert(DesC8Array[i], DesC8Array[i]);
       
  1447 		test(r==KErrNone);
       
  1448 		}
       
  1449 	for (i=0; i<4096; ++i)
       
  1450 		{
       
  1451 		const TDesC8* p = map.Find(*DesC8Array[i]);
       
  1452 		test(p == DesC8Array[(i+1)&0xfff]);
       
  1453 		const TDesC8& pp = map.FindL(*DesC8Array[i]);
       
  1454 		test(&pp == p);
       
  1455 		}
       
  1456 	TRAPD(rr,map.FindL(_L8("two")));
       
  1457 	test(rr==KErrNone);
       
  1458 	TRAP(rr,map.FindL(_L8("twx")));
       
  1459 	test(rr==KErrNotFound);
       
  1460 	CCheck(map);
       
  1461 	STRMAP8(map2);
       
  1462 	STRMAP8(mapi);
       
  1463 	STRMAP8(map3);
       
  1464 	UnionInverse<TDesC8,TDesC8>(mapi, map);
       
  1465 	UnionCompose<TDesC8,TDesC8,TDesC8>(map2, map, map);
       
  1466 	UnionCompose<TDesC8,TDesC8,TDesC8>(map3, map, mapi);
       
  1467 	CheckIdenticalMaps<TDesC8,TDesC8>(map3, id);
       
  1468 	map3.Close();
       
  1469 	UnionCompose<TDesC8,TDesC8,TDesC8>(map3, map2, mapi);
       
  1470 	CheckIdenticalMaps<TDesC8,TDesC8>(map3, map);
       
  1471 	map3.Close();
       
  1472 	for (i=0; i<4096; ++i)
       
  1473 		{
       
  1474 		TInt r = map3.Insert(DesC8Array[i], DesC8Array[(i+2)&0xfff]);
       
  1475 		test(r==KErrNone);
       
  1476 		}
       
  1477 	CheckIdenticalMaps<TDesC8,TDesC8>(map3, map2);
       
  1478 	map3.Close();
       
  1479 
       
  1480 	mapi.Close();
       
  1481 	map2.Close();
       
  1482 	id.Close();
       
  1483 	map.Close();
       
  1484 
       
  1485 	// test ResetAndDestroy()
       
  1486 	for(i=0; i<16; ++i)
       
  1487 		{
       
  1488 		TInt r = map.Insert(DesC8Array[i]->Alloc(), DesC8Array[i*i]->Alloc());
       
  1489 		test(r==KErrNone);
       
  1490 		}
       
  1491 	map.ResetAndDestroy();
       
  1492 	}
       
  1493 
       
  1494 void TestOOM()
       
  1495 	{
       
  1496 	// Max out memory and check it still works
       
  1497 	test.Next(_L("Test OOM"));
       
  1498 
       
  1499 	TInt x = 0;
       
  1500 	TInt n = 0;
       
  1501 	TInt r;
       
  1502 	INTSET(set);
       
  1503 	FOREVER
       
  1504 		{
       
  1505 		x += 0x58b90bfb;
       
  1506 		r = set.Insert(x);
       
  1507 		if (r != KErrNone)
       
  1508 			break;
       
  1509 		++n;
       
  1510 		}
       
  1511 	test(r==KErrNoMemory);
       
  1512 	TRAPD(rr,set.InsertL(x));
       
  1513 	test(rr==KErrNoMemory);
       
  1514 	test.Printf(_L("%d integers stored\n"), n);
       
  1515 	test(set.Count()==n);
       
  1516 	TRAP(rr,set.InsertL(0x58b90bfb));	// already there
       
  1517 	test(rr==KErrNone);	// should succeed
       
  1518 
       
  1519 	// final count should be a power of 2 minus 1
       
  1520 	test( (n&(n+1)) == 0 );
       
  1521 
       
  1522 	x = 0;
       
  1523 	TInt i;
       
  1524 	for (i=0; i<=n; ++i)		// check everything has been stored correctly
       
  1525 		{
       
  1526 		x += 0x58b90bfb;
       
  1527 		const TInt* p = set.Find(x);
       
  1528 		if (i < n)
       
  1529 			test(p && *p == x);
       
  1530 		else
       
  1531 			test(!p);
       
  1532 		}
       
  1533 	set.Close();
       
  1534 	TInt nn;
       
  1535 	for (nn=256; nn<=n+256; nn+=256)
       
  1536 		{
       
  1537 		r = set.Reserve(nn);
       
  1538 		set.Close();
       
  1539 		if (r!=KErrNone)
       
  1540 			break;
       
  1541 		}
       
  1542 	test.Printf(_L("Max reserve %d\n"),nn);
       
  1543 	TInt thresh = 3*((n+1)>>2);
       
  1544 	test(nn == thresh + 256);
       
  1545 	}
       
  1546 
       
  1547 class RDummyAllocator : public RAllocator
       
  1548 	{
       
  1549 public:
       
  1550 	virtual TAny* Alloc(TInt) {test(0); return 0;}
       
  1551 	virtual void Free(TAny*) {test(0);}
       
  1552 	virtual TAny* ReAlloc(TAny*, TInt, TInt) {test(0); return 0;}
       
  1553 	virtual TInt AllocLen(const TAny*) const  {test(0); return 0;}
       
  1554 	virtual TInt Compress() {test(0); return 0;}
       
  1555 	virtual void Reset() {test(0);}
       
  1556 	virtual TInt AllocSize(TInt&) const  {test(0); return 0;}
       
  1557 	virtual TInt Available(TInt&) const  {test(0); return 0;}
       
  1558 	virtual TInt DebugFunction(TInt, TAny*, TAny*) {test(0); return 0;}
       
  1559 	virtual TInt Extension_(TUint, TAny*&, TAny*) {test(0); return 0;}
       
  1560 	};
       
  1561 
       
  1562 void IntegerBenchmark(TInt aCount, TBool aReserve)
       
  1563 	{
       
  1564 	RArray<TInt> array;
       
  1565 	TUint32 before, after, diff;
       
  1566 	TInt x=0;
       
  1567 	TInt i;
       
  1568 	double avg;
       
  1569 
       
  1570 	test.Printf(_L("**** INTEGER BENCHMARKS ***\n"));
       
  1571 
       
  1572 	if (!aReserve)
       
  1573 		{
       
  1574 		before = User::NTickCount();
       
  1575 		for (i=0; i<aCount; ++i)
       
  1576 			{
       
  1577 			x += 0x58b90bfb;
       
  1578 			TInt r = array.InsertInOrder(x);
       
  1579 			test(r==KErrNone);
       
  1580 			}
       
  1581 		after = User::NTickCount();
       
  1582 		diff = after - before;
       
  1583 		diff *= NanoTickPeriod;
       
  1584 		avg = (double)diff / (double)aCount;
       
  1585 		test.Printf(_L("ARRAY:      %d insertions take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1586 
       
  1587 		x=0;
       
  1588 		before = User::NTickCount();
       
  1589 		for (i=0; i<aCount; ++i)
       
  1590 			{
       
  1591 			x += 0x58b90bfb;
       
  1592 			TInt r = array.FindInOrder(x);
       
  1593 			test(r>=0);
       
  1594 			}
       
  1595 		after = User::NTickCount();
       
  1596 		diff = after - before;
       
  1597 		diff *= NanoTickPeriod;
       
  1598 		avg = (double)diff / (double)aCount;
       
  1599 		test.Printf(_L("ARRAY:      %d successful finds take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1600 
       
  1601 		before = User::NTickCount();
       
  1602 		for (i=0; i<aCount; ++i)
       
  1603 			{
       
  1604 			x += 0x58b90bfb;
       
  1605 			TInt r = array.FindInOrder(x);
       
  1606 			test(r==KErrNotFound);
       
  1607 			}
       
  1608 		after = User::NTickCount();
       
  1609 		diff = after - before;
       
  1610 		diff *= NanoTickPeriod;
       
  1611 		avg = (double)diff / (double)aCount;
       
  1612 		test.Printf(_L("ARRAY:      %d unsuccessful finds take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1613 
       
  1614 		x=0;
       
  1615 		before = User::NTickCount();
       
  1616 		for (i=0; i<aCount; ++i)
       
  1617 			{
       
  1618 			x += 0x58b90bfb;
       
  1619 			TInt r = array.FindInOrder(x);
       
  1620 			test(r>=0);
       
  1621 			array.Remove(r);
       
  1622 			}
       
  1623 		after = User::NTickCount();
       
  1624 		diff = after - before;
       
  1625 		diff *= NanoTickPeriod;
       
  1626 		avg = (double)diff / (double)aCount;
       
  1627 		test.Printf(_L("ARRAY:      %d deletions take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1628 		array.Close();
       
  1629 		}
       
  1630 
       
  1631 	INTSET(set);
       
  1632 	x=0;
       
  1633 	RAllocator* pA = 0;
       
  1634 	RDummyAllocator da;
       
  1635 	if (aReserve)
       
  1636 		{
       
  1637 		test(set.Reserve(aCount)==KErrNone);
       
  1638 		pA = User::SwitchAllocator(&da);			// check that no memory accesses occur
       
  1639 		test(set.Reserve(10)==KErrNone);			// shouldn't need to do anything
       
  1640 		test(set.Reserve(aCount/2)==KErrNone);		// shouldn't need to do anything
       
  1641 		test(set.Reserve(aCount)==KErrNone);		// shouldn't need to do anything
       
  1642 		}
       
  1643 	before = User::NTickCount();
       
  1644 	for (i=0; i<aCount; ++i)
       
  1645 		{
       
  1646 		x += 0x58b90bfb;
       
  1647 		TInt r = set.Insert(x);	// we have no heap here so this tests that Reserve() has preallocated sufficient memory
       
  1648 		test(r==KErrNone);
       
  1649 		}
       
  1650 	after = User::NTickCount();
       
  1651 	diff = after - before;
       
  1652 	diff *= NanoTickPeriod;
       
  1653 	avg = (double)diff / (double)aCount;
       
  1654 	if (aReserve)
       
  1655 		User::SwitchAllocator(pA);
       
  1656 	test.Printf(_L("HASH TABLE: %d insertions take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1657 
       
  1658 	x=0;
       
  1659 	before = User::NTickCount();
       
  1660 	for (i=0; i<aCount; ++i)
       
  1661 		{
       
  1662 		x += 0x58b90bfb;
       
  1663 		const TInt* p = set.Find(x);
       
  1664 		test(p!=0);
       
  1665 		}
       
  1666 	after = User::NTickCount();
       
  1667 	diff = after - before;
       
  1668 	diff *= NanoTickPeriod;
       
  1669 	avg = (double)diff / (double)aCount;
       
  1670 	test.Printf(_L("HASH TABLE: %d successful finds take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1671 
       
  1672 	before = User::NTickCount();
       
  1673 	for (i=0; i<aCount; ++i)
       
  1674 		{
       
  1675 		x += 0x58b90bfb;
       
  1676 		const TInt* p = set.Find(x);
       
  1677 		test(!p);
       
  1678 		}
       
  1679 	after = User::NTickCount();
       
  1680 	diff = after - before;
       
  1681 	diff *= NanoTickPeriod;
       
  1682 	avg = (double)diff / (double)aCount;
       
  1683 	test.Printf(_L("HASH TABLE: %d unsuccessful finds take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1684 
       
  1685 	x=0;
       
  1686 	before = User::NTickCount();
       
  1687 	for (i=0; i<aCount; ++i)
       
  1688 		{
       
  1689 		x += 0x58b90bfb;
       
  1690 		TInt r = set.Remove(x);
       
  1691 		test(r==KErrNone);
       
  1692 		}
       
  1693 	after = User::NTickCount();
       
  1694 	diff = after - before;
       
  1695 	diff *= NanoTickPeriod;
       
  1696 	avg = (double)diff / (double)aCount;
       
  1697 	test.Printf(_L("HASH TABLE: %d deletions take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1698 	set.Close();
       
  1699 	}
       
  1700 
       
  1701 TInt des8order(const TDesC8& aA, const TDesC8& aB)
       
  1702 	{
       
  1703 	return aA.Compare(aB);
       
  1704 	}
       
  1705 
       
  1706 void StringBenchmark8(TInt aCount, TBool aReserve)
       
  1707 	{
       
  1708 	RPointerArray<TDesC8> array;
       
  1709 	TUint32 before, after, diff;
       
  1710 	TInt x=0;
       
  1711 	TInt i;
       
  1712 	double avg;
       
  1713 	const TDesC8** base = (const TDesC8**)&DesC8Array[0];
       
  1714 	test(base[1331] == DesC8Array[1331]);
       
  1715 
       
  1716 	test.Printf(_L("**** 8 BIT STRING BENCHMARKS ***\n"));
       
  1717 
       
  1718 	if (!aReserve)
       
  1719 		{
       
  1720 		before = User::NTickCount();
       
  1721 		for (i=0; i<aCount; ++i)
       
  1722 			{
       
  1723 			x = (x+4339)&0x3fff;
       
  1724 			TInt r = array.InsertInOrder(base[x], &des8order);
       
  1725 			test(r==KErrNone);
       
  1726 			}
       
  1727 		after = User::NTickCount();
       
  1728 		diff = after - before;
       
  1729 		diff *= NanoTickPeriod;
       
  1730 		avg = (double)diff / (double)aCount;
       
  1731 		test.Printf(_L("ARRAY:      %d insertions take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1732 
       
  1733 		x=0;
       
  1734 		before = User::NTickCount();
       
  1735 		for (i=0; i<aCount; ++i)
       
  1736 			{
       
  1737 			x = (x+4339)&0x3fff;
       
  1738 			TInt r = array.FindInOrder(base[x], &des8order);
       
  1739 			test(r>=0);
       
  1740 			}
       
  1741 		after = User::NTickCount();
       
  1742 		diff = after - before;
       
  1743 		diff *= NanoTickPeriod;
       
  1744 		avg = (double)diff / (double)aCount;
       
  1745 		test.Printf(_L("ARRAY:      %d successful finds take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1746 
       
  1747 		before = User::NTickCount();
       
  1748 		for (i=0; i<aCount; ++i)
       
  1749 			{
       
  1750 			x = (x+4339)&0x3fff;
       
  1751 			TInt r = array.FindInOrder(base[x], &des8order);
       
  1752 			test(r==KErrNotFound);
       
  1753 			}
       
  1754 		after = User::NTickCount();
       
  1755 		diff = after - before;
       
  1756 		diff *= NanoTickPeriod;
       
  1757 		avg = (double)diff / (double)aCount;
       
  1758 		test.Printf(_L("ARRAY:      %d unsuccessful finds take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1759 
       
  1760 		x=0;
       
  1761 		before = User::NTickCount();
       
  1762 		for (i=0; i<aCount; ++i)
       
  1763 			{
       
  1764 			x = (x+4339)&0x3fff;
       
  1765 			TInt r = array.FindInOrder(base[x], &des8order);
       
  1766 			test(r>=0);
       
  1767 			array.Remove(r);
       
  1768 			}
       
  1769 		after = User::NTickCount();
       
  1770 		diff = after - before;
       
  1771 		diff *= NanoTickPeriod;
       
  1772 		avg = (double)diff / (double)aCount;
       
  1773 		test.Printf(_L("ARRAY:      %d deletions take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1774 		array.Close();
       
  1775 		}
       
  1776 
       
  1777 	STRSET8(set);
       
  1778 	x=0;
       
  1779 	if (aReserve)
       
  1780 		test(set.Reserve(aCount)==KErrNone);
       
  1781 	before = User::NTickCount();
       
  1782 	for (i=0; i<aCount; ++i)
       
  1783 		{
       
  1784 		x = (x+4339)&0x3fff;
       
  1785 		TInt r = set.Insert(base[x]);
       
  1786 		test(r==KErrNone);
       
  1787 		}
       
  1788 	after = User::NTickCount();
       
  1789 	diff = after - before;
       
  1790 	diff *= NanoTickPeriod;
       
  1791 	avg = (double)diff / (double)aCount;
       
  1792 	test.Printf(_L("HASH TABLE: %d insertions take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1793 
       
  1794 	x=0;
       
  1795 	before = User::NTickCount();
       
  1796 	for (i=0; i<aCount; ++i)
       
  1797 		{
       
  1798 		x = (x+4339)&0x3fff;
       
  1799 		const TDesC8* p = set.Find(*base[x]);
       
  1800 		test(p!=0);
       
  1801 		}
       
  1802 	after = User::NTickCount();
       
  1803 	diff = after - before;
       
  1804 	diff *= NanoTickPeriod;
       
  1805 	avg = (double)diff / (double)aCount;
       
  1806 	test.Printf(_L("HASH TABLE: %d successful finds take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1807 
       
  1808 	before = User::NTickCount();
       
  1809 	for (i=0; i<aCount; ++i)
       
  1810 		{
       
  1811 		x = (x+4339)&0x3fff;
       
  1812 		const TDesC8* p = set.Find(*base[x]);
       
  1813 		test(!p);
       
  1814 		}
       
  1815 	after = User::NTickCount();
       
  1816 	diff = after - before;
       
  1817 	diff *= NanoTickPeriod;
       
  1818 	avg = (double)diff / (double)aCount;
       
  1819 	test.Printf(_L("HASH TABLE: %d unsuccessful finds take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1820 
       
  1821 	x=0;
       
  1822 	before = User::NTickCount();
       
  1823 	for (i=0; i<aCount; ++i)
       
  1824 		{
       
  1825 		x = (x+4339)&0x3fff;
       
  1826 		TInt r = set.Remove(base[x]);
       
  1827 		test(r==KErrNone);
       
  1828 		}
       
  1829 	after = User::NTickCount();
       
  1830 	diff = after - before;
       
  1831 	diff *= NanoTickPeriod;
       
  1832 	avg = (double)diff / (double)aCount;
       
  1833 	test.Printf(_L("HASH TABLE: %d deletions take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1834 	set.Close();
       
  1835 	}
       
  1836 
       
  1837 TInt des16order(const TDesC16& aA, const TDesC16& aB)
       
  1838 	{
       
  1839 	return aA.Compare(aB);
       
  1840 	}
       
  1841 
       
  1842 void StringBenchmark16(TInt aCount, TBool aReserve)
       
  1843 	{
       
  1844 	RPointerArray<TDesC16> array;
       
  1845 	TUint32 before, after, diff;
       
  1846 	TInt x=0;
       
  1847 	TInt i;
       
  1848 	double avg;
       
  1849 	const TDesC16** base = (const TDesC16**)&DesC16Array[0];
       
  1850 	test(base[1331] == DesC16Array[1331]);
       
  1851 
       
  1852 	test.Printf(_L("**** 16 BIT STRING BENCHMARKS ***\n"));
       
  1853 
       
  1854 	if (!aReserve)
       
  1855 		{
       
  1856 		before = User::NTickCount();
       
  1857 		for (i=0; i<aCount; ++i)
       
  1858 			{
       
  1859 			x = (x+4339)&0x3fff;
       
  1860 			TInt r = array.InsertInOrder(base[x], &des16order);
       
  1861 			test(r==KErrNone);
       
  1862 			}
       
  1863 		after = User::NTickCount();
       
  1864 		diff = after - before;
       
  1865 		diff *= NanoTickPeriod;
       
  1866 		avg = (double)diff / (double)aCount;
       
  1867 		test.Printf(_L("ARRAY:      %d insertions take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1868 
       
  1869 		x=0;
       
  1870 		before = User::NTickCount();
       
  1871 		for (i=0; i<aCount; ++i)
       
  1872 			{
       
  1873 			x = (x+4339)&0x3fff;
       
  1874 			TInt r = array.FindInOrder(base[x], &des16order);
       
  1875 			test(r>=0);
       
  1876 			}
       
  1877 		after = User::NTickCount();
       
  1878 		diff = after - before;
       
  1879 		diff *= NanoTickPeriod;
       
  1880 		avg = (double)diff / (double)aCount;
       
  1881 		test.Printf(_L("ARRAY:      %d successful finds take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1882 
       
  1883 		before = User::NTickCount();
       
  1884 		for (i=0; i<aCount; ++i)
       
  1885 			{
       
  1886 			x = (x+4339)&0x3fff;
       
  1887 			TInt r = array.FindInOrder(base[x], &des16order);
       
  1888 			test(r==KErrNotFound);
       
  1889 			}
       
  1890 		after = User::NTickCount();
       
  1891 		diff = after - before;
       
  1892 		diff *= NanoTickPeriod;
       
  1893 		avg = (double)diff / (double)aCount;
       
  1894 		test.Printf(_L("ARRAY:      %d unsuccessful finds take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1895 
       
  1896 		x=0;
       
  1897 		before = User::NTickCount();
       
  1898 		for (i=0; i<aCount; ++i)
       
  1899 			{
       
  1900 			x = (x+4339)&0x3fff;
       
  1901 			TInt r = array.FindInOrder(base[x], &des16order);
       
  1902 			test(r>=0);
       
  1903 			array.Remove(r);
       
  1904 			}
       
  1905 		after = User::NTickCount();
       
  1906 		diff = after - before;
       
  1907 		diff *= NanoTickPeriod;
       
  1908 		avg = (double)diff / (double)aCount;
       
  1909 		test.Printf(_L("ARRAY:      %d deletions take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1910 		array.Close();
       
  1911 		}
       
  1912 
       
  1913 	STRSET16(set);
       
  1914 	x=0;
       
  1915 	if (aReserve)
       
  1916 		test(set.Reserve(aCount)==KErrNone);
       
  1917 	before = User::NTickCount();
       
  1918 	for (i=0; i<aCount; ++i)
       
  1919 		{
       
  1920 		x = (x+4339)&0x3fff;
       
  1921 		TInt r = set.Insert(base[x]);
       
  1922 		test(r==KErrNone);
       
  1923 		}
       
  1924 	after = User::NTickCount();
       
  1925 	diff = after - before;
       
  1926 	diff *= NanoTickPeriod;
       
  1927 	avg = (double)diff / (double)aCount;
       
  1928 	test.Printf(_L("HASH TABLE: %d insertions take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1929 
       
  1930 	x=0;
       
  1931 	before = User::NTickCount();
       
  1932 	for (i=0; i<aCount; ++i)
       
  1933 		{
       
  1934 		x = (x+4339)&0x3fff;
       
  1935 		const TDesC16* p = set.Find(*base[x]);
       
  1936 		test(p!=0);
       
  1937 		}
       
  1938 	after = User::NTickCount();
       
  1939 	diff = after - before;
       
  1940 	diff *= NanoTickPeriod;
       
  1941 	avg = (double)diff / (double)aCount;
       
  1942 	test.Printf(_L("HASH TABLE: %d successful finds take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1943 
       
  1944 	before = User::NTickCount();
       
  1945 	for (i=0; i<aCount; ++i)
       
  1946 		{
       
  1947 		x = (x+4339)&0x3fff;
       
  1948 		const TDesC16* p = set.Find(*base[x]);
       
  1949 		test(!p);
       
  1950 		}
       
  1951 	after = User::NTickCount();
       
  1952 	diff = after - before;
       
  1953 	diff *= NanoTickPeriod;
       
  1954 	avg = (double)diff / (double)aCount;
       
  1955 	test.Printf(_L("HASH TABLE: %d unsuccessful finds take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1956 
       
  1957 	x=0;
       
  1958 	before = User::NTickCount();
       
  1959 	for (i=0; i<aCount; ++i)
       
  1960 		{
       
  1961 		x = (x+4339)&0x3fff;
       
  1962 		TInt r = set.Remove(base[x]);
       
  1963 		test(r==KErrNone);
       
  1964 		}
       
  1965 	after = User::NTickCount();
       
  1966 	diff = after - before;
       
  1967 	diff *= NanoTickPeriod;
       
  1968 	avg = (double)diff / (double)aCount;
       
  1969 	test.Printf(_L("HASH TABLE: %d deletions take %dus (%.2gus each)\n"), aCount, diff, avg);
       
  1970 	set.Close();
       
  1971 	}
       
  1972 
       
  1973 void Benchmark()
       
  1974 	{
       
  1975 	test.Next(_L("Benchmarks ..."));
       
  1976 
       
  1977 	IntegerBenchmark(1000, EFalse);
       
  1978 	IntegerBenchmark(1000, ETrue);
       
  1979 	IntegerBenchmark(2000, EFalse);
       
  1980 	IntegerBenchmark(2000, ETrue);
       
  1981 	IntegerBenchmark(4000, EFalse);
       
  1982 	IntegerBenchmark(4000, ETrue);
       
  1983 	IntegerBenchmark(8000, EFalse);
       
  1984 	IntegerBenchmark(8000, ETrue);
       
  1985 	IntegerBenchmark(16000, EFalse);
       
  1986 	IntegerBenchmark(16000, ETrue);
       
  1987 	IntegerBenchmark(32000, EFalse);
       
  1988 	IntegerBenchmark(32000, ETrue);
       
  1989 
       
  1990 	PopulateArray8(16384);
       
  1991 	StringBenchmark8(1000, EFalse);
       
  1992 	StringBenchmark8(2000, EFalse);
       
  1993 	StringBenchmark8(4000, EFalse);
       
  1994 	StringBenchmark8(8000, EFalse);
       
  1995 	DesC8Array.ResetAndDestroy();
       
  1996 
       
  1997 	PopulateArray16(16384);
       
  1998 	StringBenchmark16(1000, EFalse);
       
  1999 	StringBenchmark16(2000, EFalse);
       
  2000 	StringBenchmark16(4000, EFalse);
       
  2001 	StringBenchmark16(8000, EFalse);
       
  2002 	DesC16Array.ResetAndDestroy();
       
  2003 	}
       
  2004 
       
  2005 
       
  2006 TUint32 mp(TUint32 x)
       
  2007 	{
       
  2008 	TUint32 x3 = x+(x<<1);
       
  2009 	TUint32 x7 = x+(x3<<1);
       
  2010 	TUint32 x15 = x+(x7<<1);
       
  2011 	TUint32 y = x + (x7<<3) + (x3<<7) + (x15<<11) + (x7<<16) + (x3<<20) + (x15<<25) + (x<<31);
       
  2012 	return y;
       
  2013 	}
       
  2014 
       
  2015 TUint32 strh(const char* aIn)
       
  2016 	{
       
  2017 	TUint32 h = 0;
       
  2018 	TInt i = 0;
       
  2019 	while (*aIn)
       
  2020 		{
       
  2021 		if (i==0)
       
  2022 			h = mp(h);
       
  2023 		TUint32 c = *aIn++;
       
  2024 		c <<= (8*i);
       
  2025 		h ^= c;
       
  2026 		i = (i+1)&3;
       
  2027 		}
       
  2028 	return mp(h);
       
  2029 	}
       
  2030 
       
  2031 TUint32 strh(const wch* aIn)
       
  2032 	{
       
  2033 	TUint32 h = 0;
       
  2034 	TInt i = 0;
       
  2035 	while (*aIn)
       
  2036 		{
       
  2037 		if (i==0)
       
  2038 			h = mp(h);
       
  2039 		TUint32 c = *aIn++;
       
  2040 		switch (i)
       
  2041 			{
       
  2042 			case 0:		break;
       
  2043 			case 1:		c<<=16; break;
       
  2044 			case 2:		c<<=8; break;
       
  2045 			default:	c=(c<<24)|(c>>8); break;
       
  2046 			};
       
  2047 		h ^= c;
       
  2048 		i = (i+1)&3;
       
  2049 		}
       
  2050 	return mp(h);
       
  2051 	}
       
  2052 
       
  2053 void TestHash(TInt* aIn, TUint32 aExpected)
       
  2054 	{
       
  2055 	THashFunction32<TInt*> hf(&DefaultHash::IntegerPtr);
       
  2056 	TUint32 out = hf.Hash(aIn);
       
  2057 	test(aExpected == mp((TUint32)aIn));
       
  2058 	if (out != aExpected)
       
  2059 		{
       
  2060 		test.Printf(_L("Hashing %08x Expected %08x Got %08x\n"), aIn, aExpected, out);
       
  2061 		test(0);
       
  2062 		}
       
  2063 	}
       
  2064 
       
  2065 void TestHash(TDesC8* aIn, TUint32 aExpected)
       
  2066 	{
       
  2067 	THashFunction32<TDesC8*> hf(&DefaultHash::Des8Ptr);
       
  2068 	TUint32 out = hf.Hash(aIn);
       
  2069 	test(aExpected == mp((TUint32)aIn));
       
  2070 	if (out != aExpected)
       
  2071 		{
       
  2072 		test.Printf(_L("Hashing %08x Expected %08x Got %08x\n"), aIn, aExpected, out);
       
  2073 		test(0);
       
  2074 		}
       
  2075 	}
       
  2076 
       
  2077 void TestHash(TDesC16* aIn, TUint32 aExpected)
       
  2078 	{
       
  2079 	THashFunction32<TDesC16*> hf(&DefaultHash::Des16Ptr);
       
  2080 	TUint32 out = hf.Hash(aIn);
       
  2081 	test(aExpected == mp((TUint32)aIn));
       
  2082 	if (out != aExpected)
       
  2083 		{
       
  2084 		test.Printf(_L("Hashing %08x Expected %08x Got %08x\n"), aIn, aExpected, out);
       
  2085 		test(0);
       
  2086 		}
       
  2087 	}
       
  2088 
       
  2089 
       
  2090 void TestHash(TInt aIn, TUint32 aExpected)
       
  2091 	{
       
  2092 	THashFunction32<TInt> hf(&DefaultHash::Integer);
       
  2093 	TUint32 out = hf.Hash(aIn);
       
  2094 	test(aExpected == mp((TUint32)aIn));
       
  2095 	if (out != aExpected)
       
  2096 		{
       
  2097 		test.Printf(_L("Hashing %08x Expected %08x Got %08x\n"), aIn, aExpected, out);
       
  2098 		test(0);
       
  2099 		}
       
  2100 	}
       
  2101 
       
  2102 void TestHash(const char* aIn, TUint32 aExpected)
       
  2103 	{
       
  2104 	THashFunction32<TDesC8> hf(&DefaultHash::Des8);
       
  2105 	TPtrC8 p((const TUint8*)aIn);
       
  2106 	TUint32 out = hf.Hash(p);
       
  2107 	test(aExpected == strh(aIn));
       
  2108 	if (out != aExpected)
       
  2109 		{
       
  2110 		TBuf<256> buf;
       
  2111 		buf.Copy(p);
       
  2112 		test.Printf(_L("Hashing %S (len %d) Expected %08x Got %08x\n"), &buf, p.Length(), aExpected, out);
       
  2113 		test(0);
       
  2114 		}
       
  2115 	}
       
  2116 
       
  2117 void TestHash(const wch* aIn)
       
  2118 	{
       
  2119 	THashFunction32<TDesC16> hf(&DefaultHash::Des16);
       
  2120 	TPtrC16 p((const TUint16*)aIn);
       
  2121 	TUint32 out = hf.Hash(p);
       
  2122 	TUint32 exp = strh(aIn);
       
  2123 	if (out != exp)
       
  2124 		{
       
  2125 		test.Printf(_L("Hashing %S (len %d) Expected %08x Got %08x\n"), &p, p.Size(), exp, out);
       
  2126 		test(0);
       
  2127 		}
       
  2128 	}
       
  2129 
       
  2130 void TestHash()
       
  2131 	{
       
  2132 	test.Next(_L("Test integer hash"));
       
  2133 	TestHash(1,0x9e3779b9);
       
  2134 	TestHash(2,0x3c6ef372);
       
  2135 	TestHash(4,0x78dde6e4);
       
  2136 	TestHash(8,0xf1bbcdc8);
       
  2137 	TestHash(16,0xe3779b90);
       
  2138 	TestHash(0xc90fdaa2,0xbf999112);
       
  2139 	TestHash(0xb504f334,0x7fb35494);
       
  2140 	TestHash(0xddb3d743,0xd11a3a6b);
       
  2141 	TestHash(0xadf85458,0x873a8b98);
       
  2142 	TestHash(0x11730859,0xb4321951);
       
  2143 	TestHash(0x64636261,0x8628f119);
       
  2144 	}
       
  2145 
       
  2146 void TestIntegerPtrHash()
       
  2147 	{
       
  2148 	TInt i[5];
       
  2149 	TInt* ptr;
       
  2150 	test.Next(_L("Test Integer pointer hash"));
       
  2151 	for (ptr=i; ptr<i+5; ptr++)
       
  2152 		{
       
  2153 		TestHash(ptr, DefaultHash::IntegerPtr(ptr));
       
  2154 		}
       
  2155 	}
       
  2156 
       
  2157 void TestDes8PtrHash()
       
  2158 	{
       
  2159 	TBuf8<8> i[5];
       
  2160 	TDesC8* ptr;
       
  2161 	test.Next(_L("Test Des8 pointer hash"));
       
  2162 	for (ptr=i; ptr<i+5; ptr++)
       
  2163 		{
       
  2164 		TestHash(ptr, DefaultHash::Des8Ptr(ptr));
       
  2165 		}
       
  2166 	}
       
  2167 
       
  2168 void TestDes16PtrHash()
       
  2169 	{
       
  2170 	TBuf16<8> i[5];
       
  2171 	TDesC16* ptr;
       
  2172 	test.Next(_L("Test Des16 pointer hash"));
       
  2173 	for (ptr=i; ptr<i+5; ptr++)
       
  2174 		{
       
  2175 		TestHash(ptr, DefaultHash::Des16Ptr(ptr));
       
  2176 		}
       
  2177 	}
       
  2178 
       
  2179 const char teststr[] = "zyxwvutsrq";
       
  2180 void TestStringHash()
       
  2181 	{
       
  2182 	test.Next(_L("Test 8 bit string hash"));
       
  2183 	TestHash("",0x0);
       
  2184 	TestHash("a",0xf3051f19);
       
  2185 	TestHash("b",0x913c98d2);
       
  2186 	TestHash("ab",0x2f9df119);
       
  2187 	TestHash("ba",0x965bb1d2);
       
  2188 	TestHash("abc",0x4228f119);
       
  2189 	TestHash("abcd",0x8628f119);
       
  2190 	TestHash("abcde",0xb75e1e9c);
       
  2191 	TestHash("abcdef",0x3693149c);
       
  2192 	TestHash("abcdefg",0xc1c2149c);
       
  2193 	TestHash("abcdefgh",0xe9c2149c);
       
  2194 	TestHash("abcdefghi",0x5fcbf20d);
       
  2195 	TestHash("abcdefghj",0xfe036bc6);
       
  2196 
       
  2197 	TestHash(teststr, 0x108ca51e);
       
  2198 	TestHash(teststr+1, 0x551002ad);
       
  2199 	TestHash(teststr+2, 0x37dc0d6c);
       
  2200 	TestHash(teststr+3, 0x2937f92c);
       
  2201 	TestHash(teststr+4, 0xf0818a94);
       
  2202 	TestHash(teststr+5, 0xb1b25f1c);
       
  2203 	TestHash(teststr+6, 0x7a3342d4);
       
  2204 	TestHash(teststr+7, 0x81c9101b);
       
  2205 	TestHash(teststr+8, 0xf16edd62);
       
  2206 	TestHash(teststr+9, 0xd67cbaa9);
       
  2207 	TestHash(teststr+10, 0);
       
  2208 
       
  2209 	char t[16];
       
  2210 	int i,j;
       
  2211 	for (i=0; i<=4; ++i)
       
  2212 		{
       
  2213 		for (j=0; j<=10; ++j)
       
  2214 			{
       
  2215 			const char* s = teststr + j;
       
  2216 			int l = User::StringLength((const TUint8*)s);
       
  2217 			memset(t, 0xbb, 16);
       
  2218 			memcpy(t+i, s, l+1);
       
  2219 			TUint32 h;
       
  2220 			switch (j)
       
  2221 				{
       
  2222 				case 0: h = 0x108ca51e; break;
       
  2223 				case 1: h = 0x551002ad; break;
       
  2224 				case 2: h = 0x37dc0d6c; break;
       
  2225 				case 3: h = 0x2937f92c; break;
       
  2226 				case 4: h = 0xf0818a94; break;
       
  2227 				case 5: h = 0xb1b25f1c; break;
       
  2228 				case 6: h = 0x7a3342d4; break;
       
  2229 				case 7: h = 0x81c9101b; break;
       
  2230 				case 8: h = 0xf16edd62; break;
       
  2231 				case 9: h = 0xd67cbaa9; break;
       
  2232 				default: h = 0; break;
       
  2233 				};
       
  2234 			TestHash(t+i, h);
       
  2235 			}
       
  2236 		}
       
  2237 	}
       
  2238 
       
  2239 const wch wteststr[] = L"zyxwvutsrq";
       
  2240 void TestWStringHash()
       
  2241 	{
       
  2242 	test.Next(_L("Test 16 bit string hash"));
       
  2243 	TestHash(L"");
       
  2244 	TestHash(L"a");
       
  2245 	TestHash(L"b");
       
  2246 	TestHash(L"ab");
       
  2247 	TestHash(L"ba");
       
  2248 	TestHash(L"abc");
       
  2249 	TestHash(L"abcd");
       
  2250 	TestHash(L"abcde");
       
  2251 	TestHash(L"abcdef");
       
  2252 	TestHash(L"abcdefg");
       
  2253 	TestHash(L"abcdefgh");
       
  2254 	TestHash(L"abcdefghi");
       
  2255 	TestHash(L"abcdefghj");
       
  2256 
       
  2257 	TestHash(wteststr);
       
  2258 	TestHash(wteststr+1);
       
  2259 	TestHash(wteststr+2);
       
  2260 	TestHash(wteststr+3);
       
  2261 	TestHash(wteststr+4);
       
  2262 	TestHash(wteststr+5);
       
  2263 	TestHash(wteststr+6);
       
  2264 	TestHash(wteststr+7);
       
  2265 	TestHash(wteststr+8);
       
  2266 	TestHash(wteststr+9);
       
  2267 	TestHash(wteststr+10);
       
  2268 
       
  2269 	wch t[16];
       
  2270 	int i,j;
       
  2271 	for (i=0; i<=4; ++i)
       
  2272 		{
       
  2273 		for (j=0; j<=10; ++j)
       
  2274 			{
       
  2275 			const wch* s = wteststr + j;
       
  2276 			int l = User::StringLength((const TUint16*)s);
       
  2277 			memset(t, 0xbb, 2*16);
       
  2278 			memcpy(t+i, s, 2*(l+1));
       
  2279 			TestHash(t+i);
       
  2280 			}
       
  2281 		}
       
  2282 	}
       
  2283 template <class K,class V> 
       
  2284 void TestHashMapPtr(RHashMap<K,V> &map, K ptr, V* i)
       
  2285 {
       
  2286 	test(map.Reserve(5) == KErrNone);
       
  2287 	for (ptr=i;ptr<i+5;ptr++)
       
  2288 		{
       
  2289 		test(map.Insert(ptr,*ptr) == KErrNone);
       
  2290 		}
       
  2291 	for (ptr=i+4;ptr>=i;ptr--)
       
  2292 		{
       
  2293 		test(*(map.Find(ptr)) == *ptr);
       
  2294 		}
       
  2295 	test(map.Count() == 5);
       
  2296 	test(map.Remove(i) == KErrNone);
       
  2297 	test(map.Count()==4);
       
  2298 	test(map.Find(i)==NULL);
       
  2299 	map.Close();
       
  2300 }
       
  2301 
       
  2302 void TestPtrHashMaps()
       
  2303 	{	
       
  2304 
       
  2305 	test.Next(_L("Test RHashMap of default pointer types"));
       
  2306 	TInt i[5];
       
  2307 	TInt *ptr=i;
       
  2308 	RHashMap<TInt*,TInt> mp;
       
  2309 	TestHashMapPtr(mp, ptr, i);
       
  2310 
       
  2311 	TInt32 i1[5];
       
  2312 	TInt32 *ptr1=i1;
       
  2313 	RHashMap<TInt32*,TInt32> mp1;
       
  2314 	TestHashMapPtr(mp1,ptr1,i1);
       
  2315 	
       
  2316 	TUint i2[5];
       
  2317 	TUint *ptr2=i2;
       
  2318 	RHashMap<TUint*,TUint> mp2;
       
  2319 	TestHashMapPtr(mp2,ptr2,i2);
       
  2320 
       
  2321 	TUint32 i3[5];
       
  2322 	TUint32 *ptr3=i3;
       
  2323 	RHashMap<TUint32*,TUint32> mp3;
       
  2324 	TestHashMapPtr(mp3,ptr3,i3);
       
  2325 
       
  2326 	TBuf8<5> i4[5];
       
  2327 	TBuf8<5> *ptr4=i4;
       
  2328 	RHashMap<TDesC8*,TDesC8> mp4;
       
  2329 	for (ptr4=i4; ptr4 < i4+5; ptr4++) 
       
  2330 		{
       
  2331 		test(mp4.Insert(ptr4,*ptr4) == KErrNone);
       
  2332 		}
       
  2333 	for (ptr4=i4+4; ptr4 >= i4; ptr4--) 
       
  2334 		{
       
  2335 		test(*(mp4.Find(ptr4)) == *ptr4);
       
  2336 		}
       
  2337 	test(mp4.Count()==5);
       
  2338 	test(mp4.Remove(i4) == KErrNone);
       
  2339 	test(mp4.Find(i4) == NULL);
       
  2340 	test(mp4.Count()==4);
       
  2341 	mp4.Close();
       
  2342 
       
  2343 
       
  2344 	TBuf16<5> i5[5];
       
  2345 	TBuf16<5> *ptr5=i5;
       
  2346 	RHashMap<TDesC16*,TDesC16> mp5;
       
  2347 	for (ptr5=i5; ptr5 < i5+5; ptr5++) 
       
  2348 		{
       
  2349 		test(mp5.Insert(ptr5,*ptr5) == KErrNone);
       
  2350 		}
       
  2351 	for (ptr5=i5+4; ptr5 >= i5; ptr5--) 
       
  2352 		{
       
  2353 		test(*(mp5.Find(ptr5)) == *ptr5);
       
  2354 		}
       
  2355 	test(mp5.Count()==5);
       
  2356 	test(mp5.Remove(i5) == KErrNone);
       
  2357 	test(mp5.Find(i5) == NULL);
       
  2358 	test(mp5.Count()==4);
       
  2359 	mp5.Close();
       
  2360 	
       
  2361 }
       
  2362 
       
  2363 /** Tests that Reserve() will always allocate memory for new tables 
       
  2364 	even for small reserve sizes
       
  2365 	See DEF087906.
       
  2366 */
       
  2367 void TestSmallReserve()
       
  2368 	{
       
  2369 	test.Next(_L("Test RHashTableBase::Reserve preallocates memory, even for small no of elements"));
       
  2370 	RAllocator* pA = 0;
       
  2371 	RDummyAllocator da;
       
  2372 	
       
  2373 	// Reserve should allocate the memory required for the table of 1 element
       
  2374 	INTSET(set);
       
  2375 	RHashMap<TInt,TInt> hashMap;
       
  2376 	
       
  2377 	test(set.Reserve(1) == KErrNone);
       
  2378 	test(hashMap.Reserve(1) == KErrNone);
       
  2379 	
       
  2380 	pA = User::SwitchAllocator(&da);
       
  2381 	
       
  2382 	// No more memory should be allocated for the table as it should
       
  2383 	// have been already allocated by Reserve()
       
  2384 	test(set.Insert(123) == KErrNone);
       
  2385 	test(hashMap.Insert(123,456) == KErrNone);
       
  2386 	
       
  2387 	// Switch back to allow set to be closed
       
  2388 	User::SwitchAllocator(pA);
       
  2389 	set.Close();
       
  2390 	hashMap.Close();
       
  2391 	}
       
  2392 	
       
  2393 
       
  2394 TInt E32Main()
       
  2395 	{
       
  2396 	test.Title();
       
  2397 
       
  2398 	test(HAL::Get(HAL::ENanoTickPeriod, NanoTickPeriod)==KErrNone);
       
  2399 	test.Printf(_L("NanoTickPeriod %dus\n"), NanoTickPeriod);
       
  2400 
       
  2401 	__UHEAP_MARK;
       
  2402 
       
  2403 	test.Start(_L("Testing hash tables"));
       
  2404 
       
  2405 	TestHash();
       
  2406 	TestStringHash();
       
  2407 	TestWStringHash();
       
  2408 	TestIntegerPtrHash();
       
  2409 	TestDes8PtrHash();
       
  2410 	TestDes16PtrHash();
       
  2411 
       
  2412 	TestHashSet();
       
  2413 	TestHashIter();
       
  2414 	TestHashMap();
       
  2415 	TestPtrHashMaps();
       
  2416 
       
  2417 	PopulateArray8(4096);
       
  2418 	PopulateArray16(4096);
       
  2419 	TestPtrHashSet();
       
  2420 	TestPtrHashMap();
       
  2421 	DesC16Array.ResetAndDestroy();
       
  2422 	DesC8Array.ResetAndDestroy();
       
  2423 
       
  2424 
       
  2425 	TestOOM();
       
  2426 	Benchmark();
       
  2427 	TestSmallReserve();
       
  2428 
       
  2429 	test.End();
       
  2430 
       
  2431 	__UHEAP_MARKEND;
       
  2432 	return 0;
       
  2433 	}