2001-05-23 15:26:42 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: JRD Access Method
|
|
|
|
* MODULE: tra.h
|
|
|
|
* DESCRIPTION: Transaction block definitions
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Interbase Public
|
|
|
|
* License Version 1.0 (the "License"); you may not use this file
|
|
|
|
* except in compliance with the License. You may obtain a copy
|
|
|
|
* of the License at http://www.Inprise.com/IPL.html
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an
|
|
|
|
* "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either express
|
|
|
|
* or implied. See the License for the specific language governing
|
|
|
|
* rights and limitations under the License.
|
|
|
|
*
|
|
|
|
* The Original Code was created by Inprise Corporation
|
|
|
|
* and its predecessors. Portions created by Inprise Corporation are
|
|
|
|
* Copyright (C) Inprise Corporation.
|
|
|
|
*
|
|
|
|
* All Rights Reserved.
|
|
|
|
* Contributor(s): ______________________________________.
|
2002-07-01 17:07:18 +02:00
|
|
|
* 2001.6.25 Claudio Valderrama: add dfw_delete_generator and dfw_delete_udf
|
|
|
|
* to the dfw_t enumeration.
|
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
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef JRD_TRA_H
|
|
|
|
#define JRD_TRA_H
|
|
|
|
|
|
|
|
/*
|
|
|
|
* TMN: Fix this header! It should include any header it needs
|
2004-02-20 07:43:27 +01:00
|
|
|
* to define the types this header uses.
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
#include "../jrd/jrd_blks.h"
|
|
|
|
#include "../include/fb_blk.h"
|
2003-03-02 21:07:32 +01:00
|
|
|
#include "../common/classes/tree.h"
|
2004-11-24 21:48:59 +01:00
|
|
|
#include "../common/classes/GenericMap.h"
|
2003-04-25 16:51:04 +02:00
|
|
|
#include "../jrd/rpb_chain.h"
|
2004-04-18 04:50:38 +02:00
|
|
|
#include "../jrd/exe.h"
|
2004-06-22 22:13:10 +02:00
|
|
|
#include "../jrd/blb.h" // For bid structure
|
2004-09-28 08:28:38 +02:00
|
|
|
#include "../jrd/sbm.h" // For bid structure
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2006-07-17 19:44:18 +02:00
|
|
|
#include "../jrd/DatabaseSnapshot.h"
|
|
|
|
|
2004-03-20 15:57:40 +01:00
|
|
|
namespace Jrd {
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
class blb;
|
2004-03-18 06:56:06 +01:00
|
|
|
class Lock;
|
2004-03-11 06:04:26 +01:00
|
|
|
class jrd_rel;
|
2005-12-02 08:35:34 +01:00
|
|
|
template <typename T> class vec;
|
2004-03-18 06:56:06 +01:00
|
|
|
class Savepoint;
|
|
|
|
class Record;
|
|
|
|
class VerbAction;
|
2004-03-19 07:14:53 +01:00
|
|
|
class ArrayField;
|
2004-10-03 06:49:04 +02:00
|
|
|
class Attachment;
|
2004-03-19 07:14:53 +01:00
|
|
|
|
2004-06-22 22:13:10 +02:00
|
|
|
// Blobs active in transaction identified by bli_temp_id. Please keep this
|
|
|
|
// structure small as there can be huge amount of them floating in memory.
|
|
|
|
struct BlobIndex {
|
|
|
|
ULONG bli_temp_id;
|
|
|
|
bool bli_materialized;
|
|
|
|
jrd_req* bli_request;
|
|
|
|
union {
|
2004-10-03 22:18:13 +02:00
|
|
|
bid bli_blob_id; // ID of materialized blob
|
2004-06-22 22:13:10 +02:00
|
|
|
blb* bli_blob_object; // Blob object
|
|
|
|
};
|
|
|
|
static const ULONG& generate(const void *sender, const BlobIndex& item) {
|
|
|
|
return item.bli_temp_id;
|
|
|
|
}
|
|
|
|
// Empty default constructor to make it behave like POD structure
|
|
|
|
BlobIndex() {}
|
|
|
|
BlobIndex(ULONG temp_id, blb* blob_object) :
|
|
|
|
bli_temp_id(temp_id), bli_materialized(false), bli_request(NULL),
|
|
|
|
bli_blob_object(blob_object)
|
|
|
|
{ }
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef Firebird::BePlusTree<BlobIndex, ULONG, MemoryPool, BlobIndex> BlobIndexTree;
|
2004-02-20 07:43:27 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Transaction block */
|
|
|
|
|
2006-10-30 20:04:27 +01:00
|
|
|
const int DEFAULT_LOCK_TIMEOUT = -1; // infinite
|
|
|
|
|
2002-11-21 00:18:16 +01:00
|
|
|
class jrd_tra : public pool_alloc_rpt<SCHAR, type_tra>
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
2001-12-24 03:51:06 +01:00
|
|
|
public:
|
2004-10-07 11:15:32 +02:00
|
|
|
enum wait_t {
|
|
|
|
tra_no_wait,
|
|
|
|
tra_probe,
|
|
|
|
tra_wait
|
|
|
|
};
|
|
|
|
|
2006-10-07 12:53:01 +02:00
|
|
|
jrd_tra(MemoryPool& p) :
|
|
|
|
tra_blobs(&p),
|
|
|
|
tra_resources(p),
|
|
|
|
tra_context_vars(p),
|
2006-10-31 03:31:17 +01:00
|
|
|
tra_lock_timeout(DEFAULT_LOCK_TIMEOUT),
|
|
|
|
tra_stats(p)
|
2006-10-07 12:53:01 +02:00
|
|
|
{}
|
|
|
|
|
2004-10-03 06:49:04 +02:00
|
|
|
Attachment* tra_attachment; /* database attachment */
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG tra_number; /* transaction number */
|
|
|
|
SLONG tra_top; /* highest transaction in snapshot */
|
|
|
|
SLONG tra_oldest; /* oldest interesting transaction */
|
|
|
|
SLONG tra_oldest_active; /* record versions older than this can be
|
|
|
|
gargage-collected by this tx */
|
2004-02-20 07:43:27 +01:00
|
|
|
jrd_tra* tra_next; /* next transaction in database */
|
|
|
|
jrd_tra* tra_sibling; /* next transaction in group */
|
2001-12-24 03:51:06 +01:00
|
|
|
JrdMemoryPool* tra_pool; /* pool for transaction */
|
2004-06-22 22:13:10 +02:00
|
|
|
BlobIndexTree tra_blobs; /* list of active blobs */
|
2004-03-19 07:14:53 +01:00
|
|
|
ArrayField* tra_arrays; /* Linked list of active arrays */
|
2004-03-18 06:56:06 +01:00
|
|
|
Lock* tra_lock; /* lock for transaction */
|
2005-12-02 08:35:34 +01:00
|
|
|
vec<Lock*>* tra_relation_locks; /* locks for relations */
|
2004-09-28 08:28:38 +02:00
|
|
|
UInt32Bitmap* tra_commit_sub_trans; /* commited sub-transactions */
|
2007-01-06 11:18:20 +01:00
|
|
|
Savepoint* tra_save_point; /* list of savepoints */
|
|
|
|
Savepoint* tra_save_free; /* free savepoints */
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG tra_save_point_number; /* next save point number to use */
|
|
|
|
ULONG tra_flags;
|
2004-03-18 06:56:06 +01:00
|
|
|
class DeferredWork* tra_deferred_work; /* work deferred to commit time */
|
2004-04-18 04:50:38 +02:00
|
|
|
ResourceList tra_resources; /* resource existence list */
|
2004-11-22 21:22:03 +01:00
|
|
|
Firebird::StringMap tra_context_vars; // Context variables for the transaction
|
2004-03-18 06:56:06 +01:00
|
|
|
traRpbList* tra_rpblist; /* active record_param's of given transaction */
|
|
|
|
UCHAR tra_use_count; /* use count for safe AST delivery */
|
|
|
|
UCHAR tra_callback_count; /* callback count for 'execute statement' */
|
2004-10-07 11:15:32 +02:00
|
|
|
SSHORT tra_lock_timeout; /* in seconds, -1 means infinite, 0 means NOWAIT */
|
2004-03-18 06:56:06 +01:00
|
|
|
ULONG tra_next_blob_id; // ID of the previous blob or array created in this transaction
|
2006-07-17 19:44:18 +02:00
|
|
|
Firebird::TimeStamp tra_timestamp; // transaction start time
|
2004-12-07 02:19:55 +01:00
|
|
|
jrd_req* tra_requests; // Doubly linked list of requests active in this transaction
|
2006-07-17 19:44:18 +02:00
|
|
|
DatabaseSnapshot* tra_db_snapshot; // Database state snapshot (for monitoring purposes)
|
2006-10-07 12:53:01 +02:00
|
|
|
RuntimeStatistics tra_stats;
|
2001-05-23 15:26:42 +02:00
|
|
|
UCHAR tra_transactions[1];
|
2004-09-28 22:25:52 +02:00
|
|
|
|
2004-10-07 11:15:32 +02:00
|
|
|
SSHORT getLockWait() const
|
|
|
|
{
|
|
|
|
return -tra_lock_timeout;
|
|
|
|
}
|
2001-12-24 03:51:06 +01:00
|
|
|
};
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-11-06 04:20:18 +01:00
|
|
|
// System transaction is always transaction 0.
|
|
|
|
const SLONG TRA_system_transaction = 0;
|
|
|
|
|
|
|
|
// Flag definitions for tra_flags.
|
|
|
|
|
2004-05-01 00:47:16 +02:00
|
|
|
const ULONG TRA_system = 1L; /* system transaction */
|
2006-02-23 06:08:26 +01:00
|
|
|
//const ULONG TRA_update = 2L; // update is permitted
|
2004-05-01 00:47:16 +02:00
|
|
|
const ULONG TRA_prepared = 4L; /* transaction is in limbo */
|
|
|
|
const ULONG TRA_reconnected = 8L; /* reconnect in progress */
|
2006-02-23 06:08:26 +01:00
|
|
|
// How can RLCK_reserve_relation test for it if it's not set anywhere?
|
|
|
|
//const ULONG TRA_reserving = 16L; // relations explicityly locked
|
2004-05-01 00:47:16 +02:00
|
|
|
const ULONG TRA_degree3 = 32L; /* serializeable transaction */
|
2006-02-23 06:08:26 +01:00
|
|
|
//const ULONG TRA_committing = 64L; // commit in progress
|
2004-05-01 00:47:16 +02:00
|
|
|
const ULONG TRA_write = 128L; /* transaction has written */
|
|
|
|
const ULONG TRA_readonly = 256L; /* transaction is readonly */
|
2006-02-23 06:08:26 +01:00
|
|
|
//const ULONG TRA_nowait = 512L; // don't wait on relations, give up
|
2004-05-01 00:47:16 +02:00
|
|
|
const ULONG TRA_prepare2 = 1024L; /* transaction has updated RDB$TRANSACTIONS */
|
|
|
|
const ULONG TRA_ignore_limbo = 2048L; /* ignore transactions in limbo */
|
|
|
|
const ULONG TRA_invalidated = 4096L; /* transaction invalidated by failed write */
|
|
|
|
const ULONG TRA_deferred_meta = 8192L; /* deferred meta work posted */
|
2004-10-01 08:21:37 +02:00
|
|
|
//const ULONG TRA_add_log = 16384L; // write ahead log file was added
|
|
|
|
//const ULONG TRA_delete_log = 32768L; // write ahead log file was deleted
|
2004-05-01 00:47:16 +02:00
|
|
|
const ULONG TRA_read_committed = 65536L; /* can see latest committed records */
|
|
|
|
const ULONG TRA_autocommit = 131072L; /* autocommits all updates */
|
|
|
|
const ULONG TRA_perform_autocommit = 262144L; /* indicates autocommit is necessary */
|
|
|
|
const ULONG TRA_rec_version = 524288L; /* don't wait for uncommitted versions */
|
|
|
|
const ULONG TRA_restart_requests = 1048576L; /* restart all requests in attachment */
|
|
|
|
const ULONG TRA_no_auto_undo = 2097152L; /* don't start a savepoint in TRA_start */
|
2006-02-23 06:08:26 +01:00
|
|
|
// Obsolete, this was for LIBS on 16-bit.
|
|
|
|
//const ULONG TRA_sweep_at_startup = 4194304L; // sweep done at startup
|
2004-05-01 00:47:16 +02:00
|
|
|
const ULONG TRA_precommitted = 8388608L; /* transaction committed at startup */
|
|
|
|
|
|
|
|
const int TRA_MASK = 3;
|
2006-02-23 06:08:26 +01:00
|
|
|
//const int TRA_BITS_PER_TRANS = 2;
|
|
|
|
//const int TRA_TRANS_PER_BYTE = 4;
|
2004-05-01 00:47:16 +02:00
|
|
|
const int TRA_SHIFT = 2;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#define TRANS_SHIFT(number) (((number) & TRA_MASK) << 1)
|
|
|
|
#define TRANS_OFFSET(number) ((number) >> TRA_SHIFT)
|
|
|
|
|
|
|
|
/* Transaction cleanup. If a database is never quiescent, look
|
|
|
|
for "dead" active transactions every so often at transaction
|
|
|
|
startup */
|
|
|
|
|
2004-05-01 00:47:16 +02:00
|
|
|
const int TRA_ACTIVE_CLEANUP = 100;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Transaction states. The first four are states found
|
|
|
|
in the transaction inventory page; the last two are
|
|
|
|
returned internally */
|
|
|
|
|
2004-05-01 00:47:16 +02:00
|
|
|
const int tra_active = 0; /* Transaction is active */
|
|
|
|
const int tra_limbo = 1;
|
|
|
|
const int tra_dead = 2;
|
|
|
|
const int tra_committed = 3;
|
|
|
|
const int tra_us = 4; /* Transaction is us */
|
|
|
|
const int tra_precommitted = 5; /* Transaction is precommitted */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-19 00:00:21 +02:00
|
|
|
// The highest transaction number possible. This is 0x7fffffff if SLONG is 32 bits.
|
|
|
|
//#define MAX_TRA_NUMBER (~(1L << (BITS_PER_LONG - 1)))
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Savepoint block */
|
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
class Savepoint : public pool_alloc<type_sav>
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
|
|
|
public:
|
2004-03-28 11:10:30 +02:00
|
|
|
VerbAction* sav_verb_actions; /* verb action list */
|
2007-01-06 11:18:20 +01:00
|
|
|
VerbAction* sav_verb_free; /* free verb actions */
|
2004-03-18 06:56:06 +01:00
|
|
|
USHORT sav_verb_count; /* Active verb count */
|
|
|
|
SLONG sav_number; /* save point number */
|
|
|
|
Savepoint* sav_next;
|
|
|
|
USHORT sav_flags;
|
|
|
|
TEXT sav_name[32]; /* Savepoint name */
|
2001-12-24 03:51:06 +01:00
|
|
|
};
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Savepoint block flags. */
|
|
|
|
|
2004-05-01 00:47:16 +02:00
|
|
|
const int SAV_trans_level = 1; /* savepoint was started by TRA_start */
|
|
|
|
const int SAV_event_post = 2; /* event posted in the save point */
|
|
|
|
const int SAV_user = 4; /* named user savepoint as opposed to system ones */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-09-28 08:28:38 +02:00
|
|
|
/* Maximum size in bytes of transaction-level savepoint data.
|
2004-11-25 21:07:20 +01:00
|
|
|
When transaction-level savepoint gets past this size we drop it and use GC
|
|
|
|
mechanisms to clean out changes done in transaction */
|
|
|
|
const IPTR SAV_LARGE = 1024 * 32;
|
2002-11-03 18:29:51 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Deferred work blocks are used by the meta data handler to keep track
|
|
|
|
of work deferred to commit time. This are usually used to perform
|
|
|
|
meta data updates */
|
|
|
|
|
2003-08-22 12:56:55 +02:00
|
|
|
enum dfw_t {
|
2001-05-23 15:26:42 +02:00
|
|
|
dfw_null,
|
2002-12-23 16:49:14 +01:00
|
|
|
dfw_create_relation,
|
|
|
|
dfw_delete_relation,
|
|
|
|
dfw_update_format,
|
|
|
|
dfw_create_index,
|
|
|
|
dfw_delete_index,
|
|
|
|
dfw_compute_security,
|
|
|
|
dfw_add_file,
|
|
|
|
dfw_add_shadow,
|
|
|
|
dfw_delete_shadow,
|
|
|
|
dfw_modify_file,
|
|
|
|
dfw_erase_file,
|
|
|
|
dfw_delete_field,
|
|
|
|
dfw_delete_global,
|
|
|
|
dfw_delete_rfr,
|
|
|
|
dfw_post_event,
|
|
|
|
dfw_create_trigger,
|
|
|
|
dfw_delete_trigger,
|
|
|
|
dfw_modify_trigger,
|
2006-02-23 06:08:26 +01:00
|
|
|
//dfw_load_triggers,
|
2002-12-23 16:49:14 +01:00
|
|
|
dfw_grant,
|
|
|
|
dfw_revoke,
|
|
|
|
dfw_scan_relation,
|
|
|
|
dfw_create_expression_index,
|
|
|
|
dfw_delete_expression_index,
|
|
|
|
dfw_create_procedure,
|
|
|
|
dfw_modify_procedure,
|
|
|
|
dfw_delete_procedure,
|
|
|
|
dfw_delete_prm,
|
2006-08-07 18:39:21 +02:00
|
|
|
dfw_delete_collation,
|
2002-12-23 16:49:14 +01:00
|
|
|
dfw_delete_exception,
|
2006-02-23 06:08:26 +01:00
|
|
|
//dfw_unlink_file,
|
2002-12-23 16:49:14 +01:00
|
|
|
dfw_delete_generator,
|
2004-07-06 07:59:40 +02:00
|
|
|
dfw_modify_generator,
|
2003-08-06 18:30:49 +02:00
|
|
|
dfw_delete_udf,
|
|
|
|
dfw_add_difference,
|
|
|
|
dfw_delete_difference,
|
|
|
|
dfw_begin_backup,
|
2005-08-02 13:41:37 +02:00
|
|
|
dfw_end_backup,
|
|
|
|
|
2006-02-23 06:08:26 +01:00
|
|
|
// deferred works argument types
|
2005-08-02 13:41:37 +02:00
|
|
|
dfw_arg_index_name, // index name for dfw_delete_expression_index, mandatory
|
|
|
|
dfw_arg_partner_rel_id, // partner relation id for dfw_delete_index if index is FK, optional
|
2006-07-11 06:57:28 +02:00
|
|
|
dfw_arg_proc_name, // procedure name for dfw_delete_prm, mandatory
|
2006-08-19 04:52:39 +02:00
|
|
|
dfw_arg_force_computed, // we need to drop dependencies from a field that WAS computed
|
2006-11-05 19:30:36 +01:00
|
|
|
dfw_arg_check_blr, // check if BLR is still compilable
|
|
|
|
dfw_arg_rel_name, // relation name of a trigger
|
|
|
|
dfw_arg_trg_type // trigger type
|
2002-07-01 17:07:18 +02:00
|
|
|
};
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-05-12 20:28:04 +02:00
|
|
|
class DeferredWork : public pool_alloc<type_dfw>
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
2005-05-12 20:28:04 +02:00
|
|
|
public:
|
2004-02-20 07:43:27 +01:00
|
|
|
enum dfw_t dfw_type; /* type of work deferred */
|
2004-03-18 06:56:06 +01:00
|
|
|
DeferredWork* dfw_next; /* next block in transaction */
|
|
|
|
Lock* dfw_lock; /* relation creation lock */
|
|
|
|
DeferredWork* dfw_args; /* arguments */
|
2004-02-20 07:43:27 +01:00
|
|
|
SLONG dfw_sav_number; /* save point number */
|
|
|
|
USHORT dfw_id; /* object id, if appropriate */
|
|
|
|
USHORT dfw_count; /* count of block posts */
|
2005-05-12 20:28:04 +02:00
|
|
|
Firebird::string dfw_name; /* name of object */
|
|
|
|
public:
|
|
|
|
explicit DeferredWork(MemoryPool& p) : dfw_name(p) { }
|
2001-12-24 03:51:06 +01:00
|
|
|
};
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Verb actions */
|
|
|
|
|
2003-03-02 21:07:32 +01:00
|
|
|
class UndoItem {
|
|
|
|
public:
|
2004-09-28 08:28:38 +02:00
|
|
|
SINT64 rec_number;
|
2004-03-18 06:56:06 +01:00
|
|
|
Record* rec_data;
|
2004-09-28 08:28:38 +02:00
|
|
|
static const SINT64& generate(const void *sender, const UndoItem& item) {
|
2003-03-02 21:07:32 +01:00
|
|
|
return item.rec_number;
|
|
|
|
}
|
|
|
|
UndoItem() {
|
|
|
|
}
|
2004-09-28 08:28:38 +02:00
|
|
|
UndoItem(SINT64 rec_numberL, Record* rec_dataL) {
|
2004-05-12 21:39:17 +02:00
|
|
|
this->rec_number = rec_numberL;
|
|
|
|
this->rec_data = rec_dataL;
|
2003-03-02 21:07:32 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2004-09-28 08:28:38 +02:00
|
|
|
typedef Firebird::BePlusTree<UndoItem, SINT64, MemoryPool, UndoItem> UndoItemTree;
|
2003-03-02 21:07:32 +01:00
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
class VerbAction : public pool_alloc<type_vct>
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
|
|
|
public:
|
2004-03-18 06:56:06 +01:00
|
|
|
VerbAction* vct_next; /* Next action within verb */
|
2004-03-11 06:04:26 +01:00
|
|
|
jrd_rel* vct_relation; /* Relation involved */
|
2004-09-28 08:28:38 +02:00
|
|
|
RecordBitmap* vct_records; /* Record involved */
|
2004-03-11 06:04:26 +01:00
|
|
|
UndoItemTree* vct_undo; /* Data for undo records */
|
2001-12-24 03:51:06 +01:00
|
|
|
};
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-20 15:57:40 +01:00
|
|
|
} //namespace Jrd
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
#endif // JRD_TRA_H
|
|
|
|
|