mirror of
https://github.com/FirebirdSQL/firebird.git
synced 2025-01-27 07:23:04 +01:00
2506 lines
70 KiB
C++
2506 lines
70 KiB
C++
/*
|
|
* PROGRAM: JRD Access Method
|
|
* MODULE: pag.cpp
|
|
* DESCRIPTION: Page level ods manager
|
|
*
|
|
* 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): ______________________________________.
|
|
*
|
|
* 2002.02.15 Sean Leyne - Code Cleanup, removed obsolete ports:
|
|
* - EPSON, DELTA, IMP, NCR3000, M88K
|
|
* - HP9000 s300 and Apollo
|
|
*
|
|
* 2002.10.27 Sean Leyne - Completed removal of obsolete "DG_X86" port
|
|
* 2002.10.27 Sean Leyne - Code Cleanup, removed obsolete "UNIXWARE" port
|
|
* 2002.10.27 Sean Leyne - Code Cleanup, removed obsolete "Ultrix" port
|
|
* 2002.10.27 Sean Leyne - Code Cleanup, removed obsolete "Ultrix/MIPS" port
|
|
*
|
|
* 2002.10.28 Sean Leyne - Completed removal of obsolete "DGUX" port
|
|
* 2002.10.28 Sean Leyne - Code cleanup, removed obsolete "MPEXL" port
|
|
* 2002.10.28 Sean Leyne - Code cleanup, removed obsolete "DecOSF" port
|
|
* 2002.10.28 Sean Leyne - Code cleanup, removed obsolete "SGI" port
|
|
*
|
|
* 2002.10.29 Sean Leyne - Removed obsolete "Netware" port
|
|
*
|
|
*/
|
|
|
|
/*
|
|
* Modified by: Patrick J. P. Griffin
|
|
* Date: 11/29/2000
|
|
* Problem: Bug 116733 Too many generators corrupt database.
|
|
* DPM_gen_id was not calculating page and offset correctly.
|
|
* Change: Caculate pgc_gpg, number of generators per page,
|
|
* for use in DPM_gen_id.
|
|
*/
|
|
|
|
/* 2001.07.06 Sean Leyne - Code Cleanup, removed "#ifdef READONLY_DATABASE"
|
|
* conditionals, as the engine now fully supports
|
|
* readonly databases.
|
|
*
|
|
* 2001.08.07 Sean Leyne - Code Cleanup, removed "#ifdef READONLY_DATABASE"
|
|
* conditionals, second attempt
|
|
*
|
|
* 2002.02.15 Sean Leyne - Code Cleanup, removed obsolete "MAC" and "MAC_CP" defines
|
|
* 2002.02.15 Sean Leyne - Code Cleanup, removed obsolete "Apollo" port
|
|
*
|
|
*/
|
|
|
|
#include "firebird.h"
|
|
#include "../jrd/common.h"
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
|
|
#ifdef WIN_NT
|
|
#include <process.h>
|
|
#endif
|
|
|
|
#include "../common/config/config.h"
|
|
#include "../common/utils_proto.h"
|
|
#include "../jrd/fil.h"
|
|
#include "../jrd/jrd.h"
|
|
#include "../jrd/pag.h"
|
|
#include "../jrd/ods.h"
|
|
#include "../jrd/os/pio.h"
|
|
#include "../jrd/os/path_utils.h"
|
|
#include "../jrd/ibase.h"
|
|
#include "../jrd/gdsassert.h"
|
|
#include "../jrd/lck.h"
|
|
#include "../jrd/sdw.h"
|
|
#include "../jrd/cch.h"
|
|
#include "../jrd/nbak.h"
|
|
#include "../jrd/tra.h"
|
|
#ifdef VIO_DEBUG
|
|
#include "../jrd/vio_debug.h"
|
|
#endif
|
|
#include "../jrd/cch_proto.h"
|
|
#include "../jrd/dpm_proto.h"
|
|
#include "../jrd/err_proto.h"
|
|
#include "../jrd/gds_proto.h"
|
|
#include "../jrd/lck_proto.h"
|
|
#include "../jrd/met_proto.h"
|
|
#include "../jrd/mov_proto.h"
|
|
#include "../jrd/ods_proto.h"
|
|
#include "../jrd/pag_proto.h"
|
|
#include "../jrd/os/pio_proto.h"
|
|
#include "../jrd/thread_proto.h"
|
|
#include "../jrd/isc_f_proto.h"
|
|
#include "../jrd/TempSpace.h"
|
|
#include "../jrd/extds/ExtDS.h"
|
|
|
|
using namespace Jrd;
|
|
using namespace Ods;
|
|
using namespace Firebird;
|
|
|
|
static int blocking_ast_attachment(void*);
|
|
static void find_clump_space(thread_db* tdbb, SLONG, WIN*, pag**, USHORT, USHORT, const UCHAR*);
|
|
static bool find_type(thread_db* tdbb, SLONG, WIN*, pag**, USHORT, USHORT, UCHAR**, const UCHAR**);
|
|
|
|
inline void err_post_if_database_is_readonly(const Database* dbb)
|
|
{
|
|
if (dbb->dbb_flags & DBB_read_only)
|
|
ERR_post(Arg::Gds(isc_read_only_database));
|
|
}
|
|
|
|
// Class definitions (obsolete platforms are commented out)
|
|
// Class constant name consists of OS platform and CPU architecture.
|
|
//
|
|
// For ports created before Firebird 2.0 release 64-bit and 32-bit
|
|
// sub-architectures of the same CPU should use different classes.
|
|
// For 64-bit ports first created after or as a part of Firebird 2.0
|
|
// release CPU architecture may be the same for both variants.
|
|
|
|
static const int CLASS_UNKNOWN = 0;
|
|
//static const CLASS_APOLLO_68K = 1; // Apollo 68K, Dn 10K
|
|
static const int CLASS_SOLARIS_SPARC = 2; // Sun 68k, Sun Sparc, HP 9000/300, MAC AUX, IMP, DELTA, NeXT, UNIXWARE, DG_X86
|
|
static const int CLASS_SOLARIS_I386 = 3; // Sun 386i
|
|
//static const CLASS_VMS_VAX = 4; // VMS/VAX
|
|
//static const CLASS_ULTRIX_VAX = 5; // Ultrix/VAX
|
|
//static const CLASS_ULTRIX_MIPS = 6; // Ultrix/MIPS
|
|
static const int CLASS_HPUX_PA = 7; // HP-UX on PA-RISC (was: HP 900/800 (precision))
|
|
static const int CLASS_NETWARE_I386 = 8; // NetWare
|
|
//static const CLASS_MAC_OS = 9; // MAC-OS
|
|
static const int CLASS_AIX_PPC = 10; // AIX on PowerPC platform (was: IBM RS/6000)
|
|
//static const CLASS_DG_AVIION = 11; // DG AViiON
|
|
//static const CLASS_MPE_XL = 12; // MPE/XL
|
|
static const int CLASS_IRIX_MIPS = 13; // Silicon Graphics/IRIS
|
|
//static const int CLASS_CRAY = 14; // Cray
|
|
static const int CLASS_TRU64_ALPHA = 15; // Tru64 Unix running on Alpha (was: Dec OSF/1)
|
|
static const int CLASS_WINDOWS_I386 = 16; // NT -- post 4.0 (delivered 4.0 as class 8)
|
|
//static const CLASS_OS2 = 17; // OS/2
|
|
//static const CLASS_WIN16 = 18; // Windows 16 bit
|
|
static const int CLASS_LINUX_I386 = 19; // LINUX on Intel series
|
|
static const int CLASS_LINUX_SPARC = 20; // LINUX on sparc systems
|
|
static const int CLASS_FREEBSD_I386 = 21; // FreeBSD/i386
|
|
static const int CLASS_NETBSD_I386 = 22; // NetBSD/i386
|
|
static const int CLASS_DARWIN_PPC = 23; // Darwin/PowerPC
|
|
static const int CLASS_LINUX_AMD64 = 24; // LINUX on AMD64 systems
|
|
static const int CLASS_FREEBSD_AMD64 = 25;// FreeBSD/amd64
|
|
static const int CLASS_WINDOWS_AMD64 = 26;// Windows/amd64
|
|
static const int CLASS_LINUX_PPC = 27; // LINUX/PowerPC
|
|
static const int CLASS_DARWIN_I386 = 28; //Darwin/Intel
|
|
static const int CLASS_LINUX_MIPSEL = 29; // LINUX/MIPSEL
|
|
static const int CLASS_LINUX_MIPS = 30; // LINUX/MIPS
|
|
static const int CLASS_DARWIN_X64 = 31; // Darwin/x64
|
|
static const int CLASS_SOLARIS_AMD64 = 32; // Solaris/amd64
|
|
static const int CLASS_LINUX_ARM = 33; // LINUX/ARM
|
|
static const int CLASS_LINUX_IA64 = 34; // LINUX/IA64
|
|
static const int CLASS_DARWIN_PPC64 = 35; // Darwin/PowerPC64
|
|
|
|
static const int CLASS_MAX10 = CLASS_LINUX_AMD64; // This should not be changed, no new ports with ODS10
|
|
static const int CLASS_MAX = CLASS_DARWIN_PPC64;
|
|
|
|
// ARCHITECTURE COMPATIBILITY CLASSES
|
|
|
|
// For ODS10 and earlier things which normally define ODS compatibility are:
|
|
// 1) endianness (big-endian/little-endian)
|
|
// 2) alignment (32-bit or 64-bit), matters for record formats
|
|
// 3) pointer size (32-bit or 64-bit), also matters for record formats
|
|
//
|
|
// For ODS11 pointers are not stored in database and alignment is always 64-bit.
|
|
// So the only thing which normally matters for ODS11 is endiannes, but if
|
|
// endianness is wrong we are going to notice it during ODS version check,
|
|
// before architecture compatibility is tested. But we distinguish them here too,
|
|
// for consistency.
|
|
|
|
enum ArchitectureType {
|
|
archUnknown, // Unknown architecture, allow opening database only if CLASS matches exactly
|
|
archIntel86, // Little-endian platform with 32-bit pointers and 32-bit alignment (ODS10)
|
|
archLittleEndian, // Any little-endian platform with standard layout of data
|
|
archBigEndian // Any big-endian platform with standard layout of data
|
|
};
|
|
|
|
// Note that Sparc, HP and PowerPC disk structures should be compatible in theory,
|
|
// but in practice alignment on these platforms varies and actually depends on the
|
|
// compiler used to produce the build. Yes, some 32-bit RISC builds use 64-bit alignment.
|
|
// This is why we declare all such builds "Unknown" for ODS10.
|
|
|
|
static const ArchitectureType archMatrix10[CLASS_MAX10 + 1] =
|
|
{
|
|
archUnknown, // CLASS_UNKNOWN
|
|
archUnknown, // CLASS_APOLLO_68K
|
|
archUnknown, // CLASS_SOLARIS_SPARC
|
|
archIntel86, // CLASS_SOLARIS_I386
|
|
archUnknown, // CLASS_VMS_VAX
|
|
archUnknown, // CLASS_ULTRIX_VAX
|
|
archUnknown, // CLASS_ULTRIX_MIPS
|
|
archUnknown, // CLASS_HPUX_PA
|
|
archUnknown, // CLASS_NETWARE_I386
|
|
archUnknown, // CLASS_MAC_OS
|
|
archUnknown, // CLASS_AIX_PPC
|
|
archUnknown, // CLASS_DG_AVIION
|
|
archUnknown, // CLASS_MPE_XL
|
|
archUnknown, // CLASS_IRIX_MIPS
|
|
archUnknown, // CLASS_CRAY
|
|
archUnknown, // CLASS_TRU64_ALPHA
|
|
archIntel86, // CLASS_WINDOWS_I386
|
|
archUnknown, // CLASS_OS2
|
|
archUnknown, // CLASS_WIN16
|
|
archIntel86, // CLASS_LINUX_I386
|
|
archUnknown, // CLASS_LINUX_SPARC
|
|
archIntel86, // CLASS_FREEBSD_I386
|
|
archIntel86, // CLASS_NETBSD_I386
|
|
archUnknown, // CLASS_DARWIN_PPC
|
|
archUnknown // CLASS_LINUX_AMD64
|
|
};
|
|
|
|
static const ArchitectureType archMatrix[CLASS_MAX + 1] =
|
|
{
|
|
archUnknown, // CLASS_UNKNOWN
|
|
archUnknown, // CLASS_APOLLO_68K
|
|
archBigEndian, // CLASS_SOLARIS_SPARC
|
|
archLittleEndian, // CLASS_SOLARIS_I386
|
|
archUnknown, // CLASS_VMS_VAX
|
|
archUnknown, // CLASS_ULTRIX_VAX
|
|
archUnknown, // CLASS_ULTRIX_MIPS
|
|
archBigEndian, // CLASS_HPUX_PA
|
|
archUnknown, // CLASS_NETWARE_I386
|
|
archUnknown, // CLASS_MAC_OS
|
|
archBigEndian, // CLASS_AIX_PPC
|
|
archUnknown, // CLASS_DG_AVIION
|
|
archUnknown, // CLASS_MPE_XL
|
|
archBigEndian, // CLASS_IRIX_MIPS
|
|
archUnknown, // CLASS_CRAY
|
|
archBigEndian, // CLASS_TRU64_ALPHA
|
|
archLittleEndian, // CLASS_WINDOWS_I386
|
|
archUnknown, // CLASS_OS2
|
|
archUnknown, // CLASS_WIN16
|
|
archLittleEndian, // CLASS_LINUX_I386
|
|
archBigEndian, // CLASS_LINUX_SPARC
|
|
archLittleEndian, // CLASS_FREEBSD_I386
|
|
archLittleEndian, // CLASS_NETBSD_I386
|
|
archBigEndian, // CLASS_DARWIN_PPC
|
|
archLittleEndian, // CLASS_LINUX_AMD64
|
|
archLittleEndian, // CLASS_FREEBSD_AMD64
|
|
archLittleEndian, // CLASS_WINDOWS_AMD64
|
|
archBigEndian, // CLASS_LINUX_PPC
|
|
archLittleEndian, // CLASS_DARWIN_I386
|
|
archLittleEndian, // CLASS_LINUX_MIPSEL
|
|
archBigEndian, // CLASS_LINUX_MIPS
|
|
archLittleEndian, // CLASS_DARWIN_X64
|
|
archLittleEndian, // CLASS_SOLARIS_AMD64
|
|
archLittleEndian, // CLASS_LINUX_ARM
|
|
archLittleEndian, // CLASS_LINUX_IA64
|
|
archBigEndian // CLASS_DARWIN_PPC64
|
|
};
|
|
|
|
#ifdef sun
|
|
#ifdef i386
|
|
const SSHORT CLASS = CLASS_SOLARIS_I386;
|
|
#endif
|
|
#ifdef sparc
|
|
const SSHORT CLASS = CLASS_SOLARIS_SPARC;
|
|
#endif
|
|
#ifdef AMD64
|
|
const SSHORT CLASS = CLASS_SOLARIS_AMD64;
|
|
#endif
|
|
#endif
|
|
|
|
#ifdef HPUX
|
|
const SSHORT CLASS = CLASS_HPUX_PA;
|
|
#endif
|
|
|
|
#ifdef AIX_PPC
|
|
const SSHORT CLASS = CLASS_AIX_PPC;
|
|
#endif
|
|
|
|
#ifdef WIN_NT
|
|
#if defined(I386)
|
|
const SSHORT CLASS = CLASS_WINDOWS_I386;
|
|
#elif defined(AMD64)
|
|
const SSHORT CLASS = CLASS_WINDOWS_AMD64;
|
|
#else
|
|
#error no support on other hardware for Windows
|
|
#endif
|
|
#endif // WIN_NT
|
|
|
|
#ifdef LINUX
|
|
#if defined(i386) || defined(i586)
|
|
const SSHORT CLASS = CLASS_LINUX_I386;
|
|
#elif defined(sparc)
|
|
const SSHORT CLASS = CLASS_LINUX_SPARC;
|
|
#elif defined(AMD64)
|
|
const SSHORT CLASS = CLASS_LINUX_AMD64;
|
|
#elif defined(ARM)
|
|
const SSHORT CLASS = CLASS_LINUX_ARM;
|
|
#elif defined(PPC)
|
|
const SSHORT CLASS = CLASS_LINUX_PPC;
|
|
#elif defined(MIPSEL)
|
|
const SSHORT CLASS = CLASS_LINUX_MIPSEL;
|
|
#elif defined(MIPS)
|
|
const SSHORT CLASS = CLASS_LINUX_MIPS;
|
|
#elif defined(IA64)
|
|
const SSHORT CLASS = CLASS_LINUX_IA64;
|
|
#else
|
|
#error no support on other hardware for Linux
|
|
#endif
|
|
#endif // LINUX
|
|
|
|
#ifdef FREEBSD
|
|
#if defined(i386)
|
|
const SSHORT CLASS = CLASS_FREEBSD_I386;
|
|
#elif defined(AMD64)
|
|
const SSHORT CLASS = CLASS_FREEBSD_AMD64;
|
|
#else
|
|
#error no support on other hardware for FreeBSD
|
|
#endif
|
|
#endif
|
|
|
|
#ifdef NETBSD
|
|
const SSHORT CLASS = CLASS_NETBSD_I386;
|
|
#endif
|
|
|
|
#ifdef DARWIN
|
|
#if defined(i386)
|
|
const SSHORT CLASS = CLASS_DARWIN_I386;
|
|
#elif defined(DARWIN64)
|
|
const SSHORT CLASS = CLASS_DARWIN_X64;
|
|
#elif defined(powerpc)
|
|
const SSHORT CLASS = CLASS_DARWIN_PPC;
|
|
#elif defined(DARWINPPC64)
|
|
const SSHORT CLASS = CLASS_DARWIN_PPC64;
|
|
#endif
|
|
#endif // DARWIN
|
|
|
|
static const char* const SCRATCH = "fb_table_";
|
|
|
|
static const int MIN_EXTEND_BYTES = 128 * 1024; // 128KB
|
|
|
|
// CVC: Since nobody checks the result from this function (strange!), I changed
|
|
// bool to void as the return type but left the result returned as comment.
|
|
void PAG_add_clump(thread_db* tdbb,
|
|
SLONG page_num, USHORT type,
|
|
USHORT len, const UCHAR* entry, ClumpOper mode) // bool must_write
|
|
{
|
|
/***********************************************
|
|
*
|
|
* P A G _ a d d _ c l u m p
|
|
*
|
|
***********************************************
|
|
*
|
|
* Functional description
|
|
* Adds a clump to log/header page.
|
|
* mode
|
|
* 0 - add CLUMP_ADD
|
|
* 1 - replace CLUMP_REPLACE
|
|
* 2 - replace only! CLUMP_REPLACE_ONLY
|
|
* returns
|
|
* true - modified page
|
|
* false - nothing done => nobody checks this function's result.
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* dbb = tdbb->getDatabase();
|
|
CHECK_DBB(dbb);
|
|
|
|
err_post_if_database_is_readonly(dbb);
|
|
|
|
pag* page;
|
|
header_page* header = 0;
|
|
log_info_page* logp = 0;
|
|
USHORT* end_addr;
|
|
WIN window(DB_PAGE_SPACE, page_num);
|
|
if (page_num == HEADER_PAGE) {
|
|
page = CCH_FETCH(tdbb, &window, LCK_write, pag_header);
|
|
header = (header_page*) page;
|
|
end_addr = &header->hdr_end;
|
|
}
|
|
else {
|
|
page = CCH_FETCH(tdbb, &window, LCK_write, pag_log);
|
|
logp = (log_info_page*) page;
|
|
end_addr = &logp->log_end;
|
|
}
|
|
|
|
UCHAR* entry_p;
|
|
const UCHAR* clump_end;
|
|
while (mode != CLUMP_ADD)
|
|
{
|
|
const bool found =
|
|
find_type(tdbb, page_num, &window, &page, LCK_write, type, &entry_p, &clump_end);
|
|
|
|
/* If we did'nt find it and it is REPLACE_ONLY, return */
|
|
|
|
if (!found && mode == CLUMP_REPLACE_ONLY) {
|
|
CCH_RELEASE(tdbb, &window);
|
|
return; //false;
|
|
}
|
|
|
|
/* If not found, just go and add the entry */
|
|
|
|
if (!found)
|
|
break;
|
|
|
|
/* if same size, overwrite it */
|
|
|
|
if (entry_p[1] == len)
|
|
{
|
|
entry_p += 2;
|
|
if (len)
|
|
{
|
|
//if (must_write)
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
|
//else
|
|
// CCH_MARK(tdbb, &window);
|
|
|
|
memcpy(entry_p, entry, len);
|
|
}
|
|
CCH_RELEASE(tdbb, &window);
|
|
return; // true;
|
|
}
|
|
|
|
/* delete the entry
|
|
|
|
* Page is marked must write because of precedence problems. Later
|
|
* on we may allocate a new page and set up a precedence relationship.
|
|
* This may be the lower precedence page and so it cannot be dirty
|
|
*/
|
|
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
|
|
|
*end_addr -= (2 + entry_p[1]);
|
|
|
|
const UCHAR* r = entry_p + 2 + entry_p[1];
|
|
USHORT l = clump_end - r + 1;
|
|
if (l)
|
|
memmove(entry_p, r, l);
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
/* refetch the page */
|
|
|
|
window.win_page = page_num;
|
|
if (page_num == HEADER_PAGE) {
|
|
page = CCH_FETCH(tdbb, &window, LCK_write, pag_header);
|
|
header = (header_page*) page;
|
|
end_addr = &header->hdr_end;
|
|
}
|
|
else {
|
|
page = CCH_FETCH(tdbb, &window, LCK_write, pag_log);
|
|
logp = (log_info_page*) page;
|
|
end_addr = &logp->log_end;
|
|
}
|
|
break;
|
|
}
|
|
|
|
// Add the entry
|
|
|
|
find_clump_space(tdbb, page_num, &window, &page, type, len, entry);
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
return; // true;
|
|
}
|
|
|
|
|
|
USHORT PAG_add_file(thread_db* tdbb, const TEXT* file_name, SLONG start)
|
|
{
|
|
/**************************************
|
|
*
|
|
* P A G _ a d d _ f i l e
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Add a file to the current database. Return the sequence number for the new file.
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* dbb = tdbb->getDatabase();
|
|
CHECK_DBB(dbb);
|
|
|
|
err_post_if_database_is_readonly(dbb);
|
|
|
|
/* Find current last file */
|
|
|
|
PageSpace* pageSpace = dbb->dbb_page_manager.findPageSpace(DB_PAGE_SPACE);
|
|
jrd_file* file = pageSpace->file;
|
|
while (file->fil_next) {
|
|
file = file->fil_next;
|
|
}
|
|
|
|
// Verify database file path against DatabaseAccess entry of firebird.conf
|
|
if (!JRD_verify_database_access(file_name)) {
|
|
ERR_post(Arg::Gds(isc_conf_access_denied) << Arg::Str("additional database file") <<
|
|
Arg::Str(file_name));
|
|
}
|
|
|
|
/* Create the file. If the sequence number comes back zero, it didn't
|
|
work, so punt */
|
|
|
|
const USHORT sequence = PIO_add_file(dbb, pageSpace->file, file_name, start);
|
|
if (!sequence)
|
|
return 0;
|
|
|
|
/* Create header page for new file */
|
|
|
|
jrd_file* next = file->fil_next;
|
|
|
|
if (dbb->dbb_flags & (DBB_force_write | DBB_no_fs_cache))
|
|
{
|
|
PIO_force_write(next, dbb->dbb_flags & DBB_force_write, dbb->dbb_flags & DBB_no_fs_cache);
|
|
}
|
|
|
|
WIN window(DB_PAGE_SPACE, next->fil_min_page);
|
|
header_page* header = (header_page*) CCH_fake(tdbb, &window, 1);
|
|
header->hdr_header.pag_type = pag_header;
|
|
header->hdr_sequence = sequence;
|
|
header->hdr_page_size = dbb->dbb_page_size;
|
|
header->hdr_data[0] = HDR_end;
|
|
header->hdr_end = HDR_SIZE;
|
|
next->fil_sequence = sequence;
|
|
|
|
#ifdef SUPPORT_RAW_DEVICES
|
|
/* The following lines (taken from PAG_format_header) are needed to identify
|
|
this file in raw_devices_validate_database as a valid database attachment. */
|
|
*(ISC_TIMESTAMP*) header->hdr_creation_date = Firebird::TimeStamp::getCurrentTimeStamp().value();
|
|
// should we include milliseconds or not?
|
|
//Firebird::TimeStamp::round_time(header->hdr_creation_date->timestamp_time, 0);
|
|
|
|
header->hdr_ods_version = ODS_VERSION | ODS_FIREBIRD_FLAG;
|
|
header->hdr_implementation = CLASS;
|
|
header->hdr_ods_minor = ODS_CURRENT;
|
|
header->hdr_ods_minor_original = ODS_CURRENT;
|
|
if (dbb->dbb_flags & DBB_DB_SQL_dialect_3)
|
|
header->hdr_flags |= hdr_SQL_dialect_3;
|
|
#endif
|
|
|
|
header->hdr_header.pag_checksum = CCH_checksum(window.win_bdb);
|
|
PIO_write(pageSpace->file, window.win_bdb, window.win_buffer, tdbb->tdbb_status_vector);
|
|
CCH_RELEASE(tdbb, &window);
|
|
next->fil_fudge = 1;
|
|
|
|
/* Update the previous header page to point to new file */
|
|
|
|
file->fil_fudge = 0;
|
|
window.win_page = file->fil_min_page;
|
|
header = (header_page*) CCH_FETCH(tdbb, &window, LCK_write, pag_header);
|
|
if (!file->fil_min_page)
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
|
else
|
|
CCH_MARK(tdbb, &window);
|
|
|
|
--start;
|
|
|
|
if (file->fil_min_page) {
|
|
PAG_add_header_entry(tdbb, header, HDR_file, strlen(file_name),
|
|
reinterpret_cast<const UCHAR*>(file_name));
|
|
PAG_add_header_entry(tdbb, header, HDR_last_page, sizeof(SLONG), (UCHAR*) &start);
|
|
}
|
|
else {
|
|
PAG_add_clump(tdbb, HEADER_PAGE, HDR_file, strlen(file_name),
|
|
reinterpret_cast<const UCHAR*>(file_name), CLUMP_REPLACE); //, true;
|
|
PAG_add_clump(tdbb, HEADER_PAGE, HDR_last_page, sizeof(SLONG),
|
|
(UCHAR*) &start, CLUMP_REPLACE); //, true
|
|
}
|
|
|
|
header->hdr_header.pag_checksum = CCH_checksum(window.win_bdb);
|
|
PIO_write(pageSpace->file, window.win_bdb, window.win_buffer, tdbb->tdbb_status_vector);
|
|
CCH_RELEASE(tdbb, &window);
|
|
if (file->fil_min_page)
|
|
file->fil_fudge = 1;
|
|
|
|
return sequence;
|
|
}
|
|
|
|
|
|
bool PAG_add_header_entry(thread_db* tdbb, header_page* header,
|
|
USHORT type, USHORT len, const UCHAR* entry)
|
|
{
|
|
/***********************************************
|
|
*
|
|
* P A G _ a d d _ h e a d e r _ e n t r y
|
|
*
|
|
***********************************************
|
|
*
|
|
* Functional description
|
|
* Add an entry to header page.
|
|
* This will be used mainly for the shadow header page and adding
|
|
* secondary files.
|
|
* Will not follow to hdr_next_page
|
|
* Will not journal changes made to page. => obsolete
|
|
* RETURNS
|
|
* true - modified page
|
|
* false - nothing done
|
|
* CVC: Nobody checks the result of this function!
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* dbb = tdbb->getDatabase();
|
|
CHECK_DBB(dbb);
|
|
|
|
err_post_if_database_is_readonly(dbb);
|
|
|
|
UCHAR* p = header->hdr_data;
|
|
while (*p != HDR_end && *p != type)
|
|
p += 2 + p[1];
|
|
|
|
if (*p != HDR_end)
|
|
return false;
|
|
|
|
// We are at HDR_end, add the entry
|
|
|
|
const int free_space = dbb->dbb_page_size - header->hdr_end;
|
|
|
|
if (free_space > (2 + len))
|
|
{
|
|
fb_assert(type <= MAX_UCHAR);
|
|
fb_assert(len <= MAX_UCHAR);
|
|
*p++ = static_cast<UCHAR>(type);
|
|
*p++ = static_cast<UCHAR>(len);
|
|
|
|
if (len) {
|
|
if (entry) {
|
|
memcpy(p, entry, len);
|
|
}
|
|
else {
|
|
memset(p, 0, len);
|
|
}
|
|
p += len;
|
|
}
|
|
|
|
*p = HDR_end;
|
|
|
|
header->hdr_end = p - (UCHAR *) header;
|
|
|
|
return true;
|
|
}
|
|
|
|
BUGCHECK(251);
|
|
return false; /* Added to remove compiler warning */
|
|
}
|
|
|
|
|
|
void PAG_attach_temp_pages(thread_db* tdbb, USHORT pageSpaceID)
|
|
{
|
|
/***********************************************
|
|
*
|
|
* P A G _ a t t a c h _ t e m p _ p a g e s
|
|
*
|
|
***********************************************
|
|
*
|
|
* Functional description
|
|
* Attach a temporary page space
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* dbb = tdbb->getDatabase();
|
|
CHECK_DBB(dbb);
|
|
|
|
PageSpace* pageSpaceTemp = dbb->dbb_page_manager.addPageSpace(pageSpaceID);
|
|
if (!pageSpaceTemp->file)
|
|
{
|
|
Firebird::PathName file_name = TempFile::create(SCRATCH);
|
|
pageSpaceTemp->file = PIO_create(dbb, file_name, true, true, false);
|
|
PAG_format_pip(tdbb, *pageSpaceTemp);
|
|
}
|
|
}
|
|
|
|
|
|
bool PAG_replace_entry_first(thread_db* tdbb, header_page* header,
|
|
USHORT type, USHORT len, const UCHAR* entry)
|
|
{
|
|
/***********************************************
|
|
*
|
|
* P A G _ r e p l a c e _ e n t r y _ f i r s t
|
|
*
|
|
***********************************************
|
|
*
|
|
* Functional description
|
|
* Replace an entry in the header page so it will become first entry
|
|
* This will be used mainly for the clumplets used for backup purposes
|
|
* because they are needed to be read without page lock
|
|
* Will not follow to hdr_next_page
|
|
* Will not journal changes made to page. => obsolete
|
|
* RETURNS
|
|
* true - modified page
|
|
* false - nothing done
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* dbb = tdbb->getDatabase();
|
|
CHECK_DBB(dbb);
|
|
|
|
err_post_if_database_is_readonly(dbb);
|
|
|
|
UCHAR* p = header->hdr_data;
|
|
while (*p != HDR_end && *p != type) {
|
|
p += 2 + p[1];
|
|
}
|
|
|
|
// Remove item if found it somewhere
|
|
if (*p != HDR_end) {
|
|
UCHAR l = p[1] + 2;
|
|
memmove(p, p + l, header->hdr_end - (p - (UCHAR*) header) - l + 1); // to preserve HDR_end
|
|
header->hdr_end -= l;
|
|
}
|
|
|
|
if (!entry) {
|
|
return false; // We were asked just to remove item. We finished.
|
|
}
|
|
|
|
// Check if we got enough space
|
|
if (dbb->dbb_page_size - header->hdr_end <= len + 2) {
|
|
BUGCHECK(251);
|
|
}
|
|
|
|
// Actually add the item
|
|
memmove(header->hdr_data + len + 2, header->hdr_data, header->hdr_end - HDR_SIZE + 1);
|
|
header->hdr_data[0] = type;
|
|
header->hdr_data[1] = len;
|
|
memcpy(header->hdr_data + 2, entry, len);
|
|
header->hdr_end += len + 2;
|
|
|
|
return true;
|
|
}
|
|
|
|
PAG PAG_allocate(thread_db* tdbb, WIN* window)
|
|
{
|
|
/**************************************
|
|
*
|
|
* P A G _ a l l o c a t e
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Allocate a page and fake a read with a write lock. This is
|
|
* the universal sequence when allocating pages.
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* dbb = tdbb->getDatabase();
|
|
CHECK_DBB(dbb);
|
|
|
|
PageManager& pageMgr = dbb->dbb_page_manager;
|
|
PageSpace* pageSpace = pageMgr.findPageSpace(window->win_page.getPageSpaceID());
|
|
fb_assert(pageSpace);
|
|
|
|
// Not sure if this can be moved inside the loop. Maybe some data members
|
|
// should persist across iterations?
|
|
WIN pip_window(pageSpace->pageSpaceID, -1);
|
|
// CVC: Not sure of the initial value. Notice bytes and bit are used after the loop.
|
|
UCHAR* bytes = 0;
|
|
UCHAR bit = 0;
|
|
|
|
pag* new_page = 0; // NULL before the search for a new page.
|
|
|
|
// Starting from ODS 11.1 we store in pip_header.reserved number of pages
|
|
// allocated from this pointer page. There is intention to create dedicated
|
|
// field at page_inv_page for this purpose in ODS 12.
|
|
const bool isODS11_x = (dbb->dbb_ods_version == ODS_VERSION11 && dbb->dbb_minor_version >= 1);
|
|
|
|
/* Find an allocation page with something on it */
|
|
|
|
SLONG relative_bit = -1;
|
|
SLONG sequence;
|
|
SLONG pipMin;
|
|
for (sequence = pageSpace->pipHighWater; true; sequence++)
|
|
{
|
|
pip_window.win_page = (sequence == 0) ?
|
|
pageSpace->ppFirst : sequence * dbb->dbb_page_manager.pagesPerPIP - 1;
|
|
page_inv_page* pip_page = (page_inv_page*) CCH_FETCH(tdbb, &pip_window, LCK_write, pag_pages);
|
|
|
|
pipMin = MAX_SLONG;
|
|
const UCHAR* end = (UCHAR*) pip_page + dbb->dbb_page_size;
|
|
for (bytes = &pip_page->pip_bits[pip_page->pip_min >> 3]; bytes < end; bytes++)
|
|
{
|
|
if (*bytes != 0) {
|
|
/* 'byte' is not zero, so it describes at least one free page. */
|
|
bit = 1;
|
|
for (SLONG i = 0; i < 8; i++, bit <<= 1) {
|
|
if (bit & *bytes) {
|
|
relative_bit = ((bytes - pip_page->pip_bits) << 3) + i;
|
|
pipMin = MIN(pipMin, relative_bit);
|
|
|
|
const SLONG pageNum = relative_bit + sequence * pageMgr.pagesPerPIP;
|
|
window->win_page = pageNum;
|
|
new_page = CCH_fake(tdbb, window, 0); /* don't wait on latch */
|
|
if (new_page)
|
|
{
|
|
if (isODS11_x)
|
|
{
|
|
BackupManager::SharedDatabaseHolder sdbHolder(tdbb, dbb->dbb_backup_manager);
|
|
const bool nbak_stalled = (dbb->dbb_backup_manager->get_state() == nbak_state_stalled);
|
|
|
|
USHORT next_init_pages = 1;
|
|
// ensure there are space on disk for faked page
|
|
if (relative_bit + 1 > pip_page->pip_header.reserved)
|
|
{
|
|
fb_assert(relative_bit == pip_page->pip_header.reserved);
|
|
|
|
USHORT init_pages = 0;
|
|
if (!nbak_stalled)
|
|
{
|
|
init_pages = 1;
|
|
if (!(dbb->dbb_flags & DBB_no_reserve))
|
|
{
|
|
const int minExtendPages = MIN_EXTEND_BYTES / dbb->dbb_page_size;
|
|
init_pages = sequence ? 64 : MIN(pip_page->pip_header.reserved / 16, 64);
|
|
|
|
// don't touch pages belongs to the next PIP
|
|
init_pages =
|
|
MIN(init_pages, pageMgr.pagesPerPIP - pip_page->pip_header.reserved);
|
|
|
|
if (init_pages < minExtendPages)
|
|
init_pages = 1;
|
|
|
|
next_init_pages = init_pages;
|
|
}
|
|
|
|
ISC_STATUS_ARRAY status;
|
|
const ULONG start =
|
|
sequence * pageMgr.pagesPerPIP + pip_page->pip_header.reserved;
|
|
|
|
init_pages = PIO_init_data(dbb, pageSpace->file, status, start, init_pages);
|
|
}
|
|
|
|
if (init_pages) {
|
|
pip_page->pip_header.reserved += init_pages;
|
|
}
|
|
else
|
|
{
|
|
// PIO_init_data returns zero - perhaps it is not supported, no space
|
|
// left on disk or IO error occured. Try to write one page and handle
|
|
// IO errors if any
|
|
CCH_must_write(window);
|
|
try {
|
|
CCH_RELEASE(tdbb, window);
|
|
pip_page->pip_header.reserved = relative_bit + 1;
|
|
}
|
|
catch (Firebird::status_exception)
|
|
{
|
|
// forget about this page as if we never tried to fake it
|
|
CCH_forget_page(tdbb, window);
|
|
|
|
// normally all page buffers now released by CCH_unwind
|
|
// only exception is when TDBB_no_cache_unwind flag is set
|
|
if (tdbb->tdbb_flags & TDBB_no_cache_unwind)
|
|
CCH_RELEASE(tdbb, &pip_window);
|
|
|
|
throw;
|
|
}
|
|
|
|
new_page = CCH_fake(tdbb, window, 1);
|
|
}
|
|
fb_assert(new_page);
|
|
}
|
|
|
|
if (!(dbb->dbb_flags & DBB_no_reserve) && !nbak_stalled)
|
|
{
|
|
const ULONG initialized =
|
|
sequence * pageMgr.pagesPerPIP + pip_page->pip_header.reserved;
|
|
|
|
// At this point we ensure database has at least "initialized" pages
|
|
// allocated. To avoid file growth by few pages when all this space
|
|
// will be used, extend file up to initialized + next_init_pages now
|
|
pageSpace->extend(tdbb, initialized + next_init_pages);
|
|
}
|
|
}
|
|
break; /* Found a page and successfully fake-ed it */
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (new_page)
|
|
break; /* Found a page and successfully fake-ed it */
|
|
}
|
|
if (new_page)
|
|
break; /* Found a page and successfully fake-ed it */
|
|
CCH_RELEASE(tdbb, &pip_window);
|
|
}
|
|
|
|
pageSpace->pipHighWater = sequence;
|
|
|
|
CCH_MARK(tdbb, &pip_window);
|
|
*bytes &= ~bit;
|
|
page_inv_page* pip_page = (page_inv_page*) pip_window.win_buffer;
|
|
|
|
if (pipMin == relative_bit)
|
|
pipMin++;
|
|
pip_page->pip_min = pipMin;
|
|
|
|
if (relative_bit != pageMgr.pagesPerPIP - 1) {
|
|
CCH_RELEASE(tdbb, &pip_window);
|
|
CCH_precedence(tdbb, window, pip_window.win_page);
|
|
#ifdef VIO_DEBUG
|
|
if (debug_flag > DEBUG_WRITES_INFO)
|
|
printf("\tPAG_allocate: allocated page %"SLONGFORMAT"\n",
|
|
window->win_page.getPageNum());
|
|
#endif
|
|
return new_page;
|
|
}
|
|
|
|
/* We've allocated the last page on the space management page. Rather
|
|
than returning it, format it as a page inventory page, and recurse. */
|
|
|
|
page_inv_page* new_pip_page = (page_inv_page*) new_page;
|
|
new_pip_page->pip_header.pag_type = pag_pages;
|
|
const UCHAR* end = (UCHAR *) new_pip_page + dbb->dbb_page_size;
|
|
memset(new_pip_page->pip_bits, 0xff, end - new_pip_page->pip_bits);
|
|
|
|
CCH_must_write(window);
|
|
CCH_RELEASE(tdbb, window);
|
|
CCH_must_write(&pip_window);
|
|
CCH_RELEASE(tdbb, &pip_window);
|
|
|
|
return PAG_allocate(tdbb, window);
|
|
}
|
|
|
|
|
|
SLONG PAG_attachment_id(thread_db* tdbb)
|
|
{
|
|
/******************************************
|
|
*
|
|
* P A G _ a t t a c h m e n t _ i d
|
|
*
|
|
******************************************
|
|
*
|
|
* Functional description
|
|
* Get attachment id. If don't have one, get one. As a side
|
|
* effect, get a lock on it as well.
|
|
*
|
|
******************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* dbb = tdbb->getDatabase();
|
|
|
|
Attachment* attachment = tdbb->getAttachment();
|
|
WIN window(DB_PAGE_SPACE, -1);
|
|
|
|
/* If we've been here before just return the id */
|
|
|
|
if (attachment->att_id_lock)
|
|
return attachment->att_attachment_id;
|
|
|
|
/* Get new attachment id */
|
|
|
|
if (dbb->dbb_flags & DBB_read_only) {
|
|
attachment->att_attachment_id = dbb->dbb_attachment_id + dbb->genSharedUniqueNumber(tdbb);
|
|
}
|
|
else {
|
|
window.win_page = HEADER_PAGE_NUMBER;
|
|
header_page* header = (header_page*) CCH_FETCH(tdbb, &window, LCK_write, pag_header);
|
|
CCH_MARK(tdbb, &window);
|
|
attachment->att_attachment_id = ++header->hdr_attachment_id;
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
}
|
|
|
|
/* Take out lock on attachment id */
|
|
|
|
Lock* lock = FB_NEW_RPT(*dbb->dbb_permanent, sizeof(SLONG)) Lock();
|
|
attachment->att_id_lock = lock;
|
|
lock->lck_type = LCK_attachment;
|
|
lock->lck_owner_handle = LCK_get_owner_handle(tdbb, lock->lck_type);
|
|
lock->lck_parent = dbb->dbb_lock;
|
|
lock->lck_length = sizeof(SLONG);
|
|
lock->lck_key.lck_long = attachment->att_attachment_id;
|
|
lock->lck_dbb = dbb;
|
|
lock->lck_ast = blocking_ast_attachment;
|
|
lock->lck_object = attachment;
|
|
LCK_lock(tdbb, lock, LCK_EX, LCK_WAIT);
|
|
|
|
return attachment->att_attachment_id;
|
|
}
|
|
|
|
|
|
bool PAG_delete_clump_entry(thread_db* tdbb, SLONG page_num, USHORT type)
|
|
{
|
|
/***********************************************
|
|
*
|
|
* P A G _ d e l e t e _ c l u m p _ e n t r y
|
|
*
|
|
***********************************************
|
|
*
|
|
* Functional description
|
|
* Gets rid on the entry 'type' from page.
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* dbb = tdbb->getDatabase();
|
|
CHECK_DBB(dbb);
|
|
|
|
err_post_if_database_is_readonly(dbb);
|
|
|
|
WIN window(DB_PAGE_SPACE, page_num);
|
|
|
|
pag* page;
|
|
if (page_num == HEADER_PAGE)
|
|
page = CCH_FETCH(tdbb, &window, LCK_write, pag_header);
|
|
else
|
|
page = CCH_FETCH(tdbb, &window, LCK_write, pag_log);
|
|
|
|
UCHAR* entry_p;
|
|
const UCHAR* clump_end;
|
|
if (!find_type(tdbb, page_num, &window, &page, LCK_write, type, &entry_p, &clump_end))
|
|
{
|
|
CCH_RELEASE(tdbb, &window);
|
|
return false;
|
|
}
|
|
CCH_MARK(tdbb, &window);
|
|
|
|
header_page* header = 0;
|
|
log_info_page* logp = 0;
|
|
USHORT* end_addr;
|
|
if (page_num == HEADER_PAGE) {
|
|
header = (header_page*) page;
|
|
end_addr = &header->hdr_end;
|
|
}
|
|
else {
|
|
logp = (log_info_page*) page;
|
|
end_addr = &logp->log_end;
|
|
}
|
|
|
|
*end_addr -= (2 + entry_p[1]);
|
|
|
|
const UCHAR* r = entry_p + 2 + entry_p[1];
|
|
USHORT l = clump_end - r + 1;
|
|
if (l)
|
|
memmove(entry_p, r, l);
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
void PAG_format_header(thread_db* tdbb)
|
|
{
|
|
/**************************************
|
|
*
|
|
* P A G _ f o r m a t _ h e a d e r
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Create the header page for a new file.
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* const dbb = tdbb->getDatabase();
|
|
CHECK_DBB(dbb);
|
|
|
|
/* Initialize header page */
|
|
|
|
WIN window(HEADER_PAGE_NUMBER);
|
|
header_page* header = (header_page*) CCH_fake(tdbb, &window, 1);
|
|
header->hdr_header.pag_scn = 0;
|
|
*(ISC_TIMESTAMP*) header->hdr_creation_date = Firebird::TimeStamp::getCurrentTimeStamp().value();
|
|
// should we include milliseconds or not?
|
|
//Firebird::TimeStamp::round_time(header->hdr_creation_date->timestamp_time, 0);
|
|
header->hdr_header.pag_type = pag_header;
|
|
header->hdr_page_size = dbb->dbb_page_size;
|
|
header->hdr_ods_version = ODS_VERSION | ODS_FIREBIRD_FLAG;
|
|
header->hdr_implementation = CLASS;
|
|
header->hdr_ods_minor = ODS_CURRENT;
|
|
header->hdr_ods_minor_original = ODS_CURRENT;
|
|
header->hdr_oldest_transaction = 1;
|
|
header->hdr_bumped_transaction = 1;
|
|
header->hdr_end = HDR_SIZE;
|
|
header->hdr_data[0] = HDR_end;
|
|
header->hdr_flags |= hdr_force_write;
|
|
|
|
if (dbb->dbb_flags & DBB_DB_SQL_dialect_3) {
|
|
header->hdr_flags |= hdr_SQL_dialect_3;
|
|
}
|
|
|
|
dbb->dbb_ods_version = header->hdr_ods_version & ~ODS_FIREBIRD_FLAG;
|
|
dbb->dbb_minor_version = header->hdr_ods_minor;
|
|
dbb->dbb_minor_original = header->hdr_ods_minor_original;
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
}
|
|
|
|
|
|
// CVC: This function is mostly obsolete. Ann requested to keep it and the code that calls it.
|
|
// We won't read the log, anyway.
|
|
void PAG_format_log(thread_db* tdbb)
|
|
{
|
|
/***********************************************
|
|
*
|
|
* P A G _ f o r m a t _ l o g
|
|
*
|
|
***********************************************
|
|
*
|
|
* Functional description
|
|
* Initialize log page.
|
|
* Set all parameters to 0
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
|
|
WIN window(LOG_PAGE_NUMBER);
|
|
log_info_page* logp = (log_info_page*) CCH_fake(tdbb, &window, 1);
|
|
logp->log_header.pag_type = pag_log;
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
}
|
|
|
|
|
|
void PAG_format_pip(thread_db* tdbb, PageSpace& pageSpace)
|
|
{
|
|
/**************************************
|
|
*
|
|
* P A G _ f o r m a t _ p i p
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Create a page inventory page to
|
|
* complete the formatting of a new file
|
|
* into a rudimentary database.
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* dbb = tdbb->getDatabase();
|
|
CHECK_DBB(dbb);
|
|
|
|
/* Initialize Page Inventory Page */
|
|
|
|
WIN window(pageSpace.pageSpaceID, 1);
|
|
pageSpace.ppFirst = 1;
|
|
page_inv_page* pages = (page_inv_page*) CCH_fake(tdbb, &window, 1);
|
|
|
|
pages->pip_header.pag_type = pag_pages;
|
|
pages->pip_min = 4;
|
|
pages->pip_header.reserved = pages->pip_min - 1;
|
|
UCHAR* p = pages->pip_bits;
|
|
int i = dbb->dbb_page_size - OFFSETA(page_inv_page*, pip_bits);
|
|
|
|
while (i--) {
|
|
*p++ = 0xff;
|
|
}
|
|
|
|
pages->pip_bits[0] &= ~(1 | 2 | 4);
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
}
|
|
|
|
|
|
bool PAG_get_clump(thread_db* tdbb, SLONG page_num, USHORT type, USHORT* inout_len, UCHAR* entry)
|
|
{
|
|
/***********************************************
|
|
*
|
|
* P A G _ g e t _ c l u m p
|
|
*
|
|
***********************************************
|
|
*
|
|
* Functional description
|
|
* Find 'type' clump in page_num
|
|
* true - Found it
|
|
* false - Not present
|
|
* RETURNS
|
|
* value of clump in entry
|
|
* length in inout_len <-> input and output value to avoid B.O.
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
|
|
WIN window(DB_PAGE_SPACE, page_num);
|
|
|
|
pag* page;
|
|
if (page_num == HEADER_PAGE)
|
|
page = CCH_FETCH(tdbb, &window, LCK_read, pag_header);
|
|
else
|
|
page = CCH_FETCH(tdbb, &window, LCK_read, pag_log);
|
|
|
|
UCHAR* entry_p;
|
|
const UCHAR* dummy;
|
|
if (!find_type(tdbb, page_num, &window, &page, LCK_read, type, &entry_p, &dummy)) {
|
|
CCH_RELEASE(tdbb, &window);
|
|
*inout_len = 0;
|
|
return false;
|
|
}
|
|
|
|
USHORT old_len = *inout_len;
|
|
*inout_len = entry_p[1];
|
|
entry_p += 2;
|
|
|
|
if (*inout_len)
|
|
{
|
|
// Avoid the B.O. but inform the caller the buffer is bigger.
|
|
if (*inout_len < old_len)
|
|
old_len = *inout_len;
|
|
memcpy(entry, entry_p, old_len);
|
|
}
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
void PAG_header(thread_db* tdbb, bool info)
|
|
{
|
|
/**************************************
|
|
*
|
|
* P A G _ h e a d e r
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Checkout database header page.
|
|
* Done through the page cache.
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* const dbb = tdbb->getDatabase();
|
|
|
|
Attachment* attachment = tdbb->getAttachment();
|
|
fb_assert(attachment);
|
|
|
|
WIN window(HEADER_PAGE_NUMBER);
|
|
header_page* header = (header_page*) CCH_FETCH(tdbb, &window, LCK_read, pag_header);
|
|
|
|
try {
|
|
|
|
if (header->hdr_next_transaction) {
|
|
if (header->hdr_oldest_active > header->hdr_next_transaction)
|
|
BUGCHECK(266); /*next transaction older than oldest active */
|
|
|
|
if (header->hdr_oldest_transaction > header->hdr_next_transaction)
|
|
BUGCHECK(267); /* next transaction older than oldest transaction */
|
|
}
|
|
|
|
if (header->hdr_flags & hdr_SQL_dialect_3)
|
|
dbb->dbb_flags |= DBB_DB_SQL_dialect_3;
|
|
|
|
jrd_rel* relation = MET_relation(tdbb, 0);
|
|
RelationPages* relPages = relation->getBasePages();
|
|
if (!relPages->rel_pages) {
|
|
// 21-Dec-2003 Nickolay Samofatov
|
|
// No need to re-set first page for RDB$PAGES relation since
|
|
// current code cannot change its location after database creation.
|
|
// Currently, this change only affects isc_database_info call,
|
|
// the only call which may call PAG_header multiple times.
|
|
// In fact, this isc_database_info behavior seems dangerous to me,
|
|
// but let somebody else fix that problem, I just fix the memory leak.
|
|
vcl* vector = vcl::newVector(*dbb->dbb_permanent, 1);
|
|
relPages->rel_pages = vector;
|
|
(*vector)[0] = header->hdr_PAGES;
|
|
}
|
|
|
|
dbb->dbb_next_transaction = header->hdr_next_transaction;
|
|
|
|
if (!info || dbb->dbb_oldest_transaction < header->hdr_oldest_transaction) {
|
|
dbb->dbb_oldest_transaction = header->hdr_oldest_transaction;
|
|
}
|
|
if (!info || dbb->dbb_oldest_active < header->hdr_oldest_active) {
|
|
dbb->dbb_oldest_active = header->hdr_oldest_active;
|
|
}
|
|
if (!info || dbb->dbb_oldest_snapshot < header->hdr_oldest_snapshot) {
|
|
dbb->dbb_oldest_snapshot = header->hdr_oldest_snapshot;
|
|
}
|
|
|
|
dbb->dbb_attachment_id = header->hdr_attachment_id;
|
|
dbb->dbb_creation_date = *(ISC_TIMESTAMP*) header->hdr_creation_date;
|
|
|
|
if (header->hdr_flags & hdr_read_only) {
|
|
/* If Header Page flag says the database is ReadOnly, gladly accept it. */
|
|
dbb->dbb_flags &= ~DBB_being_opened_read_only;
|
|
dbb->dbb_flags |= DBB_read_only;
|
|
}
|
|
|
|
/* If hdr_read_only is not set... */
|
|
if (!(header->hdr_flags & hdr_read_only) && (dbb->dbb_flags & DBB_being_opened_read_only))
|
|
{
|
|
/* Looks like the Header page says, it is NOT ReadOnly!! But the database
|
|
* file system permission gives only ReadOnly access. Punt out with
|
|
* isc_no_priv error (no privileges)
|
|
*/
|
|
ERR_post(Arg::Gds(isc_no_priv) << Arg::Str("read-write") <<
|
|
Arg::Str("database") <<
|
|
Arg::Str(attachment->att_filename));
|
|
}
|
|
|
|
const bool useFSCache = dbb->dbb_bcb->bcb_count < Config::getMaxFileSystemCache();
|
|
|
|
if ((header->hdr_flags & hdr_force_write) || !useFSCache)
|
|
{
|
|
dbb->dbb_flags |=
|
|
(header->hdr_flags & hdr_force_write ? DBB_force_write : 0) |
|
|
(useFSCache ? 0 : DBB_no_fs_cache);
|
|
|
|
PageSpace* pageSpace = dbb->dbb_page_manager.findPageSpace(DB_PAGE_SPACE);
|
|
for (jrd_file* file = pageSpace->file; file; file = file->fil_next) {
|
|
PIO_force_write(file,
|
|
(dbb->dbb_flags & DBB_force_write) && !(header->hdr_flags & hdr_read_only),
|
|
dbb->dbb_flags & DBB_no_fs_cache);
|
|
}
|
|
}
|
|
|
|
if (header->hdr_flags & hdr_no_reserve)
|
|
dbb->dbb_flags |= DBB_no_reserve;
|
|
|
|
const USHORT sd_flags = header->hdr_flags & hdr_shutdown_mask;
|
|
if (sd_flags) {
|
|
dbb->dbb_ast_flags |= DBB_shutdown;
|
|
if (sd_flags == hdr_shutdown_full)
|
|
dbb->dbb_ast_flags |= DBB_shutdown_full;
|
|
else if (sd_flags == hdr_shutdown_single)
|
|
dbb->dbb_ast_flags |= DBB_shutdown_single;
|
|
}
|
|
|
|
} // try
|
|
catch (const Firebird::Exception&) {
|
|
CCH_RELEASE(tdbb, &window);
|
|
throw;
|
|
}
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
}
|
|
|
|
|
|
void PAG_header_init(thread_db* tdbb)
|
|
{
|
|
/**************************************
|
|
*
|
|
* P A G _ h e a d e r _ i n i t
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Checkout the core part of the database header page.
|
|
* It includes the fields required to setup the I/O layer:
|
|
* ODS version, page size, page buffers.
|
|
* Done using a physical page read.
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* const dbb = tdbb->getDatabase();
|
|
|
|
Attachment* const attachment = tdbb->getAttachment();
|
|
fb_assert(attachment);
|
|
|
|
// Allocate a spare buffer which is large enough,
|
|
// and set up to release it in case of error; note
|
|
// that dbb_page_size has not been set yet, so we
|
|
// can't depend on this.
|
|
//
|
|
// Make sure that buffer is aligned on a page boundary
|
|
// and unit of transfer is a multiple of physical disk
|
|
// sector for raw disk access.
|
|
|
|
SCHAR temp_buffer[2 * MIN_PAGE_SIZE];
|
|
SCHAR* temp_page = (SCHAR *) FB_ALIGN((IPTR) temp_buffer, MIN_PAGE_SIZE);
|
|
|
|
header_page* header = (header_page*) temp_page;
|
|
PIO_header(dbb, temp_page, MIN_PAGE_SIZE);
|
|
|
|
if (header->hdr_header.pag_type != pag_header || header->hdr_sequence) {
|
|
ERR_post(Arg::Gds(isc_bad_db_format) << Arg::Str(attachment->att_filename));
|
|
}
|
|
|
|
const USHORT ods_version = header->hdr_ods_version & ~ODS_FIREBIRD_FLAG;
|
|
|
|
if (!Ods::isSupported(header->hdr_ods_version, header->hdr_ods_minor))
|
|
{
|
|
ERR_post(Arg::Gds(isc_wrong_ods) << Arg::Str(attachment->att_filename) <<
|
|
Arg::Num(ods_version) <<
|
|
Arg::Num(header->hdr_ods_minor) <<
|
|
Arg::Num(ODS_VERSION) <<
|
|
Arg::Num(ODS_CURRENT));
|
|
}
|
|
|
|
// Note that if this check is turned on, it should be recoded in order that
|
|
// the Intel platforms can share databases. At present (Feb 95) it is possible
|
|
// to share databases between Windows and NT, but not with NetWare. Sharing
|
|
// databases with OS/2 is unknown and needs to be investigated. The CLASS was
|
|
// initially 8 for all Intel platforms, but was changed after 4.0 was released
|
|
// in order to allow differentiation between databases created on various
|
|
// platforms. This should allow us in future to identify where databases were
|
|
// created. Even when we get to the stage where databases created on PC platforms
|
|
// are sharable between all platforms, it would be useful to identify where they
|
|
// were created for debugging purposes. - Deej 2/6/95
|
|
//
|
|
// Re-enable and recode the check to avoid BUGCHECK messages when database
|
|
// is accessed with engine built for another architecture. - Nickolay 9-Feb-2005
|
|
|
|
if (header->hdr_implementation != CLASS)
|
|
{
|
|
const int classmax = ods_version < ODS_VERSION11 ? CLASS_MAX10 : CLASS_MAX;
|
|
const ArchitectureType* matrix = ods_version < ODS_VERSION11 ? archMatrix10 : archMatrix;
|
|
const int hdrImpl = header->hdr_implementation;
|
|
if (hdrImpl < 0 || hdrImpl > classmax ||
|
|
matrix[hdrImpl] == archUnknown || matrix[hdrImpl] != matrix[CLASS])
|
|
{
|
|
ERR_post(Arg::Gds(isc_bad_db_format) << Arg::Str(attachment->att_filename));
|
|
}
|
|
}
|
|
|
|
if (header->hdr_page_size < MIN_PAGE_SIZE || header->hdr_page_size > MAX_PAGE_SIZE)
|
|
{
|
|
ERR_post(Arg::Gds(isc_bad_db_format) << Arg::Str(attachment->att_filename));
|
|
}
|
|
|
|
dbb->dbb_ods_version = ods_version;
|
|
dbb->dbb_minor_version = header->hdr_ods_minor;
|
|
dbb->dbb_minor_original = header->hdr_ods_minor_original;
|
|
|
|
dbb->dbb_page_size = header->hdr_page_size;
|
|
dbb->dbb_page_buffers = header->hdr_page_buffers;
|
|
}
|
|
|
|
|
|
void PAG_init(thread_db* tdbb)
|
|
{
|
|
/**************************************
|
|
*
|
|
* P A G _ i n i t
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Initialize stuff for page handling.
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* const dbb = tdbb->getDatabase();
|
|
CHECK_DBB(dbb);
|
|
|
|
PageManager& pageMgr = dbb->dbb_page_manager;
|
|
PageSpace* pageSpace = pageMgr.findPageSpace(DB_PAGE_SPACE);
|
|
fb_assert(pageSpace);
|
|
|
|
pageMgr.bytesBitPIP = dbb->dbb_page_size - OFFSETA(page_inv_page*, pip_bits);
|
|
pageMgr.pagesPerPIP = pageMgr.bytesBitPIP * 8;
|
|
pageMgr.transPerTIP = (dbb->dbb_page_size - OFFSETA(tx_inv_page*, tip_transactions)) * 4;
|
|
pageSpace->ppFirst = 1;
|
|
/* dbb_ods_version can be 0 when a new database is being created */
|
|
if ((dbb->dbb_ods_version == 0) || (dbb->dbb_ods_version >= ODS_VERSION10))
|
|
{
|
|
pageMgr.gensPerPage =
|
|
(dbb->dbb_page_size -
|
|
OFFSETA(generator_page*, gpg_values)) / sizeof(((generator_page*) NULL)->gpg_values);
|
|
}
|
|
else {
|
|
pageMgr.gensPerPage =
|
|
(dbb->dbb_page_size -
|
|
OFFSETA(pointer_page*, ppg_page)) / sizeof(((pointer_page*) NULL)->ppg_page);
|
|
}
|
|
|
|
|
|
/* Compute the number of data pages per pointer page. Each data page
|
|
requires a 32 bit pointer and a 2 bit control field. */
|
|
|
|
dbb->dbb_dp_per_pp =
|
|
(dbb->dbb_page_size - OFFSETA(pointer_page*, ppg_page)) * 8 / (BITS_PER_LONG + 2);
|
|
|
|
/* Compute the number of records that can fit on a page using the
|
|
size of the record index (dpb_repeat) and a record header. This
|
|
gives an artificially high number, reducing the density of db_keys. */
|
|
|
|
dbb->dbb_max_records = (dbb->dbb_page_size - sizeof(data_page)) /
|
|
(sizeof(data_page::dpg_repeat) + OFFSETA(rhd*, rhd_data));
|
|
|
|
// Artifically reduce density of records to test high bits of record number
|
|
// dbb->dbb_max_records = 32000;
|
|
|
|
/* Optimize record numbers for new 64-bit sparse bitmap implementation
|
|
We need to measure if it is beneficial from performance point of view.
|
|
Price is slightly reduced density of record numbers, but for
|
|
ODS11 it doesn't matter because record numbers are 40-bit.
|
|
Benefit is ~1.5 times smaller sparse bitmaps on average and
|
|
faster bitmap iteration. */
|
|
// if (dbb->dbb_ods_version >= ODS_VERSION11)
|
|
// dbb->dbb_max_records = FB_ALIGN(dbb->dbb_max_records, 64);
|
|
|
|
/* Compute the number of index roots that will fit on an index root page,
|
|
assuming that each index has only one key */
|
|
|
|
dbb->dbb_max_idx = (dbb->dbb_page_size - OFFSETA(index_root_page*, irt_rpt)) /
|
|
(sizeof(index_root_page::irt_repeat) + (1 * (dbb->dbb_ods_version >= ODS_VERSION11) ?
|
|
sizeof(irtd) : sizeof(irtd_ods10)));
|
|
|
|
/* Compute prefetch constants from database page size and maximum prefetch
|
|
transfer size. Double pages per prefetch request so that cache reader
|
|
can overlap prefetch I/O with database computation over previously
|
|
prefetched pages. */
|
|
#ifdef SUPERSERVER_V2
|
|
dbb->dbb_prefetch_sequence = PREFETCH_MAX_TRANSFER / dbb->dbb_page_size;
|
|
dbb->dbb_prefetch_pages = dbb->dbb_prefetch_sequence * 2;
|
|
#endif
|
|
}
|
|
|
|
|
|
void PAG_init2(thread_db* tdbb, USHORT shadow_number)
|
|
{
|
|
/**************************************
|
|
*
|
|
* P A G _ i n i t 2
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Perform second phase of page initialization -- the eternal
|
|
* search for additional files.
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* const dbb = tdbb->getDatabase();
|
|
ISC_STATUS* status = tdbb->tdbb_status_vector;
|
|
|
|
/* allocate a spare buffer which is large enough,
|
|
and set up to release it in case of error. Align
|
|
the temporary page buffer for raw disk access. */
|
|
|
|
SCHAR* const temp_buffer = FB_NEW(*getDefaultMemoryPool()) SCHAR[dbb->dbb_page_size + MIN_PAGE_SIZE];
|
|
SCHAR* temp_page =
|
|
(SCHAR*) (((U_IPTR) temp_buffer + MIN_PAGE_SIZE - 1) & ~((U_IPTR) MIN_PAGE_SIZE - 1));
|
|
|
|
try {
|
|
|
|
PageSpace* pageSpace = dbb->dbb_page_manager.findPageSpace(DB_PAGE_SPACE);
|
|
jrd_file* file = pageSpace->file;
|
|
if (shadow_number) {
|
|
Shadow* shadow = dbb->dbb_shadow;
|
|
for (; shadow; shadow = shadow->sdw_next) {
|
|
if (shadow->sdw_number == shadow_number) {
|
|
file = shadow->sdw_file;
|
|
break;
|
|
}
|
|
}
|
|
if (!shadow)
|
|
BUGCHECK(161); /* msg 161 shadow block not found */
|
|
}
|
|
|
|
USHORT sequence = 1;
|
|
WIN window(DB_PAGE_SPACE, -1);
|
|
|
|
TEXT buf[MAXPATHLEN + 1];
|
|
|
|
/* Loop thru files and header pages until everything is open */
|
|
|
|
for (;;)
|
|
{
|
|
TEXT* file_name = NULL;
|
|
window.win_page = file->fil_min_page;
|
|
USHORT file_length = 0;
|
|
ULONG last_page = 0;
|
|
BufferDesc temp_bdb;
|
|
SLONG next_page = 0;
|
|
do {
|
|
/* note that we do not have to get a read lock on
|
|
the header page (except for header page 0) because
|
|
the only time it will be modified is when adding a file,
|
|
which must be done with an exclusive lock on the database --
|
|
if this changes, this policy will have to be reevaluated;
|
|
at any rate there is a problem with getting a read lock
|
|
because the corresponding page in the main database file
|
|
may not exist */
|
|
|
|
if (!file->fil_min_page)
|
|
CCH_FETCH(tdbb, &window, LCK_read, pag_header);
|
|
|
|
header_page* header = (header_page*) temp_page;
|
|
temp_bdb.bdb_buffer = (PAG) header;
|
|
temp_bdb.bdb_page = window.win_page;
|
|
temp_bdb.bdb_dbb = dbb;
|
|
|
|
/* Read the required page into the local buffer */
|
|
PIO_read(file, &temp_bdb, (PAG) header, status);
|
|
|
|
if (shadow_number && !file->fil_min_page)
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
for (const UCHAR* p = header->hdr_data; *p != HDR_end; p += 2 + p[1])
|
|
{
|
|
switch (*p)
|
|
{
|
|
case HDR_file:
|
|
file_length = p[1];
|
|
file_name = buf;
|
|
memcpy(buf, p + 2, file_length);
|
|
break;
|
|
|
|
case HDR_last_page:
|
|
memcpy(&last_page, p + 2, sizeof(last_page));
|
|
break;
|
|
|
|
case HDR_sweep_interval:
|
|
// CVC: Let's copy it always.
|
|
//if (!(dbb->dbb_flags & DBB_read_only))
|
|
memcpy(&dbb->dbb_sweep_interval, p + 2, sizeof(SLONG));
|
|
break;
|
|
}
|
|
}
|
|
|
|
next_page = header->hdr_next_page;
|
|
|
|
if (!shadow_number && !file->fil_min_page)
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
window.win_page = next_page;
|
|
|
|
/*
|
|
* Make sure the header page and all the overflow header
|
|
* pages are traversed. For V4.0, only the header page for
|
|
* the primary database page will have overflow pages.
|
|
*/
|
|
|
|
} while (next_page);
|
|
|
|
if (file->fil_min_page)
|
|
file->fil_fudge = 1;
|
|
if (!file_name)
|
|
break;
|
|
|
|
// Verify database file path against DatabaseAccess entry of firebird.conf
|
|
file_name[file_length] = 0;
|
|
if (!JRD_verify_database_access(file_name)) {
|
|
ERR_post(Arg::Gds(isc_conf_access_denied) << Arg::Str("additional database file") <<
|
|
Arg::Str(file_name));
|
|
}
|
|
|
|
file->fil_next = PIO_open(dbb, file_name, file_name, false);
|
|
file->fil_max_page = last_page;
|
|
file = file->fil_next;
|
|
if (dbb->dbb_flags & (DBB_force_write | DBB_no_fs_cache))
|
|
{
|
|
PIO_force_write(file, dbb->dbb_flags & DBB_force_write, dbb->dbb_flags & DBB_no_fs_cache);
|
|
}
|
|
file->fil_min_page = last_page + 1;
|
|
file->fil_sequence = sequence++;
|
|
}
|
|
|
|
delete[] temp_buffer;
|
|
} // try
|
|
catch (const Firebird::Exception&) {
|
|
delete[] temp_buffer;
|
|
throw;
|
|
}
|
|
}
|
|
|
|
|
|
SLONG PAG_last_page(thread_db* tdbb)
|
|
{
|
|
/**************************************
|
|
*
|
|
* P A G _ l a s t _ p a g e
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Compute the highest page allocated. This is called by the
|
|
* shadow stuff to dump a database.
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* dbb = tdbb->getDatabase();
|
|
CHECK_DBB(dbb);
|
|
|
|
PageManager& pageMgr = dbb->dbb_page_manager;
|
|
PageSpace* pageSpace = pageMgr.findPageSpace(DB_PAGE_SPACE);
|
|
fb_assert(pageSpace);
|
|
|
|
const ULONG pages_per_pip = pageMgr.pagesPerPIP;
|
|
WIN window(DB_PAGE_SPACE, -1);
|
|
|
|
/* Find the last page allocated */
|
|
|
|
ULONG relative_bit = 0;
|
|
USHORT sequence;
|
|
for (sequence = 0; true; ++sequence)
|
|
{
|
|
window.win_page = (!sequence) ? pageSpace->ppFirst : sequence * pages_per_pip - 1;
|
|
const page_inv_page* page = (page_inv_page*) CCH_FETCH(tdbb, &window, LCK_read, pag_pages);
|
|
const UCHAR* bits = page->pip_bits + (pages_per_pip >> 3) - 1;
|
|
while (*bits == (UCHAR) - 1)
|
|
--bits;
|
|
SSHORT bit;
|
|
for (bit = 7; bit >= 0; --bit) {
|
|
if (!(*bits & (1 << bit)))
|
|
break;
|
|
}
|
|
relative_bit = (bits - page->pip_bits) * 8 + bit;
|
|
CCH_RELEASE(tdbb, &window);
|
|
if (relative_bit != pages_per_pip - 1)
|
|
break;
|
|
}
|
|
|
|
return sequence * pages_per_pip + relative_bit;
|
|
}
|
|
|
|
|
|
void PAG_release_page(thread_db* tdbb, const PageNumber& number, const PageNumber& prior_page)
|
|
{
|
|
/**************************************
|
|
*
|
|
* P A G _ r e l e a s e _ p a g e
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Release a page to the free page page.
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* dbb = tdbb->getDatabase();
|
|
CHECK_DBB(dbb);
|
|
|
|
#ifdef VIO_DEBUG
|
|
if (debug_flag > DEBUG_WRITES_INFO)
|
|
printf("\tPAG_release_page: about to release page %"SLONGFORMAT"\n", number.getPageNum());
|
|
#endif
|
|
|
|
PageManager& pageMgr = dbb->dbb_page_manager;
|
|
PageSpace* pageSpace = pageMgr.findPageSpace(number.getPageSpaceID());
|
|
fb_assert(pageSpace);
|
|
|
|
const SLONG sequence = number.getPageNum() / pageMgr.pagesPerPIP;
|
|
const SLONG relative_bit = number.getPageNum() % pageMgr.pagesPerPIP;
|
|
|
|
WIN pip_window(number.getPageSpaceID(), (sequence == 0) ?
|
|
pageSpace->ppFirst : sequence * pageMgr.pagesPerPIP - 1);
|
|
|
|
page_inv_page* pages = (page_inv_page*) CCH_FETCH(tdbb, &pip_window, LCK_write, pag_pages);
|
|
CCH_precedence(tdbb, &pip_window, prior_page);
|
|
CCH_MARK(tdbb, &pip_window);
|
|
pages->pip_bits[relative_bit >> 3] |= 1 << (relative_bit & 7);
|
|
pages->pip_min = MIN(pages->pip_min, relative_bit);
|
|
|
|
CCH_RELEASE(tdbb, &pip_window);
|
|
|
|
pageSpace->pipHighWater = MIN(pageSpace->pipHighWater, sequence);
|
|
}
|
|
|
|
|
|
void PAG_set_force_write(thread_db* tdbb, bool flag)
|
|
{
|
|
/**************************************
|
|
*
|
|
* P A G _ s e t _ f o r c e _ w r i t e
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Turn on/off force write.
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* dbb = tdbb->getDatabase();
|
|
|
|
err_post_if_database_is_readonly(dbb);
|
|
|
|
WIN window(HEADER_PAGE_NUMBER);
|
|
header_page* header = (header_page*) CCH_FETCH(tdbb, &window, LCK_write, pag_header);
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
|
|
|
if (flag) {
|
|
header->hdr_flags |= hdr_force_write;
|
|
dbb->dbb_flags |= DBB_force_write;
|
|
}
|
|
else {
|
|
header->hdr_flags &= ~hdr_force_write;
|
|
dbb->dbb_flags &= ~DBB_force_write;
|
|
}
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
PageSpace* pageSpace = dbb->dbb_page_manager.findPageSpace(DB_PAGE_SPACE);
|
|
for (jrd_file* file = pageSpace->file; file; file = file->fil_next) {
|
|
PIO_force_write(file, flag, dbb->dbb_flags & DBB_no_fs_cache);
|
|
}
|
|
|
|
for (Shadow* shadow = dbb->dbb_shadow; shadow; shadow = shadow->sdw_next) {
|
|
for (jrd_file* file = shadow->sdw_file; file; file = file->fil_next) {
|
|
PIO_force_write(file, flag, dbb->dbb_flags & DBB_no_fs_cache);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void PAG_set_no_reserve(thread_db* tdbb, bool flag)
|
|
{
|
|
/**************************************
|
|
*
|
|
* P A G _ s e t _ n o _ r e s e r v e
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Turn on/off reserving space for versions
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* dbb = tdbb->getDatabase();
|
|
|
|
err_post_if_database_is_readonly(dbb);
|
|
|
|
WIN window(HEADER_PAGE_NUMBER);
|
|
header_page* header = (header_page*) CCH_FETCH(tdbb, &window, LCK_write, pag_header);
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
|
|
|
if (flag) {
|
|
header->hdr_flags |= hdr_no_reserve;
|
|
dbb->dbb_flags |= DBB_no_reserve;
|
|
}
|
|
else {
|
|
header->hdr_flags &= ~hdr_no_reserve;
|
|
dbb->dbb_flags &= ~DBB_no_reserve;
|
|
}
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
}
|
|
|
|
|
|
void PAG_set_db_readonly(thread_db* tdbb, bool flag)
|
|
{
|
|
/*********************************************
|
|
*
|
|
* P A G _ s e t _ d b _ r e a d o n l y
|
|
*
|
|
*********************************************
|
|
*
|
|
* Functional description
|
|
* Set database access mode to readonly OR readwrite
|
|
*
|
|
*********************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* dbb = tdbb->getDatabase();
|
|
|
|
WIN window(HEADER_PAGE_NUMBER);
|
|
header_page* header = (header_page*) CCH_FETCH(tdbb, &window, LCK_write, pag_header);
|
|
|
|
if (!flag) {
|
|
/* If the database is transitioning from RO to RW, reset the
|
|
* in-memory Database flag which indicates that the database is RO.
|
|
* This will allow the CCH subsystem to allow pages to be MARK'ed
|
|
* for WRITE operations
|
|
*/
|
|
header->hdr_flags &= ~hdr_read_only;
|
|
dbb->dbb_flags &= ~DBB_read_only;
|
|
}
|
|
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
|
|
|
if (flag) {
|
|
header->hdr_flags |= hdr_read_only;
|
|
dbb->dbb_flags |= DBB_read_only;
|
|
}
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
}
|
|
|
|
|
|
void PAG_set_db_SQL_dialect(thread_db* tdbb, SSHORT flag)
|
|
{
|
|
/*********************************************
|
|
*
|
|
* P A G _ s e t _ d b _ S Q L _ d i a l e c t
|
|
*
|
|
*********************************************
|
|
*
|
|
* Functional description
|
|
* Set database SQL dialect to SQL_DIALECT_V5 or SQL_DIALECT_V6
|
|
*
|
|
*********************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* dbb = tdbb->getDatabase();
|
|
|
|
const USHORT major_version = dbb->dbb_ods_version;
|
|
const USHORT minor_original = dbb->dbb_minor_original;
|
|
|
|
WIN window(HEADER_PAGE_NUMBER);
|
|
header_page* header = (header_page*) CCH_FETCH(tdbb, &window, LCK_write, pag_header);
|
|
|
|
if (flag && (ENCODE_ODS(major_version, minor_original) >= ODS_10_0))
|
|
{
|
|
switch (flag)
|
|
{
|
|
case SQL_DIALECT_V5:
|
|
|
|
if (dbb->dbb_flags & DBB_DB_SQL_dialect_3 || header->hdr_flags & hdr_SQL_dialect_3)
|
|
{
|
|
// Check the returned value here!
|
|
ERR_post_warning(Arg::Warning(isc_dialect_reset_warning));
|
|
}
|
|
|
|
dbb->dbb_flags &= ~DBB_DB_SQL_dialect_3; /* set to 0 */
|
|
header->hdr_flags &= ~hdr_SQL_dialect_3; /* set to 0 */
|
|
break;
|
|
|
|
case SQL_DIALECT_V6:
|
|
dbb->dbb_flags |= DBB_DB_SQL_dialect_3; /* set to dialect 3 */
|
|
header->hdr_flags |= hdr_SQL_dialect_3; /* set to dialect 3 */
|
|
break;
|
|
|
|
default:
|
|
CCH_RELEASE(tdbb, &window);
|
|
ERR_post(Arg::Gds(isc_inv_dialect_specified) << Arg::Num(flag) <<
|
|
Arg::Gds(isc_valid_db_dialects) << Arg::Str("1 and 3") <<
|
|
Arg::Gds(isc_dialect_not_changed));
|
|
break;
|
|
}
|
|
}
|
|
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
|
|
|
CCH_RELEASE(tdbb, &window);
|
|
}
|
|
|
|
|
|
void PAG_set_page_buffers(thread_db* tdbb, ULONG buffers)
|
|
{
|
|
/**************************************
|
|
*
|
|
* P A G _ s e t _ p a g e _ b u f f e r s
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Set database-specific page buffer cache
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* dbb = tdbb->getDatabase();
|
|
CHECK_DBB(dbb);
|
|
|
|
err_post_if_database_is_readonly(dbb);
|
|
|
|
WIN window(HEADER_PAGE_NUMBER);
|
|
header_page* header = (header_page*) CCH_FETCH(tdbb, &window, LCK_write, pag_header);
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
|
header->hdr_page_buffers = buffers;
|
|
CCH_RELEASE(tdbb, &window);
|
|
}
|
|
|
|
|
|
void PAG_sweep_interval(thread_db* tdbb, SLONG interval)
|
|
{
|
|
/**************************************
|
|
*
|
|
* P A G _ s w e e p _ i n t e r v a l
|
|
*
|
|
**************************************
|
|
*
|
|
* Functional description
|
|
* Set sweep interval.
|
|
*
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
PAG_add_clump(tdbb, HEADER_PAGE, HDR_sweep_interval, sizeof(SLONG),
|
|
(UCHAR*) &interval, CLUMP_REPLACE); //, true
|
|
}
|
|
|
|
|
|
/*
|
|
int PAG_unlicensed(thread_db* tdbb)
|
|
{
|
|
// **************************************
|
|
// *
|
|
// * P A G _ u n l i c e n s e d
|
|
// *
|
|
// **************************************
|
|
// *
|
|
// * Functional description
|
|
// * Log unlicensed activity. Return current count of this
|
|
// * sort of non-sense.
|
|
// *
|
|
// **************************************
|
|
SET_TDBB(tdbb);
|
|
|
|
WIN window(HEADER_PAGE);
|
|
CCH_FETCH(tdbb, &window, LCK_write, pag_header);
|
|
CCH_MARK_MUST_WRITE(tdbb, &window);
|
|
|
|
SLONG count;
|
|
USHORT len = sizeof(count);
|
|
if (PAG_get_clump(tdbb, HEADER_PAGE, HDR_unlicensed, &len, (UCHAR *) &count)) {
|
|
fb_assert(sizeof(count) == len);
|
|
count++;
|
|
PAG_add_clump(tdbb, HEADER_PAGE, HDR_unlicensed, sizeof(count),
|
|
(UCHAR *) &count, CLUMP_REPLACE_ONLY); //, true
|
|
}
|
|
else {
|
|
count = 1;
|
|
PAG_add_clump(tdbb, HEADER_PAGE, HDR_unlicensed, sizeof(count),
|
|
(UCHAR *) &count, CLUMP_REPLACE); //, true
|
|
}
|
|
CCH_RELEASE(tdbb, &window);
|
|
|
|
return count;
|
|
}
|
|
*/
|
|
|
|
|
|
static int blocking_ast_attachment(void* ast_object)
|
|
{
|
|
Attachment* const attachment = static_cast<Attachment*>(ast_object);
|
|
|
|
try
|
|
{
|
|
Database* const dbb = attachment->att_database;
|
|
Database::SyncGuard dsGuard(dbb, true);
|
|
|
|
ThreadContextHolder tdbb;
|
|
tdbb->setDatabase(dbb);
|
|
tdbb->setAttachment(attachment);
|
|
|
|
Jrd::ContextPoolHolder context(tdbb, dbb->dbb_permanent);
|
|
|
|
attachment->att_flags |= ATT_shutdown;
|
|
attachment->cancelExternalConnection(tdbb);
|
|
|
|
LCK_release(tdbb, attachment->att_id_lock);
|
|
}
|
|
catch (const Firebird::Exception&)
|
|
{} // no-op
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
static void find_clump_space(thread_db* tdbb,
|
|
SLONG page_num,
|
|
WIN* window,
|
|
PAG* ppage,
|
|
USHORT type,
|
|
USHORT len,
|
|
const UCHAR* entry) //USHORT must_write
|
|
{
|
|
/***********************************************
|
|
*
|
|
* f i n d _ c l u m p _ s p a c e
|
|
*
|
|
***********************************************
|
|
*
|
|
* Functional description
|
|
* Find space for the new clump.
|
|
* Add the entry at the end of clumplet list.
|
|
* Allocate a new page if required.
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
Database* dbb = tdbb->getDatabase();
|
|
CHECK_DBB(dbb);
|
|
|
|
pag* page = *ppage;
|
|
header_page* header = 0; // used after the loop
|
|
log_info_page* logp = 0; // used after the loop
|
|
|
|
while (true)
|
|
{
|
|
SLONG next_page, free_space;
|
|
USHORT* end_addr;
|
|
UCHAR* p;
|
|
|
|
if (page_num == HEADER_PAGE) {
|
|
header = (header_page*) page;
|
|
next_page = header->hdr_next_page;
|
|
free_space = dbb->dbb_page_size - header->hdr_end;
|
|
end_addr = &header->hdr_end;
|
|
p = (UCHAR *) header + header->hdr_end;
|
|
}
|
|
else {
|
|
logp = (log_info_page*) page;
|
|
next_page = logp->log_next_page;
|
|
free_space = dbb->dbb_page_size - logp->log_end;
|
|
end_addr = &logp->log_end;
|
|
p = (UCHAR *) logp + logp->log_end;
|
|
}
|
|
|
|
if (free_space > (2 + len))
|
|
{
|
|
//if (must_write)
|
|
CCH_MARK_MUST_WRITE(tdbb, window);
|
|
//else
|
|
// CCH_MARK(tdbb, window);
|
|
|
|
fb_assert(type <= MAX_UCHAR);
|
|
fb_assert(len <= MAX_UCHAR);
|
|
*p++ = static_cast<UCHAR>(type);
|
|
*p++ = static_cast<UCHAR>(len);
|
|
|
|
if (len)
|
|
{
|
|
memcpy(p, entry, len);
|
|
p += len;
|
|
}
|
|
|
|
*p = HDR_end;
|
|
|
|
*end_addr = (USHORT) (p - (UCHAR *) page);
|
|
return;
|
|
}
|
|
|
|
if (!next_page)
|
|
break;
|
|
|
|
/* Follow chain of header pages */
|
|
|
|
if (page_num == HEADER_PAGE)
|
|
*ppage = page = CCH_HANDOFF(tdbb, window, next_page, LCK_write, pag_header);
|
|
else
|
|
*ppage = page = CCH_HANDOFF(tdbb, window, next_page, LCK_write, pag_log);
|
|
}
|
|
|
|
WIN new_window(DB_PAGE_SPACE, -1);
|
|
pag* new_page = (PAG) DPM_allocate(tdbb, &new_window);
|
|
|
|
//if (must_write)
|
|
CCH_MARK_MUST_WRITE(tdbb, &new_window);
|
|
//else
|
|
// CCH_MARK(tdbb, &new_window);
|
|
|
|
|
|
header_page* new_header = 0;
|
|
log_info_page* new_logp = 0;
|
|
SLONG next_page;
|
|
USHORT* end_addr;
|
|
UCHAR* p;
|
|
if (page_num == HEADER_PAGE) {
|
|
new_header = (header_page*) new_page;
|
|
new_header->hdr_header.pag_type = pag_header;
|
|
new_header->hdr_end = HDR_SIZE;
|
|
new_header->hdr_page_size = dbb->dbb_page_size;
|
|
new_header->hdr_data[0] = HDR_end;
|
|
next_page = new_window.win_page.getPageNum();
|
|
end_addr = &new_header->hdr_end;
|
|
p = new_header->hdr_data;
|
|
}
|
|
else {
|
|
new_logp = (log_info_page*) new_page;
|
|
new_logp->log_header.pag_type = pag_log;
|
|
new_logp->log_data[0] = LOG_end;
|
|
new_logp->log_end = LIP_SIZE;
|
|
next_page = new_window.win_page.getPageNum();
|
|
end_addr = &new_logp->log_end;
|
|
p = new_logp->log_data;
|
|
}
|
|
|
|
fb_assert(type <= MAX_UCHAR);
|
|
fb_assert(len <= MAX_UCHAR);
|
|
*p++ = static_cast<UCHAR>(type);
|
|
*p++ = static_cast<UCHAR>(len);
|
|
|
|
if (len)
|
|
{
|
|
memcpy(p, entry, len);
|
|
p += len;
|
|
}
|
|
|
|
*p = HDR_end;
|
|
*end_addr = (USHORT) (p - (UCHAR *) new_page);
|
|
|
|
CCH_RELEASE(tdbb, &new_window);
|
|
|
|
CCH_precedence(tdbb, window, next_page);
|
|
|
|
CCH_MARK(tdbb, window);
|
|
|
|
if (page_num == HEADER_PAGE)
|
|
header->hdr_next_page = next_page;
|
|
else
|
|
logp->log_next_page = next_page;
|
|
}
|
|
|
|
|
|
static bool find_type(thread_db* tdbb,
|
|
SLONG page_num,
|
|
WIN* window,
|
|
PAG* ppage,
|
|
USHORT lock,
|
|
USHORT type,
|
|
UCHAR** entry_p,
|
|
const UCHAR** clump_end)
|
|
{
|
|
/***********************************************
|
|
*
|
|
* f i n d _ t y p e
|
|
*
|
|
***********************************************
|
|
*
|
|
* Functional description
|
|
* Find the requested type in a page.
|
|
* RETURNS
|
|
* pointer to type, pointer to end of page, header.
|
|
* true - Found it
|
|
* false - Not present
|
|
*
|
|
**************************************/
|
|
SET_TDBB(tdbb);
|
|
|
|
while (true)
|
|
{
|
|
header_page* header = 0;
|
|
log_info_page* logp = 0;
|
|
UCHAR* p;
|
|
SLONG next_page;
|
|
if (page_num == HEADER_PAGE) {
|
|
header = (header_page*) (*ppage);
|
|
p = header->hdr_data;
|
|
next_page = header->hdr_next_page;
|
|
}
|
|
else {
|
|
logp = (log_info_page*) (*ppage);
|
|
p = logp->log_data;
|
|
next_page = logp->log_next_page;
|
|
}
|
|
|
|
UCHAR* q = 0;
|
|
for (; (*p != HDR_end); p += 2 + p[1]) {
|
|
if (*p == type)
|
|
q = p;
|
|
}
|
|
|
|
if (q) {
|
|
*entry_p = q;
|
|
*clump_end = p;
|
|
return true;
|
|
}
|
|
|
|
/* Follow chain of pages */
|
|
|
|
if (next_page) {
|
|
if (page_num == HEADER_PAGE) {
|
|
*ppage = CCH_HANDOFF(tdbb, window, next_page, lock, pag_header);
|
|
}
|
|
else {
|
|
*ppage = CCH_HANDOFF(tdbb, window, next_page, lock, pag_log);
|
|
}
|
|
}
|
|
else
|
|
return false;
|
|
}
|
|
}
|
|
|
|
PageSpace::~PageSpace()
|
|
{
|
|
if (file) {
|
|
PIO_close(file);
|
|
|
|
while (file)
|
|
{
|
|
jrd_file* next = file->fil_next;
|
|
delete file;
|
|
file = next;
|
|
}
|
|
}
|
|
}
|
|
|
|
ULONG PageSpace::actAlloc(const USHORT pageSize)
|
|
{
|
|
/**************************************
|
|
*
|
|
* Functional description
|
|
* Compute actual number of physically allocated pages of database.
|
|
*
|
|
**************************************/
|
|
|
|
// Traverse the linked list of files and add up the
|
|
// number of pages in each file
|
|
ULONG tot_pages = 0;
|
|
for (const jrd_file* f = file; f != NULL; f = f->fil_next) {
|
|
tot_pages += PIO_get_number_of_pages(f, pageSize);
|
|
}
|
|
|
|
return tot_pages;
|
|
}
|
|
|
|
ULONG PageSpace::actAlloc(const Database* dbb)
|
|
{
|
|
PageSpace* pgSpace = dbb->dbb_page_manager.findPageSpace(DB_PAGE_SPACE);
|
|
return pgSpace->actAlloc(dbb->dbb_page_size);
|
|
}
|
|
|
|
ULONG PageSpace::maxAlloc(const USHORT pageSize)
|
|
{
|
|
/**************************************
|
|
*
|
|
* Functional description
|
|
* Compute last physically allocated page of database.
|
|
*
|
|
**************************************/
|
|
const jrd_file* f = file;
|
|
while (f->fil_next) {
|
|
f = f->fil_next;
|
|
}
|
|
|
|
const ULONG nPages = f->fil_min_page - f->fil_fudge + PIO_get_number_of_pages(f, pageSize);
|
|
|
|
if (maxPageNumber < nPages)
|
|
maxPageNumber = nPages;
|
|
|
|
return nPages;
|
|
}
|
|
|
|
ULONG PageSpace::maxAlloc(const Database* dbb)
|
|
{
|
|
PageSpace* pgSpace = dbb->dbb_page_manager.findPageSpace(DB_PAGE_SPACE);
|
|
return pgSpace->maxAlloc(dbb->dbb_page_size);
|
|
}
|
|
|
|
bool PageSpace::extend(thread_db* tdbb, const ULONG pageNum)
|
|
{
|
|
/**************************************
|
|
*
|
|
* Functional description
|
|
* Extend database file(s) up to at least pageNum pages. Number of pages to
|
|
* extend can't be less than hardcoded value MIN_EXTEND_BYTES and more than
|
|
* configured value "DatabaseGrowthIncrement" (both values in bytes).
|
|
*
|
|
* If "DatabaseGrowthIncrement" is less than MIN_EXTEND_BYTES then don't
|
|
* extend file(s)
|
|
*
|
|
**************************************/
|
|
const int MAX_EXTEND_BYTES = Config::getDatabaseGrowthIncrement();
|
|
|
|
if (pageNum < maxPageNumber || MAX_EXTEND_BYTES < MIN_EXTEND_BYTES)
|
|
return true;
|
|
|
|
Database* dbb = tdbb->getDatabase();
|
|
|
|
if (pageNum >= maxAlloc(dbb->dbb_page_size))
|
|
{
|
|
const ULONG minExtendPages = MIN_EXTEND_BYTES / dbb->dbb_page_size;
|
|
const ULONG maxExtendPages = MAX_EXTEND_BYTES / dbb->dbb_page_size;
|
|
const ULONG reqPages = pageNum - maxPageNumber + 1;
|
|
|
|
ULONG extPages;
|
|
extPages = MIN(MAX(maxPageNumber / 16, minExtendPages), maxExtendPages);
|
|
extPages = MAX(reqPages, extPages);
|
|
|
|
while (true)
|
|
{
|
|
try
|
|
{
|
|
PIO_extend(dbb, file, extPages, dbb->dbb_page_size);
|
|
break;
|
|
}
|
|
catch (Firebird::status_exception)
|
|
{
|
|
if (extPages > reqPages)
|
|
{
|
|
extPages = MAX(reqPages, extPages / 2);
|
|
fb_utils::init_status(tdbb->tdbb_status_vector);
|
|
}
|
|
else
|
|
{
|
|
gds__log("Error extending file \"%s\" by %lu page(s).\nCurrently allocated %lu pages, requested page number %lu",
|
|
file->fil_string, extPages, maxPageNumber, pageNum);
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
maxPageNumber = 0;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
PageSpace* PageManager::addPageSpace(const USHORT pageSpaceID)
|
|
{
|
|
PageSpace* newPageSpace = findPageSpace(pageSpaceID);
|
|
if (!newPageSpace)
|
|
{
|
|
newPageSpace = FB_NEW(pool) PageSpace(pageSpaceID);
|
|
pageSpaces.add(newPageSpace);
|
|
}
|
|
|
|
return newPageSpace;
|
|
}
|
|
|
|
PageSpace* PageManager::findPageSpace(const USHORT pageSpace) const
|
|
{
|
|
size_t pos;
|
|
if (pageSpaces.find(pageSpace, pos)) {
|
|
return pageSpaces[pos];
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void PageManager::delPageSpace(const USHORT pageSpace)
|
|
{
|
|
size_t pos;
|
|
if (pageSpaces.find(pageSpace, pos))
|
|
{
|
|
PageSpace* pageSpaceToDelete = pageSpaces[pos];
|
|
pageSpaces.remove(pos);
|
|
delete pageSpaceToDelete;
|
|
}
|
|
}
|
|
|
|
void PageManager::closeAll()
|
|
{
|
|
for (size_t i = 0; i < pageSpaces.getCount(); i++)
|
|
{
|
|
if (pageSpaces[i]->file) {
|
|
PIO_close(pageSpaces[i]->file);
|
|
}
|
|
}
|
|
}
|
|
|
|
void PageManager::releaseLocks()
|
|
{
|
|
#ifdef WIN_NT
|
|
for (size_t i = 0; i < pageSpaces.getCount(); i++)
|
|
{
|
|
if (pageSpaces[i]->file && pageSpaces[i]->file->fil_ext_lock) {
|
|
delete pageSpaces[i]->file->fil_ext_lock;
|
|
pageSpaces[i]->file->fil_ext_lock = NULL;
|
|
}
|
|
}
|
|
#endif
|
|
}
|
|
|
|
USHORT PageManager::getTempPageSpaceID(thread_db* tdbb)
|
|
{
|
|
USHORT result;
|
|
#ifdef SUPERSERVER
|
|
result = TEMP_PAGE_SPACE;
|
|
#else
|
|
SET_TDBB(tdbb);
|
|
Database* dbb = tdbb->getDatabase();
|
|
Attachment* att = tdbb->getAttachment();
|
|
if (!att->att_temp_pg_lock)
|
|
{
|
|
Lock* lock = FB_NEW_RPT(*dbb->dbb_permanent, sizeof(USHORT)) Lock();
|
|
lock->lck_type = LCK_page_space;
|
|
lock->lck_owner_handle = LCK_get_owner_handle(tdbb, lock->lck_type);
|
|
lock->lck_parent = dbb->dbb_lock;
|
|
lock->lck_length = sizeof(USHORT);
|
|
lock->lck_dbb = dbb;
|
|
|
|
PAG_attachment_id(tdbb);
|
|
|
|
while (true)
|
|
{
|
|
const double tmp = rand() * (MAX_USHORT - TEMP_PAGE_SPACE - 1.0) / (RAND_MAX + 1.0);
|
|
lock->lck_key.lck_long = static_cast<SLONG>(tmp) + TEMP_PAGE_SPACE + 1;
|
|
if (LCK_lock(tdbb, lock, LCK_write, LCK_NO_WAIT))
|
|
break;
|
|
}
|
|
|
|
att->att_temp_pg_lock = lock;
|
|
}
|
|
|
|
result = (USHORT) att->att_temp_pg_lock->lck_key.lck_long;
|
|
#endif
|
|
|
|
if (!this->findPageSpace(result)) {
|
|
PAG_attach_temp_pages(tdbb, result);
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
ULONG PAG_page_count(Database* database, PageCountCallback* cb)
|
|
{
|
|
/*********************************************
|
|
*
|
|
* P A G _ p a g e _ c o u n t
|
|
*
|
|
*********************************************
|
|
*
|
|
* Functional description
|
|
* Count pages, used by database
|
|
*
|
|
*********************************************/
|
|
fb_assert(cb);
|
|
|
|
const bool isODS11_x =
|
|
(database->dbb_ods_version == ODS_VERSION11 && database->dbb_minor_version >= 1);
|
|
if (!isODS11_x) {
|
|
return 0;
|
|
}
|
|
|
|
Firebird::Array<BYTE> temp;
|
|
page_inv_page* pip = (Ods::page_inv_page*) // can't reinterpret_cast<> here
|
|
FB_ALIGN((IPTR) temp.getBuffer(database->dbb_page_size + MIN_PAGE_SIZE), MIN_PAGE_SIZE);
|
|
|
|
PageSpace* pageSpace = database->dbb_page_manager.findPageSpace(DB_PAGE_SPACE);
|
|
fb_assert(pageSpace);
|
|
|
|
ULONG pageNo = pageSpace->ppFirst;
|
|
const ULONG pagesPerPip = database->dbb_page_manager.pagesPerPIP;
|
|
|
|
for (ULONG sequence = 0; true; pageNo = (pagesPerPip * ++sequence) - 1)
|
|
{
|
|
cb->newPage(pageNo, &pip->pip_header);
|
|
fb_assert(pip->pip_header.pag_type == pag_pages);
|
|
if (pip->pip_header.reserved == pagesPerPip)
|
|
{
|
|
// this is not last page, continue search
|
|
continue;
|
|
}
|
|
|
|
return pip->pip_header.reserved + pageNo + (sequence ? 1 : -1);
|
|
}
|
|
|
|
// compiler warnings silencer
|
|
return 0;
|
|
}
|