Update contrib.
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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
15 // This file is directly included in the test harness t_tbma
19 #include <kernel/kbma.h>
24 #define __TBMA_MACHINE_CODED__
28 #include <e32std_private.h>
29 #include <e32atomics.h>
31 #define __ALLOC(x) User::Alloc(x)
33 void TBmaFault(TInt aLine)
35 User::Panic(_L("TBMA"),aLine);
40 #include <kernel/kern_priv.h>
42 #define __ALLOC(x) Kern::Alloc(x)
44 void TBmaFault(TInt aLine)
46 Kern::Fault("TBMA",aLine);
51 #define TBMA_FAULT() TBmaFault(__LINE__)
53 /** Creates a new TBitMapAllocator object.
55 @param aSize The number of bit positions required, must be >0.
56 @param aState TRUE if all bit positions initially free
57 FALSE if all bit positions initially allocated.
59 @return Pointer to new object, NULL if out of memory.
61 @pre Calling thread must be in a critical section.
62 @pre No fast mutex can be held.
63 @pre Call in a thread context.
64 @pre Interrupts must be enabled.
65 @pre Kernel must be unlocked.
67 EXPORT_C TBitMapAllocator* TBitMapAllocator::New(TInt aSize, TBool aState)
69 #ifndef TBMA_TEST_CODE
70 CHECK_PRECONDITIONS(MASK_THREAD_CRITICAL,"TBitMapAllocator::New");
72 TInt nmapw=(aSize+31)>>5;
73 TInt memsz=sizeof(TBitMapAllocator)+(nmapw-1)*sizeof(TUint32);
74 TBitMapAllocator* pA=(TBitMapAllocator*)__ALLOC(memsz);
76 new(pA) TBitMapAllocator(aSize, aState);
81 /** Finds a set of consecutive bit positions with specified alignment, with
82 support for chaining multiple allocators.
84 Note that this function does not mark the positions as allocated.
87 1. Any initial run is added to the carry in
88 2. If all bits free, if BMA size+carry<=request length return 0 and leave carry alone
89 else add size to carry and return KErrOverflow
90 3. If request satisfied by initial run + carry return 0 and leave carry alone
91 4. If request satisfied by an intermediate or final run, return start pos of run and set carry=0
92 5. Otherwise carry = length of any final run and return KErrNotFound
94 With a single allocator set aCarry (and usually aBase) to zero and ignore
95 aRunLength. The return value then indicates the required position (after
96 being aligned up as necessary) or KErrNotFound.
98 With multiple allocators, this function should be called on each allocator in
99 increasing logical bit number order. The carry should be set to zero initially
100 and if there is a gap in the logical bit number between two allocators, otherwise
101 it should be left alone. The first call which returns a nonnegative value indicates
102 success and the required logical bit position is given by aligning up
103 logical bit number of first bit of allocator + return value - carry
106 1. Any initial run is added to the carry in
107 2. If all bits free, add bma length to carry and return KErrOverflow
108 3. If any run including initial+carry but not final has length >= request length
109 return start pos and length of smallest such, also set carry = length of final run
110 unless exact match found, when carry is either unchanged or set to 0
111 4. If only final run large enough, return KErrNotFound and set carry = length of final run
112 carry=0 if no final run
114 Here is an example of how to use this for multiple allocators:
116 // aLength = run length required, aAlign = alignment constraint
119 TInt minrun=KMaxTInt; // this will track the length of the shortest useful run
120 TInt minrunpos=KErrNotFound; // this will track the start position of the shortest useful run
121 TUint32 alignsize=1<<aAlign;
122 TUint32 alignmask=alignsize-1;
123 TBitMapAllocator** ppA=iBmaList; // pointer to list of TBitMapAllocator*
124 TBitMapAllocator** pE=ppA+iNumBmas; // pointer to end of list
125 TInt* pB=iBaseList; // pointer to list of initial logical bit numbers
127 for (; ppA<pE; ++ppA, ++pB)
129 TBitMapAllocator* pA=*ppA;
130 if (*pB!=base+bmalen)
132 // this BMA is not contiguous with previous one
133 // check final run of previous BMA
136 TInt fpos=base+bmalen-carry;
137 TInt lost=((fpos+base+alignmask)&~alignmask)-base-fpos;
138 if (carry-lost>=aLength)
149 TInt oldc=carry; // need to save this for the case where the best run is the initial one
150 TInt r=pA->AllocAligned(aLength,aAlign,base,ETrue,carry,l);
153 // check shortest run in this BMA
157 minrunpos=r ? (base+r) : (base-oldc);
159 break; // exact match so finish
163 // check final run of last BMA (unless exact match already found)
164 if (ppA==pE && carry<minrun)
166 TInt fpos=base+bmalen-carry;
167 TInt lost=((fpos+alignmask)&~alignmask)-fpos;
168 if (carry-lost>=aLength)
174 result = (minrunpos<0) ? minrunpos : ((minrunpos+alignmask)&~alignmask);
179 @param aLength number of consecutive bit positions to allocate.
180 @param aAlign logarithm to base 2 of the alignment required.
181 @param aBase the alignment of the first bit of this allocator - only significant modulo 2^aAlign.
182 @param aBestFit TRUE for best fit allocation strategy, FALSE for first fit.
183 @param aCarry carry in/carry out.
184 @param aRunLength Holds best run length found so far. This will be set to KMaxTInt when no
185 suitable run length has been found. In best fit mode aCarry should also be
186 checked as aRunLength will not be set if aCarry is the only suitable run length
189 @return Start position, if a suitable run was found;
190 KErrNotFound, if no suitable run was found;
191 KErrOverflow, if all positions free and best fit mode, or if all positions free
192 in first fit mode and length requested > number of positions available.
194 @see TBitMapAllocator::AllocConsecutive(TInt aLength, TBool aBestFit)
195 @see TBitMapAllocator::AllocAligned(TInt aLength, TInt aAlign, TInt aBase, TBool aBestFit)
197 EXPORT_C TInt TBitMapAllocator::AllocAligned(TInt aLength, TInt aAlign, TInt aBase, TBool aBestFit, TInt& aCarry, TInt& aRunLength) const
199 return AllocAligned(aLength, aAlign, aBase, aBestFit, aCarry, aRunLength, 0);
203 /** Allocates the next available bit position starting from the specified offset.
205 Note - If no free bit positions can be found after aOffset this method will
206 wrap around and continue the search from the start of the bit map.
208 @param aOffset The offset from the start of the bit map.
209 @return The number of the bit position allocated, -1 if all positions are occupied.
211 EXPORT_C TInt TBitMapAllocator::AllocFrom(TUint aOffset)
213 __ASSERT_ALWAYS(aOffset < (TUint)iSize, TBMA_FAULT());
218 const TUint32* pEnd = iMap + ((iSize+31)>>5);
219 TUint32* pW = iMap + (aOffset >> 5);
220 // Only check the bits after aOffset in this word.
221 TUint wordMask = 0xffffffffu >> (aOffset & 0x1f);
223 if(!((aOffset&0x1f)==0 || (wordMask&0x80000000u)==0)) // check compiler has done unsigned >>
226 TUint word = *pW & wordMask;
227 // No free bit positions in this word so search through the rest of the words.
235 TInt n = __e32_find_ms1_32(word);
237 n = (31 - n) + ((pW - iMap) << 5);
242 #if !defined( __TBMA_MACHINE_CODED__) | defined(__EABI_CTORS__)
243 /** Constructs a new TBitMapAllocator object.
245 @param aSize The number of bit positions required.
246 @param aState TRUE if all bit positions initially free;
247 FALSE if all bit positions initially allocated.
249 EXPORT_C TBitMapAllocator::TBitMapAllocator(TInt aSize, TBool aState)
251 __ASSERT_ALWAYS(aSize>0, TBMA_FAULT());
258 for (; aSize>=32; aSize-=32)
261 *pW=((0xffffffffu)<<(32-aSize));
265 TInt nmapw=(aSize+31)>>5;
267 iCheckFirst=iMap+nmapw-1;
268 memclr(iMap, nmapw*sizeof(TUint32));
274 #if !defined( __TBMA_MACHINE_CODED__)
275 /** Allocates the next available bit position.
277 @return Number of position allocated, -1 if all positions occupied.
279 EXPORT_C TInt TBitMapAllocator::Alloc()
284 TUint32* pW=iCheckFirst;
288 TInt n=__e32_find_ms1_32(*pW);
290 n=(31-n)+((pW-iMap)<<5);
295 /** Frees the specified bit position.
297 @param aPos Number of bit position to be freed; must be currently allocated.
299 EXPORT_C void TBitMapAllocator::Free(TInt aPos)
301 __ASSERT_ALWAYS(TUint(aPos)<TUint(iSize), TBMA_FAULT());
302 TUint32* pW=iMap+(aPos>>5);
303 TUint32 b=0x80000000u>>(aPos&31);
304 __ASSERT_ALWAYS(!(*pW & b), TBMA_FAULT());
312 /** Allocates a specific range of bit positions.
314 The specified range must lie within the total range for this allocator and all
315 the positions must currently be free.
317 @param aStart First position to allocate.
318 @param aLength Number of consecutive positions to allocate, must be >0.
320 EXPORT_C void TBitMapAllocator::Alloc(TInt aStart, TInt aLength)
322 __ASSERT_ALWAYS(TUint(aStart)<TUint(iSize), TBMA_FAULT());
323 __ASSERT_ALWAYS(TUint(aStart+aLength)>=TUint(aStart), TBMA_FAULT());
324 __ASSERT_ALWAYS(TUint(aStart+aLength)<=TUint(iSize), TBMA_FAULT());
327 TUint32* pW=iMap+wix;
329 TInt ebit=sbit+aLength;
332 TUint32 b=(~(0xffffffffu>>aLength)>>sbit);
334 __ASSERT_ALWAYS((w|b)==w, TBMA_FAULT());
338 TUint32 b=(0xffffffffu>>sbit);
342 __ASSERT_ALWAYS((w|b)==w, TBMA_FAULT());
352 /** Frees a specific range of bit positions.
354 The specified range must lie within the total range for this allocator and all
355 the positions must currently be allocated.
357 @param aStart First position to free.
358 @param aLength Number of consecutive positions to free, must be >0.
360 EXPORT_C void TBitMapAllocator::Free(TInt aStart, TInt aLength)
362 __ASSERT_ALWAYS(TUint(aStart)<TUint(iSize), TBMA_FAULT());
363 __ASSERT_ALWAYS(TUint(aStart+aLength)>=TUint(aStart), TBMA_FAULT());
364 __ASSERT_ALWAYS(TUint(aStart+aLength)<=TUint(iSize), TBMA_FAULT());
367 TUint32* pW=iMap+wix;
368 if (!iAvail || pW<iCheckFirst)
371 TInt ebit=sbit+aLength;
374 TUint32 b=(~(0xffffffffu>>aLength)>>sbit);
376 __ASSERT_ALWAYS((w&b)==0, TBMA_FAULT());
380 TUint32 b=(0xffffffffu>>sbit);
384 __ASSERT_ALWAYS((w&b)==0, TBMA_FAULT());
394 /** Frees a specific range of bit positions.
396 The specified range must lie within the total range for this allocator but it is
397 not necessary that all the positions are currently allocated.
399 @param aStart First position to free.
400 @param aLength Number of consecutive positions to free, must be >0.
402 EXPORT_C void TBitMapAllocator::SelectiveFree(TInt aStart, TInt aLength)
404 __ASSERT_ALWAYS(TUint(aStart)<TUint(iSize), TBMA_FAULT());
405 __ASSERT_ALWAYS(TUint(aStart+aLength)>=TUint(aStart), TBMA_FAULT());
406 __ASSERT_ALWAYS(TUint(aStart+aLength)<=TUint(iSize), TBMA_FAULT());
409 TUint32* pW=iMap+wix;
410 if (!iAvail || pW<iCheckFirst)
412 iAvail+=aLength; // update free count assuming no positions already free
413 TInt ebit=sbit+aLength;
416 TUint32 b=(~(0xffffffffu>>aLength)>>sbit);
418 *pW=w|b; // mark all positions free
419 iAvail-=__e32_bit_count_32(w&b); // reduce free count by number of positions already free
422 TUint32 b=(0xffffffffu>>sbit);
426 *pW++=w|b; // mark all positions free
427 iAvail-=__e32_bit_count_32(w&b); // reduce free count by number of positions already free
436 /** Tests whether a specific range of bit positions are all free.
438 The specified range must lie within the total range for this allocator.
440 @param aStart First position to check.
441 @param aLength Number of consecutive positions to check, must be >0.
443 @return FALSE if all positions free, TRUE if at least one is occupied.
445 EXPORT_C TBool TBitMapAllocator::NotFree(TInt aStart, TInt aLength) const
447 // Inverse logic - returns 0 if all positions free, nonzero otherwise
448 __ASSERT_ALWAYS(TUint(aStart)<TUint(iSize), TBMA_FAULT());
449 __ASSERT_ALWAYS(TUint(aStart+aLength)>=TUint(aStart), TBMA_FAULT());
450 __ASSERT_ALWAYS(TUint(aStart+aLength)<=TUint(iSize), TBMA_FAULT());
453 const TUint32* pW=iMap+wix;
454 TInt ebit=sbit+aLength;
457 TUint32 b=(~(0xffffffffu>>aLength)>>sbit);
460 TUint32 b=(0xffffffffu>>sbit);
474 /** Tests whether a specific range of bit positions are all occupied.
476 The specified range must lie within the total range for this allocator.
478 @param aStart First position to check.
479 @param aLength Number of consecutive positions to check, must be >0.
481 @return FALSE if all positions occupied, TRUE if at least one is free.
483 EXPORT_C TBool TBitMapAllocator::NotAllocated(TInt aStart, TInt aLength) const
485 // Inverse logic - returns 0 if all positions allocated, nonzero otherwise
486 __ASSERT_ALWAYS(TUint(aStart)<TUint(iSize), TBMA_FAULT());
487 __ASSERT_ALWAYS(TUint(aStart+aLength)>=TUint(aStart), TBMA_FAULT());
488 __ASSERT_ALWAYS(TUint(aStart+aLength)<=TUint(iSize), TBMA_FAULT());
491 const TUint32* pW=iMap+wix;
492 TInt ebit=sbit+aLength;
495 TUint32 b=(~(0xffffffffu>>aLength)>>sbit);
498 TUint32 b=(0xffffffffu>>sbit);
512 /** Allocates up to a specified number of available bit positions.
514 The allocated positions are not required to bear any relationship to
516 If the number of free positions is less than the number requested,
517 allocate all currently free positions.
519 @param aLength Maximum number of positions to allocate.
520 @param aList Pointer to memory area where allocated bit numbers should be stored.
522 @return The number of positions allocated.
524 EXPORT_C TInt TBitMapAllocator::AllocList(TInt aLength, TInt* aList)
526 __ASSERT_ALWAYS(aLength>0, TBMA_FAULT());
536 /** Finds a set of consecutive bit positions with specified alignment starting the
537 search from the specfied bit position offset, with support for chaining
540 For further details see:
541 TBitMapAllocator::AllocAligned(TInt aLength, TInt aAlign, TInt aBase, TBool aBestFit, TInt& aCarry, TInt& aRunLength)
543 @param aLength number of consecutive bit positions to allocate.
544 @param aAlign logarithm to base 2 of the alignment required.
545 @param aBase the alignment of the first bit of this allocator - only significant modulo 2^aAlign.
546 @param aBestFit TRUE for best fit allocation strategy, FALSE for first fit.
547 @param aCarry carry in/carry out.
548 @param aRunLength Holds best run length found so far. This will be set to KMaxTInt when no
549 suitable run length has been found. In best fit mode aCarry should also be
550 checked as aRunLength will not be set if aCarry is the only suitable run length
552 @param aOffset The bit position to start the search from, set to 0 to search all bit positions.
553 aOffset will be aligned so all bits before an aligned aOffset will be
554 ignored. This can only be non-zero if aCarry is zero as any carry in bits will be
555 ignored if aOffset is non-zero.
557 @return Start position, if a suitable run was found;
558 KErrNotFound, if no suitable run was found;
559 KErrOverflow, if all positions free and best fit mode, or if all positions free
560 in first fit mode and length requested > number of positions available.
562 @see TBitMapAllocator::AllocAligned(TInt aLength, TInt aAlign, TInt aBase, TBool aBestFit, TInt& aCarry, TInt& aRunLength)
564 EXPORT_C TInt TBitMapAllocator::AllocAligned(TInt aLength, TInt aAlign, TInt aBase, TBool aBestFit, TInt& aCarry, TInt& aRunLength, TUint aOffset) const
567 __ASSERT_ALWAYS(aLength>0, TBMA_FAULT());
568 __ASSERT_ALWAYS(TUint(aAlign)<31, TBMA_FAULT());
569 __ASSERT_ALWAYS(aOffset < (TUint)iSize, TBMA_FAULT());
570 __ASSERT_ALWAYS(!aCarry || !aOffset, TBMA_FAULT());
571 TUint32 alignsize=1<<aAlign;
572 TUint32 alignmask=alignsize-1;
576 // Align aOffset if it is set so we ignore all bits before the aligned offset.
577 aOffset = (!aOffset)? aOffset : ((aOffset + aBase + alignmask) & ~alignmask) - aBase;
578 TInt runLength = (aOffset < (TUint)iSize)? iSize - aOffset : 0;
581 TInt alignedStartPos = ((aOffset - aCarry + aBase + alignmask) & ~alignmask) - aBase;
582 TInt lost = alignedStartPos - (aOffset - aCarry);
583 if (runLength + aCarry - lost >= aLength)
585 aRunLength = runLength;
586 if (alignedStartPos >= 0)
588 aCarry=0; // clear carry if not initial run
590 return (alignedStartPos >= 0)? alignedStartPos : 0; // return start pos of exact run
597 aRunLength = KMaxTInt;
600 const TUint32* pW=aCarry?iMap:iCheckFirst;
601 const TUint32* pE=iMap+((iSize+31)>>5);
602 TInt n=((pW-iMap)<<5);
605 TUint32 s=aCarry?~0:0; // 0 when searching for 1's, FFFFFFFF when searching for 0's
607 TUint32 offsetMask = 0;
609 {// Start search from aOffset. Only align aOffset if aOffset is to
610 // be used otherwise the best fit mode may fail as aligning aOffset
611 // may cause the search to skip past parts of the bit map.
612 aOffset = ((aOffset + aBase + alignmask) & ~alignmask) - aBase;
613 const TUint32* offsetWord = iMap + (aOffset >> 5);
614 if (offsetWord >= pW)
617 n = aOffset & 0xffffffe0;
618 offsetMask = 0xffffffff >> (aOffset & 31);
619 __ASSERT_ALWAYS(offsetMask, TBMA_FAULT());
624 TUint32 word = *pW++;
626 {// Start search after bit aOffset.
627 word &= offsetMask; // Mask off any bits before the aOffset
628 offsetMask = 0; // Reset so future iterations use whole of next word.
630 if (word==s) // check if any of required bit present
632 n+=32; // if not, step bit number on by 32
636 for (TUint32 b=0x80000000; b; ++n, b>>=1)
641 break; // reached end
642 // bit found - invert search bit
645 q=n; // 1 found so save position
648 rl=n-q; // 0 found, calculate run length of 1's
649 TInt alignedStartPos = ((q + aBase + alignmask) & ~alignmask) - aBase;
650 TInt lost = alignedStartPos - q;
651 if (rl-lost>=aLength)
653 if (!aBestFit || rl==aLength)
655 // first fit or exact match - we're finished
656 if (alignedStartPos >= 0)
658 aCarry=0; // clear carry if not initial run
661 return (alignedStartPos >= 0)? alignedStartPos : 0;
665 // best fit and this run is smallest so far, so record its position and length
667 p = (alignedStartPos >= 0)? alignedStartPos : 0;
676 // exact match not found or first fit and no match found
680 // we were looking for 0, so this counts as a run
685 TInt alignedStartPos = ((q + aBase + alignmask) & ~alignmask) - aBase;
686 TInt lost = alignedStartPos - q;
687 if (rl-lost>=aLength)
688 {// BMA is not totally empty so this can't be the initial run
689 // and the final run. Therefore the start pos must be within
690 // this bma so clear carry and return start pos.
693 return alignedStartPos;
697 aCarry=rl; // set carry to length of final run or 0 if none
699 aRunLength=minrl; // return best run length found
700 return p; // return start position of run or -1 if run not found
705 /** Finds a set of consecutive free positions on a single bit map allocator.
707 @param aLength number of consecutive bit positions to allocate.
708 @param aBestFit TRUE for best fit allocation strategy, FALSE for first fit.
710 @return Start position, if a suitable run was found;
711 KErrNotFound, if no suitable run was found.
713 EXPORT_C TInt TBitMapAllocator::AllocConsecutive(TInt aLength, TBool aBestFit) const
717 TInt r=AllocAligned(aLength,0,0,aBestFit,carry,l);
720 // must check final run if any
721 if (carry>=aLength && carry<l)
730 /** Finds a set of consecutive free positions on a single bit map allocator with
733 @param aLength number of consecutive bit positions to allocate.
734 @param aAlign logarithm to base 2 of the alignment required.
735 @param aBase the alignment of the first bit of this allocator - only significant modulo 2^aAlign.
736 @param aBestFit TRUE for best fit allocation strategy, FALSE for first fit.
738 @return Start position, if a suitable run was found;
739 KErrNotFound, if no suitable run was found.
741 EXPORT_C TInt TBitMapAllocator::AllocAligned(TInt aLength, TInt aAlign, TInt aBase, TBool aBestFit) const
745 TUint32 alignsize=1<<aAlign;
746 TUint32 alignmask=alignsize-1;
748 TInt r=AllocAligned(aLength,aAlign,aBase,aBestFit,carry,l);
751 // must check final run if any
752 TInt fpos=iSize-carry;
753 TInt lost=((fpos+aBase+alignmask)&~alignmask)-aBase-fpos;
754 if (carry-lost>=aLength && carry<l)
760 r=((r+aBase+alignmask)&~alignmask)-aBase;
765 /** Copies a range from another allocator, mark remainder as occupied.
767 Values of bit positions from aFirst to aFirst+aLen-1 inclusive in allocator
768 aA are copied to bit positions in this allocator starting with aFirst mod 32.
769 Remaining bit positions in this allocator are marked as occupied.
771 @param aA Pointer to source allocator.
772 @param aFirst Number in source allocator of first bit to copy.
773 @param aLen Number of bits to copy from source allocator.
775 EXPORT_C void TBitMapAllocator::CopyAlignedRange(const TBitMapAllocator* aA, TInt aFirst, TInt aLen)
777 const TUint32* srcptr = aA->iMap + (aFirst>>5);
778 TInt last = aFirst + aLen - 1;
779 TInt len = (((last+32)&~31)-(aFirst&~31))>>3; // bytes
780 __ASSERT_ALWAYS(len<=iSize, TBMA_FAULT());
781 TInt remain = ((iSize+31)&~31)-(len<<3);
782 wordmove(iMap, srcptr, len);
783 memclr(iMap+(len>>2), remain>>3);
785 TUint32* pE = p + (len>>2);
786 *p &= (0xffffffffu >> (aFirst&31));
787 pE[-1] &= (0xffffffffu << (31-(last&31)));
797 iAvail += __e32_bit_count_32(x);