8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-26 10:43:03 +01:00
firebird-mirror/src/jrd/nbak.h
2008-12-05 00:56:15 +00:00

344 lines
11 KiB
C++

/*
* PROGRAM: JRD Access Method
* MODULE: nbak.h
* DESCRIPTION: New backup interface definitions
*
* 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.
*
* 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.
*
* The Original Code was created by Nickolay Samofatov
* for the Firebird Open Source RDBMS project.
*
* Copyright (c) 2004 Nickolay Samofatov <nickolay@broadviewsoftware.com>
* and all contributors signed below.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
*
*
*
*/
#ifndef JRD_NBAK_H
#define JRD_NBAK_H
#include "../common/classes/tree.h"
#include "../common/classes/rwlock.h"
#include "../common/classes/alloc.h"
#include "../common/classes/fb_string.h"
#include "GlobalRWLock.h"
// Uncomment this line if you need to trace backup-related activity
//#define NBAK_DEBUG
#ifdef NBAK_DEBUG
DEFINE_TRACE_ROUTINE(nbak_trace);
#define NBAK_TRACE(args) nbak_trace args
#define NBAK_TRACE_AST(message) gds__trace(message)
#else
#define NBAK_TRACE(args) /* nothing */
#define NBAK_TRACE_AST(message) /* nothing */
#endif
namespace Ods {
struct pag;
}
namespace Jrd {
class Lock;
class Record;
class thread_db;
class Database;
class jrd_file;
class AllocItem
{
public:
ULONG db_page; // Page number in the main database file
ULONG diff_page; // Page number in the difference file
Record* rec_data;
static const ULONG& generate(const void *sender, const AllocItem& item)
{
return item.db_page;
}
AllocItem() {}
AllocItem(ULONG db_pageL, ULONG diff_pageL)
{
this->db_page = db_pageL;
this->diff_page = diff_pageL;
}
};
typedef Firebird::BePlusTree<AllocItem, ULONG, MemoryPool, AllocItem> AllocItemTree;
// Class to synchronize access to backup state
class NBackupState: public GlobalRWLock
{
public:
ULONG flags;
NBackupState(thread_db* tdbb, MemoryPool& p, BackupManager *bakMan);
virtual ~NBackupState() { }
protected:
BackupManager *backup_manager;
virtual void blockingAstHandler(thread_db* tdbb);
virtual void fetch(thread_db* tdbb);
virtual void invalidate(thread_db* tdbb, bool ast_handler);
};
// Class to synchronize access to diff page allocation table
class NBackupAlloc: public GlobalRWLock
{
public:
NBackupAlloc(thread_db* tdbb, MemoryPool& p, BackupManager *bakMan);
virtual ~NBackupAlloc() { }
protected:
BackupManager *backup_manager;
virtual void fetch(thread_db* tdbb);
virtual void invalidate(thread_db* tdbb, bool ast_handler);
};
const UATOM NBAK_state_blocking = 1; // Changing the blocking state. All pages must mark by BDB_must_write flag if this flag is set
// Note this flags MUST correspond with backup mask in ods.h
const SATOM nbak_state_normal = 0x0; // Normal mode. Changes are simply written to main files
const SATOM nbak_state_stalled = 0x400; // 1024 Main files are locked. Changes are written to diff file
const SATOM nbak_state_merge = 0x800; // 2048 Merging changes from diff file into main files
const SATOM nbak_state_unknown = -1; // State is unknown. Needs to be read from disk
/*
* The functional responsibilities of NBAK are:
* 1. to redirect writes to difference files when asked (ALTER DATABASE BEGIN
* BACKUP statement)
* 2. to produce a GUID for the database snapshot and write it into the database
* header before the ALTER DATABASE BEGIN BACKUP statement returns
* 3. to merge differences into the database when asked (ALTER DATABASE END BACKUP
* statement)
* 4. to mark pages written by the engine with the current SCN [System Change
* Number] counter value for the database
* 5. to increment SCN on each change of backup state
*
* The backup state cycle is:
* nbak_state_normal -> nbak_state_stalled -> nbak_state_merge -> nbak_state_normal
* - In normal state writes go directly to the main database files.
* - In stalled state writes go to the difference file only and the main files are
* read-only.
* - In merge state new pages are not allocated from difference files. Writes go to
* the main database files. Reads of mapped pages compare both page versions and
* return the version which is fresher, because we don't know if it is merged or not.
*
* For synchronization NBAK uses 3 lock types via Firebird::GlobalRWLock:
* LCK_backup_database, LCK_backup_alloc, LCK_backup_end.
*
* LCK_backup_database protects "clean" state of database. Database is meant to be
* clean when it has no dirty pages. When attachment needs to mark a page as dirty
* (via CCH_mark) it needs to obtain READ (LCK_PR) lock of this kind. WRITE
* (LCK_EX) lock forces flush of all caches leaving no dirty pages to be written to
* database.
*
* Modification process of a page is as follows:
* CCH_fetch -> CCH_mark -> CCH_release -> write_page
*
* The dirty page is owned by the ATTACHMENT between CCH_mark and CCH_release and
* by DATABASE until write_page happens. Each dirty page owns the logical lock on
* LCK_backup_database that reflects this cycle.
*
* Header page is the special case in the above logic and needs to be locked during
* CCH_fetch(LCK_EX) phase because it needs to be modified during state transition
* and taking the lock later would cause deadlocks.
*
* AST on LCK_backup_database forces all dirty pages owned by DATABASE to be
* written to disk via write_page. Since finalizing processing of the pages owned
* by attachment may require taking new locks BDB_must_write is set for them to
* ensure that LCK_backup_database lock is released as soon as possible.
*
* To change backup state, engine takes LCK_backup_database lock first, forcing all
* dirty pages to the disk, modifies the header page reflecting the state change,
* and releases the lock allowing transaction processing to continue.
*
* LCK_backup_alloc is used to protect mapping table between difference file
* (.delta) and the database. To add new page to the mapping attachment needs to
* take WRITE lock of this kind. READ lock is necessary to read the table.
*
* LCK_backup_end is used to ensure reliable execution of state transition from
* nbak_state_merge to nbak_state_normal (MERGE process). Taking of WRITE (LCK_EX)
* lock of this kind is needed to perform the MERGE. Every new attachment attempts
* to finalize incomplete merge if the database is in nbak_state_merge mode and
* this lock is not taken.
*/
class BackupManager
{
public:
class SharedDatabaseHolder
{
public:
explicit SharedDatabaseHolder(thread_db* atdbb, BackupManager* bm)
: backupManager(bm), tdbb(atdbb)
{
backupManager->lock_shared_database(tdbb, true);
}
~SharedDatabaseHolder()
{
backupManager->unlock_shared_database(tdbb);
}
private:
// copying is prohibited
SharedDatabaseHolder(const SharedDatabaseHolder&);
SharedDatabaseHolder& operator =(const SharedDatabaseHolder&);
BackupManager* backupManager;
thread_db* tdbb;
};
// Set when db is creating. Default = false
bool dbCreating;
// Subsystem initialization
BackupManager(thread_db* tdbb, Database* _database, int ini_state);
// Release locks in response to shutdown AST
void shutdown_locks(thread_db* tdbb);
// Set difference file name in header.
// State must be locked and equal to nbak_state_normal to call this method
void set_difference(thread_db* tdbb, const char* filename);
// Return current backup state
int get_state() const
{
return backup_state;
}
// Sets current backup state
void set_state(int new_state)
{
backup_state = new_state;
}
// Return current SCN for database
ULONG get_current_scn() const
{
return current_scn;
}
// Initialize and open difference file for writing
void begin_backup(thread_db* tdbb);
// Merge difference file to main files (if needed) and unlink() difference
// file then. If merge is already in progress method silently returns false and
// does nothing (so it can be used for recovery on database startup).
void end_backup(thread_db* tdbb, bool recover);
void lock_shared_database(thread_db* tdbb, SSHORT wait);
void unlock_shared_database(thread_db* tdbb);
// Prevent allocation table from modification by other threads/processes
void lock_alloc(thread_db* tdbb, SSHORT wait);
void unlock_alloc(thread_db* tdbb);
void lock_alloc_write(thread_db* tdbb, SSHORT wait);
void unlock_alloc_write(thread_db* tdbb);
// Return page index in difference file that can be used in
// write_difference call later.
ULONG get_page_index(thread_db* tdbb, ULONG db_page);
// Return next page index in the difference file to be allocated
ULONG allocate_difference_page(thread_db* tdbb, ULONG db_page);
// Must have ISC_STATUS because it is called from write_page
bool write_difference(ISC_STATUS* status, ULONG diff_page, Ods::pag* page);
bool read_difference(thread_db* tdbb, ULONG diff_page, Ods::pag* page);
// Routines to declare and release interest in the database file
void checkout_dirty_page(thread_db* tdbb, SLONG owner_handle);
void release_dirty_page(thread_db* tdbb, SLONG owner_handle);
void change_dirty_page_owner(thread_db* tdbb, SLONG from_handle, SLONG to_handle);
// Returns difference owner handles for locks
static SLONG database_lock_handle(thread_db* tdbb)
{
return LCK_get_owner_handle_by_type(tdbb, LCK_OWNER_database);
}
static SLONG attachment_lock_handle(thread_db* tdbb)
{
return LCK_get_owner_handle_by_type(tdbb, LCK_OWNER_attachment);
}
void shutdown(thread_db* tdbb);
bool database_flush_in_progress() const
{
// NBAK_TRACE(("NBAK_state_blocking=%i", database_lock->flags & NBAK_state_blocking));
return database_lock->flags & NBAK_state_blocking;
}
// Make appropriate information up-to-date
bool actualize_state(thread_db* tdbb);
bool actualize_alloc(thread_db* tdbb);
// Get size (in pages) of locked database file
ULONG getPageCount();
// Subsystem finalization. Called from shutdown()
~BackupManager();
private:
Database* database;
jrd_file* diff_file;
AllocItemTree* alloc_table; // Cached allocation table of pages in difference file
volatile SATOM backup_state;
ULONG last_allocated_page; // Last physical page allocated in the difference file
BYTE *temp_buffers_space;
ULONG *alloc_buffer, *empty_buffer, *spare_buffer;
ULONG current_scn;
Firebird::PathName diff_name;
bool explicit_diff_name;
// Lock to protect allocation table in this process (BTree)
NBackupAlloc* alloc_lock;
NBackupState* database_lock;
void generate_filename();
// Function for force all connections to flush their caches
// and prevent them from marking new dirty pages
void lock_clean_database(thread_db* tdbb, SSHORT wait, WIN* window);
void unlock_clean_database(thread_db* tdbb);
class CleanDatabaseHolder
{
public:
explicit CleanDatabaseHolder(thread_db* atdbb, BackupManager* bm,
SSHORT wait, Jrd::WIN* window)
: backupManager(bm), tdbb(atdbb)
{
backupManager->lock_clean_database(tdbb, wait, window);
}
~CleanDatabaseHolder()
{
backupManager->unlock_clean_database(tdbb);
}
private:
// copying is prohibited
CleanDatabaseHolder(const CleanDatabaseHolder&);
CleanDatabaseHolder& operator =(const CleanDatabaseHolder&);
BackupManager* backupManager;
thread_db* tdbb;
};
};
} //namespace Jrd
#endif /* JRD_NBAK_PROTO_H */