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