8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-25 04:43:03 +01:00
firebird-mirror/src/jrd/shut.cpp

606 lines
16 KiB
C++
Raw Normal View History

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): ______________________________________.
*/
#include "firebird.h"
2004-03-22 12:38:23 +01:00
#include "../jrd/common.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/ibase.h"
#include "../jrd/nbak.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/sch_proto.h"
#include "../jrd/shut_proto.h"
#include "../jrd/thd.h"
#include "../jrd/thread_proto.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/tra_proto.h"
using namespace Jrd;
2004-05-07 00:11:24 +02:00
const SSHORT SHUT_WAIT_TIME = 5;
2001-05-23 15:26:42 +02:00
// Shutdown lock data
union shutdown_data {
struct {
SSHORT flag;
SSHORT delay;
} data_items;
SLONG data_long;
};
// Define this to true if you need to allow no-op behavior when requested shutdown mode
// matches current. Logic of jrd8_create_database may need attention in this case too
2004-05-07 00:11:24 +02:00
const bool IGNORE_SAME_MODE = false;
2007-03-09 09:27:34 +01:00
static bool bad_mode(thread_db*, bool);
static void check_backup_state(thread_db*);
2004-03-07 08:58:55 +01:00
static bool notify_shutdown(Database*, SSHORT, SSHORT);
static bool shutdown_locks(Database*, SSHORT);
2001-05-23 15:26:42 +02:00
2004-03-07 08:58:55 +01:00
bool SHUT_blocking_ast(Database* dbb)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* 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.
*
**************************************/
shutdown_data data;
2001-05-23 15:26:42 +02:00
data.data_long = LCK_read_data(dbb->dbb_lock);
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. */
// Delay of -1 means we're going online
if (delay == -1) {
2001-05-23 15:26:42 +02:00
dbb->dbb_ast_flags &=
~(DBB_shut_attach | DBB_shut_tran | DBB_shut_force |
DBB_shutdown | DBB_shutdown_single | DBB_shutdown_full);
switch (flag & isc_dpb_shut_mode_mask) {
case isc_dpb_shut_normal:
break;
case isc_dpb_shut_multi:
dbb->dbb_ast_flags |= DBB_shutdown;
break;
case isc_dpb_shut_single:
dbb->dbb_ast_flags |= DBB_shutdown | DBB_shutdown_single;
break;
case isc_dpb_shut_full:
dbb->dbb_ast_flags |= DBB_shutdown | DBB_shutdown_full;
break;
}
dbb->dbb_shutdown_delay = 0; // not tested anywhere
/* CVC: We never set it, so how could we need to unset ATT_shutdown_modify?
for (Attachment* attachment = dbb->dbb_attachments; attachment;
attachment = attachment->att_next)
{
attachment->att_flags &= ~ATT_shutdown_notify;
}
*/
return false;
2001-05-23 15:26:42 +02:00
}
if ((flag & isc_dpb_shut_force) && !delay)
return shutdown_locks(dbb, flag);
2001-05-23 15:26:42 +02:00
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; // not tested anywhere
return false;
}
}
2001-05-23 15:26:42 +02:00
2004-03-07 08:58:55 +01:00
bool SHUT_database(Database* dbb, SSHORT flag, SSHORT delay)
{
/**************************************
*
* S H U T _ d a t a b a s e
*
**************************************
*
* Functional description
* Schedule database for shutdown
*
**************************************/
thread_db* tdbb = JRD_get_thread_data();
Attachment* attachment = tdbb->tdbb_attachment;
2001-05-23 15:26:42 +02:00
/* Only platform's user locksmith can shutdown or bring online
a database. */
if (!attachment->locksmith())
{
return false;
}
// Check if requested shutdown mode is valid
// Note that if we are already in requested mode we just return true.
// This is required to ensure backward compatible behavior (gbak relies on that,
// user-written scripts may rely on this behaviour too)
int shut_mode = flag & isc_dpb_shut_mode_mask;
switch (shut_mode) {
case isc_dpb_shut_full:
2004-03-07 08:58:55 +01:00
if (dbb->dbb_ast_flags & DBB_shutdown_full)
2007-03-09 09:27:34 +01:00
return bad_mode(tdbb, IGNORE_SAME_MODE);
break;
case isc_dpb_shut_multi:
if ((dbb->dbb_ast_flags & DBB_shutdown_full) ||
(dbb->dbb_ast_flags & DBB_shutdown_single))
{
2007-03-09 09:27:34 +01:00
return bad_mode(tdbb, false);
}
if (dbb->dbb_ast_flags & DBB_shutdown)
2007-03-09 09:27:34 +01:00
return bad_mode(tdbb, IGNORE_SAME_MODE);
break;
case isc_dpb_shut_single:
if (dbb->dbb_ast_flags & DBB_shutdown_full)
2007-03-09 09:27:34 +01:00
return bad_mode(tdbb, false);
if (dbb->dbb_ast_flags & DBB_shutdown_single)
2007-03-09 09:27:34 +01:00
return bad_mode(tdbb, IGNORE_SAME_MODE);
break;
case isc_dpb_shut_normal:
if (!(dbb->dbb_ast_flags & DBB_shutdown))
2007-03-09 09:27:34 +01:00
return bad_mode(tdbb, IGNORE_SAME_MODE);
return bad_mode(tdbb, false);
default:
2007-03-09 09:27:34 +01:00
return bad_mode(tdbb, false); // unexpected mode
2001-05-23 15:26:42 +02:00
}
try {
2001-05-23 15:26:42 +02:00
2007-03-09 09:27:34 +01:00
// Reject exclusive and single-user shutdown attempts
// for a physically locked database
if (shut_mode == isc_dpb_shut_full ||
shut_mode == isc_dpb_shut_single)
{
check_backup_state(tdbb);
}
2001-05-23 15:26:42 +02:00
attachment->att_flags |= ATT_shutdown_manager;
--dbb->dbb_use_count;
/* Database is being shutdown. First notification gives shutdown
type and delay in seconds. */
bool exclusive = notify_shutdown(dbb, flag, delay);
2001-05-23 15:26:42 +02:00
/* 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. */
SSHORT timeout = delay - SHUT_WAIT_TIME;
if (!exclusive)
2001-12-24 03:51:06 +01:00
{
for (; timeout >= 0; timeout -= SHUT_WAIT_TIME)
2001-12-24 03:51:06 +01:00
{
if ((exclusive = notify_shutdown(dbb, flag, timeout)) ||
!(dbb->dbb_ast_flags & (DBB_shut_attach | DBB_shut_tran |
DBB_shut_force)))
{
break;
}
2001-12-24 03:51:06 +01:00
}
}
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) {
exclusive = false;
flag = isc_dpb_shut_force | (flag & isc_dpb_shut_mode_mask);
2001-05-23 15:26:42 +02:00
}
dbb->dbb_ast_flags |= DBB_shutdown;
dbb->dbb_ast_flags &= ~(DBB_shutdown_single | DBB_shutdown_full);
if (flag & isc_dpb_shut_single)
dbb->dbb_ast_flags |= DBB_shutdown_single;
else if (flag & isc_dpb_shut_full)
dbb->dbb_ast_flags |= DBB_shutdown_full;
2001-05-23 15:26:42 +02: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);
2006-05-22 00:07:35 +02:00
WIN window(HEADER_PAGE_NUMBER);
Ods::header_page* header =
(Ods::header_page*) CCH_FETCH(tdbb, &window, LCK_write, pag_header);
2001-05-23 15:26:42 +02:00
CCH_MARK_MUST_WRITE(tdbb, &window);
// Set appropriate shutdown mode in database header
header->hdr_flags &= ~Ods::hdr_shutdown_mask;
switch (flag & isc_dpb_shut_mode_mask) {
case isc_dpb_shut_normal:
break;
case isc_dpb_shut_multi:
header->hdr_flags |= Ods::hdr_shutdown_multi;
break;
case isc_dpb_shut_single:
header->hdr_flags |= Ods::hdr_shutdown_single;
break;
case isc_dpb_shut_full:
header->hdr_flags |= Ods::hdr_shutdown_full;
break;
}
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, &window);
CCH_release_exclusive(tdbb);
2001-12-24 03:51:06 +01:00
} // try
catch (const Firebird::Exception& ex) {
Firebird::stuff_exception(tdbb->tdbb_status_vector, ex);
return false;
2001-12-24 03:51:06 +01:00
}
return true;
2001-05-23 15:26:42 +02:00
}
2004-03-07 08:58:55 +01:00
bool SHUT_init(Database* dbb)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* S H U T _ i n i t
*
**************************************
*
* Functional description
* Read data from database lock for
* shutdown instructions.
*
**************************************/
return SHUT_blocking_ast(dbb);
}
2004-03-07 08:58:55 +01:00
bool SHUT_online(Database* dbb, SSHORT flag)
{
/**************************************
*
* S H U T _ o n l i n e
*
**************************************
*
* Functional description
* Move database to "more online" state
*
**************************************/
thread_db* tdbb = JRD_get_thread_data();
Attachment* attachment = tdbb->tdbb_attachment;
2004-03-07 08:58:55 +01:00
/* Only platform's user locksmith can shutdown or bring online
a database. */
if (!attachment->att_user->locksmith()) {
2004-03-07 08:58:55 +01:00
return false;
}
// Check if requested shutdown mode is valid
int shut_mode = flag & isc_dpb_shut_mode_mask;
switch (shut_mode) {
2004-03-07 08:58:55 +01:00
case isc_dpb_shut_normal:
if (!(dbb->dbb_ast_flags & DBB_shutdown))
2007-03-09 09:27:34 +01:00
return bad_mode(tdbb, IGNORE_SAME_MODE); // normal -> normal
2004-03-07 08:58:55 +01:00
break;
case isc_dpb_shut_multi:
if (!(dbb->dbb_ast_flags & DBB_shutdown))
2007-03-09 09:27:34 +01:00
return bad_mode(tdbb, false); // normal -> multi
2004-03-07 08:58:55 +01:00
if (!(dbb->dbb_ast_flags & DBB_shutdown_full) &&
!(dbb->dbb_ast_flags & DBB_shutdown_single))
{
2007-03-09 09:27:34 +01:00
return bad_mode(tdbb, IGNORE_SAME_MODE); // multi -> multi
2004-03-07 08:58:55 +01:00
}
break;
case isc_dpb_shut_single:
if (dbb->dbb_ast_flags & DBB_shutdown_single)
2007-03-09 09:27:34 +01:00
return bad_mode(tdbb, IGNORE_SAME_MODE); //single -> single
2004-03-07 08:58:55 +01:00
if (!(dbb->dbb_ast_flags & DBB_shutdown_full))
2007-03-09 09:27:34 +01:00
return bad_mode(tdbb, false); // !full -> single
2004-03-07 08:58:55 +01:00
break;
case isc_dpb_shut_full:
if (dbb->dbb_ast_flags & DBB_shutdown_full)
{
2007-03-09 09:27:34 +01:00
return bad_mode(tdbb, IGNORE_SAME_MODE); // full -> full
2004-03-07 08:58:55 +01:00
}
2007-03-09 09:27:34 +01:00
return bad_mode(tdbb, false);
2004-03-07 08:58:55 +01:00
default: // isc_dpb_shut_full
2007-03-09 09:27:34 +01:00
return bad_mode(tdbb, false); // unexpected mode
2004-03-07 08:58:55 +01:00
}
try {
2007-03-09 09:27:34 +01:00
// Reject exclusive and single-user shutdown attempts
// for a physically locked database
if (shut_mode == isc_dpb_shut_full ||
shut_mode == isc_dpb_shut_single)
{
check_backup_state(tdbb);
}
2004-03-07 08:58:55 +01:00
/* Clear shutdown flag on database header page */
2006-05-22 00:07:35 +02:00
WIN window(HEADER_PAGE_NUMBER);
Ods::header_page* header = (Ods::header_page*) CCH_FETCH(tdbb, &window, LCK_write, pag_header);
2004-03-07 08:58:55 +01:00
CCH_MARK_MUST_WRITE(tdbb, &window);
// Set appropriate shutdown mode in database header
header->hdr_flags &= ~Ods::hdr_shutdown_mask;
switch (shut_mode) {
2004-03-07 08:58:55 +01:00
case isc_dpb_shut_normal:
break;
case isc_dpb_shut_multi:
header->hdr_flags |= Ods::hdr_shutdown_multi;
2004-03-07 08:58:55 +01:00
break;
case isc_dpb_shut_single:
header->hdr_flags |= Ods::hdr_shutdown_single;
2004-03-07 08:58:55 +01:00
break;
case isc_dpb_shut_full:
header->hdr_flags |= Ods::hdr_shutdown_full;
2004-03-07 08:58:55 +01:00
break;
}
CCH_RELEASE(tdbb, &window);
/* Notify existing database clients that a currently
scheduled shutdown is cancelled. */
if (notify_shutdown(dbb, shut_mode, -1))
CCH_release_exclusive(tdbb);
/* Notify local attachments */
SHUT_blocking_ast(dbb);
} // try
catch (const Firebird::Exception& ex) {
2004-03-07 08:58:55 +01:00
Firebird::stuff_exception(tdbb->tdbb_status_vector, ex);
return false;
}
return true;
}
2007-03-09 09:27:34 +01:00
static bool bad_mode(thread_db* tdbb, bool ignore)
{
2004-03-07 08:58:55 +01:00
if (!ignore) {
2007-03-09 09:27:34 +01:00
Database* dbb = tdbb->tdbb_database;
2004-03-07 08:58:55 +01:00
ISC_STATUS* status = tdbb->tdbb_status_vector;
*status++ = isc_arg_gds;
*status++ = isc_bad_shutdown_mode;
*status++ = isc_arg_string;
2007-03-09 09:27:34 +01:00
*status++ = (ISC_STATUS) (IPTR) ERR_cstring(dbb->dbb_filename.c_str());
2004-03-07 08:58:55 +01:00
*status++ = isc_arg_end;
}
return ignore;
}
2007-03-09 09:27:34 +01:00
static void check_backup_state(thread_db* tdbb)
{
Database* dbb = tdbb->tdbb_database;
dbb->dbb_backup_manager->lock_shared_database(tdbb, true);
2007-03-09 09:27:34 +01:00
try {
if (dbb->dbb_backup_manager->get_state() != nbak_state_normal)
{
ERR_post(isc_bad_shutdown_mode,
isc_arg_string,
ERR_cstring(dbb->dbb_filename.c_str()),
0);
}
}
catch (const Firebird::Exception&) {
dbb->dbb_backup_manager->unlock_shared_database(tdbb);
2007-03-09 09:27:34 +01:00
throw;
}
dbb->dbb_backup_manager->unlock_shared_database(tdbb);
2007-03-09 09:27:34 +01:00
}
2004-03-07 08:58:55 +01:00
static bool notify_shutdown(Database* 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.
*
**************************************/
thread_db* tdbb = JRD_get_thread_data();
shutdown_data 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 */
bool exclusive =
CCH_exclusive(tdbb, LCK_PW, delay > 0 ? -SHUT_WAIT_TIME : LCK_NO_WAIT);
if (exclusive && (delay != -1)) {
return shutdown_locks(dbb, flag);
2001-12-24 03:51:06 +01:00
}
2003-11-08 17:40:17 +01:00
if ((flag & isc_dpb_shut_force) && !delay) {
return shutdown_locks(dbb, flag);
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)))
{
return true;
2001-12-24 03:51:06 +01:00
}
return exclusive;
2001-05-23 15:26:42 +02:00
}
2004-03-07 08:58:55 +01:00
static bool shutdown_locks(Database* dbb, SSHORT flag)
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.
*
**************************************/
thread_db* tdbb = JRD_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 | DBB_shutdown_single | DBB_shutdown_full);
switch (flag & isc_dpb_shut_mode_mask) {
case isc_dpb_shut_multi:
dbb->dbb_ast_flags |= DBB_shutdown;
break;
case isc_dpb_shut_single:
dbb->dbb_ast_flags |= DBB_shutdown | DBB_shutdown_single;
break;
case isc_dpb_shut_full:
dbb->dbb_ast_flags |= DBB_shutdown | DBB_shutdown_full;
break;
}
2001-05-23 15:26:42 +02:00
Attachment* 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 */
2004-05-15 02:58:46 +02:00
THREAD_EXIT();
2001-05-23 15:26:42 +02:00
THREAD_SLEEP(1 * 1000);
2004-05-15 02:58:46 +02:00
THREAD_ENTER();
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. */
const Attachment* shut_attachment = NULL;
2001-05-23 15:26:42 +02:00
for (attachment = dbb->dbb_attachments; attachment;
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);
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. */
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_instance_lock)
LCK_release(tdbb, dbb->dbb_instance_lock);
if (dbb->dbb_monitor_lock)
LCK_release(tdbb, dbb->dbb_monitor_lock);
if (dbb->dbb_increment_lock)
LCK_release(tdbb, dbb->dbb_increment_lock);
2001-05-23 15:26:42 +02:00
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);
dbb->dbb_backup_manager->shutdown_locks(tdbb);
2001-05-23 15:26:42 +02:00
dbb->dbb_ast_flags |= DBB_shutdown_locks;
}
return true;
2001-05-23 15:26:42 +02:00
}