Update contrib.
1 // Copyright (c) 2010 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
14 // Hexadecimal trees - implementation
18 #include <e32atomics.h>
20 EXPORT_C RHexTreeBase::RHexTreeBase(RHeap* aHeap)
23 Mem::FillZ(iRootOffsets, sizeof(iRootOffsets));
26 EXPORT_C TInt RHexTreeBase::SetAt(TUint aKey, TAny* aValue)
28 TUint mask = 0xF0000000;
29 for (TInt height = EMaxNumHexDigits; height > 1; --height)
31 if ((aKey & mask) != 0)
33 return SetAt(aKey, aValue, height);
37 return SetAt(aKey, aValue, 1);
40 EXPORT_C TAny* RHexTreeBase::At(TUint aKey) const
42 TUint mask = 0xF0000000;
43 for (TInt height = EMaxNumHexDigits; height > 1; --height)
45 if ((aKey & mask) != 0)
47 return At(aKey, height);
55 Empties this associative array and frees all memory allocated both for the
56 associative array implementation and for the values that have been added to
57 this associative array.
59 The internal state of this associative array is reset so that it can be reused
60 or allowed to go out of scope after a call to this function.
62 EXPORT_C void RHexTreeBase::ResetAndDestroy()
64 for (TInt height = 1; height <= EMaxNumHexDigits; ++height)
66 TInt offset = iRootOffsets[height - 1];
69 TAny* root = PtrAdd(this, offset);
70 ResetAndDestroy(height, root, 1);
71 iRootOffsets[height - 1] = 0;
76 TInt RHexTreeBase::SetAt(TUint aKey, TAny* aLeaf, TInt aHeight)
79 TInt offset = iRootOffsets[aHeight - 1];
82 TAny* root = iHeap->AllocZ(aHeight > 1 ? sizeof(TInt) * 15 : sizeof(TInt) * 16);
87 err = SetAt(aKey, aLeaf, aHeight, root, 1);
90 __e32_atomic_store_rel32(&iRootOffsets[aHeight - 1], reinterpret_cast<TInt>(root) - reinterpret_cast<TInt>(this));
99 TAny* root = PtrAdd(this, offset);
100 err = SetAt(aKey, aLeaf, aHeight, root, 1);
105 TInt RHexTreeBase::SetAt(TUint aKey, TAny* aLeaf, TInt aHeight, TAny* aNode, TInt aLevel)
108 TInt branch = (aKey >> ((aHeight - aLevel) << 2)) & 0xF;
109 if (aLevel == 1 && aHeight > 1)
113 TInt offset = static_cast<TInt*>(aNode)[branch];
114 if (aLevel == aHeight)
118 __e32_atomic_store_rel32(&static_cast<TInt*>(aNode)[branch], reinterpret_cast<TInt>(aLeaf) - reinterpret_cast<TInt>(aNode));
122 err = KErrAlreadyExists;
125 else if (offset == 0)
127 TAny* newNode = iHeap->AllocZ(sizeof(TInt) * 16);
132 err = SetAt(aKey, aLeaf, aHeight, newNode, aLevel + 1);
135 __e32_atomic_store_rel32(&static_cast<TInt*>(aNode)[branch], reinterpret_cast<TInt>(newNode) - reinterpret_cast<TInt>(aNode));
139 iHeap->Free(newNode);
144 TAny* nextNode = PtrAdd(aNode, offset);
145 err = SetAt(aKey, aLeaf, aHeight, nextNode, aLevel + 1);
150 TAny* RHexTreeBase::At(TUint aKey, TInt aHeight) const
152 TInt offset = __e32_atomic_load_acq32(&iRootOffsets[aHeight - 1]);
157 const TAny* node = PtrAdd(this, offset);
158 for (TInt level = 1; level <= aHeight; ++level)
160 TInt branch = (aKey >> ((aHeight - level) << 2)) & 0xF;
161 if (level == 1 && aHeight > 1)
165 offset = __e32_atomic_load_acq32(&static_cast<const TInt*>(node)[branch]);
170 node = PtrAdd(node, offset);
172 return const_cast<TAny*>(node);
175 void RHexTreeBase::ResetAndDestroy(TInt aHeight, TAny* aNode, TInt aLevel)
177 TInt maxNumBranches = (aLevel == 1 && aHeight > 1 ? 15 : 16);
178 for (TInt branch = 0; branch < maxNumBranches; ++branch)
180 TInt offset = static_cast<TInt*>(aNode)[branch];
183 TAny* nextNode = PtrAdd(aNode, offset);
184 if (aLevel == aHeight)
186 iHeap->Free(nextNode);
190 ResetAndDestroy(aHeight, nextNode, aLevel + 1);