Update contrib.
1 // Copyright (c) 1998-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 "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.
18 //#pragma message( __FILE__ " : 'TBtreePath' does not track insertion position" )
19 //#pragma message( __FILE__ " : 'TBtreePath' record of last entry unused" )
20 //#pragma message( __FILE__ " : 'TBtree' lots of opportunities for on-the-fly integrity checking" )
22 EXPORT_C void TBtreeToken::ExternalizeL(RWriteStream& aStream) const
23 /** Externalises a TBtreeToken object to a stream.
25 @param aStream Stream to which the object should be externalised. */
29 aStream.WriteInt8L(iHeight);
32 EXPORT_C void TBtreeToken::InternalizeL(RReadStream& aStream)
33 /** Internalises a TBtreeToken object from a stream.
35 @param aStream Stream from which the object should be internalised. */
39 TInt height=aStream.ReadInt8L();
40 if (height<0||height>KMaxBtreeHeight)
46 EXPORT_C void TBtreeToken::Clear()
53 void TBtreePath::Push(TPageRef aRef,TInt aEntry)
55 __ASSERT_DEBUG(aEntry==TInt(TUint8(aEntry)),Panic(EBadEntryPos));
57 __ASSERT_DEBUG(end>=0,Panic(EPathUnderflow));
59 iEntries[end]=TUint8(aEntry);
62 void TBtreePath::GotoRoot(TBtreeHeight aHeight,TPageRef aRoot)
64 // Set the end of path to height and root-node given.
67 __ASSERT_DEBUG(aHeight>0&&aHeight<=KMaxBtreeHeight,Panic(EBadTreeHeight));
69 iNodes[aHeight]=aRoot;
73 EXPORT_C TBtree::TBtree(TBtreeMode aMode)
74 : iPages(NULL),iFirst(KNullPageRef),iRoot(KNullPageRef),iHeight(0),iStatus(aMode)
75 /** Constructor that sets the B-tree mode.
77 @param aMode B-tree operating mode */
80 EXPORT_C TBtree::TBtree(const TBtreeToken& aToken,TBtreeMode aMode)
82 /** Constructor that sets the B-tree mode and initialisation parameters.
84 @param aToken Parameters with which to initialise B-tree
85 @param aMode B-tree operating mode */
88 __ASSERT_DEBUG(iHeight>=0&&iHeight<=KMaxBtreeHeight,Panic(EBadTreeHeight));
91 EXPORT_C void TBtree::Connect(MPagePool* aPool,const MBtreeKey* aKey,const MBtreeLeafOrg* aLeafOrg,const MBtreeIndexOrg* anIndexOrg)
99 EXPORT_C void TBtree::Set(const TBtreeToken& aToken,TBtreeMode aMode)
100 /** Initialises the B-tree.
102 @param aToken Parameters with which to initialise the B-tree
103 @param aMode B-tree operating mode */
105 iFirst=aToken.iFirst;
107 iHeight=aToken.iHeight;
108 iStatus=aToken.IsBroken()?aMode|EBroken:aMode;
109 __ASSERT_DEBUG(iHeight>=0&&iHeight<=KMaxBtreeHeight,Panic(EBadTreeHeight));
112 EXPORT_C TBtreeToken TBtree::Token() const
113 /** Gets an object that encapsulates the TBtree settings.
115 That object can then be used to externalise the settings.
117 @return Encapsulates the TBtree settings. */
119 return TBtreeToken(iFirst,iRoot,IsIntact()?iHeight:0);
122 EXPORT_C TBool TBtree::FirstL(TBtreePos& aPos) const
123 /** Goes to the first entry in the B-tree.
125 @param aPos On return, the position of the first entry
126 @return True if there is a first entry, otherwise false */
128 __ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
133 TBtreePath& path=aPos.iPath;
139 EXPORT_C TBool TBtree::LastL(TBtreePos& aPos) const
140 /** Goes to the last entry in the B-tree.
142 @param aPos On return, the position of the last entry
143 @return True if there are any entries, otherwise false */
145 __ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
150 TBtreePath& path=aPos.iPath;
156 EXPORT_C TBool TBtree::NextL(TBtreePos& aPos) const
157 /** Gets the position of the entry following the specified entry.
159 @param aPos On return, the position of the next entry
160 @return True if there are any more entries, otherwise false */
162 __ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
163 TBtreePath& path=aPos.iPath;
164 __ASSERT_ALWAYS(path.IsLeaf(),Panic(EInvalidTreePos));
166 if (IsEmpty()||!AscendAtLastL(path))
173 EXPORT_C TBool TBtree::PreviousL(TBtreePos& aPos) const
174 /** Gets the position of the entry preceding the specified entry.
176 @param aPos On return, the position of the preceding entry
177 @return True if there is a preceding entry, otherwise false */
179 __ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
180 TBtreePath& path=aPos.iPath;
181 __ASSERT_ALWAYS(path.IsLeaf(),Panic(EInvalidTreePos));
183 if (IsEmpty()||!AscendAtFirstL(path))
188 path.Push(ChildNodeL(path));
194 EXPORT_C void TBtree::ClearL()
195 /** Resets the B-tree to have zero-height, and a null root, and deletes all index
198 __ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
203 while (iFirst!=KNullPageRef)
205 TPageRef link=iLeafOrg->LinkNode(iPages->LockL(iFirst));
206 iPages->DeleteL(iFirst);
212 EXPORT_C TInt TBtree::RepairL()
213 /** Attempts to repair a broken B-tree.
215 If the operating mode (TBtreeMode) is set to EBtreeSecure, then the tree can
216 be repaired without any loss of data. Otherwise, the tree must be deleted
217 entirely using ClearL(), and reconstructed using other means.
219 Note that if multiple B-trees share a single store page pool, then reclaiming
220 the pool will break all the other B-trees (but the Broken flag will not be
221 set automatically). These trees can be repaired by calling MarkBroken() and
222 then RepairL(), even if they were not of the EBtreeSecure type.
224 @return Number of leaves in the tree
227 __ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
232 // create the new index set, insert pivots on the end of the index set
238 TAny* seq=iPages->LockL(iFirst);
241 count+=iLeafOrg->LastEntry(seq);
242 TPageRef link=iLeafOrg->LinkNode(seq);
243 if (link==KNullPageRef)
245 TAny* next=iPages->LockL(link);
246 NewPivot(seq,next,pivot);
250 { // first insertion, create a new index set
255 TBtreeHeight end=path.End();
256 InsertAtL(path,pivot,link);
258 { // path has been messed up by insertion, set path to last index entry
260 do path.Push(LastChildNodeL(path)); while (!path.IsLeaf());
264 path.SetEntry(path.Entry()+1);
271 EXPORT_C TBool TBtree::FindL(TBtreePos& aPos,const TAny* aKey,TFind aMode) const
272 /** Searches for an entry and returns its position.
274 @param aPos On return, the position of the entry found
275 @param aKey Pointer to the key of the entry for which to search
276 @param aMode Type of search to perform
277 @return True if search was successful, else false */
279 __ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
283 TBtreePath& path=aPos.iPath;
284 TBool match=SearchL(path,aKey,aMode==ELessEqual||aMode==EGreaterThan);
288 case EGreaterEqual: // move on if at end of node
289 if (LastEntryL(path)==path.Entry())
294 return PreviousL(aPos);
301 EXPORT_C TBool TBtree::InsertL(TBtreePos& aPos,const TAny* anEntry,TInt aLength,TAllowDuplicates aDup)
302 /** Inserts an entry into the tree.
304 @param aPos On return, the position of the entry inserted
305 @param anEntry Pointer to the entry to insert
306 @param aLength Length of entry
307 @param aDup Flag to indicate whether duplicate entries are allowed in the tree
308 @return True if successful, false if the entry was a duplicate and aDup was
309 set to ENoDuplicates */
311 __ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
313 TBtreePath& path=aPos.iPath;
321 if (SearchL(path,iKey->Key(anEntry))&&aDup==ENoDuplicates)
322 return EFalse; // oops a duplicate
324 InsertAtL(path,TPtrC8((const TUint8*)anEntry,aLength));
328 EXPORT_C TBool TBtree::DeleteL(const TAny* aKey)
329 /** Deletes an entry.
331 @param aKey Pointer to the key of the entry to delete
332 @return True if successful, false if the entry was not found */
334 __ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
337 if (IsEmpty()||SearchL(path,aKey)==0)
338 return EFalse; // not found
343 EXPORT_C void TBtree::DeleteAtL(TBtreePos& aPos)
344 /** Deletes the entry at the specified position
346 @param aPos Position of the entry to delete */
348 __ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
349 TBtreePath& path=aPos.iPath;
350 __ASSERT_ALWAYS(path.IsLeaf(),Panic(EInvalidTreePos));
355 EXPORT_C void TBtree::ExtractAtL(const TBtreePos& aPos,TAny* anEntry,TInt aMaxLength) const
356 /** Gets the entry at the specified position.
358 @param aPos Position of the entry to get
359 @param anEntry Buffer into which to copy the entry
360 @param aMaxLength Length of the anEntry buffer. If the entry is longer than
361 this, only the first aMaxLength bytes will be copied. */
363 __ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
364 const TBtreePath& path=aPos.iPath;
365 __ASSERT_ALWAYS(path.IsLeaf(),Panic(EInvalidTreePos));
367 TAny* pp=GetNodeL(path);
368 TPtrC8 entry=iLeafOrg->Entry(pp,path.Entry());
369 Mem::Copy((TUint8*)anEntry,entry.Ptr(),Min(aMaxLength,entry.Size()));
373 EXPORT_C TBool TBtree::ResetL(TBtreeMark& aMark) const
374 /** Resets an iterator to the root of the tree.
376 @param aMark Iterator to set
377 @return True if successful, false if the B-tree is empty */
379 __ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
383 TAny* first=iPages->LockL(iFirst);
385 aMark.iLink=iLeafOrg->LinkNode(first);
387 aMark.iLast=TUint8(iLeafOrg->LastEntry(first));
388 iPages->Unlock(first);
392 EXPORT_C TBool TBtree::NextL(TBtreeMark& aMark) const
393 /** Advances an iterator to the next entry.
395 @param aMark Iterator to use. On return, the iterator is advanced to the next
397 @return True if successful, false if there is no next entry */
399 __ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
404 if (aMark.iEntry<aMark.iLast)
407 if (aMark.iLink!=KNullPageRef)
409 TAny* next=iPages->LockL(aMark.iLink);
410 aMark.iLeaf=aMark.iLink;
411 aMark.iLink=iLeafOrg->LinkNode(next);
413 aMark.iLast=TUint8(iLeafOrg->LastEntry(next));
414 iPages->Unlock(next);
421 EXPORT_C void TBtree::ExtractAtL(const TBtreeMark& aMark,TAny* anEntry,TInt aMaxLength) const
422 /** Gets an entry at specified iterator position.
424 @param aMark Position of the entry to get
425 @param anEntry Buffer into which to copy the entry
426 @param aMaxLength Length of anEntry buffer. If the entry is longer than this,
427 only the first aMaxLength bytes are copied. */
429 __ASSERT_DEBUG(iPages!=0,Panic(ENoConnect));
430 TAny* pp=iPages->LockL(aMark.iLeaf);
431 TPtrC8 entry=iLeafOrg->Entry(pp,aMark.iEntry);
432 Mem::Copy((TUint8*)anEntry,entry.Ptr(),Min(aMaxLength,entry.Size()));
436 TBool TBtree::AscendAtLastL(TBtreePath& aPath) const
438 // While aPath is at the end of its node, ascend.
439 // If not at end, then move right and return true. Return false if we've run out of tree.
444 TInt last=LastEntryL(aPath);
447 TInt entry=aPath.Entry();
450 aPath.SetEntry(entry+1);
451 return ETrue; // have a next entry at this level
454 return EFalse; // end of tree
459 TBool TBtree::AscendAtFirstL(TBtreePath& aPath) const
461 // While aPath is at the beginning of its node, ascend.
462 // If not at start, then move left and return true. Return false if we've run out of tree.
467 TInt entry=aPath.Entry();
470 aPath.SetEntry(entry-1);
471 return ETrue; // have a previous entry at this level
474 return EFalse; // beginning of sequence
479 void TBtree::DescendFirstL(TBtreePath& aPath) const
481 // Descend the tree to next entry on the bottom level.
484 while (!aPath.IsLeaf())
485 aPath.Push(ChildNodeL(aPath));
488 void TBtree::DescendLastL(TBtreePath& aPath) const
490 // Descend the tree to the last entry on the bottom level.
493 while (!aPath.IsLeaf())
494 aPath.Push(LastChildNodeL(aPath));
495 aPath.SetEntry(LastEntryL(aPath)-1);
498 TBool TBtree::SearchL(TBtreePath& aPath,const TAny* aKey,TBool aAfter) const
500 // Search the B-tree for an entry. Return true if an exact match is found at base level.
503 __ASSERT_DEBUG(!IsEmpty(),Panic(ENoTree));
506 while (!aPath.IsLeaf())
508 TAny* pp=GetNodeL(aPath);
509 iIndexOrg->Search(pp,aKey,*iKey,aAfter,entry);
510 aPath.SetEntry(entry);
511 aPath.Push(iIndexOrg->ChildNode(pp,entry));
514 TAny* pp=GetNodeL(aPath);
515 TBool found=iLeafOrg->Search(pp,aKey,*iKey,aAfter,entry);
516 aPath.SetEntry(entry);
521 void TBtree::InsertAtL(TBtreePath& aPath,const TDesC8& anEntry)
523 // Insert anEntry at the current entry in aPath.
529 if (InsertAtL(aPath,anEntry,KNullPageRef,pivot,promote))
530 InsertAtL(aPath,pivot,promote);
534 void TBtree::NewPivot(const TAny* aLeftNode,const TAny* aRightNode,TBtreePivot& aPivot) const
536 // Get the key organisation to generate a new pivot between the given leaf nodes.
539 const TAny* left=iLeafOrg->EntryPtr(aLeftNode,iLeafOrg->LastEntry(aLeftNode)-1);
540 const TAny* right=iLeafOrg->EntryPtr(aRightNode,0);
541 iKey->Between(iKey->Key(left),iKey->Key(right),aPivot);
542 __DEBUG(TInt cleft=iKey->Compare(iKey->Key(left),aPivot.Ptr()));
543 __DEBUG(TInt cright=iKey->Compare(iKey->Key(right),aPivot.Ptr()));
544 __ASSERT_DEBUG((cleft<=0&&cright>0)||(cleft==0&&cright==0),Panic(EIllegalPivot));
547 void TBtree::ReplacePivotL(TBtreePath& aPath,TAny* aNode,TBtreePivot& aPivot)
549 // Replace current pivot entry in aPath with aPivot.
552 TInt entry=aPath.Entry();
553 // update the pivot in the parent
554 if (iIndexOrg->Update(aNode,entry,aPivot))
555 iPages->Unlock(aNode,EPageDirty);
557 { // have to delete the old pivot and insert new one
558 TPageRef child=iIndexOrg->ChildNode(aNode,entry+1);
559 iIndexOrg->Delete(aNode,entry);
560 iPages->Unlock(aNode,EPageDirty);
561 InsertAtL(aPath,aPivot,child);
565 void TBtree::InsertAtL(TBtreePath& aPath,TBtreePivot& aPivot,TPageRef aChild)
567 TBtreePivot togglePivot;
570 if (!InsertAtL(aPath,aPivot,aChild,togglePivot,aChild))
572 if (!InsertAtL(aPath,togglePivot,aChild,aPivot,aChild))
577 TBool TBtree::InsertAtL(TBtreePath& aPath,const TDesC8& anEntry,TPageRef aChild,TBtreePivot& aPivot,TPageRef& aPromote)
579 // Insert entry on this level. Return requirement to propagate with aPromote, aChild and aPath ready.
582 TBool leaf=aPath.IsLeaf();
583 const MBtreeNodeOrg* nodeOrg=NodeOrg(leaf);
584 TInt entry=aPath.Entry();
585 TAny* pNode=GetNodeL(aPath);
587 if (leaf?iLeafOrg->Insert(pNode,entry,anEntry):iIndexOrg->Insert(pNode,entry,anEntry,aChild))
592 // whatever happens next, we will affect the index set
594 // attempt an overflow insertion
596 { // we have an ancestor
597 TPageRef node=aPath.Node(); // save this in case we cannot overflow
599 TInt child=aPath.Entry();
600 TAny* pParent=GetNodeL(aPath);
602 if (child<iIndexOrg->LastEntry(pParent))
603 { // try to overflow to the right
604 pSibling=iPages->LockL(iIndexOrg->ChildNode(pParent,child+1));
607 if (iLeafOrg->InsertOverflow(pNode,pSibling,entry,ETrue,anEntry))
609 NewPivot(pNode,pSibling,aPivot);
610 overflowOK: PutNode(pSibling,leaf);
612 aPath.SetEntry(child);
613 ReplacePivotL(aPath,pParent,aPivot);
614 return EFalse; // completed insertion
619 const TPtrC8 pivot=iIndexOrg->Entry(pParent,child);
620 if (iIndexOrg->InsertOverflow(pNode,pSibling,entry,ETrue,anEntry,aChild,pivot,aPivot))
623 iPages->Unlock(pSibling);
627 pSibling=iPages->LockL(iIndexOrg->ChildNode(pParent,child));
630 if (iLeafOrg->InsertOverflow(pSibling,pNode,entry,EFalse,anEntry))
632 NewPivot(pSibling,pNode,aPivot);
638 const TPtrC8 pivot=iIndexOrg->Entry(pParent,child);
639 if (iIndexOrg->InsertOverflow(pSibling,pNode,entry,EFalse,anEntry,aChild,pivot,aPivot))
642 iPages->Unlock(pSibling);
644 // cannot overflow, so...
645 iPages->Unlock(pParent);
649 // do a split insertion
650 TAny* pSibling=iPages->AllocL();
651 nodeOrg->Init(pSibling);
654 iLeafOrg->InsertSplit(pNode,pSibling,entry,anEntry);
655 NewPivot(pNode,pSibling,aPivot);
656 aPromote=iPages->AssignL(pSibling); // non-reclaimable, mustn't lose track of leaves
657 iLeafOrg->SetLinkNode(pNode,aPromote); // set up sequencing
658 iPages->Unlock(pNode,EPageUpdate);
662 iIndexOrg->InsertSplit(pNode,pSibling,entry,anEntry,aChild,aPivot);
663 aPromote=iPages->AssignL(pSibling,EPageReclaimable);
664 iPages->Unlock(pNode,EPageDirty);
666 // propagate insert up the tree
668 { // new root node needed
677 void TBtree::DeleteAtL(TBtreePath& aPath)
679 // Delete the current entry on the path
682 __ASSERT_DEBUG(aPath.IsLeaf(),Panic(EInvalidTreePos));
684 const MBtreeNodeOrg* nodeOrg=iLeafOrg;
685 TAny* pNode=GetNodeL(aPath);
687 TBtreePivot newPivot;
691 TPageRef node=aPath.Node();
692 TInt entry=aPath.Entry();
693 if (nodeOrg->Delete(pNode,entry))
694 { // success, no underflow
695 if (leaf&&iLeafOrg->LastEntry(pNode)==entry)
696 { // we deleted the last entry so... replace the pivot to ensure invariant
697 TPageRef next=iLeafOrg->LinkNode(pNode);
698 if (next!=KNullPageRef)
699 { // replace the pivot
701 TAny* pSibling=iPages->LockL(next);
702 NewPivot(pNode,pSibling,newPivot);
703 iPages->Unlock(pSibling);
705 // find dividing pivot which we know is there
706 do aPath.Pop(); while (aPath.Entry()==LastEntryL(aPath));
707 ReplacePivotL(aPath,GetNodeL(aPath),newPivot);
715 { // the root! If it has entries then we're done
716 if (nodeOrg->LastEntry(pNode)>0)
720 if (--iHeight!=0) // get child as root
721 iRoot=iIndexOrg->ChildNode(pNode,0);
722 else // tree is empty
723 iRoot=iFirst=KNullPageRef;
724 iPages->DeleteL(node);
728 // will definitely affect the index set
730 // block has underflowed.. must try to redistribute or concatenate
732 TAny* pParent=GetNodeL(aPath);
737 aPath.SetEntry(--entry);
738 pSibling=iPages->LockL(iIndexOrg->ChildNode(pParent,entry)); // on left
741 { // There must be a sibling for non-root nodes!
742 __ASSERT_DEBUG(iIndexOrg->LastEntry(pParent)>0,Panic(EBadEntryCount));
744 node=iIndexOrg->ChildNode(pParent,1); // on right of first child
745 pNode=iPages->LockL(node);
747 // redistribute between nodes
750 if (iLeafOrg->Redistribute(pSibling,pNode))
752 NewPivot(pSibling,pNode,newPivot);
753 redistributeOK: PutNode(pSibling,leaf);
755 ReplacePivotL(aPath,pParent,newPivot);
758 // failed so concatenate
759 iLeafOrg->Concatenate(pSibling,pNode);
760 iPages->UpdateL(pSibling); // must change it here and now: link target is being deleted
764 TPtrC8 pivot=iIndexOrg->Entry(pParent,entry);
765 if (iIndexOrg->Redistribute(pSibling,pNode,pivot,newPivot))
767 // failed so concatenate
768 iIndexOrg->Concatenate(pSibling,pNode,pivot);
769 iPages->Unlock(pSibling,EPageDirty);
771 iPages->DeleteL(node);
779 void TBtree::DeleteIndexSetL()
781 // Destroy the index set, handle broken state too
784 __ASSERT_DEBUG(!IsEmpty(),User::Invariant());
792 DeleteIndexNodesL(path,ETrue);
795 // delete the leading edge
796 while (iRoot!=iFirst)
798 TPageRef edge=iIndexOrg->ChildNode(iPages->LockL(iRoot),0);
799 iPages->DeleteL(iRoot);
804 void TBtree::DeleteIndexNodesL(TBtreePath& aPath,TBool aLeadingEdge)
806 // Destroy the children of the node in aPath, then destroy the node itself.
807 // do not destroy the sequence set or the leading edge
812 for (TInt ii=LastEntryL(aPath);ii>=0;--ii)
814 aPath.Push(ChildNodeL(aPath,ii));
815 DeleteIndexNodesL(aPath,aLeadingEdge && ii==0);
820 iPages->DeleteL(aPath.Node());
823 void TBtree::NewTreeL()
825 // Construct the initial node on empty tree.
828 TAny* first=iPages->AllocL();
829 iLeafOrg->Init(first);
830 iRoot=iFirst=iPages->AssignL(first); // non-reclaimable, it's a leaf as well as the initial root
834 void TBtree::NewRootL()
836 // Create a new root node and set its child to the current root.
839 if (iHeight==KMaxBtreeHeight)
840 __LEAVE(KErrOverflow); // tree is max height
841 TAny* root=iPages->AllocL();
842 iIndexOrg->Init(root);
843 iIndexOrg->MakeRoot(root,iRoot);
844 iRoot=iPages->AssignL(root); // non-reclaimable, mustn't lose track of successive roots
848 void TBtree::BeginUpdateL()
850 // Prepare to update the tree. Mark it dirty and ensure locked pages are abandoned on failure.
857 void TBtree::EndUpdate()
859 // Update complete, clear the broken flag and discard page pool acquisition
866 void TBtree::CheckIntactL() const
869 __LEAVE(KErrCorrupt);
872 void TBtree::GotoRoot(TBtreePath& aPath) const
874 // Set the path to the root of the tree.
877 __ASSERT_DEBUG(!IsEmpty(),Panic(ENoTree));
878 aPath.GotoRoot(iHeight,iRoot);
881 TAny* TBtree::GetNodeL(const TBtreePath& aPath) const
883 return iPages->LockL(aPath.Node());
886 void TBtree::PutNode(TAny* aNode,TBool aLeaf) const
888 iPages->Unlock(aNode,aLeaf&&(iStatus&ESecure)?EPageUpdate:EPageDirty);
891 TInt TBtree::LastEntryL(const TBtreePath& aPath) const
893 //#pragma message( __FILE__ " : 'TBtree::LastEntryL()' should have this information without having to look at the tree" )
894 TAny* pp=GetNodeL(aPath);
895 TInt cc=NodeOrg(aPath.IsLeaf())->LastEntry(pp);
900 TPageRef TBtree::ChildNodeL(const TBtreePath& aPath) const
902 // Get current child node.
905 __ASSERT_DEBUG(!aPath.IsLeaf(),Panic(EInvalidTreePos));
906 TAny* pp=GetNodeL(aPath);
907 TPageRef pg=iIndexOrg->ChildNode(pp,aPath.Entry());
912 TPageRef TBtree::ChildNodeL(TBtreePath& aPath,TInt aChild) const
914 aPath.SetEntry(aChild);
915 return ChildNodeL(aPath);
918 TPageRef TBtree::LastChildNodeL(TBtreePath& aPath) const
920 __ASSERT_DEBUG(!aPath.IsLeaf(),Panic(EInvalidTreePos));
921 TAny* pp=GetNodeL(aPath);
922 TInt entry=iIndexOrg->LastEntry(pp);
923 aPath.SetEntry(entry);
924 TPageRef pg=iIndexOrg->ChildNode(pp,entry);