2 * Copyright (c) 2000 Apple Computer, Inc. All rights reserved.
4 * @APPLE_LICENSE_HEADER_START@
6 * The contents of this file constitute Original Code as defined in and
7 * are subject to the Apple Public Source License Version 1.1 (the
8 * "License"). You may not use this file except in compliance with the
9 * License. Please obtain a copy of the License at
10 * http://www.apple.com/publicsource and read it before using this file.
12 * This Original Code and all software distributed under the License are
13 * distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, EITHER
14 * EXPRESS OR IMPLIED, AND APPLE HEREBY DISCLAIMS ALL SUCH WARRANTIES,
15 * INCLUDING WITHOUT LIMITATION, ANY WARRANTIES OF MERCHANTABILITY,
16 * FITNESS FOR A PARTICULAR PURPOSE OR NON-INFRINGEMENT. Please see the
17 * License for the specific language governing rights and limitations
20 * @APPLE_LICENSE_HEADER_END@
25 Contains: Private interface file for the BTree Module.
27 Version: xxx put the technology version here xxx
29 Written by: Gordon Sheridan and Bill Bruffey
31 Copyright: © 1992-1999 by Apple Computer, Inc., all rights reserved.
37 Other Contact: Mark Day
39 Technology: File Systems
49 Change History (most recent first):
50 <MacOSX> 3/19/99 djb Disable MoveRecordsLeft/Right macros since bcopy is broken.
52 <MacOSX> 8/10/98 djb Removed unused BTreeIterator from BTreeControlBlock, fixed alignment.
54 <CS5> 9/4/97 djb Convert MoveRecordsLeft and GetLeftSiblingNode to macros.
55 <CS4> 7/24/97 djb Add macro for GetRecordAddress (was a function before).
56 <CS3> 7/21/97 msd GetRecordByIndex now returns an OSStatus.
57 <CS2> 7/16/97 DSH FilesInternal.i renamed FileMgrInternal.i to avoid name
59 <CS1> 4/23/97 djb first checked in
61 <HFS6> 3/17/97 DSH Added a refCon field to BTreeControlBlock, for DFA use, to point
62 to additional data. Fixed Panic macros for use with SC.
63 <HFS5> 2/19/97 djb Add InsertKey struct. Moved on-disk definitions to
65 <HFS4> 1/27/97 djb InsertTree and DeleteTree are now recursive and support variable
67 <HFS3> 1/15/97 djb Move GetFileRefNumFromFCB macro to FilesInternal.h. Added
68 kBTVariableIndexKeysMask.
69 <HFS2> 1/3/97 djb Added support for large keys.
70 <HFS1> 12/19/96 djb first checked in
72 History applicable to original Scarecrow Design:
74 <7> 10/25/96 ser Changing for new VFPI
75 <6> 10/18/96 ser Converting over VFPI changes
76 <5> 9/17/96 dkh More BTree statistics
77 <4> 9/16/96 dkh Revised BTree statistics
78 <3> 6/20/96 dkh Radar #1358740. Switch from using Pools to debug MemAllocators.
79 <2> 12/7/95 dkh D10E2 build. Changed usage of Ref data type to LogicalAddress.
80 <1> 10/18/95 rst Moved from Scarecrow project.
82 <19> 11/22/94 djb Add prototype for GetMapNode
83 <18> 11/16/94 prp Add IsItAHint routine prototype.
84 <17> 9/30/94 prp Get in sync with D2 interface changes.
85 <16> 7/25/94 wjk Eliminate usage of BytePtr in favor of UInt8 *.
86 <15> 7/22/94 wjk Convert to the new set of header files.
87 <14> 5/31/94 srs Moved Btree types to public interface
88 <13> 12/9/93 wjk Add 68k alignment pragma's around persistent structures.
89 <12> 11/30/93 wjk Move from Makefiles to BuildFiles. Fit into the ModernOS and
91 <11> 11/23/93 wjk Changes required to compile on the RS6000.
92 <10> 8/30/93 CH Removed the M_ExitOnError and M_ReturnErrorIf macros which were
93 already defined in FileSystemPriv.h (included here).
94 <9> 8/30/93 CH Added parens around the M_ReturnErrorIf macro.
95 <8> 5/21/93 gs Add kBadClose flag. Add some prototypes for internal routines.
96 <7> 5/10/93 gs Change Ptr to BytePtr. Move BTreeTypes to BTree.h. Add
98 <6> 3/23/93 gs Remove mysterious "flags" field from HeaderRec structure. Move
99 prototypes of private functions to top of respective source
101 <5> 2/8/93 gs Update to use FSAgent.h Get/Release/SetEOF/SetBlockSize
102 procPtrs. Add UpdateNode routine.
103 <4> 12/10/92 gs Add Key Descriptor function declarations.
104 <3> 12/8/92 gs Add HeaderRec structure and incorporate review feedback.
105 <2> 12/2/92 gs Add GetNode and ReleaseNode callback procptrs to BTree CB, and
106 add internal function declarations.
107 <1> 11/15/92 gs first checked in
111 #ifndef __BTREESPRIVATE__
112 #define __BTREESPRIVATE__
114 #include <sys/appleapiopts.h>
117 #ifdef __APPLE_API_PRIVATE
119 #include "../../hfs_macos_defs.h"
121 #ifndef __FILEMGRINTERNAL__
122 #include "FileMgrInternal.h"
125 #ifndef __BTREESINTERNAL__
126 #include "BTreesInternal.h"
130 /////////////////////////////////// Constants ///////////////////////////////////
132 #define kBTreeVersion 1
133 #define kMaxTreeDepth 16
136 #define kHeaderNodeNum 0
137 #define kKeyDescRecord 1
140 // Header Node Record Offsets
142 kHeaderRecOffset
= 0x000E,
143 kKeyDescRecOffset
= 0x0078,
144 kHeaderMapRecOffset
= 0x00F8
147 #define kMinNodeSize 512
149 #define kMinRecordSize 6
150 // where is minimum record size enforced?
152 // miscellaneous BTree constants
163 // illegal string attribute bits set in mask
164 #define kBadStrAttribMask 0xCF
168 //////////////////////////////////// Macros /////////////////////////////////////
170 #define M_NodesInMap(mapSize) ((mapSize) << 3)
172 #define M_ClearBitNum(integer,bitNumber) ((integer) &= (~(1<<(bitNumber))))
173 #define M_SetBitNum(integer,bitNumber) ((integer) |= (1<<(bitNumber)))
174 #define M_IsOdd(integer) (((integer) & 1) != 0)
175 #define M_IsEven(integer) (((integer) & 1) == 0)
176 #define M_BTreeHeaderDirty(btreePtr) btreePtr->flags |= kBTHeaderDirty
178 #define M_MapRecordSize(nodeSize) (nodeSize - sizeof (BTNodeDescriptor) - 6)
179 #define M_HeaderMapRecordSize(nodeSize) (nodeSize - sizeof(BTNodeDescriptor) - sizeof(BTHeaderRec) - 128 - 8)
181 #define M_SWAP_BE16_ClearBitNum(integer,bitNumber) ((integer) &= SWAP_BE16(~(1<<(bitNumber))))
182 #define M_SWAP_BE16_SetBitNum(integer,bitNumber) ((integer) |= SWAP_BE16(1<<(bitNumber)))
184 ///////////////////////////////////// Types /////////////////////////////////////
186 typedef struct BTreeControlBlock
{ // fields specific to BTree CBs
188 UInt8 reserved1
; // keep for alignment with old style fields
191 FileReference fileRefNum
; // refNum of btree file
192 KeyCompareProcPtr keyCompareProc
;
195 UInt32 firstLeafNode
;
202 UInt16 reserved3
; // 4-byte alignment
206 UInt32 flags
; // dynamic flags
207 UInt32 attributes
; // persistent flags
209 UInt32 lastfsync
; /* Last time that this was fsynced */
211 GetBlockProcPtr getBlockProc
;
212 ReleaseBlockProcPtr releaseBlockProc
;
213 SetEndOfForkProcPtr setEndOfForkProc
;
215 // statistical information
217 UInt32 numGetNewNodes
;
218 UInt32 numReleaseNodes
;
219 UInt32 numUpdateNodes
;
220 UInt32 numMapNodesRead
; // map nodes beyond header node
221 UInt32 numHintChecks
;
222 UInt32 numPossibleHints
; // Looks like a formated hint
223 UInt32 numValidHints
; // Hint used to find correct record.
225 } BTreeControlBlock
, *BTreeControlBlockPtr
;
228 UInt32
CalcKeySize(const BTreeControlBlock
*btcb
, const BTreeKey
*key
);
229 #define CalcKeySize(btcb, key) ( ((btcb)->attributes & kBTBigKeysMask) ? ((key)->length16 + 2) : ((key)->length8 + 1) )
231 UInt32
KeyLength(const BTreeControlBlock
*btcb
, const BTreeKey
*key
);
232 #define KeyLength(btcb, key) ( ((btcb)->attributes & kBTBigKeysMask) ? (key)->length16 : (key)->length8 )
237 kBTHeaderDirty
= 0x00000001
241 typedef SInt8
*NodeBuffer
;
242 typedef BlockDescriptor NodeRec
, *NodePtr
; //\80\80 remove this someday...
247 //// Tree Path Table - constructed by SearchTree, used by InsertTree and DeleteTree
250 UInt32 node
; // node number
252 UInt16 reserved
; // align size to a power of 2
253 } TreePathRecord
, *TreePathRecordPtr
;
255 typedef TreePathRecord TreePathTable
[kMaxTreeDepth
];
258 //// InsertKey - used by InsertTree, InsertLevel and InsertNode
265 Boolean replacingKey
;
269 typedef struct InsertKey InsertKey
;
272 //// For Notational Convenience
274 typedef BTNodeDescriptor
* NodeDescPtr
;
275 typedef UInt8
*RecordPtr
;
276 typedef BTreeKeyPtr KeyPtr
;
279 //////////////////////////////////// Globals ////////////////////////////////////
282 //////////////////////////////////// Macros /////////////////////////////////////
285 #define Panic( message ) DebugStr( (ConstStr255Param) message )
286 #define PanicIf( condition, message ) if ( condition != 0 ) DebugStr( message )
288 #define Panic( message )
289 #define PanicIf( condition, message )
292 // Exit function on error
293 #define M_ExitOnError( result ) if ( ( result ) != noErr ) goto ErrorExit; else ;
295 // Test for passed condition and return if true
296 #define M_ReturnErrorIf( condition, error ) if ( condition ) return( error )
298 //////////////////////////////// Key Operations /////////////////////////////////
300 SInt32
CompareKeys (BTreeControlBlockPtr btreePtr
,
304 //////////////////////////////// Map Operations /////////////////////////////////
306 OSStatus
AllocateNode (BTreeControlBlockPtr btreePtr
,
309 OSStatus
FreeNode (BTreeControlBlockPtr btreePtr
,
312 OSStatus
ExtendBTree (BTreeControlBlockPtr btreePtr
,
315 UInt32
CalcMapBits (BTreeControlBlockPtr btreePtr
);
318 //////////////////////////////// Misc Operations ////////////////////////////////
320 UInt16
CalcKeyRecordSize (UInt16 keySize
,
323 OSStatus
VerifyHeader (FCB
*filePtr
,
324 BTHeaderRec
*header
);
326 OSStatus
UpdateHeader (BTreeControlBlockPtr btreePtr
,
327 Boolean forceWrite
);
329 OSStatus
FindIteratorPosition (BTreeControlBlockPtr btreePtr
,
330 BTreeIteratorPtr iterator
,
331 BlockDescriptor
*left
,
332 BlockDescriptor
*middle
,
333 BlockDescriptor
*right
,
336 Boolean
*foundRecord
);
338 OSStatus
CheckInsertParams (FCB
*filePtr
,
339 BTreeIterator
*iterator
,
340 FSBufferDescriptor
*record
,
343 OSStatus
TrySimpleReplace (BTreeControlBlockPtr btreePtr
,
345 BTreeIterator
*iterator
,
346 FSBufferDescriptor
*record
,
348 Boolean
*recordInserted
);
350 OSStatus
IsItAHint (BTreeControlBlockPtr btreePtr
,
351 BTreeIterator
*iterator
,
354 //////////////////////////////// Node Operations ////////////////////////////////
358 OSStatus
GetNode (BTreeControlBlockPtr btreePtr
,
360 NodeRec
*returnNodePtr
);
362 OSStatus
GetLeftSiblingNode (BTreeControlBlockPtr btreePtr
,
366 #define GetLeftSiblingNode(btree,node,left) GetNode ((btree), ((NodeDescPtr)(node))->bLink, (left))
368 OSStatus
GetRightSiblingNode (BTreeControlBlockPtr btreePtr
,
372 #define GetRightSiblingNode(btree,node,right) GetNode ((btree), ((NodeDescPtr)(node))->fLink, (right))
375 OSStatus
GetNewNode (BTreeControlBlockPtr btreePtr
,
377 NodeRec
*returnNodePtr
);
379 OSStatus
ReleaseNode (BTreeControlBlockPtr btreePtr
,
382 OSStatus
TrashNode (BTreeControlBlockPtr btreePtr
,
386 void ModifyBlockStart(FileReference vp
, BlockDescPtr blockPtr
);
389 OSStatus
UpdateNode (BTreeControlBlockPtr btreePtr
,
391 UInt32 transactionID
,
394 OSStatus
GetMapNode (BTreeControlBlockPtr btreePtr
,
395 BlockDescriptor
*nodePtr
,
399 //// Node Buffer Operations
401 OSStatus
CheckNode (BTreeControlBlockPtr btreePtr
,
404 void ClearNode (BTreeControlBlockPtr btreePtr
,
407 UInt16
GetNodeDataSize (BTreeControlBlockPtr btreePtr
,
410 UInt16
GetNodeFreeSize (BTreeControlBlockPtr btreePtr
,
414 //// Record Operations
416 Boolean
InsertRecord (BTreeControlBlockPtr btreePtr
,
422 Boolean
InsertKeyRecord (BTreeControlBlockPtr btreePtr
,
430 void DeleteRecord (BTreeControlBlockPtr btree
,
435 Boolean
SearchNode (BTreeControlBlockPtr btree
,
440 OSStatus
GetRecordByIndex (BTreeControlBlockPtr btree
,
447 UInt8
* GetRecordAddress (BTreeControlBlockPtr btree
,
451 #define GetRecordAddress(btreePtr,node,index) ((UInt8 *)(node) + (*(short *) ((UInt8 *)(node) + (btreePtr)->nodeSize - ((index) << 1) - kOffsetSize)))
454 UInt16
GetRecordSize (BTreeControlBlockPtr btree
,
458 UInt32
GetChildNodeNum (BTreeControlBlockPtr btreePtr
,
462 void MoveRecordsLeft (UInt8
* src
,
464 UInt16 bytesToMove
);
466 #define MoveRecordsLeft(src,dst,bytes) bcopy((src),(dst),(bytes))
468 void MoveRecordsRight (UInt8
* src
,
470 UInt16 bytesToMove
);
472 #define MoveRecordsRight(src,dst,bytes) bcopy((src),(dst),(bytes))
475 //////////////////////////////// Tree Operations ////////////////////////////////
477 OSStatus
SearchTree (BTreeControlBlockPtr btreePtr
,
479 TreePathTable treePathTable
,
481 BlockDescriptor
*nodePtr
,
484 OSStatus
InsertTree (BTreeControlBlockPtr btreePtr
,
485 TreePathTable treePathTable
,
489 BlockDescriptor
*targetNode
,
492 Boolean replacingKey
,
493 UInt32
*insertNode
);
495 OSStatus
DeleteTree (BTreeControlBlockPtr btreePtr
,
496 TreePathTable treePathTable
,
497 BlockDescriptor
*targetNode
,
501 #endif /* __APPLE_API_PRIVATE */
503 #endif //__BTREESPRIVATE__