2001-05-23 15:26:42 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: JRD Access Method
|
2003-12-31 06:36:12 +01:00
|
|
|
* MODULE: shut.cpp
|
2001-05-23 15:26:42 +02:00
|
|
|
* DESCRIPTION: Database shutdown handler
|
|
|
|
*
|
|
|
|
* 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-29 19:42:23 +02:00
|
|
|
#include "firebird.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/jrd.h"
|
|
|
|
#include "../jrd/scl.h"
|
2003-11-08 17:40:17 +01:00
|
|
|
#include "../jrd/y_ref.h"
|
|
|
|
#include "../jrd/ibase.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/ods.h"
|
|
|
|
#include "../jrd/cch_proto.h"
|
|
|
|
#include "../jrd/cmp_proto.h"
|
|
|
|
#include "../jrd/err_proto.h"
|
|
|
|
|
|
|
|
#include "../jrd/lck_proto.h"
|
|
|
|
#include "../jrd/rlck_proto.h"
|
|
|
|
#include "../jrd/rng_proto.h"
|
|
|
|
#include "../jrd/sch_proto.h"
|
|
|
|
#include "../jrd/shut_proto.h"
|
|
|
|
#include "../jrd/thd_proto.h"
|
|
|
|
#include "../jrd/tra_proto.h"
|
|
|
|
|
|
|
|
/* Shutdown lock data */
|
|
|
|
|
|
|
|
typedef union {
|
|
|
|
struct {
|
|
|
|
SSHORT flag;
|
|
|
|
SSHORT delay;
|
|
|
|
} data_items;
|
|
|
|
SLONG data_long;
|
|
|
|
} SDATA;
|
|
|
|
|
|
|
|
#define SHUT_WAIT_TIME 5
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
static bool notify_shutdown(DBB, SSHORT, SSHORT);
|
|
|
|
static bool shutdown_locks(DBB);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
|
|
|
BOOLEAN SHUT_blocking_ast(DBB dbb)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* S H U T _ b l o c k i n g _ a s t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Read data from database lock for
|
|
|
|
* shutdown instructions.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SDATA data;
|
|
|
|
data.data_long = LCK_read_data(dbb->dbb_lock);
|
2004-02-20 07:43:27 +01:00
|
|
|
const SSHORT flag = data.data_items.flag;
|
|
|
|
const SSHORT delay = data.data_items.delay;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Database shutdown has been cancelled. */
|
|
|
|
|
|
|
|
if (!flag) {
|
|
|
|
dbb->dbb_ast_flags &=
|
|
|
|
~(DBB_shut_attach | DBB_shut_tran | DBB_shut_force |
|
|
|
|
DBB_shutdown);
|
|
|
|
dbb->dbb_shutdown_delay = 0;
|
2004-02-20 07:43:27 +01:00
|
|
|
for (att* attachment = dbb->dbb_attachments; attachment;
|
|
|
|
attachment = attachment->att_next)
|
|
|
|
{
|
|
|
|
attachment->att_flags &= ~ATT_shutdown_notify;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
if (flag & isc_dpb_shut_force && !delay)
|
2001-05-23 15:26:42 +02:00
|
|
|
return shutdown_locks(dbb);
|
|
|
|
else {
|
2003-11-08 17:40:17 +01:00
|
|
|
if (flag & isc_dpb_shut_attachment)
|
2001-05-23 15:26:42 +02:00
|
|
|
dbb->dbb_ast_flags |= DBB_shut_attach;
|
2003-11-08 17:40:17 +01:00
|
|
|
if (flag & isc_dpb_shut_force)
|
2001-05-23 15:26:42 +02:00
|
|
|
dbb->dbb_ast_flags |= DBB_shut_force;
|
2003-11-08 17:40:17 +01:00
|
|
|
if (flag & isc_dpb_shut_transaction)
|
2001-05-23 15:26:42 +02:00
|
|
|
dbb->dbb_ast_flags |= DBB_shut_tran;
|
|
|
|
dbb->dbb_shutdown_delay = delay;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BOOLEAN SHUT_database(DBB dbb, SSHORT flag, SSHORT delay)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* S H U T _ d a t a b a s e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Schedule database for shutdown
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
TDBB tdbb = GET_THREAD_DATA;
|
|
|
|
att* attachment = tdbb->tdbb_attachment;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Only platform's user locksmith can shutdown or bring online
|
|
|
|
a database. */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if (!(attachment->att_user->usr_flags & (USR_locksmith | USR_owner))) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return FALSE;
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
try {
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If shutdown flag is zero then bring database online */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if (!flag)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Clear shutdown flag on database header page */
|
|
|
|
|
2003-12-11 11:33:30 +01:00
|
|
|
WIN window(HEADER_PAGE);
|
2004-02-20 07:43:27 +01:00
|
|
|
header_page* header =
|
|
|
|
(header_page*) CCH_FETCH(tdbb, &window, LCK_write, pag_header);
|
2001-05-23 15:26:42 +02:00
|
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
|
|
|
header->hdr_flags &= ~hdr_shutdown;
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
|
|
|
|
/* Notify existing database clients that a currently
|
|
|
|
scheduled shutdown is cancelled. */
|
|
|
|
|
|
|
|
if (notify_shutdown(dbb, 0, 0))
|
|
|
|
CCH_release_exclusive(tdbb);
|
|
|
|
|
|
|
|
/* Notify local attachments */
|
|
|
|
|
|
|
|
SHUT_blocking_ast(dbb);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
attachment->att_flags |= ATT_shutdown_manager;
|
|
|
|
--dbb->dbb_use_count;
|
|
|
|
|
|
|
|
/* Database is being shutdown. First notification gives shutdown
|
|
|
|
type and delay in seconds. */
|
|
|
|
|
|
|
|
notify_shutdown(dbb, flag, delay);
|
|
|
|
|
|
|
|
/* Notify local attachments */
|
|
|
|
|
|
|
|
SHUT_blocking_ast(dbb);
|
|
|
|
|
|
|
|
/* Try to get exclusive database lock periodically up to specified delay. If we
|
|
|
|
haven't gotten it report shutdown error for weaker forms. For forced shutdown
|
|
|
|
keep notifying until successful. */
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
bool exclusive = false;
|
|
|
|
SSHORT timeout;
|
2001-05-23 15:26:42 +02:00
|
|
|
for (timeout = delay; timeout >= 0; timeout -= SHUT_WAIT_TIME)
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if ((exclusive = notify_shutdown(dbb, flag, timeout)) ||
|
|
|
|
!(dbb->dbb_ast_flags & (DBB_shut_attach | DBB_shut_tran |
|
2001-12-24 03:51:06 +01:00
|
|
|
DBB_shut_force)))
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!exclusive && (timeout > 0 ||
|
2003-11-08 17:40:17 +01:00
|
|
|
flag & (isc_dpb_shut_attachment |
|
|
|
|
isc_dpb_shut_transaction)))
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
notify_shutdown(dbb, 0, 0); /* Tell everyone we're giving up */
|
|
|
|
SHUT_blocking_ast(dbb);
|
|
|
|
attachment->att_flags &= ~ATT_shutdown_manager;
|
|
|
|
++dbb->dbb_use_count;
|
2003-11-08 17:40:17 +01:00
|
|
|
ERR_post(isc_shutfail, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Once there are no more transactions active, force all remaining
|
|
|
|
attachments to shutdown. */
|
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
if (flag & isc_dpb_shut_transaction) {
|
2004-02-20 07:43:27 +01:00
|
|
|
exclusive = false;
|
2003-11-08 17:40:17 +01:00
|
|
|
flag = isc_dpb_shut_force;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
dbb->dbb_ast_flags |= DBB_shutdown;
|
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
if (!exclusive && flag & isc_dpb_shut_force) {
|
2001-12-24 03:51:06 +01:00
|
|
|
// TMN: Ugly counting!
|
2001-05-23 15:26:42 +02:00
|
|
|
while (!notify_shutdown(dbb, flag, 0));
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
++dbb->dbb_use_count;
|
|
|
|
dbb->dbb_ast_flags &= ~(DBB_shut_force | DBB_shut_attach | DBB_shut_tran);
|
2003-12-11 11:33:30 +01:00
|
|
|
WIN window(HEADER_PAGE);
|
2004-02-20 07:43:27 +01:00
|
|
|
header_page* header =
|
|
|
|
(header_page*) CCH_FETCH(tdbb, &window, LCK_write, pag_header);
|
2001-05-23 15:26:42 +02:00
|
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
|
|
|
header->hdr_flags |= hdr_shutdown;
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
CCH_release_exclusive(tdbb);
|
|
|
|
|
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
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
BOOLEAN SHUT_init(DBB dbb)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* S H U T _ i n i t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Read data from database lock for
|
|
|
|
* shutdown instructions.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
return SHUT_blocking_ast(dbb);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
static bool notify_shutdown(DBB dbb, SSHORT flag, SSHORT delay)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* n o t i f y _ s h u t d o w n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Notify database users that shutdown
|
|
|
|
* status of a database is changing.
|
|
|
|
* Pulse database lock and pass shutdown
|
|
|
|
* flags and delay via lock data.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
TDBB tdbb = GET_THREAD_DATA;
|
|
|
|
SDATA data;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
data.data_items.flag = flag;
|
|
|
|
data.data_items.delay = delay;
|
|
|
|
|
|
|
|
LCK_write_data(dbb->dbb_lock, data.data_long);
|
|
|
|
|
|
|
|
/* Send blocking ASTs to database users */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if (CCH_exclusive(tdbb, LCK_PW, ((SSHORT) - SHUT_WAIT_TIME)) && flag) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return shutdown_locks(dbb);
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2003-11-08 17:40:17 +01:00
|
|
|
if ((flag & isc_dpb_shut_force) && !delay) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return shutdown_locks(dbb);
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2003-11-08 17:40:17 +01:00
|
|
|
if ((flag & isc_dpb_shut_transaction) &&
|
2001-12-24 03:51:06 +01:00
|
|
|
!(TRA_active_transactions(tdbb, dbb)))
|
|
|
|
{
|
2004-02-20 07:43:27 +01:00
|
|
|
return true;
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
return false;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
static bool shutdown_locks(DBB dbb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s h u t d o w n _ l o c k s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Release all attachment and database
|
|
|
|
* locks if database is quiet.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-02-20 07:43:27 +01:00
|
|
|
TDBB tdbb = GET_THREAD_DATA;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Mark database and all active attachments as shutdown. */
|
|
|
|
|
|
|
|
dbb->dbb_ast_flags |= DBB_shutdown;
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
att* attachment;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
for (attachment = dbb->dbb_attachments; attachment;
|
2003-09-13 14:03:11 +02:00
|
|
|
attachment = attachment->att_next)
|
|
|
|
{
|
|
|
|
if (!(attachment->att_flags & ATT_shutdown_manager))
|
|
|
|
attachment->att_flags |= ATT_shutdown;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (dbb->dbb_use_count) {
|
|
|
|
/* Let active database threads rundown */
|
|
|
|
|
|
|
|
THREAD_EXIT;
|
|
|
|
THREAD_SLEEP(1 * 1000);
|
|
|
|
THREAD_ENTER;
|
2004-02-20 07:43:27 +01:00
|
|
|
return false;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Since no attachment is actively running, release all
|
|
|
|
attachment-specfic locks while they're not looking. */
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
att* shut_attachment = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
for (attachment = dbb->dbb_attachments; attachment;
|
2004-02-20 07:43:27 +01:00
|
|
|
attachment = attachment->att_next)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if (attachment->att_flags & ATT_shutdown_manager) {
|
|
|
|
shut_attachment = attachment;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (attachment->att_id_lock)
|
|
|
|
LCK_release(tdbb, attachment->att_id_lock);
|
|
|
|
|
|
|
|
#ifdef PC_ENGINE
|
|
|
|
RNG_shutdown_attachment(attachment);
|
|
|
|
#endif
|
|
|
|
RLCK_shutdown_attachment(attachment);
|
|
|
|
TRA_shutdown_attachment(tdbb, attachment);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Release database locks that are shared by all attachments.
|
|
|
|
These include relation and index existence locks, as well
|
|
|
|
as, relation interest and record locking locks for PC semantic
|
|
|
|
record locking. */
|
|
|
|
|
|
|
|
RLCK_shutdown_database(dbb);
|
|
|
|
CMP_shutdown_database(tdbb);
|
|
|
|
|
|
|
|
/* If shutdown manager is here, leave enough database lock context
|
|
|
|
to run as a normal attachment. Otherwise, get rid of the rest
|
|
|
|
of the database locks.*/
|
|
|
|
|
|
|
|
if (!shut_attachment) {
|
|
|
|
CCH_shutdown_database(dbb);
|
|
|
|
if (dbb->dbb_shadow_lock)
|
|
|
|
LCK_release(tdbb, dbb->dbb_shadow_lock);
|
|
|
|
if (dbb->dbb_retaining_lock)
|
|
|
|
LCK_release(tdbb, dbb->dbb_retaining_lock);
|
|
|
|
if (dbb->dbb_lock)
|
|
|
|
LCK_release(tdbb, dbb->dbb_lock);
|
2003-08-06 18:30:49 +02:00
|
|
|
dbb->backup_manager->shutdown_locks();
|
2001-05-23 15:26:42 +02:00
|
|
|
dbb->dbb_ast_flags |= DBB_shutdown_locks;
|
|
|
|
}
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
return true;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-02-20 07:43:27 +01:00
|
|
|
|