sl@0
|
1 |
// Copyright (c) 1998-2009 Nokia Corporation and/or its subsidiary(-ies).
|
sl@0
|
2 |
// All rights reserved.
|
sl@0
|
3 |
// This component and the accompanying materials are made available
|
sl@0
|
4 |
// under the terms of "Eclipse Public License v1.0"
|
sl@0
|
5 |
// which accompanies this distribution, and is available
|
sl@0
|
6 |
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
|
sl@0
|
7 |
//
|
sl@0
|
8 |
// Initial Contributors:
|
sl@0
|
9 |
// Nokia Corporation - initial contribution.
|
sl@0
|
10 |
//
|
sl@0
|
11 |
// Contributors:
|
sl@0
|
12 |
//
|
sl@0
|
13 |
// Description:
|
sl@0
|
14 |
//
|
sl@0
|
15 |
|
sl@0
|
16 |
#include "UB_STD.H"
|
sl@0
|
17 |
|
sl@0
|
18 |
//#pragma message( __FILE__ " : 'TBtreePath' does not track insertion position" )
|
sl@0
|
19 |
//#pragma message( __FILE__ " : 'TBtreePath' record of last entry unused" )
|
sl@0
|
20 |
//#pragma message( __FILE__ " : 'TBtree' lots of opportunities for on-the-fly integrity checking" )
|
sl@0
|
21 |
|
sl@0
|
22 |
EXPORT_C void TBtreeToken::ExternalizeL(RWriteStream& aStream) const
|
sl@0
|
23 |
/** Externalises a TBtreeToken object to a stream.
|
sl@0
|
24 |
|
sl@0
|
25 |
@param aStream Stream to which the object should be externalised. */
|
sl@0
|
26 |
{
|
sl@0
|
27 |
aStream<<iFirst;
|
sl@0
|
28 |
aStream<<iRoot;
|
sl@0
|
29 |
aStream.WriteInt8L(iHeight);
|
sl@0
|
30 |
}
|
sl@0
|
31 |
|
sl@0
|
32 |
EXPORT_C void TBtreeToken::InternalizeL(RReadStream& aStream)
|
sl@0
|
33 |
/** Internalises a TBtreeToken object from a stream.
|
sl@0
|
34 |
|
sl@0
|
35 |
@param aStream Stream from which the object should be internalised. */
|
sl@0
|
36 |
{
|
sl@0
|
37 |
aStream>>iFirst;
|
sl@0
|
38 |
aStream>>iRoot;
|
sl@0
|
39 |
TInt height=aStream.ReadInt8L();
|
sl@0
|
40 |
if (height<0||height>KMaxBtreeHeight)
|
sl@0
|
41 |
__LEAVE(KErrCorrupt);
|
sl@0
|
42 |
//
|
sl@0
|
43 |
iHeight=height;
|
sl@0
|
44 |
}
|
sl@0
|
45 |
|
sl@0
|
46 |
EXPORT_C void TBtreeToken::Clear()
|
sl@0
|
47 |
{
|
sl@0
|
48 |
iFirst=KNullPageRef;
|
sl@0
|
49 |
iRoot=KNullPageRef;
|
sl@0
|
50 |
iHeight=0;
|
sl@0
|
51 |
}
|
sl@0
|
52 |
|
sl@0
|
53 |
void TBtreePath::Push(TPageRef aRef,TInt aEntry)
|
sl@0
|
54 |
{
|
sl@0
|
55 |
__ASSERT_DEBUG(aEntry==TInt(TUint8(aEntry)),Panic(EBadEntryPos));
|
sl@0
|
56 |
TInt end=--iEnd;
|
sl@0
|
57 |
__ASSERT_DEBUG(end>=0,Panic(EPathUnderflow));
|
sl@0
|
58 |
iNodes[end]=aRef;
|
sl@0
|
59 |
iEntries[end]=TUint8(aEntry);
|
sl@0
|
60 |
}
|
sl@0
|
61 |
|
sl@0
|
62 |
void TBtreePath::GotoRoot(TBtreeHeight aHeight,TPageRef aRoot)
|
sl@0
|
63 |
//
|
sl@0
|
64 |
// Set the end of path to height and root-node given.
|
sl@0
|
65 |
//
|
sl@0
|
66 |
{
|
sl@0
|
67 |
__ASSERT_DEBUG(aHeight>0&&aHeight<=KMaxBtreeHeight,Panic(EBadTreeHeight));
|
sl@0
|
68 |
iEnd=--aHeight;
|
sl@0
|
69 |
iNodes[aHeight]=aRoot;
|
sl@0
|
70 |
iEntries[aHeight]=0;
|
sl@0
|
71 |
}
|
sl@0
|
72 |
|
sl@0
|
73 |
EXPORT_C TBtree::TBtree(TBtreeMode aMode)
|
sl@0
|
74 |
: iPages(NULL),iFirst(KNullPageRef),iRoot(KNullPageRef),iHeight(0),iStatus(aMode)
|
sl@0
|
75 |
/** Constructor that sets the B-tree mode.
|
sl@0
|
76 |
|
sl@0
|
77 |
@param aMode B-tree operating mode */
|
sl@0
|
78 |
{}
|
sl@0
|
79 |
|
sl@0
|
80 |
EXPORT_C TBtree::TBtree(const TBtreeToken& aToken,TBtreeMode aMode)
|
sl@0
|
81 |
: iPages(NULL)
|
sl@0
|
82 |
/** Constructor that sets the B-tree mode and initialisation parameters.
|
sl@0
|
83 |
|
sl@0
|
84 |
@param aToken Parameters with which to initialise B-tree
|
sl@0
|
85 |
@param aMode B-tree operating mode */
|
sl@0
|
86 |
{
|
sl@0
|
87 |
Set(aToken,aMode);
|
sl@0
|
88 |
__ASSERT_DEBUG(iHeight>=0&&iHeight<=KMaxBtreeHeight,Panic(EBadTreeHeight));
|
sl@0
|
89 |
}
|
sl@0
|
90 |
|
sl@0
|
91 |
EXPORT_C void TBtree::Connect(MPagePool* aPool,const MBtreeKey* aKey,const MBtreeLeafOrg* aLeafOrg,const MBtreeIndexOrg* anIndexOrg)
|
sl@0
|
92 |
{
|
sl@0
|
93 |
iPages=aPool;
|
sl@0
|
94 |
iKey=aKey;
|
sl@0
|
95 |
iLeafOrg=aLeafOrg;
|
sl@0
|
96 |
iIndexOrg=anIndexOrg;
|
sl@0
|
97 |
}
|
sl@0
|
98 |
|
sl@0
|
99 |
EXPORT_C void TBtree::Set(const TBtreeToken& aToken,TBtreeMode aMode)
|
sl@0
|
100 |
/** Initialises the B-tree.
|
sl@0
|
101 |
|
sl@0
|
102 |
@param aToken Parameters with which to initialise the B-tree
|
sl@0
|
103 |
@param aMode B-tree operating mode */
|
sl@0
|
104 |
{
|
sl@0
|
105 |
iFirst=aToken.iFirst;
|
sl@0
|
106 |
iRoot=aToken.iRoot;
|
sl@0
|
107 |
iHeight=aToken.iHeight;
|
sl@0
|
108 |
iStatus=aToken.IsBroken()?aMode|EBroken:aMode;
|
sl@0
|
109 |
__ASSERT_DEBUG(iHeight>=0&&iHeight<=KMaxBtreeHeight,Panic(EBadTreeHeight));
|
sl@0
|
110 |
}
|
sl@0
|
111 |
|
sl@0
|
112 |
EXPORT_C TBtreeToken TBtree::Token() const
|
sl@0
|
113 |
/** Gets an object that encapsulates the TBtree settings.
|
sl@0
|
114 |
|
sl@0
|
115 |
That object can then be used to externalise the settings.
|
sl@0
|
116 |
|
sl@0
|
117 |
@return Encapsulates the TBtree settings. */
|
sl@0
|
118 |
{
|
sl@0
|
119 |
return TBtreeToken(iFirst,iRoot,IsIntact()?iHeight:0);
|
sl@0
|
120 |
}
|
sl@0
|
121 |
|
sl@0
|
122 |
EXPORT_C TBool TBtree::FirstL(TBtreePos& aPos) const
|
sl@0
|
123 |
/** Goes to the first entry in the B-tree.
|
sl@0
|
124 |
|
sl@0
|
125 |
@param aPos On return, the position of the first entry
|
sl@0
|
126 |
@return True if there is a first entry, otherwise false */
|
sl@0
|
127 |
{
|
sl@0
|
128 |
__ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
|
sl@0
|
129 |
CheckIntactL();
|
sl@0
|
130 |
if (IsEmpty())
|
sl@0
|
131 |
return EFalse;
|
sl@0
|
132 |
//
|
sl@0
|
133 |
TBtreePath& path=aPos.iPath;
|
sl@0
|
134 |
GotoRoot(path);
|
sl@0
|
135 |
DescendFirstL(path);
|
sl@0
|
136 |
return ETrue;
|
sl@0
|
137 |
}
|
sl@0
|
138 |
|
sl@0
|
139 |
EXPORT_C TBool TBtree::LastL(TBtreePos& aPos) const
|
sl@0
|
140 |
/** Goes to the last entry in the B-tree.
|
sl@0
|
141 |
|
sl@0
|
142 |
@param aPos On return, the position of the last entry
|
sl@0
|
143 |
@return True if there are any entries, otherwise false */
|
sl@0
|
144 |
{
|
sl@0
|
145 |
__ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
|
sl@0
|
146 |
CheckIntactL();
|
sl@0
|
147 |
if (IsEmpty())
|
sl@0
|
148 |
return EFalse;
|
sl@0
|
149 |
//
|
sl@0
|
150 |
TBtreePath& path=aPos.iPath;
|
sl@0
|
151 |
GotoRoot(path);
|
sl@0
|
152 |
DescendLastL(path);
|
sl@0
|
153 |
return ETrue;
|
sl@0
|
154 |
}
|
sl@0
|
155 |
|
sl@0
|
156 |
EXPORT_C TBool TBtree::NextL(TBtreePos& aPos) const
|
sl@0
|
157 |
/** Gets the position of the entry following the specified entry.
|
sl@0
|
158 |
|
sl@0
|
159 |
@param aPos On return, the position of the next entry
|
sl@0
|
160 |
@return True if there are any more entries, otherwise false */
|
sl@0
|
161 |
{
|
sl@0
|
162 |
__ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
|
sl@0
|
163 |
TBtreePath& path=aPos.iPath;
|
sl@0
|
164 |
__ASSERT_ALWAYS(path.IsLeaf(),Panic(EInvalidTreePos));
|
sl@0
|
165 |
CheckIntactL();
|
sl@0
|
166 |
if (IsEmpty()||!AscendAtLastL(path))
|
sl@0
|
167 |
return EFalse;
|
sl@0
|
168 |
//
|
sl@0
|
169 |
DescendFirstL(path);
|
sl@0
|
170 |
return ETrue;
|
sl@0
|
171 |
}
|
sl@0
|
172 |
|
sl@0
|
173 |
EXPORT_C TBool TBtree::PreviousL(TBtreePos& aPos) const
|
sl@0
|
174 |
/** Gets the position of the entry preceding the specified entry.
|
sl@0
|
175 |
|
sl@0
|
176 |
@param aPos On return, the position of the preceding entry
|
sl@0
|
177 |
@return True if there is a preceding entry, otherwise false */
|
sl@0
|
178 |
{
|
sl@0
|
179 |
__ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
|
sl@0
|
180 |
TBtreePath& path=aPos.iPath;
|
sl@0
|
181 |
__ASSERT_ALWAYS(path.IsLeaf(),Panic(EInvalidTreePos));
|
sl@0
|
182 |
CheckIntactL();
|
sl@0
|
183 |
if (IsEmpty()||!AscendAtFirstL(path))
|
sl@0
|
184 |
return EFalse;
|
sl@0
|
185 |
//
|
sl@0
|
186 |
if (!path.IsLeaf())
|
sl@0
|
187 |
{
|
sl@0
|
188 |
path.Push(ChildNodeL(path));
|
sl@0
|
189 |
DescendLastL(path);
|
sl@0
|
190 |
}
|
sl@0
|
191 |
return ETrue;
|
sl@0
|
192 |
}
|
sl@0
|
193 |
|
sl@0
|
194 |
EXPORT_C void TBtree::ClearL()
|
sl@0
|
195 |
/** Resets the B-tree to have zero-height, and a null root, and deletes all index
|
sl@0
|
196 |
pages. */
|
sl@0
|
197 |
{
|
sl@0
|
198 |
__ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
|
sl@0
|
199 |
if (IsEmpty())
|
sl@0
|
200 |
return;
|
sl@0
|
201 |
BeginUpdateL();
|
sl@0
|
202 |
DeleteIndexSetL();
|
sl@0
|
203 |
while (iFirst!=KNullPageRef)
|
sl@0
|
204 |
{
|
sl@0
|
205 |
TPageRef link=iLeafOrg->LinkNode(iPages->LockL(iFirst));
|
sl@0
|
206 |
iPages->DeleteL(iFirst);
|
sl@0
|
207 |
iFirst=link;
|
sl@0
|
208 |
}
|
sl@0
|
209 |
EndUpdate();
|
sl@0
|
210 |
}
|
sl@0
|
211 |
|
sl@0
|
212 |
EXPORT_C TInt TBtree::RepairL()
|
sl@0
|
213 |
/** Attempts to repair a broken B-tree.
|
sl@0
|
214 |
|
sl@0
|
215 |
If the operating mode (TBtreeMode) is set to EBtreeSecure, then the tree can
|
sl@0
|
216 |
be repaired without any loss of data. Otherwise, the tree must be deleted
|
sl@0
|
217 |
entirely using ClearL(), and reconstructed using other means.
|
sl@0
|
218 |
|
sl@0
|
219 |
Note that if multiple B-trees share a single store page pool, then reclaiming
|
sl@0
|
220 |
the pool will break all the other B-trees (but the Broken flag will not be
|
sl@0
|
221 |
set automatically). These trees can be repaired by calling MarkBroken() and
|
sl@0
|
222 |
then RepairL(), even if they were not of the EBtreeSecure type.
|
sl@0
|
223 |
|
sl@0
|
224 |
@return Number of leaves in the tree
|
sl@0
|
225 |
@see TBtreeMode */
|
sl@0
|
226 |
{
|
sl@0
|
227 |
__ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
|
sl@0
|
228 |
if (IsEmpty())
|
sl@0
|
229 |
return 0;
|
sl@0
|
230 |
BeginUpdateL();
|
sl@0
|
231 |
DeleteIndexSetL();
|
sl@0
|
232 |
// create the new index set, insert pivots on the end of the index set
|
sl@0
|
233 |
iHeight=1;
|
sl@0
|
234 |
iRoot=iFirst;
|
sl@0
|
235 |
TBtreePath path;
|
sl@0
|
236 |
TBtreePivot pivot;
|
sl@0
|
237 |
TInt count=0;
|
sl@0
|
238 |
TAny* seq=iPages->LockL(iFirst);
|
sl@0
|
239 |
for (;;)
|
sl@0
|
240 |
{
|
sl@0
|
241 |
count+=iLeafOrg->LastEntry(seq);
|
sl@0
|
242 |
TPageRef link=iLeafOrg->LinkNode(seq);
|
sl@0
|
243 |
if (link==KNullPageRef)
|
sl@0
|
244 |
break;
|
sl@0
|
245 |
TAny* next=iPages->LockL(link);
|
sl@0
|
246 |
NewPivot(seq,next,pivot);
|
sl@0
|
247 |
iPages->Unlock(seq);
|
sl@0
|
248 |
seq=next;
|
sl@0
|
249 |
if (iHeight==1)
|
sl@0
|
250 |
{ // first insertion, create a new index set
|
sl@0
|
251 |
iRoot=iFirst;
|
sl@0
|
252 |
NewRootL();
|
sl@0
|
253 |
GotoRoot(path);
|
sl@0
|
254 |
}
|
sl@0
|
255 |
TBtreeHeight end=path.End();
|
sl@0
|
256 |
InsertAtL(path,pivot,link);
|
sl@0
|
257 |
if (path.End()!=end)
|
sl@0
|
258 |
{ // path has been messed up by insertion, set path to last index entry
|
sl@0
|
259 |
GotoRoot(path);
|
sl@0
|
260 |
do path.Push(LastChildNodeL(path)); while (!path.IsLeaf());
|
sl@0
|
261 |
path.Pop();
|
sl@0
|
262 |
}
|
sl@0
|
263 |
else
|
sl@0
|
264 |
path.SetEntry(path.Entry()+1);
|
sl@0
|
265 |
}
|
sl@0
|
266 |
iPages->Unlock(seq);
|
sl@0
|
267 |
EndUpdate();
|
sl@0
|
268 |
return count;
|
sl@0
|
269 |
}
|
sl@0
|
270 |
|
sl@0
|
271 |
EXPORT_C TBool TBtree::FindL(TBtreePos& aPos,const TAny* aKey,TFind aMode) const
|
sl@0
|
272 |
/** Searches for an entry and returns its position.
|
sl@0
|
273 |
|
sl@0
|
274 |
@param aPos On return, the position of the entry found
|
sl@0
|
275 |
@param aKey Pointer to the key of the entry for which to search
|
sl@0
|
276 |
@param aMode Type of search to perform
|
sl@0
|
277 |
@return True if search was successful, else false */
|
sl@0
|
278 |
{
|
sl@0
|
279 |
__ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
|
sl@0
|
280 |
CheckIntactL();
|
sl@0
|
281 |
if (IsEmpty())
|
sl@0
|
282 |
return EFalse;
|
sl@0
|
283 |
TBtreePath& path=aPos.iPath;
|
sl@0
|
284 |
TBool match=SearchL(path,aKey,aMode==ELessEqual||aMode==EGreaterThan);
|
sl@0
|
285 |
switch (aMode)
|
sl@0
|
286 |
{
|
sl@0
|
287 |
case EGreaterThan:
|
sl@0
|
288 |
case EGreaterEqual: // move on if at end of node
|
sl@0
|
289 |
if (LastEntryL(path)==path.Entry())
|
sl@0
|
290 |
return NextL(aPos);
|
sl@0
|
291 |
return ETrue;
|
sl@0
|
292 |
case ELessThan:
|
sl@0
|
293 |
case ELessEqual:
|
sl@0
|
294 |
return PreviousL(aPos);
|
sl@0
|
295 |
case EEqualTo:
|
sl@0
|
296 |
break;
|
sl@0
|
297 |
}
|
sl@0
|
298 |
return match;
|
sl@0
|
299 |
}
|
sl@0
|
300 |
|
sl@0
|
301 |
EXPORT_C TBool TBtree::InsertL(TBtreePos& aPos,const TAny* anEntry,TInt aLength,TAllowDuplicates aDup)
|
sl@0
|
302 |
/** Inserts an entry into the tree.
|
sl@0
|
303 |
|
sl@0
|
304 |
@param aPos On return, the position of the entry inserted
|
sl@0
|
305 |
@param anEntry Pointer to the entry to insert
|
sl@0
|
306 |
@param aLength Length of entry
|
sl@0
|
307 |
@param aDup Flag to indicate whether duplicate entries are allowed in the tree
|
sl@0
|
308 |
@return True if successful, false if the entry was a duplicate and aDup was
|
sl@0
|
309 |
set to ENoDuplicates */
|
sl@0
|
310 |
{
|
sl@0
|
311 |
__ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
|
sl@0
|
312 |
CheckIntactL();
|
sl@0
|
313 |
TBtreePath& path=aPos.iPath;
|
sl@0
|
314 |
if (IsEmpty())
|
sl@0
|
315 |
{
|
sl@0
|
316 |
NewTreeL();
|
sl@0
|
317 |
GotoRoot(path);
|
sl@0
|
318 |
}
|
sl@0
|
319 |
else
|
sl@0
|
320 |
{
|
sl@0
|
321 |
if (SearchL(path,iKey->Key(anEntry))&&aDup==ENoDuplicates)
|
sl@0
|
322 |
return EFalse; // oops a duplicate
|
sl@0
|
323 |
}
|
sl@0
|
324 |
InsertAtL(path,TPtrC8((const TUint8*)anEntry,aLength));
|
sl@0
|
325 |
return ETrue;
|
sl@0
|
326 |
}
|
sl@0
|
327 |
|
sl@0
|
328 |
EXPORT_C TBool TBtree::DeleteL(const TAny* aKey)
|
sl@0
|
329 |
/** Deletes an entry.
|
sl@0
|
330 |
|
sl@0
|
331 |
@param aKey Pointer to the key of the entry to delete
|
sl@0
|
332 |
@return True if successful, false if the entry was not found */
|
sl@0
|
333 |
{
|
sl@0
|
334 |
__ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
|
sl@0
|
335 |
CheckIntactL();
|
sl@0
|
336 |
TBtreePath path;
|
sl@0
|
337 |
if (IsEmpty()||SearchL(path,aKey)==0)
|
sl@0
|
338 |
return EFalse; // not found
|
sl@0
|
339 |
DeleteAtL(path);
|
sl@0
|
340 |
return ETrue;
|
sl@0
|
341 |
}
|
sl@0
|
342 |
|
sl@0
|
343 |
EXPORT_C void TBtree::DeleteAtL(TBtreePos& aPos)
|
sl@0
|
344 |
/** Deletes the entry at the specified position
|
sl@0
|
345 |
|
sl@0
|
346 |
@param aPos Position of the entry to delete */
|
sl@0
|
347 |
{
|
sl@0
|
348 |
__ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
|
sl@0
|
349 |
TBtreePath& path=aPos.iPath;
|
sl@0
|
350 |
__ASSERT_ALWAYS(path.IsLeaf(),Panic(EInvalidTreePos));
|
sl@0
|
351 |
CheckIntactL();
|
sl@0
|
352 |
DeleteAtL(path);
|
sl@0
|
353 |
}
|
sl@0
|
354 |
|
sl@0
|
355 |
EXPORT_C void TBtree::ExtractAtL(const TBtreePos& aPos,TAny* anEntry,TInt aMaxLength) const
|
sl@0
|
356 |
/** Gets the entry at the specified position.
|
sl@0
|
357 |
|
sl@0
|
358 |
@param aPos Position of the entry to get
|
sl@0
|
359 |
@param anEntry Buffer into which to copy the entry
|
sl@0
|
360 |
@param aMaxLength Length of the anEntry buffer. If the entry is longer than
|
sl@0
|
361 |
this, only the first aMaxLength bytes will be copied. */
|
sl@0
|
362 |
{
|
sl@0
|
363 |
__ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
|
sl@0
|
364 |
const TBtreePath& path=aPos.iPath;
|
sl@0
|
365 |
__ASSERT_ALWAYS(path.IsLeaf(),Panic(EInvalidTreePos));
|
sl@0
|
366 |
CheckIntactL();
|
sl@0
|
367 |
TAny* pp=GetNodeL(path);
|
sl@0
|
368 |
TPtrC8 entry=iLeafOrg->Entry(pp,path.Entry());
|
sl@0
|
369 |
Mem::Copy((TUint8*)anEntry,entry.Ptr(),Min(aMaxLength,entry.Size()));
|
sl@0
|
370 |
iPages->Unlock(pp);
|
sl@0
|
371 |
}
|
sl@0
|
372 |
|
sl@0
|
373 |
EXPORT_C TBool TBtree::ResetL(TBtreeMark& aMark) const
|
sl@0
|
374 |
/** Resets an iterator to the root of the tree.
|
sl@0
|
375 |
|
sl@0
|
376 |
@param aMark Iterator to set
|
sl@0
|
377 |
@return True if successful, false if the B-tree is empty */
|
sl@0
|
378 |
{
|
sl@0
|
379 |
__ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
|
sl@0
|
380 |
if (IsEmpty())
|
sl@0
|
381 |
return EFalse;
|
sl@0
|
382 |
//
|
sl@0
|
383 |
TAny* first=iPages->LockL(iFirst);
|
sl@0
|
384 |
aMark.iLeaf=iFirst;
|
sl@0
|
385 |
aMark.iLink=iLeafOrg->LinkNode(first);
|
sl@0
|
386 |
aMark.iEntry=0;
|
sl@0
|
387 |
aMark.iLast=TUint8(iLeafOrg->LastEntry(first));
|
sl@0
|
388 |
iPages->Unlock(first);
|
sl@0
|
389 |
return ETrue;
|
sl@0
|
390 |
}
|
sl@0
|
391 |
|
sl@0
|
392 |
EXPORT_C TBool TBtree::NextL(TBtreeMark& aMark) const
|
sl@0
|
393 |
/** Advances an iterator to the next entry.
|
sl@0
|
394 |
|
sl@0
|
395 |
@param aMark Iterator to use. On return, the iterator is advanced to the next
|
sl@0
|
396 |
entry.
|
sl@0
|
397 |
@return True if successful, false if there is no next entry */
|
sl@0
|
398 |
{
|
sl@0
|
399 |
__ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
|
sl@0
|
400 |
if (IsEmpty())
|
sl@0
|
401 |
return EFalse;
|
sl@0
|
402 |
//
|
sl@0
|
403 |
++aMark.iEntry;
|
sl@0
|
404 |
if (aMark.iEntry<aMark.iLast)
|
sl@0
|
405 |
return ETrue;
|
sl@0
|
406 |
//
|
sl@0
|
407 |
if (aMark.iLink!=KNullPageRef)
|
sl@0
|
408 |
{
|
sl@0
|
409 |
TAny* next=iPages->LockL(aMark.iLink);
|
sl@0
|
410 |
aMark.iLeaf=aMark.iLink;
|
sl@0
|
411 |
aMark.iLink=iLeafOrg->LinkNode(next);
|
sl@0
|
412 |
aMark.iEntry=0;
|
sl@0
|
413 |
aMark.iLast=TUint8(iLeafOrg->LastEntry(next));
|
sl@0
|
414 |
iPages->Unlock(next);
|
sl@0
|
415 |
return ETrue;
|
sl@0
|
416 |
}
|
sl@0
|
417 |
//
|
sl@0
|
418 |
return EFalse;
|
sl@0
|
419 |
}
|
sl@0
|
420 |
|
sl@0
|
421 |
EXPORT_C void TBtree::ExtractAtL(const TBtreeMark& aMark,TAny* anEntry,TInt aMaxLength) const
|
sl@0
|
422 |
/** Gets an entry at specified iterator position.
|
sl@0
|
423 |
|
sl@0
|
424 |
@param aMark Position of the entry to get
|
sl@0
|
425 |
@param anEntry Buffer into which to copy the entry
|
sl@0
|
426 |
@param aMaxLength Length of anEntry buffer. If the entry is longer than this,
|
sl@0
|
427 |
only the first aMaxLength bytes are copied. */
|
sl@0
|
428 |
{
|
sl@0
|
429 |
__ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
|
sl@0
|
430 |
TAny* pp=iPages->LockL(aMark.iLeaf);
|
sl@0
|
431 |
TPtrC8 entry=iLeafOrg->Entry(pp,aMark.iEntry);
|
sl@0
|
432 |
Mem::Copy((TUint8*)anEntry,entry.Ptr(),Min(aMaxLength,entry.Size()));
|
sl@0
|
433 |
iPages->Unlock(pp);
|
sl@0
|
434 |
}
|
sl@0
|
435 |
|
sl@0
|
436 |
TBool TBtree::AscendAtLastL(TBtreePath& aPath) const
|
sl@0
|
437 |
//
|
sl@0
|
438 |
// While aPath is at the end of its node, ascend.
|
sl@0
|
439 |
// If not at end, then move right and return true. Return false if we've run out of tree.
|
sl@0
|
440 |
//
|
sl@0
|
441 |
{
|
sl@0
|
442 |
for (;;)
|
sl@0
|
443 |
{
|
sl@0
|
444 |
TInt last=LastEntryL(aPath);
|
sl@0
|
445 |
if (aPath.IsLeaf())
|
sl@0
|
446 |
--last;
|
sl@0
|
447 |
TInt entry=aPath.Entry();
|
sl@0
|
448 |
if (entry<last)
|
sl@0
|
449 |
{
|
sl@0
|
450 |
aPath.SetEntry(entry+1);
|
sl@0
|
451 |
return ETrue; // have a next entry at this level
|
sl@0
|
452 |
}
|
sl@0
|
453 |
if (IsRoot(aPath))
|
sl@0
|
454 |
return EFalse; // end of tree
|
sl@0
|
455 |
aPath.Pop();
|
sl@0
|
456 |
}
|
sl@0
|
457 |
}
|
sl@0
|
458 |
|
sl@0
|
459 |
TBool TBtree::AscendAtFirstL(TBtreePath& aPath) const
|
sl@0
|
460 |
//
|
sl@0
|
461 |
// While aPath is at the beginning of its node, ascend.
|
sl@0
|
462 |
// If not at start, then move left and return true. Return false if we've run out of tree.
|
sl@0
|
463 |
//
|
sl@0
|
464 |
{
|
sl@0
|
465 |
for (;;)
|
sl@0
|
466 |
{
|
sl@0
|
467 |
TInt entry=aPath.Entry();
|
sl@0
|
468 |
if (entry>0)
|
sl@0
|
469 |
{
|
sl@0
|
470 |
aPath.SetEntry(entry-1);
|
sl@0
|
471 |
return ETrue; // have a previous entry at this level
|
sl@0
|
472 |
}
|
sl@0
|
473 |
if (IsRoot(aPath))
|
sl@0
|
474 |
return EFalse; // beginning of sequence
|
sl@0
|
475 |
aPath.Pop();
|
sl@0
|
476 |
}
|
sl@0
|
477 |
}
|
sl@0
|
478 |
|
sl@0
|
479 |
void TBtree::DescendFirstL(TBtreePath& aPath) const
|
sl@0
|
480 |
//
|
sl@0
|
481 |
// Descend the tree to next entry on the bottom level.
|
sl@0
|
482 |
//
|
sl@0
|
483 |
{
|
sl@0
|
484 |
while (!aPath.IsLeaf())
|
sl@0
|
485 |
aPath.Push(ChildNodeL(aPath));
|
sl@0
|
486 |
}
|
sl@0
|
487 |
|
sl@0
|
488 |
void TBtree::DescendLastL(TBtreePath& aPath) const
|
sl@0
|
489 |
//
|
sl@0
|
490 |
// Descend the tree to the last entry on the bottom level.
|
sl@0
|
491 |
//
|
sl@0
|
492 |
{
|
sl@0
|
493 |
while (!aPath.IsLeaf())
|
sl@0
|
494 |
aPath.Push(LastChildNodeL(aPath));
|
sl@0
|
495 |
aPath.SetEntry(LastEntryL(aPath)-1);
|
sl@0
|
496 |
}
|
sl@0
|
497 |
|
sl@0
|
498 |
TBool TBtree::SearchL(TBtreePath& aPath,const TAny* aKey,TBool aAfter) const
|
sl@0
|
499 |
//
|
sl@0
|
500 |
// Search the B-tree for an entry. Return true if an exact match is found at base level.
|
sl@0
|
501 |
//
|
sl@0
|
502 |
{
|
sl@0
|
503 |
__ASSERT_DEBUG(!IsEmpty(),Panic(ENoTree));
|
sl@0
|
504 |
TInt entry;
|
sl@0
|
505 |
GotoRoot(aPath);
|
sl@0
|
506 |
while (!aPath.IsLeaf())
|
sl@0
|
507 |
{
|
sl@0
|
508 |
TAny* pp=GetNodeL(aPath);
|
sl@0
|
509 |
iIndexOrg->Search(pp,aKey,*iKey,aAfter,entry);
|
sl@0
|
510 |
aPath.SetEntry(entry);
|
sl@0
|
511 |
aPath.Push(iIndexOrg->ChildNode(pp,entry));
|
sl@0
|
512 |
iPages->Unlock(pp);
|
sl@0
|
513 |
}
|
sl@0
|
514 |
TAny* pp=GetNodeL(aPath);
|
sl@0
|
515 |
TBool found=iLeafOrg->Search(pp,aKey,*iKey,aAfter,entry);
|
sl@0
|
516 |
aPath.SetEntry(entry);
|
sl@0
|
517 |
iPages->Unlock(pp);
|
sl@0
|
518 |
return found;
|
sl@0
|
519 |
}
|
sl@0
|
520 |
|
sl@0
|
521 |
void TBtree::InsertAtL(TBtreePath& aPath,const TDesC8& anEntry)
|
sl@0
|
522 |
//
|
sl@0
|
523 |
// Insert anEntry at the current entry in aPath.
|
sl@0
|
524 |
//
|
sl@0
|
525 |
{
|
sl@0
|
526 |
TBtreePivot pivot;
|
sl@0
|
527 |
TPageRef promote;
|
sl@0
|
528 |
BeginUpdateL();
|
sl@0
|
529 |
if (InsertAtL(aPath,anEntry,KNullPageRef,pivot,promote))
|
sl@0
|
530 |
InsertAtL(aPath,pivot,promote);
|
sl@0
|
531 |
EndUpdate();
|
sl@0
|
532 |
}
|
sl@0
|
533 |
|
sl@0
|
534 |
void TBtree::NewPivot(const TAny* aLeftNode,const TAny* aRightNode,TBtreePivot& aPivot) const
|
sl@0
|
535 |
//
|
sl@0
|
536 |
// Get the key organisation to generate a new pivot between the given leaf nodes.
|
sl@0
|
537 |
//
|
sl@0
|
538 |
{
|
sl@0
|
539 |
const TAny* left=iLeafOrg->EntryPtr(aLeftNode,iLeafOrg->LastEntry(aLeftNode)-1);
|
sl@0
|
540 |
const TAny* right=iLeafOrg->EntryPtr(aRightNode,0);
|
sl@0
|
541 |
iKey->Between(iKey->Key(left),iKey->Key(right),aPivot);
|
sl@0
|
542 |
__DEBUG(TInt cleft=iKey->Compare(iKey->Key(left),aPivot.Ptr()));
|
sl@0
|
543 |
__DEBUG(TInt cright=iKey->Compare(iKey->Key(right),aPivot.Ptr()));
|
sl@0
|
544 |
__ASSERT_DEBUG((cleft<=0&&cright>0)||(cleft==0&&cright==0),Panic(EIllegalPivot));
|
sl@0
|
545 |
}
|
sl@0
|
546 |
|
sl@0
|
547 |
void TBtree::ReplacePivotL(TBtreePath& aPath,TAny* aNode,TBtreePivot& aPivot)
|
sl@0
|
548 |
//
|
sl@0
|
549 |
// Replace current pivot entry in aPath with aPivot.
|
sl@0
|
550 |
//
|
sl@0
|
551 |
{
|
sl@0
|
552 |
TInt entry=aPath.Entry();
|
sl@0
|
553 |
// update the pivot in the parent
|
sl@0
|
554 |
if (iIndexOrg->Update(aNode,entry,aPivot))
|
sl@0
|
555 |
iPages->Unlock(aNode,EPageDirty);
|
sl@0
|
556 |
else
|
sl@0
|
557 |
{ // have to delete the old pivot and insert new one
|
sl@0
|
558 |
TPageRef child=iIndexOrg->ChildNode(aNode,entry+1);
|
sl@0
|
559 |
iIndexOrg->Delete(aNode,entry);
|
sl@0
|
560 |
iPages->Unlock(aNode,EPageDirty);
|
sl@0
|
561 |
InsertAtL(aPath,aPivot,child);
|
sl@0
|
562 |
}
|
sl@0
|
563 |
}
|
sl@0
|
564 |
|
sl@0
|
565 |
void TBtree::InsertAtL(TBtreePath& aPath,TBtreePivot& aPivot,TPageRef aChild)
|
sl@0
|
566 |
{
|
sl@0
|
567 |
TBtreePivot togglePivot;
|
sl@0
|
568 |
for (;;)
|
sl@0
|
569 |
{
|
sl@0
|
570 |
if (!InsertAtL(aPath,aPivot,aChild,togglePivot,aChild))
|
sl@0
|
571 |
break;
|
sl@0
|
572 |
if (!InsertAtL(aPath,togglePivot,aChild,aPivot,aChild))
|
sl@0
|
573 |
break;
|
sl@0
|
574 |
}
|
sl@0
|
575 |
}
|
sl@0
|
576 |
|
sl@0
|
577 |
TBool TBtree::InsertAtL(TBtreePath& aPath,const TDesC8& anEntry,TPageRef aChild,TBtreePivot& aPivot,TPageRef& aPromote)
|
sl@0
|
578 |
//
|
sl@0
|
579 |
// Insert entry on this level. Return requirement to propagate with aPromote, aChild and aPath ready.
|
sl@0
|
580 |
//
|
sl@0
|
581 |
{
|
sl@0
|
582 |
TBool leaf=aPath.IsLeaf();
|
sl@0
|
583 |
const MBtreeNodeOrg* nodeOrg=NodeOrg(leaf);
|
sl@0
|
584 |
TInt entry=aPath.Entry();
|
sl@0
|
585 |
TAny* pNode=GetNodeL(aPath);
|
sl@0
|
586 |
|
sl@0
|
587 |
if (leaf?iLeafOrg->Insert(pNode,entry,anEntry):iIndexOrg->Insert(pNode,entry,anEntry,aChild))
|
sl@0
|
588 |
{
|
sl@0
|
589 |
PutNode(pNode,leaf);
|
sl@0
|
590 |
return EFalse;
|
sl@0
|
591 |
}
|
sl@0
|
592 |
// whatever happens next, we will affect the index set
|
sl@0
|
593 |
MarkBroken();
|
sl@0
|
594 |
// attempt an overflow insertion
|
sl@0
|
595 |
if (!IsRoot(aPath))
|
sl@0
|
596 |
{ // we have an ancestor
|
sl@0
|
597 |
TPageRef node=aPath.Node(); // save this in case we cannot overflow
|
sl@0
|
598 |
aPath.Pop();
|
sl@0
|
599 |
TInt child=aPath.Entry();
|
sl@0
|
600 |
TAny* pParent=GetNodeL(aPath);
|
sl@0
|
601 |
TAny* pSibling;
|
sl@0
|
602 |
if (child<iIndexOrg->LastEntry(pParent))
|
sl@0
|
603 |
{ // try to overflow to the right
|
sl@0
|
604 |
pSibling=iPages->LockL(iIndexOrg->ChildNode(pParent,child+1));
|
sl@0
|
605 |
if (leaf)
|
sl@0
|
606 |
{
|
sl@0
|
607 |
if (iLeafOrg->InsertOverflow(pNode,pSibling,entry,ETrue,anEntry))
|
sl@0
|
608 |
{
|
sl@0
|
609 |
NewPivot(pNode,pSibling,aPivot);
|
sl@0
|
610 |
overflowOK: PutNode(pSibling,leaf);
|
sl@0
|
611 |
PutNode(pNode,leaf);
|
sl@0
|
612 |
aPath.SetEntry(child);
|
sl@0
|
613 |
ReplacePivotL(aPath,pParent,aPivot);
|
sl@0
|
614 |
return EFalse; // completed insertion
|
sl@0
|
615 |
}
|
sl@0
|
616 |
}
|
sl@0
|
617 |
else
|
sl@0
|
618 |
{
|
sl@0
|
619 |
const TPtrC8 pivot=iIndexOrg->Entry(pParent,child);
|
sl@0
|
620 |
if (iIndexOrg->InsertOverflow(pNode,pSibling,entry,ETrue,anEntry,aChild,pivot,aPivot))
|
sl@0
|
621 |
goto overflowOK;
|
sl@0
|
622 |
}
|
sl@0
|
623 |
iPages->Unlock(pSibling);
|
sl@0
|
624 |
}
|
sl@0
|
625 |
if (--child>=0)
|
sl@0
|
626 |
{
|
sl@0
|
627 |
pSibling=iPages->LockL(iIndexOrg->ChildNode(pParent,child));
|
sl@0
|
628 |
if (leaf)
|
sl@0
|
629 |
{
|
sl@0
|
630 |
if (iLeafOrg->InsertOverflow(pSibling,pNode,entry,EFalse,anEntry))
|
sl@0
|
631 |
{
|
sl@0
|
632 |
NewPivot(pSibling,pNode,aPivot);
|
sl@0
|
633 |
goto overflowOK;
|
sl@0
|
634 |
}
|
sl@0
|
635 |
}
|
sl@0
|
636 |
else
|
sl@0
|
637 |
{
|
sl@0
|
638 |
const TPtrC8 pivot=iIndexOrg->Entry(pParent,child);
|
sl@0
|
639 |
if (iIndexOrg->InsertOverflow(pSibling,pNode,entry,EFalse,anEntry,aChild,pivot,aPivot))
|
sl@0
|
640 |
goto overflowOK;
|
sl@0
|
641 |
}
|
sl@0
|
642 |
iPages->Unlock(pSibling);
|
sl@0
|
643 |
}
|
sl@0
|
644 |
// cannot overflow, so...
|
sl@0
|
645 |
iPages->Unlock(pParent);
|
sl@0
|
646 |
aPath.Push(node);
|
sl@0
|
647 |
}
|
sl@0
|
648 |
|
sl@0
|
649 |
// do a split insertion
|
sl@0
|
650 |
TAny* pSibling=iPages->AllocL();
|
sl@0
|
651 |
nodeOrg->Init(pSibling);
|
sl@0
|
652 |
if (leaf)
|
sl@0
|
653 |
{
|
sl@0
|
654 |
iLeafOrg->InsertSplit(pNode,pSibling,entry,anEntry);
|
sl@0
|
655 |
NewPivot(pNode,pSibling,aPivot);
|
sl@0
|
656 |
aPromote=iPages->AssignL(pSibling); // non-reclaimable, mustn't lose track of leaves
|
sl@0
|
657 |
iLeafOrg->SetLinkNode(pNode,aPromote); // set up sequencing
|
sl@0
|
658 |
iPages->Unlock(pNode,EPageUpdate);
|
sl@0
|
659 |
}
|
sl@0
|
660 |
else
|
sl@0
|
661 |
{
|
sl@0
|
662 |
iIndexOrg->InsertSplit(pNode,pSibling,entry,anEntry,aChild,aPivot);
|
sl@0
|
663 |
aPromote=iPages->AssignL(pSibling,EPageReclaimable);
|
sl@0
|
664 |
iPages->Unlock(pNode,EPageDirty);
|
sl@0
|
665 |
}
|
sl@0
|
666 |
// propagate insert up the tree
|
sl@0
|
667 |
if (IsRoot(aPath))
|
sl@0
|
668 |
{ // new root node needed
|
sl@0
|
669 |
NewRootL();
|
sl@0
|
670 |
GotoRoot(aPath);
|
sl@0
|
671 |
}
|
sl@0
|
672 |
else
|
sl@0
|
673 |
aPath.Pop();
|
sl@0
|
674 |
return ETrue;
|
sl@0
|
675 |
}
|
sl@0
|
676 |
|
sl@0
|
677 |
void TBtree::DeleteAtL(TBtreePath& aPath)
|
sl@0
|
678 |
//
|
sl@0
|
679 |
// Delete the current entry on the path
|
sl@0
|
680 |
//
|
sl@0
|
681 |
{
|
sl@0
|
682 |
__ASSERT_DEBUG(aPath.IsLeaf(),Panic(EInvalidTreePos));
|
sl@0
|
683 |
TBool leaf=ETrue;
|
sl@0
|
684 |
const MBtreeNodeOrg* nodeOrg=iLeafOrg;
|
sl@0
|
685 |
TAny* pNode=GetNodeL(aPath);
|
sl@0
|
686 |
BeginUpdateL();
|
sl@0
|
687 |
TBtreePivot newPivot;
|
sl@0
|
688 |
//
|
sl@0
|
689 |
for (;;)
|
sl@0
|
690 |
{
|
sl@0
|
691 |
TPageRef node=aPath.Node();
|
sl@0
|
692 |
TInt entry=aPath.Entry();
|
sl@0
|
693 |
if (nodeOrg->Delete(pNode,entry))
|
sl@0
|
694 |
{ // success, no underflow
|
sl@0
|
695 |
if (leaf&&iLeafOrg->LastEntry(pNode)==entry)
|
sl@0
|
696 |
{ // we deleted the last entry so... replace the pivot to ensure invariant
|
sl@0
|
697 |
TPageRef next=iLeafOrg->LinkNode(pNode);
|
sl@0
|
698 |
if (next!=KNullPageRef)
|
sl@0
|
699 |
{ // replace the pivot
|
sl@0
|
700 |
MarkBroken();
|
sl@0
|
701 |
TAny* pSibling=iPages->LockL(next);
|
sl@0
|
702 |
NewPivot(pNode,pSibling,newPivot);
|
sl@0
|
703 |
iPages->Unlock(pSibling);
|
sl@0
|
704 |
PutNode(pNode,leaf);
|
sl@0
|
705 |
// find dividing pivot which we know is there
|
sl@0
|
706 |
do aPath.Pop(); while (aPath.Entry()==LastEntryL(aPath));
|
sl@0
|
707 |
ReplacePivotL(aPath,GetNodeL(aPath),newPivot);
|
sl@0
|
708 |
break;
|
sl@0
|
709 |
}
|
sl@0
|
710 |
}
|
sl@0
|
711 |
PutNode(pNode,leaf);
|
sl@0
|
712 |
break;
|
sl@0
|
713 |
}
|
sl@0
|
714 |
if (IsRoot(aPath))
|
sl@0
|
715 |
{ // the root! If it has entries then we're done
|
sl@0
|
716 |
if (nodeOrg->LastEntry(pNode)>0)
|
sl@0
|
717 |
PutNode(pNode,leaf);
|
sl@0
|
718 |
else
|
sl@0
|
719 |
{
|
sl@0
|
720 |
if (--iHeight!=0) // get child as root
|
sl@0
|
721 |
iRoot=iIndexOrg->ChildNode(pNode,0);
|
sl@0
|
722 |
else // tree is empty
|
sl@0
|
723 |
iRoot=iFirst=KNullPageRef;
|
sl@0
|
724 |
iPages->DeleteL(node);
|
sl@0
|
725 |
}
|
sl@0
|
726 |
break;
|
sl@0
|
727 |
}
|
sl@0
|
728 |
// will definitely affect the index set
|
sl@0
|
729 |
MarkBroken();
|
sl@0
|
730 |
// block has underflowed.. must try to redistribute or concatenate
|
sl@0
|
731 |
aPath.Pop();
|
sl@0
|
732 |
TAny* pParent=GetNodeL(aPath);
|
sl@0
|
733 |
TAny* pSibling;
|
sl@0
|
734 |
entry=aPath.Entry();
|
sl@0
|
735 |
if (entry>0)
|
sl@0
|
736 |
{
|
sl@0
|
737 |
aPath.SetEntry(--entry);
|
sl@0
|
738 |
pSibling=iPages->LockL(iIndexOrg->ChildNode(pParent,entry)); // on left
|
sl@0
|
739 |
}
|
sl@0
|
740 |
else
|
sl@0
|
741 |
{ // There must be a sibling for non-root nodes!
|
sl@0
|
742 |
__ASSERT_DEBUG(iIndexOrg->LastEntry(pParent)>0,Panic(EBadEntryCount));
|
sl@0
|
743 |
pSibling=pNode;
|
sl@0
|
744 |
node=iIndexOrg->ChildNode(pParent,1); // on right of first child
|
sl@0
|
745 |
pNode=iPages->LockL(node);
|
sl@0
|
746 |
}
|
sl@0
|
747 |
// redistribute between nodes
|
sl@0
|
748 |
if (leaf)
|
sl@0
|
749 |
{
|
sl@0
|
750 |
if (iLeafOrg->Redistribute(pSibling,pNode))
|
sl@0
|
751 |
{
|
sl@0
|
752 |
NewPivot(pSibling,pNode,newPivot);
|
sl@0
|
753 |
redistributeOK: PutNode(pSibling,leaf);
|
sl@0
|
754 |
PutNode(pNode,leaf);
|
sl@0
|
755 |
ReplacePivotL(aPath,pParent,newPivot);
|
sl@0
|
756 |
break;
|
sl@0
|
757 |
}
|
sl@0
|
758 |
// failed so concatenate
|
sl@0
|
759 |
iLeafOrg->Concatenate(pSibling,pNode);
|
sl@0
|
760 |
iPages->UpdateL(pSibling); // must change it here and now: link target is being deleted
|
sl@0
|
761 |
}
|
sl@0
|
762 |
else
|
sl@0
|
763 |
{
|
sl@0
|
764 |
TPtrC8 pivot=iIndexOrg->Entry(pParent,entry);
|
sl@0
|
765 |
if (iIndexOrg->Redistribute(pSibling,pNode,pivot,newPivot))
|
sl@0
|
766 |
goto redistributeOK;
|
sl@0
|
767 |
// failed so concatenate
|
sl@0
|
768 |
iIndexOrg->Concatenate(pSibling,pNode,pivot);
|
sl@0
|
769 |
iPages->Unlock(pSibling,EPageDirty);
|
sl@0
|
770 |
}
|
sl@0
|
771 |
iPages->DeleteL(node);
|
sl@0
|
772 |
leaf=EFalse;
|
sl@0
|
773 |
nodeOrg=iIndexOrg;
|
sl@0
|
774 |
pNode=pParent;
|
sl@0
|
775 |
}
|
sl@0
|
776 |
EndUpdate();
|
sl@0
|
777 |
}
|
sl@0
|
778 |
|
sl@0
|
779 |
void TBtree::DeleteIndexSetL()
|
sl@0
|
780 |
//
|
sl@0
|
781 |
// Destroy the index set, handle broken state too
|
sl@0
|
782 |
//
|
sl@0
|
783 |
{
|
sl@0
|
784 |
__ASSERT_DEBUG(!IsEmpty(),User::Invariant());
|
sl@0
|
785 |
if (IsIntact())
|
sl@0
|
786 |
{
|
sl@0
|
787 |
TBtreePath path;
|
sl@0
|
788 |
GotoRoot(path);
|
sl@0
|
789 |
if (!path.IsLeaf())
|
sl@0
|
790 |
{
|
sl@0
|
791 |
MarkBroken();
|
sl@0
|
792 |
DeleteIndexNodesL(path,ETrue);
|
sl@0
|
793 |
}
|
sl@0
|
794 |
}
|
sl@0
|
795 |
// delete the leading edge
|
sl@0
|
796 |
while (iRoot!=iFirst)
|
sl@0
|
797 |
{
|
sl@0
|
798 |
TPageRef edge=iIndexOrg->ChildNode(iPages->LockL(iRoot),0);
|
sl@0
|
799 |
iPages->DeleteL(iRoot);
|
sl@0
|
800 |
iRoot=edge;
|
sl@0
|
801 |
}
|
sl@0
|
802 |
}
|
sl@0
|
803 |
|
sl@0
|
804 |
void TBtree::DeleteIndexNodesL(TBtreePath& aPath,TBool aLeadingEdge)
|
sl@0
|
805 |
//
|
sl@0
|
806 |
// Destroy the children of the node in aPath, then destroy the node itself.
|
sl@0
|
807 |
// do not destroy the sequence set or the leading edge
|
sl@0
|
808 |
//
|
sl@0
|
809 |
{
|
sl@0
|
810 |
if (aPath.End()>1)
|
sl@0
|
811 |
{ // erase children
|
sl@0
|
812 |
for (TInt ii=LastEntryL(aPath);ii>=0;--ii)
|
sl@0
|
813 |
{
|
sl@0
|
814 |
aPath.Push(ChildNodeL(aPath,ii));
|
sl@0
|
815 |
DeleteIndexNodesL(aPath,aLeadingEdge && ii==0);
|
sl@0
|
816 |
aPath.Pop();
|
sl@0
|
817 |
}
|
sl@0
|
818 |
}
|
sl@0
|
819 |
if (!aLeadingEdge)
|
sl@0
|
820 |
iPages->DeleteL(aPath.Node());
|
sl@0
|
821 |
}
|
sl@0
|
822 |
|
sl@0
|
823 |
void TBtree::NewTreeL()
|
sl@0
|
824 |
//
|
sl@0
|
825 |
// Construct the initial node on empty tree.
|
sl@0
|
826 |
//
|
sl@0
|
827 |
{
|
sl@0
|
828 |
TAny* first=iPages->AllocL();
|
sl@0
|
829 |
iLeafOrg->Init(first);
|
sl@0
|
830 |
iRoot=iFirst=iPages->AssignL(first); // non-reclaimable, it's a leaf as well as the initial root
|
sl@0
|
831 |
iHeight=1;
|
sl@0
|
832 |
}
|
sl@0
|
833 |
|
sl@0
|
834 |
void TBtree::NewRootL()
|
sl@0
|
835 |
//
|
sl@0
|
836 |
// Create a new root node and set its child to the current root.
|
sl@0
|
837 |
//
|
sl@0
|
838 |
{
|
sl@0
|
839 |
if (iHeight==KMaxBtreeHeight)
|
sl@0
|
840 |
__LEAVE(KErrOverflow); // tree is max height
|
sl@0
|
841 |
TAny* root=iPages->AllocL();
|
sl@0
|
842 |
iIndexOrg->Init(root);
|
sl@0
|
843 |
iIndexOrg->MakeRoot(root,iRoot);
|
sl@0
|
844 |
iRoot=iPages->AssignL(root); // non-reclaimable, mustn't lose track of successive roots
|
sl@0
|
845 |
++iHeight;
|
sl@0
|
846 |
}
|
sl@0
|
847 |
|
sl@0
|
848 |
void TBtree::BeginUpdateL()
|
sl@0
|
849 |
//
|
sl@0
|
850 |
// Prepare to update the tree. Mark it dirty and ensure locked pages are abandoned on failure.
|
sl@0
|
851 |
//
|
sl@0
|
852 |
{
|
sl@0
|
853 |
iPages->PushL();
|
sl@0
|
854 |
MarkDirty();
|
sl@0
|
855 |
}
|
sl@0
|
856 |
|
sl@0
|
857 |
void TBtree::EndUpdate()
|
sl@0
|
858 |
//
|
sl@0
|
859 |
// Update complete, clear the broken flag and discard page pool acquisition
|
sl@0
|
860 |
//
|
sl@0
|
861 |
{
|
sl@0
|
862 |
iPages->Pop();
|
sl@0
|
863 |
MarkIntact();
|
sl@0
|
864 |
}
|
sl@0
|
865 |
|
sl@0
|
866 |
void TBtree::CheckIntactL() const
|
sl@0
|
867 |
{
|
sl@0
|
868 |
if (IsBroken())
|
sl@0
|
869 |
__LEAVE(KErrCorrupt);
|
sl@0
|
870 |
}
|
sl@0
|
871 |
|
sl@0
|
872 |
void TBtree::GotoRoot(TBtreePath& aPath) const
|
sl@0
|
873 |
//
|
sl@0
|
874 |
// Set the path to the root of the tree.
|
sl@0
|
875 |
//
|
sl@0
|
876 |
{
|
sl@0
|
877 |
__ASSERT_DEBUG(!IsEmpty(),Panic(ENoTree));
|
sl@0
|
878 |
aPath.GotoRoot(iHeight,iRoot);
|
sl@0
|
879 |
}
|
sl@0
|
880 |
|
sl@0
|
881 |
TAny* TBtree::GetNodeL(const TBtreePath& aPath) const
|
sl@0
|
882 |
{
|
sl@0
|
883 |
return iPages->LockL(aPath.Node());
|
sl@0
|
884 |
}
|
sl@0
|
885 |
|
sl@0
|
886 |
void TBtree::PutNode(TAny* aNode,TBool aLeaf) const
|
sl@0
|
887 |
{
|
sl@0
|
888 |
iPages->Unlock(aNode,aLeaf&&(iStatus&ESecure)?EPageUpdate:EPageDirty);
|
sl@0
|
889 |
}
|
sl@0
|
890 |
|
sl@0
|
891 |
TInt TBtree::LastEntryL(const TBtreePath& aPath) const
|
sl@0
|
892 |
{
|
sl@0
|
893 |
//#pragma message( __FILE__ " : 'TBtree::LastEntryL()' should have this information without having to look at the tree" )
|
sl@0
|
894 |
TAny* pp=GetNodeL(aPath);
|
sl@0
|
895 |
TInt cc=NodeOrg(aPath.IsLeaf())->LastEntry(pp);
|
sl@0
|
896 |
iPages->Unlock(pp);
|
sl@0
|
897 |
return cc;
|
sl@0
|
898 |
}
|
sl@0
|
899 |
|
sl@0
|
900 |
TPageRef TBtree::ChildNodeL(const TBtreePath& aPath) const
|
sl@0
|
901 |
//
|
sl@0
|
902 |
// Get current child node.
|
sl@0
|
903 |
//
|
sl@0
|
904 |
{
|
sl@0
|
905 |
__ASSERT_DEBUG(!aPath.IsLeaf(),Panic(EInvalidTreePos));
|
sl@0
|
906 |
TAny* pp=GetNodeL(aPath);
|
sl@0
|
907 |
TPageRef pg=iIndexOrg->ChildNode(pp,aPath.Entry());
|
sl@0
|
908 |
iPages->Unlock(pp);
|
sl@0
|
909 |
return pg;
|
sl@0
|
910 |
}
|
sl@0
|
911 |
|
sl@0
|
912 |
TPageRef TBtree::ChildNodeL(TBtreePath& aPath,TInt aChild) const
|
sl@0
|
913 |
{
|
sl@0
|
914 |
aPath.SetEntry(aChild);
|
sl@0
|
915 |
return ChildNodeL(aPath);
|
sl@0
|
916 |
}
|
sl@0
|
917 |
|
sl@0
|
918 |
TPageRef TBtree::LastChildNodeL(TBtreePath& aPath) const
|
sl@0
|
919 |
{
|
sl@0
|
920 |
__ASSERT_DEBUG(!aPath.IsLeaf(),Panic(EInvalidTreePos));
|
sl@0
|
921 |
TAny* pp=GetNodeL(aPath);
|
sl@0
|
922 |
TInt entry=iIndexOrg->LastEntry(pp);
|
sl@0
|
923 |
aPath.SetEntry(entry);
|
sl@0
|
924 |
TPageRef pg=iIndexOrg->ChildNode(pp,entry);
|
sl@0
|
925 |
iPages->Unlock(pp);
|
sl@0
|
926 |
return pg;
|
sl@0
|
927 |
}
|
sl@0
|
928 |
|