8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-27 18:03:04 +01:00
firebird-mirror/src/jrd/btr.h
hvlad 714e6c2eaf Remove LCK_idx_reserve introduced with 64-bit transaction ID.
Restore original logic in BTR_next_index (wait for tx lock when index build is in progress). 
Use irt_repeat::irt_root to store high 32 bit of tra_num. 
It also fixed deadlock related with LCK_idx_reserve lock (reported privately).
2016-01-27 11:11:38 +00:00

303 lines
9.2 KiB
C++

/*
* PROGRAM: JRD Access Method
* MODULE: btr.h
* DESCRIPTION: Index walking data structures
*
* The contents of this file are subject to the Interbase 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.Inprise.com/IPL.html
*
* Software distributed under the License is distributed on an
* "AS IS" basis, 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 Inprise Corporation
* and its predecessors. Portions created by Inprise Corporation are
* Copyright (C) Inprise Corporation.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
*
* 2002.10.28 Sean Leyne - Code cleanup, removed obsolete "DecOSF" port
*
*/
#ifndef JRD_BTR_H
#define JRD_BTR_H
#include "../jrd/constants.h"
#include "../common/classes/array.h"
#include "../include/fb_blk.h"
#include "../jrd/err_proto.h" // Index error types
#include "../jrd/RecordNumber.h"
#include "../jrd/sbm.h"
#include "../jrd/lck.h"
struct dsc;
namespace Jrd {
class jrd_rel;
class jrd_tra;
template <typename T> class vec;
class JrdStatement;
struct temporary_key;
class jrd_tra;
class BtrPageGCLock;
class Sort;
// Index descriptor block -- used to hold info from index root page
struct index_desc
{
ULONG idx_root; // Index root
float idx_selectivity; // selectivity of index
USHORT idx_id;
UCHAR idx_flags;
UCHAR idx_runtime_flags; // flags used at runtime, not stored on disk
USHORT idx_primary_index; // id for primary key partner index
USHORT idx_primary_relation; // id for primary key partner relation
USHORT idx_count; // number of keys
vec<int>* idx_foreign_primaries; // ids for primary/unique indexes with partners
vec<int>* idx_foreign_relations; // ids for foreign key partner relations
vec<int>* idx_foreign_indexes; // ids for foreign key partner indexes
ValueExprNode* idx_expression; // node tree for indexed expresssion
dsc idx_expression_desc; // descriptor for expression result
JrdStatement* idx_expression_statement; // stored statement for expression evaluation
// This structure should exactly match IRTD structure for current ODS
struct idx_repeat
{
USHORT idx_field; // field id
USHORT idx_itype; // data of field in index
float idx_selectivity; // segment selectivity
} idx_rpt[MAX_INDEX_SEGMENTS];
};
struct IndexDescAlloc : public pool_alloc_rpt<index_desc>
{
index_desc items[1];
};
const USHORT idx_invalid = USHORT(~0); // Applies to idx_id as special value
// index types and flags
// See jrd/intl.h for notes on idx_itype and dsc_sub_type considerations
// idx_numeric .. idx_byte_array values are compatible with VMS values
const int idx_numeric = 0;
const int idx_string = 1;
// value of 2 was used in ODS < 10
const int idx_byte_array = 3;
const int idx_metadata = 4;
const int idx_sql_date = 5;
const int idx_sql_time = 6;
const int idx_timestamp = 7;
const int idx_numeric2 = 8; // Introduced for 64-bit Integer support
const int idx_boolean = 9;
// idx_itype space for future expansion
const int idx_first_intl_string = 64; // .. MAX (short) Range of computed key strings
const int idx_offset_intl_range = (0x7FFF + idx_first_intl_string);
// these flags must match the irt_flags (see ods.h)
const int idx_unique = 1;
const int idx_descending = 2;
const int idx_in_progress = 4;
const int idx_foreign = 8;
const int idx_primary = 16;
const int idx_expressn = 32;
// these flags are for idx_runtime_flags
const int idx_plan_dont_use = 1; // index is not mentioned in user-specified access plan
const int idx_plan_navigate = 2; // plan specifies index to be used for ordering
const int idx_used = 4; // index was in fact selected for retrieval
const int idx_navigate = 8; // index was in fact selected for navigation
const int idx_marker = 16; // marker used in procedure sort_indices
// Index insertion block -- parameter block for index insertions
struct index_insertion
{
RecordNumber iib_number; // record number (or lower level page)
ULONG iib_sibling; // right sibling page
index_desc* iib_descriptor; // index descriptor
jrd_rel* iib_relation; // relation block
temporary_key* iib_key; // varying string for insertion
RecordBitmap* iib_duplicates; // spare bit map of duplicates
jrd_tra* iib_transaction; // insertion transaction
BtrPageGCLock* iib_dont_gc_lock; // lock to prevent removal of splitted page
UCHAR iib_btr_level; // target level to propagate split page to
};
// these flags are for the key_flags
const int key_empty = 1; // Key contains empty data / empty string
// Temporary key block
struct temporary_key
{
USHORT key_length;
UCHAR key_data[MAX_KEY + 1];
UCHAR key_flags;
USHORT key_nulls; // bitmap of encountered null segments,
// USHORT is enough to store MAX_INDEX_SEGMENTS bits
};
// Index Sort Record -- fix part of sort record for index fast load
// hvlad: index_sort_record structure is stored in sort scratch file so we
// don't want to grow sort file with padding added by compiler to please
// alignment rules.
#pragma pack(1)
struct index_sort_record
{
// RecordNumber should be at the first place, because it's used
// for determing sort by creating index (see idx.cpp)
SINT64 isr_record_number;
USHORT isr_key_length;
USHORT isr_flags;
};
#pragma pack()
const int ISR_secondary = 1; // Record is secondary version
const int ISR_null = 2; // Record consists of NULL values only
// Index retrieval block -- hold stuff for index retrieval
class IndexRetrieval
{
public:
IndexRetrieval(jrd_rel* relation, const index_desc* idx, USHORT count, temporary_key* key)
: irb_relation(relation), irb_index(idx->idx_id),
irb_generic(0), irb_lower_count(count), irb_upper_count(count), irb_key(key),
irb_name(NULL), irb_value(NULL)
{
memcpy(&irb_desc, idx, sizeof(irb_desc));
}
IndexRetrieval(MemoryPool& pool, jrd_rel* relation, const index_desc* idx,
const Firebird::MetaName& name)
: irb_relation(relation), irb_index(idx->idx_id),
irb_generic(0), irb_lower_count(0), irb_upper_count(0), irb_key(NULL),
irb_name(FB_NEW_POOL(pool) Firebird::MetaName(name)),
irb_value(FB_NEW_POOL(pool) ValueExprNode*[idx->idx_count * 2])
{
memcpy(&irb_desc, idx, sizeof(irb_desc));
}
~IndexRetrieval()
{
delete irb_name;
delete[] irb_value;
}
index_desc irb_desc; // Index descriptor
jrd_rel* irb_relation; // Relation for retrieval
USHORT irb_index; // Index id
USHORT irb_generic; // Flags for generic search
USHORT irb_lower_count; // Number of segments for retrieval
USHORT irb_upper_count; // Number of segments for retrieval
temporary_key* irb_key; // Key for equality retrieval
Firebird::MetaName* irb_name; // Index name
ValueExprNode** irb_value;
};
// Flag values for irb_generic
const int irb_partial = 1; // Partial match: not all segments or starting of key only
const int irb_starting = 2; // Only compute "starting with" key for index segment
const int irb_equality = 4; // Probing index for equality match
const int irb_ignore_null_value_key = 8; // if lower bound is specified and upper bound unspecified,
// ignore looking at null value keys
const int irb_descending = 16; // Base index uses descending order
const int irb_exclude_lower = 32; // exclude lower bound keys while scanning index
const int irb_exclude_upper = 64; // exclude upper bound keys while scanning index
typedef Firebird::HalfStaticArray<float, 4> SelectivityList;
class BtrPageGCLock : public Lock
{
// We want to put 8 bytes (PageNumber) in lock key. One long is already
// reserved by Lock::lck_long, this is the second long. It is really unused
// as second long needed for 8-byte key already "allocated" by compiler
// because of alignment rules. Anyway, to be formally correct, let introduce
// 4-byte field for guarantee we have space for lock key.
#ifdef __clang__
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wunused-private-field"
#endif
ULONG unused;
#ifdef __clang__
#pragma clang diagnostic pop
#endif
public:
explicit BtrPageGCLock(thread_db* tdbb);
~BtrPageGCLock();
void disablePageGC(thread_db* tdbb, const PageNumber &page);
void enablePageGC(thread_db* tdbb);
static bool isPageGCAllowed(thread_db* tdbb, const PageNumber& page);
};
// Struct used for index creation
struct IndexCreation
{
jrd_rel* relation;
index_desc* index;
jrd_tra* transaction;
USHORT key_length;
Firebird::AutoPtr<Sort> sort;
};
// Class used to report any index related errors
class IndexErrorContext
{
struct Location
{
jrd_rel* relation;
USHORT indexId;
};
public:
IndexErrorContext(jrd_rel* relation, index_desc* index, const char* indexName = NULL)
: m_relation(relation), m_index(index), m_indexName(indexName), isLocationDefined(false)
{}
void setErrorLocation(jrd_rel* relation, USHORT indexId)
{
isLocationDefined = true;
m_location.relation = relation;
m_location.indexId = indexId;
}
void raise(thread_db*, idx_e, Record*);
private:
jrd_rel* const m_relation;
index_desc* const m_index;
const char* const m_indexName;
Location m_location;
bool isLocationDefined;
};
} //namespace Jrd
#endif // JRD_BTR_H