8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-25 00:03:03 +01:00
firebird-mirror/src/jrd/pag.h

333 lines
7.9 KiB
C++

/*
* PROGRAM: JRD Access Method
* MODULE: pag.h
* DESCRIPTION: Page interface definitions
*
* 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): ______________________________________.
*/
/*
* 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: Add pgc_gpg, number of generators per page,
* for use in DPM_gen_id.
*/
#ifndef JRD_PAG_H
#define JRD_PAG_H
#include "../include/fb_blk.h"
#include "../common/classes/array.h"
#include "../common/classes/locks.h"
#include "../jrd/ods.h"
#include "../jrd/lls.h"
namespace Jrd {
// Page control block -- used by PAG to keep track of critical constants
/**
class PageControl : public pool_alloc<type_pgc>
{
public:
SLONG pgc_high_water; // Lowest PIP with space
SLONG pgc_ppp; // Pages per pip
SLONG pgc_pip; // First pointer page
ULONG pgc_bytes; // Number of bytes of bit in PIP
ULONG pgc_tpt; // Transactions per TIP
ULONG pgc_gpg; // Generators per generator page
};
**/
// page spaces below TRANS_PAGE_SPACE contain regular database pages
// TEMP_PAGE_SPACE and page spaces above TEMP_PAGE_SPACE contain temporary pages
// TRANS_PAGE_SPACE is pseudo space to store transaction numbers in precedence stack
// INVALID_PAGE_SPACE is to ???
const USHORT INVALID_PAGE_SPACE = 0;
const USHORT DB_PAGE_SPACE = 1;
const USHORT TRANS_PAGE_SPACE = 255;
const USHORT TEMP_PAGE_SPACE = 256;
const USHORT PAGES_IN_EXTENT = 8;
class jrd_file;
class Database;
class thread_db;
class PageManager;
class PageSpace : public pool_alloc<type_PageSpace>
{
public:
explicit PageSpace(Database* aDbb, USHORT aPageSpaceID)
{
pageSpaceID = aPageSpaceID;
pipHighWater = 0;
pipWithExtent = 0;
pipFirst = 0;
scnFirst = 0;
file = 0;
dbb = aDbb;
maxPageNumber = 0;
pipMaxKnown = 0;
}
~PageSpace();
USHORT pageSpaceID;
Firebird::AtomicCounter pipHighWater; // Lowest PIP with space
Firebird::AtomicCounter pipWithExtent; // Lowest PIP with free extent
ULONG pipFirst; // First pointer page
ULONG scnFirst; // First SCN's page
jrd_file* file;
static inline bool isTemporary(USHORT aPageSpaceID)
{
return (aPageSpaceID >= TEMP_PAGE_SPACE);
}
inline bool isTemporary() const
{
return isTemporary(pageSpaceID);
}
static inline SLONG generate(const PageSpace* Item)
{
return Item->pageSpaceID;
}
// how many pages allocated
ULONG actAlloc();
static ULONG actAlloc(const Database* dbb);
// number of last allocated page
ULONG maxAlloc();
static ULONG maxAlloc(const Database* dbb);
// number of last used page
ULONG lastUsedPage();
static ULONG lastUsedPage(const Database* dbb);
// number of used pages
ULONG usedPages();
static ULONG usedPages(const Database* dbb);
// extend page space
bool extend(thread_db*, const ULONG, const bool);
// get SCN's page number
ULONG getSCNPageNum(ULONG sequence);
static ULONG getSCNPageNum(const Database* dbb, ULONG sequence);
// is pagespace on raw device
bool onRawDevice() const;
private:
ULONG maxPageNumber;
Database* dbb;
ULONG pipMaxKnown;
};
class PageManager : public pool_alloc<type_PageManager>
{
public:
explicit PageManager(Database* aDbb, Firebird::MemoryPool& aPool) :
dbb(aDbb),
pageSpaces(aPool),
pool(aPool)
{
pagesPerPIP = 0;
bytesBitPIP = 0;
transPerTIP = 0;
gensPerPage = 0;
pagesPerSCN = 0;
tempPageSpaceID = 0;
tempFileCreated = false;
addPageSpace(DB_PAGE_SPACE);
}
~PageManager()
{
while (pageSpaces.hasData())
delete pageSpaces.pop();
}
PageSpace* findPageSpace(const USHORT pageSpaceID) const;
void initTempPageSpace(thread_db* tdbb);
USHORT getTempPageSpaceID(thread_db* tdbb);
void closeAll();
ULONG pagesPerPIP; // Pages per pip
ULONG bytesBitPIP; // Number of bytes of bit in PIP
ULONG transPerTIP; // Transactions per TIP
ULONG gensPerPage; // Generators per generator page
ULONG pagesPerSCN; // Slots per SCN's page
private:
typedef Firebird::SortedArray<PageSpace*, Firebird::EmptyStorage<PageSpace*>,
USHORT, PageSpace> PageSpaceArray;
PageSpace* addPageSpace(const USHORT pageSpaceID);
void delPageSpace(const USHORT pageSpaceID);
Database* dbb;
PageSpaceArray pageSpaces;
Firebird::MemoryPool& pool;
Firebird::Mutex initTmpMtx;
USHORT tempPageSpaceID;
bool tempFileCreated;
};
class PageNumber
{
public:
// CVC: To be completely in sync, the second param would have to be TraNumber
inline PageNumber(const USHORT aPageSpace, const ULONG aPageNum)
: pageNum(aPageNum), pageSpaceID(aPageSpace)
{
// Some asserts are commented cause 0 was also used as 'does not matter' pagespace
// fb_assert(pageSpaceID != INVALID_PAGE_SPACE);
}
// Required to be able to keep it in Firebird::Stack
inline PageNumber()
: pageNum(0), pageSpaceID(INVALID_PAGE_SPACE)
{ }
inline PageNumber(const PageNumber& from)
: pageNum(from.pageNum), pageSpaceID(from.pageSpaceID)
{ }
inline ULONG getPageNum() const
{
// fb_assert(pageSpaceID != INVALID_PAGE_SPACE);
return pageNum;
}
inline USHORT getPageSpaceID() const
{
fb_assert(pageSpaceID != INVALID_PAGE_SPACE);
return pageSpaceID;
}
inline USHORT setPageSpaceID(const USHORT aPageSpaceID)
{
fb_assert(aPageSpaceID != INVALID_PAGE_SPACE);
pageSpaceID = aPageSpaceID;
return pageSpaceID;
}
inline bool isTemporary() const
{
fb_assert(pageSpaceID != INVALID_PAGE_SPACE);
return PageSpace::isTemporary(pageSpaceID);
}
inline static USHORT getLockLen()
{
return 2 * sizeof(ULONG);
}
inline void getLockStr(UCHAR* str) const
{
fb_assert(pageSpaceID != INVALID_PAGE_SPACE);
memcpy(str, &pageNum, sizeof(ULONG));
str += sizeof(ULONG);
const ULONG val = pageSpaceID;
memcpy(str, &val, sizeof(ULONG));
}
inline PageNumber& operator=(const PageNumber& from)
{
pageSpaceID = from.pageSpaceID;
// fb_assert(pageSpaceID != INVALID_PAGE_SPACE);
pageNum = from.pageNum;
return *this;
}
inline ULONG operator=(const ULONG from)
{
// fb_assert(pageSpaceID != INVALID_PAGE_SPACE);
pageNum = from;
return pageNum;
}
inline bool operator==(const PageNumber& other) const
{
return (pageNum == other.pageNum) && (pageSpaceID == other.pageSpaceID);
}
inline bool operator!=(const PageNumber& other) const
{
return !(*this == other);
}
inline bool operator>(const PageNumber& other) const
{
fb_assert(pageSpaceID != INVALID_PAGE_SPACE);
fb_assert(other.pageSpaceID != INVALID_PAGE_SPACE);
return (pageSpaceID > other.pageSpaceID) ||
((pageSpaceID == other.pageSpaceID) && (pageNum > other.pageNum));
}
inline bool operator>=(const PageNumber& other) const
{
fb_assert(pageSpaceID != INVALID_PAGE_SPACE);
fb_assert(other.pageSpaceID != INVALID_PAGE_SPACE);
return (pageSpaceID > other.pageSpaceID) ||
((pageSpaceID == other.pageSpaceID) && (pageNum >= other.pageNum));
}
inline bool operator<(const PageNumber& other) const
{
return !(*this >= other);
}
inline bool operator<=(const PageNumber& other) const
{
return !(*this > other);
}
/*
inline operator ULONG() const
{
return pageNum;
}
*/
private:
ULONG pageNum;
USHORT pageSpaceID;
};
const PageNumber ZERO_PAGE_NUMBER(DB_PAGE_SPACE, 0);
const PageNumber HEADER_PAGE_NUMBER(DB_PAGE_SPACE, HEADER_PAGE);
typedef Firebird::Stack<PageNumber> PageStack;
} //namespace Jrd
#endif // JRD_PAG_H