2003-08-06 18:30:49 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: JRD Access Method
|
|
|
|
* MODULE: nbak.cpp
|
2009-03-17 08:39:55 +01:00
|
|
|
* DESCRIPTION: Incremental backup technology
|
2003-08-06 18:30:49 +02:00
|
|
|
*
|
2004-06-30 03:45:18 +02:00
|
|
|
* The contents of this file are subject to the Initial
|
|
|
|
* Developer's 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.ibphoenix.com/main.nfs?a=ibphoenix&page=ibp_idpl.
|
2003-08-06 18:30:49 +02:00
|
|
|
*
|
2004-06-30 03:45:18 +02:00
|
|
|
* Software distributed under the License is distributed AS IS,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing rights
|
|
|
|
* and limitations under the License.
|
2003-08-06 18:30:49 +02:00
|
|
|
*
|
2004-06-30 03:45:18 +02:00
|
|
|
* The Original Code was created by Nickolay Samofatov
|
|
|
|
* for the Firebird Open Source RDBMS project.
|
2003-08-06 18:30:49 +02:00
|
|
|
*
|
2004-06-30 03:45:18 +02:00
|
|
|
* Copyright (c) 2004 Nickolay Samofatov <nickolay@broadviewsoftware.com>
|
|
|
|
* and all contributors signed below.
|
|
|
|
*
|
|
|
|
* All Rights Reserved.
|
2009-03-17 08:39:55 +01:00
|
|
|
* Contributor(s):
|
2004-06-30 03:45:18 +02:00
|
|
|
*
|
2009-03-17 08:39:55 +01:00
|
|
|
* Roman Simakov <roman-simakov@users.sourceforge.net>
|
2009-09-29 11:24:40 +02:00
|
|
|
* Khorsun Vladyslav <hvlad@users.sourceforge.net>
|
2004-06-30 03:45:18 +02:00
|
|
|
*
|
2003-08-06 18:30:49 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "firebird.h"
|
2004-03-22 12:38:23 +01:00
|
|
|
#include "common.h"
|
2003-08-06 18:30:49 +02:00
|
|
|
#include "jrd.h"
|
2005-11-22 00:33:20 +01:00
|
|
|
#include "nbak.h"
|
2003-08-06 18:30:49 +02:00
|
|
|
#include "ods.h"
|
|
|
|
#include "lck.h"
|
|
|
|
#include "cch.h"
|
|
|
|
#include "lck_proto.h"
|
|
|
|
#include "pag_proto.h"
|
|
|
|
#include "err_proto.h"
|
|
|
|
#include "cch_proto.h"
|
|
|
|
#include "isc_proto.h"
|
2004-05-18 00:30:09 +02:00
|
|
|
#include "../jrd/thread_proto.h"
|
2003-08-06 18:30:49 +02:00
|
|
|
#include "os/pio_proto.h"
|
2003-11-11 13:19:20 +01:00
|
|
|
#include "gen/iberror.h"
|
2003-08-06 18:30:49 +02:00
|
|
|
#include "gds_proto.h"
|
|
|
|
#include "os/guid.h"
|
2003-09-08 22:23:46 +02:00
|
|
|
#include "os/isc_i_proto.h"
|
2003-08-06 18:30:49 +02:00
|
|
|
|
2006-04-09 12:59:43 +02:00
|
|
|
#ifdef HAVE_SYS_TYPES_H
|
|
|
|
#include <sys/types.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef HAVE_SYS_STAT_H
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#endif
|
|
|
|
|
2003-08-06 22:22:52 +02:00
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
|
|
|
|
2006-04-09 12:59:43 +02:00
|
|
|
#ifdef HAVE_ERRNO_H
|
|
|
|
#include <errno.h>
|
|
|
|
#endif
|
|
|
|
|
2003-09-08 22:23:46 +02:00
|
|
|
#ifdef NBAK_DEBUG
|
2008-08-27 14:20:47 +02:00
|
|
|
#include <stdarg.h>
|
2004-06-14 01:47:02 +02:00
|
|
|
IMPLEMENT_TRACE_ROUTINE(nbak_trace, "NBAK")
|
2003-09-08 22:23:46 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-03-20 15:57:40 +01:00
|
|
|
using namespace Jrd;
|
2008-08-27 14:20:47 +02:00
|
|
|
using namespace Firebird;
|
2004-03-20 15:57:40 +01:00
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
|
|
|
|
/******************************** NBackupStateLock ******************************/
|
|
|
|
|
2009-03-18 03:43:15 +01:00
|
|
|
NBackupStateLock::NBackupStateLock(thread_db* tdbb, MemoryPool& p, BackupManager* bakMan):
|
2009-03-17 08:39:55 +01:00
|
|
|
GlobalRWLock(tdbb, p, LCK_backup_database, LCK_OWNER_database), backup_manager(bakMan)
|
2004-03-18 06:56:06 +01:00
|
|
|
{
|
2003-09-08 22:23:46 +02:00
|
|
|
}
|
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
bool NBackupStateLock::fetch(thread_db* tdbb)
|
2004-03-18 06:56:06 +01:00
|
|
|
{
|
2009-03-17 08:39:55 +01:00
|
|
|
backup_manager->endFlush();
|
|
|
|
NBAK_TRACE( ("backup_manager->endFlush()") );
|
|
|
|
if (!backup_manager->actualizeState(tdbb))
|
2007-04-24 16:05:46 +02:00
|
|
|
{
|
2009-03-17 08:39:55 +01:00
|
|
|
ERR_bugcheck_msg("Can't actualize backup state");
|
2003-09-08 22:23:46 +02:00
|
|
|
}
|
2009-03-17 08:39:55 +01:00
|
|
|
return true;
|
2003-09-08 22:23:46 +02:00
|
|
|
}
|
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
void NBackupStateLock::invalidate(thread_db* tdbb)
|
2004-03-18 06:56:06 +01:00
|
|
|
{
|
2009-03-17 08:39:55 +01:00
|
|
|
GlobalRWLock::invalidate(tdbb);
|
|
|
|
NBAK_TRACE( ("invalidate state stateLock(%p)", this) );
|
|
|
|
backup_manager->setState(nbak_state_unknown);
|
|
|
|
backup_manager->closeDelta();
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
void NBackupStateLock::blockingAstHandler(thread_db* tdbb)
|
2004-03-18 06:56:06 +01:00
|
|
|
{
|
2009-09-29 10:13:48 +02:00
|
|
|
const bool wasWrite = (cachedLock->lck_physical == LCK_write);
|
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
if (!backup_manager->databaseFlushInProgress())
|
|
|
|
{
|
|
|
|
backup_manager->beginFlush();
|
|
|
|
NBAK_TRACE_AST( ("backup_manager->beginFlush()") );
|
2007-04-24 16:05:46 +02:00
|
|
|
CCH_flush_ast(tdbb);
|
|
|
|
NBAK_TRACE_AST(("database FLUSHED"));
|
2003-09-08 22:23:46 +02:00
|
|
|
}
|
2009-09-30 03:10:11 +02:00
|
|
|
|
2007-04-24 16:05:46 +02:00
|
|
|
GlobalRWLock::blockingAstHandler(tdbb);
|
2009-09-29 10:13:48 +02:00
|
|
|
|
|
|
|
if (wasWrite && (cachedLock->lck_physical == LCK_read))
|
|
|
|
backup_manager->endFlush();
|
2007-04-24 16:05:46 +02:00
|
|
|
}
|
2003-08-06 18:30:49 +02:00
|
|
|
|
2007-04-24 16:05:46 +02:00
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
/******************************** NBackupAllocLock ******************************/
|
2003-08-06 18:30:49 +02:00
|
|
|
|
2009-03-18 03:43:15 +01:00
|
|
|
NBackupAllocLock::NBackupAllocLock(thread_db* tdbb, MemoryPool& p, BackupManager* bakMan):
|
2009-03-17 08:39:55 +01:00
|
|
|
GlobalRWLock(tdbb, p, LCK_backup_alloc, LCK_OWNER_database), backup_manager(bakMan)
|
2004-03-18 06:56:06 +01:00
|
|
|
{
|
2003-09-08 22:23:46 +02:00
|
|
|
}
|
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
bool NBackupAllocLock::fetch(thread_db* tdbb)
|
2004-03-18 06:56:06 +01:00
|
|
|
{
|
2009-03-17 08:39:55 +01:00
|
|
|
if (!backup_manager->actualizeAlloc(tdbb))
|
|
|
|
ERR_bugcheck_msg("Can't actualize alloc table");
|
|
|
|
return true;
|
2003-09-08 22:23:46 +02:00
|
|
|
}
|
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
/******************************** BackupManager::StateWriteGuard ******************************/
|
2008-12-05 02:20:14 +01:00
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
BackupManager::StateWriteGuard::StateWriteGuard(thread_db* _tdbb, Jrd::WIN* wnd)
|
|
|
|
: tdbb(_tdbb), window(NULL)
|
2007-04-24 16:05:46 +02:00
|
|
|
{
|
2009-03-17 08:39:55 +01:00
|
|
|
Database* dbb = tdbb->getDatabase();
|
|
|
|
dbb->dbb_backup_manager->beginFlush();
|
|
|
|
CCH_flush(tdbb, FLUSH_ALL, 0); // Flush local cache to release all dirty pages
|
2007-04-24 16:05:46 +02:00
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
if (!tdbb->getAttachment()->backupStateWriteLock(tdbb, true))
|
|
|
|
ERR_bugcheck_msg("Can't lock state for write");
|
2003-09-08 22:23:46 +02:00
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
dbb->dbb_backup_manager->endFlush();
|
|
|
|
NBAK_TRACE(("backup state locked for write"));
|
|
|
|
CCH_FETCH(tdbb, wnd, LCK_write, pag_header);
|
|
|
|
window = wnd;
|
2003-09-08 22:23:46 +02:00
|
|
|
}
|
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
void BackupManager::StateWriteGuard::releaseHeader()
|
2003-09-08 22:23:46 +02:00
|
|
|
{
|
2009-03-17 08:39:55 +01:00
|
|
|
if (window)
|
|
|
|
{
|
|
|
|
CCH_RELEASE(tdbb, window);
|
|
|
|
window = NULL;
|
2003-09-08 22:23:46 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
/********************************** CORE LOGIC ********************************/
|
2003-09-08 22:23:46 +02:00
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
void BackupManager::generateFilename()
|
2003-09-08 22:23:46 +02:00
|
|
|
{
|
2009-03-17 08:39:55 +01:00
|
|
|
diff_name = database->dbb_filename + ".delta";
|
|
|
|
explicit_diff_name = false;
|
2003-09-08 22:23:46 +02:00
|
|
|
}
|
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
void BackupManager::openDelta()
|
2007-04-24 16:05:46 +02:00
|
|
|
{
|
2009-03-17 08:39:55 +01:00
|
|
|
fb_assert(!diff_file);
|
|
|
|
diff_file = PIO_open(database, diff_name, diff_name, false);
|
2009-09-30 03:10:11 +02:00
|
|
|
|
|
|
|
if (database->dbb_flags & (DBB_force_write | DBB_no_fs_cache))
|
|
|
|
{
|
|
|
|
PIO_force_write(diff_file,
|
|
|
|
database->dbb_flags & DBB_force_write,
|
2009-09-29 11:24:40 +02:00
|
|
|
database->dbb_flags & DBB_no_fs_cache);
|
|
|
|
}
|
2007-04-24 16:05:46 +02:00
|
|
|
}
|
2003-09-08 22:23:46 +02:00
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
void BackupManager::closeDelta()
|
2004-03-18 06:56:06 +01:00
|
|
|
{
|
2009-11-17 11:33:54 +01:00
|
|
|
if (diff_file)
|
|
|
|
{
|
2009-03-17 08:39:55 +01:00
|
|
|
PIO_close(diff_file);
|
|
|
|
diff_file = NULL;
|
|
|
|
}
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Initialize and open difference file for writing
|
2009-03-17 08:39:55 +01:00
|
|
|
void BackupManager::beginBackup(thread_db* tdbb)
|
2004-03-18 06:56:06 +01:00
|
|
|
{
|
2009-03-17 08:39:55 +01:00
|
|
|
NBAK_TRACE(("beginBackup"));
|
2003-08-10 17:43:23 +02:00
|
|
|
|
2008-12-14 11:19:27 +01:00
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2007-10-26 15:21:16 +02:00
|
|
|
// Check for raw device
|
|
|
|
if ((!explicit_diff_name) && database->onRawDevice()) {
|
2008-08-27 14:20:47 +02:00
|
|
|
ERR_post(Arg::Gds(isc_need_difference));
|
2007-10-26 15:21:16 +02:00
|
|
|
}
|
|
|
|
|
2006-05-22 00:07:35 +02:00
|
|
|
WIN window(HEADER_PAGE_NUMBER);
|
2007-04-24 16:05:46 +02:00
|
|
|
|
2010-03-14 16:27:04 +01:00
|
|
|
StateWriteGuard stateGuard(tdbb, &window);
|
|
|
|
Ods::header_page* header = (Ods::header_page*) window.win_buffer;
|
2003-08-10 17:43:23 +02:00
|
|
|
|
2010-03-14 16:27:04 +01:00
|
|
|
// Check state
|
|
|
|
if (backup_state != nbak_state_normal)
|
|
|
|
{
|
|
|
|
NBAK_TRACE(("end backup - invalid state %d", backup_state));
|
|
|
|
return;
|
|
|
|
}
|
2007-04-24 16:05:46 +02:00
|
|
|
|
2010-03-14 16:27:04 +01:00
|
|
|
try
|
|
|
|
{
|
2008-12-05 02:20:14 +01:00
|
|
|
// Create file
|
2007-04-24 16:05:46 +02:00
|
|
|
NBAK_TRACE(("Creating difference file %s", diff_name.c_str()));
|
|
|
|
diff_file = PIO_create(database, diff_name, true, false, false);
|
2010-03-14 16:27:04 +01:00
|
|
|
}
|
|
|
|
catch (const Firebird::Exception&)
|
|
|
|
{
|
2010-03-15 07:38:49 +01:00
|
|
|
// no reasons to set it to unknown if we just failed to create difference file
|
2010-03-14 16:27:04 +01:00
|
|
|
backup_state = nbak_state_normal;
|
|
|
|
throw;
|
|
|
|
}
|
2009-09-30 03:10:11 +02:00
|
|
|
|
2010-03-14 16:27:04 +01:00
|
|
|
try
|
|
|
|
{
|
2009-09-30 03:10:11 +02:00
|
|
|
if (database->dbb_flags & (DBB_force_write | DBB_no_fs_cache))
|
|
|
|
{
|
|
|
|
PIO_force_write(diff_file,
|
|
|
|
database->dbb_flags & DBB_force_write,
|
2009-09-29 11:24:40 +02:00
|
|
|
database->dbb_flags & DBB_no_fs_cache);
|
|
|
|
}
|
2009-09-30 03:10:11 +02:00
|
|
|
|
2008-12-05 02:20:14 +01:00
|
|
|
#ifdef UNIX
|
2006-04-09 12:59:43 +02:00
|
|
|
// adjust difference file access rights to make it match main DB ones
|
2009-01-20 09:33:59 +01:00
|
|
|
if (diff_file && geteuid() == 0)
|
|
|
|
{
|
2006-04-09 12:59:43 +02:00
|
|
|
struct stat st;
|
2009-11-17 11:33:54 +01:00
|
|
|
PageSpace* pageSpace = database->dbb_page_manager.findPageSpace(DB_PAGE_SPACE);
|
|
|
|
while (fstat(pageSpace->file->fil_desc, &st) != 0)
|
|
|
|
{
|
2006-04-09 12:59:43 +02:00
|
|
|
if (errno != EINTR) {
|
|
|
|
Firebird::system_call_failed::raise("fstat");
|
|
|
|
}
|
|
|
|
}
|
2009-11-17 11:33:54 +01:00
|
|
|
while (fchown(diff_file->fil_desc, st.st_uid, st.st_gid) != 0)
|
|
|
|
{
|
2006-04-09 12:59:43 +02:00
|
|
|
if (errno != EINTR) {
|
|
|
|
Firebird::system_call_failed::raise("fchown");
|
|
|
|
}
|
|
|
|
}
|
2009-11-17 11:33:54 +01:00
|
|
|
while (fchmod(diff_file->fil_desc, st.st_mode) != 0)
|
|
|
|
{
|
2006-04-09 12:59:43 +02:00
|
|
|
if (errno != EINTR) {
|
|
|
|
Firebird::system_call_failed::raise("fchmod");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2009-09-30 03:10:11 +02:00
|
|
|
|
2003-08-06 18:30:49 +02:00
|
|
|
// Zero out first page (empty allocation table)
|
2004-03-18 06:56:06 +01:00
|
|
|
BufferDesc temp_bdb;
|
2003-08-06 18:30:49 +02:00
|
|
|
temp_bdb.bdb_page = 0;
|
|
|
|
temp_bdb.bdb_dbb = database;
|
2004-03-20 15:57:40 +01:00
|
|
|
temp_bdb.bdb_buffer = reinterpret_cast<Ods::pag*>(alloc_buffer);
|
2003-08-06 18:30:49 +02:00
|
|
|
memset(alloc_buffer, 0, database->dbb_page_size);
|
2003-09-08 22:23:46 +02:00
|
|
|
if (!PIO_write(diff_file, &temp_bdb, temp_bdb.bdb_buffer, tdbb->tdbb_status_vector))
|
2003-08-06 18:30:49 +02:00
|
|
|
ERR_punt();
|
2003-09-08 22:23:46 +02:00
|
|
|
NBAK_TRACE(("Set backup state in header"));
|
2003-08-06 18:30:49 +02:00
|
|
|
FB_GUID guid;
|
|
|
|
GenerateGuid(&guid);
|
|
|
|
// Set state in database header page. All changes are written to main database file yet.
|
|
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
2009-04-01 12:49:54 +02:00
|
|
|
const int newState = nbak_state_stalled; // Should be USHORT?
|
2004-05-01 00:47:16 +02:00
|
|
|
header->hdr_flags = (header->hdr_flags & ~Ods::hdr_backup_mask) | newState;
|
2004-07-10 05:20:33 +02:00
|
|
|
const ULONG adjusted_scn = ++header->hdr_header.pag_scn; // Generate new SCN
|
2008-12-14 11:19:27 +01:00
|
|
|
PAG_replace_entry_first(tdbb, header, Ods::HDR_backup_guid, sizeof(guid),
|
2003-12-31 06:36:12 +01:00
|
|
|
reinterpret_cast<const UCHAR*>(&guid));
|
2003-08-10 17:43:23 +02:00
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
stateGuard.releaseHeader();
|
2008-03-20 14:08:10 +01:00
|
|
|
|
2003-08-06 18:30:49 +02:00
|
|
|
backup_state = newState;
|
|
|
|
current_scn = adjusted_scn;
|
2007-04-24 16:05:46 +02:00
|
|
|
|
2007-11-09 16:47:48 +01:00
|
|
|
// All changes go to the difference file now
|
2006-05-20 05:55:54 +02:00
|
|
|
}
|
2008-12-05 02:20:14 +01:00
|
|
|
catch (const Firebird::Exception&)
|
2008-04-16 10:46:13 +02:00
|
|
|
{
|
2009-03-17 08:39:55 +01:00
|
|
|
NBAK_TRACE( ("invalidate state E1") );
|
2003-08-06 18:30:49 +02:00
|
|
|
backup_state = nbak_state_unknown;
|
|
|
|
throw;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-11-09 16:47:48 +01:00
|
|
|
|
|
|
|
// Determine actual DB size (raw devices support)
|
|
|
|
ULONG BackupManager::getPageCount()
|
|
|
|
{
|
|
|
|
if (backup_state != nbak_state_stalled)
|
|
|
|
{
|
|
|
|
// calculate pages only when database is locked for backup:
|
|
|
|
// other case such service is just dangerous
|
|
|
|
return 0;
|
|
|
|
}
|
2008-03-20 14:08:10 +01:00
|
|
|
|
2007-11-12 18:24:22 +01:00
|
|
|
class PioCount : public Jrd::PageCountCallback
|
2007-11-09 16:47:48 +01:00
|
|
|
{
|
2007-11-12 18:24:22 +01:00
|
|
|
private:
|
|
|
|
BufferDesc temp_bdb;
|
|
|
|
PageSpace* pageSpace;
|
2007-11-09 16:47:48 +01:00
|
|
|
|
2007-11-12 18:24:22 +01:00
|
|
|
public:
|
2007-11-17 11:18:10 +01:00
|
|
|
explicit PioCount(Database* d)
|
2007-11-09 16:47:48 +01:00
|
|
|
{
|
2007-11-17 11:18:10 +01:00
|
|
|
fb_assert(d);
|
2007-11-12 18:24:22 +01:00
|
|
|
temp_bdb.bdb_dbb = d;
|
|
|
|
pageSpace = d->dbb_page_manager.findPageSpace(DB_PAGE_SPACE);
|
|
|
|
fb_assert(pageSpace);
|
2007-11-09 16:47:48 +01:00
|
|
|
}
|
2007-11-12 18:24:22 +01:00
|
|
|
virtual void newPage(const SLONG pageNum, Ods::pag* buf)
|
2007-11-09 16:47:48 +01:00
|
|
|
{
|
2007-11-12 18:24:22 +01:00
|
|
|
temp_bdb.bdb_buffer = buf;
|
|
|
|
temp_bdb.bdb_page = pageNum;
|
|
|
|
ISC_STATUS_ARRAY status;
|
2008-12-05 02:20:14 +01:00
|
|
|
if (!PIO_read(pageSpace->file, &temp_bdb, temp_bdb.bdb_buffer, status))
|
2007-11-12 18:24:22 +01:00
|
|
|
{
|
|
|
|
Firebird::status_exception::raise(status);
|
|
|
|
}
|
2007-11-09 16:47:48 +01:00
|
|
|
}
|
2007-11-12 18:24:22 +01:00
|
|
|
};
|
|
|
|
PioCount pioCount(database);
|
2008-03-20 14:08:10 +01:00
|
|
|
|
2007-11-12 18:24:22 +01:00
|
|
|
return PAG_page_count(database, &pioCount);
|
2007-11-09 16:47:48 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-03-18 03:43:15 +01:00
|
|
|
// Merge difference file to main files (if needed) and unlink() difference
|
|
|
|
// file then. If merge is already in progress method silently returns and
|
|
|
|
// does nothing (so it can be used for recovery on database startup).
|
|
|
|
void BackupManager::endBackup(thread_db* tdbb, bool recover)
|
2005-11-22 00:33:20 +01:00
|
|
|
{
|
2007-04-24 16:05:46 +02:00
|
|
|
NBAK_TRACE(("end_backup, recover=%i", recover));
|
|
|
|
|
|
|
|
// Check for recover
|
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
GlobalRWLock endLock(tdbb, *database->dbb_permanent, LCK_backup_end,
|
|
|
|
LCK_OWNER_attachment, false);
|
2007-04-24 16:05:46 +02:00
|
|
|
|
2009-11-17 11:33:54 +01:00
|
|
|
if (!endLock.lockWrite(tdbb, LCK_NO_WAIT))
|
|
|
|
{
|
2007-04-24 19:43:11 +02:00
|
|
|
// Someboby holds write lock on LCK_backup_end. We need not to do end_backup
|
2007-04-24 16:05:46 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// STEP 1. Change state in header to "merge"
|
|
|
|
WIN window(HEADER_PAGE_NUMBER);
|
|
|
|
Ods::header_page* header;
|
|
|
|
|
2003-08-06 18:30:49 +02:00
|
|
|
ULONG adjusted_scn; // We use this value to prevent race conditions.
|
|
|
|
// They are possible because we release state lock
|
|
|
|
// for some instants and anything is possible at
|
|
|
|
// that times.
|
2003-08-10 17:43:23 +02:00
|
|
|
|
2008-12-05 02:20:14 +01:00
|
|
|
try
|
2008-04-16 10:46:13 +02:00
|
|
|
{
|
2007-04-24 16:05:46 +02:00
|
|
|
// Check state under PR lock of backup state for speed
|
2008-04-16 10:46:13 +02:00
|
|
|
{ // scope
|
2009-03-17 08:39:55 +01:00
|
|
|
StateReadGuard stateGuard(tdbb);
|
2009-03-18 03:43:15 +01:00
|
|
|
// Nobody is doing end_backup but database isn't in merge state.
|
|
|
|
if ( (recover || backup_state != nbak_state_stalled) && (backup_state != nbak_state_merge ) )
|
2008-04-16 10:46:13 +02:00
|
|
|
{
|
|
|
|
NBAK_TRACE(("invalid state %d", backup_state));
|
2009-03-17 08:39:55 +01:00
|
|
|
endLock.unlockWrite(tdbb);
|
2008-04-16 10:46:13 +02:00
|
|
|
return;
|
|
|
|
}
|
2004-01-28 08:50:41 +01:00
|
|
|
}
|
2008-04-16 10:46:13 +02:00
|
|
|
|
2007-04-24 16:05:46 +02:00
|
|
|
// Here backup state can be changed. Need to check it again after lock
|
2009-03-17 08:39:55 +01:00
|
|
|
StateWriteGuard stateGuard(tdbb, &window);
|
2008-04-16 10:46:13 +02:00
|
|
|
|
2008-12-05 02:20:14 +01:00
|
|
|
if ( (recover || backup_state != nbak_state_stalled) && (backup_state != nbak_state_merge ) )
|
2008-04-16 10:46:13 +02:00
|
|
|
{
|
2003-09-08 22:23:46 +02:00
|
|
|
NBAK_TRACE(("invalid state %d", backup_state));
|
2009-03-17 08:39:55 +01:00
|
|
|
endLock.unlockWrite(tdbb);
|
2003-08-10 17:43:23 +02:00
|
|
|
return;
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
2007-04-24 16:05:46 +02:00
|
|
|
header = (Ods::header_page*) window.win_buffer;
|
|
|
|
|
|
|
|
NBAK_TRACE(("difference file %s, current backup state is %d", diff_name.c_str(), backup_state));
|
|
|
|
|
2003-08-06 18:30:49 +02:00
|
|
|
// Set state in database header
|
|
|
|
backup_state = nbak_state_merge;
|
|
|
|
adjusted_scn = ++current_scn;
|
2003-09-08 22:23:46 +02:00
|
|
|
NBAK_TRACE(("New state is getting to become %d", backup_state));
|
2003-08-06 18:30:49 +02:00
|
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
|
|
|
// Generate new SCN
|
2004-07-10 05:20:33 +02:00
|
|
|
header->hdr_header.pag_scn = current_scn;
|
2005-07-20 01:24:28 +02:00
|
|
|
NBAK_TRACE(("new SCN=%d is getting written to header", header->hdr_header.pag_scn));
|
2003-08-06 18:30:49 +02:00
|
|
|
// Adjust state
|
2004-05-01 00:47:16 +02:00
|
|
|
header->hdr_flags = (header->hdr_flags & ~Ods::hdr_backup_mask) | backup_state;
|
2007-04-24 16:05:46 +02:00
|
|
|
NBAK_TRACE(("Setting state %d in header page is over", backup_state));
|
2006-05-20 05:55:54 +02:00
|
|
|
}
|
2009-11-17 11:33:54 +01:00
|
|
|
catch (const Firebird::Exception&)
|
|
|
|
{
|
2009-03-17 08:39:55 +01:00
|
|
|
NBAK_TRACE( ("invalidate state E2") );
|
2003-08-06 18:30:49 +02:00
|
|
|
backup_state = nbak_state_unknown;
|
2009-03-17 08:39:55 +01:00
|
|
|
endLock.unlockWrite(tdbb);
|
2003-08-06 18:30:49 +02:00
|
|
|
throw;
|
2003-08-10 17:43:23 +02:00
|
|
|
}
|
2007-02-08 15:32:07 +01:00
|
|
|
|
2007-04-24 16:05:46 +02:00
|
|
|
// STEP 2. Merging database and delta
|
2003-08-06 18:30:49 +02:00
|
|
|
// Here comes the dirty work. We need to reapply all changes from difference file to database
|
2008-12-05 02:20:14 +01:00
|
|
|
// Release write state lock and get read lock.
|
2003-08-06 18:30:49 +02:00
|
|
|
// Merge process should not inhibit normal operations.
|
2007-04-24 16:05:46 +02:00
|
|
|
|
2008-12-05 02:20:14 +01:00
|
|
|
try
|
2008-04-16 10:46:13 +02:00
|
|
|
{
|
|
|
|
NBAK_TRACE(("database locked to merge"));
|
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
StateReadGuard stateGuard(tdbb);
|
2008-04-16 10:46:13 +02:00
|
|
|
|
2003-09-08 22:23:46 +02:00
|
|
|
NBAK_TRACE(("Merge. State=%d, current_scn=%d, adjusted_scn=%d",
|
2008-12-05 02:20:14 +01:00
|
|
|
backup_state, current_scn, adjusted_scn));
|
2007-02-08 15:32:07 +01:00
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
actualizeAlloc(tdbb);
|
2008-12-05 02:20:14 +01:00
|
|
|
NBAK_TRACE(("Merge. Alloc table is actualized."));
|
2007-04-24 16:05:46 +02:00
|
|
|
AllocItemTree::Accessor all(alloc_table);
|
2008-03-20 14:08:10 +01:00
|
|
|
|
2009-11-17 11:33:54 +01:00
|
|
|
if (all.getFirst())
|
|
|
|
{
|
2003-12-11 11:33:30 +01:00
|
|
|
do {
|
2007-04-24 16:05:46 +02:00
|
|
|
WIN window2(DB_PAGE_SPACE, all.current().db_page);
|
2005-07-20 01:24:28 +02:00
|
|
|
NBAK_TRACE(("Merge page %d, diff=%d", all.current().db_page, all.current().diff_page));
|
2009-03-17 08:39:55 +01:00
|
|
|
Ods::pag* page = CCH_FETCH_MERGE(tdbb, &window2, LCK_write, pag_undefined);
|
2007-04-24 16:05:46 +02:00
|
|
|
NBAK_TRACE(("Merge: page %d is fetched", all.current().db_page));
|
2009-11-17 11:33:54 +01:00
|
|
|
if (page->pag_scn != current_scn)
|
|
|
|
{
|
2004-05-12 21:39:17 +02:00
|
|
|
CCH_MARK(tdbb, &window2);
|
2007-04-24 16:05:46 +02:00
|
|
|
NBAK_TRACE(("Merge: page %d is marked", all.current().db_page));
|
|
|
|
}
|
2004-05-12 21:39:17 +02:00
|
|
|
CCH_RELEASE(tdbb, &window2);
|
2007-04-24 16:05:46 +02:00
|
|
|
NBAK_TRACE(("Merge: page %d is released", all.current().db_page));
|
2004-02-02 12:02:12 +01:00
|
|
|
} while (all.getNext());
|
2003-12-11 11:33:30 +01:00
|
|
|
}
|
2007-02-08 15:32:07 +01:00
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
CCH_flush(tdbb, FLUSH_ALL, 0);
|
2007-04-24 16:05:46 +02:00
|
|
|
NBAK_TRACE(("Merging is over. Database unlocked"));
|
2006-05-20 05:55:54 +02:00
|
|
|
}
|
2009-11-17 11:33:54 +01:00
|
|
|
catch (const Firebird::Exception&)
|
|
|
|
{
|
2009-03-17 08:39:55 +01:00
|
|
|
endLock.unlockWrite(tdbb);
|
2003-08-06 18:30:49 +02:00
|
|
|
throw;
|
|
|
|
}
|
2007-02-08 15:32:07 +01:00
|
|
|
|
2007-04-24 16:05:46 +02:00
|
|
|
// STEP 3. Change state in header to "normal"
|
2003-08-07 23:48:54 +02:00
|
|
|
// We finished. We need to reflect it in our database header page
|
2003-08-06 18:30:49 +02:00
|
|
|
try {
|
2007-04-24 16:05:46 +02:00
|
|
|
window.win_page = HEADER_PAGE;
|
|
|
|
window.win_flags = 0;
|
2008-04-16 10:46:13 +02:00
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
StateWriteGuard stateGuard(tdbb, &window);
|
2008-04-16 10:46:13 +02:00
|
|
|
|
2007-04-24 16:05:46 +02:00
|
|
|
header = (Ods::header_page*) window.win_buffer;
|
|
|
|
|
2003-08-06 18:30:49 +02:00
|
|
|
// Set state in database header
|
|
|
|
backup_state = nbak_state_normal;
|
|
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
|
|
|
// Adjust state
|
2004-05-01 00:47:16 +02:00
|
|
|
header->hdr_flags = (header->hdr_flags & ~Ods::hdr_backup_mask) | backup_state;
|
2003-09-08 22:23:46 +02:00
|
|
|
NBAK_TRACE(("Set state %d in header page", backup_state));
|
|
|
|
// Generate new SCN
|
2004-07-10 05:20:33 +02:00
|
|
|
header->hdr_header.pag_scn = ++current_scn;
|
2005-07-20 01:24:28 +02:00
|
|
|
NBAK_TRACE(("new SCN=%d is getting written to header", header->hdr_header.pag_scn));
|
2009-03-17 08:39:55 +01:00
|
|
|
|
|
|
|
stateGuard.releaseHeader();
|
2008-03-20 14:08:10 +01:00
|
|
|
|
2003-08-06 18:30:49 +02:00
|
|
|
// Page allocation table cache is no longer valid
|
2007-04-24 16:05:46 +02:00
|
|
|
NBAK_TRACE(("Dropping alloc table"));
|
2003-08-06 18:30:49 +02:00
|
|
|
delete alloc_table;
|
|
|
|
alloc_table = NULL;
|
|
|
|
last_allocated_page = 0;
|
2009-03-17 08:39:55 +01:00
|
|
|
if (!allocLock->tryReleaseLock(tdbb))
|
2007-04-24 16:05:46 +02:00
|
|
|
ERR_bugcheck_msg("There are holders of alloc_lock after end_backup finish");
|
2008-03-20 14:08:10 +01:00
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
closeDelta();
|
2005-11-22 00:33:20 +01:00
|
|
|
unlink(diff_name.c_str());
|
2008-03-20 14:08:10 +01:00
|
|
|
|
2007-04-24 16:05:46 +02:00
|
|
|
NBAK_TRACE(("backup is over"));
|
2009-03-17 08:39:55 +01:00
|
|
|
endLock.unlockWrite(tdbb);
|
2006-05-20 05:55:54 +02:00
|
|
|
}
|
2009-11-17 11:33:54 +01:00
|
|
|
catch (const Firebird::Exception&)
|
|
|
|
{
|
2009-03-17 08:39:55 +01:00
|
|
|
NBAK_TRACE( ("invalidate state E3") );
|
2003-08-06 18:30:49 +02:00
|
|
|
backup_state = nbak_state_unknown;
|
2009-03-17 08:39:55 +01:00
|
|
|
endLock.unlockWrite(tdbb);
|
2003-08-06 18:30:49 +02:00
|
|
|
throw;
|
|
|
|
}
|
2007-04-24 16:05:46 +02:00
|
|
|
|
2003-08-10 17:43:23 +02:00
|
|
|
return;
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
2008-12-05 02:20:14 +01:00
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
bool BackupManager::actualizeAlloc(thread_db* tdbb)
|
2004-03-18 06:56:06 +01:00
|
|
|
{
|
2005-11-22 00:33:20 +01:00
|
|
|
ISC_STATUS *status_vector = tdbb->tdbb_status_vector;
|
2003-09-08 22:23:46 +02:00
|
|
|
try {
|
2008-12-05 02:20:14 +01:00
|
|
|
NBAK_TRACE(("actualize_alloc last_allocated_page=%d alloc_table=%p",
|
2003-09-08 22:23:46 +02:00
|
|
|
last_allocated_page, alloc_table));
|
2008-12-05 02:20:14 +01:00
|
|
|
// For SuperServer this routine is really executed only at database startup when
|
2003-09-08 22:23:46 +02:00
|
|
|
// it has exlock or when exclusive access to database is enabled
|
|
|
|
if (!alloc_table)
|
|
|
|
alloc_table = FB_NEW(*database->dbb_permanent) AllocItemTree(database->dbb_permanent);
|
2009-01-20 09:33:59 +01:00
|
|
|
while (true)
|
|
|
|
{
|
2004-03-18 06:56:06 +01:00
|
|
|
BufferDesc temp_bdb;
|
2003-09-08 22:23:46 +02:00
|
|
|
// Difference file pointer pages have one ULONG as number of pages allocated on the page and
|
|
|
|
// then go physical numbers of pages from main database file. Offsets of numbers correspond
|
|
|
|
// to difference file pages.
|
2008-03-20 14:08:10 +01:00
|
|
|
|
2003-09-08 22:23:46 +02:00
|
|
|
// Get offset of pointer page. We can do so because page sizes are powers of 2
|
2004-10-08 10:13:22 +02:00
|
|
|
temp_bdb.bdb_page = last_allocated_page & ~(database->dbb_page_size / sizeof(ULONG) - 1);
|
2003-09-08 22:23:46 +02:00
|
|
|
temp_bdb.bdb_dbb = database;
|
2004-03-20 15:57:40 +01:00
|
|
|
temp_bdb.bdb_buffer = reinterpret_cast<Ods::pag*>(alloc_buffer);
|
2008-03-20 14:08:10 +01:00
|
|
|
|
2007-04-24 16:05:46 +02:00
|
|
|
if (!PIO_read(diff_file, &temp_bdb, temp_bdb.bdb_buffer, status_vector)) {
|
2003-09-08 22:23:46 +02:00
|
|
|
return false;
|
2007-04-24 16:05:46 +02:00
|
|
|
}
|
|
|
|
for (ULONG i = last_allocated_page - temp_bdb.bdb_page.getPageNum(); i < alloc_buffer[0]; i++)
|
2004-03-18 06:56:06 +01:00
|
|
|
{
|
2007-04-24 16:05:46 +02:00
|
|
|
NBAK_TRACE(("alloc item page=%d, diff=%d", alloc_buffer[i + 1], temp_bdb.bdb_page.getPageNum() + i + 1));
|
|
|
|
if (!alloc_table->add(AllocItem(alloc_buffer[i + 1], temp_bdb.bdb_page.getPageNum() + i + 1)))
|
2004-02-20 07:43:27 +01:00
|
|
|
{
|
2003-09-08 22:23:46 +02:00
|
|
|
database->dbb_flags |= DBB_bugcheck;
|
2008-12-05 02:20:14 +01:00
|
|
|
ERR_build_status(status_vector,
|
2008-08-27 14:20:47 +02:00
|
|
|
Arg::Gds(isc_bug_check) << Arg::Str("Duplicated item in allocation table detected"));
|
2003-09-08 22:23:46 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2006-05-22 00:07:35 +02:00
|
|
|
last_allocated_page = temp_bdb.bdb_page.getPageNum() + alloc_buffer[0];
|
2004-03-18 06:56:06 +01:00
|
|
|
if (alloc_buffer[0] == database->dbb_page_size / sizeof(ULONG) - 1)
|
2008-08-31 03:10:41 +02:00
|
|
|
last_allocated_page++; // if page is full adjust position for next pointer page
|
2003-09-08 22:23:46 +02:00
|
|
|
else
|
2008-08-31 03:10:41 +02:00
|
|
|
break; // We finished reading allocation table
|
2003-09-08 22:23:46 +02:00
|
|
|
}
|
2006-05-20 05:55:54 +02:00
|
|
|
}
|
2009-11-17 11:33:54 +01:00
|
|
|
catch (const Firebird::Exception& ex)
|
|
|
|
{
|
2004-03-01 04:35:23 +01:00
|
|
|
// Handle out of memory error, etc
|
2003-08-06 18:30:49 +02:00
|
|
|
delete alloc_table;
|
2004-03-01 04:35:23 +01:00
|
|
|
Firebird::stuff_exception(status_vector, ex);
|
2003-08-06 18:30:49 +02:00
|
|
|
alloc_table = NULL;
|
|
|
|
last_allocated_page = 0;
|
2007-04-24 16:05:46 +02:00
|
|
|
return false;
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2008-12-05 02:20:14 +01:00
|
|
|
// Return page index in difference file that can be used in
|
2009-03-17 08:39:55 +01:00
|
|
|
// writeDifference call later.
|
2010-09-24 11:53:51 +02:00
|
|
|
ULONG BackupManager::getPageIndex(thread_db* /*tdbb*/, ULONG db_page)
|
2004-03-18 06:56:06 +01:00
|
|
|
{
|
2007-04-24 16:05:46 +02:00
|
|
|
NBAK_TRACE(("get_page_index"));
|
|
|
|
|
2003-08-06 18:30:49 +02:00
|
|
|
AllocItemTree::Accessor a(alloc_table);
|
2007-04-24 16:05:46 +02:00
|
|
|
ULONG diff_page = a.locate(db_page) ? a.current().diff_page : 0;
|
|
|
|
|
|
|
|
return diff_page;
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Mark next difference page as used by some database page
|
2009-03-17 08:39:55 +01:00
|
|
|
ULONG BackupManager::allocateDifferencePage(thread_db* tdbb, ULONG db_page)
|
2005-11-22 00:33:20 +01:00
|
|
|
{
|
2007-04-24 16:05:46 +02:00
|
|
|
// This page may be allocated by other
|
2009-03-17 08:39:55 +01:00
|
|
|
if (ULONG diff_page = getPageIndex(tdbb, db_page)) {
|
2007-04-24 16:05:46 +02:00
|
|
|
return diff_page;
|
|
|
|
}
|
|
|
|
|
|
|
|
NBAK_TRACE(("allocate_difference_page"));
|
2004-03-18 06:56:06 +01:00
|
|
|
fb_assert(last_allocated_page % (database->dbb_page_size / sizeof(ULONG)) == alloc_buffer[0]);
|
2003-09-08 22:23:46 +02:00
|
|
|
|
2005-11-22 00:33:20 +01:00
|
|
|
ISC_STATUS* status_vector = tdbb->tdbb_status_vector;
|
2003-08-06 18:30:49 +02:00
|
|
|
// Grow file first. This is done in such order to keep difference
|
2008-12-05 02:20:14 +01:00
|
|
|
// file consistent in case of write error. We should always be able
|
2003-08-06 18:30:49 +02:00
|
|
|
// to read next alloc page when previous one is full.
|
2004-03-18 06:56:06 +01:00
|
|
|
BufferDesc temp_bdb;
|
2004-10-08 10:13:22 +02:00
|
|
|
temp_bdb.bdb_page = last_allocated_page + 1;
|
2003-09-08 22:23:46 +02:00
|
|
|
temp_bdb.bdb_dbb = database;
|
2004-03-20 15:57:40 +01:00
|
|
|
temp_bdb.bdb_buffer = reinterpret_cast<Ods::pag*>(empty_buffer);
|
2007-04-24 16:05:46 +02:00
|
|
|
if (!PIO_write(diff_file, &temp_bdb, (Ods::pag*)empty_buffer, status_vector)) {
|
2003-09-08 22:23:46 +02:00
|
|
|
return 0;
|
2007-04-24 16:05:46 +02:00
|
|
|
}
|
2008-03-20 14:08:10 +01:00
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
const bool alloc_page_full = alloc_buffer[0] == database->dbb_page_size / sizeof(ULONG) - 2;
|
2009-11-17 11:33:54 +01:00
|
|
|
if (alloc_page_full)
|
|
|
|
{
|
2003-08-06 18:30:49 +02:00
|
|
|
// Pointer page is full. Its time to create new one.
|
2004-03-18 06:56:06 +01:00
|
|
|
temp_bdb.bdb_page = last_allocated_page + 2;
|
2003-08-06 18:30:49 +02:00
|
|
|
temp_bdb.bdb_dbb = database;
|
2004-03-20 15:57:40 +01:00
|
|
|
temp_bdb.bdb_buffer = reinterpret_cast<Ods::pag*>(empty_buffer);
|
2007-04-24 16:05:46 +02:00
|
|
|
if (!PIO_write(diff_file, &temp_bdb, (Ods::pag*)empty_buffer, status_vector)) {
|
2003-09-08 22:23:46 +02:00
|
|
|
return 0;
|
2007-04-24 16:05:46 +02:00
|
|
|
}
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// Write new item to the allocation table
|
2004-03-18 06:56:06 +01:00
|
|
|
temp_bdb.bdb_page = last_allocated_page & ~(database->dbb_page_size / sizeof(ULONG) - 1);
|
2003-08-06 18:30:49 +02:00
|
|
|
temp_bdb.bdb_dbb = database;
|
2004-03-20 15:57:40 +01:00
|
|
|
temp_bdb.bdb_buffer = reinterpret_cast<Ods::pag*>(alloc_buffer);
|
2003-08-06 18:30:49 +02:00
|
|
|
alloc_buffer[++alloc_buffer[0]] = db_page;
|
2007-04-24 16:05:46 +02:00
|
|
|
if (!PIO_write(diff_file, &temp_bdb, temp_bdb.bdb_buffer, status_vector)) {
|
2003-09-08 22:23:46 +02:00
|
|
|
return 0;
|
2007-04-24 16:05:46 +02:00
|
|
|
}
|
2003-08-06 18:30:49 +02:00
|
|
|
last_allocated_page++;
|
|
|
|
// Register new page in the alloc table
|
2003-09-08 22:23:46 +02:00
|
|
|
try {
|
|
|
|
alloc_table->add(AllocItem(db_page, last_allocated_page));
|
2006-05-20 05:55:54 +02:00
|
|
|
}
|
2009-11-17 11:33:54 +01:00
|
|
|
catch (const Firebird::Exception& ex)
|
|
|
|
{
|
2003-09-08 22:23:46 +02:00
|
|
|
// Handle out of memory error
|
|
|
|
delete alloc_table;
|
|
|
|
alloc_table = NULL;
|
|
|
|
last_allocated_page = 0;
|
2004-03-01 04:35:23 +01:00
|
|
|
Firebird::stuff_exception(status_vector, ex);
|
2003-09-08 22:23:46 +02:00
|
|
|
return 0;
|
|
|
|
}
|
2003-08-06 18:30:49 +02:00
|
|
|
// Adjust buffer and counters if we allocated new alloc page earlier
|
2009-11-17 11:33:54 +01:00
|
|
|
if (alloc_page_full)
|
|
|
|
{
|
2003-08-06 18:30:49 +02:00
|
|
|
last_allocated_page++;
|
2003-09-08 22:23:46 +02:00
|
|
|
memset(alloc_buffer, 0, database->dbb_page_size);
|
2004-10-08 10:13:22 +02:00
|
|
|
return last_allocated_page - 1;
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
2008-03-20 14:08:10 +01:00
|
|
|
|
2003-09-08 22:23:46 +02:00
|
|
|
return last_allocated_page;
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
bool BackupManager::writeDifference(ISC_STATUS* status, ULONG diff_page, Ods::pag* page)
|
2003-09-08 22:23:46 +02:00
|
|
|
{
|
2007-04-24 16:05:46 +02:00
|
|
|
NBAK_TRACE(("write_diff"));
|
2004-03-18 06:56:06 +01:00
|
|
|
BufferDesc temp_bdb;
|
2003-08-06 18:30:49 +02:00
|
|
|
temp_bdb.bdb_page = diff_page;
|
|
|
|
temp_bdb.bdb_dbb = database;
|
|
|
|
temp_bdb.bdb_buffer = page;
|
2008-04-16 11:16:45 +02:00
|
|
|
// Check that diff page is not allocation page
|
2008-04-18 03:37:44 +02:00
|
|
|
fb_assert(diff_page % (database->dbb_page_size / sizeof(ULONG)));
|
2003-08-06 18:30:49 +02:00
|
|
|
if (!PIO_write(diff_file, &temp_bdb, page, status))
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
2008-12-05 02:20:14 +01:00
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
bool BackupManager::readDifference(thread_db* tdbb, ULONG diff_page, Ods::pag* page)
|
2003-09-08 22:23:46 +02:00
|
|
|
{
|
2007-04-24 16:05:46 +02:00
|
|
|
NBAK_TRACE(("read_diff"));
|
2004-03-18 06:56:06 +01:00
|
|
|
BufferDesc temp_bdb;
|
2003-08-06 18:30:49 +02:00
|
|
|
temp_bdb.bdb_page = diff_page;
|
|
|
|
temp_bdb.bdb_dbb = database;
|
|
|
|
temp_bdb.bdb_buffer = page;
|
2008-12-05 02:20:14 +01:00
|
|
|
if (!PIO_read(diff_file, &temp_bdb, page, tdbb->tdbb_status_vector))
|
2003-08-06 18:30:49 +02:00
|
|
|
return false;
|
2008-12-05 02:20:14 +01:00
|
|
|
return true;
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
2008-12-05 02:20:14 +01:00
|
|
|
|
2005-11-22 00:33:20 +01:00
|
|
|
BackupManager::BackupManager(thread_db* tdbb, Database* _database, int ini_state) :
|
2007-07-26 09:58:48 +02:00
|
|
|
dbCreating(false), database(_database), diff_file(NULL), alloc_table(NULL),
|
2007-10-26 15:21:16 +02:00
|
|
|
last_allocated_page(0), current_scn(0), diff_name(*_database->dbb_permanent),
|
2009-03-17 08:39:55 +01:00
|
|
|
explicit_diff_name(false), flushInProgress(false),
|
|
|
|
stateLock(FB_NEW(*database->dbb_permanent) NBackupStateLock(tdbb, *database->dbb_permanent, this)),
|
|
|
|
allocLock(FB_NEW(*database->dbb_permanent) NBackupAllocLock(tdbb, *database->dbb_permanent, this))
|
2003-08-06 18:30:49 +02:00
|
|
|
{
|
2003-09-08 22:23:46 +02:00
|
|
|
// Allocate various database page buffers needed for operation
|
2007-11-12 18:24:22 +01:00
|
|
|
temp_buffers_space = FB_NEW(*database->dbb_permanent) BYTE[database->dbb_page_size * 3 + MIN_PAGE_SIZE];
|
2007-10-05 16:37:33 +02:00
|
|
|
// Align it at sector boundary for faster IO (also guarantees correct alignment for ULONG later)
|
2004-03-13 02:57:52 +01:00
|
|
|
BYTE *temp_buffers = reinterpret_cast<BYTE*>(
|
2004-12-25 10:44:03 +01:00
|
|
|
FB_ALIGN(reinterpret_cast<U_IPTR>(temp_buffers_space), MIN_PAGE_SIZE));
|
2004-03-13 02:57:52 +01:00
|
|
|
memset(temp_buffers, 0, database->dbb_page_size * 3);
|
2008-03-20 14:08:10 +01:00
|
|
|
|
2007-04-24 16:05:46 +02:00
|
|
|
backup_state = ini_state;
|
2007-02-08 15:32:07 +01:00
|
|
|
|
2004-03-13 02:57:52 +01:00
|
|
|
empty_buffer = reinterpret_cast<ULONG*>(temp_buffers);
|
|
|
|
spare_buffer = reinterpret_cast<ULONG*>(temp_buffers + database->dbb_page_size);
|
2007-02-08 15:32:07 +01:00
|
|
|
alloc_buffer = reinterpret_cast<ULONG*>(temp_buffers + database->dbb_page_size * 2);
|
2003-09-08 22:23:46 +02:00
|
|
|
|
2007-04-24 16:05:46 +02:00
|
|
|
NBAK_TRACE(("Create BackupManager, database=%s", database->dbb_filename.c_str()));
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
|
|
|
|
2004-03-18 06:56:06 +01:00
|
|
|
BackupManager::~BackupManager()
|
|
|
|
{
|
2009-03-17 08:39:55 +01:00
|
|
|
delete stateLock;
|
|
|
|
delete allocLock;
|
2004-03-12 21:19:41 +01:00
|
|
|
delete alloc_table;
|
|
|
|
delete[] temp_buffers_space;
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
void BackupManager::setDifference(thread_db* tdbb, const char* filename)
|
2005-11-22 00:33:20 +01:00
|
|
|
{
|
2008-12-14 11:19:27 +01:00
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2009-11-17 11:33:54 +01:00
|
|
|
if (filename)
|
|
|
|
{
|
2006-05-22 00:07:35 +02:00
|
|
|
WIN window(HEADER_PAGE_NUMBER);
|
2004-03-20 15:57:40 +01:00
|
|
|
Ods::header_page* header =
|
|
|
|
(Ods::header_page*) CCH_FETCH(tdbb, &window, LCK_write, pag_header);
|
2003-08-06 18:30:49 +02:00
|
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
2008-12-14 11:19:27 +01:00
|
|
|
PAG_replace_entry_first(tdbb, header, Ods::HDR_difference_file,
|
2003-12-31 06:36:12 +01:00
|
|
|
strlen(filename), reinterpret_cast<const UCHAR*>(filename));
|
2003-08-06 18:30:49 +02:00
|
|
|
CCH_RELEASE(tdbb, &window);
|
2005-11-22 00:33:20 +01:00
|
|
|
diff_name = filename;
|
2007-10-26 15:21:16 +02:00
|
|
|
explicit_diff_name = true;
|
2004-01-28 08:50:41 +01:00
|
|
|
}
|
2009-11-17 11:33:54 +01:00
|
|
|
else
|
|
|
|
{
|
2009-11-28 09:56:39 +01:00
|
|
|
PAG_delete_clump_entry(tdbb, Ods::HDR_difference_file);
|
2009-03-17 08:39:55 +01:00
|
|
|
generateFilename();
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-03-17 08:39:55 +01:00
|
|
|
bool BackupManager::actualizeState(thread_db* tdbb)
|
2005-11-22 00:33:20 +01:00
|
|
|
{
|
2003-09-08 22:23:46 +02:00
|
|
|
// State is unknown. We need to read it from the disk.
|
|
|
|
// We cannot use CCH for this because of likely recursion.
|
2009-03-17 08:39:55 +01:00
|
|
|
NBAK_TRACE(("actualizeState: current_state=%i", backup_state));
|
2007-02-08 15:32:07 +01:00
|
|
|
|
2009-11-17 11:33:54 +01:00
|
|
|
if (dbCreating)
|
|
|
|
{
|
2007-04-24 16:05:46 +02:00
|
|
|
backup_state = nbak_state_normal;
|
|
|
|
return true;
|
|
|
|
}
|
2007-02-08 15:32:07 +01:00
|
|
|
|
2008-12-14 10:28:25 +01:00
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
2007-04-24 16:05:46 +02:00
|
|
|
ISC_STATUS *status = tdbb->tdbb_status_vector;
|
2008-03-20 14:08:10 +01:00
|
|
|
|
2003-09-08 22:23:46 +02:00
|
|
|
// Read original page from database file or shadows.
|
|
|
|
SSHORT retryCount = 0;
|
2004-03-20 15:57:40 +01:00
|
|
|
Ods::header_page* header = reinterpret_cast<Ods::header_page*>(spare_buffer);
|
2004-03-18 06:56:06 +01:00
|
|
|
BufferDesc temp_bdb;
|
2006-05-22 00:07:35 +02:00
|
|
|
temp_bdb.bdb_page = HEADER_PAGE_NUMBER;
|
2003-09-08 22:23:46 +02:00
|
|
|
temp_bdb.bdb_dbb = database;
|
2004-07-10 05:20:33 +02:00
|
|
|
temp_bdb.bdb_buffer = reinterpret_cast<Ods::pag*>(header);
|
2008-08-31 03:10:41 +02:00
|
|
|
PageSpace* pageSpace = database->dbb_page_manager.findPageSpace(DB_PAGE_SPACE);
|
2007-04-24 16:05:46 +02:00
|
|
|
fb_assert(pageSpace);
|
2006-05-22 00:07:35 +02:00
|
|
|
jrd_file* file = pageSpace->file;
|
2009-01-20 09:33:59 +01:00
|
|
|
while (!PIO_read(file, &temp_bdb, temp_bdb.bdb_buffer, status))
|
|
|
|
{
|
2009-11-17 11:33:54 +01:00
|
|
|
if (!CCH_rollover_to_shadow(tdbb, database, file, false))
|
|
|
|
{
|
2003-09-08 22:23:46 +02:00
|
|
|
NBAK_TRACE(("Shadow change error"));
|
|
|
|
return false;
|
|
|
|
}
|
2006-05-22 00:07:35 +02:00
|
|
|
if (file != pageSpace->file)
|
|
|
|
file = pageSpace->file;
|
2009-11-17 11:33:54 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
if (retryCount++ == 3)
|
|
|
|
{
|
2003-09-08 22:23:46 +02:00
|
|
|
NBAK_TRACE(("IO error"));
|
2003-08-06 18:30:49 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
2003-09-08 22:23:46 +02:00
|
|
|
}
|
2003-08-06 18:30:49 +02:00
|
|
|
|
2009-11-24 12:42:56 +01:00
|
|
|
const int new_backup_state = header->hdr_flags & Ods::hdr_backup_mask;
|
2003-09-08 22:23:46 +02:00
|
|
|
NBAK_TRACE(("backup state read from header is %d", new_backup_state));
|
|
|
|
// Check is we missed lock/unlock cycle and need to invalidate
|
2008-12-05 02:20:14 +01:00
|
|
|
// our allocation table and file handle
|
2004-07-10 05:20:33 +02:00
|
|
|
const bool missed_cycle = (header->hdr_header.pag_scn - current_scn) > 1;
|
|
|
|
current_scn = header->hdr_header.pag_scn;
|
2003-08-06 18:30:49 +02:00
|
|
|
|
2003-09-08 22:23:46 +02:00
|
|
|
// Read difference file name from header clumplets
|
2007-10-26 15:21:16 +02:00
|
|
|
explicit_diff_name = false;
|
2004-03-18 06:56:06 +01:00
|
|
|
const UCHAR* p = header->hdr_data;
|
2009-11-17 11:33:54 +01:00
|
|
|
while (true)
|
|
|
|
{
|
2009-01-20 09:33:59 +01:00
|
|
|
switch (*p)
|
|
|
|
{
|
2004-05-01 00:47:16 +02:00
|
|
|
case Ods::HDR_backup_guid:
|
2004-03-18 06:56:06 +01:00
|
|
|
p += p[1] + 2;
|
2003-09-08 22:23:46 +02:00
|
|
|
continue;
|
2004-05-01 00:47:16 +02:00
|
|
|
case Ods::HDR_difference_file:
|
2007-10-26 15:21:16 +02:00
|
|
|
explicit_diff_name = true;
|
2005-11-22 00:33:20 +01:00
|
|
|
diff_name.assign(reinterpret_cast<const char*>(p + 2), p[1]);
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
2003-09-08 22:23:46 +02:00
|
|
|
break;
|
|
|
|
}
|
2007-10-26 15:21:16 +02:00
|
|
|
if (!explicit_diff_name)
|
2009-03-17 08:39:55 +01:00
|
|
|
generateFilename();
|
2008-03-20 14:08:10 +01:00
|
|
|
|
2009-01-20 09:33:59 +01:00
|
|
|
if (new_backup_state == nbak_state_normal || missed_cycle)
|
|
|
|
{
|
2003-09-08 22:23:46 +02:00
|
|
|
// Page allocation table cache is no longer valid.
|
2009-11-17 11:33:54 +01:00
|
|
|
if (alloc_table)
|
|
|
|
{
|
2003-09-08 22:23:46 +02:00
|
|
|
NBAK_TRACE(("Dropping alloc table"));
|
|
|
|
delete alloc_table;
|
|
|
|
alloc_table = NULL;
|
|
|
|
last_allocated_page = 0;
|
2009-03-17 08:39:55 +01:00
|
|
|
if (!allocLock->tryReleaseLock(tdbb))
|
2007-04-24 16:05:46 +02:00
|
|
|
ERR_bugcheck_msg("There are holders of alloc_lock after end_backup finish");
|
2003-09-08 22:23:46 +02:00
|
|
|
}
|
|
|
|
}
|
2007-04-24 16:05:46 +02:00
|
|
|
|
2009-01-20 09:33:59 +01:00
|
|
|
if (new_backup_state != nbak_state_normal && !diff_file)
|
2009-03-17 08:39:55 +01:00
|
|
|
openDelta();
|
2003-09-08 22:23:46 +02:00
|
|
|
// Adjust state at the very and to ensure proper error handling
|
|
|
|
backup_state = new_backup_state;
|
2007-04-24 16:05:46 +02:00
|
|
|
|
2003-08-06 18:30:49 +02:00
|
|
|
return true;
|
|
|
|
}
|
2004-03-18 06:56:06 +01:00
|
|
|
|
2010-09-24 11:53:51 +02:00
|
|
|
void BackupManager::shutdown(thread_db* /*tdbb*/)
|
2005-11-22 00:33:20 +01:00
|
|
|
{
|
2009-03-17 08:39:55 +01:00
|
|
|
closeDelta();
|
|
|
|
stateLock->shutdownLock();
|
|
|
|
allocLock->shutdownLock();
|
2005-11-22 00:33:20 +01:00
|
|
|
}
|