mirror of
https://github.com/FirebirdSQL/firebird.git
synced 2025-01-27 20:03:03 +01:00
5729790ed6
2. Small generic cleanup
1015 lines
27 KiB
C++
1015 lines
27 KiB
C++
/*
|
|
* PROGRAM: JRD Access Method
|
|
* MODULE: rlck.cpp
|
|
* DESCRIPTION: Record and Relation Lock 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.06 Sean Leyne - Code Cleanup, removed "#ifdef READONLY_DATABASE"
|
|
* conditionals, as the engine now fully supports
|
|
* readonly databases.
|
|
*/
|
|
|
|
#include "firebird.h"
|
|
#include "../jrd/common.h"
|
|
#include "../jrd/jrd.h"
|
|
#include "../jrd/tra.h"
|
|
#include "../jrd/lck.h"
|
|
#include "../jrd/req.h"
|
|
#include "gen/iberror.h"
|
|
#include "../jrd/all.h"
|
|
#include "../jrd/all_proto.h"
|
|
#include "../jrd/err_proto.h"
|
|
#include "../jrd/isc_proto.h"
|
|
|
|
#include "../jrd/lck_proto.h"
|
|
#include "../jrd/rlck_proto.h"
|
|
#include "../jrd/thd.h"
|
|
#include "../jrd/vio_proto.h"
|
|
|
|
using namespace Jrd;
|
|
|
|
#ifdef PC_ENGINE
|
|
static Lock* allocate_record_lock(jrd_tra*, record_param*);
|
|
#endif
|
|
static Lock* allocate_relation_lock(MemoryPool*, jrd_rel*);
|
|
#ifdef PC_ENGINE
|
|
static Lock* attachment_relation_lock(jrd_rel*);
|
|
static void drop_record_lock(Lock*);
|
|
static Lock* find_record_lock(record_param*);
|
|
static bool obtain_lock(jrd_tra*, Lock*, USHORT);
|
|
static int start_record_locking(void*);
|
|
#endif
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
Lock* RLCK_lock_record(record_param* rpb,
|
|
USHORT lock_level, lock_ast_t ast, BLK ast_arg)
|
|
{
|
|
/**************************************
|
|
*
|
|
* R L C K _ l o c k _ r e c o r d
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Lock the current record in the provided
|
|
* record parameter block at the specified level.
|
|
*
|
|
**************************************/
|
|
// SLONG process_count;
|
|
|
|
|
|
/* first get a record lock on the desired record;
|
|
if we can't get one then there is no point
|
|
in signalling that this process needs to acquire locks */
|
|
|
|
Lock* lock = RLCK_lock_record_implicit(0, rpb, lock_level, ast, ast_arg);
|
|
if (!lock)
|
|
return NULL;
|
|
|
|
thread_db* tdbb = JRD_get_thread_data();
|
|
|
|
/*
|
|
if the record is trying to be locked, check
|
|
whether the record has already been updated on page
|
|
by a transaction whose updates we cannot see yet--
|
|
in which case we effectively cannot get the lock */
|
|
|
|
if (VIO_check_if_updated(tdbb, rpb)) {
|
|
RLCK_unlock_record_implicit(lock, rpb);
|
|
return NULL;
|
|
}
|
|
|
|
/* if this process has no record locks outstanding,
|
|
indicate through the lock manager that we are
|
|
starting the record locking protocol */
|
|
|
|
jrd_rel* relation = rpb->rpb_relation;
|
|
if (!relation->rel_explicit_locks) {
|
|
Lock* record_locking = RLCK_record_locking(relation);
|
|
|
|
/* get a shared write lock to be compatible with other processes
|
|
that are doing record locking, but incompatible with those who aren't */
|
|
|
|
if (record_locking->lck_logical == LCK_none)
|
|
LCK_lock_non_blocking(tdbb, record_locking, LCK_SW, LCK_WAIT);
|
|
else
|
|
LCK_convert_non_blocking(tdbb, tdbb, record_locking, LCK_SW,
|
|
LCK_WAIT);
|
|
}
|
|
|
|
relation->rel_explicit_locks++;
|
|
|
|
return lock;
|
|
}
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
Lock* RLCK_lock_record_implicit(jrd_tra* transaction,
|
|
record_param* rpb,
|
|
USHORT lock_level,
|
|
lock_ast_t ast, BLK ast_arg)
|
|
{
|
|
/**************************************
|
|
*
|
|
* R L C K _ l o c k _ r e c o r d _ i m p l i c i t
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Lock the given record for implicit access, a subset
|
|
* of what we need to do to lock it explicitly.
|
|
*
|
|
* transaction is NULL for explicit record locking.
|
|
*
|
|
**************************************/
|
|
USHORT interest_lock_level = 0;
|
|
|
|
Lock* lock = allocate_record_lock(transaction, rpb);
|
|
lock->lck_ast = ast;
|
|
lock->lck_object = ast_arg;
|
|
|
|
/* To ensure that relation and record locks respect each other,
|
|
utilize a multigranularity locking scheme, establishing
|
|
an interest lock level in the parent relation according to
|
|
the following scheme:
|
|
read lock on record := protected read
|
|
(implies) interest read lock on relation := shared read
|
|
write lock on record := exclusive
|
|
(implies) interest write lock on relation := shared write
|
|
Then as long as PR and EX locks are used to implement read
|
|
and write locking on relations, the relation locks will
|
|
be respected.
|
|
|
|
This also guarantees that implicit record locks won't result
|
|
in interest locks being taken out, since implicit record locks
|
|
are SW. Relations are already implicitly reserved in
|
|
RLCK_reserve_relation(), so there is no need for an additional
|
|
interest lock.
|
|
implicit read lock on record := NO LOCK
|
|
(implies) NO interest read lock on relation
|
|
implicit write lock on record := shared write
|
|
(implies) interest write lock on relation := shared write
|
|
*/
|
|
|
|
jrd_rel* relation = rpb->rpb_relation;
|
|
if (lock_level == LCK_EX) {
|
|
interest_lock_level = LCK_SW;
|
|
relation->rel_write_locks++;
|
|
relation->rel_lock_total++;
|
|
}
|
|
else if (lock_level == LCK_PR) {
|
|
interest_lock_level = LCK_SR;
|
|
relation->rel_read_locks++;
|
|
relation->rel_lock_total++;
|
|
}
|
|
|
|
/* first attempt to get the parent lock then get the record lock,
|
|
using two-phase locking to help prevent deadlocks */
|
|
|
|
if (interest_lock_level
|
|
&& !obtain_lock(transaction, lock->lck_parent, interest_lock_level)
|
|
|| !obtain_lock(transaction, lock, lock_level))
|
|
{
|
|
RLCK_unlock_record_implicit(lock, rpb);
|
|
return NULL;
|
|
}
|
|
|
|
return lock;
|
|
}
|
|
#endif
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
Lock* RLCK_lock_relation(jrd_rel* relation,
|
|
USHORT lock_level, lock_ast_t ast, BLK ast_arg)
|
|
{
|
|
/**************************************
|
|
*
|
|
* R L C K _ l o c k _ r e l a t i o n
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Lock a relation at the specified level.
|
|
*
|
|
**************************************/
|
|
|
|
/* allocate a relation lock hanging off the attachment
|
|
block, then keep a count of the number of times it
|
|
is used so that we know when to release it (bug #7478) */
|
|
|
|
Lock* lock = attachment_relation_lock(relation);
|
|
lock->lck_count++;
|
|
lock->lck_ast = ast;
|
|
lock->lck_object = ast_arg;
|
|
|
|
if (!obtain_lock(0, lock, lock_level))
|
|
{
|
|
ERR_post(isc_relation_lock, isc_arg_string,
|
|
relation->rel_name.c_str(), 0);
|
|
}
|
|
return lock;
|
|
}
|
|
#endif
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
Lock* RLCK_range_relation(jrd_tra* transaction,
|
|
jrd_rel* relation, lock_ast_t ast, BLK ast_arg)
|
|
{
|
|
/**************************************
|
|
*
|
|
* R L C K _ r a n g e _ r e l a t i o n
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Lock a relation for a refresh range.
|
|
*
|
|
**************************************/
|
|
thread_db* tdbb = JRD_get_thread_data();
|
|
Attachment* attachment = tdbb->tdbb_attachment;
|
|
|
|
if (transaction->tra_flags & TRA_system)
|
|
return NULL;
|
|
|
|
Lock* lock = allocate_relation_lock(transaction->tra_pool, relation);
|
|
lock->lck_owner = attachment;
|
|
lock->lck_ast = ast;
|
|
lock->lck_object = ast_arg;
|
|
lock->lck_type = LCK_range_relation;
|
|
|
|
const USHORT level = LCK_PR;
|
|
|
|
/* get lock */
|
|
|
|
const USHORT result =
|
|
LCK_lock_non_blocking(tdbb, lock, level, transaction->getLockWait());
|
|
|
|
if (result)
|
|
return lock;
|
|
else
|
|
return NULL;
|
|
}
|
|
#endif
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
Lock* RLCK_record_locking(jrd_rel* relation)
|
|
{
|
|
/**************************************
|
|
*
|
|
* R L C K _ r e c o r d _ l o c k i n g
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Allocate a record locking lock, which determines
|
|
* when record locking is necessary for a particular
|
|
* relation.
|
|
*
|
|
**************************************/
|
|
if (relation->rel_record_locking)
|
|
return relation->rel_record_locking;
|
|
|
|
thread_db* tdbb = JRD_get_thread_data();
|
|
Database* dbb = GET_DBB();
|
|
|
|
Lock* lock = FB_NEW_RPT(*dbb->dbb_permanent, sizeof(SLONG)) Lock();
|
|
lock->lck_parent = dbb->dbb_lock;
|
|
lock->lck_dbb = dbb;
|
|
lock->lck_type = LCK_record_locking;
|
|
lock->lck_owner_handle = LCK_get_owner_handle(tdbb, lock->lck_type);
|
|
|
|
/* make the lock specific to the relation */
|
|
|
|
lock->lck_length = sizeof(SLONG);
|
|
lock->lck_key.lck_long = relation->rel_id;
|
|
relation->rel_record_locking = lock;
|
|
|
|
/* set up an ast to start record locking if
|
|
someone gets an incompatible lock */
|
|
|
|
lock->lck_ast = start_record_locking;
|
|
lock->lck_object = relation;
|
|
|
|
/* now attempt to get a PR on the lock to detect when
|
|
anyone locks a record explicitly */
|
|
|
|
LCK_lock(tdbb, _non_blocking(lock, LCK_PR, LCK_NO_WAIT));
|
|
return lock;
|
|
}
|
|
#endif
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
void RLCK_release_lock(Lock* lock)
|
|
{
|
|
/**************************************
|
|
*
|
|
* R L C K _ r e l e a s e _ l o c k
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Release a lock of record or relation
|
|
* type. The lock handle came from the
|
|
* user, so validate it carefully.
|
|
*
|
|
**************************************/
|
|
|
|
/* first do basic validation of the handle */
|
|
|
|
if (!lock)
|
|
ERR_post(isc_bad_lock_handle, 0);
|
|
if (((BLK) lock)->blk_type != (UCHAR) type_lck)
|
|
ERR_post(isc_bad_lock_handle, 0);
|
|
/* now use the lock type to determine the type
|
|
of lock to release */
|
|
if (lock->lck_type == LCK_relation)
|
|
RLCK_unlock_relation(0, (jrd_rel*) lock->lck_object);
|
|
else if (lock->lck_type == LCK_record)
|
|
RLCK_unlock_record(lock, 0);
|
|
else
|
|
ERR_post(isc_bad_lock_handle, 0);
|
|
}
|
|
#endif
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
void RLCK_release_locks(Attachment* attachment)
|
|
{
|
|
/**************************************
|
|
*
|
|
* R L C K _ r e l e a s e _ l o c k s
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Release all relation and record locks
|
|
* explicitly taken out during this attachment.
|
|
*
|
|
**************************************/
|
|
|
|
/* unlock all explicit relation locks */
|
|
vec* vector = attachment->att_relation_locks;
|
|
if (vector) {
|
|
vec::iterator lptr = vector->begin();
|
|
for (const vec::const_iterator lend = vector->end(); lptr < lend; lptr++)
|
|
{
|
|
Lock* lock = (Lock*)(*lptr);
|
|
if (lock)
|
|
RLCK_unlock_relation(0, (jrd_rel*) lock->lck_object);
|
|
}
|
|
}
|
|
/* unlock all explicit record locks */
|
|
Lock* lock;
|
|
while (lock = attachment->att_record_locks) {
|
|
RLCK_unlock_record(lock, 0);
|
|
}
|
|
/* clear the vector of user locks */
|
|
if (vector = attachment->att_lck_quick_ref) {
|
|
vec::iterator lptr = vector->begin();
|
|
for (const vec::const_iterator lend = vector->end(); lptr < lend; lptr++)
|
|
{
|
|
*lptr = NULL;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
Lock* RLCK_reserve_relation(thread_db* tdbb,
|
|
jrd_tra* transaction,
|
|
jrd_rel* relation, bool write_flag, bool error_flag)
|
|
{
|
|
/**************************************
|
|
*
|
|
* R L C K _ r e s e r v e _ r e l a t i o n
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Lock a relation within a transaction. If the relation
|
|
* is already locked at a lower level, upgrade the lock.
|
|
*
|
|
**************************************/
|
|
if (transaction->tra_flags & TRA_system)
|
|
return NULL;
|
|
if (write_flag && (tdbb->tdbb_database->dbb_flags & DBB_read_only))
|
|
ERR_post(isc_read_only_database, 0);
|
|
if (write_flag && (transaction->tra_flags & TRA_readonly))
|
|
ERR_post(isc_read_only_trans, 0);
|
|
Lock* lock = RLCK_transaction_relation_lock(transaction, relation);
|
|
|
|
/* Next, figure out what kind of lock we need */
|
|
USHORT level;
|
|
if (transaction->tra_flags & TRA_degree3) {
|
|
if (write_flag)
|
|
level = LCK_EX;
|
|
else
|
|
level = LCK_PR;
|
|
}
|
|
else {
|
|
if (write_flag)
|
|
level = LCK_SW;
|
|
else
|
|
level = LCK_none;
|
|
}
|
|
|
|
/* If the lock is already "good enough", we're done */
|
|
|
|
if (level <= lock->lck_logical)
|
|
return lock;
|
|
if (transaction->tra_flags & TRA_reserving)
|
|
{
|
|
ERR_post(isc_unres_rel, isc_arg_string,
|
|
relation->rel_name.c_str(), 0);
|
|
}
|
|
/* get lock */
|
|
USHORT result;
|
|
if (lock->lck_logical)
|
|
result = LCK_convert_non_blocking(NULL, lock, level,
|
|
transaction->getLockWait());
|
|
else
|
|
result = LCK_lock_non_blocking(NULL, lock, level,
|
|
transaction->getLockWait());
|
|
if (result)
|
|
return lock;
|
|
else {
|
|
if (error_flag)
|
|
ERR_punt();
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
|
|
void RLCK_shutdown_attachment(Attachment* attachment)
|
|
{
|
|
/**************************************
|
|
*
|
|
* R L C K _ s h u t d o w n _ a t t a c h m e n t
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Shutdown the attachment's persistent record
|
|
* and relation locks. This runs at AST level.
|
|
*
|
|
**************************************/
|
|
thread_db* tdbb = JRD_get_thread_data();
|
|
/* Release child record locks before parent relation locks */
|
|
for (Lock* record_lock = attachment->att_record_locks;
|
|
record_lock;
|
|
record_lock = record_lock->lck_att_next)
|
|
{
|
|
LCK_release(tdbb, record_lock);
|
|
}
|
|
vec* lock_vector = attachment->att_relation_locks;
|
|
if (lock_vector) {
|
|
for (vec::iterator lock = lock_vector->begin();
|
|
lock != lock_vector->end(); ++lock)
|
|
{
|
|
if (*lock) {
|
|
LCK_release(tdbb, (Lock*)(*lock));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void RLCK_shutdown_database(Database* dbb)
|
|
{
|
|
/**************************************
|
|
*
|
|
* R L C K _ s h u t d o w n _ d a t a b a s e
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Shutdown the database locks shared for relation
|
|
* interest and record locking locks. This can be called
|
|
* at AST level.
|
|
*
|
|
**************************************/
|
|
thread_db* tdbb = JRD_get_thread_data();
|
|
vec* vector = dbb->dbb_relations;
|
|
if (!vector)
|
|
return;
|
|
|
|
vec::iterator ptr = vector->begin();
|
|
for (const vec::const_iterator end = vector->end(); ptr < end; ptr++)
|
|
{
|
|
jrd_rel* relation = (jrd_rel*)(*ptr);
|
|
if (relation) {
|
|
if (relation->rel_record_locking)
|
|
LCK_release(tdbb, relation->rel_record_locking);
|
|
if (relation->rel_interest_lock)
|
|
LCK_release(tdbb, relation->rel_interest_lock);
|
|
relation->rel_explicit_locks = 0;
|
|
relation->rel_read_locks = 0;
|
|
relation->rel_write_locks = 0;
|
|
relation->rel_lock_total = 0;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
void RLCK_signal_refresh(jrd_tra* transaction)
|
|
{
|
|
/**************************************
|
|
*
|
|
* R L C K _ s i g n a l _ r e f r e s h
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* For each reserved relation, get a refresh relation
|
|
* lock to signal possible refresh range users.
|
|
*
|
|
**************************************/
|
|
thread_db* tdbb = JRD_get_thread_data();
|
|
Database* dbb = tdbb->tdbb_database;
|
|
/* for each relation, take out a range relation lock and then release it */
|
|
vec* vector = transaction->tra_relation_locks;
|
|
if (vector) {
|
|
|
|
/* allocate a local lock */
|
|
|
|
Lock* local_lock = FB_NEW_RPT(*dbb->dbb_permanent, sizeof(SLONG)) Lock();
|
|
local_lock->lck_dbb = dbb;
|
|
local_lock->lck_length = sizeof(SLONG);
|
|
local_lock->lck_type = LCK_range_relation;
|
|
local_lock->lck_owner_handle =
|
|
LCK_get_owner_handle(tdbb, local_lock->lck_type);
|
|
local_lock->lck_parent = dbb->dbb_lock;
|
|
local_lock->lck_compatible = tdbb->tdbb_attachment;
|
|
for (size_t i = 0; i < vector->count(); i++) {
|
|
Lock* lock = (Lock**) ((*vector)[i]);
|
|
if (lock) {
|
|
jrd_rel* relation = (jrd_rel*) lock->lck_object;
|
|
local_lock->lck_key.lck_long = relation->rel_id;
|
|
local_lock->lck_object = relation;
|
|
LCK_lock_non_blocking(tdbb, local_lock, LCK_SW, LCK_NO_WAIT);
|
|
LCK_release(tdbb, local_lock);
|
|
}
|
|
}
|
|
ALL_release(local_lock);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
Lock* RLCK_transaction_relation_lock(jrd_tra* transaction, jrd_rel* relation)
|
|
{
|
|
/**************************************
|
|
*
|
|
* R L C K _ t r a n s a c t i o n _ r e l a t i o n _ l o c k
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Take out a relation lock within the context of
|
|
* a transaction.
|
|
*
|
|
**************************************/
|
|
Lock* lock;
|
|
vec* vector = transaction->tra_relation_locks;
|
|
if (vector &&
|
|
(relation->rel_id < vector->count()) &&
|
|
(lock = (Lock*) (*vector)[relation->rel_id]))
|
|
{
|
|
return lock;
|
|
}
|
|
|
|
vector = transaction->tra_relation_locks =
|
|
vec::newVector(*transaction->tra_pool, transaction->tra_relation_locks,
|
|
relation->rel_id + 1);
|
|
|
|
if ( (lock = (Lock*) (*vector)[relation->rel_id]) )
|
|
return lock;
|
|
|
|
lock = allocate_relation_lock(transaction->tra_pool, relation);
|
|
lock->lck_owner = transaction;
|
|
/* for relations locked within a transaction, add a second level of
|
|
compatibility within the intra-process lock manager which specifies
|
|
that relation locks are incompatible with locks taken out by other
|
|
transactions, if a transaction is specified */
|
|
lock->lck_compatible2 = transaction;
|
|
(*vector)[relation->rel_id] = (BLK) lock;
|
|
|
|
return lock;
|
|
}
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
void RLCK_unlock_record(Lock* lock, record_param* rpb)
|
|
{
|
|
/**************************************
|
|
*
|
|
* R L C K _ u n l o c k _ r e c o r d
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Unlock the specified record lock, or if
|
|
* it's not available use the current record
|
|
* in the specified record parameter block.
|
|
*
|
|
**************************************/
|
|
jrd_rel* relation;
|
|
if (rpb)
|
|
relation = rpb->rpb_relation;
|
|
else if (lock)
|
|
relation = (jrd_rel*) lock->lck_parent->lck_object;
|
|
else
|
|
relation = NULL; /* theoretically impossible */
|
|
|
|
RLCK_unlock_record_implicit(lock, rpb);
|
|
thread_db* tdbb = JRD_get_thread_data();
|
|
Attachment* attachment = tdbb->tdbb_attachment;
|
|
if (attachment->att_flags & ATT_shutdown)
|
|
return;
|
|
/* decrement the count of explicit locks this process has taken out;
|
|
if there are none, go back to a PR on the lock--
|
|
if we cannot obtain this, it means someone else has explicit
|
|
record locks taken out and we should just release the lock */
|
|
if (relation && !--relation->rel_explicit_locks) {
|
|
Lock* record_locking = relation->rel_record_locking;
|
|
if (!LCK_convert_non_blocking(tdbb, record_locking, LCK_PR, LCK_NO_WAIT))
|
|
LCK_release(tdbb, record_locking);
|
|
}
|
|
|
|
}
|
|
#endif
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
void RLCK_unlock_record_implicit(Lock* lock, record_param* rpb)
|
|
{
|
|
/**************************************
|
|
*
|
|
* R L C K _ u n l o c k _ r e c o r d _ i m p l i c i t
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Unlock a record-level lock.
|
|
*
|
|
**************************************/
|
|
thread_db* tdbb = JRD_get_thread_data();
|
|
if (!lock)
|
|
lock = find_record_lock(rpb);
|
|
const USHORT lock_level = lock->lck_logical;
|
|
drop_record_lock(lock);
|
|
LCK_release(tdbb, lock);
|
|
Attachment* attachment = tdbb->tdbb_attachment;
|
|
if (attachment->att_flags & ATT_shutdown) {
|
|
ALL_release(lock);
|
|
return;
|
|
}
|
|
|
|
/* handle the release half of the multigranularity locking scheme:
|
|
if there are no more write locks, downgrade the interest
|
|
lock on the parent relation; if there are no more read or
|
|
write locks, release the lock entirely
|
|
|
|
For implicit locks, there are no relation intrest locks.
|
|
So do nothing.
|
|
*/
|
|
|
|
jrd_rel* relation = (jrd_rel*) lock->lck_parent->lck_object;
|
|
if (lock_level == LCK_EX) {
|
|
if (!--relation->rel_write_locks)
|
|
if (!relation->rel_read_locks)
|
|
LCK_release(tdbb, relation->rel_interest_lock);
|
|
else
|
|
LCK_convert_non_blocking(tdbb, relation->rel_interest_lock,
|
|
LCK_SR, LCK_WAIT);
|
|
}
|
|
else if (lock_level == LCK_PR) {
|
|
if (!--relation->rel_read_locks && !relation->rel_write_locks)
|
|
LCK_release(tdbb, relation->rel_interest_lock);
|
|
}
|
|
|
|
ALL_release(lock);
|
|
}
|
|
#endif
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
void RLCK_unlock_relation(Lock* lock, jrd_rel* relation)
|
|
{
|
|
/**************************************
|
|
*
|
|
* R L C K _ u n l o c k _ r e l a t i o n
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Release the attachment's persistent lock
|
|
* on the specified relation.
|
|
*
|
|
**************************************/
|
|
thread_db* tdbb = JRD_get_thread_data();
|
|
Attachment* attachment = tdbb->tdbb_attachment;
|
|
vec* vector = attachment->att_relation_locks;
|
|
if (!vector)
|
|
return;
|
|
|
|
if (relation) {
|
|
const USHORT id = relation->rel_id;
|
|
if (id >= vector->count())
|
|
return;
|
|
lock = (Lock*) (*vector)[id];
|
|
}
|
|
else {
|
|
for (int id = 0; id < vector->count(); id++)
|
|
if (lock == (Lock*) (*vector)[id])
|
|
break;
|
|
}
|
|
if (!lock)
|
|
return;
|
|
/* decrement the use count; if it goes to zero,
|
|
there are no further locks taken out in this
|
|
attachment so we can release the lock (bug #7478) */
|
|
if (lock->lck_count > 1) {
|
|
lock->lck_count--;
|
|
return;
|
|
}
|
|
|
|
LCK_release(tdbb, lock);
|
|
ALL_release(lock);
|
|
(*vector)[id] = NULL;
|
|
}
|
|
#endif
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
static Lock* allocate_record_lock(jrd_tra* transaction, record_param* rpb)
|
|
{
|
|
/**************************************
|
|
*
|
|
* a l l o c a t e _ r e c o r d _ l o c k
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Create a record lock block for the current
|
|
* record in the passed rpb.
|
|
* transaction used only for implicit record locks.
|
|
*
|
|
**************************************/
|
|
thread_db* tdbb = JRD_get_thread_data();
|
|
Database* dbb = tdbb->tdbb_database;
|
|
Attachment* attachment = tdbb->tdbb_attachment;
|
|
if (!rpb->rpb_record)
|
|
ERR_post(isc_no_cur_rec, 0);
|
|
/* allocate a lock block for the record lock */
|
|
Lock* lock = FB_NEW_RPT(*dbb->dbb_permanent, sizeof(SLONG)) Lock();
|
|
lock->lck_dbb = dbb;
|
|
lock->lck_object = reinterpret_cast<blk*>(dbb);
|
|
lock->lck_type = LCK_record;
|
|
lock->lck_owner_handle = LCK_get_owner_handle(tdbb, lock->lck_type);
|
|
/* use the relation lock as the lock parent */
|
|
jrd_rel* relation = rpb->rpb_relation;
|
|
if (transaction)
|
|
lock->lck_parent =
|
|
RLCK_transaction_relation_lock(transaction, relation);
|
|
else {
|
|
if (!relation->rel_interest_lock)
|
|
relation->rel_interest_lock =
|
|
allocate_relation_lock(dbb->dbb_permanent, relation);
|
|
lock->lck_parent = relation->rel_interest_lock;
|
|
}
|
|
|
|
/* indicate that this lock should be compatible at the attachment
|
|
level--meaning that two record locks taken out within the same
|
|
attachment should always be compatible; ditto for the interest lock */
|
|
|
|
lock->lck_compatible = attachment;
|
|
lock->lck_parent->lck_compatible = attachment;
|
|
/* link in the record lock with the other record locks
|
|
taken out by this attachment */
|
|
lock->lck_att_next = attachment->att_record_locks;
|
|
attachment->att_record_locks = lock;
|
|
/* fill in the lock value using the record number */
|
|
lock->lck_length = sizeof(SLONG);
|
|
lock->lck_key.lck_long = rpb->rpb_number;
|
|
return lock;
|
|
}
|
|
#endif
|
|
|
|
|
|
static Lock* allocate_relation_lock(MemoryPool* pool, jrd_rel* relation)
|
|
{
|
|
/**************************************
|
|
*
|
|
* a l l o c a t e _ r e l a t i o n _ l o c k
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Allocate a lock block for a relation lock.
|
|
*
|
|
**************************************/
|
|
thread_db* tdbb = JRD_get_thread_data();
|
|
Database* dbb = tdbb->tdbb_database;
|
|
Lock* lock = FB_NEW_RPT(*pool, sizeof(SLONG)) Lock();
|
|
lock->lck_dbb = dbb;
|
|
lock->lck_length = sizeof(SLONG);
|
|
lock->lck_key.lck_long = relation->rel_id;
|
|
lock->lck_type = LCK_relation;
|
|
lock->lck_owner_handle = LCK_get_owner_handle(tdbb, lock->lck_type);
|
|
lock->lck_parent = dbb->dbb_lock;
|
|
/* enter all relation locks into the intra-process lock manager and treat
|
|
them as compatible within the attachment according to IPLM rules */
|
|
lock->lck_compatible = tdbb->tdbb_attachment;
|
|
/* the lck_object is used here to find the relation
|
|
block from the lock block */
|
|
lock->lck_object = relation;
|
|
return lock;
|
|
}
|
|
|
|
#ifdef PC_ENGINE
|
|
static Lock* attachment_relation_lock(jrd_rel* relation)
|
|
{
|
|
/**************************************
|
|
*
|
|
* a t t a c h m e n t _ r e l a t i o n _ l o c k
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Take out a persistent relation lock at the
|
|
* attachment level.
|
|
*
|
|
**************************************/
|
|
thread_db* tdbb = JRD_get_thread_data();
|
|
Database* dbb = tdbb->tdbb_database;
|
|
Attachment* attachment = tdbb->tdbb_attachment;
|
|
|
|
Lock* lock;
|
|
vec* vector = attachment->att_relation_locks;
|
|
if (vector &&
|
|
(relation->rel_id < vector->count()) &&
|
|
(lock = (Lock*) (*vector)[relation->rel_id]))
|
|
{
|
|
return lock;
|
|
}
|
|
|
|
vector = attachment->att_relation_locks =
|
|
vec::newVector(*dbb->dbb_permanent, attachment->att_relation_locks,
|
|
relation->rel_id + 1);
|
|
|
|
if ( (lock = (Lock*) (*vector)[relation->rel_id]) )
|
|
return lock;
|
|
|
|
lock = allocate_relation_lock(dbb->dbb_permanent, relation);
|
|
lock->lck_owner = attachment;
|
|
(*vector)[relation->rel_id] = (BLK) lock;
|
|
|
|
return lock;
|
|
}
|
|
#endif
|
|
|
|
#ifdef PC_ENGINE
|
|
static void drop_record_lock(Lock* record_lock)
|
|
{
|
|
/**************************************
|
|
*
|
|
* d r o p _ r e c o r d _ l o c k
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Find the record lock in the linked list off
|
|
* the attachment block, and drop it from the list.
|
|
*
|
|
**************************************/
|
|
thread_db* tdbb = JRD_get_thread_data();
|
|
/* look through all the record locks taken out by this attachment
|
|
looking for one with the same record number and relation id */
|
|
Attachment* attachment = tdbb->tdbb_attachment;
|
|
for (Lock** lock = &attachment->att_record_locks; *lock;
|
|
lock = &(*lock)->lck_att_next)
|
|
{
|
|
if (*lock == record_lock) {
|
|
*lock = (*lock)->lck_att_next;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
static Lock* find_record_lock(record_param* rpb)
|
|
{
|
|
/**************************************
|
|
*
|
|
* f i n d _ r e c o r d _ l o c k
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Find the record lock previously
|
|
* defined for a record.
|
|
*
|
|
**************************************/
|
|
thread_db* tdbb = JRD_get_thread_data();
|
|
/* look through all the record locks taken out by this attachment
|
|
looking for one with the same record number and relation */
|
|
Attachment* attachment = tdbb->tdbb_attachment;
|
|
Lock* lock;
|
|
for (lock = attachment->att_record_locks; lock; lock = lock->lck_att_next)
|
|
{
|
|
if ((rpb->rpb_number == lock->lck_key.lck_long)
|
|
&& (rpb->rpb_relation == (jrd_rel*) lock->lck_parent->lck_object))
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
return lock;
|
|
}
|
|
#endif
|
|
|
|
#ifdef PC_ENGINE
|
|
static bool obtain_lock(jrd_tra* transaction, Lock* lock, USHORT lock_level)
|
|
{
|
|
/**************************************
|
|
*
|
|
* o b t a i n _ l o c k
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Obtain the specified lock at the
|
|
* necessary level.
|
|
*
|
|
**************************************/
|
|
const SSHORT wait_flag =
|
|
transaction ? transaction->getLockWait() : 0;
|
|
/* return if lock level OK and if the lock has not been released
|
|
(like as part of a refresh range) */
|
|
if ((lock_level <= lock->lck_logical) && (lock->lck_id != -1))
|
|
return true;
|
|
if ((lock->lck_logical) && (lock->lck_id != -1)) {
|
|
if (LCK_convert_non_blocking(NULL, lock, lock_level, wait_flag))
|
|
return true;
|
|
}
|
|
else if (LCK_lock_non_blocking(NULL, lock, lock_level, wait_flag))
|
|
return true;
|
|
return false;
|
|
}
|
|
#endif
|
|
|
|
#ifdef PC_ENGINE
|
|
static int start_record_locking(void* relation_void)
|
|
{
|
|
/**************************************
|
|
*
|
|
* s t a r t _ r e c o r d _ l o c k i n g
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* A blocking AST has been issued to give up
|
|
* the lock on the record locking semaphore.
|
|
* Flag the fact that record locking is now
|
|
* necessary for reading and writing records.
|
|
*
|
|
**************************************/
|
|
jrd_rel* relation = static_cast<jrd_rel*>(relation_void);
|
|
Lock* record_locking = relation->rel_record_locking;
|
|
/* if we have shared write, it means we have records
|
|
locked; we won't give up this lock for anyone! */
|
|
if (record_locking->lck_physical == LCK_SW)
|
|
return 0;
|
|
ISC_ast_enter();
|
|
LCK_release(NULL, record_locking);
|
|
ISC_ast_exit();
|
|
return 0;
|
|
}
|
|
#endif
|
|
|