2003-12-01 11:25:41 +01:00
|
|
|
/*
|
|
|
|
* PROGRAM: JRD Access Method
|
|
|
|
* MODULE: btn.cpp
|
|
|
|
* DESCRIPTION: B-tree management code
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Initial
|
|
|
|
* Developer's Public License Version 1.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the
|
|
|
|
* License. You may obtain a copy of the License at
|
|
|
|
* http://www.ibphoenix.com/main.nfs?a=ibphoenix&page=ibp_idpl.
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed AS IS,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing rights
|
|
|
|
* and limitations under the License.
|
|
|
|
*
|
|
|
|
* The Original Code was created by Arno Brinkman
|
|
|
|
* for the Firebird Open Source RDBMS project.
|
|
|
|
*
|
|
|
|
* Copyright (c) 2003 Arno Brinkman and all contributors
|
|
|
|
* signed below.
|
|
|
|
*
|
|
|
|
* All Rights Reserved.
|
|
|
|
* Contributor(s): ______________________________________.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "firebird.h" // needed for get_long
|
|
|
|
#include "memory_routines.h" // needed for get_long
|
|
|
|
|
|
|
|
#include "../jrd/common.h" // needed for SINT64
|
2008-12-05 02:20:14 +01:00
|
|
|
#include "../jrd/ods.h"
|
|
|
|
#include "../jrd/btn.h"
|
2003-12-01 11:25:41 +01:00
|
|
|
|
|
|
|
namespace BTreeNode {
|
|
|
|
|
2004-03-20 15:57:40 +01:00
|
|
|
using namespace Ods;
|
2003-12-01 11:25:41 +01:00
|
|
|
|
2008-12-05 02:20:14 +01:00
|
|
|
USHORT computePrefix(const UCHAR* prevString, USHORT prevLength,
|
2003-12-03 09:44:28 +01:00
|
|
|
const UCHAR* string, USHORT length)
|
2003-12-01 11:25:41 +01:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c o m p u t e P r e f i x
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2008-12-05 02:20:14 +01:00
|
|
|
* Compute and return prefix common
|
2003-12-01 11:25:41 +01:00
|
|
|
* to two strings.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-12-03 09:44:28 +01:00
|
|
|
USHORT l = MIN(prevLength, length);
|
|
|
|
if (!l) {
|
2003-12-01 11:25:41 +01:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2003-12-03 09:44:28 +01:00
|
|
|
const UCHAR* p = prevString;
|
2003-12-01 11:25:41 +01:00
|
|
|
|
2009-05-01 19:21:36 +02:00
|
|
|
while (*p == *string)
|
|
|
|
{
|
2003-12-03 09:44:28 +01:00
|
|
|
++p;
|
|
|
|
++string;
|
2003-12-01 11:25:41 +01:00
|
|
|
if (!--l) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return (p - prevString);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-02-02 12:02:12 +01:00
|
|
|
SLONG findPageInDuplicates(const btree_page* page, UCHAR* pointer,
|
2004-09-28 08:28:38 +02:00
|
|
|
SLONG previousNumber, RecordNumber findRecordNumber)
|
2003-12-01 11:25:41 +01:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* f i n d P a g e I n D u p l i c a t e s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Return the first page number
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-12-03 09:44:28 +01:00
|
|
|
const bool leafPage = (page->btr_level == 0);
|
2006-04-19 09:10:53 +02:00
|
|
|
const UCHAR flags = page->btr_header.pag_flags;
|
2004-09-24 02:11:32 +02:00
|
|
|
//const UCHAR* endPointer = (UCHAR*)page + page->btr_length;
|
2003-12-01 11:25:41 +01:00
|
|
|
|
2003-12-03 09:44:28 +01:00
|
|
|
IndexNode node, previousNode;
|
2003-12-01 11:25:41 +01:00
|
|
|
pointer = readNode(&node, pointer, flags, leafPage);
|
2008-12-05 02:20:14 +01:00
|
|
|
|
2004-09-24 02:11:32 +02:00
|
|
|
// Check if pointer is still valid
|
|
|
|
//if (pointer > endPointer) {
|
|
|
|
// BUGCHECK(204); // msg 204 index inconsistent
|
|
|
|
//}
|
2003-12-01 11:25:41 +01:00
|
|
|
|
2009-05-01 19:21:36 +02:00
|
|
|
while (true)
|
|
|
|
{
|
2008-12-05 02:20:14 +01:00
|
|
|
// loop through duplicates until
|
2003-12-01 11:25:41 +01:00
|
|
|
// correct node is found.
|
|
|
|
// If this is an end bucket marker then return
|
|
|
|
// the previous passed page number.
|
2004-03-07 22:50:53 +01:00
|
|
|
if (node.isEndBucket) {
|
2003-12-01 11:25:41 +01:00
|
|
|
return previousNumber;
|
|
|
|
}
|
2004-09-24 02:11:32 +02:00
|
|
|
if (findRecordNumber <= node.recordNumber) {
|
2003-12-01 11:25:41 +01:00
|
|
|
// If first record number on page is higher
|
|
|
|
// then record number must be at the previous
|
|
|
|
// passed page number.
|
|
|
|
return previousNumber;
|
2008-12-05 02:20:14 +01:00
|
|
|
}
|
|
|
|
// Save current page number and fetch next node
|
2003-12-01 11:25:41 +01:00
|
|
|
// for comparision.
|
|
|
|
previousNumber = node.pageNumber;
|
|
|
|
previousNode = node;
|
|
|
|
pointer = BTreeNode::readNode(&node, pointer, flags, leafPage);
|
2004-09-24 02:11:32 +02:00
|
|
|
|
|
|
|
// Check if pointer is still valid
|
|
|
|
//if (pointer > endPointer) {
|
|
|
|
// BUGCHECK(204); // msg 204 index inconsistent
|
|
|
|
//}
|
|
|
|
|
2008-12-05 02:20:14 +01:00
|
|
|
// We're done if end level marker is reached or this
|
2003-12-01 11:25:41 +01:00
|
|
|
// isn't a equal node anymore.
|
2008-12-05 02:20:14 +01:00
|
|
|
if ((node.isEndLevel) ||
|
|
|
|
(node.length != 0) ||
|
2003-12-01 11:25:41 +01:00
|
|
|
(node.prefix != (previousNode.length + previousNode.prefix)))
|
|
|
|
{
|
|
|
|
return previousNumber;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// We never reach this point
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-04-19 09:10:53 +02:00
|
|
|
USHORT getJumpNodeSize(const IndexJumpNode* jumpNode, UCHAR flags)
|
2003-12-01 11:25:41 +01:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* g e t J u m p N o d e S i z e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2008-12-05 02:20:14 +01:00
|
|
|
* Return the size needed to store
|
2003-12-01 11:25:41 +01:00
|
|
|
* this node.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
USHORT result = 0;
|
2009-05-01 19:21:36 +02:00
|
|
|
if (flags & btr_large_keys)
|
|
|
|
{
|
2003-12-01 11:25:41 +01:00
|
|
|
// Size needed for prefix
|
2004-02-25 23:14:19 +01:00
|
|
|
USHORT number = jumpNode->prefix;
|
2004-03-07 22:50:53 +01:00
|
|
|
if (number & 0xC000) {
|
|
|
|
result += 3;
|
|
|
|
}
|
|
|
|
else if (number & 0xFF80) {
|
|
|
|
result += 2;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2004-03-07 22:50:53 +01:00
|
|
|
else {
|
|
|
|
result += 1;
|
|
|
|
}
|
|
|
|
|
2003-12-01 11:25:41 +01:00
|
|
|
// Size needed for length
|
2004-02-25 23:14:19 +01:00
|
|
|
number = jumpNode->length;
|
2004-03-07 22:50:53 +01:00
|
|
|
if (number & 0xC000) {
|
|
|
|
result += 3;
|
|
|
|
}
|
|
|
|
else if (number & 0xFF80) {
|
|
|
|
result += 2;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
result += 1;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
2009-05-01 19:21:36 +02:00
|
|
|
else
|
|
|
|
{
|
2003-12-01 11:25:41 +01:00
|
|
|
// Size needed for prefix
|
|
|
|
result++;
|
|
|
|
// Size needed for length
|
|
|
|
result++;
|
|
|
|
}
|
|
|
|
// Size needed for offset
|
2004-02-25 23:14:19 +01:00
|
|
|
// NOTE! offset can be unknown when this function is called,
|
|
|
|
// therefor we can't use a compression method.
|
2003-12-01 11:25:41 +01:00
|
|
|
result += sizeof(USHORT);
|
|
|
|
// Size needed for data
|
|
|
|
result += jumpNode->length;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-04-19 09:10:53 +02:00
|
|
|
USHORT getNodeSize(const IndexNode* indexNode, UCHAR flags, bool leafNode)
|
2003-12-01 11:25:41 +01:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* g e t N o d e S i z e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2008-12-05 02:20:14 +01:00
|
|
|
* Return the size needed to store
|
2003-12-01 11:25:41 +01:00
|
|
|
* this node.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
USHORT result = 0;
|
2009-05-01 19:21:36 +02:00
|
|
|
if (flags & btr_large_keys)
|
|
|
|
{
|
2004-02-25 23:14:19 +01:00
|
|
|
|
2008-12-05 02:20:14 +01:00
|
|
|
// Determine flags
|
2004-02-25 23:14:19 +01:00
|
|
|
UCHAR internalFlags = 0;
|
2004-09-24 02:11:32 +02:00
|
|
|
if (indexNode->isEndLevel) {
|
|
|
|
internalFlags = BTN_END_LEVEL_FLAG;
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
2004-09-24 02:11:32 +02:00
|
|
|
else if (indexNode->isEndBucket) {
|
|
|
|
internalFlags = BTN_END_BUCKET_FLAG;
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
2009-05-01 19:21:36 +02:00
|
|
|
else if (indexNode->length == 0)
|
|
|
|
{
|
2004-11-19 00:00:20 +01:00
|
|
|
if (indexNode->prefix == 0) {
|
|
|
|
internalFlags = BTN_ZERO_PREFIX_ZERO_LENGTH_FLAG;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
internalFlags = BTN_ZERO_LENGTH_FLAG;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (indexNode->length == 1) {
|
|
|
|
internalFlags = BTN_ONE_LENGTH_FLAG;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
|
|
|
|
2004-11-19 00:00:20 +01:00
|
|
|
// Store internal flags + 5 bits from number
|
2004-09-28 08:28:38 +02:00
|
|
|
SINT64 number = indexNode->recordNumber.getValue();
|
2004-02-25 23:14:19 +01:00
|
|
|
if (number < 0) {
|
|
|
|
number = 0;
|
|
|
|
}
|
|
|
|
result++;
|
2004-11-19 00:00:20 +01:00
|
|
|
// If this is a END_LEVEL marker then we're done
|
|
|
|
if (indexNode->isEndLevel) {
|
|
|
|
return result;
|
|
|
|
}
|
2004-02-25 23:14:19 +01:00
|
|
|
|
2004-11-19 00:00:20 +01:00
|
|
|
number >>= 5;
|
2004-02-25 23:14:19 +01:00
|
|
|
// Get size for storing remaining bits for number
|
2004-09-28 08:28:38 +02:00
|
|
|
// 5 bytes should be enough to fit remaining 34 bits of record number
|
|
|
|
if (number & QUADCONST(0xFFF0000000)) {
|
2004-02-26 01:07:37 +01:00
|
|
|
result += 5;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2004-09-28 08:28:38 +02:00
|
|
|
else if (number & QUADCONST(0xFFFFE00000)) {
|
2004-02-26 01:07:37 +01:00
|
|
|
result += 4;
|
|
|
|
}
|
2004-09-28 08:28:38 +02:00
|
|
|
else if (number & QUADCONST(0xFFFFFFC000)) {
|
2004-02-26 01:07:37 +01:00
|
|
|
result += 3;
|
|
|
|
}
|
2004-09-28 08:28:38 +02:00
|
|
|
else if (number & QUADCONST(0xFFFFFFFF80)) {
|
2004-02-26 01:07:37 +01:00
|
|
|
result += 2;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2004-02-26 01:07:37 +01:00
|
|
|
else {
|
|
|
|
result += 1;
|
|
|
|
}
|
|
|
|
|
2009-05-01 19:21:36 +02:00
|
|
|
if (!leafNode)
|
|
|
|
{
|
2004-02-25 23:14:19 +01:00
|
|
|
// Size needed for page number
|
|
|
|
number = indexNode->pageNumber;
|
|
|
|
if (number < 0) {
|
2008-12-05 02:20:14 +01:00
|
|
|
number = 0;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
|
|
|
|
2004-02-26 01:07:37 +01:00
|
|
|
if (number & 0xF0000000) {
|
|
|
|
result += 5;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2004-02-26 01:07:37 +01:00
|
|
|
else if (number & 0xFFE00000) {
|
|
|
|
result += 4;
|
|
|
|
}
|
|
|
|
else if (number & 0xFFFFC000) {
|
|
|
|
result += 3;
|
|
|
|
}
|
|
|
|
else if (number & 0xFFFFFF80) {
|
|
|
|
result += 2;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
result += 1;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-05-01 19:21:36 +02:00
|
|
|
if (internalFlags != BTN_ZERO_PREFIX_ZERO_LENGTH_FLAG)
|
|
|
|
{
|
2008-12-05 02:20:14 +01:00
|
|
|
// Size needed for prefix
|
2004-02-25 23:14:19 +01:00
|
|
|
number = indexNode->prefix;
|
2004-03-07 22:50:53 +01:00
|
|
|
if (number & 0xFFFFC000) {
|
|
|
|
result += 3;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2004-03-07 22:50:53 +01:00
|
|
|
else if (number & 0xFFFFFF80) {
|
|
|
|
result += 2;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
result += 1;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2004-11-19 00:00:20 +01:00
|
|
|
}
|
2004-02-25 23:14:19 +01:00
|
|
|
|
2004-11-19 00:00:20 +01:00
|
|
|
if ((internalFlags != BTN_ZERO_PREFIX_ZERO_LENGTH_FLAG) &&
|
|
|
|
(internalFlags != BTN_ZERO_LENGTH_FLAG) &&
|
2008-12-05 02:20:14 +01:00
|
|
|
(internalFlags != BTN_ONE_LENGTH_FLAG))
|
2004-11-19 00:00:20 +01:00
|
|
|
{
|
2008-12-05 02:20:14 +01:00
|
|
|
// Size needed for length
|
2004-02-25 23:14:19 +01:00
|
|
|
number = indexNode->length;
|
2004-03-07 22:50:53 +01:00
|
|
|
if (number & 0xFFFFC000) {
|
|
|
|
result += 3;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2004-03-07 22:50:53 +01:00
|
|
|
else if (number & 0xFFFFFF80) {
|
|
|
|
result += 2;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
result += 1;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
2004-02-25 23:14:19 +01:00
|
|
|
|
|
|
|
result += indexNode->length;
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
2009-05-01 19:21:36 +02:00
|
|
|
else
|
|
|
|
{
|
2008-12-05 02:20:14 +01:00
|
|
|
// Size needed for prefix
|
2003-12-01 11:25:41 +01:00
|
|
|
result++;
|
2008-12-05 02:20:14 +01:00
|
|
|
// Size needed for length
|
2003-12-01 11:25:41 +01:00
|
|
|
result++;
|
2008-12-05 02:20:14 +01:00
|
|
|
// Size needed for page number
|
2003-12-01 11:25:41 +01:00
|
|
|
result += sizeof(SLONG);
|
|
|
|
|
2004-02-25 23:14:19 +01:00
|
|
|
result += indexNode->length;
|
2003-12-01 11:25:41 +01:00
|
|
|
|
2004-09-24 02:11:32 +02:00
|
|
|
if ((flags & btr_all_record_number) &&
|
|
|
|
(!leafNode ||
|
2008-12-05 02:20:14 +01:00
|
|
|
(leafNode && indexNode->isEndBucket && (indexNode->length == 0))))
|
2004-09-24 02:11:32 +02:00
|
|
|
{
|
2004-02-25 23:14:19 +01:00
|
|
|
// Size needed for record number
|
|
|
|
result += sizeof(SLONG);
|
|
|
|
}
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-02-02 12:02:12 +01:00
|
|
|
UCHAR* getPointerFirstNode(btree_page* page, IndexJumpInfo* jumpInfo)
|
2003-12-01 11:25:41 +01:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* g e t P o i n t e r F i r s t N o d e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2008-12-05 02:20:14 +01:00
|
|
|
* Return the pointer to first data node
|
2003-12-01 11:25:41 +01:00
|
|
|
* in the bucket. When a jumpInfo is
|
|
|
|
* given, then pointer of first jump
|
|
|
|
* node is returned.
|
|
|
|
*
|
|
|
|
**************************************/
|
2008-01-16 09:31:31 +01:00
|
|
|
if (page->btr_header.pag_flags & btr_jump_info)
|
|
|
|
{
|
2003-12-01 11:25:41 +01:00
|
|
|
if (jumpInfo) {
|
2003-12-03 09:44:28 +01:00
|
|
|
UCHAR* pointer = reinterpret_cast<UCHAR*>(page->btr_nodes);
|
2003-12-01 11:25:41 +01:00
|
|
|
return readJumpInfo(jumpInfo, pointer);
|
|
|
|
}
|
2008-01-16 09:31:31 +01:00
|
|
|
|
|
|
|
IndexJumpInfo jumpInformation;
|
|
|
|
UCHAR* pointer = reinterpret_cast<UCHAR*>(page->btr_nodes);
|
|
|
|
readJumpInfo(&jumpInformation, pointer);
|
|
|
|
|
|
|
|
return reinterpret_cast<UCHAR*>(page) + jumpInformation.firstNodeOffset;
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
2008-01-16 09:31:31 +01:00
|
|
|
|
|
|
|
return reinterpret_cast<UCHAR*>(page->btr_nodes);
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-12-03 09:44:28 +01:00
|
|
|
bool keyEquality(USHORT length, const UCHAR* data, const IndexNode* indexNode)
|
2003-12-01 11:25:41 +01:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* k e y E q u a l i t y
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Check a B-tree node against a key for equality.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
if (length != indexNode->length + indexNode->prefix) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
USHORT l = indexNode->length;
|
|
|
|
if (!l) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2003-12-03 09:44:28 +01:00
|
|
|
const UCHAR* p = indexNode->data;
|
|
|
|
const UCHAR* q = data + indexNode->prefix;
|
2003-12-01 11:25:41 +01:00
|
|
|
while (l) {
|
|
|
|
if (*p++ != *q++) {
|
|
|
|
return false;
|
|
|
|
}
|
2003-12-03 09:44:28 +01:00
|
|
|
--l;
|
2004-11-07 11:47:20 +01:00
|
|
|
}
|
2003-12-01 11:25:41 +01:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
2004-03-28 11:10:30 +02:00
|
|
|
UCHAR* lastNode(btree_page* page, exp_index_buf* expanded_page, btree_exp** expanded_node)
|
2003-12-01 11:25:41 +01:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* l a s t N o d e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
2008-12-05 02:20:14 +01:00
|
|
|
* Functional description
|
2003-12-01 11:25:41 +01:00
|
|
|
* Find the last node on a page. Used when walking
|
2008-12-05 02:20:14 +01:00
|
|
|
* down the right side of an index tree.
|
2003-12-01 11:25:41 +01:00
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2008-12-05 02:20:14 +01:00
|
|
|
// the last expanded node is always at the end of the page
|
2004-03-28 11:10:30 +02:00
|
|
|
// minus the size of a btree_exp, since there is always an extra
|
|
|
|
// btree_exp node with zero-length tail at the end of the page
|
|
|
|
btree_exp* enode = (btree_exp*) ((UCHAR*) expanded_page + expanded_page->exp_length - BTX_SIZE);
|
2003-12-01 11:25:41 +01:00
|
|
|
|
|
|
|
// starting at the end of the page, find the
|
|
|
|
// first node that is not an end marker
|
|
|
|
UCHAR* pointer = ((UCHAR*) page + page->btr_length);
|
2008-01-16 09:31:31 +01:00
|
|
|
const UCHAR flags = page->btr_header.pag_flags;
|
2003-12-01 11:25:41 +01:00
|
|
|
IndexNode node;
|
2009-05-01 19:21:36 +02:00
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
pointer = previousNode(/*&node,*/ pointer, /*flags,*/ &enode);
|
|
|
|
if (!node.isEndBucket && !node.isEndLevel)
|
2003-12-01 11:25:41 +01:00
|
|
|
{
|
|
|
|
if (expanded_node) {
|
|
|
|
*expanded_node = enode;
|
|
|
|
}
|
|
|
|
return node.nodePointer;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2008-12-05 02:20:14 +01:00
|
|
|
UCHAR* nextNode(IndexNode* node, UCHAR* pointer,
|
2006-04-19 09:10:53 +02:00
|
|
|
UCHAR flags, btree_exp** expanded_node)
|
2003-12-01 11:25:41 +01:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* n e x t N o d e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
2008-12-05 02:20:14 +01:00
|
|
|
* Functional description
|
2003-12-01 11:25:41 +01:00
|
|
|
* Find the next node on both the index page
|
|
|
|
* and its associated expanded buffer.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
pointer = readNode(node, pointer, flags, true);
|
|
|
|
|
|
|
|
if (*expanded_node) {
|
2004-03-28 11:10:30 +02:00
|
|
|
*expanded_node = (btree_exp*) ((UCHAR*) (*expanded_node)->btx_data +
|
2003-12-01 11:25:41 +01:00
|
|
|
node->prefix + node->length);
|
|
|
|
}
|
|
|
|
|
|
|
|
return pointer;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-05-01 19:21:36 +02:00
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
UCHAR* previousNode(/*IndexNode* node,*/ UCHAR* pointer,
|
|
|
|
/*UCHAR flags,*/ btree_exp** expanded_node)
|
2003-12-01 11:25:41 +01:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p r e v i o u s N o d e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
2008-12-05 02:20:14 +01:00
|
|
|
* Functional description
|
2003-12-01 11:25:41 +01:00
|
|
|
* Find the previous node on a page. Used when walking
|
2008-12-05 02:20:14 +01:00
|
|
|
* an index backwards.
|
2003-12-01 11:25:41 +01:00
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
pointer = (pointer - (*expanded_node)->btx_btr_previous_length);
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
*expanded_node = (btree_exp*) ((UCHAR*) *expanded_node - (*expanded_node)->btx_previous_length);
|
2003-12-01 11:25:41 +01:00
|
|
|
|
|
|
|
return pointer;
|
|
|
|
}
|
2009-05-01 19:21:36 +02:00
|
|
|
#endif
|
2003-12-01 11:25:41 +01:00
|
|
|
|
|
|
|
|
|
|
|
UCHAR* readJumpInfo(IndexJumpInfo* jumpInfo, UCHAR* pagePointer)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e a d J u m p I n f o
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2008-12-05 02:20:14 +01:00
|
|
|
* Read jump information from the page
|
|
|
|
* by the given pagePointer and the
|
|
|
|
* return the remaining position after
|
2003-12-01 11:25:41 +01:00
|
|
|
* the read.
|
|
|
|
*
|
|
|
|
**************************************/
|
2006-01-26 11:45:02 +01:00
|
|
|
jumpInfo->firstNodeOffset = get_short(pagePointer);
|
2003-12-01 11:25:41 +01:00
|
|
|
pagePointer += sizeof(USHORT);
|
2006-01-26 11:45:02 +01:00
|
|
|
jumpInfo->jumpAreaSize = get_short(pagePointer);
|
2003-12-01 11:25:41 +01:00
|
|
|
pagePointer += sizeof(USHORT);
|
2006-01-26 11:45:02 +01:00
|
|
|
jumpInfo->jumpers = (USHORT)(*pagePointer++);
|
2003-12-01 11:25:41 +01:00
|
|
|
return pagePointer;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-05-01 19:21:36 +02:00
|
|
|
UCHAR* readJumpNode(IndexJumpNode* jumpNode, UCHAR* pagePointer, UCHAR flags)
|
2003-12-01 11:25:41 +01:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e a d J u m p N o d e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Read a jump node from the page by the
|
|
|
|
* given pagePointer and the return the
|
|
|
|
* remaining position after the read.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
jumpNode->nodePointer = pagePointer;
|
2009-05-01 19:21:36 +02:00
|
|
|
if (flags & btr_large_keys)
|
|
|
|
{
|
2004-02-25 23:14:19 +01:00
|
|
|
// Get prefix
|
2006-01-26 11:45:02 +01:00
|
|
|
UCHAR tmp = *pagePointer++;
|
2004-02-25 23:14:19 +01:00
|
|
|
jumpNode->prefix = (tmp & 0x7F);
|
|
|
|
if (tmp & 0x80) {
|
2006-01-26 11:45:02 +01:00
|
|
|
tmp = *pagePointer++;
|
2004-02-25 23:14:19 +01:00
|
|
|
jumpNode->prefix |= (tmp & 0x7F) << 7; // We get 14 bits at this point
|
|
|
|
}
|
|
|
|
// Get length
|
2006-01-26 11:45:02 +01:00
|
|
|
tmp = *pagePointer++;
|
2004-02-25 23:14:19 +01:00
|
|
|
jumpNode->length = (tmp & 0x7F);
|
|
|
|
if (tmp & 0x80) {
|
2006-01-26 11:45:02 +01:00
|
|
|
tmp = *pagePointer++;
|
2004-02-25 23:14:19 +01:00
|
|
|
jumpNode->length |= (tmp & 0x7F) << 7; // We get 14 bits at this point
|
|
|
|
}
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
|
|
|
else {
|
2006-01-26 11:45:02 +01:00
|
|
|
jumpNode->prefix = (USHORT)(*pagePointer++);
|
|
|
|
jumpNode->length = (USHORT)(*pagePointer++);
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
2006-01-26 11:45:02 +01:00
|
|
|
jumpNode->offset = get_short(pagePointer);
|
2003-12-01 11:25:41 +01:00
|
|
|
pagePointer += sizeof(USHORT);
|
|
|
|
jumpNode->data = pagePointer;
|
|
|
|
pagePointer += jumpNode->length;
|
|
|
|
return pagePointer;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-02-02 12:02:12 +01:00
|
|
|
UCHAR* writeJumpInfo(btree_page* page, const IndexJumpInfo* jumpInfo)
|
2003-12-01 11:25:41 +01:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* w r i t e J u m p I n f o
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Write jump information to the page by the
|
|
|
|
* given pointer.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-18 06:56:06 +01:00
|
|
|
UCHAR* pointer = reinterpret_cast<UCHAR*>(page->btr_nodes);
|
2006-01-26 11:45:02 +01:00
|
|
|
put_short(pointer, jumpInfo->firstNodeOffset);
|
2003-12-01 11:25:41 +01:00
|
|
|
pointer += sizeof(USHORT);
|
2006-01-26 11:45:02 +01:00
|
|
|
put_short(pointer, jumpInfo->jumpAreaSize);
|
2003-12-01 11:25:41 +01:00
|
|
|
pointer += sizeof(USHORT);
|
2006-01-26 11:45:02 +01:00
|
|
|
*pointer++ = (UCHAR) jumpInfo->jumpers;
|
2003-12-01 11:25:41 +01:00
|
|
|
return pointer;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-05-01 19:21:36 +02:00
|
|
|
UCHAR* writeJumpNode(IndexJumpNode* jumpNode, UCHAR* pagePointer, UCHAR flags)
|
2003-12-01 11:25:41 +01:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* w r i t e J u m p N o d e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Write jump information to the page by the
|
|
|
|
* given pointer.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
jumpNode->nodePointer = pagePointer;
|
2009-05-01 19:21:36 +02:00
|
|
|
if (flags & btr_large_keys)
|
|
|
|
{
|
2004-02-25 23:14:19 +01:00
|
|
|
// Write prefix, maximum 14 bits
|
|
|
|
USHORT number = jumpNode->prefix;
|
|
|
|
UCHAR tmp = (number & 0x7F);
|
|
|
|
number >>= 7;
|
|
|
|
if (number > 0) {
|
|
|
|
tmp |= 0x80;
|
|
|
|
}
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = tmp;
|
2004-02-25 23:14:19 +01:00
|
|
|
if (tmp & 0x80) {
|
|
|
|
tmp = (number & 0x7F);
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = tmp;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Write length, maximum 14 bits
|
|
|
|
number = jumpNode->length;
|
|
|
|
tmp = (number & 0x7F);
|
|
|
|
number >>= 7;
|
|
|
|
if (number > 0) {
|
|
|
|
tmp |= 0x80;
|
|
|
|
}
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = tmp;
|
2004-02-25 23:14:19 +01:00
|
|
|
if (tmp & 0x80) {
|
|
|
|
tmp = (number & 0x7F);
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = tmp;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
|
|
|
else {
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = (UCHAR) jumpNode->prefix;
|
|
|
|
*pagePointer++ = (UCHAR) jumpNode->length;
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
2006-01-26 11:45:02 +01:00
|
|
|
put_short(pagePointer, jumpNode->offset);
|
2003-12-01 11:25:41 +01:00
|
|
|
pagePointer += sizeof(USHORT);
|
2004-02-25 23:14:19 +01:00
|
|
|
memmove(pagePointer, jumpNode->data, jumpNode->length);
|
|
|
|
pagePointer += jumpNode->length;
|
2003-12-01 11:25:41 +01:00
|
|
|
return pagePointer;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-04-19 09:10:53 +02:00
|
|
|
UCHAR* writeNode(IndexNode* indexNode, UCHAR* pagePointer, UCHAR flags,
|
2003-12-01 11:25:41 +01:00
|
|
|
bool leafNode, bool withData)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* w r i t e N o d e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Write a leaf/page node to the page by the
|
|
|
|
* given page_pointer.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
indexNode->nodePointer = pagePointer;
|
2009-05-01 19:21:36 +02:00
|
|
|
if (flags & btr_large_keys)
|
|
|
|
{
|
2004-02-25 23:14:19 +01:00
|
|
|
|
|
|
|
// AB: 2004-02-22
|
|
|
|
// To allow as much as compression possible we
|
|
|
|
// store numbers per 7 bit and the 8-th bit tell us
|
|
|
|
// if we need to go on reading or we're done.
|
2008-12-05 02:20:14 +01:00
|
|
|
// Also for duplicate node entries (length and prefix
|
2004-02-25 23:14:19 +01:00
|
|
|
// are zero) we don't store the length and prefix
|
|
|
|
// information. This will save at least 2 bytes per node.
|
|
|
|
|
2009-05-01 19:21:36 +02:00
|
|
|
if (!withData)
|
|
|
|
{
|
2004-09-24 02:11:32 +02:00
|
|
|
// First move data so we can't override it.
|
|
|
|
// For older structure node was always the same, but length
|
|
|
|
// from new structure depends on the values.
|
|
|
|
const USHORT offset = getNodeSize(indexNode, flags, leafNode) - indexNode->length;
|
|
|
|
pagePointer += offset; // set pointer to right position
|
|
|
|
memmove(pagePointer, indexNode->data, indexNode->length);
|
|
|
|
pagePointer -= offset; // restore pointer to original position
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
2004-09-24 02:11:32 +02:00
|
|
|
|
2004-02-25 23:14:19 +01:00
|
|
|
// Internal flags
|
|
|
|
UCHAR internalFlags = 0;
|
2004-09-24 02:11:32 +02:00
|
|
|
if (indexNode->isEndLevel) {
|
2004-02-25 23:14:19 +01:00
|
|
|
internalFlags = BTN_END_LEVEL_FLAG;
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
2004-09-24 02:11:32 +02:00
|
|
|
else if (indexNode->isEndBucket) {
|
2004-02-25 23:14:19 +01:00
|
|
|
internalFlags = BTN_END_BUCKET_FLAG;
|
|
|
|
}
|
2009-05-01 19:21:36 +02:00
|
|
|
else if (indexNode->length == 0)
|
|
|
|
{
|
2004-11-19 00:00:20 +01:00
|
|
|
if (indexNode->prefix == 0) {
|
|
|
|
internalFlags = BTN_ZERO_PREFIX_ZERO_LENGTH_FLAG;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
internalFlags = BTN_ZERO_LENGTH_FLAG;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (indexNode->length == 1) {
|
|
|
|
internalFlags = BTN_ONE_LENGTH_FLAG;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2004-09-24 02:11:32 +02:00
|
|
|
|
2004-09-28 08:28:38 +02:00
|
|
|
SINT64 number = indexNode->recordNumber.getValue();
|
2004-02-25 23:14:19 +01:00
|
|
|
if (number < 0) {
|
|
|
|
number = 0;
|
|
|
|
}
|
|
|
|
// Store internal flags + 6 bits from number
|
|
|
|
UCHAR tmp = internalFlags;
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = ((tmp << 5) | (number & 0x1F));
|
2004-02-25 23:14:19 +01:00
|
|
|
|
2004-11-19 00:00:20 +01:00
|
|
|
if (indexNode->isEndLevel) {
|
|
|
|
return pagePointer;
|
|
|
|
}
|
|
|
|
|
2004-02-25 23:14:19 +01:00
|
|
|
// Store remaining bits from number
|
2004-11-19 00:00:20 +01:00
|
|
|
number >>= 5;
|
2004-02-25 23:14:19 +01:00
|
|
|
tmp = (number & 0x7F);
|
2004-11-19 00:00:20 +01:00
|
|
|
number >>= 7; //12
|
|
|
|
if (number == 0) {
|
|
|
|
*pagePointer++ = tmp;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2009-05-01 19:21:36 +02:00
|
|
|
else
|
|
|
|
{
|
2004-11-19 00:00:20 +01:00
|
|
|
*pagePointer++ = tmp | 0x80;
|
2004-02-25 23:14:19 +01:00
|
|
|
tmp = (number & 0x7F);
|
2004-11-19 00:00:20 +01:00
|
|
|
number >>= 7; //19
|
|
|
|
if (number == 0) {
|
|
|
|
*pagePointer++ = tmp;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2009-05-01 19:21:36 +02:00
|
|
|
else
|
|
|
|
{
|
2004-11-19 00:00:20 +01:00
|
|
|
*pagePointer++ = tmp | 0x80;
|
2004-02-25 23:14:19 +01:00
|
|
|
tmp = (number & 0x7F);
|
2004-11-19 00:00:20 +01:00
|
|
|
number >>= 7; //26
|
|
|
|
if (number == 0) {
|
|
|
|
*pagePointer++ = tmp;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2009-05-01 19:21:36 +02:00
|
|
|
else
|
|
|
|
{
|
2004-11-19 00:00:20 +01:00
|
|
|
*pagePointer++ = tmp | 0x80;
|
2004-02-25 23:14:19 +01:00
|
|
|
tmp = (number & 0x7F);
|
2004-11-19 00:00:20 +01:00
|
|
|
number >>= 7; //33
|
|
|
|
if (number == 0) {
|
|
|
|
*pagePointer++ = tmp;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2009-05-01 19:21:36 +02:00
|
|
|
else
|
|
|
|
{
|
2004-11-19 00:00:20 +01:00
|
|
|
*pagePointer++ = tmp | 0x80;
|
2004-02-25 23:14:19 +01:00
|
|
|
tmp = (number & 0x7F);
|
2004-11-19 00:00:20 +01:00
|
|
|
number >>= 7; //40
|
|
|
|
if (number == 0) {
|
|
|
|
*pagePointer++ = tmp;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2008-12-05 02:20:14 +01:00
|
|
|
/*
|
2004-11-19 00:00:20 +01:00
|
|
|
Enable this if you need more bits in record number
|
2009-05-01 19:21:36 +02:00
|
|
|
else
|
|
|
|
{
|
2004-11-19 00:00:20 +01:00
|
|
|
*pagePointer++ = tmp | 0x80;
|
2004-02-25 23:14:19 +01:00
|
|
|
tmp = (number & 0x7F);
|
2004-11-19 00:00:20 +01:00
|
|
|
number >>= 7; //47
|
|
|
|
if (number == 0) {
|
|
|
|
*pagePointer++ = tmp;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2009-05-01 19:21:36 +02:00
|
|
|
else
|
|
|
|
{
|
2004-11-19 00:00:20 +01:00
|
|
|
*pagePointer++ = tmp | 0x80;
|
2004-02-25 23:14:19 +01:00
|
|
|
tmp = (number & 0x7F);
|
2004-11-19 00:00:20 +01:00
|
|
|
number >>= 7; //54
|
|
|
|
if (number == 0) {
|
|
|
|
*pagePointer++ = tmp;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2009-05-01 19:21:36 +02:00
|
|
|
else
|
|
|
|
{
|
2004-11-19 00:00:20 +01:00
|
|
|
*pagePointer++ = tmp | 0x80;
|
2004-02-25 23:14:19 +01:00
|
|
|
tmp = (number & 0x7F);
|
2004-11-19 00:00:20 +01:00
|
|
|
number >>= 7; //61
|
|
|
|
if (number == 0) {
|
|
|
|
*pagePointer++ = tmp;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// ....
|
|
|
|
}
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2004-11-19 00:00:20 +01:00
|
|
|
*/
|
|
|
|
}
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
|
|
|
}
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
2004-02-25 23:14:19 +01:00
|
|
|
|
2009-05-01 19:21:36 +02:00
|
|
|
if (!leafNode)
|
|
|
|
{
|
2004-02-25 23:14:19 +01:00
|
|
|
// Store page number for non-leaf pages
|
|
|
|
number = indexNode->pageNumber;
|
|
|
|
if (number < 0) {
|
2004-09-24 02:11:32 +02:00
|
|
|
number = 0;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
|
|
|
tmp = (number & 0x7F);
|
|
|
|
number >>= 7;
|
|
|
|
if (number > 0) {
|
|
|
|
tmp |= 0x80;
|
|
|
|
}
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = tmp;
|
2009-05-01 19:21:36 +02:00
|
|
|
if (number > 0)
|
|
|
|
{
|
2004-02-25 23:14:19 +01:00
|
|
|
tmp = (number & 0x7F);
|
|
|
|
number >>= 7; //14
|
|
|
|
if (number > 0) {
|
|
|
|
tmp |= 0x80;
|
|
|
|
}
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = tmp;
|
2009-05-01 19:21:36 +02:00
|
|
|
if (number > 0)
|
|
|
|
{
|
2004-02-25 23:14:19 +01:00
|
|
|
tmp = (number & 0x7F);
|
|
|
|
number >>= 7; //21
|
|
|
|
if (number > 0) {
|
|
|
|
tmp |= 0x80;
|
|
|
|
}
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = tmp;
|
2009-05-01 19:21:36 +02:00
|
|
|
if (number > 0)
|
|
|
|
{
|
2004-02-25 23:14:19 +01:00
|
|
|
tmp = (number & 0x7F);
|
|
|
|
number >>= 7; //28
|
|
|
|
if (number > 0) {
|
|
|
|
tmp |= 0x80;
|
|
|
|
}
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = tmp;
|
2009-05-01 19:21:36 +02:00
|
|
|
if (number > 0)
|
|
|
|
{
|
2004-02-29 03:00:57 +01:00
|
|
|
tmp = (number & 0x0F);
|
|
|
|
number >>= 7; //35
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = tmp;
|
2004-02-29 03:00:57 +01:00
|
|
|
/*
|
|
|
|
Change number to 64-bit type and enable this for 64-bit support
|
2004-02-25 23:14:19 +01:00
|
|
|
tmp = (number & 0x7F);
|
|
|
|
number >>= 7; //35
|
|
|
|
if (number > 0) {
|
|
|
|
tmp |= 0x80;
|
|
|
|
}
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = tmp;
|
2009-05-01 19:21:36 +02:00
|
|
|
if (number > 0)
|
|
|
|
{
|
2004-02-25 23:14:19 +01:00
|
|
|
tmp = (number & 0x7F);
|
|
|
|
number >>= 7; //42
|
|
|
|
if (number > 0) {
|
|
|
|
tmp |= 0x80;
|
|
|
|
}
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = tmp;
|
2009-05-01 19:21:36 +02:00
|
|
|
if (number > 0)
|
|
|
|
{
|
2004-02-25 23:14:19 +01:00
|
|
|
tmp = (number & 0x7F);
|
|
|
|
number >>= 7; //49
|
|
|
|
if (number > 0) {
|
|
|
|
tmp |= 0x80;
|
|
|
|
}
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = tmp;
|
2009-05-01 19:21:36 +02:00
|
|
|
if (number > 0)
|
|
|
|
{
|
2004-02-25 23:14:19 +01:00
|
|
|
tmp = (number & 0x7F);
|
|
|
|
number >>= 7; //56
|
|
|
|
if (number > 0) {
|
|
|
|
tmp |= 0x80;
|
|
|
|
}
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = tmp;
|
2004-09-28 08:28:38 +02:00
|
|
|
if (number > 0) {
|
2004-02-25 23:14:19 +01:00
|
|
|
tmp = (number & 0x7F);
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = tmp;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2006-01-29 02:52:12 +01:00
|
|
|
}
|
|
|
|
*/
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
2004-02-25 23:14:19 +01:00
|
|
|
|
2009-05-01 19:21:36 +02:00
|
|
|
if (internalFlags != BTN_ZERO_PREFIX_ZERO_LENGTH_FLAG)
|
|
|
|
{
|
2004-02-25 23:14:19 +01:00
|
|
|
// Write prefix, maximum 14 bits
|
|
|
|
number = indexNode->prefix;
|
|
|
|
tmp = (number & 0x7F);
|
|
|
|
number >>= 7;
|
|
|
|
if (number > 0) {
|
|
|
|
tmp |= 0x80;
|
|
|
|
}
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = tmp;
|
2004-11-19 00:00:20 +01:00
|
|
|
if (number > 0) {
|
2004-02-25 23:14:19 +01:00
|
|
|
tmp = (number & 0x7F);
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = tmp;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2004-11-19 00:00:20 +01:00
|
|
|
}
|
2004-02-25 23:14:19 +01:00
|
|
|
|
2004-11-19 00:00:20 +01:00
|
|
|
if ((internalFlags != BTN_ZERO_PREFIX_ZERO_LENGTH_FLAG) &&
|
|
|
|
(internalFlags != BTN_ZERO_LENGTH_FLAG) &&
|
2008-12-05 02:20:14 +01:00
|
|
|
(internalFlags != BTN_ONE_LENGTH_FLAG))
|
2004-11-19 00:00:20 +01:00
|
|
|
{
|
2008-12-05 02:20:14 +01:00
|
|
|
// Write length, maximum 14 bits
|
2004-02-25 23:14:19 +01:00
|
|
|
number = indexNode->length;
|
|
|
|
tmp = (number & 0x7F);
|
|
|
|
number >>= 7;
|
|
|
|
if (number > 0) {
|
|
|
|
tmp |= 0x80;
|
|
|
|
}
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = tmp;
|
2004-11-19 00:00:20 +01:00
|
|
|
if (number > 0) {
|
2004-02-25 23:14:19 +01:00
|
|
|
tmp = (number & 0x7F);
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = tmp;
|
2004-02-25 23:14:19 +01:00
|
|
|
}
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
2004-02-25 23:14:19 +01:00
|
|
|
|
|
|
|
// Store data
|
|
|
|
if (withData) {
|
|
|
|
memcpy(pagePointer, indexNode->data, indexNode->length);
|
2008-12-05 02:20:14 +01:00
|
|
|
}
|
2004-02-25 23:14:19 +01:00
|
|
|
pagePointer += indexNode->length;
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
2009-05-01 19:21:36 +02:00
|
|
|
else
|
|
|
|
{
|
2008-12-05 02:20:14 +01:00
|
|
|
// Write prefix
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = (UCHAR)indexNode->prefix;
|
2008-12-05 02:20:14 +01:00
|
|
|
// Write length
|
2006-01-26 11:45:02 +01:00
|
|
|
*pagePointer++ = (UCHAR)indexNode->length;
|
|
|
|
|
2004-09-24 02:11:32 +02:00
|
|
|
if (indexNode->isEndLevel) {
|
2006-01-26 11:45:02 +01:00
|
|
|
put_long(pagePointer, END_LEVEL);
|
2004-09-24 02:11:32 +02:00
|
|
|
}
|
|
|
|
else if (indexNode->isEndBucket) {
|
2006-01-26 11:45:02 +01:00
|
|
|
put_long(pagePointer, END_BUCKET);
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
2009-05-01 19:21:36 +02:00
|
|
|
else
|
|
|
|
{
|
2004-09-24 02:11:32 +02:00
|
|
|
if (leafNode) {
|
|
|
|
// Write record number
|
2006-01-26 11:45:02 +01:00
|
|
|
put_long(pagePointer, indexNode->recordNumber.getValue());
|
2004-09-24 02:11:32 +02:00
|
|
|
}
|
|
|
|
else {
|
2008-12-05 02:20:14 +01:00
|
|
|
// Write page number
|
2006-01-26 11:45:02 +01:00
|
|
|
put_long(pagePointer, indexNode->pageNumber);
|
2004-09-24 02:11:32 +02:00
|
|
|
}
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
2008-12-05 02:20:14 +01:00
|
|
|
pagePointer += sizeof(SLONG);
|
2004-02-25 23:14:19 +01:00
|
|
|
|
2009-05-01 19:21:36 +02:00
|
|
|
if (withData)
|
|
|
|
{
|
2004-02-25 23:14:19 +01:00
|
|
|
USHORT size = indexNode->length;
|
|
|
|
const UCHAR* ptr = indexNode->data;
|
|
|
|
while (size) {
|
|
|
|
*pagePointer++ = *ptr++;
|
|
|
|
size--;
|
|
|
|
}
|
2008-12-05 02:20:14 +01:00
|
|
|
}
|
2003-12-01 11:25:41 +01:00
|
|
|
else {
|
2004-02-25 23:14:19 +01:00
|
|
|
pagePointer += indexNode->length;
|
|
|
|
}
|
|
|
|
|
2008-12-05 02:20:14 +01:00
|
|
|
if ((flags & btr_all_record_number) &&
|
2004-09-24 02:11:32 +02:00
|
|
|
(!leafNode ||
|
2008-12-05 02:20:14 +01:00
|
|
|
(leafNode && indexNode->isEndBucket && (indexNode->length == 0))))
|
2004-09-24 02:11:32 +02:00
|
|
|
{
|
2008-12-05 02:20:14 +01:00
|
|
|
// Write record number
|
2006-01-26 11:45:02 +01:00
|
|
|
put_long(pagePointer, indexNode->recordNumber.getValue());
|
2004-02-25 23:14:19 +01:00
|
|
|
pagePointer += sizeof(SLONG);
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return pagePointer;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-05-01 19:21:36 +02:00
|
|
|
void setEndBucket(IndexNode* indexNode) //, bool leafNode)
|
2003-12-01 11:25:41 +01:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e t E n d B u c k e t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-07 22:50:53 +01:00
|
|
|
indexNode->isEndBucket = true;
|
|
|
|
indexNode->isEndLevel = false;
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-05-01 19:21:36 +02:00
|
|
|
void setEndLevel(IndexNode* indexNode) //, bool leafNode)
|
2003-12-01 11:25:41 +01:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e t E n d L e v e l
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-07 22:50:53 +01:00
|
|
|
indexNode->isEndBucket = false;
|
|
|
|
indexNode->isEndLevel = true;
|
2003-12-01 11:25:41 +01:00
|
|
|
indexNode->prefix = 0;
|
|
|
|
indexNode->length = 0;
|
2004-09-24 02:11:32 +02:00
|
|
|
indexNode->pageNumber = 0;
|
2004-09-28 08:28:38 +02:00
|
|
|
indexNode->recordNumber.setValue(0);
|
2004-09-24 02:11:32 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-12-05 02:20:14 +01:00
|
|
|
void setNode(IndexNode* indexNode, USHORT prefix, USHORT length,
|
2004-09-28 08:28:38 +02:00
|
|
|
RecordNumber recordNumber, SLONG pageNumber,
|
2004-09-24 02:11:32 +02:00
|
|
|
bool isEndBucket, bool isEndLevel)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e t N o d e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
indexNode->isEndBucket = isEndBucket;
|
|
|
|
indexNode->isEndLevel = isEndLevel;
|
|
|
|
indexNode->prefix = prefix;
|
|
|
|
indexNode->length = length;
|
|
|
|
indexNode->recordNumber = recordNumber;
|
|
|
|
indexNode->pageNumber = pageNumber;
|
2003-12-01 11:25:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
} // namespace BTreeNode
|