/* * 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 { 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 { 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 { 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, 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 PageStack; } //namespace Jrd #endif // JRD_PAG_H