kernel/eka/common/heap.cpp
branchRCL_3
changeset 44 3e88ff8f41d5
equal deleted inserted replaced
43:c1f20ce4abcf 44:3e88ff8f41d5
       
     1 // Copyright (c) 1994-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 // e32\common\heap.cpp
       
    15 // 
       
    16 //
       
    17 
       
    18 #include "common.h"
       
    19 #ifdef __KERNEL_MODE__
       
    20 #include <kernel/kern_priv.h>
       
    21 #endif
       
    22 
       
    23 #ifdef _DEBUG
       
    24 #define __SIMULATE_ALLOC_FAIL(s)	if (CheckForSimulatedAllocFail()) {s}
       
    25 #define	__CHECK_CELL(p)				CheckCell(p)
       
    26 #define	__ZAP_CELL(p)				memset( ((TUint8*)p) + RHeap::EAllocCellSize, 0xde, p->len - RHeap::EAllocCellSize)
       
    27 #define __DEBUG_SAVE(p)				TInt dbgNestLevel = ((SDebugCell*)p)->nestingLevel
       
    28 #define __DEBUG_RESTORE(p)			((SDebugCell*)(((TUint8*)p)-EAllocCellSize))->nestingLevel = dbgNestLevel
       
    29 #else
       
    30 #define __SIMULATE_ALLOC_FAIL(s)
       
    31 #define	__CHECK_CELL(p)
       
    32 #define	__ZAP_CELL(p)
       
    33 #define __DEBUG_SAVE(p)
       
    34 #define __DEBUG_RESTORE(p)
       
    35 #endif
       
    36 
       
    37 #define __NEXT_CELL(p)				((SCell*)(((TUint8*)p)+p->len))
       
    38 
       
    39 #define __POWER_OF_2(x)				((TUint32)((x)^((x)-1))>=(TUint32)(x))
       
    40 
       
    41 #define __MEMORY_MONITOR_CHECK_CELL(p) \
       
    42 					{ \
       
    43 					TLinAddr m = TLinAddr(iAlign-1); \
       
    44 					SCell* c = (SCell*)(((TUint8*)p)-EAllocCellSize); \
       
    45 					if((c->len & m) || (c->len<iMinCell) || ((TUint8*)c<iBase) || ((TUint8*)__NEXT_CELL(c)>iTop)) \
       
    46 						BTraceContext12(BTrace::EHeap, BTrace::EHeapCorruption, (TUint32)this, (TUint32)p, (TUint32)c->len-EAllocCellSize); \
       
    47 					}
       
    48 					
       
    49 /**
       
    50 @SYMPatchable
       
    51 @publishedPartner
       
    52 @released
       
    53 
       
    54 Defines the minimum cell size of  a heap.
       
    55 
       
    56 The constant can be changed at ROM build time using patchdata OBY keyword.
       
    57 */
       
    58 #ifdef __X86GCC__	// For X86GCC we dont use the proper data import attribute
       
    59 #undef IMPORT_D		// since the constant is not really imported. GCC doesn't 
       
    60 #define IMPORT_D	// allow imports from self.
       
    61 #endif
       
    62 IMPORT_D extern const TInt KHeapMinCellSize;
       
    63 
       
    64 /**
       
    65 @SYMPatchable
       
    66 @publishedPartner
       
    67 @released
       
    68 
       
    69 This constant defines the ratio that determines the amount of hysteresis between heap growing and heap
       
    70 shrinking.
       
    71 It is a 32-bit fixed point number where the radix point is defined to be
       
    72 between bits 7 and 8 (where the LSB is bit 0) i.e. using standard notation, a Q8 or a fx24.8
       
    73 fixed point number.  For example, for a ratio of 2.0, set KHeapShrinkHysRatio=0x200.
       
    74 
       
    75 The heap shrinking hysteresis value is calculated to be:
       
    76 @code
       
    77 KHeapShrinkHysRatio*(iGrowBy>>8)
       
    78 @endcode
       
    79 where iGrowBy is a page aligned value set by the argument, aGrowBy, to the RHeap constructor.
       
    80 The default hysteresis value is iGrowBy bytes i.e. KHeapShrinkHysRatio=2.0.
       
    81 
       
    82 Memory usage may be improved by reducing the heap shrinking hysteresis
       
    83 by setting 1.0 < KHeapShrinkHysRatio < 2.0.  Heap shrinking hysteresis is disabled/removed
       
    84 when KHeapShrinkHysRatio <= 1.0.
       
    85 
       
    86 The constant can be changed at ROM build time using patchdata OBY keyword.
       
    87 */
       
    88 IMPORT_D extern const TInt KHeapShrinkHysRatio;
       
    89 
       
    90 #pragma warning( disable : 4705 )	// statement has no effect
       
    91 UEXPORT_C RHeap::RHeap(TInt aMaxLength, TInt aAlign, TBool aSingleThread)
       
    92 /**
       
    93 @internalComponent
       
    94 */
       
    95 //
       
    96 // Constructor for fixed size heap
       
    97 //
       
    98 	:	iMinLength(aMaxLength), iMaxLength(aMaxLength), iOffset(0), iGrowBy(0), iChunkHandle(0),
       
    99 		iNestingLevel(0), iAllocCount(0), iFailType(ENone), iTestData(NULL)
       
   100 	{
       
   101 	iAlign = aAlign ? aAlign : ECellAlignment;
       
   102 	iPageSize = 0;
       
   103 	iFlags = aSingleThread ? (ESingleThreaded|EFixedSize) : EFixedSize;
       
   104 	Initialise();
       
   105 	}
       
   106 #pragma warning( default : 4705 )
       
   107 
       
   108 
       
   109 
       
   110 
       
   111 UEXPORT_C RHeap::RHeap(TInt aChunkHandle, TInt aOffset, TInt aMinLength, TInt aMaxLength, TInt aGrowBy, TInt aAlign, TBool aSingleThread)
       
   112 /**
       
   113 @internalComponent
       
   114 */
       
   115 //
       
   116 // Constructor for chunk heaps.
       
   117 //
       
   118 	:	iOffset(aOffset), iChunkHandle(aChunkHandle),
       
   119 		iNestingLevel(0), iAllocCount(0), iFailType(ENone), iTestData(NULL)
       
   120 	{
       
   121 	TInt sz = iBase - ((TUint8*)this - iOffset);
       
   122 	GET_PAGE_SIZE(iPageSize);
       
   123 	__ASSERT_ALWAYS(iOffset>=0, HEAP_PANIC(ETHeapNewBadOffset));
       
   124 	iMinLength = Max(aMinLength, sz + EAllocCellSize);
       
   125 	iMinLength = _ALIGN_UP(iMinLength, iPageSize);
       
   126 	iMaxLength = Max(aMaxLength, iMinLength);
       
   127 	iMaxLength = _ALIGN_UP(iMaxLength, iPageSize);
       
   128 	iGrowBy = _ALIGN_UP(aGrowBy, iPageSize);
       
   129 	iFlags = aSingleThread ? ESingleThreaded : 0;
       
   130 	iAlign = aAlign ? aAlign : ECellAlignment;
       
   131 	Initialise();
       
   132 	}
       
   133 
       
   134 
       
   135 
       
   136 
       
   137 UEXPORT_C TAny* RHeap::operator new(TUint aSize, TAny* aBase) __NO_THROW
       
   138 /**
       
   139 @internalComponent
       
   140 */
       
   141 	{
       
   142 	__ASSERT_ALWAYS(aSize>=sizeof(RHeap), HEAP_PANIC(ETHeapNewBadSize));
       
   143 	RHeap* h = (RHeap*)aBase;
       
   144 	h->iAlign = 0x80000000;	// garbage value
       
   145 	h->iBase = ((TUint8*)aBase) + aSize;
       
   146 	return aBase;
       
   147 	}
       
   148 
       
   149 void RHeap::Initialise()
       
   150 //
       
   151 // Initialise the heap.
       
   152 //
       
   153 	{
       
   154 
       
   155 	__ASSERT_ALWAYS((TUint32)iAlign>=sizeof(TAny*) && __POWER_OF_2(iAlign), HEAP_PANIC(ETHeapNewBadAlignment));
       
   156 	iCellCount = 0;
       
   157 	iTotalAllocSize = 0;
       
   158 	iBase = (TUint8*)Align(iBase + EAllocCellSize);
       
   159 	iBase -= EAllocCellSize;
       
   160 	TInt b = iBase - ((TUint8*)this - iOffset);
       
   161 	TInt len = _ALIGN_DOWN(iMinLength - b, iAlign);
       
   162 	iTop = iBase + len;
       
   163 	iMinLength = iTop - ((TUint8*)this - iOffset);
       
   164 	iMinCell = Align(KHeapMinCellSize + Max((TInt)EAllocCellSize, (TInt)EFreeCellSize));
       
   165 #ifdef _DEBUG
       
   166 	memset(iBase, 0xa5, len);
       
   167 #endif
       
   168 	SCell* pM=(SCell*)iBase; // First free cell
       
   169 	iFree.next=pM; // Free list points to first free cell
       
   170 	iFree.len=0; // Stop free from joining this with a free block
       
   171 	pM->next=NULL; // Terminate the free list
       
   172 	pM->len=len; // Set the size of the free cell
       
   173 	}
       
   174 
       
   175 #ifdef _DEBUG
       
   176 void RHeap::CheckCell(const SCell* aCell) const
       
   177 	{
       
   178 	TLinAddr m = TLinAddr(iAlign - 1);
       
   179 
       
   180 	__ASSERT_DEBUG(!(aCell->len & m), HEAP_PANIC(ETHeapBadCellAddress));
       
   181 	__ASSERT_DEBUG(aCell->len >= iMinCell, HEAP_PANIC(ETHeapBadCellAddress));
       
   182 	__ASSERT_DEBUG((TUint8*)aCell>=iBase, HEAP_PANIC(ETHeapBadCellAddress));
       
   183 	__ASSERT_DEBUG((TUint8*)__NEXT_CELL(aCell)<=iTop, HEAP_PANIC(ETHeapBadCellAddress));
       
   184 	}
       
   185 #endif
       
   186 
       
   187 UEXPORT_C RHeap::SCell* RHeap::GetAddress(const TAny* aCell) const
       
   188 //
       
   189 // As much as possible, check a cell address and backspace it
       
   190 // to point at the cell header.
       
   191 //
       
   192 	{
       
   193 
       
   194 	TLinAddr m = TLinAddr(iAlign - 1);
       
   195 	__ASSERT_ALWAYS(!(TLinAddr(aCell)&m), HEAP_PANIC(ETHeapBadCellAddress));
       
   196 
       
   197 	SCell* pC = (SCell*)(((TUint8*)aCell)-EAllocCellSize);
       
   198 	__CHECK_CELL(pC);
       
   199 
       
   200 	return pC;
       
   201 	}
       
   202 
       
   203 
       
   204 
       
   205 
       
   206 UEXPORT_C TInt RHeap::AllocLen(const TAny* aCell) const
       
   207 /**
       
   208 Gets the length of the available space in the specified allocated cell.
       
   209 
       
   210 @param aCell A pointer to the allocated cell.
       
   211 
       
   212 @return The length of the available space in the allocated cell.
       
   213 
       
   214 @panic USER 42 if aCell does not point to  a valid cell.
       
   215 */
       
   216 	{
       
   217 
       
   218 	SCell* pC = GetAddress(aCell);
       
   219 	return pC->len - EAllocCellSize;
       
   220 	}
       
   221 
       
   222 
       
   223 
       
   224 
       
   225 
       
   226 #if !defined(__HEAP_MACHINE_CODED__) || defined(_DEBUG)
       
   227 RHeap::SCell* RHeap::DoAlloc(TInt aSize, SCell*& aLastFree)
       
   228 //
       
   229 // Allocate without growing. aSize includes cell header and alignment.
       
   230 // Lock already held.
       
   231 //
       
   232 	{
       
   233 	SCell* pP = &iFree;
       
   234 	SCell* pC = pP->next;
       
   235 	for (; pC; pP=pC, pC=pC->next) // Scan the free list
       
   236 		{
       
   237 		__CHECK_CELL(pC);
       
   238 		SCell* pE;
       
   239 		if (pC->len >= aSize)				// Block size bigger than request
       
   240 			{
       
   241 			if (pC->len - aSize < iMinCell)	// Leftover must be large enough to hold an SCell
       
   242 			   	{
       
   243 			   	aSize = pC->len;			// It isn't, so take it all
       
   244 			   	pE = pC->next;				// Set the next field
       
   245 			   	}
       
   246 			else
       
   247 			   	{
       
   248 			   	pE = (SCell*)(((TUint8*)pC)+aSize); // Take amount required
       
   249 			   	pE->len = pC->len - aSize;	// Initialize new free cell
       
   250 			   	pE->next = pC->next;
       
   251 			   	}
       
   252 			pP->next = pE;					// Update previous pointer
       
   253 			pC->len = aSize;				// Set control size word
       
   254 #if defined(_DEBUG)														
       
   255 			((SDebugCell*)pC)->nestingLevel = iNestingLevel;
       
   256 			((SDebugCell*)pC)->allocCount = ++iAllocCount;
       
   257 #endif
       
   258 			return pC;
       
   259 			}
       
   260 		}
       
   261 	aLastFree = pP;
       
   262 	return NULL;
       
   263 	}
       
   264 #endif
       
   265 
       
   266 
       
   267 
       
   268 
       
   269 UEXPORT_C TAny* RHeap::Alloc(TInt aSize)
       
   270 /**
       
   271 Allocates a cell of the specified size from the heap.
       
   272 
       
   273 If there is insufficient memory available on the heap from which to allocate
       
   274 a cell of the required size, the function returns NULL.
       
   275 
       
   276 The cell is aligned according to the alignment value specified at construction,
       
   277 or the default alignment value, if an explict value was not specified.
       
   278 
       
   279 The resulting size of the allocated cell may be rounded up to a
       
   280 value greater than aSize, but is guaranteed to be not less than aSize.
       
   281 
       
   282 @param aSize The 
       
   283 size of the cell to be allocated from the heap
       
   284 
       
   285 @return A pointer to the allocated cell. NULL if there is insufficient memory 
       
   286         available.
       
   287         
       
   288 @panic USER 47 if the maximum unsigned value of aSize is greater than or equal
       
   289        to the value of KMaxTInt/2; for example, calling Alloc(-1) raises
       
   290        this panic.
       
   291        
       
   292 @see KMaxTInt        
       
   293 */
       
   294 	{
       
   295 
       
   296 	__CHECK_THREAD_STATE;
       
   297 	__ASSERT_ALWAYS((TUint)aSize<(KMaxTInt/2),HEAP_PANIC(ETHeapBadAllocatedCellSize));
       
   298 	__SIMULATE_ALLOC_FAIL(return NULL;)
       
   299 	
       
   300 	TInt origSize = aSize;
       
   301 	aSize = Max(Align(aSize + EAllocCellSize), iMinCell);
       
   302 	SCell* pL = NULL;
       
   303 	Lock();
       
   304 	SCell* pC = (SCell*)DoAlloc(aSize, pL);
       
   305 	if (!pC && !(iFlags & EFixedSize))
       
   306 		{
       
   307 		// try to grow chunk heap
       
   308 		TInt r = TryToGrowHeap(aSize, pL);
       
   309 		if (r==KErrNone)
       
   310 			pC = DoAlloc(aSize, pL);
       
   311 		}
       
   312 	if (pC)
       
   313 		++iCellCount, iTotalAllocSize += (pC->len - EAllocCellSize);
       
   314 	Unlock();
       
   315 	if (pC)
       
   316 		{
       
   317 		TAny* result=((TUint8*)pC) + EAllocCellSize;
       
   318 		if (iFlags & ETraceAllocs)
       
   319 			{
       
   320 			TUint32 traceData[2];
       
   321 			traceData[0] = AllocLen(result);
       
   322 			traceData[1] = origSize;
       
   323 			BTraceContextN(BTrace::EHeap, BTrace::EHeapAlloc, (TUint32)this, (TUint32)result, traceData, sizeof(traceData));
       
   324 			}
       
   325 #ifdef __KERNEL_MODE__
       
   326 		memclr(result, pC->len - EAllocCellSize);
       
   327 #endif		
       
   328 		return result;
       
   329 		}
       
   330 	if (iFlags & ETraceAllocs)						
       
   331 			BTraceContext8(BTrace::EHeap, BTrace::EHeapAllocFail, (TUint32)this, (TUint32)origSize);
       
   332 	return NULL;
       
   333 	}
       
   334 
       
   335 
       
   336 
       
   337 
       
   338 TInt RHeap::TryToGrowHeap(TInt aSize, SCell* aLastFree)
       
   339 	{
       
   340 	TBool at_end = IsLastCell(aLastFree);
       
   341 	TInt extra = at_end ? aSize - aLastFree->len : aSize;
       
   342 	extra = (extra + iGrowBy - 1) / iGrowBy;
       
   343 	extra *= iGrowBy;
       
   344 	TInt cur_len = _ALIGN_UP(iTop - ((TUint8*)this - iOffset), iPageSize);
       
   345 	TInt new_len = cur_len + extra;
       
   346 	TInt r = KErrNoMemory;
       
   347 	if (new_len <= iMaxLength)
       
   348 		{
       
   349 		r = SetBrk(new_len);
       
   350 		if (r == KErrNone)
       
   351 			{
       
   352 			if (at_end)
       
   353 				aLastFree->len += extra;
       
   354 			else
       
   355 				{
       
   356 				SCell* pC = (SCell*)iTop;
       
   357 				pC->len = extra;
       
   358 				pC->next = NULL;
       
   359 				aLastFree->next = pC;
       
   360 				}
       
   361 			iTop += extra;
       
   362 			}
       
   363 		}
       
   364 	return r;
       
   365 	}
       
   366 
       
   367 
       
   368 
       
   369 
       
   370 #ifndef __KERNEL_MODE__
       
   371 EXPORT_C TInt RHeap::Compress()
       
   372 /**
       
   373 Compresses the heap.
       
   374 
       
   375 The function frees excess committed space from the top 
       
   376 of the heap. The size of the heap is never reduced below the minimum size 
       
   377 specified during creation of the heap.
       
   378 
       
   379 @return The space reclaimed. If no space can be reclaimed, then this value 
       
   380         is zero.
       
   381 */
       
   382 	{
       
   383 
       
   384 	if (iFlags & EFixedSize)
       
   385 		return 0;
       
   386 	TInt r = 0;
       
   387 	Lock();
       
   388 	SCell* pC = &iFree;
       
   389 	for (; pC->next; pC=pC->next) {}
       
   390 	if (pC!=&iFree)
       
   391 		{
       
   392 		__CHECK_CELL(pC);
       
   393 		if (IsLastCell(pC))
       
   394 			r = Reduce(pC);
       
   395 		}
       
   396 	Unlock();
       
   397 	return r;
       
   398 	}
       
   399 #endif
       
   400 
       
   401 
       
   402 
       
   403 
       
   404 #if !defined(__HEAP_MACHINE_CODED__) || defined(_DEBUG)
       
   405 void RHeap::DoFree(SCell* pC)
       
   406 	{
       
   407 	__ZAP_CELL(pC);
       
   408 
       
   409 	SCell* pP = &iFree;
       
   410 	SCell* pE = pP->next;
       
   411 	for (; pE && pE<pC; pP=pE, pE=pE->next) {}
       
   412 	if (pE)			// Is there a following free cell?
       
   413 		{
       
   414 		SCell* pN = __NEXT_CELL(pC);
       
   415 		__ASSERT_ALWAYS(pN<=pE, HEAP_PANIC(ETHeapFreeBadNextCell)); // Following cell overlaps
       
   416 		if (pN==pE) // Is it adjacent
       
   417 			{
       
   418 			pC->len += pE->len; // Yes - coalesce adjacent free cells
       
   419 			pC->next = pE->next;
       
   420 			}
       
   421 		else					// pN<pE, non-adjacent free cells
       
   422 			pC->next = pE;		// Otherwise just point to it
       
   423 		}
       
   424 	else
       
   425 		pC->next = NULL;		// No following free cell
       
   426 	SCell* pN = __NEXT_CELL(pP);	// pN=pP=&iFree if no preceding free cell
       
   427 	__ASSERT_ALWAYS(pN<=pC, HEAP_PANIC(ETHeapFreeBadPrevCell)); // Previous cell overlaps
       
   428 	if (pN==pC) // Is it adjacent
       
   429 		{
       
   430 		pP->len += pC->len;		// Yes - coalesce adjacent free cells
       
   431 		pP->next = pC->next;
       
   432 		pC = pP;				// for size reduction check
       
   433 		}
       
   434 	else						// pN<pC, non-adjacent free cells
       
   435 		pP->next = pC;			// point previous cell to the one being freed
       
   436 	pN = __NEXT_CELL(pC);		// End of amalgamated free cell
       
   437 	if ((TUint8*)pN==iTop && !(iFlags & EFixedSize) && 
       
   438 		pC->len >= KHeapShrinkHysRatio*(iGrowBy>>8))
       
   439 		Reduce(pC);
       
   440 	}
       
   441 #endif
       
   442 
       
   443 
       
   444 
       
   445 
       
   446 UEXPORT_C void RHeap::Free(TAny* aCell)
       
   447 /**
       
   448 Frees the specified cell and returns it to the heap.
       
   449 
       
   450 @param aCell A pointer to a valid cell; this pointer can also be NULL,
       
   451              in which case the function does nothing and just returns.
       
   452 
       
   453 @panic USER 42 if aCell points to an invalid cell.
       
   454 */
       
   455 	{
       
   456 	__CHECK_THREAD_STATE;
       
   457 	if (!aCell)
       
   458 		return;
       
   459 	Lock();
       
   460 	if (iFlags & EMonitorMemory)
       
   461 		__MEMORY_MONITOR_CHECK_CELL(aCell);
       
   462 	SCell* pC = GetAddress(aCell);
       
   463 	--iCellCount;
       
   464 	iTotalAllocSize -= (pC->len - EAllocCellSize);
       
   465 	DoFree(pC);
       
   466 	if (iFlags & ETraceAllocs)
       
   467 		BTraceContext8(BTrace::EHeap, BTrace::EHeapFree, (TUint32)this, (TUint32)aCell);
       
   468 	Unlock();
       
   469 	}
       
   470 
       
   471 
       
   472 
       
   473 
       
   474 TInt RHeap::Reduce(SCell* aCell)
       
   475 	{
       
   476 	TInt reduce=0;
       
   477 	TInt offset=((TUint8*)aCell)-((TUint8*)this - iOffset);
       
   478 	if (offset>=iMinLength)
       
   479 		reduce = aCell->len;						// length of entire free cell
       
   480 	else
       
   481 		reduce = offset + aCell->len - iMinLength;	// length of free cell past minimum heap size
       
   482 	reduce = _ALIGN_DOWN(reduce, iPageSize);		// round down to page multiple
       
   483 	if (reduce<=0)
       
   484 		return 0;									// can't reduce this heap
       
   485 	TInt new_cell_len = aCell->len - reduce;		// length of last free cell after reduction
       
   486 	if (new_cell_len == 0)
       
   487 		{
       
   488 		// the free cell can be entirely eliminated
       
   489 		SCell* pP = &iFree;
       
   490 		for (; pP->next!=aCell; pP=pP->next) {}
       
   491 		pP->next = NULL;
       
   492 		}
       
   493 	else
       
   494 		{
       
   495 		if (new_cell_len < iMinCell)
       
   496 			{
       
   497 			// max reduction would leave a cell too small
       
   498 			reduce -= iPageSize;
       
   499 			new_cell_len += iPageSize;
       
   500 			}
       
   501 		aCell->len = new_cell_len;	// reduce the cell length
       
   502 		}
       
   503 	iTop -= reduce;
       
   504 	TInt new_len = _ALIGN_UP(iTop - ((TUint8*)this - iOffset), iPageSize);
       
   505 	TInt r = SetBrk(new_len);
       
   506 	__ASSERT_ALWAYS(r==KErrNone, HEAP_PANIC(ETHeapReduceFailed));
       
   507 	return reduce;
       
   508 	}
       
   509 
       
   510 
       
   511 
       
   512 
       
   513 #ifndef __KERNEL_MODE__
       
   514 EXPORT_C void RHeap::Reset()
       
   515 /**
       
   516 Frees all allocated cells on this heap.
       
   517 */
       
   518 	{
       
   519 
       
   520 	Lock();
       
   521 	if (!(iFlags & EFixedSize))
       
   522 		{
       
   523 		TInt r = SetBrk(iMinLength);
       
   524 		__ASSERT_ALWAYS(r==KErrNone, HEAP_PANIC(ETHeapResetFailed));
       
   525 		}
       
   526 	Initialise();
       
   527 	Unlock();
       
   528 	}
       
   529 #endif
       
   530 
       
   531 
       
   532 
       
   533 
       
   534 inline void RHeap::FindFollowingFreeCell(SCell* aCell, SCell*& aPrev, SCell*& aNext)
       
   535 //
       
   536 // Find the free cell that immediately follows aCell, if one exists
       
   537 // If found, aNext is set to point to it, else it is set to NULL.
       
   538 // aPrev is set to the free cell before aCell or the dummy free cell where there are no free cells before aCell.
       
   539 // Called with lock enabled.
       
   540 //
       
   541 	{
       
   542 	aPrev = &iFree;
       
   543 	aNext = aPrev->next;
       
   544 	for (; aNext && aNext<aCell; aPrev=aNext, aNext=aNext->next) {}	
       
   545 	
       
   546 	if (aNext) // If there is a following free cell, check its directly after aCell.
       
   547 		{
       
   548 			SCell* pNextCell = __NEXT_CELL(aCell);			// end of this cell
       
   549 			__ASSERT_ALWAYS(pNextCell<=aNext, (Unlock(), HEAP_PANIC(ETHeapReAllocBadNextCell)));	// Following free cell overlaps
       
   550 			if (pNextCell!=aNext) 
       
   551 				aNext=NULL;		
       
   552 		}
       
   553 	}
       
   554 
       
   555 
       
   556 
       
   557 
       
   558 TInt RHeap::TryToGrowCell(SCell* aCell,SCell* aPrev, SCell* aNext, TInt aSize)
       
   559 //
       
   560 // Try to grow the heap cell 'aCell' in place, to size 'aSize'.
       
   561 // Requires the free cell immediately after aCell (aNext), and the free cell prior to
       
   562 // that (aPrev), to be provided.  (As found by FindFollowingFreeCell)
       
   563 //
       
   564 
       
   565 	{
       
   566 	TInt extra = aSize - aCell->len;
       
   567 	if (aNext && (aNext->len>=extra)) // Is there a following free cell big enough?
       
   568 		{
       
   569 		if (aNext->len - extra >= iMinCell)	// take part of free cell ?
       
   570 			{
       
   571 			SCell* pX = (SCell*)((TUint8*)aNext + extra);	// remainder of free cell
       
   572 			pX->next = aNext->next;			// remainder->next = original free cell->next
       
   573 			pX->len = aNext->len - extra;		// remainder length = original free cell length - extra
       
   574 			aPrev->next = pX;					// put remainder into free chain
       
   575 			}
       
   576 		else
       
   577 			{
       
   578 			extra = aNext->len;					// Take whole free cell
       
   579 			aPrev->next = aNext->next;			// remove from free chain
       
   580 			}
       
   581 #ifdef __KERNEL_MODE__
       
   582 		memclr(((TUint8*)aCell) + aCell->len, extra);
       
   583 #endif		
       
   584 		aCell->len += extra;					// update reallocated cell length
       
   585 		iTotalAllocSize += extra;
       
   586 		return KErrNone;
       
   587 		}
       
   588 	return KErrGeneral;  // No space to grow cell
       
   589 	}
       
   590 
       
   591 
       
   592 
       
   593 
       
   594 // UEXPORT_C TAny* RHeap::ReAlloc(TAny* aCell, TInt aSize, TInt aMode)
       
   595 /**
       
   596 Increases or decreases the size of an existing cell in the heap.
       
   597 
       
   598 If the cell is being decreased in size, then it is guaranteed not to move,
       
   599 and the function returns the pointer originally passed in aCell. Note that the
       
   600 length of the cell will be the same if the difference between the old size
       
   601 and the new size is smaller than the minimum cell size.
       
   602 
       
   603 If the cell is being increased in size, i.e. aSize is bigger than its
       
   604 current size, then the function tries to grow the cell in place.
       
   605 If successful, then the function returns the pointer originally
       
   606 passed in aCell. If unsuccessful, then:
       
   607 
       
   608 1. if the cell cannot be moved, i.e. aMode has the ENeverMove bit set, then
       
   609    the function returns NULL.
       
   610 2. if the cell can be moved, i.e. aMode does not have the ENeverMove bit set,
       
   611    then the function tries to allocate a new replacement cell, and, if
       
   612    successful, returns a pointer to the new cell; if unsuccessful, it
       
   613    returns NULL.
       
   614 
       
   615 Note that in debug mode, the function returns NULL if the cell cannot be grown
       
   616 in place, regardless of whether the ENeverMove bit is set.
       
   617 
       
   618 If the reallocated cell is at a different location from the original cell, then
       
   619 the content of the original cell is copied to the reallocated cell.
       
   620 
       
   621 If the supplied pointer, aCell is NULL, then the function attempts to allocate
       
   622 a new cell, but only if the cell can be moved, i.e. aMode does not have
       
   623 the ENeverMove bit set.
       
   624 
       
   625 Note the following general points:
       
   626 
       
   627 1. If reallocation fails, the content of the original cell is preserved.
       
   628 
       
   629 2. The resulting size of the re-allocated cell may be rounded up to a value
       
   630    greater than aSize, but is guaranteed to be not less than aSize.
       
   631  
       
   632 @param aCell A pointer to the cell to be reallocated. This may be NULL.
       
   633 
       
   634 @param aSize The new size of the cell. This may be bigger or smaller than the
       
   635              size of the original cell.
       
   636              
       
   637 @param aMode Flags controlling the reallocation. The only bit which has any
       
   638              effect on this function is that defined by the enumeration
       
   639              ENeverMove of the enum RAllocator::TReAllocMode.
       
   640              If this is set, then any successful reallocation guarantees not
       
   641              to have changed the start address of the cell.
       
   642              By default, this parameter is zero.
       
   643 
       
   644 @return A pointer to the reallocated cell. This may be the same as the original
       
   645         pointer supplied through aCell. NULL if there is insufficient memory to
       
   646         reallocate the cell, or to grow it in place.
       
   647 
       
   648 @panic USER 42, if aCell is not NULL, and does not point to a valid cell.
       
   649 @panic USER 47, if the maximum unsigned value of aSize is greater
       
   650                 than or equal to KMaxTInt/2. For example,
       
   651                 calling ReAlloc(someptr,-1) raises this panic.
       
   652 
       
   653 @see RAllocator::TReAllocMode
       
   654 */
       
   655 UEXPORT_C TAny* RHeap::ReAlloc(TAny* aCell, TInt aSize, TInt aMode)
       
   656 	{
       
   657 	if (aCell && iFlags&EMonitorMemory)
       
   658 		__MEMORY_MONITOR_CHECK_CELL(aCell);
       
   659 	TAny* retval = ReAllocImpl(aCell, aSize, aMode);
       
   660 	if (iFlags & ETraceAllocs)
       
   661 		{
       
   662 		if (retval)
       
   663 			{
       
   664 			TUint32 traceData[3];
       
   665 			traceData[0] = AllocLen(retval);
       
   666 			traceData[1] = aSize;
       
   667 			traceData[2] = (TUint32)aCell;
       
   668 			BTraceContextN(BTrace::EHeap, BTrace::EHeapReAlloc,(TUint32)this, (TUint32)retval,traceData, sizeof(traceData));
       
   669 			}
       
   670 		else
       
   671 			BTraceContext12(BTrace::EHeap, BTrace::EHeapReAllocFail, (TUint32)this, (TUint32)aCell, (TUint32)aSize);
       
   672 		}
       
   673 	return retval;
       
   674 	}
       
   675 inline TAny* RHeap::ReAllocImpl(TAny* aCell, TInt aSize, TInt aMode)
       
   676 	{
       
   677 	__CHECK_THREAD_STATE;
       
   678 	if (!aCell)
       
   679 		return (aMode & ENeverMove) ? NULL : Alloc(aSize);
       
   680 	__ASSERT_ALWAYS((TUint)aSize<(KMaxTInt/2),HEAP_PANIC(ETHeapBadAllocatedCellSize));
       
   681 	Lock();
       
   682 	SCell* pC = GetAddress(aCell);
       
   683 	TInt old_len = pC->len;
       
   684 	__DEBUG_SAVE(pC);
       
   685 	aSize = Max(Align(aSize + EAllocCellSize), iMinCell);
       
   686 	if (aSize > old_len)	// Trying to grow cell
       
   687 		{
       
   688 		__SIMULATE_ALLOC_FAIL({	Unlock(); return NULL;})			
       
   689 		
       
   690 		// Try to grow cell in place, without reallocation
       
   691 		SCell* pPrev;
       
   692 		SCell* pNext;
       
   693 		FindFollowingFreeCell(pC,pPrev, pNext);
       
   694 		TInt r = TryToGrowCell(pC, pPrev, pNext, aSize);
       
   695 		
       
   696 		if (r==KErrNone) 
       
   697 			{
       
   698 			Unlock();
       
   699 			return aCell;
       
   700 			}
       
   701 
       
   702 		if (!(aMode & ENeverMove))
       
   703 		// If moving allowed, try re-alloc. 
       
   704 		// If we need to extend heap,and cell is at the end, try and grow in place
       
   705 			{
       
   706 			SCell* pLastFree;
       
   707 			SCell* pNewCell = (SCell*)DoAlloc(aSize, pLastFree);
       
   708 			if (!pNewCell && !(iFlags & EFixedSize))
       
   709 			// if we need to extend the heap to alloc
       
   710 				{
       
   711 				if (IsLastCell(pC) || (pNext && IsLastCell(pNext)))
       
   712 				// if last used Cell, try and extend heap and then cell 
       
   713 					{
       
   714 					TInt r = TryToGrowHeap(aSize - old_len, pLastFree);
       
   715 					if (r==KErrNone)
       
   716 						{
       
   717 						r = TryToGrowCell(pC, pPrev, pPrev->next, aSize);
       
   718 						Unlock();
       
   719 						__ASSERT_DEBUG(r == KErrNone, HEAP_PANIC(ETHeapCellDidntGrow));						
       
   720 						return aCell;
       
   721 						}
       
   722 					}
       
   723 				else
       
   724 				// try to grow chunk heap and Alloc on it
       
   725 					{
       
   726 					TInt r = TryToGrowHeap(aSize, pLastFree);
       
   727 					if (r==KErrNone)
       
   728 						pNewCell = DoAlloc(aSize, pLastFree);
       
   729 					}
       
   730 				}
       
   731 
       
   732 			if (pNewCell)
       
   733 			// if we created a new cell, adjust tellies, copy the contents and delete old cell.
       
   734 				{
       
   735 				iCellCount++;
       
   736 				iTotalAllocSize += (pNewCell->len - EAllocCellSize);
       
   737 
       
   738 				Unlock();
       
   739 				TUint8* raw = ((TUint8*) pNewCell);
       
   740 				
       
   741 				memcpy(raw + EAllocCellSize, aCell, old_len - EAllocCellSize);
       
   742 #ifdef __KERNEL_MODE__
       
   743 				memclr(raw + old_len, pNewCell->len - old_len);
       
   744 #endif		
       
   745 				Free(aCell);
       
   746 				__DEBUG_RESTORE(raw + EAllocCellSize);
       
   747 				return raw + EAllocCellSize;
       
   748 				}
       
   749 			}
       
   750 		else 
       
   751 		// No moving, but still posible to extend the heap (if heap extendable)
       
   752 			{
       
   753 			if (!(iFlags & EFixedSize) && (IsLastCell(pC) || (pNext && IsLastCell(pNext))))
       
   754 				{
       
   755 				SCell* pLastFree = pNext ? pNext : pPrev;
       
   756 				TInt r = TryToGrowHeap(aSize - old_len, pLastFree);
       
   757 				if (r==KErrNone)
       
   758 					{
       
   759 					r = TryToGrowCell(pC, pPrev, pPrev->next, aSize);
       
   760 					Unlock();
       
   761 					__ASSERT_DEBUG(r==KErrNone, HEAP_PANIC(ETHeapCellDidntGrow));					
       
   762 					return aCell;
       
   763 					}
       
   764 				}
       
   765 			}			
       
   766 		Unlock();
       
   767 		return NULL;
       
   768 		}
       
   769 	if (old_len - aSize >= iMinCell)
       
   770 		{
       
   771 		// cell shrinking, remainder big enough to form a new free cell
       
   772 		SCell* pX = (SCell*)((TUint8*)pC + aSize);	// pointer to new free cell
       
   773 		pC->len = aSize;			// update cell size
       
   774 		pX->len = old_len - aSize;	// size of remainder
       
   775 		iTotalAllocSize -= pX->len;
       
   776 		DoFree(pX);					// link new free cell into chain, shrink heap if necessary
       
   777 		}
       
   778 	Unlock();
       
   779 	return aCell;
       
   780 	}
       
   781 
       
   782 
       
   783 
       
   784 
       
   785 #ifndef __KERNEL_MODE__
       
   786 
       
   787 EXPORT_C TInt RHeap::Available(TInt& aBiggestBlock) const
       
   788 /**
       
   789 Gets the total free space currently available on the heap and the space 
       
   790 available in the largest free block.
       
   791 
       
   792 The space available represents the total space which can be allocated.
       
   793 
       
   794 Note that compressing the heap may reduce the total free space available and 
       
   795 the space available in the largest free block.
       
   796 
       
   797 @param aBiggestBlock On return, contains the space available 
       
   798                      in the largest free block on the heap.
       
   799                      
       
   800 @return The total free space currently available on the heap.
       
   801 */
       
   802 	{
       
   803 
       
   804 	TInt total = 0;
       
   805 	TInt max = 0;
       
   806 	Lock();
       
   807 	SCell* pC = iFree.next;
       
   808 	for (; pC; pC=pC->next)
       
   809 		{
       
   810 		TInt l = pC->len - EAllocCellSize;
       
   811 		if (l > max)
       
   812 			max = l;
       
   813 		total += l;
       
   814 		}
       
   815 	Unlock();
       
   816 	aBiggestBlock = max;
       
   817 	return total;
       
   818 	}
       
   819 
       
   820 
       
   821 
       
   822 
       
   823 EXPORT_C TInt RHeap::AllocSize(TInt& aTotalAllocSize) const
       
   824 /**
       
   825 Gets the number of cells allocated on this heap, and the total space 
       
   826 allocated to them.
       
   827 
       
   828 @param aTotalAllocSize On return, contains the total space allocated
       
   829                        to the cells.
       
   830 
       
   831 @return The number of cells allocated on this heap.
       
   832 */
       
   833 	{
       
   834 	Lock();
       
   835 	TInt c = iCellCount;
       
   836 	aTotalAllocSize = iTotalAllocSize;
       
   837 	Unlock();
       
   838 	return c;
       
   839 	}
       
   840 
       
   841 
       
   842 
       
   843 
       
   844 EXPORT_C RHeap* UserHeap::FixedHeap(TAny* aBase, TInt aMaxLength, TInt aAlign, TBool aSingleThread)
       
   845 /**
       
   846 Creates a fixed length heap at a specified location.
       
   847 
       
   848 On successful return from this function, aMaxLength bytes are committed by the chunk.
       
   849 The heap cannot be extended.
       
   850 
       
   851 @param aBase         A pointer to the location where the heap is to be constructed.
       
   852 @param aMaxLength    The length of the heap. If the supplied value is less
       
   853                      than KMinHeapSize, it is discarded and the value KMinHeapSize
       
   854                      is used instead.
       
   855 @param aAlign        The alignment of heap cells.
       
   856 @param aSingleThread Indicates whether single threaded or not.
       
   857 
       
   858 @return A pointer to the new heap, or NULL if the heap could not be created.
       
   859 
       
   860 @panic USER 56 if aMaxLength is negative.
       
   861 @panic USER 172 if aAlign is not a power of 2 or is less than the size of a TAny*.
       
   862 */
       
   863 //
       
   864 // Force construction of the fixed memory.
       
   865 //
       
   866 	{
       
   867 
       
   868 	__ASSERT_ALWAYS(aMaxLength>=0, ::Panic(ETHeapMaxLengthNegative));
       
   869 	if (aMaxLength<KMinHeapSize)
       
   870 		aMaxLength=KMinHeapSize;
       
   871 	RHeap* h = new(aBase) RHeap(aMaxLength, aAlign, aSingleThread);
       
   872 	if (!aSingleThread)
       
   873 		{
       
   874 		TInt r = h->iLock.CreateLocal();
       
   875 		if (r!=KErrNone)
       
   876 			return NULL;
       
   877 		h->iHandles = (TInt*)&h->iLock;
       
   878 		h->iHandleCount = 1;
       
   879 		}
       
   880 	return h;
       
   881 	}
       
   882 
       
   883 
       
   884 /**
       
   885 Constructor where minimum and maximum length of the heap can be defined.
       
   886 It defaults the chunk heap to be created to have use a new local chunk, 
       
   887 to have a grow by value of KMinHeapGrowBy, to be unaligned, not to be 
       
   888 single threaded and not to have any mode flags set.
       
   889 
       
   890 @param aMinLength    The minimum length of the heap to be created.
       
   891 @param aMaxLength    The maximum length to which the heap to be created can grow.
       
   892                      If the supplied value is less than KMinHeapSize, then it
       
   893                      is discarded and the value KMinHeapSize used instead.
       
   894 */
       
   895 EXPORT_C TChunkHeapCreateInfo::TChunkHeapCreateInfo(TInt aMinLength, TInt aMaxLength) :
       
   896 	iVersionNumber(EVersion0), iMinLength(aMinLength), iMaxLength(aMaxLength),
       
   897 	iAlign(0), iGrowBy(1), iSingleThread(EFalse), 
       
   898 	iOffset(0), iPaging(EUnspecified), iMode(0), iName(NULL)
       
   899 	{
       
   900 	}
       
   901 
       
   902 
       
   903 /**
       
   904 Sets the chunk heap to create a new chunk with the specified name.
       
   905 
       
   906 This overriddes any previous call to TChunkHeapCreateInfo::SetNewChunkHeap() or
       
   907 TChunkHeapCreateInfo::SetExistingChunkHeap() for this TChunkHeapCreateInfo object.
       
   908 
       
   909 @param aName	The name to be given to the chunk heap to be created
       
   910 				If NULL, the function constructs a local chunk to host the heap.
       
   911 				If not NULL, a pointer to a descriptor containing the name to be 
       
   912 				assigned to the global chunk hosting the heap.
       
   913 */
       
   914 EXPORT_C void TChunkHeapCreateInfo::SetCreateChunk(const TDesC* aName)
       
   915 	{
       
   916 	iName = (TDesC*)aName;
       
   917 	iChunk.SetHandle(KNullHandle);
       
   918 	}
       
   919 
       
   920 
       
   921 /**
       
   922 Sets the chunk heap to be created to use the chunk specified.
       
   923 
       
   924 This overriddes any previous call to TChunkHeapCreateInfo::SetNewChunkHeap() or
       
   925 TChunkHeapCreateInfo::SetExistingChunkHeap() for this TChunkHeapCreateInfo object.
       
   926 
       
   927 @param aChunk	A handle to the chunk to use for the heap.
       
   928 */
       
   929 EXPORT_C void TChunkHeapCreateInfo::SetUseChunk(const RChunk aChunk)
       
   930 	{
       
   931 	iName = NULL;
       
   932 	iChunk = aChunk;
       
   933 	}
       
   934 
       
   935 
       
   936 /**
       
   937 Creates a chunk heap of the type specified by the parameter aCreateInfo.
       
   938 
       
   939 @param aCreateInfo	A reference to a TChunkHeapCreateInfo object specifying the
       
   940 					type of chunk heap to create.
       
   941 
       
   942 @return A pointer to the new heap or NULL if the heap could not be created.
       
   943 
       
   944 @panic USER 41 if the heap's specified minimum length is greater than the specified maximum length.
       
   945 @panic USER 55 if the heap's specified minimum length is negative.
       
   946 @panic USER 172 if the heap's specified alignment is not a power of 2 or is less than the size of a TAny*.
       
   947 */
       
   948 EXPORT_C RHeap* UserHeap::ChunkHeap(const TChunkHeapCreateInfo& aCreateInfo)
       
   949 	{
       
   950 	// aCreateInfo must have been configured to use a new chunk or an exiting chunk.
       
   951 	__ASSERT_ALWAYS(!(aCreateInfo.iMode & (TUint32)~EChunkHeapMask), ::Panic(EHeapCreateInvalidMode));
       
   952 	RHeap* h = NULL;
       
   953 
       
   954 	if (aCreateInfo.iChunk.Handle() == KNullHandle)
       
   955 		{// A new chunk is to be created for this heap.
       
   956 		__ASSERT_ALWAYS(aCreateInfo.iMinLength >= 0, ::Panic(ETHeapMinLengthNegative));
       
   957 		__ASSERT_ALWAYS(aCreateInfo.iMaxLength >= aCreateInfo.iMinLength, ::Panic(ETHeapCreateMaxLessThanMin));
       
   958 
       
   959 		TInt maxLength = aCreateInfo.iMaxLength;
       
   960 		if (maxLength < KMinHeapSize)
       
   961 			maxLength = KMinHeapSize;
       
   962 
       
   963 		TChunkCreateInfo chunkInfo;
       
   964 		chunkInfo.SetNormal(0, maxLength);
       
   965 		chunkInfo.SetOwner((aCreateInfo.iSingleThread)? EOwnerThread : EOwnerProcess);
       
   966 		if (aCreateInfo.iName)
       
   967 			chunkInfo.SetGlobal(*aCreateInfo.iName);
       
   968 		// Set the paging attributes of the chunk.
       
   969 		if (aCreateInfo.iPaging == TChunkHeapCreateInfo::EPaged)
       
   970 			chunkInfo.SetPaging(TChunkCreateInfo::EPaged);
       
   971 		if (aCreateInfo.iPaging == TChunkHeapCreateInfo::EUnpaged)
       
   972 			chunkInfo.SetPaging(TChunkCreateInfo::EUnpaged);
       
   973 		// Create the chunk.
       
   974 		RChunk chunk;
       
   975 		if (chunk.Create(chunkInfo) != KErrNone)
       
   976 			return NULL;
       
   977 		// Create the heap using the new chunk.
       
   978 		TUint mode = aCreateInfo.iMode | EChunkHeapDuplicate;	// Must duplicate the handle.
       
   979 		h = OffsetChunkHeap(chunk, aCreateInfo.iMinLength, aCreateInfo.iOffset,
       
   980 							aCreateInfo.iGrowBy, maxLength, aCreateInfo.iAlign,
       
   981 							aCreateInfo.iSingleThread, mode);
       
   982 		chunk.Close();
       
   983 		}
       
   984 	else
       
   985 		{
       
   986 		h = OffsetChunkHeap(aCreateInfo.iChunk, aCreateInfo.iMinLength, aCreateInfo.iOffset,
       
   987 							aCreateInfo.iGrowBy, aCreateInfo.iMaxLength, aCreateInfo.iAlign,
       
   988 							aCreateInfo.iSingleThread, aCreateInfo.iMode);
       
   989 		}
       
   990 	return h;
       
   991 	}
       
   992 
       
   993 
       
   994 EXPORT_C RHeap* UserHeap::ChunkHeap(const TDesC* aName, TInt aMinLength, TInt aMaxLength, TInt aGrowBy, TInt aAlign, TBool aSingleThread)
       
   995 /**
       
   996 Creates a heap in a local or global chunk.
       
   997 
       
   998 The chunk hosting the heap can be local or global.
       
   999 
       
  1000 A local chunk is one which is private to the process creating it and is not
       
  1001 intended for access by other user processes.
       
  1002 A global chunk is one which is visible to all processes.
       
  1003 
       
  1004 The hosting chunk is local, if the pointer aName is NULL, otherwise
       
  1005 the hosting chunk is global and the descriptor *aName is assumed to contain
       
  1006 the name to be assigned to it.
       
  1007 
       
  1008 Ownership of the host chunk is vested in the current process.
       
  1009 
       
  1010 A minimum and a maximum size for the heap can be specified. On successful
       
  1011 return from this function, the size of the heap is at least aMinLength.
       
  1012 If subsequent requests for allocation of memory from the heap cannot be
       
  1013 satisfied by compressing the heap, the size of the heap is extended in
       
  1014 increments of aGrowBy until the request can be satisfied. Attempts to extend
       
  1015 the heap causes the size of the host chunk to be adjusted.
       
  1016 
       
  1017 Note that the size of the heap cannot be adjusted by more than aMaxLength.
       
  1018 
       
  1019 @param aName         If NULL, the function constructs a local chunk to host
       
  1020                      the heap.
       
  1021                      If not NULL, a pointer to a descriptor containing the name
       
  1022                      to be assigned to the global chunk hosting the heap.
       
  1023 @param aMinLength    The minimum length of the heap.
       
  1024 @param aMaxLength    The maximum length to which the heap can grow.
       
  1025                      If the supplied value is less than KMinHeapSize, then it
       
  1026                      is discarded and the value KMinHeapSize used instead.
       
  1027 @param aGrowBy       The increments to the size of the host chunk. If a value is
       
  1028                      not explicitly specified, the value KMinHeapGrowBy is taken
       
  1029                      by default
       
  1030 @param aAlign        The alignment of heap cells.
       
  1031 @param aSingleThread Indicates whether single threaded or not.
       
  1032 
       
  1033 @return A pointer to the new heap or NULL if the heap could not be created.
       
  1034 
       
  1035 @panic USER 41 if aMinLength is greater than the supplied value of aMaxLength.
       
  1036 @panic USER 55 if aMinLength is negative.
       
  1037 @panic USER 172 if aAlign is not a power of 2 or is less than the size of a TAny*.
       
  1038 */
       
  1039 //
       
  1040 // Allocate a Chunk of the requested size and force construction.
       
  1041 //
       
  1042 	{
       
  1043 	TChunkHeapCreateInfo createInfo(aMinLength, aMaxLength);
       
  1044 	createInfo.SetCreateChunk(aName);
       
  1045 	createInfo.SetGrowBy(aGrowBy);
       
  1046 	createInfo.SetAlignment(aAlign);
       
  1047 	createInfo.SetSingleThread(aSingleThread);
       
  1048 	return ChunkHeap(createInfo);
       
  1049 	}
       
  1050 
       
  1051 
       
  1052 
       
  1053 
       
  1054 EXPORT_C RHeap* UserHeap::ChunkHeap(RChunk aChunk, TInt aMinLength, TInt aGrowBy, TInt aMaxLength, TInt aAlign, TBool aSingleThread, TUint32 aMode)
       
  1055 /**
       
  1056 Creates a heap in an existing chunk.
       
  1057 
       
  1058 This function is intended to be used to create a heap in a user writable code
       
  1059 chunk as created by a call to RChunk::CreateLocalCode().
       
  1060 This type of heap can be used to hold code fragments from a JIT compiler.
       
  1061 
       
  1062 The maximum length to which the heap can grow is the same as
       
  1063 the maximum size of the chunk.
       
  1064 
       
  1065 @param aChunk        The chunk that will host the heap.
       
  1066 @param aMinLength    The minimum length of the heap.
       
  1067 @param aGrowBy       The increments to the size of the host chunk. 
       
  1068 @param aMaxLength    The maximum length to which the heap can grow.
       
  1069 @param aAlign        The alignment of heap cells.
       
  1070 @param aSingleThread Indicates whether single threaded or not.
       
  1071 @param aMode         Flags controlling the heap creation.  This should be set 
       
  1072 					 from one or more of the values in TChunkHeapCreateMode.
       
  1073                      
       
  1074 @return A pointer to the new heap or NULL if the heap could not be created.
       
  1075 
       
  1076 @panic USER 172 if aAlign is not a power of 2 or is less than the size of a TAny*.
       
  1077 */
       
  1078 //
       
  1079 // Construct a heap in an already existing chunk
       
  1080 //
       
  1081 	{
       
  1082 	
       
  1083 	return OffsetChunkHeap(aChunk, aMinLength, 0, aGrowBy, aMaxLength, aAlign, aSingleThread, aMode);
       
  1084 	}
       
  1085 
       
  1086 
       
  1087 
       
  1088 
       
  1089 EXPORT_C RHeap* UserHeap::OffsetChunkHeap(RChunk aChunk, TInt aMinLength, TInt aOffset, TInt aGrowBy, TInt aMaxLength, TInt aAlign, TBool aSingleThread, TUint32 aMode)
       
  1090 /**
       
  1091 Creates a heap in an existing chunk, offset from the beginning of the chunk.
       
  1092 
       
  1093 This function is intended to be used to create a heap where a fixed amount of
       
  1094 additional data must be stored at a known location. The additional data can be
       
  1095 placed at the base address of the chunk, allowing it to be located without
       
  1096 depending on the internals of the heap structure.
       
  1097 
       
  1098 The maximum length to which the heap can grow is the maximum size of the chunk,
       
  1099 minus the offset.
       
  1100 
       
  1101 @param aChunk        The chunk that will host the heap.
       
  1102 @param aMinLength    The minimum length of the heap.
       
  1103 @param aOffset       The offset from the start of the chunk, to the start of the heap.
       
  1104 @param aGrowBy       The increments to the size of the host chunk. 
       
  1105 @param aMaxLength    The maximum length to which the heap can grow.
       
  1106 @param aAlign        The alignment of heap cells.
       
  1107 @param aSingleThread Indicates whether single threaded or not.
       
  1108 @param aMode         Flags controlling the heap creation.  This should be set 
       
  1109 					 from one or more of the values in TChunkHeapCreateMode.
       
  1110                      
       
  1111 @return A pointer to the new heap or NULL if the heap could not be created.
       
  1112 
       
  1113 @panic USER 172 if aAlign is not a power of 2 or is less than the size of a TAny*.
       
  1114 */
       
  1115 //
       
  1116 // Construct a heap in an already existing chunk
       
  1117 //
       
  1118 	{
       
  1119 
       
  1120 	TInt page_size;
       
  1121 	UserHal::PageSizeInBytes(page_size);
       
  1122 	if (!aAlign)
       
  1123 		aAlign = RHeap::ECellAlignment;
       
  1124 	TInt maxLength = aChunk.MaxSize();
       
  1125 	TInt round_up = Max(aAlign, page_size);
       
  1126 	TInt min_cell = _ALIGN_UP(Max((TInt)RHeap::EAllocCellSize, (TInt)RHeap::EFreeCellSize), aAlign);
       
  1127 	aOffset = _ALIGN_UP(aOffset, 8);
       
  1128 	if (aMaxLength && aMaxLength+aOffset<maxLength)
       
  1129 		maxLength = _ALIGN_UP(aMaxLength+aOffset, round_up);
       
  1130 	__ASSERT_ALWAYS(aMinLength>=0, ::Panic(ETHeapMinLengthNegative));
       
  1131 	__ASSERT_ALWAYS(maxLength>=aMinLength, ::Panic(ETHeapCreateMaxLessThanMin));
       
  1132 	aMinLength = _ALIGN_UP(Max(aMinLength, (TInt)sizeof(RHeap) + min_cell) + aOffset, round_up);
       
  1133 	TInt r=aChunk.Adjust(aMinLength);
       
  1134 	if (r!=KErrNone)
       
  1135 		return NULL;
       
  1136 
       
  1137 	RHeap* h = new (aChunk.Base() + aOffset) RHeap(aChunk.Handle(), aOffset, aMinLength, maxLength, aGrowBy, aAlign, aSingleThread);
       
  1138 
       
  1139 	TBool duplicateLock = EFalse;
       
  1140 	if (!aSingleThread)
       
  1141 		{
       
  1142 		duplicateLock = aMode & EChunkHeapSwitchTo;
       
  1143 		if(h->iLock.CreateLocal(duplicateLock ? EOwnerThread : EOwnerProcess)!=KErrNone)
       
  1144 			{
       
  1145 			h->iChunkHandle = 0;
       
  1146 			return NULL;
       
  1147 			}
       
  1148 		}
       
  1149 
       
  1150 	if (aMode & EChunkHeapSwitchTo)
       
  1151 		User::SwitchHeap(h);
       
  1152 
       
  1153 	h->iHandles = &h->iChunkHandle;
       
  1154 	if (!aSingleThread)
       
  1155 		{
       
  1156 		// now change the thread-relative chunk/semaphore handles into process-relative handles
       
  1157 		h->iHandleCount = 2;
       
  1158 		if(duplicateLock)
       
  1159 			{
       
  1160 			RHandleBase s = h->iLock;
       
  1161 			r = h->iLock.Duplicate(RThread());
       
  1162 			s.Close();
       
  1163 			}
       
  1164 		if (r==KErrNone && (aMode & EChunkHeapDuplicate))
       
  1165 			{
       
  1166 			r = ((RChunk*)&h->iChunkHandle)->Duplicate(RThread());
       
  1167 			if (r!=KErrNone)
       
  1168 				h->iLock.Close(), h->iChunkHandle=0;
       
  1169 			}
       
  1170 		}
       
  1171 	else
       
  1172 		{
       
  1173 		h->iHandleCount = 1;
       
  1174 		if (aMode & EChunkHeapDuplicate)
       
  1175 			r = ((RChunk*)&h->iChunkHandle)->Duplicate(RThread(), EOwnerThread);
       
  1176 		}
       
  1177 
       
  1178 	// return the heap address
       
  1179 	return (r==KErrNone) ? h : NULL;
       
  1180 	}
       
  1181 
       
  1182 
       
  1183 
       
  1184 #define UserTestDebugMaskBit(bit) (TBool)(UserSvr::DebugMask(bit>>5) & (1<<(bit&31)))
       
  1185 
       
  1186 _LIT(KLitDollarHeap,"$HEAP");
       
  1187 EXPORT_C TInt UserHeap::CreateThreadHeap(SStdEpocThreadCreateInfo& aInfo, RHeap*& aHeap, TInt aAlign, TBool aSingleThread)
       
  1188 /**
       
  1189 @internalComponent
       
  1190 */
       
  1191 //
       
  1192 // Create a user-side heap
       
  1193 //
       
  1194 	{
       
  1195 	TInt page_size;
       
  1196 	UserHal::PageSizeInBytes(page_size);
       
  1197 	TInt minLength = _ALIGN_UP(aInfo.iHeapInitialSize, page_size);
       
  1198 	TInt maxLength = Max(aInfo.iHeapMaxSize, minLength);
       
  1199 	if (UserTestDebugMaskBit(96)) // 96 == KUSERHEAPTRACE in nk_trace.h
       
  1200 		aInfo.iFlags |= ETraceHeapAllocs;
       
  1201 
       
  1202 	// Create the thread's heap chunk.
       
  1203 	RChunk c;
       
  1204 	TChunkCreateInfo createInfo;
       
  1205 	createInfo.SetThreadHeap(0, maxLength, KLitDollarHeap());	// Initialise with no memory committed.
       
  1206 
       
  1207 	// Set the paging policy of the heap chunk based on the thread's paging policy.
       
  1208 	TUint pagingflags = aInfo.iFlags & EThreadCreateFlagPagingMask;
       
  1209 	switch (pagingflags)
       
  1210 		{
       
  1211 		case EThreadCreateFlagPaged:
       
  1212 			createInfo.SetPaging(TChunkCreateInfo::EPaged);
       
  1213 			break;
       
  1214 		case EThreadCreateFlagUnpaged:
       
  1215 			createInfo.SetPaging(TChunkCreateInfo::EUnpaged);
       
  1216 			break;
       
  1217 		case EThreadCreateFlagPagingUnspec:
       
  1218 			// Leave the chunk paging policy unspecified so the process's 
       
  1219 			// paging policy is used.
       
  1220 			break;
       
  1221 		}
       
  1222 
       
  1223 	TInt r = c.Create(createInfo);
       
  1224 	if (r!=KErrNone)
       
  1225 		return r;
       
  1226 
       
  1227 	aHeap = ChunkHeap(c, minLength, page_size, maxLength, aAlign, aSingleThread, EChunkHeapSwitchTo|EChunkHeapDuplicate);
       
  1228 	c.Close();
       
  1229 	if (!aHeap)
       
  1230 		return KErrNoMemory;
       
  1231 	if (aInfo.iFlags & ETraceHeapAllocs)
       
  1232 		{
       
  1233 		aHeap->iFlags |= RHeap::ETraceAllocs;
       
  1234 		BTraceContext8(BTrace::EHeap, BTrace::EHeapCreate,(TUint32)aHeap, RHeap::EAllocCellSize);
       
  1235 		TInt handle = aHeap->ChunkHandle();
       
  1236 		TInt chunkId = ((RHandleBase&)handle).BTraceId();
       
  1237 		BTraceContext8(BTrace::EHeap, BTrace::EHeapChunkCreate, (TUint32)aHeap, chunkId);
       
  1238 		}
       
  1239 	if (aInfo.iFlags & EMonitorHeapMemory)
       
  1240 		aHeap->iFlags |= RHeap::EMonitorMemory;
       
  1241 	return KErrNone;
       
  1242 	}
       
  1243 
       
  1244 #endif	// __KERNEL_MODE__
       
  1245 
       
  1246 void RHeap::WalkCheckCell(TAny* aPtr, TCellType aType, TAny* aCell, TInt aLen)
       
  1247 	{
       
  1248 	(void)aCell;
       
  1249 	SHeapCellInfo& info = *(SHeapCellInfo*)aPtr;
       
  1250 	switch(aType)
       
  1251 		{
       
  1252 		case EGoodAllocatedCell:
       
  1253 			{
       
  1254 			++info.iTotalAlloc;
       
  1255 			info.iTotalAllocSize += (aLen-EAllocCellSize);
       
  1256 #if defined(_DEBUG)
       
  1257 			RHeap& h = *info.iHeap;
       
  1258 			if ( ((SDebugCell*)aCell)->nestingLevel == h.iNestingLevel )
       
  1259 				{
       
  1260 				if (++info.iLevelAlloc==1)
       
  1261 					info.iStranded = (SDebugCell*)aCell;
       
  1262 #ifdef __KERNEL_MODE__
       
  1263 				if (KDebugNum(KSERVER) || KDebugNum(KTESTFAST))
       
  1264 					{
       
  1265 //				__KTRACE_OPT(KSERVER,Kern::Printf("LEAKED KERNEL HEAP CELL @ %08x : len=%d", aCell, aLen));
       
  1266 					Kern::Printf("LEAKED KERNEL HEAP CELL @ %08x : len=%d", aCell, aLen);
       
  1267 					TLinAddr base = ((TLinAddr)aCell)&~0x0f;
       
  1268 					TLinAddr end = ((TLinAddr)aCell)+(TLinAddr)aLen;
       
  1269 					while(base<end)
       
  1270 						{
       
  1271 						const TUint32* p = (const TUint32*)base;
       
  1272 						Kern::Printf("%08x: %08x %08x %08x %08x", p, p[0], p[1], p[2], p[3]);
       
  1273 						base += 16;
       
  1274 						}
       
  1275 					}
       
  1276 #endif
       
  1277 				}
       
  1278 #endif	
       
  1279 			break;
       
  1280 			}
       
  1281 		case EGoodFreeCell:
       
  1282 			++info.iTotalFree;
       
  1283 			break;
       
  1284 		case EBadAllocatedCellSize:
       
  1285 			HEAP_PANIC(ETHeapBadAllocatedCellSize);
       
  1286 		case EBadAllocatedCellAddress:
       
  1287 			HEAP_PANIC(ETHeapBadAllocatedCellAddress);
       
  1288 		case EBadFreeCellAddress:
       
  1289 			HEAP_PANIC(ETHeapBadFreeCellAddress);
       
  1290 		case EBadFreeCellSize:
       
  1291 			HEAP_PANIC(ETHeapBadFreeCellSize);
       
  1292 		default:
       
  1293 			HEAP_PANIC(ETHeapWalkBadCellType);
       
  1294 		}
       
  1295 	}
       
  1296 
       
  1297 TInt RHeap::DoCountAllocFree(TInt& aFree)
       
  1298 	{
       
  1299 	SHeapCellInfo info;
       
  1300 	memclr(&info, sizeof(info));
       
  1301 	info.iHeap = this;
       
  1302 	Walk(&WalkCheckCell, &info);
       
  1303 	aFree = info.iTotalFree;
       
  1304 	return info.iTotalAlloc;
       
  1305 	}
       
  1306 
       
  1307 
       
  1308 UEXPORT_C TInt RHeap::DebugFunction(TInt aFunc, TAny* a1, TAny* a2)
       
  1309 /**
       
  1310 @internalComponent
       
  1311 */
       
  1312 	{
       
  1313 	TInt r = KErrNone;
       
  1314 	switch(aFunc)
       
  1315 		{
       
  1316 		case RAllocator::ECount:
       
  1317 			r = DoCountAllocFree(*(TInt*)a1);
       
  1318 			break;
       
  1319 		case RAllocator::EMarkStart:
       
  1320 			__DEBUG_ONLY(DoMarkStart());
       
  1321 			break;
       
  1322 		case RAllocator::EMarkEnd:
       
  1323 			__DEBUG_ONLY( r = DoMarkEnd((TInt)a1) );
       
  1324 			break;
       
  1325 		case RAllocator::ECheck:
       
  1326 			r = DoCheckHeap((SCheckInfo*)a1);
       
  1327 			break;
       
  1328 		case RAllocator::ESetFail:
       
  1329 			__DEBUG_ONLY(DoSetAllocFail((TAllocFail)(TInt)a1, (TInt)a2));
       
  1330 			break;
       
  1331 		case RAllocator::ESetBurstFail:
       
  1332 #if _DEBUG
       
  1333 			{
       
  1334 			SRAllocatorBurstFail* fail = (SRAllocatorBurstFail*) a2;
       
  1335 			DoSetAllocFail((TAllocFail)(TInt)a1, fail->iRate, fail->iBurst);
       
  1336 			}
       
  1337 #endif
       
  1338 			break;
       
  1339 
       
  1340 		case RAllocator::ECheckFailure:
       
  1341 				// iRand will be incremented for each EFailNext, EBurstFailNext,
       
  1342 				// EDeterministic and EBurstDeterministic failure.
       
  1343 				r = iRand;
       
  1344 				break;
       
  1345 
       
  1346 		case RAllocator::ECopyDebugInfo:
       
  1347 			{
       
  1348 			TInt nestingLevel = ((SDebugCell*)a1)[-1].nestingLevel;
       
  1349 			((SDebugCell*)a2)[-1].nestingLevel = nestingLevel;
       
  1350 			break;
       
  1351 			}
       
  1352 		case RHeap::EWalk:
       
  1353 			Walk((TWalkFunc)a1, a2);
       
  1354 			break;
       
  1355 		default:
       
  1356 			return KErrNotSupported;
       
  1357 		}
       
  1358 	return r;
       
  1359 	}
       
  1360 
       
  1361 
       
  1362 
       
  1363 
       
  1364 void RHeap::Walk(TWalkFunc aFunc, TAny* aPtr)
       
  1365 //
       
  1366 // Walk the heap calling the info function.
       
  1367 //
       
  1368 	{
       
  1369 
       
  1370 	Lock();
       
  1371 	SCell* pC = (SCell*)iBase;		// allocated cells
       
  1372 	SCell* pF = &iFree;				// free cells
       
  1373 	FOREVER
       
  1374 		{
       
  1375 		pF = pF->next;				// next free cell
       
  1376 		if (!pF)
       
  1377 			pF = (SCell*)iTop;		// to make size checking work
       
  1378 		else if ( (TUint8*)pF>=iTop || (pF->next && pF->next<=pF) )
       
  1379 			{
       
  1380 			if (iFlags & ETraceAllocs)
       
  1381 				BTraceContext12(BTrace::EHeap, BTrace::EHeapCorruption, (TUint32)this, (TUint32)pF+EFreeCellSize, 0);
       
  1382 			// free cell pointer off the end or going backwards
       
  1383 			Unlock();
       
  1384 			(*aFunc)(aPtr, EBadFreeCellAddress, pF, 0);
       
  1385 			return;
       
  1386 			}
       
  1387 		else
       
  1388 			{
       
  1389 			TInt l = pF->len;
       
  1390 			if (l<iMinCell || (l & (iAlign-1)))
       
  1391 				{
       
  1392 				if (iFlags & ETraceAllocs)
       
  1393 					BTraceContext12(BTrace::EHeap, BTrace::EHeapCorruption, (TUint32)this, (TUint32)pF+EFreeCellSize, l-EFreeCellSize);
       
  1394 				// free cell length invalid
       
  1395 				Unlock();
       
  1396 				(*aFunc)(aPtr, EBadFreeCellSize, pF, l);
       
  1397 				return;
       
  1398 				}
       
  1399 			}
       
  1400 		while (pC!=pF)				// walk allocated cells up to next free cell
       
  1401 			{
       
  1402 			TInt l = pC->len;
       
  1403 			if (l<iMinCell || (l & (iAlign-1)))
       
  1404 				{
       
  1405 				if (iFlags & ETraceAllocs)
       
  1406 					BTraceContext12(BTrace::EHeap, BTrace::EHeapCorruption, (TUint32)this, (TUint32)pC+EAllocCellSize, l-EAllocCellSize);
       
  1407 				// allocated cell length invalid
       
  1408 				Unlock();
       
  1409 				(*aFunc)(aPtr, EBadAllocatedCellSize, pC, l);
       
  1410 				return;
       
  1411 				}
       
  1412 			(*aFunc)(aPtr, EGoodAllocatedCell, pC, l);
       
  1413 			SCell* pN = __NEXT_CELL(pC);
       
  1414 			if (pN > pF)
       
  1415 				{
       
  1416 				if (iFlags & ETraceAllocs)
       
  1417 					BTraceContext12(BTrace::EHeap, BTrace::EHeapCorruption, (TUint32)this, (TUint32)pC+EAllocCellSize, l-EAllocCellSize);			
       
  1418 				// cell overlaps next free cell
       
  1419 				Unlock();
       
  1420 				(*aFunc)(aPtr, EBadAllocatedCellAddress, pC, l);
       
  1421 				return;
       
  1422 				}
       
  1423 			pC = pN;
       
  1424 			}
       
  1425 		if ((TUint8*)pF == iTop)
       
  1426 			break;		// reached end of heap
       
  1427 		pC = __NEXT_CELL(pF);	// step to next allocated cell
       
  1428 		(*aFunc)(aPtr, EGoodFreeCell, pF, pF->len);
       
  1429 		}
       
  1430 	Unlock();
       
  1431 	}
       
  1432 
       
  1433 TInt RHeap::DoCheckHeap(SCheckInfo* aInfo)
       
  1434 	{
       
  1435 	(void)aInfo;
       
  1436 	SHeapCellInfo info;
       
  1437 	memclr(&info, sizeof(info));
       
  1438 	info.iHeap = this;
       
  1439 	Walk(&WalkCheckCell, &info);
       
  1440 #if defined(_DEBUG)
       
  1441 	if (!aInfo)
       
  1442 		return KErrNone;
       
  1443 	TInt expected = aInfo->iCount;
       
  1444 	TInt actual = aInfo->iAll ? info.iTotalAlloc : info.iLevelAlloc;
       
  1445 	if (actual!=expected && !iTestData)
       
  1446 		{
       
  1447 #ifdef __KERNEL_MODE__
       
  1448 		Kern::Fault("KERN-ALLOC COUNT", (expected<<16)|actual );
       
  1449 #else
       
  1450 		User::Panic(_L("ALLOC COUNT"), (expected<<16)|actual );
       
  1451 #endif
       
  1452 		}
       
  1453 #endif
       
  1454 	return KErrNone;
       
  1455 	}
       
  1456 
       
  1457 #ifdef _DEBUG
       
  1458 void RHeap::DoMarkStart()
       
  1459 	{
       
  1460 	if (iNestingLevel==0)
       
  1461 		iAllocCount=0;
       
  1462 	iNestingLevel++;
       
  1463 	}
       
  1464 
       
  1465 TUint32 RHeap::DoMarkEnd(TInt aExpected)
       
  1466 	{
       
  1467 	if (iNestingLevel==0)
       
  1468 		return 0;
       
  1469 	SHeapCellInfo info;
       
  1470 	SHeapCellInfo* p = iTestData ? (SHeapCellInfo*)iTestData : &info;
       
  1471 	memclr(p, sizeof(info));
       
  1472 	p->iHeap = this;
       
  1473 	Walk(&WalkCheckCell, p);
       
  1474 	if (p->iLevelAlloc != aExpected && !iTestData)
       
  1475 		return (TUint32)(p->iStranded + 1);
       
  1476 	if (--iNestingLevel == 0)
       
  1477 		iAllocCount = 0;
       
  1478 	return 0;
       
  1479 	}
       
  1480 
       
  1481 void ResetAllocCellLevels(TAny* aPtr, RHeap::TCellType aType, TAny* aCell, TInt aLen)
       
  1482 	{
       
  1483 	(void)aPtr;
       
  1484 	(void)aLen;
       
  1485 	RHeap::SDebugCell* cell = (RHeap::SDebugCell*)aCell;
       
  1486 	if (aType == RHeap::EGoodAllocatedCell)
       
  1487 		{
       
  1488 		cell->nestingLevel = 0;
       
  1489 		}
       
  1490 	}
       
  1491 
       
  1492 void RHeap::DoSetAllocFail(TAllocFail aType, TInt aRate)
       
  1493 	{// Default to a burst mode of 1, as aType may be a burst type.
       
  1494 	DoSetAllocFail(aType, aRate, 1);
       
  1495 	}
       
  1496 
       
  1497 // Don't change as the ETHeapBadDebugFailParameter check below and the API 
       
  1498 // documentation rely on this being 16 for RHeap.
       
  1499 LOCAL_D const TInt KBurstFailRateShift = 16;
       
  1500 LOCAL_D const TInt KBurstFailRateMask = (1 << KBurstFailRateShift) - 1;
       
  1501 
       
  1502 void RHeap::DoSetAllocFail(TAllocFail aType, TInt aRate, TUint aBurst)
       
  1503 	{
       
  1504 	if (aType==EReset)
       
  1505 		{
       
  1506 		// reset levels of all allocated cells to 0
       
  1507 		// this should prevent subsequent tests failing unnecessarily
       
  1508 		iFailed = EFalse;		// Reset for ECheckFailure relies on this.
       
  1509 		Walk(&ResetAllocCellLevels, NULL);
       
  1510 		// reset heap allocation mark as well
       
  1511 		iNestingLevel=0;
       
  1512 		iAllocCount=0;
       
  1513 		aType=ENone;
       
  1514 		}
       
  1515 
       
  1516 	switch (aType)
       
  1517 		{
       
  1518 		case EBurstRandom:
       
  1519 		case EBurstTrueRandom:
       
  1520 		case EBurstDeterministic:
       
  1521 		case EBurstFailNext:
       
  1522 			// If the fail type is a burst type then iFailRate is split in 2:
       
  1523 			// the 16 lsbs are the fail rate and the 16 msbs are the burst length.
       
  1524 			if (TUint(aRate) > (TUint)KMaxTUint16 || aBurst > KMaxTUint16)
       
  1525 				HEAP_PANIC(ETHeapBadDebugFailParameter);
       
  1526 
       
  1527 			iFailed = EFalse;
       
  1528 			iFailType = aType;
       
  1529 			iFailRate = (aRate == 0) ? 1 : aRate;
       
  1530 			iFailAllocCount = -iFailRate;
       
  1531 			iFailRate = iFailRate | (aBurst << KBurstFailRateShift);
       
  1532 			break;
       
  1533 
       
  1534 		default:
       
  1535 			iFailed = EFalse;
       
  1536 			iFailType = aType;
       
  1537 			iFailRate = (aRate == 0) ? 1 : aRate; // A rate of <1 is meaningless
       
  1538 			iFailAllocCount = 0;
       
  1539 			break;
       
  1540 		}
       
  1541 
       
  1542 	// Set up iRand for either:
       
  1543 	//		- random seed value, or
       
  1544 	//		- a count of the number of failures so far.
       
  1545 	iRand = 0;
       
  1546 #ifndef __KERNEL_MODE__
       
  1547 	switch (iFailType)
       
  1548 		{
       
  1549 		case ETrueRandom:
       
  1550 		case EBurstTrueRandom:
       
  1551 			{
       
  1552 			TTime time;
       
  1553 			time.HomeTime();
       
  1554 			TInt64 seed = time.Int64();
       
  1555 			iRand = Math::Rand(seed);
       
  1556 			break;
       
  1557 			}
       
  1558 		case ERandom:
       
  1559 		case EBurstRandom:
       
  1560 	        {
       
  1561 	        TInt64 seed = 12345;
       
  1562 			iRand = Math::Rand(seed);
       
  1563 			break;
       
  1564 	        }
       
  1565 		default:
       
  1566 			break;
       
  1567 		}
       
  1568 #endif
       
  1569 	}
       
  1570 
       
  1571 TBool RHeap::CheckForSimulatedAllocFail()
       
  1572 //
       
  1573 // Check to see if the user has requested simulated alloc failure, and if so possibly 
       
  1574 // Return ETrue indicating a failure.
       
  1575 //
       
  1576 	{
       
  1577 	// For burst mode failures iFailRate is shared
       
  1578 	TUint16 rate  = (TUint16)(iFailRate &  KBurstFailRateMask);
       
  1579 	TUint16 burst = (TUint16)(iFailRate >> KBurstFailRateShift);
       
  1580 	TBool r = EFalse;
       
  1581 	switch (iFailType)
       
  1582 		{
       
  1583 #ifndef __KERNEL_MODE__
       
  1584 		case ERandom:
       
  1585 		case ETrueRandom:
       
  1586 			if (++iFailAllocCount>=iFailRate) 
       
  1587 				{	
       
  1588 				iFailAllocCount=0;
       
  1589 				if (!iFailed) // haven't failed yet after iFailRate allocations so fail now
       
  1590 					return(ETrue); 
       
  1591 				iFailed=EFalse;
       
  1592 				}
       
  1593 			else   
       
  1594 				{
       
  1595 				if (!iFailed)
       
  1596 					{
       
  1597 	                TInt64 seed=iRand;
       
  1598 					iRand=Math::Rand(seed);
       
  1599 					if (iRand%iFailRate==0)
       
  1600 						{
       
  1601 						iFailed=ETrue;
       
  1602 						return(ETrue);
       
  1603 						}
       
  1604 					}
       
  1605 				}
       
  1606 			break;
       
  1607 
       
  1608 		case EBurstRandom:
       
  1609 		case EBurstTrueRandom:
       
  1610 			if (++iFailAllocCount < 0) 
       
  1611 				{
       
  1612 				// We haven't started failing yet so should we now?
       
  1613 				TInt64 seed = iRand;
       
  1614 				iRand = Math::Rand(seed);
       
  1615 				if (iRand % rate == 0)
       
  1616 					{// Fail now.  Reset iFailAllocCount so we fail burst times
       
  1617 					iFailAllocCount = 0;
       
  1618 					r = ETrue;
       
  1619 					}
       
  1620 				}
       
  1621 			else
       
  1622 				{
       
  1623 				if (iFailAllocCount < burst)
       
  1624 					{// Keep failing for burst times
       
  1625 					r = ETrue;
       
  1626 					}
       
  1627 				else
       
  1628 					{// We've now failed burst times so start again.
       
  1629 					iFailAllocCount = -(rate - 1);
       
  1630 					}
       
  1631 				}
       
  1632 			break;
       
  1633 #endif
       
  1634 		case EDeterministic:
       
  1635 			if (++iFailAllocCount%iFailRate==0)
       
  1636 				{
       
  1637 				r=ETrue;
       
  1638 				iRand++;	// Keep count of how many times we have failed
       
  1639 				}
       
  1640 			break;
       
  1641 
       
  1642 		case EBurstDeterministic:
       
  1643 			// This will fail burst number of times, every rate attempts.
       
  1644 			if (++iFailAllocCount >= 0)
       
  1645 				{
       
  1646 				if (iFailAllocCount == burst - 1)
       
  1647 					{// This is the burst time we have failed so make it the last by
       
  1648 					// reseting counts so we next fail after rate attempts.
       
  1649 					iFailAllocCount = -rate;
       
  1650 					}
       
  1651 				r = ETrue;
       
  1652 				iRand++;	// Keep count of how many times we have failed
       
  1653 				}
       
  1654 			break;
       
  1655 
       
  1656 		case EFailNext:
       
  1657 			if ((++iFailAllocCount%iFailRate)==0)
       
  1658 				{
       
  1659 				iFailType=ENone;
       
  1660 				r=ETrue;
       
  1661 				iRand++;	// Keep count of how many times we have failed
       
  1662 				}
       
  1663 			break;
       
  1664 
       
  1665 		case EBurstFailNext:
       
  1666 			if (++iFailAllocCount >= 0)
       
  1667 				{
       
  1668 				if (iFailAllocCount == burst - 1)
       
  1669 					{// This is the burst time we have failed so make it the last.
       
  1670 					iFailType = ENone;
       
  1671 					}
       
  1672 				r = ETrue;
       
  1673 				iRand++;	// Keep count of how many times we have failed
       
  1674 				}
       
  1675 			break;
       
  1676 		default:
       
  1677 			break;
       
  1678 		}
       
  1679 	return r;
       
  1680 	}
       
  1681 #endif	// ifdef _DEBUG
       
  1682 
       
  1683 UEXPORT_C TInt RHeap::Extension_(TUint aExtensionId, TAny*& a0, TAny* a1)
       
  1684 	{
       
  1685 	return RAllocator::Extension_(aExtensionId, a0, a1);
       
  1686 	}
       
  1687 
       
  1688 #if defined(__HEAP_MACHINE_CODED__) && !defined(_DEBUG)
       
  1689 GLDEF_C void RHeap_PanicBadAllocatedCellSize()
       
  1690 	{
       
  1691 	HEAP_PANIC(ETHeapBadAllocatedCellSize);
       
  1692 	}
       
  1693 
       
  1694 GLDEF_C void RHeap_PanicBadNextCell()
       
  1695 	{
       
  1696 	HEAP_PANIC(ETHeapFreeBadNextCell);
       
  1697 	}
       
  1698 
       
  1699 GLDEF_C void RHeap_PanicBadPrevCell()
       
  1700 	{
       
  1701 	HEAP_PANIC(ETHeapFreeBadPrevCell);
       
  1702 	}
       
  1703 
       
  1704 GLDEF_C void RHeap_PanicBadCellAddress()
       
  1705 	{
       
  1706 	HEAP_PANIC(ETHeapBadCellAddress);
       
  1707 	}
       
  1708 #endif
       
  1709 
       
  1710 
       
  1711 
       
  1712 
       
  1713