2001-05-23 15:26:42 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: JRD Access Method
|
|
|
|
* MODULE: tra.c
|
|
|
|
* DESCRIPTION: Transaction 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): ______________________________________.
|
2001-07-10 19:35:13 +02:00
|
|
|
* 2001.07.06 Sean Leyne - Code Cleanup, removed "#ifdef READONLY_DATABASE"
|
|
|
|
* conditionals, as the engine now fully supports
|
|
|
|
* readonly databases.
|
2002-10-29 21:20:44 +01:00
|
|
|
* 2002.10.29 Nickolay Samofatov: Added support for savepoints
|
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>
|
|
|
|
#include "../jrd/jrd.h"
|
|
|
|
#include "../jrd/tra.h"
|
|
|
|
#include "../jrd/ods.h"
|
|
|
|
#include "../jrd/pag.h"
|
|
|
|
#include "../jrd/lck.h"
|
2001-07-29 19:42:23 +02:00
|
|
|
#include "../jrd/gds.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/jrn.h"
|
|
|
|
#include "../jrd/lls.h"
|
|
|
|
#include "../jrd/all.h"
|
|
|
|
#include "../jrd/req.h"
|
|
|
|
#include "../jrd/exe.h"
|
|
|
|
#include "../jrd/rse.h"
|
2003-02-13 23:38:04 +01:00
|
|
|
#include "../jrd/jrd_pwd.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/all_proto.h"
|
|
|
|
#include "../jrd/blb_proto.h"
|
|
|
|
#include "../jrd/cch_proto.h"
|
|
|
|
#include "../jrd/cmp_proto.h"
|
|
|
|
#include "../jrd/dfw_proto.h"
|
|
|
|
#include "../jrd/dpm_proto.h"
|
|
|
|
#include "../jrd/err_proto.h"
|
|
|
|
#include "../jrd/exe_proto.h"
|
|
|
|
#include "../jrd/ext_proto.h"
|
|
|
|
#include "../jrd/gds_proto.h"
|
|
|
|
#include "../jrd/isc_proto.h"
|
|
|
|
#include "../jrd/lck_proto.h"
|
|
|
|
#include "../jrd/met_proto.h"
|
|
|
|
#include "../jrd/mov_proto.h"
|
|
|
|
#include "../jrd/rlck_proto.h"
|
|
|
|
#include "../jrd/sbm_proto.h"
|
|
|
|
#include "../jrd/sch_proto.h"
|
|
|
|
#include "../jrd/thd_proto.h"
|
|
|
|
#include "../jrd/tpc_proto.h"
|
|
|
|
#include "../jrd/tra_proto.h"
|
|
|
|
#include "../jrd/vio_proto.h"
|
|
|
|
#include "../jrd/enc_proto.h"
|
|
|
|
|
|
|
|
#ifndef VMS
|
|
|
|
#include "../lock/lock_proto.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "../jrd/ail.h"
|
|
|
|
|
|
|
|
#define DYN_MSG_FAC 8
|
|
|
|
|
|
|
|
#ifdef VMS
|
|
|
|
#include ssdef
|
|
|
|
#include lckdef
|
|
|
|
|
|
|
|
#define EVENT_FLAG 15
|
|
|
|
|
2003-02-13 11:11:35 +01:00
|
|
|
static const SCHAR lock_types[] =
|
2001-05-24 16:54:26 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
0,
|
|
|
|
LCK$K_NLMODE,
|
|
|
|
LCK$K_CRMODE,
|
|
|
|
LCK$K_CWMODE,
|
|
|
|
LCK$K_PRMODE,
|
|
|
|
LCK$K_PWMODE,
|
|
|
|
LCK$K_EXMODE
|
|
|
|
};
|
|
|
|
#endif /* VMS */
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER_V2
|
|
|
|
static SLONG bump_transaction_id(TDBB, WIN *);
|
|
|
|
#else
|
|
|
|
static HDR bump_transaction_id(TDBB, WIN *);
|
|
|
|
#endif
|
2002-11-17 01:13:59 +01:00
|
|
|
static void retain_context(TDBB, JRD_TRA, USHORT);
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef VMS
|
2002-11-17 01:13:59 +01:00
|
|
|
static void compute_oldest_retaining(TDBB, JRD_TRA, BOOLEAN);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2002-11-17 01:13:59 +01:00
|
|
|
static void downgrade_lock(JRD_TRA);
|
|
|
|
static void expand_view_lock(JRD_TRA, JRD_REL, SCHAR);
|
2001-05-23 15:26:42 +02:00
|
|
|
static TIP fetch_inventory_page(TDBB, WIN *, SLONG, USHORT);
|
2002-07-01 17:07:18 +02:00
|
|
|
static SLONG inventory_page(TDBB, SLONG);
|
2001-05-23 15:26:42 +02:00
|
|
|
static SSHORT limbo_transaction(TDBB, SLONG);
|
2002-11-17 01:13:59 +01:00
|
|
|
static void restart_requests(TDBB, JRD_TRA);
|
2001-05-23 15:26:42 +02:00
|
|
|
static BOOLEAN start_sweeper(TDBB, DBB);
|
|
|
|
static void THREAD_ROUTINE sweep_database(UCHAR *);
|
2002-11-17 01:13:59 +01:00
|
|
|
static void transaction_options(TDBB, JRD_TRA, UCHAR *, USHORT);
|
2001-05-23 15:26:42 +02:00
|
|
|
static BOOLEAN vms_convert(LCK, SLONG *, SCHAR, BOOLEAN);
|
|
|
|
|
2003-02-13 11:11:35 +01:00
|
|
|
static const UCHAR sweep_tpb[] = { isc_tpb_version1, isc_tpb_read,
|
2001-05-23 15:26:42 +02:00
|
|
|
isc_tpb_read_committed, isc_tpb_rec_version
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
BOOLEAN TRA_active_transactions(TDBB tdbb, DBB dbb)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ a c t i v e _ t r a n s a c t i o n s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Determine if any transactions are active.
|
|
|
|
* Return TRUE is active transactions; otherwise
|
|
|
|
* return FALSE if no active transactions.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
#ifndef VMS
|
|
|
|
return ((LCK_query_data(dbb->dbb_lock, LCK_tra, LCK_ANY)) ? TRUE : FALSE);
|
|
|
|
#else
|
|
|
|
WIN window;
|
|
|
|
HDR header;
|
2002-11-17 01:13:59 +01:00
|
|
|
JRD_TRA trans;
|
2001-05-23 15:26:42 +02:00
|
|
|
struct lck temp_lock;
|
|
|
|
USHORT shift, state;
|
|
|
|
ULONG byte, oldest, number, base, active;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
window.win_flags = 0;
|
|
|
|
|
|
|
|
/* Read header page and allocate transaction number. */
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER_V2
|
|
|
|
number = dbb->dbb_next_transaction;
|
|
|
|
oldest = dbb->dbb_oldest_transaction;
|
|
|
|
active = MAX(dbb->dbb_oldest_active, dbb->dbb_oldest_transaction);
|
|
|
|
#else
|
|
|
|
if (dbb->dbb_flags & DBB_read_only) {
|
|
|
|
number = dbb->dbb_next_transaction;
|
|
|
|
oldest = dbb->dbb_oldest_transaction;
|
|
|
|
active = MAX(dbb->dbb_oldest_active, dbb->dbb_oldest_transaction);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
window.win_page = HEADER_PAGE;
|
|
|
|
header = (HDR) CCH_FETCH(tdbb, &window, LCK_read, pag_header);
|
|
|
|
number = header->hdr_next_transaction;
|
|
|
|
oldest = header->hdr_oldest_transaction;
|
|
|
|
active =
|
|
|
|
MAX(header->hdr_oldest_active, header->hdr_oldest_transaction);
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
}
|
|
|
|
#endif /* SUPERSERVER_V2 */
|
|
|
|
|
|
|
|
base = oldest & ~TRA_MASK;
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
trans = FB_NEW_RPT(*dbb->dbb_permanent, (number - base + TRA_MASK) / 4) jrd_tra();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Build transaction bitmap to scan for active transactions. */
|
|
|
|
|
|
|
|
TRA_get_inventory(tdbb, trans->tra_transactions, base, number);
|
|
|
|
|
|
|
|
temp_lock.lck_dbb = dbb;
|
2002-03-31 01:40:08 +01:00
|
|
|
temp_lock.lck_object = reinterpret_cast<blk*>(trans);
|
2001-05-23 15:26:42 +02:00
|
|
|
temp_lock.lck_type = LCK_tra;
|
|
|
|
temp_lock.lck_owner_handle =
|
|
|
|
LCK_get_owner_handle(tdbb, temp_lock.lck_type);
|
|
|
|
temp_lock.lck_parent = dbb->dbb_lock;
|
|
|
|
temp_lock.lck_length = sizeof(SLONG);
|
|
|
|
|
|
|
|
for (; active <= number; active++) {
|
|
|
|
byte = TRANS_OFFSET(active - base);
|
|
|
|
shift = TRANS_SHIFT(active);
|
|
|
|
state = (trans->tra_transactions[byte] >> shift) & TRA_MASK;
|
|
|
|
if (state == tra_active) {
|
|
|
|
temp_lock.lck_key.lck_long = active;
|
|
|
|
if (!LCK_lock_non_blocking(tdbb, &temp_lock, LCK_read, FALSE)) {
|
2001-12-24 03:51:06 +01:00
|
|
|
delete trans;
|
2001-05-23 15:26:42 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
LCK_release(tdbb, &temp_lock);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
delete trans;
|
2001-05-23 15:26:42 +02:00
|
|
|
return FALSE;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void TRA_cleanup(TDBB tdbb)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ c l e a n u p
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* TRA_cleanup is called at startup while an exclusive lock is
|
|
|
|
* held on the database. Because we haven't started a transaction,
|
|
|
|
* and we have an exclusive lock on the db, any transactions marked
|
|
|
|
* as active on the transaction inventory pages are indeed dead.
|
|
|
|
* Mark them so.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DBB dbb;
|
|
|
|
WIN window;
|
|
|
|
HDR header;
|
|
|
|
TIP tip;
|
|
|
|
ATT attachment;
|
|
|
|
UCHAR *byte;
|
2002-07-01 17:07:18 +02:00
|
|
|
SSHORT shift, state;
|
|
|
|
SLONG number, ceiling, active, trans_per_tip;
|
|
|
|
SLONG max, limbo, sequence, last, trans_offset;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
CHECK_DBB(dbb);
|
|
|
|
|
|
|
|
/* Return without cleaning up the TIP's for a ReadOnly database */
|
|
|
|
if (dbb->dbb_flags & DBB_read_only)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* First, make damn sure there are no outstanding transactions */
|
|
|
|
|
|
|
|
for (attachment = dbb->dbb_attachments; attachment;
|
|
|
|
attachment = attachment->att_next) if (attachment->att_transactions)
|
|
|
|
return;
|
|
|
|
|
|
|
|
trans_per_tip = dbb->dbb_pcontrol->pgc_tpt;
|
|
|
|
|
|
|
|
/* Read header page and allocate transaction number. Since
|
|
|
|
the transaction inventory page was initialized to zero, it
|
|
|
|
transaction is automatically marked active. */
|
|
|
|
|
|
|
|
window.win_page = HEADER_PAGE;
|
|
|
|
window.win_flags = 0;
|
|
|
|
header = (HDR) CCH_FETCH(tdbb, &window, LCK_read, pag_header);
|
|
|
|
ceiling = header->hdr_next_transaction;
|
|
|
|
active = header->hdr_oldest_active;
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
|
|
|
|
if (ceiling == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* Zip thru transactions from the "oldest active" to the next looking for
|
|
|
|
active transactions. When one is found, declare it dead. */
|
|
|
|
|
|
|
|
last = ceiling / trans_per_tip;
|
|
|
|
number = active % trans_per_tip;
|
|
|
|
limbo = 0;
|
|
|
|
|
|
|
|
for (sequence = active / trans_per_tip; sequence <= last;
|
|
|
|
sequence++, number = 0) {
|
|
|
|
window.win_page = inventory_page(tdbb, sequence);
|
|
|
|
tip = (TIP) CCH_FETCH(tdbb, &window, LCK_write, pag_transactions);
|
|
|
|
max = ceiling - (sequence * trans_per_tip);
|
|
|
|
if (max > trans_per_tip)
|
|
|
|
max = trans_per_tip - 1;
|
|
|
|
for (; number <= max; number++) {
|
|
|
|
trans_offset = TRANS_OFFSET(number);
|
|
|
|
byte = tip->tip_transactions + trans_offset;
|
|
|
|
shift = TRANS_SHIFT(number);
|
|
|
|
state = (*byte >> shift) & TRA_MASK;
|
|
|
|
if (state == tra_limbo && limbo == 0)
|
|
|
|
limbo = sequence * trans_per_tip + number;
|
|
|
|
else if (state == tra_active) {
|
|
|
|
CCH_MARK(tdbb, &window);
|
|
|
|
*byte &= ~(TRA_MASK << shift);
|
|
|
|
*byte |= tra_dead << shift;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#ifdef SUPERSERVER_V2
|
|
|
|
if (sequence == last) {
|
|
|
|
CCH_MARK(tdbb, &window);
|
|
|
|
for (; number < trans_per_tip; number++) {
|
|
|
|
trans_offset = TRANS_OFFSET(number);
|
|
|
|
byte = tip->tip_transactions + trans_offset;
|
|
|
|
shift = TRANS_SHIFT(number);
|
|
|
|
*byte &= ~(TRA_MASK << shift);
|
|
|
|
if (tip->tip_next)
|
|
|
|
*byte |= tra_committed << shift;
|
|
|
|
else
|
2002-04-29 17:05:11 +02:00
|
|
|
*byte |= tra_active << shift;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER_V2
|
|
|
|
window.win_page = inventory_page(tdbb, last);
|
|
|
|
tip = (TIP) CCH_FETCH(tdbb, &window, LCK_write, pag_transactions);
|
|
|
|
|
|
|
|
while (tip->tip_next) {
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
window.win_page = inventory_page(tdbb, ++last);
|
|
|
|
tip = (TIP) CCH_FETCH(tdbb, &window, LCK_write, pag_transactions);
|
|
|
|
CCH_MARK(tdbb, &window);
|
|
|
|
for (number = 0; number < trans_per_tip; number++) {
|
|
|
|
trans_offset = TRANS_OFFSET(number);
|
|
|
|
byte = tip->tip_transactions + trans_offset;
|
|
|
|
shift = TRANS_SHIFT(number);
|
|
|
|
*byte &= ~(TRA_MASK << shift);
|
|
|
|
if (tip->tip_next || !number)
|
|
|
|
*byte |= tra_committed << shift;
|
|
|
|
else
|
|
|
|
*byte |= tra_active << shift;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!tip->tip_next)
|
|
|
|
dbb->dbb_next_transaction = last * trans_per_tip;
|
|
|
|
}
|
|
|
|
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-17 01:13:59 +01:00
|
|
|
void TRA_commit(TDBB tdbb, JRD_TRA transaction, USHORT retaining_flag)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ c o m m i t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Commit a transaction.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
LCK lock;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
/* If this is a commit retaining, and no updates have been performed,
|
|
|
|
and no events have been posted (via stored procedures etc)
|
|
|
|
no-op the operation */
|
|
|
|
|
|
|
|
if (retaining_flag
|
|
|
|
&& !(transaction->tra_flags & TRA_write
|
|
|
|
|| transaction->tra_deferred_work)) {
|
|
|
|
transaction->tra_flags &= ~TRA_prepared;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (transaction->tra_flags & TRA_invalidated)
|
|
|
|
ERR_post(gds_trans_invalid, 0);
|
|
|
|
|
|
|
|
tdbb->tdbb_default = transaction->tra_pool;
|
|
|
|
|
|
|
|
/* Perform any meta data work deferred */
|
|
|
|
|
|
|
|
if (!(transaction->tra_flags & TRA_prepared))
|
|
|
|
DFW_perform_work(transaction);
|
|
|
|
|
|
|
|
if (transaction->tra_flags & (TRA_prepare2 | TRA_reconnected))
|
|
|
|
MET_update_transaction(tdbb, transaction, TRUE);
|
|
|
|
|
|
|
|
/* Check in with external file system */
|
|
|
|
|
|
|
|
EXT_trans_commit(transaction);
|
|
|
|
|
|
|
|
#ifdef GARBAGE_THREAD
|
|
|
|
/* Flush pages if transaction logically modified data */
|
|
|
|
|
|
|
|
if (transaction->tra_flags & TRA_write)
|
|
|
|
#endif
|
|
|
|
CCH_flush(tdbb, (USHORT) FLUSH_TRAN, transaction->tra_number);
|
|
|
|
#ifdef GARBAGE_THREAD
|
|
|
|
else if (transaction->tra_flags & (TRA_prepare2 | TRA_reconnected)) {
|
|
|
|
/* If the transaction only read data but is a member of a
|
|
|
|
multi-database transaction with a transaction description
|
|
|
|
message then flush RDB$TRANSACTIONS. */
|
|
|
|
|
|
|
|
CCH_flush(tdbb, (USHORT) FLUSH_SYSTEM, 0);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (transaction->tra_flags & TRA_delete_log)
|
|
|
|
AIL_drop_log();
|
|
|
|
|
|
|
|
if (transaction->tra_flags & TRA_add_log)
|
|
|
|
AIL_add_log();
|
|
|
|
|
|
|
|
/* signal refresh range relations for ExpressLink */
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
|
|
RLCK_signal_refresh(transaction);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (retaining_flag) {
|
|
|
|
retain_context(tdbb, transaction, TRUE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set the state on the inventory page to be committed */
|
|
|
|
|
|
|
|
TRA_set_state(tdbb, transaction, transaction->tra_number, tra_committed);
|
|
|
|
|
|
|
|
/* Perform any post commit work */
|
|
|
|
|
|
|
|
DFW_perform_post_commit_work(transaction);
|
|
|
|
|
|
|
|
/* notify any waiting locks that this transaction is committing;
|
|
|
|
there could be no lock if this transaction is being reconnected */
|
|
|
|
|
|
|
|
++transaction->tra_use_count;
|
|
|
|
if ((lock = transaction->tra_lock) && (lock->lck_logical < LCK_write))
|
|
|
|
LCK_convert(tdbb, lock, LCK_write, TRUE);
|
|
|
|
--transaction->tra_use_count;
|
|
|
|
|
|
|
|
TRA_release_transaction(tdbb, transaction);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void TRA_extend_tip(TDBB tdbb, ULONG sequence, WIN * precedence_window)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ e x t e n d _ t i p
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Allocate and link in new TIP (transaction inventory page).
|
|
|
|
* This is called from TRA_start and from validate/repair.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DBB dbb;
|
|
|
|
WIN window, prior_window;
|
|
|
|
TIP tip, prior_tip;
|
|
|
|
VCL vector;
|
|
|
|
JRNI record;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
CHECK_DBB(dbb);
|
|
|
|
window.win_flags = prior_window.win_flags = 0;
|
|
|
|
|
|
|
|
/* Start by fetching prior transaction page, if any */
|
|
|
|
|
|
|
|
if (sequence)
|
|
|
|
prior_tip =
|
|
|
|
fetch_inventory_page(tdbb, &prior_window, (SLONG) (sequence - 1),
|
|
|
|
LCK_write);
|
|
|
|
|
|
|
|
/* Allocate and format new page */
|
|
|
|
|
|
|
|
tip = (TIP) DPM_allocate(tdbb, &window);
|
|
|
|
tip->tip_header.pag_type = pag_transactions;
|
|
|
|
|
|
|
|
CCH_must_write(&window);
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
|
|
|
|
/* Release prior page */
|
|
|
|
|
|
|
|
if (sequence) {
|
|
|
|
CCH_MARK_MUST_WRITE(tdbb, &prior_window);
|
|
|
|
prior_tip->tip_next = window.win_page;
|
|
|
|
if (dbb->dbb_wal) {
|
|
|
|
record.jrni_type = JRNP_NEXT_TIP;
|
|
|
|
record.jrni_position = 0;
|
|
|
|
record.jrni_states = 0;
|
|
|
|
record.jrni_transaction = prior_tip->tip_next;
|
|
|
|
CCH_journal_record(tdbb, &prior_window,
|
|
|
|
reinterpret_cast < UCHAR * >(&record),
|
|
|
|
JRNI_SIZE, 0, 0);
|
|
|
|
}
|
|
|
|
CCH_RELEASE(tdbb, &prior_window);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Link into internal data structures */
|
|
|
|
|
|
|
|
if (!(vector = dbb->dbb_t_pages))
|
2001-12-24 03:51:06 +01:00
|
|
|
vector = vcl::newVector(*dbb->dbb_permanent, sequence + 1);
|
|
|
|
else if (sequence >= vector->count())
|
|
|
|
vector->resize(sequence + 1);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
(*vector)[sequence] = window.win_page;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Write into pages relation */
|
|
|
|
|
|
|
|
DPM_pages(tdbb, 0, pag_transactions, sequence, window.win_page);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int TRA_fetch_state(TDBB tdbb, SLONG number)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ f e t c h _ s t a t e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2001-07-10 19:35:13 +02:00
|
|
|
* Physically fetch the state of a given
|
|
|
|
* transaction on the transaction inventory
|
|
|
|
* page.
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DBB dbb;
|
2002-07-01 17:07:18 +02:00
|
|
|
SLONG trans_per_tip;
|
|
|
|
ULONG tip_number, tip_seq;
|
2001-05-23 15:26:42 +02:00
|
|
|
WIN window;
|
|
|
|
TIP tip;
|
|
|
|
ULONG byte;
|
|
|
|
USHORT shift, state;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
CHECK_DBB(dbb);
|
|
|
|
|
|
|
|
/* locate and fetch the proper TIP page */
|
|
|
|
|
2002-07-01 17:07:18 +02:00
|
|
|
tip_number = (ULONG) number;
|
2001-05-23 15:26:42 +02:00
|
|
|
trans_per_tip = dbb->dbb_pcontrol->pgc_tpt;
|
2002-07-01 17:07:18 +02:00
|
|
|
tip_seq = tip_number / trans_per_tip;
|
2001-05-23 15:26:42 +02:00
|
|
|
window.win_flags = 0;
|
|
|
|
tip = fetch_inventory_page(tdbb, &window, tip_seq, LCK_read);
|
|
|
|
|
|
|
|
/* calculate the state of the desired transaction */
|
|
|
|
|
2002-07-01 17:07:18 +02:00
|
|
|
byte = TRANS_OFFSET(tip_number % trans_per_tip);
|
|
|
|
shift = TRANS_SHIFT(tip_number);
|
2001-05-23 15:26:42 +02:00
|
|
|
state = (tip->tip_transactions[byte] >> shift) & TRA_MASK;
|
|
|
|
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void TRA_get_inventory(TDBB tdbb, UCHAR * bit_vector, ULONG base, ULONG top)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ g e t _ i n v e n t o r y
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Get an inventory of the state of all transactions
|
|
|
|
* between the base and top transactions passed.
|
|
|
|
* To get a consistent view of the transaction
|
2001-07-10 19:35:13 +02:00
|
|
|
* inventory (in case we ever implement sub-transactions),
|
2001-05-23 15:26:42 +02:00
|
|
|
* do handoffs to read the pages in order.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DBB dbb;
|
|
|
|
ULONG trans_per_tip;
|
2002-07-01 17:07:18 +02:00
|
|
|
ULONG sequence, last, l;
|
2001-05-23 15:26:42 +02:00
|
|
|
WIN window;
|
|
|
|
TIP tip;
|
|
|
|
UCHAR *p, *q;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
CHECK_DBB(dbb);
|
|
|
|
|
|
|
|
trans_per_tip = dbb->dbb_pcontrol->pgc_tpt;
|
|
|
|
sequence = base / trans_per_tip;
|
|
|
|
last = top / trans_per_tip;
|
|
|
|
|
|
|
|
/* fetch the first inventory page */
|
|
|
|
|
|
|
|
window.win_flags = 0;
|
|
|
|
tip = fetch_inventory_page(tdbb, &window, (SLONG) sequence++, LCK_read);
|
|
|
|
|
|
|
|
/* move the first page into the bit vector */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if ( (p = bit_vector) ) {
|
2001-05-23 15:26:42 +02:00
|
|
|
l = base % trans_per_tip;
|
|
|
|
q = tip->tip_transactions + TRANS_OFFSET(l);
|
|
|
|
l = TRANS_OFFSET(MIN((top + TRA_MASK - base), trans_per_tip - l));
|
|
|
|
MOVE_FAST(q, p, l);
|
|
|
|
p += l;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* move successive pages into the bit vector */
|
|
|
|
|
|
|
|
while (sequence <= last) {
|
|
|
|
base = sequence * trans_per_tip;
|
|
|
|
|
|
|
|
/* release the read lock as we go, so that some one else can
|
|
|
|
* commit without having to signal all other transactions.
|
|
|
|
*/
|
|
|
|
|
|
|
|
tip =
|
|
|
|
(TIP) CCH_HANDOFF(tdbb, &window, inventory_page(tdbb, sequence++),
|
|
|
|
LCK_read, pag_transactions);
|
|
|
|
TPC_update_cache(tdbb, tip, sequence - 1);
|
|
|
|
if (p) {
|
|
|
|
l = TRANS_OFFSET(MIN((top + TRA_MASK - base), trans_per_tip));
|
|
|
|
MOVE_FAST(tip->tip_transactions, p, l);
|
|
|
|
p += l;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int TRA_get_state(TDBB tdbb, SLONG number)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ g e t _ s t a t e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2001-07-10 19:35:13 +02:00
|
|
|
* Get the state of a given transaction on the
|
2001-05-23 15:26:42 +02:00
|
|
|
* transaction inventory page.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DBB dbb;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
CHECK_DBB(dbb);
|
|
|
|
|
|
|
|
if (dbb->dbb_tip_cache)
|
|
|
|
return TPC_snapshot_state(tdbb, number);
|
|
|
|
|
|
|
|
if (number && dbb->dbb_pc_transactions)
|
|
|
|
if (TRA_precommited(tdbb, number, number))
|
|
|
|
return tra_precommitted;
|
|
|
|
|
|
|
|
return TRA_fetch_state(tdbb, number);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER_V2
|
|
|
|
void TRA_header_write(TDBB tdbb, DBB dbb, SLONG number)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ h e a d e r _ w r i t e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Force transaction ID on header to disk.
|
|
|
|
* Do post fetch check of the transaction
|
|
|
|
* ID header write as a concurrent thread
|
|
|
|
* might have written the header page
|
|
|
|
* while blocked on the latch.
|
|
|
|
*
|
|
|
|
* The idea is to amortize the cost of
|
|
|
|
* header page I/O across multiple transactions.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
WIN window;
|
|
|
|
HDR header;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
/* If transaction number is already on disk just return. */
|
|
|
|
|
|
|
|
if (!number || dbb->dbb_last_header_write < number) {
|
|
|
|
window.win_page = HEADER_PAGE;
|
|
|
|
window.win_flags = 0;
|
|
|
|
header = (HDR) CCH_FETCH(tdbb, &window, LCK_write, pag_header);
|
|
|
|
|
|
|
|
if (header->hdr_next_transaction) {
|
|
|
|
if (header->hdr_oldest_active > header->hdr_next_transaction)
|
|
|
|
BUGCHECK(266); /*next transaction older than oldest active */
|
|
|
|
|
|
|
|
if (header->hdr_oldest_transaction > header->hdr_next_transaction)
|
|
|
|
BUGCHECK(267); /* next transaction older than oldest transaction */
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The header page might have been written while waiting
|
|
|
|
for the latch; perform a post fetch check and optimize
|
|
|
|
this case by not writing the page again. */
|
|
|
|
|
|
|
|
if (!number || dbb->dbb_last_header_write < number) {
|
|
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
|
|
|
if (dbb->dbb_next_transaction > header->hdr_next_transaction)
|
|
|
|
header->hdr_next_transaction = dbb->dbb_next_transaction;
|
|
|
|
|
|
|
|
if (dbb->dbb_oldest_active > header->hdr_oldest_active)
|
|
|
|
header->hdr_oldest_active = dbb->dbb_oldest_active;
|
|
|
|
|
|
|
|
if (dbb->dbb_oldest_transaction > header->hdr_oldest_transaction)
|
|
|
|
header->hdr_oldest_transaction = dbb->dbb_oldest_transaction;
|
|
|
|
|
|
|
|
if (dbb->dbb_oldest_snapshot > header->hdr_oldest_snapshot)
|
|
|
|
header->hdr_oldest_snapshot = dbb->dbb_oldest_snapshot;
|
|
|
|
}
|
|
|
|
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
void TRA_init(TDBB tdbb)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ i n i t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* "Start" the system transaction.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DBB dbb;
|
2002-11-17 01:13:59 +01:00
|
|
|
JRD_TRA trans;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
CHECK_DBB(dbb);
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
dbb->dbb_sys_trans = trans = FB_NEW_RPT(*dbb->dbb_permanent, 0) jrd_tra();
|
2001-05-23 15:26:42 +02:00
|
|
|
trans->tra_flags |= TRA_system | TRA_ignore_limbo;
|
|
|
|
trans->tra_pool = dbb->dbb_permanent;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void TRA_invalidate(DBB database, ULONG mask)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ i n v a l i d a t e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Invalidate any active transactions that may have
|
|
|
|
* modified a page that couldn't be written.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
ATT attachment;
|
2002-11-17 01:13:59 +01:00
|
|
|
JRD_TRA transaction;
|
2001-05-23 15:26:42 +02:00
|
|
|
ULONG transaction_mask;
|
|
|
|
|
|
|
|
for (attachment = database->dbb_attachments; attachment;
|
|
|
|
attachment = attachment->att_next) for (transaction =
|
|
|
|
attachment->att_transactions;
|
|
|
|
transaction;
|
|
|
|
transaction =
|
|
|
|
transaction->tra_next) {
|
|
|
|
transaction_mask =
|
|
|
|
1L << (transaction->tra_number & (BITS_PER_LONG - 1));
|
|
|
|
if (transaction_mask & mask && transaction->tra_flags & TRA_write)
|
|
|
|
transaction->tra_flags |= TRA_invalidated;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-17 01:13:59 +01:00
|
|
|
void TRA_link_transaction(TDBB tdbb, JRD_TRA transaction)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ l i n k _ t r a n s a c t i o n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Link transaction block into database attachment.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
ATT attachment;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
transaction->tra_attachment = attachment = tdbb->tdbb_attachment;
|
|
|
|
transaction->tra_next = attachment->att_transactions;
|
|
|
|
attachment->att_transactions = transaction;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-17 01:13:59 +01:00
|
|
|
void TRA_post_resources(TDBB tdbb, JRD_TRA transaction, RSC resources)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ p o s t _ r e s o u r c e s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Post interest in relation/procedure existence to transaction.
|
|
|
|
* This guarantees that the relation/procedure won't be dropped
|
|
|
|
* out from under the transaction.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
RSC rsc, tra_rsc, new_rsc;
|
2001-12-24 03:51:06 +01:00
|
|
|
JrdMemoryPool *old_pool;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
old_pool = tdbb->tdbb_default;
|
|
|
|
tdbb->tdbb_default = transaction->tra_pool;
|
|
|
|
|
|
|
|
for (rsc = resources; rsc; rsc = rsc->rsc_next)
|
|
|
|
if (rsc->rsc_type == rsc_relation || rsc->rsc_type == rsc_procedure) {
|
|
|
|
for (tra_rsc = transaction->tra_resources; tra_rsc;
|
|
|
|
tra_rsc =
|
|
|
|
tra_rsc->rsc_next) if (rsc->rsc_id == tra_rsc->rsc_id) break;
|
|
|
|
if (!tra_rsc) {
|
2002-09-25 19:12:16 +02:00
|
|
|
new_rsc = FB_NEW(*tdbb->tdbb_default) Rsc();
|
2001-05-23 15:26:42 +02:00
|
|
|
new_rsc->rsc_next = transaction->tra_resources;
|
|
|
|
transaction->tra_resources = new_rsc;
|
|
|
|
new_rsc->rsc_id = rsc->rsc_id;
|
|
|
|
new_rsc->rsc_type = rsc->rsc_type;
|
|
|
|
switch (rsc->rsc_type) {
|
|
|
|
case rsc_relation:
|
|
|
|
new_rsc->rsc_rel = rsc->rsc_rel;
|
|
|
|
MET_post_existence(tdbb, new_rsc->rsc_rel);
|
|
|
|
break;
|
|
|
|
case rsc_procedure:
|
|
|
|
new_rsc->rsc_prc = rsc->rsc_prc;
|
|
|
|
new_rsc->rsc_prc->prc_use_count++;
|
|
|
|
#ifdef DEBUG_PROCS
|
|
|
|
{
|
|
|
|
char buffer[256];
|
|
|
|
sprintf(buffer,
|
|
|
|
"Called from TRA_post_resources():\n\t Incrementing use count of %s\n",
|
|
|
|
new_rsc->rsc_prc->prc_name->str_data);
|
|
|
|
JRD_print_procedure_info(tdbb, buffer);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
break;
|
2001-12-24 03:51:06 +01:00
|
|
|
default: /* Shut up compiler warning */
|
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
tdbb->tdbb_default = old_pool;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BOOLEAN TRA_precommited(TDBB tdbb, SLONG old_number, SLONG new_number)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ p r e c o m m i t e d (s i c)
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2001-07-10 19:35:13 +02:00
|
|
|
* Maintain a vector of active precommitted
|
|
|
|
* transactions. If old_number <> new_number
|
2001-05-23 15:26:42 +02:00
|
|
|
* then swap old_number with new_number in
|
|
|
|
* the vector. If old_number equals new_number
|
|
|
|
* then test for that number's presence in
|
|
|
|
* the vector.
|
|
|
|
*
|
|
|
|
**************************************/
|
2001-12-24 03:51:06 +01:00
|
|
|
vcl::iterator p, end;
|
|
|
|
SLONG *zp;
|
2001-05-23 15:26:42 +02:00
|
|
|
VCL vector;
|
|
|
|
DBB dbb;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
CHECK_DBB(dbb);
|
|
|
|
|
|
|
|
if (!(vector = dbb->dbb_pc_transactions)) {
|
|
|
|
if (old_number == new_number)
|
|
|
|
return FALSE;
|
2001-12-24 03:51:06 +01:00
|
|
|
vector = dbb->dbb_pc_transactions = vcl::newVector(*dbb->dbb_permanent, 1);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
zp = 0;
|
2001-12-24 03:51:06 +01:00
|
|
|
for (p = vector->begin(), end = vector->end(); p < end; p++) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (*p == old_number)
|
|
|
|
return (*p = new_number) ? TRUE : FALSE;
|
|
|
|
if (!zp && !*p)
|
2002-01-04 12:34:22 +01:00
|
|
|
zp = &*p;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (old_number == new_number || new_number == 0)
|
|
|
|
return FALSE;
|
|
|
|
if (zp)
|
|
|
|
*zp = new_number;
|
|
|
|
else {
|
2001-12-24 03:51:06 +01:00
|
|
|
vector->resize(vector->count() + 1);
|
|
|
|
(*vector)[vector->count() - 1] = new_number;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-17 01:13:59 +01:00
|
|
|
void TRA_prepare(TDBB tdbb, JRD_TRA transaction, USHORT length, UCHAR * msg)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ p r e p a r e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Put a transaction into limbo.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
if (transaction->tra_flags & TRA_prepared)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (transaction->tra_flags & TRA_invalidated)
|
|
|
|
ERR_post(gds_trans_invalid, 0);
|
|
|
|
|
2001-07-10 19:35:13 +02:00
|
|
|
/* If there's a transaction description message, log it to RDB$TRANSACTION
|
2001-05-23 15:26:42 +02:00
|
|
|
We should only log a message to RDB$TRANSACTION if there is a message
|
|
|
|
to log (if the length = 0, we won't log the transaction in RDB$TRANSACTION)
|
|
|
|
These messages are used to recover transactions in limbo. The message indicates
|
|
|
|
the action that is to be performed (hence, if nothing is getting logged, don't
|
|
|
|
bother).
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Make sure that if msg is NULL there is no length. The two
|
|
|
|
should go hand in hand
|
2001-07-10 19:35:13 +02:00
|
|
|
msg == NULL || *msg == NULL
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
|
|
|
assert(!(!msg && length) || (msg && (!*msg && length)));
|
|
|
|
|
|
|
|
if (msg && length) {
|
|
|
|
MET_prepare(tdbb, transaction, length, msg);
|
|
|
|
transaction->tra_flags |= TRA_prepare2;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check in with external file system */
|
|
|
|
|
|
|
|
EXT_trans_prepare(transaction);
|
|
|
|
|
|
|
|
/* Perform any meta data work deferred */
|
|
|
|
|
|
|
|
DFW_perform_work(transaction);
|
|
|
|
|
|
|
|
#ifdef GARBAGE_THREAD
|
|
|
|
/* Flush pages if transaction logically modified data */
|
|
|
|
|
|
|
|
if (transaction->tra_flags & TRA_write)
|
|
|
|
#endif
|
|
|
|
CCH_flush(tdbb, (USHORT) FLUSH_TRAN, transaction->tra_number);
|
|
|
|
#ifdef GARBAGE_THREAD
|
|
|
|
else if (transaction->tra_flags & TRA_prepare2) {
|
|
|
|
/* If the transaction only read data but is a member of a
|
|
|
|
multi-database transaction with a transaction description
|
|
|
|
message then flush RDB$TRANSACTIONS. */
|
|
|
|
|
|
|
|
CCH_flush(tdbb, (USHORT) FLUSH_SYSTEM, 0);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Set the state on the inventory page to be limbo */
|
|
|
|
|
|
|
|
transaction->tra_flags |= TRA_prepared;
|
|
|
|
TRA_set_state(tdbb, transaction, transaction->tra_number, tra_limbo);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-17 01:13:59 +01:00
|
|
|
JRD_TRA TRA_reconnect(TDBB tdbb, UCHAR * id, USHORT length)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ r e c o n n e c t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Reconnect to a transaction in limbo.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DBB dbb;
|
2002-11-17 01:13:59 +01:00
|
|
|
JRD_TRA trans;
|
2001-05-23 15:26:42 +02:00
|
|
|
UCHAR state;
|
|
|
|
TEXT text[128];
|
|
|
|
USHORT flags, message;
|
|
|
|
SLONG number;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
CHECK_DBB(dbb);
|
|
|
|
|
|
|
|
/* Cannot work on limbo transactions for ReadOnly database */
|
|
|
|
if (dbb->dbb_flags & DBB_read_only)
|
|
|
|
ERR_post(isc_read_only_database, 0);
|
|
|
|
|
|
|
|
|
2003-01-16 18:47:10 +01:00
|
|
|
tdbb->tdbb_default = JrdMemoryPool::createPool();
|
2002-11-21 00:18:16 +01:00
|
|
|
trans = FB_NEW_RPT(*tdbb->tdbb_default, 0) jrd_tra();
|
2001-05-23 15:26:42 +02:00
|
|
|
trans->tra_pool = tdbb->tdbb_default;
|
|
|
|
trans->tra_number = gds__vax_integer(id, length);
|
|
|
|
trans->tra_flags |= TRA_prepared | TRA_reconnected | TRA_write;
|
|
|
|
|
|
|
|
if ((state = limbo_transaction(tdbb, trans->tra_number)) != tra_limbo) {
|
|
|
|
switch (state) {
|
|
|
|
case tra_active:
|
|
|
|
message = 262; /* ACTIVE */
|
|
|
|
break;
|
|
|
|
case tra_dead:
|
|
|
|
message = 264; /* ROLLED BACK */
|
|
|
|
break;
|
|
|
|
case tra_committed:
|
|
|
|
message = 263; /* COMMITTED */
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
message = 265; /* ILL DEFINED */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
number = trans->tra_number;
|
2003-01-16 18:47:10 +01:00
|
|
|
JrdMemoryPool::deletePool(trans->tra_pool);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
gds__msg_lookup(0, JRD_BUGCHK, message, sizeof(text), text, &flags);
|
|
|
|
|
|
|
|
ERR_post(gds_no_recon,
|
|
|
|
gds_arg_gds, gds_tra_state,
|
|
|
|
gds_arg_number, number,
|
|
|
|
gds_arg_string, ERR_cstring(text), 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
TRA_link_transaction(tdbb, trans);
|
|
|
|
|
|
|
|
return trans;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-17 01:13:59 +01:00
|
|
|
void TRA_release_transaction(TDBB tdbb, JRD_TRA transaction)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ r e l e a s e _ t r a n s a c t i o n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Cleanup a transaction. This is called by both COMMIT and
|
|
|
|
* ROLLBACK as well as code in JRD to get rid of remote
|
|
|
|
* transactions.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
VEC vector;
|
2001-12-24 03:51:06 +01:00
|
|
|
vec::iterator lock;
|
2002-11-17 01:13:59 +01:00
|
|
|
JRD_TRA *ptr;
|
2001-05-23 15:26:42 +02:00
|
|
|
RSC rsc;
|
|
|
|
USHORT i;
|
2001-12-24 03:51:06 +01:00
|
|
|
JrdMemoryPool *tra_pool;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
while (transaction->tra_blobs)
|
|
|
|
BLB_cancel(tdbb, transaction->tra_blobs);
|
|
|
|
|
|
|
|
while (transaction->tra_arrays)
|
|
|
|
BLB_release_array(transaction->tra_arrays);
|
|
|
|
|
|
|
|
if (transaction->tra_pool)
|
|
|
|
CMP_expunge_transaction(transaction);
|
|
|
|
|
|
|
|
/* Release interest in relation/procedure existence for transaction */
|
|
|
|
|
|
|
|
for (rsc = transaction->tra_resources; rsc; rsc = rsc->rsc_next)
|
|
|
|
switch (rsc->rsc_type) {
|
|
|
|
case rsc_procedure:
|
|
|
|
CMP_decrement_prc_use_count(tdbb, rsc->rsc_prc);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
MET_release_existence(rsc->rsc_rel);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Release the locks associated with the transaction */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if ( (vector = transaction->tra_relation_locks) )
|
|
|
|
for (i = 0, lock = vector->begin(); i < vector->count();
|
2001-05-23 15:26:42 +02:00
|
|
|
i++, lock++)
|
|
|
|
if (*lock)
|
2001-12-24 03:51:06 +01:00
|
|
|
LCK_release(tdbb, (LCK)*lock);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
++transaction->tra_use_count;
|
|
|
|
if (transaction->tra_lock)
|
|
|
|
LCK_release(tdbb, transaction->tra_lock);
|
|
|
|
--transaction->tra_use_count;
|
|
|
|
|
|
|
|
/* release the sparse bit map used for commit retain transaction */
|
|
|
|
|
|
|
|
SBM_release(transaction->tra_commit_sub_trans);
|
|
|
|
|
|
|
|
if (transaction->tra_flags & TRA_precommitted)
|
|
|
|
(void) TRA_precommited(tdbb, transaction->tra_number, 0);
|
|
|
|
|
|
|
|
/* Unlink the transaction from the database block */
|
|
|
|
|
|
|
|
for (ptr = &tdbb->tdbb_attachment->att_transactions;
|
|
|
|
*ptr; ptr = &(*ptr)->tra_next) if (*ptr == transaction) {
|
|
|
|
*ptr = transaction->tra_next;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Release the transaction pool. */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if ( (tra_pool = transaction->tra_pool) )
|
2003-01-16 18:47:10 +01:00
|
|
|
JrdMemoryPool::deletePool(tra_pool);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-17 01:13:59 +01:00
|
|
|
void TRA_rollback(TDBB tdbb, JRD_TRA transaction, USHORT retaining_flag)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ r o l l b a c k
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Rollback a transaction.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
tdbb->tdbb_default = transaction->tra_pool;
|
|
|
|
|
|
|
|
/* Check in with external file system */
|
|
|
|
|
|
|
|
EXT_trans_rollback(transaction);
|
|
|
|
|
|
|
|
/* If no writes have been made, commit the transaction instead. */
|
|
|
|
|
|
|
|
if (transaction->tra_flags & (TRA_prepare2 | TRA_reconnected))
|
|
|
|
MET_update_transaction(tdbb, transaction, FALSE);
|
|
|
|
|
2002-11-03 18:29:51 +01:00
|
|
|
/* Find out if there is a transaction savepoint we can use to rollback our transaction */
|
2002-10-29 21:20:44 +01:00
|
|
|
BOOLEAN tran_sav = FALSE;
|
|
|
|
for (SAV temp = transaction->tra_save_point; temp; temp=temp->sav_next)
|
2002-11-03 18:29:51 +01:00
|
|
|
if (temp->sav_flags & SAV_trans_level) {
|
2002-10-29 21:20:44 +01:00
|
|
|
tran_sav = TRUE;
|
2002-11-03 18:29:51 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Measure transaction savepoint size if there is one. We'll use it for undo
|
|
|
|
only if it is small enough */
|
|
|
|
SLONG count = SAV_LARGE;
|
|
|
|
if (tran_sav) {
|
|
|
|
for (SAV temp = transaction->tra_save_point; temp; temp=temp->sav_next) {
|
|
|
|
count = VIO_savepoint_large(temp, count);
|
|
|
|
if (count < 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// We are going to use savepoint to undo transaction
|
|
|
|
if (tran_sav && count > 0) {
|
|
|
|
// Undo all user savepoints work
|
2002-10-29 21:20:44 +01:00
|
|
|
while (transaction->tra_save_point->sav_flags & SAV_user) {
|
|
|
|
++transaction->tra_save_point->sav_verb_count; /* cause undo */
|
|
|
|
VIO_verb_cleanup(tdbb, transaction);
|
2002-11-03 18:29:51 +01:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Free all savepoint data
|
|
|
|
// We can do it in reverse order because nothing except simple deallocation
|
|
|
|
// of memory is really done in VIO_verb_cleanup when we pass NULL as sav_next
|
|
|
|
while (transaction->tra_save_point && transaction->tra_save_point->sav_flags & SAV_user) {
|
|
|
|
SAV next=transaction->tra_save_point->sav_next;
|
|
|
|
transaction->tra_save_point->sav_next = NULL;
|
|
|
|
VIO_verb_cleanup(tdbb, transaction);
|
|
|
|
transaction->tra_save_point = next;
|
2002-10-29 21:20:44 +01:00
|
|
|
}
|
2002-11-03 18:29:51 +01:00
|
|
|
if (transaction->tra_save_point) {
|
|
|
|
if (!(transaction->tra_save_point->sav_flags & SAV_trans_level))
|
|
|
|
BUGCHECK(287); /* Too many savepoints */
|
|
|
|
/* This transaction savepoint contains wrong data now. Clean it up */
|
|
|
|
VIO_verb_cleanup(tdbb, transaction); /* get rid of transaction savepoint */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Only transaction savepoint could be there
|
|
|
|
if (transaction->tra_save_point)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!(transaction->tra_save_point->sav_flags & SAV_trans_level))
|
|
|
|
BUGCHECK(287); /* Too many savepoints */
|
|
|
|
|
|
|
|
/* Make sure that any error during savepoint undo is handled by marking
|
|
|
|
the transaction as dead. */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
try {
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* In an attempt to avoid deadlocks, clear the precedence by writing
|
|
|
|
all dirty buffers for this transaction. */
|
|
|
|
|
|
|
|
if (transaction->tra_flags & TRA_write) {
|
|
|
|
CCH_flush(tdbb, (USHORT) FLUSH_TRAN, transaction->tra_number);
|
|
|
|
++transaction->tra_save_point->sav_verb_count; /* cause undo */
|
|
|
|
VIO_verb_cleanup(tdbb, transaction);
|
|
|
|
CCH_flush(tdbb, (USHORT) FLUSH_TRAN, transaction->tra_number);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
VIO_verb_cleanup(tdbb, transaction);
|
|
|
|
|
|
|
|
/* If this a abort retaining transaction, do not
|
|
|
|
* set the TIP bits. That will be done in retain_context ().
|
|
|
|
* Only ExpressLink runs in this mode and only DDL is
|
|
|
|
* executed in this manner.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (!retaining_flag)
|
|
|
|
TRA_set_state(tdbb, transaction, transaction->tra_number,
|
|
|
|
tra_committed);
|
|
|
|
}
|
2003-02-13 14:33:57 +01:00
|
|
|
catch (const std::exception&) {
|
2001-12-24 03:51:06 +01:00
|
|
|
/* Prevent a bugcheck in TRA_set_state to cause a loop */
|
|
|
|
/* Clear the error because the rollback will succeed. */
|
|
|
|
tdbb->tdbb_status_vector[0] = gds_arg_gds;
|
|
|
|
tdbb->tdbb_status_vector[1] = 0;
|
|
|
|
tdbb->tdbb_status_vector[2] = gds_arg_end;
|
|
|
|
TRA_set_state(tdbb, transaction, transaction->tra_number,
|
|
|
|
tra_dead);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* Set the state on the inventory page to be dead */
|
|
|
|
TRA_set_state(tdbb, transaction, transaction->tra_number, tra_dead);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if this is a rollback retain (used only by ExpressLink), abort
|
|
|
|
* this transaction and start a new one.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (retaining_flag) {
|
|
|
|
retain_context(tdbb, transaction, FALSE);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
TRA_release_transaction(tdbb, transaction);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-17 01:13:59 +01:00
|
|
|
void TRA_set_state(TDBB tdbb, JRD_TRA transaction, SLONG number, SSHORT state)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ s e t _ s t a t e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Set the state of a transaction in the inventory page.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DBB dbb;
|
|
|
|
WIN window;
|
|
|
|
TIP tip;
|
|
|
|
JRNI record;
|
|
|
|
UCHAR *address;
|
2002-07-01 17:07:18 +02:00
|
|
|
SSHORT shift;
|
|
|
|
SLONG sequence;
|
2001-05-23 15:26:42 +02:00
|
|
|
ULONG byte, trans_per_tip;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
CHECK_DBB(dbb);
|
|
|
|
|
|
|
|
/* If we're terminating ourselves and we've
|
|
|
|
been precommitted then just return. */
|
|
|
|
|
|
|
|
if (transaction &&
|
|
|
|
transaction->tra_number == number &&
|
|
|
|
transaction->tra_flags & TRA_precommitted) return;
|
|
|
|
|
|
|
|
/* If it is a ReadOnly DB, set the new state in the TIP cache and return */
|
|
|
|
if ((dbb->dbb_flags & DBB_read_only) && dbb->dbb_tip_cache) {
|
|
|
|
TPC_set_state(tdbb, number, state);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
trans_per_tip = dbb->dbb_pcontrol->pgc_tpt;
|
|
|
|
sequence = number / trans_per_tip;
|
|
|
|
trans_per_tip = dbb->dbb_pcontrol->pgc_tpt;
|
|
|
|
byte = TRANS_OFFSET(number % trans_per_tip);
|
|
|
|
shift = TRANS_SHIFT(number);
|
|
|
|
|
|
|
|
window.win_flags = 0;
|
|
|
|
tip = fetch_inventory_page(tdbb, &window, (SLONG) sequence, LCK_write);
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER_V2
|
|
|
|
CCH_MARK(tdbb, &window);
|
|
|
|
ULONG generation = tip->tip_header.pag_generation;
|
|
|
|
#else
|
|
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* set the state on the TIP page */
|
|
|
|
|
|
|
|
address = tip->tip_transactions + byte;
|
|
|
|
*address &= ~(TRA_MASK << shift);
|
|
|
|
*address |= state << shift;
|
|
|
|
|
|
|
|
/* set the new state in the TIP cache as well */
|
|
|
|
|
|
|
|
if (dbb->dbb_tip_cache)
|
|
|
|
TPC_set_state(tdbb, number, state);
|
|
|
|
|
|
|
|
/* If journalling is enabled, journal the change */
|
|
|
|
|
|
|
|
if ((dbb->dbb_wal) && (state == tra_committed)) {
|
|
|
|
record.jrni_type = JRNP_TRANSACTION;
|
|
|
|
record.jrni_position = byte;
|
|
|
|
record.jrni_states = *address;
|
|
|
|
record.jrni_transaction = number;
|
|
|
|
CCH_journal_record(tdbb, &window,
|
|
|
|
reinterpret_cast < UCHAR * >(&record),
|
|
|
|
sizeof(record), 0, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER_V2
|
|
|
|
/* Let the TIP be lazily updated for read-only queries.
|
|
|
|
To amortize write of TIP page for update transactions,
|
|
|
|
exit engine to allow other transactions to update the TIP
|
|
|
|
and use page generation to determine if page was written. */
|
|
|
|
|
|
|
|
if (transaction && !(transaction->tra_flags & TRA_write))
|
|
|
|
return;
|
|
|
|
else {
|
|
|
|
THREAD_EXIT;
|
|
|
|
THREAD_ENTER;
|
2002-04-29 17:05:11 +02:00
|
|
|
tip = reinterpret_cast<TIP>(CCH_FETCH(tdbb, &window, LCK_write, pag_transactions));
|
2001-05-23 15:26:42 +02:00
|
|
|
if (generation == tip->tip_header.pag_generation)
|
|
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if ((dbb->dbb_wal) && (state == tra_committed)) {
|
|
|
|
AIL_commit(number);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void TRA_shutdown_attachment(TDBB tdbb, ATT attachment)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ s h u t d o w n _ a t t a c h m e n t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Release locks associated with transactions for attachment.
|
|
|
|
*
|
|
|
|
**************************************/
|
2002-11-17 01:13:59 +01:00
|
|
|
JRD_TRA transaction;
|
2001-05-23 15:26:42 +02:00
|
|
|
VEC vector;
|
2001-12-24 03:51:06 +01:00
|
|
|
vec::iterator lock;
|
2001-05-23 15:26:42 +02:00
|
|
|
USHORT i;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
for (transaction = attachment->att_transactions; transaction;
|
|
|
|
transaction = transaction->tra_next) {
|
|
|
|
/* Release the relation locks associated with the transaction */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if ( (vector = transaction->tra_relation_locks) )
|
|
|
|
for (i = 0, lock = vector->begin();
|
|
|
|
i < vector->count(); i++, lock++)
|
2001-05-23 15:26:42 +02:00
|
|
|
if (*lock)
|
2001-12-24 03:51:06 +01:00
|
|
|
LCK_release(tdbb, (LCK)*lock);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Release transaction lock itself */
|
|
|
|
|
|
|
|
++transaction->tra_use_count;
|
|
|
|
if (transaction->tra_lock)
|
|
|
|
LCK_release(tdbb, transaction->tra_lock);
|
|
|
|
--transaction->tra_use_count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-17 01:13:59 +01:00
|
|
|
int TRA_snapshot_state(TDBB tdbb, JRD_TRA trans, SLONG number)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ s n a p s h o t _ s t a t e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Get the state of a numbered transaction when a
|
|
|
|
* transaction started.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
if (number && TRA_precommited(tdbb, number, number))
|
|
|
|
return tra_precommitted;
|
|
|
|
|
|
|
|
if (number == trans->tra_number)
|
|
|
|
return tra_us;
|
|
|
|
|
2001-07-10 19:35:13 +02:00
|
|
|
/* If the transaction is older than the oldest
|
2001-05-23 15:26:42 +02:00
|
|
|
interesting transaction, it must be committed. */
|
|
|
|
|
|
|
|
if (number < trans->tra_oldest)
|
|
|
|
return tra_committed;
|
|
|
|
|
|
|
|
/* If the transaction is the system transaction, it is considered committed. */
|
|
|
|
|
|
|
|
if (number == 0 || (trans->tra_flags & TRA_system))
|
|
|
|
return tra_committed;
|
|
|
|
|
|
|
|
/* If the transaction is a commited sub-transction - do the easy lookup.
|
2001-07-10 19:35:13 +02:00
|
|
|
Since this is not the most common case, and looking up the
|
2001-05-23 15:26:42 +02:00
|
|
|
transaction cache for read committed transactions is equally
|
|
|
|
fast, just to that instead. */
|
|
|
|
|
|
|
|
if (trans->tra_flags & TRA_read_committed)
|
|
|
|
return TPC_snapshot_state(tdbb, number);
|
|
|
|
|
|
|
|
if (trans->tra_commit_sub_trans &&
|
|
|
|
SBM_test(trans->tra_commit_sub_trans, number)) return tra_committed;
|
|
|
|
|
|
|
|
/* If the transaction is younger than we are, it must be considered
|
|
|
|
active. */
|
|
|
|
|
|
|
|
if (number > trans->tra_top)
|
|
|
|
return tra_active;
|
|
|
|
|
|
|
|
return TRA_state(trans->tra_transactions, trans->tra_oldest, number);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-17 01:13:59 +01:00
|
|
|
JRD_TRA TRA_start(TDBB tdbb, int tpb_length, SCHAR * tpb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ s t a r t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Start a user transaction.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DBB dbb;
|
|
|
|
ATT attachment;
|
|
|
|
WIN window;
|
|
|
|
HDR header;
|
2002-11-17 01:13:59 +01:00
|
|
|
JRD_TRA trans, temp;
|
2001-05-23 15:26:42 +02:00
|
|
|
LCK lock;
|
|
|
|
struct lck temp_lock;
|
|
|
|
USHORT shift, oldest_state, cleanup;
|
|
|
|
ULONG byte, oldest, number, base, active, oldest_active, oldest_snapshot;
|
|
|
|
SLONG data;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
attachment = tdbb->tdbb_attachment;
|
|
|
|
window.win_flags = 0;
|
|
|
|
|
|
|
|
if (dbb->dbb_ast_flags & DBB_shut_tran)
|
|
|
|
ERR_post(gds_shutinprog, gds_arg_cstring,
|
2002-06-05 11:53:59 +02:00
|
|
|
tdbb->tdbb_attachment->att_filename->str_length,
|
|
|
|
tdbb->tdbb_attachment->att_filename->str_data,
|
2001-05-23 15:26:42 +02:00
|
|
|
0);
|
|
|
|
|
|
|
|
/* To handle the problems of relation locks, allocate a temporary
|
|
|
|
transaction block first, sieze relation locks, the go ahead and
|
|
|
|
make up the real transaction block. */
|
|
|
|
|
2003-01-16 18:47:10 +01:00
|
|
|
tdbb->tdbb_default = JrdMemoryPool::createPool();
|
2002-11-21 00:18:16 +01:00
|
|
|
temp = FB_NEW_RPT(*tdbb->tdbb_default, 0) jrd_tra;
|
2001-05-23 15:26:42 +02:00
|
|
|
temp->tra_pool = tdbb->tdbb_default;
|
|
|
|
transaction_options(tdbb, temp, reinterpret_cast < UCHAR * >(tpb),
|
|
|
|
tpb_length);
|
|
|
|
|
|
|
|
lock = TRA_transaction_lock(tdbb, reinterpret_cast < blk * >(temp));
|
|
|
|
|
|
|
|
/* Read header page and allocate transaction number. Since
|
|
|
|
the transaction inventory page was initialized to zero, it
|
|
|
|
transaction is automatically marked active. */
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER_V2
|
|
|
|
number = bump_transaction_id(tdbb, &window);
|
|
|
|
oldest = dbb->dbb_oldest_transaction;
|
|
|
|
active = MAX(dbb->dbb_oldest_active, dbb->dbb_oldest_transaction);
|
|
|
|
oldest_active = dbb->dbb_oldest_active;
|
|
|
|
oldest_snapshot = dbb->dbb_oldest_snapshot;
|
|
|
|
|
|
|
|
#else /* SUPERSERVER_V2 */
|
|
|
|
if (dbb->dbb_flags & DBB_read_only) {
|
|
|
|
number = ++dbb->dbb_next_transaction;
|
|
|
|
oldest = dbb->dbb_oldest_transaction;
|
|
|
|
active = MAX(dbb->dbb_oldest_active, dbb->dbb_oldest_transaction);
|
|
|
|
oldest_active = dbb->dbb_oldest_active;
|
|
|
|
oldest_snapshot = dbb->dbb_oldest_snapshot;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
header = bump_transaction_id(tdbb, &window);
|
|
|
|
number = header->hdr_next_transaction;
|
|
|
|
oldest = header->hdr_oldest_transaction;
|
|
|
|
active =
|
|
|
|
MAX(header->hdr_oldest_active, header->hdr_oldest_transaction);
|
|
|
|
oldest_active = header->hdr_oldest_active;
|
|
|
|
oldest_snapshot = header->hdr_oldest_snapshot;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* SUPERSERVER_V2 */
|
|
|
|
|
|
|
|
/* Allocate pool and transactions block. Since, by policy,
|
|
|
|
all transactions older than the oldest are either committed
|
|
|
|
or cleaned up, they can be all considered as committed. To
|
|
|
|
make everything simpler, round down the oldest to a multiple
|
|
|
|
of four, which ib_puts the transaction on a byte boundary. */
|
|
|
|
|
|
|
|
base = oldest & ~TRA_MASK;
|
|
|
|
|
|
|
|
if (temp->tra_flags & TRA_read_committed)
|
2002-11-21 00:18:16 +01:00
|
|
|
trans = FB_NEW_RPT(*tdbb->tdbb_default, 0) jrd_tra;
|
2001-05-23 15:26:42 +02:00
|
|
|
else {
|
2002-11-21 00:18:16 +01:00
|
|
|
trans = FB_NEW_RPT(*tdbb->tdbb_default, (number - base + TRA_MASK) / 4) jrd_tra;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
trans->tra_pool = temp->tra_pool;
|
|
|
|
trans->tra_relation_locks = temp->tra_relation_locks;
|
|
|
|
trans->tra_flags = temp->tra_flags;
|
|
|
|
trans->tra_number = number;
|
|
|
|
trans->tra_top = number;
|
|
|
|
trans->tra_oldest = oldest;
|
|
|
|
trans->tra_oldest_active = active;
|
2001-12-24 03:51:06 +01:00
|
|
|
delete temp;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
trans->tra_lock = lock;
|
|
|
|
lock->lck_key.lck_long = number;
|
|
|
|
|
|
|
|
/* Support refresh range notification in ExpressLink */
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
|
|
lock->lck_ast = downgrade_lock;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Put the TID of the oldest active transaction (from the header page)
|
|
|
|
in the new transaction's lock. */
|
|
|
|
|
|
|
|
lock->lck_data = active;
|
2002-03-31 01:40:08 +01:00
|
|
|
lock->lck_object = reinterpret_cast<blk*>(trans);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!LCK_lock_non_blocking(tdbb, lock, LCK_write, TRUE)) {
|
|
|
|
#ifndef SUPERSERVER_V2
|
|
|
|
if (!(dbb->dbb_flags & DBB_read_only))
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
#endif
|
2001-12-24 03:51:06 +01:00
|
|
|
delete trans;
|
2001-05-23 15:26:42 +02:00
|
|
|
ERR_post(gds_lock_conflict, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Link the transaction to the attachment block before releasing
|
|
|
|
header page for handling signals. */
|
|
|
|
|
|
|
|
TRA_link_transaction(tdbb, trans);
|
|
|
|
|
|
|
|
#ifndef SUPERSERVER_V2
|
|
|
|
if (!(dbb->dbb_flags & DBB_read_only))
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (dbb->dbb_flags & DBB_read_only) {
|
|
|
|
/* Set transaction flags to TRA_precommitted, TRA_readonly */
|
|
|
|
trans->tra_flags |= (TRA_readonly | TRA_precommitted);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Next, take a snapshot of all transactions between the oldest interesting
|
|
|
|
transaction and the current. Don't bother to get a snapshot for
|
|
|
|
read-committed transactions; they use the snapshot off the dbb block
|
|
|
|
since they need to know what is currently committed. */
|
|
|
|
|
|
|
|
if (trans->tra_flags & TRA_read_committed)
|
|
|
|
TPC_initialize_tpc(tdbb, number);
|
|
|
|
else
|
|
|
|
TRA_get_inventory(tdbb, trans->tra_transactions, base, number);
|
|
|
|
|
|
|
|
/* Next task is to find the oldest active transaction on the system. This
|
|
|
|
is needed for garbage collection. Things are made ever so slightly
|
|
|
|
more complicated by the fact that existing transaction may have oldest
|
|
|
|
actives older than they are. */
|
|
|
|
|
|
|
|
temp_lock.lck_dbb = dbb;
|
2002-03-31 01:40:08 +01:00
|
|
|
temp_lock.lck_object = reinterpret_cast<blk*>(trans);
|
2001-05-23 15:26:42 +02:00
|
|
|
temp_lock.lck_type = LCK_tra;
|
|
|
|
temp_lock.lck_owner_handle =
|
|
|
|
LCK_get_owner_handle(tdbb, temp_lock.lck_type);
|
|
|
|
temp_lock.lck_parent = dbb->dbb_lock;
|
|
|
|
temp_lock.lck_length = sizeof(SLONG);
|
|
|
|
|
|
|
|
trans->tra_oldest_active = number;
|
|
|
|
base = oldest & ~TRA_MASK;
|
|
|
|
oldest_active = number;
|
|
|
|
cleanup = (number % TRA_ACTIVE_CLEANUP) ? FALSE : TRUE;
|
|
|
|
|
|
|
|
for (; active < number; active++) {
|
|
|
|
if (trans->tra_flags & TRA_read_committed)
|
|
|
|
oldest_state = TPC_cache_state(tdbb, active);
|
|
|
|
else {
|
|
|
|
byte = TRANS_OFFSET(active - base);
|
|
|
|
shift = TRANS_SHIFT(active);
|
|
|
|
oldest_state =
|
|
|
|
(trans->tra_transactions[byte] >> shift) & TRA_MASK;
|
|
|
|
}
|
|
|
|
if (oldest_state == tra_active) {
|
|
|
|
temp_lock.lck_key.lck_long = active;
|
|
|
|
if (!(data = LCK_read_data(&temp_lock))) {
|
|
|
|
if (cleanup) {
|
|
|
|
if (TRA_wait(tdbb, trans, active, FALSE) == tra_committed)
|
|
|
|
cleanup = FALSE;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
data = active;
|
|
|
|
}
|
|
|
|
|
|
|
|
oldest_active = MIN(oldest_active, active);
|
|
|
|
|
|
|
|
/* Find the oldest record version that cannot be garbage collected yet
|
2001-07-10 19:35:13 +02:00
|
|
|
by taking the minimum of all all versions needed by all active
|
2001-05-23 15:26:42 +02:00
|
|
|
transactions. */
|
|
|
|
|
|
|
|
if (data < trans->tra_oldest_active)
|
|
|
|
trans->tra_oldest_active = data;
|
|
|
|
|
|
|
|
/* If the lock data for any active transaction matches a previously
|
|
|
|
computed value then there is no need to continue. There can't be
|
|
|
|
an older lock data in the remaining active transactions. */
|
|
|
|
|
|
|
|
if (trans->tra_oldest_active == oldest_snapshot)
|
|
|
|
break;
|
|
|
|
#ifndef VMS
|
|
|
|
/* Query the minimum lock data for all active transaction locks.
|
|
|
|
This will be the oldest active snapshot used for regulating
|
|
|
|
garbage collection. */
|
|
|
|
|
|
|
|
data = LCK_query_data(dbb->dbb_lock, LCK_tra, LCK_MIN);
|
|
|
|
if (data && data < trans->tra_oldest_active)
|
|
|
|
trans->tra_oldest_active = data;
|
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Put the TID of the oldest active transaction (just calculated)
|
|
|
|
in the new transaction's lock. */
|
|
|
|
|
|
|
|
if (lock->lck_data != oldest_active)
|
|
|
|
LCK_write_data(lock, oldest_active);
|
|
|
|
|
|
|
|
/* Scan commit retaining transactions which have started after us but which
|
|
|
|
want to preserve an oldest active from an already committed transaction.
|
|
|
|
If a previously computed oldest snapshot was matched then there's no
|
|
|
|
need to worry about commit retaining transactions. */
|
|
|
|
|
|
|
|
#ifdef VMS
|
|
|
|
if (trans->tra_oldest_active != oldest_snapshot)
|
|
|
|
compute_oldest_retaining(tdbb, trans, FALSE);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Finally, scan transactions looking for the oldest interesting transaction -- the oldest
|
|
|
|
non-commited transaction. This will not be updated immediately, but saved until the
|
|
|
|
next update access to the header page */
|
|
|
|
|
|
|
|
oldest_state = tra_committed;
|
|
|
|
|
|
|
|
for (oldest = trans->tra_oldest; oldest < number; oldest++) {
|
|
|
|
if (trans->tra_flags & TRA_read_committed)
|
|
|
|
oldest_state = TPC_cache_state(tdbb, oldest);
|
|
|
|
else {
|
|
|
|
byte = TRANS_OFFSET(oldest - base);
|
|
|
|
shift = TRANS_SHIFT(oldest);
|
|
|
|
oldest_state =
|
|
|
|
(trans->tra_transactions[byte] >> shift) & TRA_MASK;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (oldest_state != tra_committed && oldest_state != tra_precommitted)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef MULTI_THREAD
|
|
|
|
if (--oldest > dbb->dbb_oldest_transaction)
|
|
|
|
dbb->dbb_oldest_transaction = oldest;
|
|
|
|
|
|
|
|
if (oldest_active > dbb->dbb_oldest_active)
|
|
|
|
dbb->dbb_oldest_active = oldest_active;
|
|
|
|
#else
|
|
|
|
dbb->dbb_oldest_transaction = oldest - 1;
|
|
|
|
dbb->dbb_oldest_active = oldest_active;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (trans->tra_oldest_active > dbb->dbb_oldest_snapshot)
|
|
|
|
dbb->dbb_oldest_snapshot = trans->tra_oldest_active;
|
|
|
|
|
|
|
|
/* If the transaction block is getting out of hand, force a sweep */
|
|
|
|
|
|
|
|
if (dbb->dbb_sweep_interval &&
|
|
|
|
(trans->tra_oldest_active - trans->tra_oldest >
|
|
|
|
dbb->dbb_sweep_interval) && oldest_state != tra_limbo) {
|
|
|
|
#ifndef SWEEP_THREAD
|
|
|
|
|
|
|
|
/* force a sweep */
|
|
|
|
|
|
|
|
TRA_sweep(tdbb, trans);
|
|
|
|
|
|
|
|
#else
|
|
|
|
start_sweeper(tdbb, dbb);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check in with external file system */
|
|
|
|
|
|
|
|
EXT_trans_start(trans);
|
|
|
|
|
|
|
|
/* Start a 'transaction-level' savepoint, unless this is the
|
|
|
|
system transaction, or unless the transactions doesn't want
|
|
|
|
a savepoint to be started. This savepoint will be used to
|
|
|
|
undo the transaction if it rolls back. */
|
|
|
|
|
|
|
|
if ((trans != dbb->dbb_sys_trans) &&
|
|
|
|
!(trans->tra_flags & TRA_no_auto_undo)) {
|
2002-11-03 18:29:51 +01:00
|
|
|
VIO_start_save_point(tdbb, trans);
|
2001-05-23 15:26:42 +02:00
|
|
|
trans->tra_save_point->sav_flags |= SAV_trans_level;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if the user asked us to restart all requests in this attachment,
|
|
|
|
do so now using the new transaction */
|
|
|
|
|
|
|
|
if (trans->tra_flags & TRA_restart_requests)
|
|
|
|
restart_requests(tdbb, trans);
|
|
|
|
|
|
|
|
/* If the transaction is read-only and read committed, it can be
|
|
|
|
precommitted because it can't modify any records and doesn't
|
|
|
|
need a snapshot preserved. This is being enabled for internal
|
|
|
|
transactions used by the sweeper threads. This transaction type
|
|
|
|
can run forever without impacting garbage collection or causing
|
|
|
|
transaction bitmap growth. It can be turned on for external use
|
|
|
|
by removing the test for TDBB_sweeper. */
|
|
|
|
|
2002-07-01 17:07:18 +02:00
|
|
|
if (trans->tra_flags & TRA_readonly &&
|
2001-05-23 15:26:42 +02:00
|
|
|
trans->tra_flags & TRA_read_committed) {
|
|
|
|
TRA_set_state(tdbb, trans, trans->tra_number, tra_committed);
|
|
|
|
LCK_release(tdbb, trans->tra_lock);
|
2001-12-24 03:51:06 +01:00
|
|
|
delete trans->tra_lock;
|
2001-05-23 15:26:42 +02:00
|
|
|
trans->tra_lock = 0;
|
|
|
|
trans->tra_flags |= TRA_precommitted;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (trans->tra_flags & TRA_precommitted)
|
|
|
|
(void) TRA_precommited(tdbb, 0, trans->tra_number);
|
|
|
|
|
|
|
|
return trans;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int TRA_state(UCHAR * bit_vector, ULONG oldest, ULONG number)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ s t a t e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Get the state of a transaction from a cached
|
|
|
|
* bit vector.
|
|
|
|
* NOTE: This code is reproduced elsewhere in
|
|
|
|
* this module for speed. If changes are made
|
|
|
|
* to this code make them in the replicated code also.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
ULONG base, byte;
|
|
|
|
USHORT shift;
|
|
|
|
|
|
|
|
base = oldest & ~TRA_MASK;
|
|
|
|
byte = TRANS_OFFSET(number - base);
|
|
|
|
shift = TRANS_SHIFT(number);
|
|
|
|
|
|
|
|
return (bit_vector[byte] >> shift) & TRA_MASK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-17 01:13:59 +01:00
|
|
|
int TRA_sweep(TDBB tdbb, JRD_TRA trans)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ s w e e p
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Make a garbage collection pass thru database.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DBB dbb;
|
2002-11-17 01:13:59 +01:00
|
|
|
JRD_TRA transaction = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
HDR header;
|
|
|
|
WIN window;
|
|
|
|
USHORT shift;
|
|
|
|
ULONG byte, active, base;
|
|
|
|
SLONG transaction_oldest_active;
|
|
|
|
struct lck temp_lock;
|
|
|
|
JRNDH journal;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
CHECK_DBB(dbb);
|
|
|
|
|
|
|
|
/* No point trying to sweep a ReadOnly database */
|
|
|
|
if (dbb->dbb_flags & DBB_read_only)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (dbb->dbb_flags & DBB_sweep_in_progress)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
/* fill out a lock block, zeroing it out first */
|
|
|
|
|
|
|
|
temp_lock.lck_dbb = dbb;
|
2002-03-31 01:40:08 +01:00
|
|
|
temp_lock.lck_object = reinterpret_cast<blk*>(trans);
|
2001-05-23 15:26:42 +02:00
|
|
|
temp_lock.lck_type = LCK_sweep;
|
|
|
|
temp_lock.lck_owner_handle =
|
|
|
|
LCK_get_owner_handle(tdbb, temp_lock.lck_type);
|
|
|
|
temp_lock.lck_parent = dbb->dbb_lock;
|
|
|
|
temp_lock.lck_length = sizeof(SLONG);
|
|
|
|
|
|
|
|
if (!LCK_lock_non_blocking
|
|
|
|
(tdbb, &temp_lock, LCK_EX, (trans) ? FALSE : TRUE)) return TRUE;
|
|
|
|
|
|
|
|
dbb->dbb_flags |= DBB_sweep_in_progress;
|
|
|
|
|
|
|
|
/* The following line seems to fix a bug that appears on VMS and AIX
|
|
|
|
(and perhaps MPE XL). It probably has to do with the fact that
|
|
|
|
the error handler below used to contain the first reference to
|
|
|
|
variable transaction, which is actually initialized a few lines
|
|
|
|
below that. */
|
|
|
|
|
|
|
|
transaction = *(&transaction);
|
|
|
|
|
|
|
|
/* Clean up the temporary locks we've gotten in case anything goes wrong */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
try {
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Identify ourselves as a sweeper thread. This accomplishes
|
|
|
|
two goals: 1) Sweep transaction is started "precommitted"
|
|
|
|
and 2) Execution is throttled in JRD_reschedule() by
|
|
|
|
yielding the processor when our quantum expires. */
|
|
|
|
|
|
|
|
tdbb->tdbb_flags |= TDBB_sweeper;
|
|
|
|
|
|
|
|
/* Start a transaction, if necessary, to perform the sweep.
|
|
|
|
Save the transaction's oldest snapshot as it is refreshed
|
|
|
|
during the course of the database sweep. Since it is used
|
|
|
|
below to advance the OIT we must save it before it changes. */
|
|
|
|
|
|
|
|
if (!(transaction = trans))
|
|
|
|
transaction = TRA_start(tdbb,
|
|
|
|
sizeof(sweep_tpb),
|
|
|
|
reinterpret_cast < char *>(const_cast <
|
|
|
|
UCHAR *
|
|
|
|
>(sweep_tpb)));
|
|
|
|
|
|
|
|
transaction_oldest_active = transaction->tra_oldest_active;
|
|
|
|
|
|
|
|
#ifdef GARBAGE_THREAD
|
|
|
|
/* The garbage collector runs asynchronously with respect to
|
|
|
|
our database sweep. This isn't good enough since we must
|
|
|
|
be absolutely certain that all dead transactions have been
|
|
|
|
swept from the database before advancing the OIT. Turn off
|
|
|
|
the "notify garbage collector" flag for the attachment and
|
|
|
|
synchronously perform the garbage collection ourselves. */
|
|
|
|
|
|
|
|
transaction->tra_attachment->att_flags &= ~ATT_notify_gc;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (VIO_sweep(tdbb, transaction)) {
|
|
|
|
base = transaction->tra_oldest & ~TRA_MASK;
|
|
|
|
|
|
|
|
if (transaction->tra_flags & TRA_sweep_at_startup)
|
|
|
|
active = transaction->tra_oldest_active;
|
|
|
|
else {
|
|
|
|
for (active = transaction->tra_oldest;
|
|
|
|
active < transaction->tra_top; active++) {
|
|
|
|
if (transaction->tra_flags & TRA_read_committed) {
|
|
|
|
if (TPC_cache_state(tdbb, active) == tra_limbo)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
byte = TRANS_OFFSET(active - base);
|
|
|
|
shift = TRANS_SHIFT(active);
|
|
|
|
if (
|
|
|
|
((transaction->tra_transactions[byte] >> shift) &
|
|
|
|
TRA_MASK) == tra_limbo) break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Flush page buffers to insure that no dangling records from
|
|
|
|
dead transactions are left on-disk. This must be done before
|
|
|
|
the OIT is advanced and the header page is written to disk.
|
|
|
|
If the header page was written before flushing the page buffers
|
|
|
|
and there was a server crash, the dead records would appear
|
|
|
|
committed since their TID would now be less than the OIT recorded
|
|
|
|
in the database. */
|
|
|
|
|
|
|
|
CCH_flush(tdbb, (USHORT) FLUSH_SWEEP, 0);
|
|
|
|
|
|
|
|
window.win_page = HEADER_PAGE;
|
|
|
|
window.win_flags = 0;
|
|
|
|
header = (HDR) CCH_FETCH(tdbb, &window, LCK_write, pag_header);
|
|
|
|
|
|
|
|
if (header->hdr_oldest_transaction < --transaction_oldest_active) {
|
|
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
|
|
|
header->hdr_oldest_transaction =
|
|
|
|
MIN(active, transaction_oldest_active);
|
|
|
|
if (dbb->dbb_wal) {
|
|
|
|
journal.jrndh_type = JRNP_DB_HEADER;
|
|
|
|
journal.jrndh_nti = header->hdr_next_transaction;
|
|
|
|
journal.jrndh_oit = header->hdr_oldest_transaction;
|
|
|
|
journal.jrndh_oat = header->hdr_oldest_active;
|
|
|
|
CCH_journal_record(tdbb, &window,
|
|
|
|
reinterpret_cast < UCHAR * >(&journal),
|
|
|
|
JRNDH_SIZE, 0, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!trans)
|
|
|
|
TRA_commit(tdbb, transaction, FALSE);
|
|
|
|
|
|
|
|
LCK_release(tdbb, &temp_lock);
|
|
|
|
dbb->dbb_flags &= ~DBB_sweep_in_progress;
|
|
|
|
|
|
|
|
tdbb->tdbb_flags &= ~TDBB_sweeper;
|
2001-12-24 03:51:06 +01:00
|
|
|
} // try
|
2003-02-13 14:33:57 +01:00
|
|
|
catch (const std::exception&) {
|
2001-12-24 03:51:06 +01:00
|
|
|
try {
|
|
|
|
if (!trans)
|
|
|
|
{
|
|
|
|
TRA_commit(tdbb, transaction, FALSE);
|
|
|
|
}
|
|
|
|
}
|
2003-02-13 14:33:57 +01:00
|
|
|
catch (const std::exception&) {
|
2001-12-24 03:51:06 +01:00
|
|
|
LCK_release(tdbb, &temp_lock);
|
|
|
|
dbb->dbb_flags &= ~DBB_sweep_in_progress;
|
|
|
|
tdbb->tdbb_flags &= ~TDBB_sweeper;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
LCK TRA_transaction_lock(TDBB tdbb, BLK object)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ t r a n s a c t i o n _ l o c k
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Allocate a transaction lock block.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DBB dbb;
|
|
|
|
LCK lock;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
|
2002-09-25 19:12:16 +02:00
|
|
|
lock = FB_NEW_RPT(*tdbb->tdbb_default, sizeof(SLONG)) lck();
|
2001-05-23 15:26:42 +02:00
|
|
|
lock->lck_type = LCK_tra;
|
|
|
|
lock->lck_owner_handle = LCK_get_owner_handle(tdbb, lock->lck_type);
|
|
|
|
lock->lck_length = sizeof(SLONG);
|
|
|
|
|
|
|
|
lock->lck_dbb = dbb;
|
|
|
|
lock->lck_parent = dbb->dbb_lock;
|
|
|
|
lock->lck_object = object;
|
|
|
|
|
|
|
|
return lock;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-17 01:13:59 +01:00
|
|
|
int TRA_wait(TDBB tdbb, JRD_TRA trans, SLONG number, USHORT wait)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* T R A _ w a i t ( n o n - G A T E W A Y )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Wait for a given transaction to drop into a stable state (i.e. non-active)
|
|
|
|
* state. To do this, we first wait on the transaction number. When we
|
|
|
|
* are able to get the lock, the transaction is not longer bona fide
|
|
|
|
* active. Next, we determine the state of the transaction from the
|
|
|
|
* transaction inventory page. If either committed, dead, or limbo,
|
|
|
|
* we return the state. If the transaction is still marked active,
|
|
|
|
* however, declare the transaction dead, and mark the transaction
|
|
|
|
* inventory page accordingly.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DBB dbb;
|
|
|
|
USHORT shift, state, wait_flag;
|
|
|
|
ULONG byte;
|
|
|
|
struct lck temp_lock;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
CHECK_DBB(dbb);
|
|
|
|
|
|
|
|
/* Create, wait on, and release lock on target transaction. If
|
|
|
|
we can't get the lock due to deadlock */
|
|
|
|
|
|
|
|
if (wait) {
|
|
|
|
temp_lock.lck_dbb = dbb;
|
|
|
|
temp_lock.lck_type = LCK_tra;
|
|
|
|
temp_lock.lck_owner_handle =
|
|
|
|
LCK_get_owner_handle(tdbb, temp_lock.lck_type);
|
|
|
|
temp_lock.lck_parent = dbb->dbb_lock;
|
|
|
|
temp_lock.lck_length = sizeof(SLONG);
|
|
|
|
temp_lock.lck_key.lck_long = number;
|
|
|
|
temp_lock.lck_owner = (BLK) trans;
|
|
|
|
|
|
|
|
wait_flag = (trans->tra_flags & TRA_nowait) ? FALSE : TRUE;
|
|
|
|
if (!LCK_lock_non_blocking(tdbb, &temp_lock, LCK_read, wait_flag))
|
|
|
|
return tra_active;
|
|
|
|
|
|
|
|
LCK_release(tdbb, &temp_lock);
|
|
|
|
}
|
|
|
|
|
|
|
|
state = TRA_get_state(tdbb, number);
|
|
|
|
|
|
|
|
if (wait && state == tra_committed)
|
|
|
|
return state;
|
|
|
|
|
|
|
|
if (state == tra_precommitted)
|
|
|
|
return state;
|
|
|
|
|
|
|
|
/* If the recorded state of the transaction is active, we know better. If
|
|
|
|
it were active, he'd be alive now. Mark him dead. */
|
|
|
|
|
|
|
|
if (state == tra_active) {
|
|
|
|
state = tra_dead;
|
|
|
|
TRA_set_state(tdbb, 0, number, tra_dead);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (number > trans->tra_top)
|
|
|
|
return state;
|
|
|
|
|
|
|
|
/* If the transaction disppeared into limbo, died, for constructively
|
|
|
|
died, tweak the transaction state snapshot to reflect the new state.
|
|
|
|
This is guarenteed safe. */
|
|
|
|
|
|
|
|
byte = TRANS_OFFSET(number - (trans->tra_oldest & ~TRA_MASK));
|
|
|
|
shift = TRANS_SHIFT(number);
|
|
|
|
|
|
|
|
if (trans->tra_flags & TRA_read_committed)
|
|
|
|
TPC_set_state(tdbb, number, state);
|
|
|
|
else {
|
|
|
|
trans->tra_transactions[byte] &= ~(TRA_MASK << shift);
|
|
|
|
trans->tra_transactions[byte] |= state << shift;
|
|
|
|
}
|
|
|
|
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER_V2
|
|
|
|
static SLONG bump_transaction_id(TDBB tdbb, WIN * window)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* b u m p _ t r a n s a c t i o n _ i d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Fetch header and bump next transaction id. If necessary,
|
|
|
|
* extend TIP.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DBB dbb;
|
|
|
|
HDR header;
|
|
|
|
ULONG fake_tid, number;
|
|
|
|
JRNDH journal;
|
|
|
|
SLONG sequence;
|
|
|
|
SSHORT new_tip;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
CHECK_DBB(dbb);
|
|
|
|
|
|
|
|
number = ++dbb->dbb_next_transaction;
|
|
|
|
|
|
|
|
/* No need to write TID onto the TIP page, for a RO DB */
|
|
|
|
if (dbb->dbb_flags & DBB_read_only)
|
|
|
|
return number;
|
|
|
|
|
|
|
|
/* If this is the first transaction on a TIP, allocate the TIP now. */
|
|
|
|
|
|
|
|
if (new_tip = ((number == 1)
|
|
|
|
|| ((number % dbb->dbb_pcontrol->pgc_tpt) == 0)))
|
|
|
|
TRA_extend_tip(tdbb,
|
|
|
|
(ULONG) (number / dbb->dbb_pcontrol->pgc_tpt), window);
|
|
|
|
|
|
|
|
/* Transaction start will be logged every MOD_START_TRAN or when a
|
|
|
|
* new tip page is allocated.
|
|
|
|
* Make sure that the fake_tid does not roll over to the next tip page.
|
|
|
|
* If that is the case, use last tid in this page.
|
|
|
|
*/
|
|
|
|
if (dbb->dbb_wal) {
|
|
|
|
if ((number % MOD_START_TRAN == 0) || (new_tip)) {
|
|
|
|
fake_tid = number + MOD_START_TRAN;
|
|
|
|
sequence = number / dbb->dbb_pcontrol->pgc_tpt;
|
|
|
|
if (sequence != ((fake_tid + 1) / dbb->dbb_pcontrol->pgc_tpt))
|
|
|
|
fake_tid = ((sequence + 1) * dbb->dbb_pcontrol->pgc_tpt) - 1;
|
|
|
|
header = (HDR) CCH_FETCH(tdbb, window, LCK_write, pag_header);
|
|
|
|
CCH_MARK_MUST_WRITE(tdbb, window);
|
|
|
|
header->hdr_bumped_transaction = fake_tid;
|
|
|
|
|
|
|
|
journal.jrndh_type = JRNP_DB_HEADER;
|
|
|
|
journal.jrndh_nti = header->hdr_bumped_transaction;
|
|
|
|
journal.jrndh_oit = dbb->dbb_oldest_transaction;
|
|
|
|
journal.jrndh_oat = dbb->dbb_oldest_active;
|
2002-04-29 17:05:11 +02:00
|
|
|
CCH_journal_record(tdbb, window,
|
|
|
|
reinterpret_cast<unsigned char*>(&journal), JRNDH_SIZE, 0, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
CCH_RELEASE(tdbb, window);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return number;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
|
|
|
|
|
|
|
|
static HDR bump_transaction_id(TDBB tdbb, WIN * window)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* b u m p _ t r a n s a c t i o n _ i d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Fetch header and bump next transaction id. If necessary,
|
|
|
|
* extend TIP.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DBB dbb;
|
|
|
|
HDR header;
|
|
|
|
ULONG fake_tid, number;
|
|
|
|
JRNDH journal;
|
|
|
|
SLONG sequence;
|
|
|
|
SSHORT new_tip;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
CHECK_DBB(dbb);
|
|
|
|
|
|
|
|
window->win_page = HEADER_PAGE;
|
|
|
|
header = (HDR) CCH_FETCH(tdbb, window, LCK_write, pag_header);
|
|
|
|
|
|
|
|
/* Before incrementing the next transaction Id, make sure the current one is valid */
|
|
|
|
if (header->hdr_next_transaction) {
|
|
|
|
if (header->hdr_oldest_active > header->hdr_next_transaction)
|
|
|
|
BUGCHECK(266); /*next transaction older than oldest active */
|
|
|
|
|
|
|
|
if (header->hdr_oldest_transaction > header->hdr_next_transaction)
|
|
|
|
BUGCHECK(267); /* next transaction older than oldest transaction */
|
|
|
|
}
|
|
|
|
|
|
|
|
number = header->hdr_next_transaction + 1;
|
|
|
|
|
|
|
|
/* If this is the first transaction on a TIP, allocate the TIP now. */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if ( (new_tip = ((number == 1))
|
2001-05-23 15:26:42 +02:00
|
|
|
|| ((number % dbb->dbb_pcontrol->pgc_tpt) == 0)))
|
|
|
|
TRA_extend_tip(tdbb,
|
|
|
|
(ULONG) (number / dbb->dbb_pcontrol->pgc_tpt), window);
|
|
|
|
|
|
|
|
/* Extend, if necessary, has apparently succeeded. Next, update header
|
|
|
|
page */
|
|
|
|
|
|
|
|
CCH_MARK_MUST_WRITE(tdbb, window);
|
|
|
|
header->hdr_next_transaction = number;
|
|
|
|
|
|
|
|
if (dbb->dbb_oldest_active > header->hdr_oldest_active)
|
|
|
|
header->hdr_oldest_active = dbb->dbb_oldest_active;
|
|
|
|
|
|
|
|
if (dbb->dbb_oldest_transaction > header->hdr_oldest_transaction)
|
|
|
|
header->hdr_oldest_transaction = dbb->dbb_oldest_transaction;
|
|
|
|
|
|
|
|
if (dbb->dbb_oldest_snapshot > header->hdr_oldest_snapshot)
|
|
|
|
header->hdr_oldest_snapshot = dbb->dbb_oldest_snapshot;
|
|
|
|
|
|
|
|
/* Transaction start will be logged every MOD_START_TRAN or when a
|
|
|
|
* new tip page is allocated.
|
|
|
|
* Make sure that the fake_tid does not roll over to the next tip page.
|
|
|
|
* If that is the case, use last tid in this page.
|
|
|
|
*/
|
|
|
|
if (dbb->dbb_wal) {
|
|
|
|
if ((number % MOD_START_TRAN == 0) || (new_tip)) {
|
|
|
|
fake_tid = number + MOD_START_TRAN;
|
|
|
|
sequence = number / dbb->dbb_pcontrol->pgc_tpt;
|
|
|
|
if (sequence != ((fake_tid + 1) / dbb->dbb_pcontrol->pgc_tpt))
|
|
|
|
fake_tid = ((sequence + 1) * dbb->dbb_pcontrol->pgc_tpt) - 1;
|
|
|
|
header->hdr_bumped_transaction = fake_tid;
|
|
|
|
|
|
|
|
journal.jrndh_type = JRNP_DB_HEADER;
|
|
|
|
journal.jrndh_nti = header->hdr_bumped_transaction;
|
|
|
|
journal.jrndh_oit = header->hdr_oldest_transaction;
|
|
|
|
journal.jrndh_oat = header->hdr_oldest_active;
|
|
|
|
CCH_journal_record(tdbb, window,
|
|
|
|
reinterpret_cast < UCHAR * >(&journal),
|
|
|
|
JRNDH_SIZE, 0, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return header;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef VMS
|
|
|
|
static void compute_oldest_retaining(
|
|
|
|
TDBB tdbb,
|
2002-11-17 01:13:59 +01:00
|
|
|
JRD_TRA transaction, BOOLEAN write_flag)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c o m p u t e _ o l d e s t _ r e t a i n i n g
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Read the oldest active for all transactions
|
|
|
|
* younger than us up to the youngest retaining
|
|
|
|
* transaction. If an "older" oldest active is
|
|
|
|
* found, by all means use it. Write flag is TRUE
|
|
|
|
* to write retaining lock and FALSE to read it.
|
|
|
|
* The retaining lock holds the youngest commit
|
|
|
|
* retaining transaction.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DBB dbb;
|
|
|
|
LCK lock;
|
|
|
|
SLONG data, number, youngest_retaining;
|
|
|
|
struct lck temp_lock;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
CHECK_DBB(dbb);
|
|
|
|
|
|
|
|
/* Get a commit retaining lock, if not present. */
|
|
|
|
|
|
|
|
if (!(lock = dbb->dbb_retaining_lock)) {
|
2002-09-25 19:12:16 +02:00
|
|
|
lock = FB_NEW_RPT(*dbb->dbb_permanent, sizeof(SLONG)) lck();
|
2001-05-23 15:26:42 +02:00
|
|
|
lock->lck_dbb = dbb;
|
|
|
|
lock->lck_type = LCK_retaining;
|
|
|
|
lock->lck_owner_handle = LCK_get_owner_handle(tdbb, lock->lck_type);
|
|
|
|
lock->lck_parent = dbb->dbb_lock;
|
|
|
|
lock->lck_length = sizeof(SLONG);
|
2002-03-31 01:40:08 +01:00
|
|
|
lock->lck_object = reinterpret_cast<blk*>(dbb);
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef VMS
|
|
|
|
if (LCK_lock(tdbb, lock, LCK_EX, FALSE)) {
|
|
|
|
number = 0;
|
|
|
|
vms_convert(lock, &number, LCK_SR, TRUE);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
LCK_lock(tdbb, lock, LCK_SR, TRUE);
|
|
|
|
#else
|
|
|
|
LCK_lock_non_blocking(tdbb, lock, LCK_SR, TRUE);
|
|
|
|
#endif
|
|
|
|
dbb->dbb_retaining_lock = lock;
|
|
|
|
}
|
|
|
|
|
|
|
|
number = transaction->tra_number;
|
|
|
|
|
|
|
|
/* Writers must synchronize their lock update so that
|
|
|
|
an older retaining is not written over a younger retaining.
|
|
|
|
In any case, lock types have been selected so that
|
|
|
|
readers and writers don't interfere. */
|
|
|
|
|
|
|
|
if (write_flag) {
|
|
|
|
#ifdef VMS
|
|
|
|
vms_convert(lock, &youngest_retaining, LCK_PW, TRUE);
|
|
|
|
if (number > youngest_retaining)
|
|
|
|
vms_convert(lock, &number, LCK_SR, TRUE);
|
|
|
|
else
|
|
|
|
vms_convert(lock, 0, LCK_SR, TRUE);
|
|
|
|
#else
|
|
|
|
LCK_convert(tdbb, lock, LCK_PW, TRUE);
|
|
|
|
youngest_retaining = LOCK_read_data(lock->lck_id);
|
|
|
|
if (number > youngest_retaining)
|
|
|
|
LCK_write_data(lock, number);
|
|
|
|
LCK_convert(tdbb, lock, LCK_SR, TRUE);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
#ifdef VMS
|
|
|
|
vms_convert(lock, &youngest_retaining, LCK_SR, TRUE);
|
|
|
|
#else
|
|
|
|
youngest_retaining = LOCK_read_data(lock->lck_id);
|
|
|
|
#endif
|
|
|
|
if (number > youngest_retaining)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* fill out a lock block, zeroing it out first */
|
|
|
|
|
|
|
|
temp_lock.lck_dbb = dbb;
|
|
|
|
temp_lock.lck_type = LCK_tra;
|
|
|
|
temp_lock.lck_owner_handle =
|
|
|
|
LCK_get_owner_handle(tdbb, temp_lock.lck_type);
|
|
|
|
temp_lock.lck_parent = dbb->dbb_lock;
|
|
|
|
temp_lock.lck_length = sizeof(SLONG);
|
2002-03-31 01:40:08 +01:00
|
|
|
temp_lock.lck_object = reinterpret_cast<blk*>(transaction);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
while (number < youngest_retaining) {
|
|
|
|
temp_lock.lck_key.lck_long = ++number;
|
|
|
|
if (data = LCK_read_data(&temp_lock))
|
|
|
|
if (data < transaction->tra_oldest_active)
|
|
|
|
transaction->tra_oldest_active = data;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2002-11-17 01:13:59 +01:00
|
|
|
static void downgrade_lock(JRD_TRA transaction)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* d o w n g r a d e _ l o c k
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Someone is trying to establish an interest
|
|
|
|
* lock in this transaction. Downgrade to a
|
|
|
|
* shared write, to allow transactions to wait
|
2001-07-10 19:35:13 +02:00
|
|
|
* on this transaction or, alternatively, be
|
2001-05-23 15:26:42 +02:00
|
|
|
* notified if and when the transaction commits.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
LCK lock;
|
|
|
|
struct tdbb thd_context, *tdbb;
|
|
|
|
|
|
|
|
ISC_ast_enter();
|
|
|
|
|
|
|
|
/* Since this routine will be called asynchronously, we must establish
|
|
|
|
a thread context. */
|
|
|
|
|
|
|
|
SET_THREAD_DATA;
|
|
|
|
|
|
|
|
/* Ignore the request if the transaction or lock block does not appear
|
|
|
|
to be valid or if the lock is not a write lock. */
|
|
|
|
|
|
|
|
if (transaction->tra_use_count);
|
|
|
|
else {
|
|
|
|
tdbb->tdbb_database = transaction->tra_attachment->att_database;
|
|
|
|
tdbb->tdbb_attachment = transaction->tra_attachment;
|
|
|
|
tdbb->tdbb_quantum = QUANTUM;
|
|
|
|
tdbb->tdbb_request = NULL;
|
|
|
|
tdbb->tdbb_transaction = transaction;
|
|
|
|
|
|
|
|
++transaction->tra_use_count;
|
|
|
|
if ((lock = transaction->tra_lock) && lock->lck_logical == LCK_write) {
|
|
|
|
lock->lck_ast = NULL;
|
|
|
|
LCK_convert(tdbb, lock, LCK_SW, TRUE);
|
|
|
|
}
|
|
|
|
--transaction->tra_use_count;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Restore the prior thread context */
|
|
|
|
|
|
|
|
RESTORE_THREAD_DATA;
|
|
|
|
|
|
|
|
ISC_ast_exit();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-17 01:13:59 +01:00
|
|
|
static void expand_view_lock(JRD_TRA transaction, JRD_REL relation, SCHAR lock_type)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* e x p a n d _ v i e w _ l o c k
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* A view in a RESERVING will lead to all tables in the
|
|
|
|
* view being locked.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
TDBB tdbb = GET_THREAD_DATA;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
/* set up the lock on the relation/view */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
LCK lock = RLCK_transaction_relation_lock(transaction, relation);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
lock->lck_logical = lock_type;
|
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
VCX ctx = relation->rel_view_contexts;
|
|
|
|
if (!ctx) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return;
|
2002-03-31 01:40:08 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
for (; ctx; ctx = ctx->vcx_next)
|
|
|
|
{
|
2002-11-17 01:13:59 +01:00
|
|
|
JRD_REL rel = MET_lookup_relation(tdbb,
|
2002-03-31 01:40:08 +01:00
|
|
|
reinterpret_cast<const char*>(ctx->vcx_relation_name->str_data));
|
|
|
|
if (!rel)
|
|
|
|
{
|
|
|
|
ERR_post(gds_bad_tpb_content, /* should be a BUGCHECK */
|
|
|
|
gds_arg_gds,
|
|
|
|
gds_relnotdef,
|
|
|
|
gds_arg_string,
|
|
|
|
ERR_cstring(reinterpret_cast<char*>(ctx->vcx_relation_name->str_data)),
|
|
|
|
0);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* force a scan to read view information */
|
|
|
|
MET_scan_relation(tdbb, rel);
|
|
|
|
|
|
|
|
expand_view_lock(transaction, rel, lock_type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static TIP fetch_inventory_page(
|
|
|
|
TDBB tdbb,
|
|
|
|
WIN * window,
|
|
|
|
SLONG sequence, USHORT lock_level)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* f e t c h _ i n v e n t o r y _ p a g e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Fetch a transaction inventory page.
|
|
|
|
* Use the opportunity to cache the info
|
|
|
|
* in the TIP cache.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
TIP tip;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
window->win_page = inventory_page(tdbb, (int) sequence);
|
|
|
|
tip = (TIP) CCH_FETCH(tdbb, window, lock_level, pag_transactions);
|
|
|
|
|
|
|
|
TPC_update_cache(tdbb, tip, sequence);
|
|
|
|
|
|
|
|
return tip;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-07-01 17:07:18 +02:00
|
|
|
static SLONG inventory_page(TDBB tdbb, SLONG sequence)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n v e n t o r y _ p a g e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Get the physical page number of the n-th transaction inventory
|
|
|
|
* page. If not found, try to reconstruct using sibling pointer
|
|
|
|
* from last known TIP page.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DBB dbb;
|
|
|
|
TIP tip;
|
|
|
|
VCL vector;
|
|
|
|
WIN window;
|
|
|
|
SLONG next;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
CHECK_DBB(dbb);
|
|
|
|
|
|
|
|
window.win_flags = 0;
|
2001-12-24 03:51:06 +01:00
|
|
|
while (!(vector = dbb->dbb_t_pages) || sequence >= vector->count()) {
|
2001-05-23 15:26:42 +02:00
|
|
|
DPM_scan_pages(tdbb);
|
2001-12-24 03:51:06 +01:00
|
|
|
if ((vector = dbb->dbb_t_pages) && sequence < vector->count())
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2001-12-24 03:51:06 +01:00
|
|
|
if (!vector)
|
2001-05-23 15:26:42 +02:00
|
|
|
BUGCHECK(165); /* msg 165 cannot find tip page */
|
2001-12-24 03:51:06 +01:00
|
|
|
window.win_page = (*vector)[vector->count() - 1];
|
2001-05-23 15:26:42 +02:00
|
|
|
tip = (TIP) CCH_FETCH(tdbb, &window, LCK_read, pag_transactions);
|
|
|
|
next = tip->tip_next;
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
if (!(window.win_page = next))
|
|
|
|
BUGCHECK(165); /* msg 165 cannot find tip page */
|
|
|
|
tip = (TIP) CCH_FETCH(tdbb, &window, LCK_read, pag_transactions); /* Type check it */
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
2001-12-24 03:51:06 +01:00
|
|
|
DPM_pages(tdbb, 0, pag_transactions, vector->count(),
|
2001-05-23 15:26:42 +02:00
|
|
|
window.win_page);
|
|
|
|
}
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
return (*vector)[sequence];
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static SSHORT limbo_transaction(TDBB tdbb, SLONG id)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* l i m b o _ t r a n s a c t i o n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
* limbo_state is called when reconnecting
|
|
|
|
* to an existing transaction to assure that
|
|
|
|
* the transaction is actually in limbo.
|
|
|
|
* It returns the transaction state.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DBB dbb;
|
|
|
|
WIN window;
|
|
|
|
TIP tip;
|
|
|
|
UCHAR *byte;
|
2002-07-01 17:07:18 +02:00
|
|
|
SSHORT shift, state;
|
|
|
|
SLONG page, number, trans_per_tip, trans_offset;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
CHECK_DBB(dbb);
|
|
|
|
|
|
|
|
trans_per_tip = dbb->dbb_pcontrol->pgc_tpt;
|
|
|
|
|
|
|
|
page = id / trans_per_tip;
|
|
|
|
number = id % trans_per_tip;
|
|
|
|
|
|
|
|
window.win_flags = 0;
|
|
|
|
tip = fetch_inventory_page(tdbb, &window, page, LCK_write);
|
|
|
|
|
|
|
|
trans_offset = TRANS_OFFSET(number);
|
|
|
|
byte = tip->tip_transactions + trans_offset;
|
|
|
|
shift = TRANS_SHIFT(number);
|
|
|
|
state = (*byte >> shift) & TRA_MASK;
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-17 01:13:59 +01:00
|
|
|
static void restart_requests(TDBB tdbb, JRD_TRA trans)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e s t a r t _ r e q u e s t s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2001-07-10 19:35:13 +02:00
|
|
|
* Restart all requests in the current
|
|
|
|
* attachment to utilize the passed
|
2001-05-23 15:26:42 +02:00
|
|
|
* transaction.
|
|
|
|
*
|
|
|
|
**************************************/
|
2002-11-17 01:13:59 +01:00
|
|
|
JRD_REQ request, clone;
|
2001-05-23 15:26:42 +02:00
|
|
|
VEC vector;
|
|
|
|
USHORT level;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
for (request = trans->tra_attachment->att_requests; request;
|
|
|
|
request = request->req_request) {
|
|
|
|
if (request->req_transaction) {
|
|
|
|
EXE_unwind(tdbb, request);
|
|
|
|
EXE_start(tdbb, request, trans);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* now take care of any other request levels;
|
|
|
|
start at level 1 since level 0 was just handled */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if ( (vector = request->req_sub_requests) )
|
|
|
|
for (level = 1; level < vector->count(); level++)
|
2002-11-17 01:13:59 +01:00
|
|
|
if ((clone = (JRD_REQ) (*vector)[level]) &&
|
2001-05-23 15:26:42 +02:00
|
|
|
clone->req_transaction) {
|
|
|
|
EXE_unwind(tdbb, clone);
|
|
|
|
EXE_start(tdbb, clone, trans);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2002-11-17 01:13:59 +01:00
|
|
|
static void retain_context(TDBB tdbb, JRD_TRA transaction, USHORT commit)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e t a i n _ c o n t e x t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* If 'commit' flag is true, commit the transaction,
|
|
|
|
* else rollback the transaction.
|
|
|
|
*
|
|
|
|
* Commit/rollback a transaction while preserving the
|
|
|
|
* context, in particular, its snapshot. The
|
|
|
|
* trick is to insure that the transaction's
|
|
|
|
* oldest active is seen by other transactions
|
|
|
|
* simultaneously starting up.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
DBB dbb;
|
|
|
|
HDR header;
|
|
|
|
SLONG new_number, old_number;
|
|
|
|
WIN window;
|
|
|
|
LCK new_lock, old_lock;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
dbb = tdbb->tdbb_database;
|
|
|
|
CHECK_DBB(dbb);
|
|
|
|
|
|
|
|
/* The new transaction needs to remember the 'commit-retained' transaction
|
|
|
|
because it must see the operations of the 'commit-retained' transaction and
|
|
|
|
its snapshot doesn't contain these operations. */
|
|
|
|
|
|
|
|
if (commit)
|
|
|
|
SBM_set(tdbb, &transaction->tra_commit_sub_trans,
|
|
|
|
transaction->tra_number);
|
|
|
|
|
|
|
|
/* Create a new transaction lock, inheriting oldest active
|
|
|
|
from transaction being committed. */
|
|
|
|
|
|
|
|
window.win_flags = 0;
|
|
|
|
#ifdef SUPERSERVER_V2
|
|
|
|
new_number = bump_transaction_id(tdbb, &window);
|
|
|
|
#else
|
|
|
|
if (dbb->dbb_flags & DBB_read_only)
|
|
|
|
new_number = ++dbb->dbb_next_transaction;
|
|
|
|
else {
|
|
|
|
header = bump_transaction_id(tdbb, &window);
|
|
|
|
new_number = header->hdr_next_transaction;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if ( (old_lock = transaction->tra_lock) ) {
|
2001-05-23 15:26:42 +02:00
|
|
|
new_lock =
|
|
|
|
TRA_transaction_lock(tdbb,
|
|
|
|
reinterpret_cast < blk * >(transaction));
|
|
|
|
new_lock->lck_key.lck_long = new_number;
|
|
|
|
new_lock->lck_data = transaction->tra_lock->lck_data;
|
|
|
|
|
|
|
|
/* Support refresh range notification in ExpressLink */
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
|
|
new_lock->lck_ast = downgrade_lock;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (!LCK_lock_non_blocking(tdbb, new_lock, LCK_write, TRUE)) {
|
|
|
|
#ifndef SUPERSERVER_V2
|
|
|
|
if (!(dbb->dbb_flags & DBB_read_only))
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
#endif
|
|
|
|
ERR_post(gds_lock_conflict, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef SUPERSERVER_V2
|
|
|
|
if (!(dbb->dbb_flags & DBB_read_only))
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Update database notion of the youngest commit retaining
|
|
|
|
transaction before committing the first transaction. This
|
|
|
|
secures the original snapshot by insuring the oldest active
|
|
|
|
is seen by other transactions. */
|
|
|
|
|
|
|
|
old_number = transaction->tra_number;
|
|
|
|
#ifdef VMS
|
|
|
|
transaction->tra_number = new_number;
|
|
|
|
compute_oldest_retaining(tdbb, transaction, TRUE);
|
|
|
|
transaction->tra_number = old_number;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (!(dbb->dbb_flags & DBB_read_only)) {
|
|
|
|
/* Set the state on the inventory page */
|
|
|
|
if (commit)
|
|
|
|
TRA_set_state(tdbb, transaction, old_number, tra_committed);
|
|
|
|
else
|
|
|
|
TRA_set_state(tdbb, transaction, old_number, tra_dead);
|
|
|
|
}
|
|
|
|
transaction->tra_number = new_number;
|
|
|
|
|
|
|
|
/* Release transaction lock since it isn't needed
|
|
|
|
anymore and the new one is already in place. */
|
|
|
|
|
|
|
|
if (old_lock) {
|
|
|
|
++transaction->tra_use_count;
|
|
|
|
LCK_release(tdbb, old_lock);
|
|
|
|
transaction->tra_lock = new_lock;
|
|
|
|
--transaction->tra_use_count;
|
2001-12-24 03:51:06 +01:00
|
|
|
delete old_lock;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Perform any post commit work OR delete entries from deferred list */
|
|
|
|
|
|
|
|
if (commit)
|
|
|
|
DFW_perform_post_commit_work(transaction);
|
|
|
|
else
|
|
|
|
DFW_delete_deferred(transaction, -1);
|
|
|
|
|
|
|
|
transaction->tra_flags &= ~(TRA_write | TRA_prepared);
|
|
|
|
|
|
|
|
/* We have to mimic a TRA_commit and a TRA_start while reusing the
|
|
|
|
'transaction' control block: get rid of the transaction-level
|
|
|
|
savepoint and possibly start a new transaction-level savepoint. */
|
|
|
|
|
2002-11-03 18:29:51 +01:00
|
|
|
// Get rid of all user savepoints
|
|
|
|
// Why we can do this in reverse order described in commit method
|
|
|
|
while (transaction->tra_save_point && transaction->tra_save_point->sav_flags & SAV_user) {
|
|
|
|
SAV next=transaction->tra_save_point->sav_next;
|
|
|
|
transaction->tra_save_point->sav_next = NULL;
|
2002-10-29 21:20:44 +01:00
|
|
|
VIO_verb_cleanup(tdbb, transaction);
|
2002-11-03 18:29:51 +01:00
|
|
|
transaction->tra_save_point = next;
|
|
|
|
}
|
2002-10-29 21:20:44 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
if (transaction->tra_save_point) {
|
|
|
|
if (!(transaction->tra_save_point->sav_flags & SAV_trans_level))
|
|
|
|
BUGCHECK(287); /* Too many savepoints */
|
2002-11-03 18:29:51 +01:00
|
|
|
VIO_verb_cleanup(tdbb, transaction); /* get rid of transaction savepoint */
|
|
|
|
VIO_start_save_point(tdbb, transaction); /* start new savepoint */
|
2001-05-23 15:26:42 +02:00
|
|
|
transaction->tra_save_point->sav_flags |= SAV_trans_level;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (transaction->tra_flags & TRA_precommitted) {
|
|
|
|
if (!(dbb->dbb_flags & DBB_read_only))
|
|
|
|
{
|
|
|
|
transaction->tra_flags &= ~TRA_precommitted;
|
|
|
|
TRA_set_state(tdbb, transaction, new_number, tra_committed);
|
|
|
|
transaction->tra_flags |= TRA_precommitted;
|
|
|
|
}
|
|
|
|
|
|
|
|
(void) TRA_precommited(tdbb, old_number, new_number);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SWEEP_THREAD
|
|
|
|
static BOOLEAN start_sweeper(TDBB tdbb, DBB dbb)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s t a r t _ s w e e p e r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Start a thread to sweep the database.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2002-09-08 18:33:46 +02:00
|
|
|
if ((dbb->dbb_flags & DBB_sweep_in_progress)
|
|
|
|
|| (dbb->dbb_ast_flags & DBB_shutdown))
|
2001-05-24 16:54:26 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
return FALSE;
|
2001-05-24 16:54:26 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
// fill out the lock block
|
|
|
|
lck temp_lock;
|
|
|
|
temp_lock.lck_dbb = dbb;
|
|
|
|
temp_lock.lck_type = LCK_sweep;
|
|
|
|
temp_lock.lck_owner_handle = LCK_get_owner_handle(tdbb, temp_lock.lck_type);
|
|
|
|
temp_lock.lck_parent = dbb->dbb_lock;
|
|
|
|
temp_lock.lck_length = sizeof(SLONG);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!LCK_lock(tdbb, &temp_lock, LCK_EX, FALSE))
|
2001-05-24 16:54:26 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
return FALSE;
|
2001-05-24 16:54:26 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
LCK_release(tdbb, &temp_lock);
|
|
|
|
|
2001-05-24 16:54:26 +02:00
|
|
|
/* allocate space for the string and a null at the end */
|
|
|
|
const char* pszFilename =
|
2002-08-21 15:34:53 +02:00
|
|
|
reinterpret_cast<const char*>(tdbb->tdbb_attachment->att_filename->str_data);
|
2001-05-24 16:54:26 +02:00
|
|
|
|
|
|
|
char* database = (char*)gds__alloc(strlen(pszFilename) + 1);
|
|
|
|
|
|
|
|
if (!database)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
ERR_log(0, 0, "cannot start sweep thread, Out of Memory");
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2001-05-24 16:54:26 +02:00
|
|
|
strcpy(database, pszFilename);
|
|
|
|
if (gds__thread_start(reinterpret_cast<FPTR_INT_VOID_PTR>(sweep_database),
|
|
|
|
database,
|
|
|
|
THREAD_low,
|
|
|
|
0,
|
|
|
|
0))
|
|
|
|
{
|
|
|
|
gds__free(database);
|
|
|
|
ERR_log(0, 0, "cannot start sweep thread");
|
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
static void THREAD_ROUTINE sweep_database(UCHAR* database)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s w e e p _ d a t a b a s e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Sweep database.
|
|
|
|
*
|
|
|
|
**************************************/
|
2001-05-24 16:54:26 +02:00
|
|
|
STATUS status_vector[ISC_STATUS_LENGTH];
|
|
|
|
char* q;
|
2001-05-23 15:26:42 +02:00
|
|
|
UCHAR sweep_dpb[100];
|
|
|
|
|
2001-05-24 16:54:26 +02:00
|
|
|
isc_db_handle db_handle = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-05-24 16:54:26 +02:00
|
|
|
UCHAR* dpb = sweep_dpb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
*dpb++ = gds_dpb_version1;
|
|
|
|
*dpb++ = gds_dpb_user_name;
|
|
|
|
q = "sweeper";
|
|
|
|
*dpb++ = strlen (q);
|
|
|
|
while (*q)
|
|
|
|
{
|
|
|
|
*dpb++ = *q++;
|
|
|
|
}
|
|
|
|
|
|
|
|
*dpb++ = gds_dpb_password;
|
|
|
|
q = "none";
|
|
|
|
*dpb++ = strlen(q);
|
|
|
|
while (*q)
|
|
|
|
{
|
|
|
|
*dpb++ = *q++;
|
|
|
|
}
|
|
|
|
|
|
|
|
*dpb++ = gds_dpb_sweep;
|
|
|
|
*dpb++ = 1;
|
|
|
|
*dpb++ = gds_dpb_records;
|
2001-07-10 19:35:13 +02:00
|
|
|
|
2001-05-24 16:54:26 +02:00
|
|
|
const SSHORT dpb_length = dpb - sweep_dpb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Register as internal database handle */
|
2001-05-24 16:54:26 +02:00
|
|
|
IHNDL ihandle;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
THREAD_ENTER;
|
|
|
|
for (ihandle = internal_db_handles; ihandle; ihandle = ihandle->ihndl_next)
|
|
|
|
{
|
|
|
|
if (ihandle->ihndl_object == NULL)
|
|
|
|
{
|
|
|
|
ihandle->ihndl_object = &db_handle;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!ihandle)
|
|
|
|
{
|
|
|
|
ihandle = (IHNDL) gds__alloc ((SLONG) sizeof (struct ihndl));
|
|
|
|
ihandle->ihndl_object = &db_handle;
|
|
|
|
ihandle->ihndl_next = internal_db_handles;
|
|
|
|
internal_db_handles = ihandle;
|
|
|
|
}
|
|
|
|
THREAD_EXIT;
|
|
|
|
|
|
|
|
gds__attach_database(status_vector,
|
|
|
|
0,
|
|
|
|
reinterpret_cast<char*>(GDS_VAL(database)),
|
2002-12-16 17:25:10 +01:00
|
|
|
GDS_REF(db_handle),
|
2001-05-23 15:26:42 +02:00
|
|
|
dpb_length,
|
|
|
|
reinterpret_cast<char*>(GDS_VAL(sweep_dpb)));
|
|
|
|
|
|
|
|
assert (ihandle->ihndl_object == &db_handle);
|
|
|
|
ihandle->ihndl_object = NULL;
|
|
|
|
|
|
|
|
if (db_handle)
|
|
|
|
{
|
|
|
|
gds__detach_database(status_vector,
|
2002-12-16 17:25:10 +01:00
|
|
|
GDS_REF(db_handle));
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
gds__free(database);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
static void transaction_options(
|
|
|
|
TDBB tdbb,
|
2002-11-17 01:13:59 +01:00
|
|
|
JRD_TRA transaction,
|
2001-05-23 15:26:42 +02:00
|
|
|
UCHAR * tpb, USHORT tpb_length)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* t r a n s a c t i o n _ o p t i o n s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Process transaction options.
|
|
|
|
*
|
|
|
|
**************************************/
|
2002-11-17 01:13:59 +01:00
|
|
|
JRD_REL relation;
|
2001-05-23 15:26:42 +02:00
|
|
|
VEC vector;
|
|
|
|
LCK lock;
|
|
|
|
UCHAR *p, *end;
|
|
|
|
TEXT name[32], text[128];
|
|
|
|
USHORT flags;
|
|
|
|
USHORT l, level, op, wait;
|
|
|
|
int id;
|
|
|
|
SCHAR lock_type;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
if (!tpb_length)
|
|
|
|
return;
|
|
|
|
|
|
|
|
wait = 1;
|
|
|
|
end = tpb + tpb_length;
|
|
|
|
|
|
|
|
if (*tpb != gds_tpb_version3 && *tpb != gds_tpb_version1)
|
|
|
|
ERR_post(gds_bad_tpb_form, gds_arg_gds, gds_wrotpbver, 0);
|
|
|
|
|
|
|
|
++tpb;
|
|
|
|
|
|
|
|
while (tpb < end) {
|
|
|
|
switch (op = *tpb++) {
|
|
|
|
case gds_tpb_consistency:
|
|
|
|
transaction->tra_flags |= TRA_degree3;
|
|
|
|
transaction->tra_flags &= ~TRA_read_committed;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gds_tpb_concurrency:
|
|
|
|
transaction->tra_flags &= ~TRA_degree3;
|
|
|
|
transaction->tra_flags &= ~TRA_read_committed;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gds_tpb_read_committed:
|
|
|
|
transaction->tra_flags &= ~TRA_degree3;
|
|
|
|
transaction->tra_flags |= TRA_read_committed;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gds_tpb_shared:
|
|
|
|
case gds_tpb_protected:
|
|
|
|
case gds_tpb_exclusive:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gds_tpb_wait:
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gds_tpb_rec_version:
|
|
|
|
transaction->tra_flags |= TRA_rec_version;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gds_tpb_no_rec_version:
|
|
|
|
transaction->tra_flags &= ~TRA_rec_version;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gds_tpb_nowait:
|
|
|
|
wait = 0;
|
|
|
|
transaction->tra_flags |= TRA_nowait;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gds_tpb_read:
|
|
|
|
transaction->tra_flags |= TRA_readonly;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gds_tpb_write:
|
|
|
|
transaction->tra_flags &= ~TRA_readonly;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gds_tpb_ignore_limbo:
|
|
|
|
transaction->tra_flags |= TRA_ignore_limbo;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gds_tpb_no_auto_undo:
|
|
|
|
transaction->tra_flags |= TRA_no_auto_undo;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gds_tpb_lock_write:
|
|
|
|
case gds_tpb_lock_read:
|
|
|
|
p = reinterpret_cast < UCHAR * >(name);
|
2001-12-24 03:51:06 +01:00
|
|
|
if ( (l = *tpb++) ) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (l >= sizeof(name)) {
|
|
|
|
gds__msg_lookup(0, DYN_MSG_FAC, 159, sizeof(text),
|
|
|
|
text, &flags);
|
|
|
|
/* msg 159: Name longer than database column size */
|
|
|
|
ERR_post(gds_bad_tpb_content, gds_arg_gds, gds_random,
|
|
|
|
gds_arg_string, ERR_cstring(text), 0);
|
|
|
|
}
|
|
|
|
do
|
|
|
|
*p++ = *tpb++;
|
|
|
|
while (--l);
|
|
|
|
}
|
|
|
|
*p = 0;
|
|
|
|
if (!(relation = MET_lookup_relation(tdbb, name)))
|
|
|
|
ERR_post(gds_bad_tpb_content,
|
|
|
|
gds_arg_gds, gds_relnotdef, gds_arg_string,
|
|
|
|
ERR_cstring(name), 0);
|
|
|
|
|
|
|
|
/* force a scan to read view information */
|
|
|
|
MET_scan_relation(tdbb, relation);
|
|
|
|
|
|
|
|
lock_type = (op == gds_tpb_lock_read) ? LCK_none : LCK_SW;
|
|
|
|
if (tpb < end) {
|
|
|
|
if (*tpb == gds_tpb_shared)
|
|
|
|
tpb++;
|
|
|
|
else if (*tpb == gds_tpb_protected
|
|
|
|
|| *tpb == gds_tpb_exclusive) {
|
|
|
|
tpb++;
|
|
|
|
lock_type = (lock_type == LCK_SW) ? LCK_EX : LCK_PR;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
expand_view_lock(transaction, relation, lock_type);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gds_tpb_verb_time:
|
|
|
|
case gds_tpb_commit_time:
|
|
|
|
l = *tpb++;
|
|
|
|
tpb += l;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gds_tpb_autocommit:
|
|
|
|
transaction->tra_flags |= TRA_autocommit;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case gds_tpb_restart_requests:
|
|
|
|
transaction->tra_flags |= TRA_restart_requests;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
ERR_post(gds_bad_tpb_form, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If there aren't any relation locks to seize, we're done. */
|
|
|
|
|
|
|
|
if (!(vector = transaction->tra_relation_locks))
|
|
|
|
return;
|
|
|
|
|
2001-07-10 19:35:13 +02:00
|
|
|
/* Try to seize all relation locks.
|
2001-05-23 15:26:42 +02:00
|
|
|
If any can't be seized, release all and try again. */
|
|
|
|
|
|
|
|
id = 0;
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
for (; id < vector->count(); id++) {
|
|
|
|
if (!(lock = (LCK) (*vector)[id]))
|
2001-05-23 15:26:42 +02:00
|
|
|
continue;
|
|
|
|
level = lock->lck_logical;
|
|
|
|
if (level == LCK_none
|
|
|
|
|| LCK_lock_non_blocking(tdbb, lock, level, wait)) continue;
|
|
|
|
for (l = 0; l < id; l++)
|
2001-12-24 03:51:06 +01:00
|
|
|
if ( (lock = (LCK) (*vector)[l]) ) {
|
2001-05-23 15:26:42 +02:00
|
|
|
level = lock->lck_logical;
|
|
|
|
LCK_release(tdbb, lock);
|
|
|
|
lock->lck_logical = level;
|
|
|
|
}
|
|
|
|
id = 0;
|
|
|
|
if (!wait)
|
|
|
|
ERR_post(gds_lock_conflict, 0);
|
|
|
|
else
|
|
|
|
ERR_post(gds_deadlock, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef VMS
|
|
|
|
static BOOLEAN vms_convert(LCK lock, SLONG * data, SCHAR type, BOOLEAN wait)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* v m s _ c o n v e r t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Comply with VMS protocol for lock I/O.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SLONG status, flags;
|
|
|
|
LKSB lksb;
|
|
|
|
|
|
|
|
lksb.lksb_lock_id = lock->lck_id;
|
|
|
|
|
|
|
|
if (data && type < lock->lck_physical)
|
|
|
|
lksb.lksb_value[0] = *data;
|
|
|
|
|
|
|
|
flags = LCK$M_CONVERT;
|
|
|
|
|
|
|
|
if (data)
|
|
|
|
flags |= LCK$M_VALBLK;
|
|
|
|
|
|
|
|
if (!wait)
|
|
|
|
flags |= LCK$M_NOQUEUE;
|
|
|
|
|
|
|
|
status = sys$enqw(EVENT_FLAG, lock_types[type], &lksb, flags, NULL, NULL, NULL, /* AST routine when granted */
|
|
|
|
NULL, /* ast_argument */
|
|
|
|
NULL, /* ast_routine */
|
|
|
|
NULL, NULL);
|
|
|
|
|
|
|
|
if (!wait && status == SS$_NOTQUEUED)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (!(status & 1) || !((status = lksb.lksb_status) & 1))
|
|
|
|
ERR_post(gds__sys_request, gds_arg_string,
|
|
|
|
"sys$enqw (commit retaining lock)", gds_arg_vms, status, 0);
|
|
|
|
|
|
|
|
if (data && type >= lock->lck_physical)
|
|
|
|
*data = lksb.lksb_value[0];
|
|
|
|
|
|
|
|
lock->lck_physical = lock->lck_logical = type;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
#endif
|