8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-27 17:23:03 +01:00
firebird-mirror/src/jrd/btn.cpp
dimitr 05c27aa212 Integrated the jumping information natively into the index bucket.
WARNING: This is an ODS change, databases must be recreated.
2012-03-14 14:33:41 +00:00

632 lines
13 KiB
C++

/*
* 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/ods.h"
#include "../jrd/btn.h"
using namespace Jrd;
using namespace Ods;
USHORT IndexNode::computePrefix(const UCHAR* prevString, USHORT prevLength,
const UCHAR* string, USHORT length)
{
/**************************************
*
* c o m p u t e P r e f i x
*
**************************************
*
* Functional description
* Compute and return prefix common
* to two strings.
*
**************************************/
USHORT l = MIN(prevLength, length);
if (!l) {
return 0;
}
const UCHAR* p = prevString;
while (*p == *string)
{
++p;
++string;
if (!--l) {
break;
}
}
return (p - prevString);
}
SLONG IndexNode::findPageInDuplicates(const btree_page* page, UCHAR* pointer,
SLONG previousNumber, RecordNumber findRecordNumber)
{
/**************************************
*
* 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
*
**************************************/
const bool leafPage = (page->btr_level == 0);
IndexNode node, previousNode;
pointer = node.readNode(pointer, leafPage);
while (true)
{
// loop through duplicates until
// correct node is found.
// If this is an end bucket marker then return
// the previous passed page number.
if (node.isEndBucket) {
return previousNumber;
}
if (findRecordNumber <= node.recordNumber) {
// If first record number on page is higher
// then record number must be at the previous
// passed page number.
return previousNumber;
}
// Save current page number and fetch next node
// for comparision.
previousNumber = node.pageNumber;
previousNode = node;
pointer = node.readNode(pointer, leafPage);
// Check if pointer is still valid
//if (pointer > endPointer) {
// BUGCHECK(204); // msg 204 index inconsistent
//}
// We're done if end level marker is reached or this
// isn't a equal node anymore.
if ((node.isEndLevel) ||
(node.length != 0) ||
(node.prefix != (previousNode.length + previousNode.prefix)))
{
return previousNumber;
}
}
// We never reach this point
}
USHORT IndexJumpNode::getJumpNodeSize() const
{
/**************************************
*
* g e t J u m p N o d e S i z e
*
**************************************
*
* Functional description
* Return the size needed to store
* this node.
*
**************************************/
USHORT result = 0;
// Size needed for prefix
USHORT number = prefix;
if (number & 0xC000) {
result += 3;
}
else if (number & 0xFF80) {
result += 2;
}
else {
result += 1;
}
// Size needed for length
number = length;
if (number & 0xC000) {
result += 3;
}
else if (number & 0xFF80) {
result += 2;
}
else {
result += 1;
}
// Size needed for offset
// NOTE! offset can be unknown when this function is called,
// therefor we can't use a compression method.
result += sizeof(USHORT);
// Size needed for data
result += length;
return result;
}
USHORT IndexNode::getNodeSize(bool leafNode) const
{
/**************************************
*
* g e t N o d e S i z e
*
**************************************
*
* Functional description
* Return the size needed to store
* this node.
*
**************************************/
USHORT result = 0;
// Determine flags
UCHAR internalFlags = 0;
if (isEndLevel) {
internalFlags = BTN_END_LEVEL_FLAG;
}
else if (isEndBucket) {
internalFlags = BTN_END_BUCKET_FLAG;
}
else if (length == 0)
{
if (prefix == 0) {
internalFlags = BTN_ZERO_PREFIX_ZERO_LENGTH_FLAG;
}
else {
internalFlags = BTN_ZERO_LENGTH_FLAG;
}
}
else if (length == 1) {
internalFlags = BTN_ONE_LENGTH_FLAG;
}
// Store internal flags + 5 bits from number
SINT64 number = recordNumber.getValue();
if (number < 0) {
number = 0;
}
result++;
// If this is a END_LEVEL marker then we're done
if (isEndLevel) {
return result;
}
number >>= 5;
// Get size for storing remaining bits for number
// 5 bytes should be enough to fit remaining 34 bits of record number
if (number & QUADCONST(0xFFF0000000)) {
result += 5;
}
else if (number & QUADCONST(0xFFFFE00000)) {
result += 4;
}
else if (number & QUADCONST(0xFFFFFFC000)) {
result += 3;
}
else if (number & QUADCONST(0xFFFFFFFF80)) {
result += 2;
}
else {
result += 1;
}
if (!leafNode)
{
// Size needed for page number
number = pageNumber;
if (number < 0) {
number = 0;
}
if (number & 0xF0000000) {
result += 5;
}
else if (number & 0xFFE00000) {
result += 4;
}
else if (number & 0xFFFFC000) {
result += 3;
}
else if (number & 0xFFFFFF80) {
result += 2;
}
else {
result += 1;
}
}
if (internalFlags != BTN_ZERO_PREFIX_ZERO_LENGTH_FLAG)
{
// Size needed for prefix
number = prefix;
if (number & 0xFFFFC000) {
result += 3;
}
else if (number & 0xFFFFFF80) {
result += 2;
}
else {
result += 1;
}
}
if ((internalFlags != BTN_ZERO_PREFIX_ZERO_LENGTH_FLAG) &&
(internalFlags != BTN_ZERO_LENGTH_FLAG) &&
(internalFlags != BTN_ONE_LENGTH_FLAG))
{
// Size needed for length
number = length;
if (number & 0xFFFFC000) {
result += 3;
}
else if (number & 0xFFFFFF80) {
result += 2;
}
else {
result += 1;
}
}
result += length;
return result;
}
UCHAR* IndexJumpNode::readJumpNode(UCHAR* pagePointer)
{
/**************************************
*
* 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.
*
**************************************/
nodePointer = pagePointer;
// Get prefix
UCHAR tmp = *pagePointer++;
prefix = (tmp & 0x7F);
if (tmp & 0x80)
{
tmp = *pagePointer++;
prefix |= (tmp & 0x7F) << 7; // We get 14 bits at this point
}
// Get length
tmp = *pagePointer++;
length = (tmp & 0x7F);
if (tmp & 0x80)
{
tmp = *pagePointer++;
length |= (tmp & 0x7F) << 7; // We get 14 bits at this point
}
offset = get_short(pagePointer);
pagePointer += sizeof(USHORT);
data = pagePointer;
pagePointer += length;
return pagePointer;
}
UCHAR* IndexJumpNode::writeJumpNode(UCHAR* pagePointer)
{
/**************************************
*
* 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.
*
**************************************/
nodePointer = pagePointer;
// Write prefix, maximum 14 bits
USHORT number = prefix;
UCHAR tmp = (number & 0x7F);
number >>= 7;
if (number > 0) {
tmp |= 0x80;
}
*pagePointer++ = tmp;
if (tmp & 0x80)
{
tmp = (number & 0x7F);
*pagePointer++ = tmp;
}
// Write length, maximum 14 bits
number = length;
tmp = (number & 0x7F);
number >>= 7;
if (number > 0) {
tmp |= 0x80;
}
*pagePointer++ = tmp;
if (tmp & 0x80)
{
tmp = (number & 0x7F);
*pagePointer++ = tmp;
}
put_short(pagePointer, offset);
pagePointer += sizeof(USHORT);
memmove(pagePointer, data, length);
pagePointer += length;
return pagePointer;
}
UCHAR* IndexNode::writeNode(UCHAR* pagePointer, 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.
*
**************************************/
nodePointer = pagePointer;
// 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.
// Also for duplicate node entries (length and prefix
// are zero) we don't store the length and prefix
// information. This will save at least 2 bytes per node.
if (!withData)
{
// 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(leafNode) - length;
pagePointer += offset; // set pointer to right position
memmove(pagePointer, data, length);
pagePointer -= offset; // restore pointer to original position
}
// Internal flags
UCHAR internalFlags = 0;
if (isEndLevel) {
internalFlags = BTN_END_LEVEL_FLAG;
}
else if (isEndBucket) {
internalFlags = BTN_END_BUCKET_FLAG;
}
else if (length == 0)
{
if (prefix == 0) {
internalFlags = BTN_ZERO_PREFIX_ZERO_LENGTH_FLAG;
}
else {
internalFlags = BTN_ZERO_LENGTH_FLAG;
}
}
else if (length == 1) {
internalFlags = BTN_ONE_LENGTH_FLAG;
}
SINT64 number = recordNumber.getValue();
if (number < 0) {
number = 0;
}
// Store internal flags + 6 bits from number
UCHAR tmp = internalFlags;
*pagePointer++ = ((tmp << 5) | (number & 0x1F));
if (isEndLevel) {
return pagePointer;
}
// Store remaining bits from number
number >>= 5;
tmp = (number & 0x7F);
number >>= 7; //12
if (number == 0) {
*pagePointer++ = tmp;
}
else
{
*pagePointer++ = tmp | 0x80;
tmp = (number & 0x7F);
number >>= 7; //19
if (number == 0) {
*pagePointer++ = tmp;
}
else
{
*pagePointer++ = tmp | 0x80;
tmp = (number & 0x7F);
number >>= 7; //26
if (number == 0) {
*pagePointer++ = tmp;
}
else
{
*pagePointer++ = tmp | 0x80;
tmp = (number & 0x7F);
number >>= 7; //33
if (number == 0) {
*pagePointer++ = tmp;
}
else
{
*pagePointer++ = tmp | 0x80;
tmp = (number & 0x7F);
number >>= 7; //40
if (number == 0) {
*pagePointer++ = tmp;
}
/*
Enable this if you need more bits in record number
else
{
*pagePointer++ = tmp | 0x80;
tmp = (number & 0x7F);
number >>= 7; //47
if (number == 0) {
*pagePointer++ = tmp;
}
else
{
*pagePointer++ = tmp | 0x80;
tmp = (number & 0x7F);
number >>= 7; //54
if (number == 0) {
*pagePointer++ = tmp;
}
else
{
*pagePointer++ = tmp | 0x80;
tmp = (number & 0x7F);
number >>= 7; //61
if (number == 0) {
*pagePointer++ = tmp;
}
else {
// ....
}
}
}
}
*/
}
}
}
}
if (!leafNode)
{
// Store page number for non-leaf pages
number = pageNumber;
if (number < 0) {
number = 0;
}
tmp = (number & 0x7F);
number >>= 7;
if (number > 0) {
tmp |= 0x80;
}
*pagePointer++ = tmp;
if (number > 0)
{
tmp = (number & 0x7F);
number >>= 7; //14
if (number > 0) {
tmp |= 0x80;
}
*pagePointer++ = tmp;
if (number > 0)
{
tmp = (number & 0x7F);
number >>= 7; //21
if (number > 0) {
tmp |= 0x80;
}
*pagePointer++ = tmp;
if (number > 0)
{
tmp = (number & 0x7F);
number >>= 7; //28
if (number > 0) {
tmp |= 0x80;
}
*pagePointer++ = tmp;
if (number > 0)
{
tmp = (number & 0x0F);
number >>= 7; //35
*pagePointer++ = tmp;
}
}
}
}
}
if (internalFlags != BTN_ZERO_PREFIX_ZERO_LENGTH_FLAG)
{
// Write prefix, maximum 14 bits
number = prefix;
tmp = (number & 0x7F);
number >>= 7;
if (number > 0) {
tmp |= 0x80;
}
*pagePointer++ = tmp;
if (number > 0)
{
tmp = (number & 0x7F);
*pagePointer++ = tmp;
}
}
if ((internalFlags != BTN_ZERO_PREFIX_ZERO_LENGTH_FLAG) &&
(internalFlags != BTN_ZERO_LENGTH_FLAG) &&
(internalFlags != BTN_ONE_LENGTH_FLAG))
{
// Write length, maximum 14 bits
number = length;
tmp = (number & 0x7F);
number >>= 7;
if (number > 0) {
tmp |= 0x80;
}
*pagePointer++ = tmp;
if (number > 0)
{
tmp = (number & 0x7F);
*pagePointer++ = tmp;
}
}
// Store data
if (withData) {
memcpy(pagePointer, data, length);
}
pagePointer += length;
return pagePointer;
}