2001-05-23 15:26:42 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: JRD Access Method
|
2003-11-07 09:06:35 +01:00
|
|
|
* MODULE: idx.cpp
|
2001-05-23 15:26:42 +02:00
|
|
|
* DESCRIPTION: Index manager
|
|
|
|
*
|
|
|
|
* 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): ______________________________________.
|
2003-03-05 12:23:08 +01:00
|
|
|
*
|
|
|
|
* 2003.03.04 Dmitry Yemanov: Added support for NULLs in unique indices.
|
|
|
|
* Done in two stages:
|
|
|
|
* 1. Restored old behaviour of having only _one_
|
|
|
|
* NULL key allowed (i.e. two NULLs are considered
|
|
|
|
* duplicates). idx_e_nullunique error was removed.
|
|
|
|
* 2. Changed algorithms in IDX_create_index() and
|
|
|
|
* check_duplicates() to ignore NULL key duplicates.
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
|
|
|
|
2001-07-29 19:42:23 +02:00
|
|
|
#include "firebird.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include <string.h>
|
2004-03-22 12:38:23 +01:00
|
|
|
#include "../jrd/common.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/jrd.h"
|
|
|
|
#include "../jrd/val.h"
|
|
|
|
#include "../jrd/intl.h"
|
|
|
|
#include "../jrd/req.h"
|
|
|
|
#include "../jrd/ods.h"
|
|
|
|
#include "../jrd/btr.h"
|
|
|
|
#include "../jrd/sort.h"
|
|
|
|
#include "../jrd/lls.h"
|
|
|
|
#include "../jrd/tra.h"
|
2003-11-11 13:19:20 +01:00
|
|
|
#include "gen/iberror.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/sbm.h"
|
|
|
|
#include "../jrd/exe.h"
|
|
|
|
#include "../jrd/scl.h"
|
|
|
|
#include "../jrd/lck.h"
|
|
|
|
#include "../jrd/rse.h"
|
|
|
|
#include "../jrd/cch.h"
|
|
|
|
#include "../jrd/gdsassert.h"
|
|
|
|
#include "../jrd/all_proto.h"
|
|
|
|
#include "../jrd/btr_proto.h"
|
|
|
|
#include "../jrd/cch_proto.h"
|
|
|
|
#include "../jrd/cmp_proto.h"
|
|
|
|
#include "../jrd/dpm_proto.h"
|
|
|
|
#include "../jrd/err_proto.h"
|
|
|
|
#include "../jrd/evl_proto.h"
|
|
|
|
#include "../jrd/gds_proto.h"
|
|
|
|
#include "../jrd/idx_proto.h"
|
|
|
|
#include "../jrd/jrd_proto.h"
|
|
|
|
#include "../jrd/lck_proto.h"
|
|
|
|
#include "../jrd/met_proto.h"
|
|
|
|
#include "../jrd/mov_proto.h"
|
|
|
|
#include "../jrd/sort_proto.h"
|
2004-06-08 15:41:08 +02:00
|
|
|
#include "../jrd/thd.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/vio_proto.h"
|
|
|
|
|
|
|
|
|
2004-03-20 16:33:30 +01:00
|
|
|
using namespace Jrd;
|
|
|
|
typedef Ods::index_root_page index_root_page;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-02-13 11:11:35 +01:00
|
|
|
static const SCHAR NULL_STR = '\0';
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Data to be passed to index fast load duplicates routine */
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
struct index_fast_load {
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG ifl_duplicates;
|
|
|
|
USHORT ifl_key_length;
|
2004-03-28 11:10:30 +02:00
|
|
|
};
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static IDX_E check_duplicates(thread_db*, Record*, index_desc*, index_insertion*, jrd_rel*);
|
|
|
|
static IDX_E check_foreign_key(thread_db*, Record*, jrd_rel*, jrd_tra*, index_desc*, jrd_rel**, USHORT *);
|
|
|
|
static IDX_E check_partner_index(thread_db*, jrd_rel*, Record*, jrd_tra*, index_desc*, jrd_rel*, SSHORT);
|
2003-12-11 11:33:30 +01:00
|
|
|
static bool duplicate_key(const UCHAR*, const UCHAR*, void*);
|
2004-03-28 11:10:30 +02:00
|
|
|
static SLONG get_root_page(thread_db*, const jrd_rel*);
|
2004-03-18 06:56:06 +01:00
|
|
|
static int index_block_flush(void*);
|
2004-03-28 11:10:30 +02:00
|
|
|
static IDX_E insert_key(thread_db*, jrd_rel*, Record*, jrd_tra*, WIN *, index_insertion*, jrd_rel**, USHORT *);
|
|
|
|
static bool key_equal(const temporary_key*, const temporary_key*);
|
2004-03-11 06:04:26 +01:00
|
|
|
static void signal_index_deletion(thread_db*, jrd_rel*, USHORT);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
void IDX_check_access(thread_db* tdbb, CompilerScratch* csb, jrd_rel* view, jrd_rel* relation,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_fld* field)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I D X _ c h e c k _ a c c e s s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Check the various indices in a relation
|
|
|
|
* to see if we need REFERENCES access to fields
|
|
|
|
* in the primary key. Don't call this routine for
|
|
|
|
* views or external relations, since the mechanism
|
|
|
|
* ain't there.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
index_desc idx;
|
2003-08-18 23:13:56 +02:00
|
|
|
idx.idx_id = (USHORT) -1;
|
2003-12-11 11:33:30 +01:00
|
|
|
WIN window(-1);
|
|
|
|
WIN referenced_window(-1);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
while (BTR_next_index(tdbb, relation, 0, &idx, &window))
|
|
|
|
if (idx.idx_flags & idx_foreign) {
|
|
|
|
/* find the corresponding primary key index */
|
|
|
|
|
2003-12-11 11:33:30 +01:00
|
|
|
if (!MET_lookup_partner(tdbb, relation, &idx, &NULL_STR)) {
|
|
|
|
continue;
|
|
|
|
}
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_rel* referenced_relation =
|
2001-05-23 15:26:42 +02:00
|
|
|
MET_relation(tdbb, idx.idx_primary_relation);
|
|
|
|
MET_scan_relation(tdbb, referenced_relation);
|
2003-12-22 11:00:59 +01:00
|
|
|
const USHORT index_id = (USHORT) idx.idx_primary_index;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* get the description of the primary key index */
|
|
|
|
|
|
|
|
referenced_window.win_page =
|
|
|
|
get_root_page(tdbb, referenced_relation);
|
|
|
|
referenced_window.win_flags = 0;
|
2004-02-20 07:43:27 +01:00
|
|
|
index_root_page* referenced_root =
|
|
|
|
(index_root_page*) CCH_FETCH(tdbb, &referenced_window, LCK_read, pag_root);
|
2004-03-28 11:10:30 +02:00
|
|
|
index_desc referenced_idx;
|
2004-08-19 20:41:19 +02:00
|
|
|
if (!BTR_description(tdbb, referenced_relation, referenced_root,
|
|
|
|
&referenced_idx, index_id))
|
2003-12-11 11:33:30 +01:00
|
|
|
{
|
|
|
|
BUGCHECK(173); /* msg 173 referenced index description not found */
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* post references access to each field in the index */
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
const index_desc::idx_repeat* idx_desc = referenced_idx.idx_rpt;
|
2003-12-22 11:00:59 +01:00
|
|
|
for (USHORT i = 0; i < referenced_idx.idx_count; i++, idx_desc++) {
|
|
|
|
const jrd_fld* referenced_field =
|
2001-05-23 15:26:42 +02:00
|
|
|
MET_get_field(referenced_relation, idx_desc->idx_field);
|
|
|
|
CMP_post_access(tdbb, csb,
|
2003-08-14 22:45:36 +02:00
|
|
|
referenced_relation->rel_security_name,
|
|
|
|
(view ? view->rel_id : 0),
|
2004-04-18 04:50:38 +02:00
|
|
|
SCL_sql_references, "TABLE",
|
2001-05-23 15:26:42 +02:00
|
|
|
referenced_relation->rel_name);
|
|
|
|
CMP_post_access(tdbb, csb,
|
2004-04-18 04:50:38 +02:00
|
|
|
referenced_field->fld_security_name, 0,
|
2001-05-23 15:26:42 +02:00
|
|
|
SCL_sql_references, "COLUMN",
|
|
|
|
referenced_field->fld_name);
|
|
|
|
}
|
|
|
|
|
|
|
|
CCH_RELEASE(tdbb, &referenced_window);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void IDX_create_index(
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_rel* relation,
|
2004-03-28 11:10:30 +02:00
|
|
|
index_desc* idx,
|
2003-12-11 11:33:30 +01:00
|
|
|
const TEXT* index_name,
|
2003-12-03 09:19:24 +01:00
|
|
|
USHORT* index_id,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_tra* transaction,
|
2003-11-30 22:14:30 +01:00
|
|
|
SelectivityList& selectivity)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I D X _ c r e a t e _ i n d e x
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Create and populate index.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
IDX_E result = idx_e_ok;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
if (relation->rel_file) {
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_no_meta_update, isc_arg_gds, isc_extfile_uns_op,
|
|
|
|
isc_arg_string, ERR_cstring(relation->rel_name), 0);
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
BTR_reserve_slot(tdbb, relation, transaction, idx);
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if (index_id) {
|
2001-05-23 15:26:42 +02:00
|
|
|
*index_id = idx->idx_id;
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
record_param primary, secondary;
|
2001-12-24 03:51:06 +01:00
|
|
|
secondary.rpb_relation = relation;
|
|
|
|
primary.rpb_relation = relation;
|
2004-09-28 08:28:38 +02:00
|
|
|
primary.rpb_number.setValue(BOF_NUMBER);
|
2004-03-11 06:04:26 +01:00
|
|
|
//primary.rpb_window.win_flags = secondary.rpb_window.win_flags = 0; redundant
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
const USHORT key_length = ROUNDUP(BTR_key_length(relation, idx), sizeof(SLONG));
|
2003-12-01 03:37:25 +01:00
|
|
|
|
|
|
|
USHORT max_key_size = MAX_KEY_LIMIT;
|
|
|
|
if (dbb->dbb_ods_version < ODS_VERSION11) {
|
|
|
|
max_key_size = MAX_KEY_PRE_ODS11;
|
|
|
|
}
|
|
|
|
if (key_length >= max_key_size) {
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_no_meta_update,
|
|
|
|
isc_arg_gds,
|
|
|
|
isc_keytoobig,
|
|
|
|
isc_arg_string,
|
2003-12-11 11:33:30 +01:00
|
|
|
ERR_cstring(index_name), 0);
|
2003-12-01 03:37:25 +01:00
|
|
|
}
|
|
|
|
|
2004-04-18 16:22:27 +02:00
|
|
|
RecordStack stack;
|
2003-12-22 11:00:59 +01:00
|
|
|
const UCHAR pad = (idx->idx_flags & idx_descending) ? -1 : 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-21 08:16:17 +02:00
|
|
|
index_fast_load ifl_data;
|
2001-05-23 15:26:42 +02:00
|
|
|
ifl_data.ifl_duplicates = 0;
|
|
|
|
ifl_data.ifl_key_length = key_length;
|
|
|
|
|
2003-09-28 16:03:10 +02:00
|
|
|
bool key_is_null = false;
|
2003-03-05 12:23:08 +01:00
|
|
|
|
2004-10-05 23:09:51 +02:00
|
|
|
sort_key_def key_desc[2];
|
|
|
|
// Key sort description
|
|
|
|
key_desc[0].skd_dtype = SKD_bytes;
|
|
|
|
key_desc[0].skd_flags = SKD_ascending;
|
|
|
|
key_desc[0].skd_length = key_length;
|
|
|
|
key_desc[0].skd_offset = 0;
|
|
|
|
key_desc[0].skd_vary_offset = 0;
|
|
|
|
// RecordNumber sort description
|
|
|
|
key_desc[1].skd_dtype = SKD_int64;
|
|
|
|
key_desc[1].skd_flags = SKD_ascending;
|
|
|
|
key_desc[1].skd_length = sizeof(RecordNumber);
|
2004-10-06 13:44:03 +02:00
|
|
|
key_desc[1].skd_offset = key_length;
|
2004-10-05 23:09:51 +02:00
|
|
|
key_desc[1].skd_vary_offset = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-11 11:33:30 +01:00
|
|
|
FPTR_REJECT_DUP_CALLBACK callback =
|
|
|
|
(idx->idx_flags & idx_unique) ? duplicate_key : NULL;
|
2003-09-28 16:41:58 +02:00
|
|
|
void* callback_arg =
|
|
|
|
(idx->idx_flags & idx_unique) ? &ifl_data : NULL;
|
|
|
|
|
2004-03-19 07:14:53 +01:00
|
|
|
sort_context* sort_handle = SORT_init(tdbb->tdbb_status_vector,
|
2004-03-28 11:10:30 +02:00
|
|
|
key_length + sizeof(index_sort_record),
|
2004-10-07 10:33:51 +02:00
|
|
|
2, key_desc, callback, callback_arg,
|
2003-09-28 16:41:58 +02:00
|
|
|
tdbb->tdbb_attachment, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!sort_handle)
|
|
|
|
ERR_punt();
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
jrd_rel* partner_relation = 0;
|
|
|
|
USHORT partner_index_id = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (idx->idx_flags & idx_foreign) {
|
2003-12-11 11:33:30 +01:00
|
|
|
if (!MET_lookup_partner(tdbb, relation, idx, index_name)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
BUGCHECK(173); /* msg 173 referenced index description not found */
|
2003-12-11 11:33:30 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
partner_relation = MET_relation(tdbb, idx->idx_primary_relation);
|
|
|
|
partner_index_id = (USHORT) idx->idx_primary_index;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Checkout a garbage collect record block for fetching data. */
|
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
Record* gc_record = VIO_gc_record(tdbb, relation);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Unless this is the only attachment or a database restore, worry about
|
|
|
|
preserving the page working sets of other attachments. */
|
2004-03-18 06:56:06 +01:00
|
|
|
Attachment* attachment = tdbb->tdbb_attachment;
|
2003-12-22 11:00:59 +01:00
|
|
|
if ((attachment) &&
|
|
|
|
(attachment != dbb->dbb_attachments || attachment->att_next))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if (attachment->att_flags & ATT_gbak_attachment ||
|
2003-12-22 11:00:59 +01:00
|
|
|
DPM_data_pages(tdbb, relation) > (SLONG) dbb->dbb_bcb->bcb_count)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
primary.rpb_window.win_flags = secondary.rpb_window.win_flags =
|
|
|
|
WIN_large_scan;
|
|
|
|
primary.rpb_org_scans = secondary.rpb_org_scans =
|
|
|
|
relation->rel_scan_count++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Loop thru the relation computing index keys. If there are old versions,
|
|
|
|
find them, too. */
|
2004-02-20 07:43:27 +01:00
|
|
|
bool cancel = false;
|
2004-03-28 11:10:30 +02:00
|
|
|
temporary_key key;
|
2004-02-20 07:43:27 +01:00
|
|
|
while (!cancel && DPM_next(tdbb, &primary, LCK_read, false, false)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (transaction && !VIO_garbage_collect(tdbb, &primary, transaction))
|
|
|
|
continue;
|
|
|
|
if (primary.rpb_flags & rpb_deleted)
|
|
|
|
CCH_RELEASE(tdbb, &primary.rpb_window);
|
|
|
|
else {
|
|
|
|
primary.rpb_record = gc_record;
|
2004-08-30 20:11:08 +02:00
|
|
|
VIO_data(tdbb, &primary, dbb->dbb_permanent);
|
2001-05-23 15:26:42 +02:00
|
|
|
gc_record = primary.rpb_record;
|
2004-04-18 16:22:27 +02:00
|
|
|
stack.push(primary.rpb_record);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
secondary.rpb_page = primary.rpb_b_page;
|
|
|
|
secondary.rpb_line = primary.rpb_b_line;
|
|
|
|
secondary.rpb_prior = primary.rpb_prior;
|
|
|
|
while (secondary.rpb_page) {
|
|
|
|
if (!DPM_fetch(tdbb, &secondary, LCK_read))
|
|
|
|
break; /* must be garbage collected */
|
|
|
|
secondary.rpb_record = NULL;
|
2004-08-30 20:11:08 +02:00
|
|
|
VIO_data(tdbb, &secondary, tdbb->getDefaultPool());
|
2004-04-18 16:22:27 +02:00
|
|
|
stack.push(secondary.rpb_record);
|
2001-05-23 15:26:42 +02:00
|
|
|
secondary.rpb_page = secondary.rpb_b_page;
|
|
|
|
secondary.rpb_line = secondary.rpb_b_line;
|
|
|
|
}
|
|
|
|
|
2004-05-27 18:26:52 +02:00
|
|
|
while (stack.hasData())
|
2004-04-21 16:14:36 +02:00
|
|
|
{
|
2004-04-18 16:22:27 +02:00
|
|
|
Record* record = stack.pop();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If foreign key index is being defined, make sure foreign
|
|
|
|
key definition will not be violated */
|
|
|
|
|
2004-04-21 16:14:36 +02:00
|
|
|
if (idx->idx_flags & idx_foreign)
|
|
|
|
{
|
2003-03-09 22:07:29 +01:00
|
|
|
idx_null_state null_state;
|
2001-05-23 15:26:42 +02:00
|
|
|
/* find out if there is a null segment by faking uniqueness --
|
|
|
|
if there is one, don't bother to check the primary key */
|
|
|
|
|
|
|
|
if (!(idx->idx_flags & idx_unique)) {
|
|
|
|
idx->idx_flags |= idx_unique;
|
2003-03-09 22:07:29 +01:00
|
|
|
result = BTR_key(tdbb, relation, record, idx, &key, &null_state);
|
2001-05-23 15:26:42 +02:00
|
|
|
idx->idx_flags &= ~idx_unique;
|
2003-03-09 22:07:29 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
result = BTR_key(tdbb, relation, record, idx, &key, &null_state);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2003-03-09 22:07:29 +01:00
|
|
|
if (null_state != idx_nulls_none) {
|
|
|
|
result = idx_e_ok;
|
|
|
|
}
|
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
result =
|
|
|
|
check_partner_index(tdbb, relation, record,
|
|
|
|
transaction, idx,
|
|
|
|
partner_relation,
|
|
|
|
partner_index_id);
|
2003-03-05 12:23:08 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2003-03-05 12:23:08 +01:00
|
|
|
if (result == idx_e_ok) {
|
2003-03-09 22:07:29 +01:00
|
|
|
idx_null_state null_state;
|
|
|
|
BTR_key(tdbb, relation, record, idx, &key, &null_state);
|
2003-09-28 16:03:10 +02:00
|
|
|
key_is_null = (null_state == idx_nulls_all);
|
2003-03-05 12:23:08 +01:00
|
|
|
}
|
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
do {
|
|
|
|
if (record != gc_record)
|
2001-12-24 03:51:06 +01:00
|
|
|
delete record;
|
2004-05-27 18:26:52 +02:00
|
|
|
} while (stack.hasData() && (record = stack.pop()));
|
2001-05-23 15:26:42 +02:00
|
|
|
SORT_fini(sort_handle, tdbb->tdbb_attachment);
|
|
|
|
gc_record->rec_flags &= ~REC_gc_active;
|
|
|
|
if (primary.rpb_window.win_flags & WIN_large_scan)
|
|
|
|
--relation->rel_scan_count;
|
2003-03-05 12:23:08 +01:00
|
|
|
ERR_duplicate_error(result, partner_relation,
|
|
|
|
partner_index_id);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2003-09-17 01:19:46 +02:00
|
|
|
if (key.key_length > key_length) {
|
2001-05-23 15:26:42 +02:00
|
|
|
do {
|
|
|
|
if (record != gc_record)
|
2001-12-24 03:51:06 +01:00
|
|
|
delete record;
|
2004-05-27 18:26:52 +02:00
|
|
|
} while (stack.hasData() && (record = stack.pop()));
|
2001-05-23 15:26:42 +02:00
|
|
|
SORT_fini(sort_handle, tdbb->tdbb_attachment);
|
|
|
|
gc_record->rec_flags &= ~REC_gc_active;
|
|
|
|
if (primary.rpb_window.win_flags & WIN_large_scan)
|
|
|
|
--relation->rel_scan_count;
|
|
|
|
BUGCHECK(174); /* msg 174 index key too big */
|
|
|
|
}
|
2003-09-28 20:49:21 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
UCHAR* p;
|
2003-09-28 20:49:21 +02:00
|
|
|
SORT_put(tdbb->tdbb_status_vector, sort_handle,
|
|
|
|
reinterpret_cast<ULONG**>(&p));
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* try to catch duplicates early */
|
|
|
|
|
2003-09-28 16:41:58 +02:00
|
|
|
if (ifl_data.ifl_duplicates > 0) {
|
2001-05-23 15:26:42 +02:00
|
|
|
do {
|
|
|
|
if (record != gc_record)
|
2001-12-24 03:51:06 +01:00
|
|
|
delete record;
|
2004-05-27 18:26:52 +02:00
|
|
|
} while (stack.hasData() && (record = stack.pop()));
|
2001-05-23 15:26:42 +02:00
|
|
|
SORT_fini(sort_handle, tdbb->tdbb_attachment);
|
|
|
|
gc_record->rec_flags &= ~REC_gc_active;
|
|
|
|
if (primary.rpb_window.win_flags & WIN_large_scan)
|
|
|
|
--relation->rel_scan_count;
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_no_dup, isc_arg_string,
|
2003-12-11 11:33:30 +01:00
|
|
|
ERR_cstring(index_name), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
USHORT l = key.key_length;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-01 03:37:25 +01:00
|
|
|
if (l > 0) {
|
2003-12-22 11:00:59 +01:00
|
|
|
const UCHAR* q = key.key_data;
|
2003-12-03 09:19:24 +01:00
|
|
|
do {
|
2003-12-01 03:37:25 +01:00
|
|
|
*p++ = *q++;
|
2003-12-03 09:19:24 +01:00
|
|
|
} while (--l);
|
2003-12-01 03:37:25 +01:00
|
|
|
}
|
2003-12-03 09:19:24 +01:00
|
|
|
if ( (l = key_length - key.key_length) ) {
|
|
|
|
do {
|
2001-05-23 15:26:42 +02:00
|
|
|
*p++ = pad;
|
2003-12-03 09:19:24 +01:00
|
|
|
} while (--l);
|
|
|
|
}
|
2004-03-28 11:10:30 +02:00
|
|
|
index_sort_record* isr = (index_sort_record*) p;
|
2001-05-23 15:26:42 +02:00
|
|
|
isr->isr_record_number = primary.rpb_number;
|
2004-10-05 23:09:51 +02:00
|
|
|
isr->isr_key_length = key.key_length;
|
2004-05-27 18:26:52 +02:00
|
|
|
isr->isr_flags = (stack.hasData() ? ISR_secondary : 0) | (key_is_null ? ISR_null : 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (record != gc_record)
|
2001-12-24 03:51:06 +01:00
|
|
|
delete record;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2003-03-24 15:41:42 +01:00
|
|
|
#ifdef SUPERSERVER
|
2001-05-23 15:26:42 +02:00
|
|
|
if (--tdbb->tdbb_quantum < 0 && !tdbb->tdbb_inhibit)
|
2004-03-11 06:04:26 +01:00
|
|
|
cancel = JRD_reschedule(tdbb, 0, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
gc_record->rec_flags &= ~REC_gc_active;
|
|
|
|
if (primary.rpb_window.win_flags & WIN_large_scan)
|
|
|
|
--relation->rel_scan_count;
|
|
|
|
|
2003-09-28 20:49:21 +02:00
|
|
|
if (cancel || !SORT_sort(tdbb->tdbb_status_vector, sort_handle)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
SORT_fini(sort_handle, tdbb->tdbb_attachment);
|
|
|
|
ERR_punt();
|
|
|
|
}
|
|
|
|
|
2003-09-28 16:41:58 +02:00
|
|
|
if (ifl_data.ifl_duplicates > 0) {
|
2001-05-23 15:26:42 +02:00
|
|
|
SORT_fini(sort_handle, tdbb->tdbb_attachment);
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_no_dup, isc_arg_string,
|
2003-12-11 11:33:30 +01:00
|
|
|
ERR_cstring(index_name), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
BTR_create(tdbb, relation, idx, key_length, sort_handle, selectivity);
|
|
|
|
|
2003-09-28 16:41:58 +02:00
|
|
|
if (ifl_data.ifl_duplicates > 0) {
|
2003-09-28 17:39:10 +02:00
|
|
|
// we don't need SORT_fini() here, as it's called inside BTR_create()
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_no_dup, isc_arg_string,
|
2003-12-11 11:33:30 +01:00
|
|
|
ERR_cstring(index_name), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-19 07:14:53 +01:00
|
|
|
IndexBlock* IDX_create_index_block(thread_db* tdbb, jrd_rel* relation, USHORT id)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I D X _ c r e a t e _ i n d e x _ b l o c k
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Create an index block and an associated
|
|
|
|
* lock block for the specified index.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
CHECK_DBB(dbb);
|
|
|
|
|
2004-03-19 07:14:53 +01:00
|
|
|
IndexBlock* index_block = FB_NEW(*dbb->dbb_permanent) IndexBlock();
|
2001-05-23 15:26:42 +02:00
|
|
|
index_block->idb_id = id;
|
|
|
|
|
|
|
|
/* link the block in with the relation linked list */
|
|
|
|
|
|
|
|
index_block->idb_next = relation->rel_index_blocks;
|
|
|
|
relation->rel_index_blocks = index_block;
|
|
|
|
|
|
|
|
/* create a shared lock for the index, to coordinate
|
|
|
|
any modification to the index so that the cached information
|
|
|
|
about the index will be discarded */
|
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
Lock* lock = FB_NEW_RPT(*dbb->dbb_permanent, 0) Lock;
|
2004-02-20 07:43:27 +01:00
|
|
|
index_block->idb_lock = lock;
|
2001-05-23 15:26:42 +02:00
|
|
|
lock->lck_parent = dbb->dbb_lock;
|
|
|
|
lock->lck_dbb = dbb;
|
|
|
|
lock->lck_key.lck_long = index_block->idb_id;
|
|
|
|
lock->lck_length = sizeof(lock->lck_key.lck_long);
|
|
|
|
lock->lck_type = LCK_expression;
|
|
|
|
lock->lck_owner_handle = LCK_get_owner_handle(tdbb, lock->lck_type);
|
2003-05-16 22:35:19 +02:00
|
|
|
lock->lck_ast = index_block_flush;
|
2004-03-18 06:56:06 +01:00
|
|
|
lock->lck_object = index_block;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return index_block;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void IDX_delete_index(thread_db* tdbb, jrd_rel* relation, USHORT id)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I D X _ d e l e t e _ i n d e x
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Delete a single index.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
signal_index_deletion(tdbb, relation, id);
|
|
|
|
|
2003-12-11 11:33:30 +01:00
|
|
|
WIN window(relation->rel_index_root);
|
2001-05-23 15:26:42 +02:00
|
|
|
CCH_FETCH(tdbb, &window, LCK_write, pag_root);
|
|
|
|
|
|
|
|
BTR_delete_index(tdbb, &window, id);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void IDX_delete_indices(thread_db* tdbb, jrd_rel* relation)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I D X _ d e l e t e _ i n d i c e s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Delete all known indices in preparation for deleting a
|
|
|
|
* complete relation.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SSHORT i;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2003-12-11 11:33:30 +01:00
|
|
|
WIN window(relation->rel_index_root);
|
2004-02-20 07:43:27 +01:00
|
|
|
index_root_page* root = (index_root_page*) CCH_FETCH(tdbb, &window, LCK_write, pag_root);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
for (i = 0; i < root->irt_count; i++) {
|
|
|
|
BTR_delete_index(tdbb, &window, i);
|
2004-02-20 07:43:27 +01:00
|
|
|
root = (index_root_page*) CCH_FETCH(tdbb, &window, LCK_write, pag_root);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
IDX_E IDX_erase(thread_db* tdbb,
|
2004-03-18 06:56:06 +01:00
|
|
|
record_param* rpb,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_tra* transaction, jrd_rel** bad_relation, USHORT * bad_index)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I D X _ e r a s e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Check the various indices prior to an ERASE operation.
|
|
|
|
* If one is a primary key, check its partner for
|
|
|
|
* a duplicate record.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-28 11:10:30 +02:00
|
|
|
index_desc idx;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
IDX_E error_code = idx_e_ok;
|
2003-08-18 23:13:56 +02:00
|
|
|
idx.idx_id = (USHORT) -1;
|
2003-12-11 11:33:30 +01:00
|
|
|
WIN window(-1);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-13 14:03:11 +02:00
|
|
|
while (BTR_next_index(tdbb, rpb->rpb_relation, transaction, &idx, &window))
|
|
|
|
if (idx.idx_flags & (idx_primary | idx_unique)) {
|
|
|
|
error_code = check_foreign_key(tdbb, rpb->rpb_record,
|
|
|
|
rpb->rpb_relation, transaction,
|
|
|
|
&idx, bad_relation, bad_index);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (idx_e_ok != error_code) {
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return error_code;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-04-18 16:22:27 +02:00
|
|
|
void IDX_garbage_collect(thread_db* tdbb,
|
|
|
|
record_param* rpb,
|
2004-05-09 07:48:33 +02:00
|
|
|
RecordStack& going,
|
|
|
|
RecordStack& staying)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I D X _ g a r b a g e _ c o l l e c t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Perform garbage collection for a bunch of records. Scan
|
|
|
|
* through the indices defined for a relation. Garbage collect
|
|
|
|
* each.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-28 11:10:30 +02:00
|
|
|
index_desc idx;
|
|
|
|
temporary_key key1, key2;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
index_insertion insertion;
|
2001-05-23 15:26:42 +02:00
|
|
|
insertion.iib_descriptor = &idx;
|
|
|
|
insertion.iib_number = rpb->rpb_number;
|
|
|
|
insertion.iib_relation = rpb->rpb_relation;
|
|
|
|
insertion.iib_key = &key1;
|
|
|
|
|
2003-12-11 11:33:30 +01:00
|
|
|
WIN window(rpb->rpb_relation->rel_index_root);
|
2004-02-20 07:43:27 +01:00
|
|
|
index_root_page* root = (index_root_page*) CCH_FETCH(tdbb, &window, LCK_read, pag_root);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
for (USHORT i = 0; i < root->irt_count; i++) {
|
2004-08-19 20:41:19 +02:00
|
|
|
if (BTR_description(tdbb, rpb->rpb_relation, root, &idx, i)) {
|
2004-05-27 18:26:52 +02:00
|
|
|
for (RecordStack::iterator stack1(going); stack1.hasData(); ++stack1) {
|
2004-04-18 16:22:27 +02:00
|
|
|
Record* rec1 = stack1.object();
|
2003-03-05 12:23:08 +01:00
|
|
|
BTR_key(tdbb, rpb->rpb_relation, rec1, &idx, &key1, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-05-27 16:59:13 +02:00
|
|
|
/* Cancel index if there are duplicates in the remaining records */
|
|
|
|
|
2004-10-05 23:09:51 +02:00
|
|
|
|
2004-04-18 16:22:27 +02:00
|
|
|
RecordStack::iterator stack2(stack1);
|
2004-05-27 18:26:52 +02:00
|
|
|
for (++stack2; stack2.hasData(); ++stack2)
|
2004-02-02 12:02:12 +01:00
|
|
|
{
|
2004-04-18 16:22:27 +02:00
|
|
|
Record* rec2 = stack2.object();
|
2003-05-27 16:59:13 +02:00
|
|
|
if (rec2->rec_number == rec1->rec_number) {
|
|
|
|
BTR_key(tdbb, rpb->rpb_relation, rec2, &idx, &key2, 0);
|
|
|
|
if (key_equal(&key1, &key2))
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2004-05-27 18:26:52 +02:00
|
|
|
if (stack2.hasData())
|
2003-05-27 16:59:13 +02:00
|
|
|
continue;
|
2004-10-05 23:09:51 +02:00
|
|
|
|
2003-05-27 16:59:13 +02:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Make sure the index doesn't exist in any record remaining */
|
|
|
|
|
2004-04-18 16:22:27 +02:00
|
|
|
RecordStack::iterator stack3(staying);
|
2004-05-27 18:26:52 +02:00
|
|
|
for (; stack3.hasData(); ++stack3) {
|
2004-04-18 16:22:27 +02:00
|
|
|
Record* rec3 = stack3.object();
|
|
|
|
BTR_key(tdbb, rpb->rpb_relation, rec3, &idx, &key2, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (key_equal(&key1, &key2))
|
|
|
|
break;
|
|
|
|
}
|
2004-05-27 18:26:52 +02:00
|
|
|
if (stack3.hasData())
|
2001-05-23 15:26:42 +02:00
|
|
|
continue;
|
|
|
|
|
|
|
|
/* Get rid of index node */
|
|
|
|
|
|
|
|
BTR_remove(tdbb, &window, &insertion);
|
2004-02-20 07:43:27 +01:00
|
|
|
root = (index_root_page*) CCH_FETCH(tdbb, &window, LCK_read, pag_root);
|
2004-04-23 10:02:52 +02:00
|
|
|
if (stack1.hasMore(1))
|
2004-04-21 16:14:36 +02:00
|
|
|
{
|
2004-08-19 20:41:19 +02:00
|
|
|
BTR_description(tdbb, rpb->rpb_relation, root, &idx, i);
|
2004-04-21 16:14:36 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
IDX_E IDX_modify(thread_db* tdbb,
|
2004-03-18 06:56:06 +01:00
|
|
|
record_param* org_rpb,
|
|
|
|
record_param* new_rpb,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_tra* transaction, jrd_rel** bad_relation, USHORT * bad_index)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I D X _ m o d i f y
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Update the various indices after a MODIFY operation. If a duplicate
|
|
|
|
* index is violated, return the index number. If successful, return
|
|
|
|
* -1.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
temporary_key key1;
|
|
|
|
index_desc idx;
|
|
|
|
|
|
|
|
index_insertion insertion;
|
2001-05-23 15:26:42 +02:00
|
|
|
insertion.iib_relation = org_rpb->rpb_relation;
|
|
|
|
insertion.iib_number = org_rpb->rpb_number;
|
|
|
|
insertion.iib_key = &key1;
|
|
|
|
insertion.iib_descriptor = &idx;
|
|
|
|
insertion.iib_transaction = transaction;
|
2003-12-22 11:00:59 +01:00
|
|
|
IDX_E error_code = idx_e_ok;
|
2003-08-18 23:13:56 +02:00
|
|
|
idx.idx_id = (USHORT) -1;
|
2003-12-11 11:33:30 +01:00
|
|
|
WIN window(-1);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
temporary_key key2;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
while (BTR_next_index
|
2004-03-11 06:04:26 +01:00
|
|
|
(tdbb, org_rpb->rpb_relation, transaction, &idx, &window))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
*bad_index = idx.idx_id;
|
|
|
|
*bad_relation = new_rpb->rpb_relation;
|
2001-12-24 03:51:06 +01:00
|
|
|
if ( (error_code =
|
2001-05-23 15:26:42 +02:00
|
|
|
BTR_key(tdbb, new_rpb->rpb_relation, new_rpb->rpb_record, &idx,
|
2004-03-11 06:04:26 +01:00
|
|
|
&key1, 0)) )
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
BTR_key(tdbb, org_rpb->rpb_relation, org_rpb->rpb_record, &idx,
|
2003-03-05 12:23:08 +01:00
|
|
|
&key2, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!key_equal(&key1, &key2)) {
|
2001-12-24 03:51:06 +01:00
|
|
|
if (( error_code =
|
2001-05-23 15:26:42 +02:00
|
|
|
insert_key(tdbb, new_rpb->rpb_relation, new_rpb->rpb_record,
|
|
|
|
transaction, &window, &insertion, bad_relation,
|
2004-03-11 06:04:26 +01:00
|
|
|
bad_index)) )
|
|
|
|
{
|
|
|
|
return error_code;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return error_code;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
IDX_E IDX_modify_check_constraints(thread_db* tdbb,
|
2004-03-18 06:56:06 +01:00
|
|
|
record_param* org_rpb,
|
|
|
|
record_param* new_rpb,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_tra* transaction,
|
|
|
|
jrd_rel** bad_relation, USHORT * bad_index)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I D X _ m o d i f y _ c h e c k _ c o n s t r a i n t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Check for foreign key constraint after a modify statement
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-28 11:10:30 +02:00
|
|
|
temporary_key key1, key2;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
IDX_E error_code = idx_e_ok;
|
2004-03-28 11:10:30 +02:00
|
|
|
index_desc idx;
|
2003-08-18 23:13:56 +02:00
|
|
|
idx.idx_id = (USHORT) -1;
|
2003-12-11 11:33:30 +01:00
|
|
|
WIN window(-1);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If relation's primary/unique keys have no dependencies by other
|
|
|
|
relations' foreign keys then don't bother cycling thru all index
|
|
|
|
descriptions. */
|
|
|
|
|
|
|
|
if (!(org_rpb->rpb_relation->rel_flags & REL_check_partners) &&
|
|
|
|
!(org_rpb->rpb_relation->rel_primary_dpnds.prim_reference_ids))
|
2003-12-11 11:33:30 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
return error_code;
|
2003-12-11 11:33:30 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Now check all the foreign key constraints. Referential integrity relation
|
|
|
|
could be established by primary key/foreign key or unique key/foreign key */
|
|
|
|
|
|
|
|
while (BTR_next_index
|
2003-12-11 11:33:30 +01:00
|
|
|
(tdbb, org_rpb->rpb_relation, transaction, &idx, &window))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!(idx.idx_flags & (idx_primary | idx_unique))
|
|
|
|
|| !MET_lookup_partner(tdbb, org_rpb->rpb_relation, &idx,
|
2003-12-11 11:33:30 +01:00
|
|
|
&NULL_STR))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
continue;
|
2003-12-11 11:33:30 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
*bad_index = idx.idx_id;
|
|
|
|
*bad_relation = new_rpb->rpb_relation;
|
|
|
|
if (
|
|
|
|
(error_code =
|
|
|
|
BTR_key(tdbb, new_rpb->rpb_relation, new_rpb->rpb_record, &idx,
|
2003-03-05 12:23:08 +01:00
|
|
|
&key1, 0))
|
2001-05-23 15:26:42 +02:00
|
|
|
|| (error_code =
|
|
|
|
BTR_key(tdbb, org_rpb->rpb_relation, org_rpb->rpb_record,
|
2003-12-22 11:00:59 +01:00
|
|
|
&idx, &key2, 0)))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (!key_equal(&key1, &key2)) {
|
|
|
|
error_code =
|
|
|
|
check_foreign_key(tdbb, org_rpb->rpb_record,
|
|
|
|
org_rpb->rpb_relation, transaction, &idx,
|
|
|
|
bad_relation, bad_index);
|
|
|
|
if (idx_e_ok != error_code) {
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
return error_code;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return error_code;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-21 08:16:17 +02:00
|
|
|
void IDX_statistics(thread_db* tdbb, const jrd_rel* relation, USHORT id,
|
2004-03-28 11:10:30 +02:00
|
|
|
SelectivityList& selectivity)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I D X _ s t a t i s t i c s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Scan index pages recomputing
|
|
|
|
* selectivity.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2003-11-30 22:14:30 +01:00
|
|
|
BTR_selectivity(tdbb, relation, id, selectivity);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
IDX_E IDX_store(thread_db* tdbb,
|
2004-03-18 06:56:06 +01:00
|
|
|
record_param* rpb,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_tra* transaction, jrd_rel** bad_relation, USHORT * bad_index)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I D X _ s t o r e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Update the various indices after a STORE operation. If a duplicate
|
|
|
|
* index is violated, return the index number. If successful, return
|
|
|
|
* -1.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
temporary_key key;
|
|
|
|
index_desc idx;
|
|
|
|
|
|
|
|
index_insertion insertion;
|
2001-05-23 15:26:42 +02:00
|
|
|
insertion.iib_relation = rpb->rpb_relation;
|
|
|
|
insertion.iib_number = rpb->rpb_number;
|
|
|
|
insertion.iib_key = &key;
|
|
|
|
insertion.iib_descriptor = &idx;
|
|
|
|
insertion.iib_transaction = transaction;
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
IDX_E error_code = idx_e_ok;
|
2003-08-18 23:13:56 +02:00
|
|
|
idx.idx_id = (USHORT) -1;
|
2003-12-11 11:33:30 +01:00
|
|
|
WIN window(-1);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
while (BTR_next_index
|
2003-12-11 11:33:30 +01:00
|
|
|
(tdbb, rpb->rpb_relation, transaction, &idx, &window))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
*bad_index = idx.idx_id;
|
|
|
|
*bad_relation = rpb->rpb_relation;
|
2001-12-24 03:51:06 +01:00
|
|
|
if ( (error_code =
|
2003-12-11 11:33:30 +01:00
|
|
|
BTR_key(tdbb, rpb->rpb_relation, rpb->rpb_record, &idx, &key, 0)) )
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
break;
|
|
|
|
}
|
2001-12-24 03:51:06 +01:00
|
|
|
if ( (error_code =
|
2001-05-23 15:26:42 +02:00
|
|
|
insert_key(tdbb, rpb->rpb_relation, rpb->rpb_record, transaction,
|
2001-12-24 03:51:06 +01:00
|
|
|
&window, &insertion, bad_relation, bad_index)) )
|
2003-12-11 11:33:30 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
return error_code;
|
2003-12-11 11:33:30 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return error_code;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static IDX_E check_duplicates(
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb,
|
2004-03-18 06:56:06 +01:00
|
|
|
Record* record,
|
2004-03-28 11:10:30 +02:00
|
|
|
index_desc* record_idx,
|
|
|
|
index_insertion* insertion, jrd_rel* relation_2)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c h e c k _ d u p l i c a t e s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Make sure there aren't any active duplicates for
|
|
|
|
* a unique index or a foreign key.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DSC desc1, desc2;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
IDX_E result = idx_e_ok;
|
2004-03-28 11:10:30 +02:00
|
|
|
index_desc* insertion_idx = insertion->iib_descriptor;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
record_param rpb;
|
2001-05-23 15:26:42 +02:00
|
|
|
rpb.rpb_relation = insertion->iib_relation;
|
|
|
|
rpb.rpb_record = NULL;
|
2004-03-11 06:04:26 +01:00
|
|
|
// rpb.rpb_window.win_flags = 0; redundant.
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_rel* relation_1 = insertion->iib_relation;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-09-28 08:28:38 +02:00
|
|
|
RecordBitmap::Accessor accessor(insertion->iib_duplicates);
|
|
|
|
|
|
|
|
if (accessor.getFirst())
|
|
|
|
do {
|
|
|
|
rpb.rpb_number.setValue(accessor.current());
|
2001-05-23 15:26:42 +02:00
|
|
|
if (rpb.rpb_number != insertion->iib_number
|
|
|
|
&& VIO_get_current(tdbb, &rpb, insertion->iib_transaction,
|
2004-08-30 20:11:08 +02:00
|
|
|
tdbb->getDefaultPool(),
|
2004-02-20 07:43:27 +01:00
|
|
|
(record_idx->idx_flags & idx_foreign) != 0))
|
2003-11-01 12:29:14 +01:00
|
|
|
{
|
2003-05-29 13:59:49 +02:00
|
|
|
// dimitr: we shouldn't ignore status exceptions which take place
|
2003-11-11 13:19:20 +01:00
|
|
|
// inside the lock manager. Namely, they are: isc_deadlock,
|
|
|
|
// isc_lock_conflict, isc_lock_timeout. Otherwise we may
|
2003-05-29 13:59:49 +02:00
|
|
|
// have logically corrupted database as a result. If any
|
|
|
|
// of the mentioned errors appeared, it means that there's
|
|
|
|
// an active transaction out there which has modified our
|
|
|
|
// record. For "nowait" transaction, it means we have
|
|
|
|
// an update conflict.
|
|
|
|
//
|
|
|
|
// was: if (rpb.rpb_flags & rpb_deleted) {
|
|
|
|
//
|
|
|
|
// P.S. I think the check for a status vector should be enough,
|
|
|
|
// but for sure let's keep the old one as well.
|
|
|
|
// 2003.05.27
|
2003-11-01 12:29:14 +01:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
const bool lock_error =
|
2003-11-11 13:19:20 +01:00
|
|
|
(tdbb->tdbb_status_vector[1] == isc_deadlock ||
|
|
|
|
tdbb->tdbb_status_vector[1] == isc_lock_conflict ||
|
|
|
|
tdbb->tdbb_status_vector[1] == isc_lock_timeout);
|
2003-11-01 12:29:14 +01:00
|
|
|
// the above errors are not thrown but returned silently
|
|
|
|
|
|
|
|
if (rpb.rpb_flags & rpb_deleted || lock_error) {
|
2001-05-23 15:26:42 +02:00
|
|
|
result = idx_e_duplicate;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check the values of the fields in the record being inserted with the
|
|
|
|
record retrieved -- for unique indexes the insertion index and the
|
|
|
|
record index are the same, but for foreign keys they are different */
|
2003-03-09 22:07:29 +01:00
|
|
|
|
2004-09-03 07:27:44 +02:00
|
|
|
if (record_idx->idx_flags & idx_expressn)
|
|
|
|
{
|
|
|
|
DSC *desc_ptr1, *desc_ptr2;
|
|
|
|
|
|
|
|
fb_assert(insertion_idx->idx_expression != NULL);
|
|
|
|
|
|
|
|
fb_assert(insertion_idx->idx_expression_request->req_caller == NULL);
|
|
|
|
insertion_idx->idx_expression_request->req_caller = tdbb->tdbb_request;
|
|
|
|
|
|
|
|
if (tdbb->tdbb_request)
|
|
|
|
{
|
|
|
|
insertion_idx->idx_expression_request->req_transaction =
|
|
|
|
tdbb->tdbb_request->req_transaction;
|
|
|
|
}
|
|
|
|
|
|
|
|
tdbb->tdbb_request = insertion_idx->idx_expression_request;
|
|
|
|
tdbb->tdbb_request->req_rpb[0].rpb_record = rpb.rpb_record;
|
|
|
|
tdbb->tdbb_request->req_flags &= ~req_null;
|
|
|
|
{
|
2004-09-03 08:10:23 +02:00
|
|
|
Jrd::ContextPoolHolder context(tdbb, tdbb->tdbb_request->req_pool);
|
2004-09-03 07:27:44 +02:00
|
|
|
|
|
|
|
if (!(desc_ptr1 = EVL_expr(tdbb, insertion_idx->idx_expression)))
|
|
|
|
desc_ptr1 = &insertion_idx->idx_expression_desc;
|
|
|
|
|
|
|
|
}
|
|
|
|
const bool flag1 = !(tdbb->tdbb_request->req_flags & req_null);
|
|
|
|
|
|
|
|
tdbb->tdbb_request = insertion_idx->idx_expression_request->req_caller;
|
|
|
|
insertion_idx->idx_expression_request->req_caller = NULL;
|
|
|
|
|
|
|
|
fb_assert(record_idx->idx_expression != NULL);
|
|
|
|
|
|
|
|
fb_assert(record_idx->idx_expression_request->req_caller == NULL);
|
|
|
|
record_idx->idx_expression_request->req_caller = tdbb->tdbb_request;
|
|
|
|
|
|
|
|
if (tdbb->tdbb_request)
|
|
|
|
{
|
|
|
|
record_idx->idx_expression_request->req_transaction =
|
|
|
|
tdbb->tdbb_request->req_transaction;
|
|
|
|
}
|
|
|
|
|
2004-09-03 08:44:11 +02:00
|
|
|
tdbb->tdbb_request = record_idx->idx_expression_request;
|
2004-09-03 07:27:44 +02:00
|
|
|
tdbb->tdbb_request->req_rpb[0].rpb_record = record;
|
|
|
|
tdbb->tdbb_request->req_flags &= ~req_null;
|
|
|
|
{
|
2004-09-03 08:10:23 +02:00
|
|
|
Jrd::ContextPoolHolder context(tdbb, tdbb->tdbb_request->req_pool);
|
2004-09-03 07:27:44 +02:00
|
|
|
|
|
|
|
if (!(desc_ptr2 = EVL_expr(tdbb, record_idx->idx_expression)))
|
|
|
|
desc_ptr2 = &record_idx->idx_expression_desc;
|
|
|
|
}
|
|
|
|
const bool flag2 = !(tdbb->tdbb_request->req_flags & req_null);
|
|
|
|
|
|
|
|
tdbb->tdbb_request = record_idx->idx_expression_request->req_caller;
|
|
|
|
record_idx->idx_expression_request->req_caller = NULL;
|
|
|
|
|
|
|
|
if (flag1 && flag2 && !MOV_compare(desc_ptr1, desc_ptr2)) {
|
|
|
|
result = idx_e_duplicate;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-09-03 07:27:44 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
bool all_nulls = true;
|
|
|
|
USHORT i;
|
|
|
|
for (i = 0; i < insertion_idx->idx_count; i++) {
|
|
|
|
USHORT field_id = insertion_idx->idx_rpt[i].idx_field;
|
|
|
|
/* In order to "map a null to a default" value (in EVL_field()),
|
|
|
|
* the relation block is referenced.
|
|
|
|
* Reference: Bug 10116, 10424
|
|
|
|
*/
|
|
|
|
const bool flag =
|
|
|
|
EVL_field(relation_1, rpb.rpb_record, field_id, &desc1);
|
|
|
|
|
|
|
|
field_id = record_idx->idx_rpt[i].idx_field;
|
|
|
|
const bool flag_2 = EVL_field(relation_2, record, field_id, &desc2);
|
|
|
|
|
|
|
|
if (flag != flag_2 || MOV_compare(&desc1, &desc2) != 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
all_nulls = all_nulls && !flag && !flag_2;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-09-03 07:27:44 +02:00
|
|
|
if (i >= insertion_idx->idx_count && !all_nulls) {
|
|
|
|
result = idx_e_duplicate;
|
|
|
|
break;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
2004-09-28 08:28:38 +02:00
|
|
|
} while (accessor.getNext());
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (rpb.rpb_record)
|
2001-12-24 03:51:06 +01:00
|
|
|
delete rpb.rpb_record;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static IDX_E check_foreign_key(
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb,
|
2004-03-18 06:56:06 +01:00
|
|
|
Record* record,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_rel* relation,
|
|
|
|
jrd_tra* transaction,
|
2004-03-28 11:10:30 +02:00
|
|
|
index_desc* idx,
|
|
|
|
jrd_rel** bad_relation, USHORT* bad_index)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c h e c k _ f o r e i g n _ k e y
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* The passed index participates in a foreign key.
|
|
|
|
* Check the passed record to see if a corresponding
|
|
|
|
* record appears in the partner index.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
IDX_E result = idx_e_ok;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-11 11:33:30 +01:00
|
|
|
if (!MET_lookup_partner(tdbb, relation, idx, &NULL_STR)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return result;
|
2003-12-11 11:33:30 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
jrd_rel* partner_relation;
|
2003-12-22 11:00:59 +01:00
|
|
|
USHORT index_id = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (idx->idx_flags & idx_foreign) {
|
|
|
|
partner_relation = MET_relation(tdbb, idx->idx_primary_relation);
|
|
|
|
index_id = (USHORT) idx->idx_primary_index;
|
|
|
|
result =
|
|
|
|
check_partner_index(tdbb, relation, record, transaction, idx,
|
|
|
|
partner_relation, index_id);
|
|
|
|
}
|
2003-12-22 11:00:59 +01:00
|
|
|
else if (idx->idx_flags & (idx_primary | idx_unique)) {
|
2004-03-28 11:10:30 +02:00
|
|
|
for (int index_number = 0;
|
2003-08-09 23:15:32 +02:00
|
|
|
index_number < (int) idx->idx_foreign_primaries->count();
|
2003-12-22 11:00:59 +01:00
|
|
|
index_number++)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if (idx->idx_id !=
|
2004-01-21 08:18:30 +01:00
|
|
|
(UCHAR)(IPTR) (*idx->idx_foreign_primaries)[index_number])
|
2003-12-22 11:00:59 +01:00
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
partner_relation =
|
|
|
|
MET_relation(tdbb,
|
2004-01-21 08:18:30 +01:00
|
|
|
(IPTR) (*idx->idx_foreign_relations)[index_number]);
|
2001-05-23 15:26:42 +02:00
|
|
|
index_id =
|
2004-01-21 08:18:30 +01:00
|
|
|
(USHORT)(IPTR) (*idx->idx_foreign_indexes)[index_number];
|
2001-12-24 03:51:06 +01:00
|
|
|
if ( (result =
|
2001-05-23 15:26:42 +02:00
|
|
|
check_partner_index(tdbb, relation, record, transaction, idx,
|
2001-12-24 03:51:06 +01:00
|
|
|
partner_relation, index_id)) )
|
2004-01-03 11:59:52 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (result) {
|
|
|
|
if (idx->idx_flags & idx_foreign) {
|
|
|
|
*bad_relation = relation;
|
|
|
|
*bad_index = idx->idx_id;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
*bad_relation = partner_relation;
|
|
|
|
*bad_index = index_id;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static IDX_E check_partner_index(
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_rel* relation,
|
2004-03-18 06:56:06 +01:00
|
|
|
Record* record,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_tra* transaction,
|
2004-03-28 11:10:30 +02:00
|
|
|
index_desc* idx,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_rel* partner_relation, SSHORT index_id)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c h e c k _ p a r t n e r _ i n d e x
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* The passed index participates in a foreign key.
|
|
|
|
* Check the passed record to see if a corresponding
|
|
|
|
* record appears in the partner index.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-28 11:10:30 +02:00
|
|
|
index_desc partner_idx;
|
|
|
|
index_insertion insertion;
|
|
|
|
temporary_key key;
|
2004-03-19 07:14:53 +01:00
|
|
|
IndexRetrieval retrieval;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
IDX_E result = idx_e_ok;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* get the index root page for the partner relation */
|
|
|
|
|
2003-12-11 11:33:30 +01:00
|
|
|
WIN window(get_root_page(tdbb, partner_relation));
|
2004-02-20 07:43:27 +01:00
|
|
|
index_root_page* root = (index_root_page*) CCH_FETCH(tdbb, &window, LCK_read, pag_root);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* get the description of the partner index */
|
|
|
|
|
2004-08-19 20:41:19 +02:00
|
|
|
if (!BTR_description(tdbb, partner_relation, root, &partner_idx, index_id))
|
2001-05-23 15:26:42 +02:00
|
|
|
BUGCHECK(175); /* msg 175 partner index description not found */
|
|
|
|
|
|
|
|
/* get the key in the original index */
|
|
|
|
|
2003-03-05 12:23:08 +01:00
|
|
|
result = BTR_key(tdbb, relation, record, idx, &key, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
|
|
|
|
/* now check for current duplicates */
|
|
|
|
|
|
|
|
if (result == idx_e_ok) {
|
|
|
|
/* fill out a retrieval block for the purpose of
|
|
|
|
generating a bitmap of duplicate records */
|
|
|
|
|
2004-03-19 07:14:53 +01:00
|
|
|
MOVE_CLEAR(&retrieval, sizeof(IndexRetrieval));
|
2001-12-24 03:51:06 +01:00
|
|
|
//retrieval.blk_type = type_irb;
|
2001-05-23 15:26:42 +02:00
|
|
|
retrieval.irb_index = partner_idx.idx_id;
|
|
|
|
MOVE_FAST(&partner_idx, &retrieval.irb_desc,
|
|
|
|
sizeof(retrieval.irb_desc));
|
|
|
|
retrieval.irb_generic = irb_equality;
|
|
|
|
retrieval.irb_relation = partner_relation;
|
|
|
|
retrieval.irb_key = &key;
|
2003-12-02 16:08:10 +01:00
|
|
|
if (partner_idx.idx_flags & idx_descending) {
|
|
|
|
retrieval.irb_generic |= irb_descending;
|
|
|
|
retrieval.irb_upper_count = retrieval.irb_lower_count = 0;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
retrieval.irb_upper_count =
|
|
|
|
retrieval.irb_lower_count = idx->idx_count;
|
|
|
|
}
|
2004-03-28 11:10:30 +02:00
|
|
|
|
2004-09-28 08:28:38 +02:00
|
|
|
RecordBitmap* bitmap = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
BTR_evaluate(tdbb, &retrieval, &bitmap);
|
|
|
|
|
|
|
|
/* if there is a bitmap, it means duplicates were found */
|
|
|
|
|
|
|
|
if (bitmap) {
|
|
|
|
insertion.iib_descriptor = &partner_idx;
|
|
|
|
insertion.iib_relation = partner_relation;
|
2004-09-28 08:28:38 +02:00
|
|
|
insertion.iib_number.setValue(BOF_NUMBER);
|
2001-05-23 15:26:42 +02:00
|
|
|
insertion.iib_duplicates = bitmap;
|
|
|
|
insertion.iib_transaction = transaction;
|
|
|
|
result =
|
|
|
|
check_duplicates(tdbb, record, idx, &insertion, relation);
|
|
|
|
if (idx->idx_flags & (idx_primary | idx_unique))
|
2004-09-01 00:29:59 +02:00
|
|
|
result = result ? idx_e_foreign_references_present : idx_e_ok;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (idx->idx_flags & idx_foreign)
|
2004-09-01 00:29:59 +02:00
|
|
|
result = result ? idx_e_ok : idx_e_foreign_target_doesnt_exist;
|
2004-09-28 08:28:38 +02:00
|
|
|
delete bitmap;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
else if (idx->idx_flags & idx_foreign) {
|
2004-09-01 00:29:59 +02:00
|
|
|
result = idx_e_foreign_target_doesnt_exist;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2003-12-11 11:33:30 +01:00
|
|
|
static bool duplicate_key(const UCHAR* record1, const UCHAR* record2, void* ifl_void)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* d u p l i c a t e _ k e y
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Callback routine for duplicate keys during index creation. Just
|
|
|
|
* bump a counter.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-28 11:10:30 +02:00
|
|
|
index_fast_load* ifl_data = static_cast<index_fast_load*>(ifl_void);
|
|
|
|
const index_sort_record* rec1 =
|
|
|
|
(index_sort_record*) (record1 + ifl_data->ifl_key_length);
|
|
|
|
const index_sort_record* rec2 =
|
|
|
|
(index_sort_record*) (record2 + ifl_data->ifl_key_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-28 16:03:10 +02:00
|
|
|
if (!(rec1->isr_flags & (ISR_secondary | ISR_null)) &&
|
2003-12-11 11:33:30 +01:00
|
|
|
!(rec2->isr_flags & (ISR_secondary | ISR_null)))
|
|
|
|
{
|
2003-03-05 12:23:08 +01:00
|
|
|
++ifl_data->ifl_duplicates;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-11 11:33:30 +01:00
|
|
|
return false;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static SLONG get_root_page(thread_db* tdbb, const jrd_rel* relation)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* g e t _ r o o t _ p a g e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Find the root page for a relation.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
SLONG page = relation->rel_index_root;
|
|
|
|
if (!page) {
|
2001-05-23 15:26:42 +02:00
|
|
|
DPM_scan_pages(tdbb);
|
|
|
|
page = relation->rel_index_root;
|
|
|
|
}
|
|
|
|
|
|
|
|
return page;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
static int index_block_flush(void* ast_object)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n d e x _ b l o c k _ f l u s h
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* An exclusive lock has been requested on the
|
|
|
|
* index block. The information in the cached
|
|
|
|
* index block is no longer valid, so clear it
|
|
|
|
* out and release the lock.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-19 07:14:53 +01:00
|
|
|
IndexBlock* index_block = static_cast<IndexBlock*>(ast_object);
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db thd_context, *tdbb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Since this routine will be called asynchronously, we must establish
|
|
|
|
a thread context. */
|
|
|
|
|
2004-05-24 03:03:41 +02:00
|
|
|
JRD_set_thread_data(tdbb, thd_context);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
Lock* lock = index_block->idb_lock;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (lock->lck_attachment) {
|
2001-05-23 15:26:42 +02:00
|
|
|
tdbb->tdbb_database = lock->lck_attachment->att_database;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
tdbb->tdbb_attachment = lock->lck_attachment;
|
|
|
|
tdbb->tdbb_quantum = QUANTUM;
|
|
|
|
tdbb->tdbb_request = NULL;
|
|
|
|
tdbb->tdbb_transaction = NULL;
|
|
|
|
|
|
|
|
/* release the index expression request, which also has
|
|
|
|
the effect of releasing the expression tree */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (index_block->idb_expression_request) {
|
2001-05-23 15:26:42 +02:00
|
|
|
CMP_release(tdbb, index_block->idb_expression_request);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
index_block->idb_expression_request = NULL;
|
|
|
|
index_block->idb_expression = NULL;
|
|
|
|
MOVE_CLEAR(&index_block->idb_expression_desc, sizeof(struct dsc));
|
|
|
|
|
|
|
|
LCK_release(tdbb, lock);
|
|
|
|
|
|
|
|
/* Restore the prior thread context */
|
|
|
|
|
2004-05-23 05:18:10 +02:00
|
|
|
JRD_restore_thread_data();
|
2003-05-16 22:35:19 +02:00
|
|
|
|
|
|
|
return 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static IDX_E insert_key(
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_rel* relation,
|
2004-03-18 06:56:06 +01:00
|
|
|
Record* record,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_tra* transaction,
|
2001-05-23 15:26:42 +02:00
|
|
|
WIN * window_ptr,
|
2004-03-28 11:10:30 +02:00
|
|
|
index_insertion* insertion,
|
2004-01-03 11:59:52 +01:00
|
|
|
jrd_rel** bad_relation,
|
2003-03-05 12:23:08 +01:00
|
|
|
USHORT * bad_index)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n s e r t _ k e y
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Insert a key in the index.
|
|
|
|
* If this is a unique index, check for active duplicates.
|
|
|
|
* If this is a foreign key, check for duplicates in the
|
|
|
|
* primary key index.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
IDX_E result = idx_e_ok;
|
2004-03-28 11:10:30 +02:00
|
|
|
index_desc* idx = insertion->iib_descriptor;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-02 12:02:12 +01:00
|
|
|
/* Insert the key into the index. If the index is unique, btr
|
2001-05-23 15:26:42 +02:00
|
|
|
will keep track of duplicates. */
|
|
|
|
|
|
|
|
insertion->iib_duplicates = NULL;
|
|
|
|
BTR_insert(tdbb, window_ptr, insertion);
|
|
|
|
|
|
|
|
if (insertion->iib_duplicates) {
|
|
|
|
result = check_duplicates(tdbb, record, idx, insertion, NULL);
|
2004-09-28 08:28:38 +02:00
|
|
|
delete insertion->iib_duplicates;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (result != idx_e_ok) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return result;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* if we are dealing with a foreign key index,
|
|
|
|
check for an insert into the corresponding
|
|
|
|
primary key index */
|
|
|
|
if (idx->idx_flags & idx_foreign) {
|
|
|
|
/* find out if there is a null segment by faking uniqueness --
|
|
|
|
if there is one, don't bother to check the primary key */
|
|
|
|
|
|
|
|
idx->idx_flags |= idx_unique;
|
|
|
|
CCH_FETCH(tdbb, window_ptr, LCK_read, pag_root);
|
2004-03-28 11:10:30 +02:00
|
|
|
temporary_key key;
|
2003-12-22 11:00:59 +01:00
|
|
|
idx_null_state null_state;
|
2003-03-09 22:07:29 +01:00
|
|
|
result = BTR_key(tdbb, relation, record, idx, &key, &null_state);
|
2001-05-23 15:26:42 +02:00
|
|
|
CCH_RELEASE(tdbb, window_ptr);
|
|
|
|
idx->idx_flags &= ~idx_unique;
|
2003-03-09 22:07:29 +01:00
|
|
|
if (null_state == idx_nulls_none) {
|
2003-03-05 12:23:08 +01:00
|
|
|
result =
|
|
|
|
check_foreign_key(tdbb, record, insertion->iib_relation,
|
|
|
|
transaction, idx, bad_relation, bad_index);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-28 11:10:30 +02:00
|
|
|
static bool key_equal(const temporary_key* key1, const temporary_key* key2)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* k e y _ e q u a l
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Compare two keys for equality.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-12-22 11:00:59 +01:00
|
|
|
USHORT l = key1->key_length;
|
|
|
|
if (l != key2->key_length)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (l) {
|
|
|
|
const UCHAR* p = key1->key_data;
|
|
|
|
const UCHAR* q = key2->key_data;
|
|
|
|
do {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (*p++ != *q++)
|
2003-12-22 11:00:59 +01:00
|
|
|
return false;
|
|
|
|
} while (--l);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
return true;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static void signal_index_deletion(thread_db* tdbb, jrd_rel* relation, USHORT id)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s i g n a l _ i n d e x _ d e l e t i o n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* On delete of an index, force all
|
|
|
|
* processes to get rid of index info.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-19 07:14:53 +01:00
|
|
|
IndexBlock* index_block;
|
2004-03-18 06:56:06 +01:00
|
|
|
Lock* lock = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
/* get an exclusive lock on the associated index
|
|
|
|
block (if it exists) to make sure that all other
|
|
|
|
processes flush their cached information about
|
|
|
|
this index */
|
|
|
|
|
|
|
|
for (index_block = relation->rel_index_blocks; index_block;
|
2003-09-13 14:03:11 +02:00
|
|
|
index_block = index_block->idb_next)
|
|
|
|
{
|
|
|
|
if (index_block->idb_id == id) {
|
2001-05-23 15:26:42 +02:00
|
|
|
lock = index_block->idb_lock;
|
|
|
|
break;
|
|
|
|
}
|
2003-09-13 14:03:11 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* if one didn't exist, create it */
|
|
|
|
|
|
|
|
if (!index_block) {
|
|
|
|
index_block = IDX_create_index_block(tdbb, relation, id);
|
|
|
|
lock = index_block->idb_lock;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* signal other processes to clear out the index block */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (lock->lck_physical == LCK_SR) {
|
2004-10-07 11:15:32 +02:00
|
|
|
LCK_convert_non_blocking(tdbb, lock, LCK_EX, LCK_WAIT);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else {
|
2004-10-07 11:15:32 +02:00
|
|
|
LCK_lock_non_blocking(tdbb, lock, LCK_EX, LCK_WAIT);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* and clear out our index block as well */
|
|
|
|
|
|
|
|
index_block_flush(index_block);
|
|
|
|
}
|
|
|
|
|