8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-23 17:23:04 +01:00

Defer the process block cleanup till its death. This is dumber but more robust. It avoids races at LCK_fini() and fixes a rare deadlock there.

This commit is contained in:
dimitr 2008-03-04 19:07:55 +00:00
parent 5def15f13e
commit 0a3826d5e6

View File

@ -157,7 +157,6 @@ static void bug(ISC_STATUS*, const TEXT*);
#ifdef DEV_BUILD
static void bug_assert(const TEXT*, ULONG);
#endif
static void cleanup_process(ISC_STATUS*, prc*);
static bool convert(SRQ_PTR, UCHAR, SSHORT, lock_ast_t, void*, ISC_STATUS*);
static bool create_owner(ISC_STATUS*, LOCK_OWNER_T, UCHAR, SRQ_PTR*);
static bool create_process(ISC_STATUS*);
@ -200,6 +199,7 @@ static void validate_owner(const SRQ_PTR, USHORT);
static void validate_request(const SRQ_PTR, USHORT, USHORT);
static void validate_shb(const SRQ_PTR);
#endif
static void shutdown_blocking_thread(ISC_STATUS*);
static bool signal_owner(own*, SRQ_PTR);
static USHORT wait_for_request(lrq*, SSHORT, ISC_STATUS*);
@ -638,15 +638,8 @@ void LOCK_fini(ISC_STATUS* status_vector,
acquire(offset);
owner = (own*) SRQ_ABS_PTR(offset); // Re-init after a potential remap
fb_assert(owner->own_process == LOCK_process_offset);
prc* const process = (prc*) SRQ_ABS_PTR(owner->own_process);
fb_assert(process->prc_process_id == LOCK_pid);
purge_owner(offset, owner);
if (SRQ_EMPTY(process->prc_owners))
cleanup_process(status_vector, process);
release_mutex();
*owner_offset = (SRQ_PTR) 0;
@ -1126,7 +1119,8 @@ static UCHAR* alloc(SSHORT size, ISC_STATUS* status_vector)
// Post remapping notifications
remap_local_owners();
// Remap the shared memory region
const ULONG length = LOCK_data.sh_mem_length_mapped + Config::getLockMemSize();
const ULONG length = LOCK_data.sh_mem_length_mapped +
(LOCK_header->lhb_used > LOCK_header->lhb_length) ? Config::getLockMemSize() : 0;
lhb* header = (lhb*) ISC_remap_file(status_vector, &LOCK_data, length, true);
if (header) {
LOCK_header = header;
@ -1458,48 +1452,6 @@ static void bug(ISC_STATUS* status_vector, const TEXT* string)
}
static void cleanup_process(ISC_STATUS* status_vector, prc* process)
{
/**************************************
*
* c l e a n u p _ p r o c e s s
*
**************************************
*
* Functional description
* Purge the process, shutdown the blocking thread,
* unmap the process pointer.
*
**************************************/
LOCK_process_offset = 0;
if (LOCK_process) {
#ifdef USE_BLOCKING_THREAD
// Wait for AST thread to start (or 5 secs)
localMutex->leave();
startupSemaphore->tryEnter(5);
localMutex->enter();
// Wakeup the AST thread - it might be blocking or stalled
ISC_event_post(&LOCK_process->prc_blocking);
// Wait for the AST thread to finish cleanup or for 5 seconds
localMutex->leave();
cleanupSemaphore->tryEnter(5);
localMutex->enter();
#endif
#if defined HAVE_MMAP || defined WIN_NT
ISC_unmap_object(status_vector, &LOCK_data, (UCHAR**) &LOCK_process, sizeof(prc));
#else
LOCK_process = NULL;
#endif
}
purge_process(process);
}
static bool convert(SRQ_PTR request_offset,
UCHAR type,
SSHORT lck_wait,
@ -2060,21 +2012,26 @@ static void exit_handler(void* arg)
* by the cleanup handler.
*
**************************************/
if (!LOCK_header)
return;
ISC_STATUS_ARRAY local_status;
Firebird::MutexLockGuard guard(localMutex);
// Get rid of all the owners belonging to the current process
ISC_STATUS_ARRAY local_status;
const SRQ_PTR process_offset = LOCK_process_offset;
if (LOCK_process_offset) {
if (process_offset)
{
shutdown_blocking_thread(local_status);
acquire(DUMMY_OWNER);
prc* const process = (prc*) SRQ_ABS_PTR(LOCK_process_offset);
cleanup_process(local_status, process);
prc* const process = (prc*) SRQ_ABS_PTR(process_offset);
purge_process(process);
release_mutex();
}
ISC_unmap_file(local_status, &LOCK_data, 0);
if (LOCK_header)
{
ISC_unmap_file(local_status, &LOCK_data, 0);
}
}
@ -3145,6 +3102,46 @@ static void release_request(lrq* request)
}
static void shutdown_blocking_thread(ISC_STATUS* status_vector)
{
/**************************************
*
* s h u t d o w n _ b l o c k i n g _ t h r e a d
*
**************************************
*
* Functional description
* Shutdown the blocking thread and
* unmap the process pointer.
*
**************************************/
LOCK_process_offset = 0;
if (LOCK_process) {
#ifdef USE_BLOCKING_THREAD
// Wait for AST thread to start (or 5 secs)
localMutex->leave();
startupSemaphore->tryEnter(5);
localMutex->enter();
// Wakeup the AST thread - it might be blocking
ISC_event_post(&LOCK_process->prc_blocking);
// Wait for the AST thread to finish cleanup or for 5 seconds
localMutex->leave();
cleanupSemaphore->tryEnter(5);
localMutex->enter();
#endif
#if defined HAVE_MMAP || defined WIN_NT
ISC_unmap_object(status_vector, &LOCK_data, (UCHAR**) &LOCK_process, sizeof(prc));
#else
LOCK_process = NULL;
#endif
}
}
static bool signal_owner(own* blocking_owner, SRQ_PTR blocked_owner_offset)
{
/**************************************