8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-30 19:23:03 +01:00
firebird-mirror/src/jrd/dpm.epp

3431 lines
95 KiB
Plaintext
Raw Normal View History

2001-05-23 15:26:42 +02:00
/*
* PROGRAM: JRD Access Method
* MODULE: dpm.epp
2001-05-23 15:26:42 +02:00
* DESCRIPTION: Data page 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-10-30 07:40:58 +01:00
*
* 2002.10.29 Sean Leyne - Removed obsolete "Netware" port
*
* 2002.10.30 Sean Leyne - Removed support for obsolete "PC_PLATFORM" define
*
2001-05-23 15:26:42 +02:00
*/
/*
* 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: Corrected routine to use new variables from PAG_init.
*/
#include "firebird.h"
2004-03-22 12:38:23 +01:00
#include "../jrd/common.h"
2004-04-29 00:43:34 +02:00
#include <stdio.h>
2001-05-23 15:26:42 +02:00
#include <string.h>
#include "../jrd/jrd.h"
#include "../jrd/ods.h"
#include "../jrd/req.h"
2003-11-08 17:40:17 +01:00
#include "../jrd/ibase.h"
2001-05-23 15:26:42 +02:00
#include "../jrd/sqz.h"
#include "../jrd/irq.h"
#include "../jrd/blb.h"
#include "../jrd/tra.h"
#include "../jrd/lls.h"
#include "../jrd/lck.h"
#include "../jrd/cch.h"
#include "../jrd/pag.h"
#include "../jrd/rse.h"
2006-05-31 10:29:29 +02:00
#include "../jrd/val.h"
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
#include "../jrd/vio_debug.h"
#endif
#include "../jrd/cch_proto.h"
#include "../jrd/cmp_proto.h"
#include "../jrd/dpm_proto.h"
#include "../jrd/err_proto.h"
#include "../jrd/exe_proto.h"
#include "../jrd/met_proto.h"
#include "../jrd/mov_proto.h"
#include "../jrd/pag_proto.h"
#include "../jrd/sqz_proto.h"
#include "../common/StatusArg.h"
2001-05-23 15:26:42 +02:00
#ifdef DEV_BUILD
#include "../jrd/dbg_proto.h"
#endif
2001-05-23 15:26:42 +02:00
DATABASE DB = FILENAME "ODS.RDB";
#define DECOMPOSE(n, divisor, q, r) {r = n % divisor; q = n / divisor;}
2009-06-25 12:59:10 +02:00
//#define DECOMPOSE_QUOTIENT(n, divisor, q) {q = n / divisor;}
2004-02-02 12:02:12 +01:00
#define HIGH_WATER(x) ((SSHORT) sizeof (data_page) + (SSHORT) sizeof (data_page::dpg_repeat) * (x - 1))
2001-05-23 15:26:42 +02:00
#define SPACE_FUDGE RHDF_SIZE
using namespace Jrd;
using namespace Ods;
using namespace Firebird;
static void check_swept(thread_db*, record_param*);
2006-05-22 00:07:35 +02:00
static void delete_tail(thread_db*, rhdf*, const USHORT, USHORT);
static void fragment(thread_db*, record_param*, SSHORT, DataComprControl*, SSHORT, const jrd_tra*);
static void extend_relation(thread_db*, jrd_rel*, WIN*, USHORT);
static UCHAR* find_space(thread_db*, record_param*, SSHORT, PageStack&, Record*, USHORT);
2009-08-20 13:18:57 +02:00
static bool get_header(WIN*, SSHORT, record_param*);
static pointer_page* get_pointer_page(thread_db*, jrd_rel*, RelationPages*, WIN*, USHORT, USHORT);
static rhd* locate_space(thread_db*, record_param*, SSHORT, PageStack&, Record*, USHORT);
static void mark_full(thread_db*, record_param*);
static void store_big_record(thread_db*, record_param*, PageStack&, DataComprControl*, USHORT, USHORT);
2001-05-23 15:26:42 +02:00
PAG DPM_allocate(thread_db* tdbb, WIN* window)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ a l l o c a t e
*
**************************************
*
* Functional description
* Allocate a data page.
2001-05-23 15:26:42 +02:00
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf("DPM_allocate (window page %"SLONGFORMAT")\n",
2006-05-22 00:07:35 +02:00
window ? window->win_page.getPageNum() : 0);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
pag* page = PAG_allocate(tdbb, window);
2001-05-23 15:26:42 +02:00
return page;
}
void DPM_backout( thread_db* tdbb, record_param* rpb)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ b a c k o u t
*
**************************************
*
* Functional description
* Backout a record where the record and previous version are on
* the same page.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_WRITES)
printf("DPM_backout (record_param %"QUADFORMAT"d)\n", rpb->rpb_number.getValue());
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf
2003-04-01 19:58:19 +02:00
(" record %"SLONGFORMAT":%d transaction %"SLONGFORMAT" back %"
SLONGFORMAT":%d fragment %"SLONGFORMAT":%d flags %d\n",
rpb->rpb_page, rpb->rpb_line, rpb->rpb_transaction_nr,
2001-05-23 15:26:42 +02:00
rpb->rpb_b_page, rpb->rpb_b_line, rpb->rpb_f_page,
rpb->rpb_f_line, rpb->rpb_flags);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
2006-05-22 00:07:35 +02:00
CCH_MARK(tdbb, &rpb->getWindow(tdbb));
data_page* page = (data_page*) rpb->getWindow(tdbb).win_buffer;
data_page::dpg_repeat* index1 = page->dpg_rpt + rpb->rpb_line;
2004-02-02 12:02:12 +01:00
data_page::dpg_repeat* index2 = page->dpg_rpt + rpb->rpb_b_line;
2001-05-23 15:26:42 +02:00
*index1 = *index2;
index2->dpg_offset = index2->dpg_length = 0;
rhd* header = (rhd*) ((SCHAR *) page + index1->dpg_offset);
2001-05-23 15:26:42 +02:00
header->rhd_flags &= ~(rhd_chain | rhd_gc_active);
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf
2003-04-01 19:58:19 +02:00
(" old record %"SLONGFORMAT":%d, new record %"SLONGFORMAT
":%d, old dpg_count %d, ",
rpb->rpb_page, rpb->rpb_line, rpb->rpb_b_page,
2003-04-01 19:58:19 +02:00
rpb->rpb_b_line, page->dpg_count);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
2009-08-20 13:18:57 +02:00
// Check to see if the index got shorter
2003-12-31 06:36:12 +01:00
USHORT n;
2009-06-23 06:12:59 +02:00
for (n = page->dpg_count; --n;)
{
2001-05-23 15:26:42 +02:00
if (page->dpg_rpt[n].dpg_length)
break;
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
page->dpg_count = n + 1;
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_WRITES_INFO)
2004-04-29 00:43:34 +02:00
printf(" new dpg_count %d\n", page->dpg_count);
2001-05-23 15:26:42 +02:00
#endif
fb_assert((page->dpg_header.pag_flags & dpg_swept) == 0);
2006-05-22 00:07:35 +02:00
CCH_RELEASE(tdbb, &rpb->getWindow(tdbb));
2001-05-23 15:26:42 +02:00
}
2005-03-29 11:43:31 +02:00
double DPM_cardinality(thread_db* tdbb, jrd_rel* relation, const Format* format)
{
/**************************************
*
* D P M _ c a r d i n a l i t y
*
**************************************
*
* Functional description
* Estimate cardinality for the given relation.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2005-03-29 11:43:31 +02:00
// Estimated number of total records for this relation,
2005-03-29 11:43:31 +02:00
// we assume that the records are compressed to 50%
// Every record has also a header and a jump section (13 + 4)
USHORT minRecordSize = sizeof(Ods::data_page::dpg_repeat) + RHD_SIZE;
if (!(dbb->dbb_flags & DBB_no_reserve)) {
minRecordSize += RHDF_SIZE;
}
// Get the number of data-pages for this relation
SLONG dataPages = DPM_data_pages(tdbb, relation);
// AB: If we have only 1 data-page then the cardinality calculation
2009-06-23 06:12:59 +02:00
// is to worse to be useful, therefore rely on the record count
2005-03-29 11:43:31 +02:00
// from the data-page.
if (dataPages == 1)
{
2006-05-22 00:07:35 +02:00
RelationPages* relPages = relation->getPages(tdbb);
vcl* vector = relPages->rel_pages;
2009-06-23 06:12:59 +02:00
if (vector)
{
2006-05-22 00:07:35 +02:00
WIN window(relPages->rel_pg_space_id, (*vector)[0]);
Ods::pointer_page* ppage =
2005-03-29 11:43:31 +02:00
(Ods::pointer_page*) CCH_FETCH(tdbb, &window, LCK_read, pag_pointer);
USHORT recordCount = 0;
const SLONG* page = ppage->ppg_page;
2009-06-23 06:12:59 +02:00
if (*page)
{
2005-03-29 11:43:31 +02:00
Ods::data_page* dpage =
(Ods::data_page*) CCH_HANDOFF(tdbb, &window, *page, LCK_read, pag_data);
recordCount = dpage->dpg_count;
}
CCH_RELEASE(tdbb, &window);
return (double) recordCount;
}
}
if (!format) {
format = relation->rel_current_format;
}
return (double) dataPages * (dbb->dbb_page_size - DPG_SIZE) /
2005-03-29 11:43:31 +02:00
(minRecordSize + (format->fmt_length * 0.5));
}
bool DPM_chain( thread_db* tdbb, record_param* org_rpb, record_param* new_rpb)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ c h a i n
*
**************************************
*
* Functional description
* Start here with a plausible, but non-active record_param.
2001-05-23 15:26:42 +02:00
*
* We need to create a new version of a record. If the new version
* fits on the same page as the old record, things are simple and
* quick. If not, return false and let somebody else suffer.
2001-05-23 15:26:42 +02:00
*
* Note that we also return false if the record fetched doesn't
2001-05-23 15:26:42 +02:00
* match the state of the input rpb, or if there is no record for
* that record number. The caller has to check the results to
2009-06-07 11:49:58 +02:00
* see what failed if false is returned. At the moment, there is
* only one caller, VIO_erase.
2001-05-23 15:26:42 +02:00
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES)
{
printf("DPM_chain (org_rpb %"QUADFORMAT"d, new_rpb %"
QUADFORMAT"d)\n", org_rpb->rpb_number.getValue(),
new_rpb ? new_rpb->rpb_number.getValue() : 0);
2003-12-31 06:36:12 +01:00
}
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf
2003-04-01 19:58:19 +02:00
(" org record %"SLONGFORMAT":%d transaction %"SLONGFORMAT
" back %"SLONGFORMAT":%d fragment %"SLONGFORMAT":%d flags %d\n",
org_rpb->rpb_page, org_rpb->rpb_line, org_rpb->rpb_transaction_nr,
2001-05-23 15:26:42 +02:00
org_rpb->rpb_b_page, org_rpb->rpb_b_line, org_rpb->rpb_f_page,
org_rpb->rpb_f_line, org_rpb->rpb_flags);
2009-06-23 06:12:59 +02:00
if (new_rpb)
{
2004-04-29 00:43:34 +02:00
printf(" new record length %d transaction %"SLONGFORMAT
2003-04-01 19:58:19 +02:00
" flags %d\n",
new_rpb->rpb_length, new_rpb->rpb_transaction_nr,
2001-05-23 15:26:42 +02:00
new_rpb->rpb_flags);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
}
#endif
record_param temp = *org_rpb;
DataComprControl dcc(*tdbb->getDefaultPool());
const USHORT size = SQZ_length((SCHAR*) new_rpb->rpb_address, (int) new_rpb->rpb_length, &dcc);
2001-05-23 15:26:42 +02:00
2009-06-23 06:12:59 +02:00
if (!DPM_get(tdbb, org_rpb, LCK_write))
{
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_WRITES_INFO)
2004-04-29 00:43:34 +02:00
printf(" record not found in DPM_chain\n");
2001-05-23 15:26:42 +02:00
#endif
return false;
2001-05-23 15:26:42 +02:00
}
2009-08-20 13:18:57 +02:00
// if somebody has modified the record since we looked last, stop now!
2001-05-23 15:26:42 +02:00
if (temp.rpb_transaction_nr != org_rpb->rpb_transaction_nr ||
2001-05-23 15:26:42 +02:00
temp.rpb_b_page != org_rpb->rpb_b_page ||
2003-12-31 06:36:12 +01:00
temp.rpb_b_line != org_rpb->rpb_b_line)
{
2006-05-22 00:07:35 +02:00
CCH_RELEASE(tdbb, &org_rpb->getWindow(tdbb));
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_WRITES_INFO)
2004-04-29 00:43:34 +02:00
printf(" record changed in DPM_chain\n");
2001-05-23 15:26:42 +02:00
#endif
return false;
2001-05-23 15:26:42 +02:00
}
if ((org_rpb->rpb_flags & rpb_delta) && temp.rpb_prior) {
2001-05-23 15:26:42 +02:00
org_rpb->rpb_prior = temp.rpb_prior;
}
2009-06-23 06:12:59 +02:00
else if (org_rpb->rpb_flags & rpb_delta)
{
2006-05-22 00:07:35 +02:00
CCH_RELEASE(tdbb, &org_rpb->getWindow(tdbb));
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_WRITES_INFO)
2004-04-29 00:43:34 +02:00
printf(" record delta state changed\n");
2001-05-23 15:26:42 +02:00
#endif
return false;
2001-05-23 15:26:42 +02:00
}
2006-05-22 00:07:35 +02:00
data_page* page = (data_page*) org_rpb->getWindow(tdbb).win_buffer;
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// If the record obviously isn't going to fit, don't even try
2001-05-23 15:26:42 +02:00
2009-06-23 06:12:59 +02:00
if (size > dbb->dbb_page_size - (sizeof(data_page) + RHD_SIZE))
{
2006-05-22 00:07:35 +02:00
CCH_RELEASE(tdbb, &org_rpb->getWindow(tdbb));
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_WRITES_INFO)
2004-04-29 00:43:34 +02:00
printf(" insufficient room found in DPM_chain\n");
2001-05-23 15:26:42 +02:00
#endif
return false;
2001-05-23 15:26:42 +02:00
}
2009-08-20 13:18:57 +02:00
// The record must be long enough to permit fragmentation later. If it's
// too small, compute the number of pad bytes required
2001-05-23 15:26:42 +02:00
2003-12-31 06:36:12 +01:00
SLONG fill = (RHDF_SIZE - RHD_SIZE) - size;
if (fill < 0 || (new_rpb->rpb_flags & rpb_deleted)) {
2001-05-23 15:26:42 +02:00
fill = 0;
}
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// Accomodate max record size i.e. 64K
2003-12-31 06:36:12 +01:00
const SLONG length = ROUNDUP(RHD_SIZE + size + fill, ODS_ALIGNMENT);
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// Find space on page and open slot
2001-05-23 15:26:42 +02:00
2003-12-31 06:36:12 +01:00
SSHORT slot = page->dpg_count;
SSHORT space = dbb->dbb_page_size;
SSHORT top = HIGH_WATER(page->dpg_count);
SSHORT available = dbb->dbb_page_size - top;
2001-05-23 15:26:42 +02:00
2003-12-31 06:36:12 +01:00
SSHORT n = 0;
2004-02-02 12:02:12 +01:00
const data_page::dpg_repeat* index = page->dpg_rpt;
for (const data_page::dpg_repeat* const end = index + page->dpg_count;
2003-12-31 06:36:12 +01:00
index < end; index++, n++)
{
if (!index->dpg_length && slot == page->dpg_count) {
2001-05-23 15:26:42 +02:00
slot = n;
}
2003-12-31 06:36:12 +01:00
SSHORT offset;
2009-06-23 06:12:59 +02:00
if (index->dpg_length && (offset = index->dpg_offset))
{
2001-05-23 15:26:42 +02:00
available -= ROUNDUP(index->dpg_length, ODS_ALIGNMENT);
space = MIN(space, offset);
}
}
2009-06-23 06:12:59 +02:00
if (slot == page->dpg_count)
{
2004-02-02 12:02:12 +01:00
top += sizeof(data_page::dpg_repeat);
available -= sizeof(data_page::dpg_repeat);
2001-05-23 15:26:42 +02:00
}
2009-08-20 13:18:57 +02:00
// If the record doesn't fit, punt
2001-05-23 15:26:42 +02:00
2009-06-23 06:12:59 +02:00
if (length > available)
{
2006-05-22 00:07:35 +02:00
CCH_RELEASE(tdbb, &org_rpb->getWindow(tdbb));
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_WRITES_INFO)
2004-04-29 00:43:34 +02:00
printf(" compressed page doesn't have room in DPM_chain\n");
2001-05-23 15:26:42 +02:00
#endif
return false;
2001-05-23 15:26:42 +02:00
}
2006-05-22 00:07:35 +02:00
CCH_precedence(tdbb, &org_rpb->getWindow(tdbb), -org_rpb->rpb_transaction_nr);
CCH_MARK(tdbb, &org_rpb->getWindow(tdbb));
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// Record fits, in theory. Check to see if the page needs compression
2001-05-23 15:26:42 +02:00
space -= length;
if (space < top) {
2001-05-23 15:26:42 +02:00
space = DPM_compress(tdbb, page) - length;
}
2001-05-23 15:26:42 +02:00
if (slot == page->dpg_count) {
2001-05-23 15:26:42 +02:00
++page->dpg_count;
}
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// Swap the old record into the new slot and the new record into the old slot
2001-05-23 15:26:42 +02:00
new_rpb->rpb_b_page = new_rpb->rpb_page = org_rpb->rpb_page;
2001-05-23 15:26:42 +02:00
new_rpb->rpb_b_line = slot;
new_rpb->rpb_line = org_rpb->rpb_line;
2001-05-23 15:26:42 +02:00
data_page::dpg_repeat* index2 = page->dpg_rpt + org_rpb->rpb_line;
rhd* header = (rhd*) ((SCHAR *) page + index2->dpg_offset);
2001-05-23 15:26:42 +02:00
header->rhd_flags |= rhd_chain;
2003-12-31 06:36:12 +01:00
page->dpg_rpt[slot] = *index2;
2001-05-23 15:26:42 +02:00
2003-12-31 06:36:12 +01:00
index2->dpg_offset = space;
index2->dpg_length = RHD_SIZE + size + fill;
2001-05-23 15:26:42 +02:00
header = (rhd*) ((SCHAR *) page + space);
2001-05-23 15:26:42 +02:00
header->rhd_flags = new_rpb->rpb_flags;
header->rhd_transaction = new_rpb->rpb_transaction_nr;
2001-05-23 15:26:42 +02:00
header->rhd_format = new_rpb->rpb_format_number;
header->rhd_b_page = new_rpb->rpb_b_page;
header->rhd_b_line = new_rpb->rpb_b_line;
SQZ_fast(&dcc, (SCHAR*) new_rpb->rpb_address, (SCHAR*) header->rhd_data);
2001-05-23 15:26:42 +02:00
if (fill) {
memset(header->rhd_data + size, 0, fill);
2001-05-23 15:26:42 +02:00
}
if (page->dpg_header.pag_flags & dpg_swept)
{
page->dpg_header.pag_flags &= ~dpg_swept;
mark_full(tdbb, org_rpb);
}
else
2009-12-28 12:46:55 +01:00
CCH_RELEASE(tdbb, &org_rpb->getWindow(tdbb));
2001-05-23 15:26:42 +02:00
return true;
2001-05-23 15:26:42 +02:00
}
int DPM_compress( thread_db* tdbb, data_page* page)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ c o m p r e s s
*
**************************************
*
* Functional description
* Compress a data page. Return the high water mark.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_TRACE_ALL)
2004-04-29 00:43:34 +02:00
printf("compress (page)\n");
2001-05-23 15:26:42 +02:00
if (debug_flag > DEBUG_TRACE_ALL_INFO)
2004-04-29 00:43:34 +02:00
printf(" sequence %"SLONGFORMAT"\n", page->dpg_sequence);
2001-05-23 15:26:42 +02:00
#endif
2003-12-31 06:36:12 +01:00
UCHAR temp_page[MAX_PAGE_SIZE];
2001-12-24 03:51:06 +01:00
if (dbb->dbb_page_size > sizeof(temp_page)) {
2009-08-20 13:18:57 +02:00
BUGCHECK(250); // msg 250 temporary page buffer too small
2001-12-24 03:51:06 +01:00
}
2003-12-31 06:36:12 +01:00
SSHORT space = dbb->dbb_page_size;
2004-02-02 12:02:12 +01:00
const data_page::dpg_repeat* const end = page->dpg_rpt + page->dpg_count;
2001-05-23 15:26:42 +02:00
2004-02-02 12:02:12 +01:00
for (data_page::dpg_repeat* index = page->dpg_rpt; index < end; index++)
2001-12-24 03:51:06 +01:00
{
if (index->dpg_offset)
{
// 11-Aug-2004. Nickolay Samofatov.
// Copy block of pre-aligned length to avoid putting rubbish from stack into database
// This should also work just a little bit faster too.
const SSHORT l = ROUNDUP(index->dpg_length, ODS_ALIGNMENT);
space -= l;
memcpy(temp_page + space, (UCHAR *) page + index->dpg_offset, l);
2001-05-23 15:26:42 +02:00
index->dpg_offset = space;
}
2001-12-24 03:51:06 +01:00
}
2001-05-23 15:26:42 +02:00
memcpy((UCHAR *) page + space, temp_page + space, dbb->dbb_page_size - space);
2001-05-23 15:26:42 +02:00
if (page->dpg_header.pag_type != pag_data) {
2009-08-20 13:18:57 +02:00
BUGCHECK(251); // msg 251 damaged data page
}
2001-05-23 15:26:42 +02:00
return space;
}
void DPM_create_relation( thread_db* tdbb, jrd_rel* relation)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ c r e a t e _ r e l a t i o n
*
**************************************
*
* Functional description
* Create a new relation.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_TRACE_ALL)
2004-04-29 00:43:34 +02:00
printf("DPM_create_relation (relation %d)\n", relation->rel_id);
2001-05-23 15:26:42 +02:00
#endif
2006-05-22 00:07:35 +02:00
RelationPages* relPages = relation->getBasePages();
DPM_create_relation_pages(tdbb, relation, relPages);
2009-08-20 13:18:57 +02:00
// Store page numbers in RDB$PAGES
2006-05-22 00:07:35 +02:00
DPM_pages(tdbb, relation->rel_id, pag_pointer, (ULONG) 0,
(*relPages->rel_pages)[0] /*window.win_page*/);
DPM_pages(tdbb, relation->rel_id, pag_root, (ULONG) 0,
relPages->rel_index_root /*root_window.win_page*/);
}
void DPM_create_relation_pages(thread_db* tdbb, jrd_rel* relation, RelationPages* relPages)
{
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2006-05-22 00:07:35 +02:00
CHECK_DBB(dbb);
2009-08-20 13:18:57 +02:00
// Allocate first pointer page
2006-05-22 00:07:35 +02:00
WIN window(relPages->rel_pg_space_id, -1);
2004-02-02 12:02:12 +01:00
pointer_page* page = (pointer_page*) DPM_allocate(tdbb, &window);
page->ppg_header.pag_type = pag_pointer;
2001-05-23 15:26:42 +02:00
page->ppg_relation = relation->rel_id;
page->ppg_header.pag_flags = ppg_eof;
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, &window);
2009-08-20 13:18:57 +02:00
// If this is relation 0 (RDB$PAGES), update the header
2001-05-23 15:26:42 +02:00
2009-06-23 06:12:59 +02:00
if (relation->rel_id == 0)
{
2006-05-22 00:07:35 +02:00
WIN root_window(HEADER_PAGE_NUMBER);
2008-12-18 11:47:25 +01:00
header_page* header = (header_page*) CCH_FETCH(tdbb, &root_window, LCK_write, pag_header);
2001-05-23 15:26:42 +02:00
CCH_MARK(tdbb, &root_window);
2006-05-22 00:07:35 +02:00
header->hdr_PAGES = window.win_page.getPageNum();
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, &root_window);
}
2009-08-20 13:18:57 +02:00
// Keep track in memory of the first pointer page
2001-05-23 15:26:42 +02:00
2009-06-23 06:12:59 +02:00
if (!relPages->rel_pages)
{
2006-05-22 00:07:35 +02:00
vcl* vector = vcl::newVector(*dbb->dbb_permanent, 1);
relPages->rel_pages = vector;
}
(*relPages->rel_pages)[0] = window.win_page.getPageNum();
2001-05-23 15:26:42 +02:00
// CVC: AFAIK, DPM_allocate calls PAG_allocate and neither of them cares about win_page.
// Therefore, I decided that the root_window in the if() above and this one aren't related.
2009-08-20 13:18:57 +02:00
// Create an index root page
2006-05-22 00:07:35 +02:00
WIN root_window(relPages->rel_pg_space_id, -1);
index_root_page* root = (index_root_page*) DPM_allocate(tdbb, &root_window);
root->irt_header.pag_type = pag_root;
2001-05-23 15:26:42 +02:00
root->irt_relation = relation->rel_id;
2009-08-20 13:18:57 +02:00
//root->irt_count = 0;
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, &root_window);
2006-05-22 00:07:35 +02:00
relPages->rel_index_root = root_window.win_page.getPageNum();
2001-05-23 15:26:42 +02:00
}
SLONG DPM_data_pages(thread_db* tdbb, jrd_rel* relation)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ d a t a _ p a g e s
*
**************************************
*
* Functional description
* Compute and return the number of data pages in a relation.
* Some poor sucker is going to use this information to make
* an educated guess at the number of records in the relation.
*
**************************************/
SET_TDBB(tdbb);
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_TRACE_ALL)
2004-04-29 00:43:34 +02:00
printf("DPM_data_pages (relation %d)\n", relation->rel_id);
2001-05-23 15:26:42 +02:00
#endif
2006-05-22 00:07:35 +02:00
RelationPages* relPages = relation->getPages(tdbb);
SLONG pages = relPages->rel_data_pages;
if (!pages)
{
2006-05-22 00:07:35 +02:00
WIN window(relPages->rel_pg_space_id, -1);
for (USHORT sequence = 0; true; sequence++)
{
2004-02-02 12:02:12 +01:00
const pointer_page* ppage =
2006-05-22 00:07:35 +02:00
get_pointer_page(tdbb, relation, relPages, &window, sequence, LCK_read);
2009-06-23 06:12:59 +02:00
if (!ppage)
{
BUGCHECK(243);
2009-08-20 13:18:57 +02:00
// msg 243 missing pointer page in DPM_data_pages
}
2003-12-31 06:36:12 +01:00
const SLONG* page = ppage->ppg_page;
const SLONG* const end_page = page + ppage->ppg_count;
2009-06-23 06:12:59 +02:00
while (page < end_page)
{
if (*page++) {
2001-05-23 15:26:42 +02:00
pages++;
}
2003-12-31 06:36:12 +01:00
}
if (ppage->ppg_header.pag_flags & ppg_eof) {
2001-05-23 15:26:42 +02:00
break;
}
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, &window);
}
CCH_RELEASE(tdbb, &window);
#ifdef SUPERSERVER
2006-05-22 00:07:35 +02:00
relPages->rel_data_pages = pages;
2001-05-23 15:26:42 +02:00
#endif
}
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_TRACE_ALL)
2004-04-29 00:43:34 +02:00
printf(" returned pages: %"SLONGFORMAT"\n", pages);
2001-05-23 15:26:42 +02:00
#endif
return pages;
}
void DPM_delete( thread_db* tdbb, record_param* rpb, SLONG prior_page)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ d e l e t e
*
**************************************
*
* Functional description
* Delete a fragment from data page. Assume the page has
* already been fetched (but not marked) for write. Copy the
* record header into the record parameter block before deleting
* it. If the record goes empty, release the page. Release the
* page when we're done.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES)
{
2006-01-18 11:43:13 +01:00
printf("DPM_delete (record_param %"QUADFORMAT", prior_page %"SLONGFORMAT")\n",
rpb->rpb_number.getValue(), prior_page);
2003-12-31 06:36:12 +01:00
}
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf
2003-04-01 19:58:19 +02:00
(" record %"SLONGFORMAT":%d transaction %"SLONGFORMAT" back %"
SLONGFORMAT":%d fragment %"SLONGFORMAT":%d flags %d\n",
rpb->rpb_page, rpb->rpb_line, rpb->rpb_transaction_nr,
2001-05-23 15:26:42 +02:00
rpb->rpb_b_page, rpb->rpb_b_line, rpb->rpb_f_page,
rpb->rpb_f_line, rpb->rpb_flags);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
2009-06-25 12:59:10 +02:00
WIN* const window = &rpb->getWindow(tdbb);
2004-02-02 12:02:12 +01:00
data_page* page = (data_page*) window->win_buffer;
2003-12-31 06:36:12 +01:00
SLONG sequence = page->dpg_sequence;
data_page::dpg_repeat* index = &page->dpg_rpt[rpb->rpb_line];
const RecordNumber number = rpb->rpb_number;
2001-05-23 15:26:42 +02:00
2009-06-23 06:12:59 +02:00
if (!get_header(window, rpb->rpb_line, rpb))
{
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, window);
2009-08-20 13:18:57 +02:00
BUGCHECK(244); // msg 244 Fragment does not exist
2001-05-23 15:26:42 +02:00
}
#ifdef VIO_DEBUG
2003-12-31 06:36:12 +01:00
if (debug_flag > DEBUG_WRITES_INFO) {
2009-06-23 06:12:59 +02:00
printf(" record length as stored in record header: %d\n", rpb->rpb_length);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
rpb->rpb_number = number;
CCH_precedence(tdbb, window, prior_page);
CCH_MARK(tdbb, window);
index->dpg_offset = 0;
index->dpg_length = 0;
2009-08-20 13:18:57 +02:00
// Compute the highest line number level on page
2001-05-23 15:26:42 +02:00
2008-12-18 11:47:25 +01:00
for (index = &page->dpg_rpt[page->dpg_count]; index > page->dpg_rpt; --index)
2003-09-13 14:03:11 +02:00
{
if (index[-1].dpg_offset)
2001-05-23 15:26:42 +02:00
break;
2003-09-13 14:03:11 +02:00
}
2001-05-23 15:26:42 +02:00
2006-05-25 10:40:23 +02:00
USHORT count;
2001-05-23 15:26:42 +02:00
page->dpg_count = count = index - page->dpg_rpt;
2009-08-20 13:18:57 +02:00
// If the page is not empty and used to be marked as full, change the
// state of both the page and the appropriate pointer page.
2001-05-23 15:26:42 +02:00
2009-06-23 06:12:59 +02:00
if (count && (page->dpg_header.pag_flags & dpg_full))
{
DEBUG
page->dpg_header.pag_flags &= ~dpg_full;
2001-05-23 15:26:42 +02:00
mark_full(tdbb, rpb);
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_WRITES_INFO)
2004-04-29 00:43:34 +02:00
printf(" page is no longer full\n");
2001-05-23 15:26:42 +02:00
#endif
return;
}
const UCHAR flags = page->dpg_header.pag_flags;
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, window);
2009-08-20 13:18:57 +02:00
// If the page is non-empty, we're done.
2001-05-23 15:26:42 +02:00
if (count)
return;
2009-06-23 06:12:59 +02:00
if (flags & dpg_orphan)
{
2009-08-20 13:18:57 +02:00
// The page inventory page will be written after the page being
// released, which will be written after the pages from which earlier
// fragments were deleted, which will be written after the page from
// which the first fragment is deleted.
// The resulting 'must-be-written-after' graph is:
// pip --> deallocated page --> prior_page
PAG_release_page(tdbb, window->win_page, window->win_page);
2001-05-23 15:26:42 +02:00
return;
}
2006-05-25 10:40:23 +02:00
// Data page has gone empty. Refetch page thru pointer page. If it's
// still empty, remove page from relation.
2006-05-25 10:40:23 +02:00
pointer_page* ppage;
SSHORT slot;
USHORT pp_sequence;
2001-05-23 15:26:42 +02:00
DECOMPOSE(sequence, dbb->dbb_dp_per_pp, pp_sequence, slot);
2009-06-25 12:59:10 +02:00
RelationPages* relPages = NULL;
WIN pwindow(DB_PAGE_SPACE, -1);
2001-05-23 15:26:42 +02:00
2009-06-25 12:59:10 +02:00
for (;;)
2003-12-31 06:36:12 +01:00
{
2009-06-25 12:59:10 +02:00
relPages = rpb->rpb_relation->getPages(tdbb, rpb->rpb_transaction_nr);
pwindow = WIN(relPages->rel_pg_space_id, -1);
2009-06-27 05:03:56 +02:00
if (!(ppage = get_pointer_page(tdbb, rpb->rpb_relation, relPages, &pwindow,
pp_sequence, LCK_write)))
2009-06-25 12:59:10 +02:00
{
2009-08-20 13:18:57 +02:00
BUGCHECK(245); // msg 245 pointer page disappeared in DPM_delete
2009-06-25 12:59:10 +02:00
}
2001-05-23 15:26:42 +02:00
2009-06-25 12:59:10 +02:00
if (slot >= ppage->ppg_count || !(window->win_page = ppage->ppg_page[slot]))
{
CCH_RELEASE(tdbb, &pwindow);
return;
}
2009-08-20 13:18:57 +02:00
// Since this fetch for exclusive access follows a (pointer page) fetch for
// exclusive access, put a timeout on this fetch to be able to recover from
// possible deadlocks.
2009-06-25 12:59:10 +02:00
page = (data_page*) CCH_FETCH_TIMEOUT(tdbb, window, LCK_write, pag_data, -1);
if (!page) {
CCH_RELEASE(tdbb, &pwindow);
}
else
break;
2001-05-23 15:26:42 +02:00
}
2009-06-25 12:59:10 +02:00
2009-06-23 06:12:59 +02:00
if (page->dpg_count)
{
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, &pwindow);
CCH_RELEASE(tdbb, window);
return;
}
2009-08-20 13:18:57 +02:00
// Data page is still empty and still in the relation. Eliminate the
// pointer to the data page then release the page.
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf
2003-04-01 19:58:19 +02:00
("\tDPM_delete: page %"SLONGFORMAT
" is empty and about to be released from relation %d\n",
2006-05-22 04:36:53 +02:00
window->win_page.getPageNum(), rpb->rpb_relation->rel_id);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
2009-08-20 13:18:57 +02:00
// Make sure that the pointer page is written after the data page.
// The resulting 'must-be-written-after' graph is:
// pip --> pp --> deallocated page --> prior_page
2001-05-23 15:26:42 +02:00
CCH_precedence(tdbb, &pwindow, window->win_page);
CCH_MARK(tdbb, &pwindow);
ppage->ppg_page[slot] = 0;
2009-06-25 12:59:10 +02:00
const SLONG* ptr;
2003-09-13 14:03:11 +02:00
for (ptr = &ppage->ppg_page[ppage->ppg_count]; ptr > ppage->ppg_page; --ptr)
2003-12-31 06:36:12 +01:00
{
2003-09-13 14:03:11 +02:00
if (ptr[-1])
2001-05-23 15:26:42 +02:00
break;
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
ppage->ppg_count = count = ptr - ppage->ppg_page;
if (count) {
2001-05-23 15:26:42 +02:00
count--;
}
2001-05-23 15:26:42 +02:00
ppage->ppg_min_space = MIN(ppage->ppg_min_space, count);
2006-05-25 10:40:23 +02:00
//jrd_rel* relation = rpb->rpb_relation;
2006-05-22 00:07:35 +02:00
relPages->rel_slot_space = MIN(relPages->rel_slot_space, pp_sequence);
if (relPages->rel_data_pages) {
--relPages->rel_data_pages;
2001-05-23 15:26:42 +02:00
}
CCH_RELEASE(tdbb, &pwindow);
CCH_RELEASE(tdbb, window);
2009-08-20 13:18:57 +02:00
// Make sure that the page inventory page is written after the pointer page.
// Earlier, we make sure that the pointer page is written after the data
// page being released.
PAG_release_page(tdbb, window->win_page, pwindow.win_page);
2001-05-23 15:26:42 +02:00
}
void DPM_delete_relation( thread_db* tdbb, jrd_rel* relation)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ d e l e t e _ r e l a t i o n
*
**************************************
*
* Functional description
* Get rid of an unloved, unwanted relation.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2006-05-22 00:07:35 +02:00
RelationPages* relPages = relation->getBasePages();
DPM_delete_relation_pages(tdbb, relation, relPages);
2009-08-20 13:18:57 +02:00
// Next, cancel out stuff from RDB$PAGES
2006-05-22 00:07:35 +02:00
jrd_req* handle = NULL;
FOR(REQUEST_HANDLE handle) X IN RDB$PAGES WITH
2006-05-25 10:40:23 +02:00
X.RDB$RELATION_ID EQ relation->rel_id
ERASE X;
2006-05-22 00:07:35 +02:00
END_FOR;
CMP_release(tdbb, handle);
CCH_flush(tdbb, FLUSH_ALL, 0);
}
2009-11-25 09:38:52 +01:00
void DPM_delete_relation_pages(Jrd::thread_db* tdbb, Jrd::jrd_rel* relation,
Jrd::RelationPages* relPages)
2006-05-22 00:07:35 +02:00
{
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2006-05-22 00:07:35 +02:00
WIN window(relPages->rel_pg_space_id, -1), data_window(relPages->rel_pg_space_id, -1);
2001-05-23 15:26:42 +02:00
window.win_flags = data_window.win_flags = WIN_large_scan;
window.win_scans = data_window.win_scans = 1;
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_TRACE_ALL)
2004-04-29 00:43:34 +02:00
printf("DPM_delete_relation (relation %d)\n", relation->rel_id);
2001-05-23 15:26:42 +02:00
#endif
2009-08-20 13:18:57 +02:00
// Delete all data and pointer pages
2001-05-23 15:26:42 +02:00
2003-12-31 06:36:12 +01:00
for (USHORT sequence = 0; true; sequence++)
2001-05-23 15:26:42 +02:00
{
2004-02-02 12:02:12 +01:00
const pointer_page* ppage =
2006-05-22 00:07:35 +02:00
get_pointer_page(tdbb, relation, relPages, &window, sequence, LCK_read);
2001-05-23 15:26:42 +02:00
if (!ppage)
{
2009-08-20 13:18:57 +02:00
BUGCHECK(246); // msg 246 pointer page lost from DPM_delete_relation
2001-05-23 15:26:42 +02:00
}
2003-12-31 06:36:12 +01:00
const SLONG* page = ppage->ppg_page;
const UCHAR* flags = (UCHAR *) (ppage->ppg_page + dbb->dbb_dp_per_pp);
for (USHORT i = 0; i < ppage->ppg_count; i++, page++)
2001-05-23 15:26:42 +02:00
{
if (!*page) {
continue;
}
// if (flags[i >> 2] & (2 << ((i & 3) << 1)))
if (PPG_DP_BIT_TEST(flags, i, ppg_dp_large))
2001-05-23 15:26:42 +02:00
{
data_window.win_page = *page;
2008-12-18 11:47:25 +01:00
data_page* dpage = (data_page*) CCH_FETCH(tdbb, &data_window, LCK_write, pag_data);
2004-02-02 12:02:12 +01:00
const data_page::dpg_repeat* line = dpage->dpg_rpt;
const data_page::dpg_repeat* const end_line = line + dpage->dpg_count;
2001-05-23 15:26:42 +02:00
for (; line < end_line; line++)
{
if (line->dpg_length)
{
rhd* header = (rhd*) ((UCHAR *) dpage + line->dpg_offset);
2001-05-23 15:26:42 +02:00
if (header->rhd_flags & rhd_large)
{
delete_tail(tdbb, (rhdf*)header, relPages->rel_pg_space_id,
2008-12-18 11:47:25 +01:00
line->dpg_length);
2001-05-23 15:26:42 +02:00
}
}
}
CCH_RELEASE_TAIL(tdbb, &data_window);
}
PAG_release_page(tdbb, PageNumber(relPages->rel_pg_space_id, *page), ZERO_PAGE_NUMBER);
2001-05-23 15:26:42 +02:00
}
const UCHAR pag_flags = ppage->ppg_header.pag_flags;
2001-05-23 15:26:42 +02:00
CCH_RELEASE_TAIL(tdbb, &window);
PAG_release_page(tdbb, window.win_page, ZERO_PAGE_NUMBER);
2001-05-23 15:26:42 +02:00
if (pag_flags & ppg_eof)
{
break;
}
}
2006-05-22 00:07:35 +02:00
delete relPages->rel_pages;
relPages->rel_pages = NULL;
relPages->rel_data_pages = 0;
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// Now get rid of the index root page
2001-05-23 15:26:42 +02:00
PAG_release_page(tdbb,
2008-01-16 09:46:02 +01:00
PageNumber(relPages->rel_pg_space_id, relPages->rel_index_root), ZERO_PAGE_NUMBER);
2006-05-22 00:07:35 +02:00
relPages->rel_index_root = 0;
2001-05-23 15:26:42 +02:00
}
bool DPM_fetch(thread_db* tdbb, record_param* rpb, USHORT lock)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ f e t c h
*
**************************************
*
* Functional description
* Fetch a particular record fragment from page and line numbers.
* Get various header stuff, but don't change the record number.
*
* return: true if the fragment is returned.
* false if the fragment is not found.
2001-05-23 15:26:42 +02:00
*
**************************************/
SET_TDBB(tdbb);
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_READS)
{
printf("DPM_fetch (record_param %"QUADFORMAT"d, lock %d)\n",
rpb->rpb_number.getValue(), lock);
}
2001-05-23 15:26:42 +02:00
if (debug_flag > DEBUG_READS_INFO)
printf(" record %"SLONGFORMAT":%d\n", rpb->rpb_page, rpb->rpb_line);
2001-05-23 15:26:42 +02:00
#endif
const RecordNumber number = rpb->rpb_number;
2006-05-22 00:07:35 +02:00
RelationPages* relPages = rpb->rpb_relation->getPages(tdbb);
rpb->getWindow(tdbb).win_page = PageNumber(relPages->rel_pg_space_id, rpb->rpb_page);
2006-05-22 00:07:35 +02:00
CCH_FETCH(tdbb, &rpb->getWindow(tdbb), lock, pag_data);
2001-05-23 15:26:42 +02:00
2009-06-23 06:12:59 +02:00
if (!get_header(&rpb->getWindow(tdbb), rpb->rpb_line, rpb))
{
2006-05-22 00:07:35 +02:00
CCH_RELEASE(tdbb, &rpb->getWindow(tdbb));
return false;
2001-05-23 15:26:42 +02:00
}
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_READS_INFO)
{
2004-04-29 00:43:34 +02:00
printf
2003-04-01 19:58:19 +02:00
(" record %"SLONGFORMAT":%d transaction %"SLONGFORMAT" back %"
SLONGFORMAT":%d fragment %"SLONGFORMAT":%d flags %d\n",
rpb->rpb_page, rpb->rpb_line, rpb->rpb_transaction_nr,
2001-05-23 15:26:42 +02:00
rpb->rpb_b_page, rpb->rpb_b_line, rpb->rpb_f_page,
rpb->rpb_f_line, rpb->rpb_flags);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
rpb->rpb_number = number;
return true;
2001-05-23 15:26:42 +02:00
}
2009-11-25 09:38:52 +01:00
SSHORT DPM_fetch_back(thread_db* tdbb, record_param* rpb, USHORT lock, SSHORT latch_wait)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ f e t c h _ b a c k
*
**************************************
*
* Functional description
* Chase a backpointer with a handoff.
*
* input:
* latch_wait: 1 => Wait as long as necessary to get the latch.
* This can cause deadlocks of course.
* 0 => If the latch can't be acquired immediately,
* give up and return 0.
* <negative number> => Latch timeout interval in seconds.
*
* return:
* 1: fetch back was successful.
* 0: unsuccessful (only possible is latch_wait <> 1),
* The latch timed out.
* The latch on the fetched page is downgraded to shared.
* The fetched page is unmarked.
*
**************************************/
SET_TDBB(tdbb);
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_READS)
{
printf("DPM_fetch_back (record_param %"QUADFORMAT"d, lock %d)\n",
rpb->rpb_number.getValue(), lock);
2001-05-23 15:26:42 +02:00
}
if (debug_flag > DEBUG_READS_INFO)
{
2004-04-29 00:43:34 +02:00
printf(" record %"SLONGFORMAT":%d transaction %"SLONGFORMAT
2003-04-01 19:58:19 +02:00
" back %"SLONGFORMAT":%d\n",
rpb->rpb_page, rpb->rpb_line, rpb->rpb_transaction_nr,
2001-05-23 15:26:42 +02:00
rpb->rpb_b_page, rpb->rpb_b_line);
}
#endif
2009-08-20 13:18:57 +02:00
// Possibly allow a latch timeout to occur. Return error if that is the case.
2001-05-23 15:26:42 +02:00
if (!(CCH_HANDOFF_TIMEOUT(tdbb,
2008-12-18 11:47:25 +01:00
&rpb->getWindow(tdbb),
rpb->rpb_b_page,
lock,
pag_data,
latch_wait)))
2001-05-23 15:26:42 +02:00
{
return 0;
}
const RecordNumber number = rpb->rpb_number;
rpb->rpb_page = rpb->rpb_b_page;
rpb->rpb_line = rpb->rpb_b_line;
if (!get_header(&rpb->getWindow(tdbb), rpb->rpb_line, rpb))
2001-05-23 15:26:42 +02:00
{
2006-05-22 00:07:35 +02:00
CCH_RELEASE(tdbb, &rpb->getWindow(tdbb));
2009-08-20 13:18:57 +02:00
BUGCHECK(291); // msg 291 cannot find record back version
2001-05-23 15:26:42 +02:00
}
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_READS_INFO)
{
2004-04-29 00:43:34 +02:00
printf(" record fetched %"SLONGFORMAT":%d transaction %"
2003-04-01 19:58:19 +02:00
SLONGFORMAT" back %"SLONGFORMAT":%d fragment %"SLONGFORMAT
":%d flags %d\n",
rpb->rpb_page, rpb->rpb_line, rpb->rpb_transaction_nr,
2001-05-23 15:26:42 +02:00
rpb->rpb_b_page, rpb->rpb_b_line, rpb->rpb_f_page,
rpb->rpb_f_line, rpb->rpb_flags);
}
#endif
rpb->rpb_number = number;
return 1;
}
void DPM_fetch_fragment( thread_db* tdbb, record_param* rpb, USHORT lock)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ f e t c h _f r a g m e n t
*
**************************************
*
* Functional description
* Chase a fragment pointer with a handoff.
*
**************************************/
SET_TDBB(tdbb);
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_READS)
printf("DPM_fetch_fragment (record_param %"QUADFORMAT"d, lock %d)\n",
rpb->rpb_number.getValue(), lock);
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_READS_INFO)
{
2004-04-29 00:43:34 +02:00
printf(" record %"SLONGFORMAT":%d transaction %"SLONGFORMAT
2003-04-01 19:58:19 +02:00
" back %"SLONGFORMAT":%d\n",
rpb->rpb_page, rpb->rpb_line, rpb->rpb_transaction_nr,
2001-05-23 15:26:42 +02:00
rpb->rpb_b_page, rpb->rpb_b_line);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
const RecordNumber number = rpb->rpb_number;
rpb->rpb_page = rpb->rpb_f_page;
rpb->rpb_line = rpb->rpb_f_line;
CCH_HANDOFF(tdbb, &rpb->getWindow(tdbb), rpb->rpb_page, lock, pag_data);
2001-05-23 15:26:42 +02:00
2009-06-23 06:12:59 +02:00
if (!get_header(&rpb->getWindow(tdbb), rpb->rpb_line, rpb))
{
2006-05-22 00:07:35 +02:00
CCH_RELEASE(tdbb, &rpb->getWindow(tdbb));
2009-08-20 13:18:57 +02:00
BUGCHECK(248); // msg 248 cannot find record fragment
2001-05-23 15:26:42 +02:00
}
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_READS_INFO)
{
2004-04-29 00:43:34 +02:00
printf
2003-04-01 19:58:19 +02:00
(" record fetched %"SLONGFORMAT":%d transaction %"SLONGFORMAT
" back %"SLONGFORMAT":%d fragment %"SLONGFORMAT":%d flags %d\n",
rpb->rpb_page, rpb->rpb_line, rpb->rpb_transaction_nr,
2001-05-23 15:26:42 +02:00
rpb->rpb_b_page, rpb->rpb_b_line, rpb->rpb_f_page,
rpb->rpb_f_line, rpb->rpb_flags);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
rpb->rpb_number = number;
}
2004-07-07 05:47:12 +02:00
SINT64 DPM_gen_id(thread_db* tdbb, SLONG generator, bool initialize, SINT64 val)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ g e n _ i d
*
**************************************
*
* Functional description
* Generate relation specific value.
* If initialize is set then value of generator is made
* equal to val else generator is incremented by val.
2001-05-23 15:26:42 +02:00
* The resulting value is the result of the function.
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_TRACE_ALL)
{
printf("DPM_gen_id (generator %"SLONGFORMAT", val %" QUADFORMAT "d)\n", generator, val);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
2006-05-22 00:07:35 +02:00
const USHORT sequence = generator / dbb->dbb_page_manager.gensPerPage;
const USHORT offset = generator % dbb->dbb_page_manager.gensPerPage;
2001-05-23 15:26:42 +02:00
2006-05-22 00:07:35 +02:00
WIN window(DB_PAGE_SPACE, -1);
2003-12-31 06:36:12 +01:00
vcl* vector = dbb->dbb_gen_id_pages;
if (!vector || (sequence >= vector->count()) || !((*vector)[sequence]))
{
2001-05-23 15:26:42 +02:00
DPM_scan_pages(tdbb);
if (!(vector = dbb->dbb_gen_id_pages) ||
(sequence >= vector->count()) || !((*vector)[sequence]))
{
2004-02-02 12:02:12 +01:00
generator_page* page = (generator_page*) DPM_allocate(tdbb, &window);
page->gpg_header.pag_type = pag_ids;
2001-05-23 15:26:42 +02:00
page->gpg_sequence = sequence;
CCH_must_write(&window);
CCH_RELEASE(tdbb, &window);
2006-05-22 00:07:35 +02:00
DPM_pages(tdbb, 0, pag_ids, (ULONG) sequence, window.win_page.getPageNum());
vector = dbb->dbb_gen_id_pages =
2008-12-18 11:47:25 +01:00
vcl::newVector(*dbb->dbb_permanent, dbb->dbb_gen_id_pages, sequence + 1);
2006-05-22 00:07:35 +02:00
(*vector)[sequence] = window.win_page.getPageNum();
2001-05-23 15:26:42 +02:00
}
}
2001-12-24 03:51:06 +01:00
window.win_page = (*vector)[sequence];
2001-05-23 15:26:42 +02:00
window.win_flags = 0;
2004-02-02 12:02:12 +01:00
generator_page* page;
if (dbb->dbb_flags & DBB_read_only) {
2004-02-02 12:02:12 +01:00
page = (generator_page*) CCH_FETCH(tdbb, &window, LCK_read, pag_ids);
}
else {
2004-02-02 12:02:12 +01:00
page = (generator_page*) CCH_FETCH(tdbb, &window, LCK_write, pag_ids);
}
2001-05-23 15:26:42 +02:00
2009-06-23 06:12:59 +02:00
/* If we are in ODS >= 10, then we have a pointer to an int64 value in the
* generator page: if earlier than 10, it's a pointer to a long value.
* Pick up the right kind of pointer, based on the ODS version.
* The conditions were commented out 1999-05-14 by ChrisJ, because we
* decided that the V6 engine would only access an ODS-10 database.
* (and uncommented 2000-05-05, also by ChrisJ, when minds changed.)
*/
SINT64* const ptr = ((SINT64*) (page->gpg_values)) + offset;
2001-05-23 15:26:42 +02:00
if (val || initialize)
{
2009-06-23 06:12:59 +02:00
if (dbb->dbb_flags & DBB_read_only)
{
2004-01-09 02:21:02 +01:00
CCH_RELEASE(tdbb, &window);
ERR_post(Arg::Gds(isc_read_only_database));
}
CCH_MARK_SYSTEM(tdbb, &window);
2001-05-23 15:26:42 +02:00
if (initialize) {
*ptr = val;
}
else {
*ptr += val;
2001-05-23 15:26:42 +02:00
}
if (tdbb->getTransaction()) {
tdbb->getTransaction()->tra_flags |= TRA_write;
}
2001-05-23 15:26:42 +02:00
}
SINT64 value = *ptr;
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, &window);
return value;
}
2009-11-25 09:38:52 +01:00
bool DPM_get(thread_db* tdbb, record_param* rpb, SSHORT lock_type)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ g e t
*
**************************************
*
* Functional description
* Get a specific record in a relation. If it doesn't exit,
* just return false.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_READS)
{
printf("DPM_get (record_param %"QUADFORMAT"d, lock type %d)\n",
rpb->rpb_number.getValue(), lock_type);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
2006-05-22 00:07:35 +02:00
WIN* window = &rpb->getWindow(tdbb);
2001-05-23 15:26:42 +02:00
rpb->rpb_prior = NULL;
2009-08-20 13:18:57 +02:00
// Find starting point
USHORT pp_sequence;
SSHORT slot, line;
rpb->rpb_number.decompose(dbb->dbb_max_records, dbb->dbb_dp_per_pp, line, slot, pp_sequence);
2009-08-20 13:18:57 +02:00
// Check if the record number is OK
if (rpb->rpb_number.getValue() < 0) {
return false;
}
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// Find the next pointer page, data page, and record
pointer_page* page = get_pointer_page(tdbb, rpb->rpb_relation,
2006-05-22 00:07:35 +02:00
rpb->rpb_relation->getPages(tdbb), window, pp_sequence, LCK_read);
if (!page) {
return false;
}
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_READS_INFO)
2004-04-29 00:43:34 +02:00
printf(" record %"SLONGFORMAT":%d\n", page->ppg_page[slot], line);
2001-05-23 15:26:42 +02:00
#endif
const SLONG page_number = page->ppg_page[slot];
2009-06-23 06:12:59 +02:00
if (page_number)
{
2001-05-23 15:26:42 +02:00
CCH_HANDOFF(tdbb, window, page_number, lock_type, pag_data);
if (get_header(window, line, rpb) &&
!(rpb->rpb_flags & (rpb_blob | rpb_chained | rpb_fragment)))
{
return true;
}
2001-05-23 15:26:42 +02:00
}
CCH_RELEASE(tdbb, window);
return false;
2001-05-23 15:26:42 +02:00
}
ULONG DPM_get_blob(thread_db* tdbb,
blb* blob,
RecordNumber record_number, bool delete_flag, SLONG prior_page)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ g e t _ b l o b
*
**************************************
*
* Functional description
* Given a blob block, find the associated blob. If blob is level 0,
* get the data clump, otherwise get the vector of pointers.
2001-05-23 15:26:42 +02:00
*
* If the delete flag is set, delete the blob header after access
* and return the page number. This is a kludge, but less code
* a completely separate delete blob call.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
record_param rpb;
2006-05-22 00:07:35 +02:00
rpb.rpb_relation = blob->blb_relation;
rpb.getWindow(tdbb).win_flags = WIN_secondary;
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_READS)
{
2004-04-29 00:43:34 +02:00
printf
("DPM_get_blob (blob, record_number %"QUADFORMAT
"d, delete_flag %d, prior_page %"SLONGFORMAT")\n",
record_number.getValue(), (int) delete_flag, prior_page);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
2009-08-20 13:18:57 +02:00
// Find starting point
2003-12-31 06:36:12 +01:00
USHORT pp_sequence;
SSHORT slot, line;
record_number.decompose(dbb->dbb_max_records, dbb->dbb_dp_per_pp, line, slot, pp_sequence);
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// Find the next pointer page, data page, and record. If the page or
// record doesn't exist, or the record isn't a blob, give up and
// let somebody else complain.
2001-05-23 15:26:42 +02:00
pointer_page* ppage = get_pointer_page(tdbb, blob->blb_relation,
2008-01-16 09:46:02 +01:00
blob->blb_relation->getPages(tdbb), &rpb.getWindow(tdbb), pp_sequence, LCK_read);
2009-06-23 06:12:59 +02:00
if (!ppage)
{
2001-05-23 15:26:42 +02:00
blob->blb_flags |= BLB_damaged;
2002-09-26 11:26:40 +02:00
return 0UL;
2001-05-23 15:26:42 +02:00
}
// Do not delete this scope. It makes goto compatible with variables
// defined in their minimal scope.
{
2006-07-21 03:35:17 +02:00
const SLONG page_number = ppage->ppg_page[slot];
if (!page_number)
{
goto punt;
}
2001-05-23 15:26:42 +02:00
2006-07-21 03:35:17 +02:00
data_page* page = (data_page*) CCH_HANDOFF(tdbb,
2006-05-22 00:07:35 +02:00
&rpb.getWindow(tdbb),
2001-05-23 15:26:42 +02:00
page_number,
2009-03-01 16:42:23 +01:00
(SSHORT) (delete_flag ? LCK_write : LCK_read),
2001-05-23 15:26:42 +02:00
pag_data);
2006-07-21 03:35:17 +02:00
if (line >= page->dpg_count) {
goto punt;
}
2001-05-23 15:26:42 +02:00
2006-07-21 03:35:17 +02:00
data_page::dpg_repeat* index = &page->dpg_rpt[line];
if (index->dpg_offset == 0) {
goto punt;
}
2001-05-23 15:26:42 +02:00
2006-07-21 03:35:17 +02:00
blh* header = (blh*) ((SCHAR *) page + index->dpg_offset);
if (!(header->blh_flags & rhd_blob)) {
goto punt;
}
2001-05-23 15:26:42 +02:00
2006-07-21 03:35:17 +02:00
// We've got the blob header and everything looks ducky. Get the header
// fields.
2001-05-23 15:26:42 +02:00
2006-07-21 03:35:17 +02:00
blob->blb_lead_page = header->blh_lead_page;
blob->blb_max_sequence = header->blh_max_sequence;
blob->blb_count = header->blh_count;
blob->blb_length = header->blh_length;
blob->blb_max_segment = header->blh_max_segment;
blob->blb_level = header->blh_level;
blob->blb_sub_type = header->blh_sub_type;
blob->blb_charset = header->blh_charset;
2001-05-23 15:26:42 +02:00
2006-07-21 03:35:17 +02:00
// Unless this is the only attachment, don't allow the sequential scan
// of very large blobs to flush pages used by other attachments.
2001-05-23 15:26:42 +02:00
Jrd::Attachment* attachment = tdbb->getAttachment();
2008-12-18 11:47:25 +01:00
if (attachment && (attachment != dbb->dbb_attachments || attachment->att_next))
{
2009-08-20 13:18:57 +02:00
// If the blob has more pages than the page buffer cache then mark
// it as large. If this is a database backup then mark any blob as
// large as the cumulative effect of scanning many small blobs is
// equivalent to scanning single large blobs.
2006-07-21 03:35:17 +02:00
if (blob->blb_max_sequence > dbb->dbb_bcb->bcb_count ||
attachment->att_flags & ATT_gbak_attachment)
{
blob->blb_flags |= BLB_large_scan;
}
}
2001-05-23 15:26:42 +02:00
2006-07-21 03:35:17 +02:00
if (header->blh_flags & rhd_stream_blob) {
blob->blb_flags |= BLB_stream;
}
2001-05-23 15:26:42 +02:00
2006-07-21 03:35:17 +02:00
if (header->blh_flags & rhd_damaged) {
goto punt;
}
2001-05-23 15:26:42 +02:00
2006-07-21 03:35:17 +02:00
// Retrieve the data either into page clump (level 0) or page vector (levels
// 1 and 2).
2001-05-23 15:26:42 +02:00
2006-07-21 03:35:17 +02:00
const USHORT length = index->dpg_length - BLH_SIZE;
const UCHAR* q = (UCHAR *) header->blh_page;
2001-05-23 15:26:42 +02:00
2009-06-23 06:12:59 +02:00
if (blob->blb_level == 0)
{
2006-07-21 03:35:17 +02:00
blob->blb_space_remaining = length;
2008-01-16 09:46:02 +01:00
if (length)
memcpy(blob->getBuffer(), q, length);
}
2009-06-23 06:12:59 +02:00
else
{
2006-07-21 03:35:17 +02:00
vcl* vector = blob->blb_pages;
if (!vector) {
2008-12-18 11:47:25 +01:00
vector = blob->blb_pages = vcl::newVector(*blob->blb_transaction->tra_pool, 0);
2006-07-21 03:35:17 +02:00
}
vector->resize(length / sizeof(SLONG));
2008-01-16 09:46:02 +01:00
memcpy(vector->memPtr(), q, length);
2001-05-23 15:26:42 +02:00
}
2009-06-23 06:12:59 +02:00
if (!delete_flag)
{
2006-07-21 03:35:17 +02:00
CCH_RELEASE(tdbb, &rpb.getWindow(tdbb));
return 0UL;
}
2001-05-23 15:26:42 +02:00
2006-07-21 03:35:17 +02:00
// We've been asked (nicely) to delete the blob. So do so.
2001-05-23 15:26:42 +02:00
2006-07-21 03:35:17 +02:00
rpb.rpb_relation = blob->blb_relation;
rpb.rpb_page = rpb.getWindow(tdbb).win_page.getPageNum();
rpb.rpb_line = line;
2006-07-21 03:35:17 +02:00
DPM_delete(tdbb, &rpb, prior_page);
return rpb.rpb_page;
} // scope
2001-05-23 15:26:42 +02:00
punt:
2006-05-22 00:07:35 +02:00
CCH_RELEASE(tdbb, &rpb.getWindow(tdbb));
2001-05-23 15:26:42 +02:00
blob->blb_flags |= BLB_damaged;
2002-09-26 11:26:40 +02:00
return 0UL;
2001-05-23 15:26:42 +02:00
}
2009-11-25 09:38:52 +01:00
bool DPM_next(thread_db* tdbb, record_param* rpb, USHORT lock_type, bool onepage)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ n e x t
*
**************************************
*
* Functional description
* Get the next record in a stream.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_READS)
printf("DPM_next (record_param %"QUADFORMAT"d)\n", rpb->rpb_number.getValue());
2001-05-23 15:26:42 +02:00
#endif
2006-05-22 00:07:35 +02:00
WIN* window = &rpb->getWindow(tdbb);
RelationPages* relPages = rpb->rpb_relation->getPages(tdbb);
2009-06-23 06:12:59 +02:00
if (window->win_flags & WIN_large_scan)
{
2009-08-20 13:18:57 +02:00
// Try to account for staggered execution of large sequential scans.
2001-05-23 15:26:42 +02:00
2008-12-18 11:47:25 +01:00
window->win_scans = rpb->rpb_relation->rel_scan_count - rpb->rpb_org_scans;
if (window->win_scans < 1) {
2001-05-23 15:26:42 +02:00
window->win_scans = rpb->rpb_relation->rel_scan_count;
}
2001-05-23 15:26:42 +02:00
}
rpb->rpb_prior = NULL;
2009-08-20 13:18:57 +02:00
// Find starting point
2001-05-23 15:26:42 +02:00
rpb->rpb_number.increment();
2001-05-23 15:26:42 +02:00
2003-12-31 06:36:12 +01:00
SSHORT slot, line;
USHORT pp_sequence;
rpb->rpb_number.decompose(dbb->dbb_max_records, dbb->dbb_dp_per_pp, line, slot, pp_sequence);
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
2003-12-31 06:36:12 +01:00
if (debug_flag > DEBUG_READS_INFO) {
2009-06-23 06:12:59 +02:00
printf(" pointer, slot, and line %d:%d%d\n", pp_sequence, slot, line);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
// If i'm a sweeper i don't need to look at swept pages. Also i should
2009-12-28 12:46:55 +01:00
// check processed pages if they were swept.
const bool sweeper = (tdbb->tdbb_flags & TDBB_sweeper);
if (sweeper && (pp_sequence || slot) && !line)
{
// The last record at previous data page was returned to caller.
// It is time now to check if previous data page was swept.
const RecordNumber saveRecNo = rpb->rpb_number;
rpb->rpb_number.decrement();
check_swept(tdbb, rpb);
rpb->rpb_number = saveRecNo;
}
2009-08-20 13:18:57 +02:00
// Find the next pointer page, data page, and record
2001-05-23 15:26:42 +02:00
while (true)
{
const pointer_page* ppage = get_pointer_page(tdbb, rpb->rpb_relation,
2006-05-22 00:07:35 +02:00
relPages, window, pp_sequence, LCK_read);
if (!ppage) {
2009-08-20 13:18:57 +02:00
BUGCHECK(249); // msg 249 pointer page vanished from DPM_next
}
2006-05-22 00:07:35 +02:00
for (; slot >= 0 && slot < ppage->ppg_count;)
{
2003-12-31 06:36:12 +01:00
const SLONG page_number = ppage->ppg_page[slot];
2009-12-18 00:59:07 +01:00
const UCHAR* bits = (UCHAR*) (ppage->ppg_page + dbb->dbb_dp_per_pp);
if (page_number && !PPG_DP_BIT_TEST(bits, slot, ppg_dp_secondary) &&
2009-12-28 12:46:55 +01:00
(!sweeper || !PPG_DP_BIT_TEST(bits, slot, ppg_dp_swept)) )
{
2001-05-23 15:26:42 +02:00
#ifdef SUPERSERVER_V2
2009-08-20 13:18:57 +02:00
// Perform sequential prefetch of relation's data pages.
// This may need more work for scrollable cursors.
2001-05-23 15:26:42 +02:00
if (!onepage && !line)
{
if (!(slot % dbb->dbb_prefetch_sequence))
{
2003-12-31 06:36:12 +01:00
SLONG pages[PREFETCH_MAX_PAGES + 1];
USHORT slot2 = slot;
USHORT i;
2008-12-18 11:47:25 +01:00
for (i = 0; i < dbb->dbb_prefetch_pages && slot2 < ppage->ppg_count;)
2003-12-31 06:36:12 +01:00
{
pages[i++] = ppage->ppg_page[slot2++];
}
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// If no more data pages, piggyback next pointer page.
2001-05-23 15:26:42 +02:00
if (slot2 >= ppage->ppg_count) {
2001-05-23 15:26:42 +02:00
pages[i++] = ppage->ppg_next;
}
2001-05-23 15:26:42 +02:00
CCH_PREFETCH(tdbb, pages, i);
}
}
#endif
const data_page* dpage = (data_page*) CCH_HANDOFF(tdbb, window,
page_number, lock_type, pag_data);
2009-06-25 04:29:13 +02:00
for (; line >= 0 && line < dpage->dpg_count; ++line)
{
2008-12-18 11:47:25 +01:00
if (get_header(window, line, rpb) &&
!(rpb->rpb_flags & (rpb_blob | rpb_chained | rpb_fragment)))
{
2008-12-18 11:47:25 +01:00
rpb->rpb_number.compose(dbb->dbb_max_records, dbb->dbb_dp_per_pp,
line, slot, pp_sequence);
return true;
2001-05-23 15:26:42 +02:00
}
}
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// Prevent large relations from emptying cache. When scrollable
// cursors are surfaced, this logic may need to be revisited.
2001-05-23 15:26:42 +02:00
if (window->win_flags & WIN_large_scan) {
2001-05-23 15:26:42 +02:00
CCH_RELEASE_TAIL(tdbb, window);
}
2001-05-23 15:26:42 +02:00
else if (window->win_flags & WIN_garbage_collector &&
window->win_flags & WIN_garbage_collect)
{
2001-05-23 15:26:42 +02:00
CCH_RELEASE_TAIL(tdbb, window);
window->win_flags &= ~WIN_garbage_collect;
}
else {
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, window);
}
2001-05-23 15:26:42 +02:00
if (sweeper)
{
// The last record at data page was not returned to caller.
// It is time now to check if this data page was swept.
const RecordNumber saveRecNo = rpb->rpb_number;
rpb->rpb_number.compose(dbb->dbb_max_records, dbb->dbb_dp_per_pp,
line, slot, pp_sequence);
rpb->rpb_number.decrement();
check_swept(tdbb, rpb);
rpb->rpb_number = saveRecNo;
}
if (onepage) {
return false;
}
2001-05-23 15:26:42 +02:00
2008-12-18 11:47:25 +01:00
if (!(ppage = get_pointer_page(tdbb, rpb->rpb_relation, relPages, window,
pp_sequence, LCK_read)))
{
2009-08-20 13:18:57 +02:00
BUGCHECK(249); // msg 249 pointer page vanished from DPM_next
2008-12-18 11:47:25 +01:00
}
2001-05-23 15:26:42 +02:00
}
2009-06-23 06:12:59 +02:00
if (onepage)
{
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, window);
return false;
2001-05-23 15:26:42 +02:00
}
slot++;
line = 0;
2001-05-23 15:26:42 +02:00
}
const UCHAR flags = ppage->ppg_header.pag_flags;
pp_sequence++;
slot = 0;
line = 0;
2001-05-23 15:26:42 +02:00
if (window->win_flags & WIN_large_scan) {
2001-05-23 15:26:42 +02:00
CCH_RELEASE_TAIL(tdbb, window);
}
else {
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, window);
}
2006-05-22 04:36:53 +02:00
if (flags & ppg_eof || onepage) {
return false;
}
2001-05-23 15:26:42 +02:00
}
}
2008-12-18 11:47:25 +01:00
void DPM_pages(thread_db* tdbb, SSHORT rel_id, int type, ULONG sequence, SLONG page)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ p a g e s
*
**************************************
*
* Functional description
* Write a record in RDB$PAGES.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_TRACE_ALL)
{
printf("DPM_pages (rel_id %d, type %d, sequence %"ULONGFORMAT", page %"SLONGFORMAT")\n",
rel_id, type, sequence, page);
}
2001-05-23 15:26:42 +02:00
#endif
jrd_req* request = CMP_find_request(tdbb, irq_s_pages, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
STORE(REQUEST_HANDLE request)
2008-01-16 09:46:02 +01:00
X IN RDB$PAGES
X.RDB$RELATION_ID = rel_id;
2001-05-23 15:26:42 +02:00
X.RDB$PAGE_TYPE = type;
X.RDB$PAGE_SEQUENCE = sequence;
X.RDB$PAGE_NUMBER = page;
END_STORE
if (!REQUEST(irq_s_pages))
REQUEST(irq_s_pages) = request;
2001-05-23 15:26:42 +02:00
}
#ifdef SUPERSERVER_V2
SLONG DPM_prefetch_bitmap(thread_db* tdbb, jrd_rel* relation, PageBitmap* bitmap, RecordNumber number)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ p r e f e t c h _ b i t m a p
*
**************************************
*
* Functional description
* Generate a vector of corresponding data page
* numbers from a bitmap of relation record numbers.
* Return the bitmap record number of where we left
* off.
*
**************************************/
SET_TDBB(tdbb);
2009-08-20 13:18:57 +02:00
// Empty and singular bitmaps aren't worth prefetch effort.
2001-05-23 15:26:42 +02:00
if (!bitmap || bitmap->sbm_state != SBM_PLURAL) {
2001-05-23 15:26:42 +02:00
return number;
}
2001-05-23 15:26:42 +02:00
Database* dbb = tdbb->getDatabase();
WIN window(-1);
2003-12-31 06:36:12 +01:00
SLONG pages[PREFETCH_MAX_PAGES];
// CVC: If this routine is activated some day, what's the default value
// of this variable? I set it to -1.
SLONG prefetch_number = -1;
2001-05-23 15:26:42 +02:00
2003-12-31 06:36:12 +01:00
USHORT i;
for (i = 0; i < dbb->dbb_prefetch_pages;)
{
2003-12-31 06:36:12 +01:00
SLONG dp_sequence;
SSHORT line, slot;
USHORT pp_sequence;
2001-05-23 15:26:42 +02:00
DECOMPOSE(number, dbb->dbb_max_records, dp_sequence, line);
DECOMPOSE(dp_sequence, dbb->dbb_dp_per_pp, pp_sequence, slot);
2008-12-18 11:47:25 +01:00
const pointer_page* ppage = get_pointer_page(tdbb, relation, &window, pp_sequence, LCK_read);
2009-06-23 06:12:59 +02:00
if (!ppage)
{
BUGCHECK(249);
2009-08-20 13:18:57 +02:00
// msg 249 pointer page vanished from DPM_prefetch_bitmap
}
2008-12-18 11:47:25 +01:00
pages[i] = (slot >= 0 && slot < ppage->ppg_count) ? ppage->ppg_page[slot] : 0;
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, &window);
if (i++ < dbb->dbb_prefetch_sequence) {
2001-05-23 15:26:42 +02:00
prefetch_number = number;
}
2001-05-23 15:26:42 +02:00
number = ((dp_sequence + 1) * dbb->dbb_max_records) - 1;
if (!SBM_next(bitmap, &number, RSE_get_forward)) {
2001-05-23 15:26:42 +02:00
break;
}
2001-05-23 15:26:42 +02:00
}
CCH_PREFETCH(tdbb, pages, i);
return prefetch_number;
}
#endif
void DPM_scan_pages( thread_db* tdbb)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ s c a n _ p a g e s
*
**************************************
*
* Functional description
* Scan RDB$PAGES.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_TRACE_ALL)
2004-04-29 00:43:34 +02:00
printf("DPM_scan_pages ()\n");
2001-05-23 15:26:42 +02:00
#endif
2009-08-20 13:18:57 +02:00
// Special case update of RDB$PAGES pointer page vector to avoid
// infinite recursion from this internal request when RDB$PAGES
// has been extended with another pointer page.
2001-05-23 15:26:42 +02:00
2003-12-31 06:36:12 +01:00
jrd_rel* relation = MET_relation(tdbb, 0);
2006-05-22 00:07:35 +02:00
RelationPages* relPages = relation->getBasePages();
vcl** address = &relPages->rel_pages;
2003-12-31 06:36:12 +01:00
vcl* vector = *address;
size_t sequence = vector->count() - 1;
2006-05-22 00:07:35 +02:00
WIN window(relPages->rel_pg_space_id, (*vector)[sequence]);
2004-02-02 12:02:12 +01:00
pointer_page* ppage = (pointer_page*) CCH_FETCH(tdbb, &window, LCK_read, pag_pointer);
2001-05-23 15:26:42 +02:00
2009-06-23 06:12:59 +02:00
while (ppage->ppg_next)
{
2001-05-23 15:26:42 +02:00
++sequence;
2001-12-24 03:51:06 +01:00
vector->resize(sequence + 1);
(*vector)[sequence] = ppage->ppg_next;
2008-12-18 11:47:25 +01:00
ppage = (pointer_page*) CCH_HANDOFF(tdbb, &window, ppage->ppg_next, LCK_read, pag_pointer);
2001-05-23 15:26:42 +02:00
}
CCH_RELEASE(tdbb, &window);
jrd_req* request = CMP_find_request(tdbb, irq_r_pages, IRQ_REQUESTS);
2001-05-23 15:26:42 +02:00
FOR(REQUEST_HANDLE request) X IN RDB$PAGES
if (!REQUEST(irq_r_pages))
REQUEST(irq_r_pages) = request;
2001-05-23 15:26:42 +02:00
relation = MET_relation(tdbb, X.RDB$RELATION_ID);
2006-05-22 04:36:53 +02:00
relPages = relation->getBasePages();
2001-05-23 15:26:42 +02:00
sequence = X.RDB$PAGE_SEQUENCE;
2009-01-20 09:33:59 +01:00
switch (X.RDB$PAGE_TYPE)
{
2001-05-23 15:26:42 +02:00
case pag_root:
2006-05-22 00:07:35 +02:00
relPages->rel_index_root = X.RDB$PAGE_NUMBER;
2001-05-23 15:26:42 +02:00
continue;
case pag_pointer:
2006-05-22 00:07:35 +02:00
address = &relPages->rel_pages;
2001-05-23 15:26:42 +02:00
break;
case pag_transactions:
address = &dbb->dbb_t_pages;
break;
case pag_ids:
address = &dbb->dbb_gen_id_pages;
break;
default:
2009-08-20 13:18:57 +02:00
CORRUPT(257); // msg 257 bad record in RDB$PAGES
2001-05-23 15:26:42 +02:00
}
2008-12-18 11:47:25 +01:00
vector = *address = vcl::newVector(*dbb->dbb_permanent, *address, sequence + 1);
2001-12-24 03:51:06 +01:00
(*vector)[sequence] = X.RDB$PAGE_NUMBER;
2001-05-23 15:26:42 +02:00
END_FOR;
if (!REQUEST(irq_r_pages))
REQUEST(irq_r_pages) = request;
2001-05-23 15:26:42 +02:00
}
void DPM_store( thread_db* tdbb, record_param* rpb, PageStack& stack, USHORT type)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ s t o r e
*
**************************************
*
* Functional description
* Store a new record in a relation. If we can put it on a
* specific page, so much the better.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES)
{
printf("DPM_store (record_param %"QUADFORMAT"d, stack, type %d)\n",
rpb->rpb_number.getValue(), type);
2003-12-31 06:36:12 +01:00
}
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf
2003-04-01 19:58:19 +02:00
(" record to store %"SLONGFORMAT":%d transaction %"SLONGFORMAT
" back %"SLONGFORMAT":%d fragment %"SLONGFORMAT":%d flags %d\n",
rpb->rpb_page, rpb->rpb_line, rpb->rpb_transaction_nr,
2001-05-23 15:26:42 +02:00
rpb->rpb_b_page, rpb->rpb_b_line, rpb->rpb_f_page,
rpb->rpb_f_line, rpb->rpb_flags);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
DataComprControl dcc(*tdbb->getDefaultPool());
const USHORT size = SQZ_length((SCHAR*) rpb->rpb_address, (int) rpb->rpb_length, &dcc);
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// If the record isn't going to fit on a page, even if fragmented,
// handle it a little differently.
2001-05-23 15:26:42 +02:00
2009-06-23 06:12:59 +02:00
if (size > dbb->dbb_page_size - (sizeof(data_page) + RHD_SIZE))
{
store_big_record(tdbb, rpb, stack, &dcc, size, type);
2001-05-23 15:26:42 +02:00
return;
}
2003-12-31 06:36:12 +01:00
SLONG fill = (RHDF_SIZE - RHD_SIZE) - size;
if (fill < 0) {
2001-05-23 15:26:42 +02:00
fill = 0;
}
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// Accomodate max record size i.e. 64K
2003-12-31 06:36:12 +01:00
const SLONG length = RHD_SIZE + size + fill;
rhd* header = locate_space(tdbb, rpb, (SSHORT)length, stack, NULL, type);
2001-05-23 15:26:42 +02:00
header->rhd_flags = rpb->rpb_flags;
header->rhd_transaction = rpb->rpb_transaction_nr;
2001-05-23 15:26:42 +02:00
header->rhd_format = rpb->rpb_format_number;
header->rhd_b_page = rpb->rpb_b_page;
header->rhd_b_line = rpb->rpb_b_line;
SQZ_fast(&dcc, (SCHAR*) rpb->rpb_address, (SCHAR*) header->rhd_data);
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf
2003-04-01 19:58:19 +02:00
(" record %"SLONGFORMAT":%d, length %"SLONGFORMAT
", rpb_flags %d, f_page %"SLONGFORMAT":%d, b_page %"SLONGFORMAT
":%d\n",
rpb->rpb_page, rpb->rpb_line, length, rpb->rpb_flags,
2001-05-23 15:26:42 +02:00
rpb->rpb_f_page, rpb->rpb_f_line, rpb->rpb_b_page,
rpb->rpb_b_line);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
if (fill) {
memset(header->rhd_data + size, 0, fill);
2001-05-23 15:26:42 +02:00
}
Ods::pag *page = rpb->getWindow(tdbb).win_buffer;
if (page->pag_flags & dpg_swept)
{
page->pag_flags &= ~dpg_swept;
mark_full(tdbb, rpb);
}
else
2009-12-28 12:46:55 +01:00
CCH_RELEASE(tdbb, &rpb->getWindow(tdbb));
2001-05-23 15:26:42 +02:00
}
RecordNumber DPM_store_blob(thread_db* tdbb, blb* blob, Record* record)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ s t o r e _ b l o b
*
**************************************
*
* Functional description
* Store a blob on a data page. Not so hard, all in all.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
record_param rpb;
2006-05-22 00:07:35 +02:00
//rpb.getWindow(tdbb).win_flags = 0; redundant.
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_WRITES)
2004-04-29 00:43:34 +02:00
printf("DPM_store_blob (blob, record)\n");
2001-05-23 15:26:42 +02:00
#endif
2009-08-20 13:18:57 +02:00
// Figure out length of blob on page. Remember that blob can either
// be a clump of data or a vector of page pointers.
2003-12-31 06:36:12 +01:00
vcl* vector = 0;
USHORT length;
const UCHAR* q;
PageStack stack;
2008-01-16 09:46:02 +01:00
Firebird::Array<UCHAR> buffer;
2008-01-16 09:46:02 +01:00
if (blob->blb_level == 0)
{
2001-05-23 15:26:42 +02:00
length = blob->blb_clump_size - blob->blb_space_remaining;
2008-01-16 09:46:02 +01:00
if (!blob->hasBuffer())
{
if (blob->blb_temp_size > 0)
{
blob->blb_transaction->getBlobSpace()->read(
2008-12-18 11:47:25 +01:00
blob->blb_temp_offset, buffer.getBuffer(blob->blb_temp_size), blob->blb_temp_size);
2008-01-16 09:46:02 +01:00
q = buffer.begin();
}
else
{
fb_assert(length == 0);
q = NULL;
}
}
else
q = blob->getBuffer();
if (q)
q = (UCHAR*) ((blob_page*) q)->blp_page;
2001-05-23 15:26:42 +02:00
}
2008-01-16 09:46:02 +01:00
else
{
2001-05-23 15:26:42 +02:00
vector = blob->blb_pages;
2001-12-24 03:51:06 +01:00
length = vector->count() * sizeof(SLONG);
q = (UCHAR *) (vector->begin());
2009-08-20 13:18:57 +02:00
// Figure out precedence pages, if any
2003-12-31 06:36:12 +01:00
vcl::iterator ptr, end;
2004-05-09 07:48:33 +02:00
for (ptr = vector->begin(), end = vector->end(); ptr < end; ++ptr) {
stack.push(*ptr);
}
}
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// Locate space to store blob
2001-05-23 15:26:42 +02:00
rpb.rpb_relation = blob->blb_relation;
2008-12-18 11:47:25 +01:00
blh* header = (blh*) locate_space(tdbb, &rpb, (SSHORT)(BLH_SIZE + length),
stack, record, DPM_other);
2001-05-23 15:26:42 +02:00
header->blh_flags = rhd_blob;
if (blob->blb_flags & BLB_stream) {
2001-05-23 15:26:42 +02:00
header->blh_flags |= rhd_stream_blob;
}
2001-05-23 15:26:42 +02:00
if (blob->blb_level) {
2001-05-23 15:26:42 +02:00
header->blh_flags |= rhd_large;
}
2001-05-23 15:26:42 +02:00
header->blh_lead_page = blob->blb_lead_page;
header->blh_max_sequence = blob->blb_max_sequence;
header->blh_count = blob->blb_count;
header->blh_max_segment = blob->blb_max_segment;
header->blh_length = blob->blb_length;
header->blh_level = blob->blb_level;
header->blh_sub_type = blob->blb_sub_type;
header->blh_charset = blob->blb_charset;
2001-05-23 15:26:42 +02:00
2008-01-16 09:46:02 +01:00
if (length)
memcpy(header->blh_page, q, length);
2001-05-23 15:26:42 +02:00
2006-05-22 00:07:35 +02:00
data_page* page = (data_page*) rpb.getWindow(tdbb).win_buffer;
2009-06-23 06:12:59 +02:00
if (blob->blb_level && !(page->dpg_header.pag_flags & dpg_large))
{
page->dpg_header.pag_flags |= dpg_large;
2001-05-23 15:26:42 +02:00
mark_full(tdbb, &rpb);
}
else {
2006-05-22 00:07:35 +02:00
CCH_RELEASE(tdbb, &rpb.getWindow(tdbb));
}
2001-05-23 15:26:42 +02:00
return rpb.rpb_number;
}
void DPM_rewrite_header( thread_db* tdbb, record_param* rpb)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ r e w r i t e _ h e a d e r
*
**************************************
*
* Functional description
* Re-write benign fields in record header. This is mostly used
* to purge out old versions.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_WRITES)
printf("DPM_rewrite_header (record_param %"QUADFORMAT"d)\n", rpb->rpb_number.getValue());
2001-05-23 15:26:42 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
printf(" record %"SLONGFORMAT":%d\n", rpb->rpb_page, rpb->rpb_line);
2001-05-23 15:26:42 +02:00
#endif
2006-05-22 00:07:35 +02:00
WIN* window = &rpb->getWindow(tdbb);
2004-02-02 12:02:12 +01:00
data_page* page = (data_page*) window->win_buffer;
rhd* header = (rhd*) ((SCHAR *) page + page->dpg_rpt[rpb->rpb_line].dpg_offset);
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf
2003-04-01 19:58:19 +02:00
(" old flags %d, old transaction %"SLONGFORMAT
", old format %d, old back record %"SLONGFORMAT":%d\n",
2004-03-30 06:10:52 +02:00
header->rhd_flags, header->rhd_transaction, (int) header->rhd_format,
2001-05-23 15:26:42 +02:00
header->rhd_b_page, header->rhd_b_line);
2004-04-29 00:43:34 +02:00
printf
2003-04-01 19:58:19 +02:00
(" new flags %d, new transaction %"SLONGFORMAT
", new format %d, new back record %"SLONGFORMAT":%d\n",
rpb->rpb_flags, rpb->rpb_transaction_nr, rpb->rpb_format_number,
2001-05-23 15:26:42 +02:00
rpb->rpb_b_page, rpb->rpb_b_line);
}
#endif
header->rhd_flags = rpb->rpb_flags;
header->rhd_transaction = rpb->rpb_transaction_nr;
2001-05-23 15:26:42 +02:00
header->rhd_format = rpb->rpb_format_number;
header->rhd_b_page = rpb->rpb_b_page;
header->rhd_b_line = rpb->rpb_b_line;
}
2009-11-25 09:38:52 +01:00
void DPM_update( thread_db* tdbb, record_param* rpb, PageStack* stack, const jrd_tra* transaction)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* D P M _ u p d a t e
*
**************************************
*
* Functional description
* Replace an existing record.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES)
{
printf("DPM_update (record_param %"QUADFORMAT"d, stack, transaction %"SLONGFORMAT")\n",
rpb->rpb_number.getValue(), transaction ? transaction->tra_number : 0);
2003-12-31 06:36:12 +01:00
}
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf
2003-04-01 19:58:19 +02:00
(" record %"SLONGFORMAT":%d transaction %"SLONGFORMAT" back %"
SLONGFORMAT":%d fragment %"SLONGFORMAT":%d flags %d\n",
rpb->rpb_page, rpb->rpb_line, rpb->rpb_transaction_nr,
2001-05-23 15:26:42 +02:00
rpb->rpb_b_page, rpb->rpb_b_line, rpb->rpb_f_page,
rpb->rpb_f_line, rpb->rpb_flags);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
2009-08-20 13:18:57 +02:00
// Mark the page as modified, then figure out the compressed length of the
// replacement record.
2001-05-23 15:26:42 +02:00
DEBUG
if (stack)
{
while (stack->hasData())
{
2006-05-22 00:07:35 +02:00
CCH_precedence(tdbb, &rpb->getWindow(tdbb), stack->pop());
}
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
2006-05-22 00:07:35 +02:00
CCH_precedence(tdbb, &rpb->getWindow(tdbb), -rpb->rpb_transaction_nr);
CCH_MARK(tdbb, &rpb->getWindow(tdbb));
data_page* page = (data_page*) rpb->getWindow(tdbb).win_buffer;
DataComprControl dcc(*tdbb->getDefaultPool());
const USHORT size = SQZ_length((SCHAR*) rpb->rpb_address, (int) rpb->rpb_length, &dcc);
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// It is critical that the record be padded, if necessary, to the length of
// a fragmented record header. Compute the amount of fill required.
2001-05-23 15:26:42 +02:00
2003-12-31 06:36:12 +01:00
SLONG fill = (RHDF_SIZE - RHD_SIZE) - size;
if (fill < 0) {
2001-05-23 15:26:42 +02:00
fill = 0;
}
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// Accomodate max record size i.e. 64K
2003-12-31 06:36:12 +01:00
const SLONG length = ROUNDUP(RHD_SIZE + size + fill, ODS_ALIGNMENT);
const SSHORT slot = rpb->rpb_line;
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// Find space on page
2003-12-31 06:36:12 +01:00
SSHORT space = dbb->dbb_page_size;
const SSHORT top = HIGH_WATER(page->dpg_count);
SSHORT available = dbb->dbb_page_size - top;
const SSHORT old_length = page->dpg_rpt[slot].dpg_length;
2001-05-23 15:26:42 +02:00
page->dpg_rpt[slot].dpg_length = 0;
2004-02-02 12:02:12 +01:00
const data_page::dpg_repeat* index = page->dpg_rpt;
2008-12-18 11:47:25 +01:00
for (const data_page::dpg_repeat* const end = index + page->dpg_count; index < end; index++)
2003-12-31 06:36:12 +01:00
{
const SSHORT offset = index->dpg_offset;
2009-06-23 06:12:59 +02:00
if (offset)
{
2001-05-23 15:26:42 +02:00
available -= ROUNDUP(index->dpg_length, ODS_ALIGNMENT);
space = MIN(space, offset);
}
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
2009-06-23 06:12:59 +02:00
if (length > available)
{
2001-05-23 15:26:42 +02:00
fragment(tdbb, rpb, available, &dcc, old_length, transaction);
return;
}
space -= length;
if (space < top) {
2001-05-23 15:26:42 +02:00
space = DPM_compress(tdbb, page) - length;
}
2001-05-23 15:26:42 +02:00
page->dpg_rpt[slot].dpg_offset = space;
page->dpg_rpt[slot].dpg_length = RHD_SIZE + size + fill;
rhd* header = (rhd*) ((SCHAR *) page + space);
2001-05-23 15:26:42 +02:00
header->rhd_flags = rpb->rpb_flags;
header->rhd_transaction = rpb->rpb_transaction_nr;
2001-05-23 15:26:42 +02:00
header->rhd_format = rpb->rpb_format_number;
header->rhd_b_page = rpb->rpb_b_page;
header->rhd_b_line = rpb->rpb_b_line;
SQZ_fast(&dcc, (SCHAR*) rpb->rpb_address, (SCHAR*) header->rhd_data);
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf
2003-04-01 19:58:19 +02:00
(" record %"SLONGFORMAT
":%d, dpg_length %d, rpb_flags %d, rpb_f record %"SLONGFORMAT
":%d, rpb_b record %"SLONGFORMAT":%d\n",
rpb->rpb_page, rpb->rpb_line, page->dpg_rpt[slot].dpg_length,
2001-05-23 15:26:42 +02:00
rpb->rpb_flags, rpb->rpb_f_page, rpb->rpb_f_line,
rpb->rpb_b_page, rpb->rpb_b_line);
}
2001-05-23 15:26:42 +02:00
#endif
if (fill) {
memset(header->rhd_data + size, 0, fill);
2001-05-23 15:26:42 +02:00
}
if (page->dpg_header.pag_flags & dpg_swept)
{
page->dpg_header.pag_flags &= ~dpg_swept;
mark_full(tdbb, rpb);
}
else
2009-12-28 12:46:55 +01:00
CCH_RELEASE(tdbb, &rpb->getWindow(tdbb));
2001-05-23 15:26:42 +02:00
}
static void check_swept(thread_db* tdbb, record_param* rpb)
{
/**************************************
*
* c h e c k _ s w e p t
*
**************************************
*
* Functional description
2009-12-28 12:46:55 +01:00
* Check if data page has primary record versions only and all of them
* created by committed transactions. Such data page should be skipped
* by sweep as sweep have nothing to do on it.
* Mark swept data page and its pointer page by corresponding flag.
*
**************************************/
Database* dbb = tdbb->getDatabase();
jrd_tra *transaction = tdbb->getTransaction();
WIN* window = &rpb->getWindow(tdbb);
RelationPages* relPages = rpb->rpb_relation->getPages(tdbb);
USHORT pp_sequence;
SSHORT slot, line;
2009-12-28 12:46:55 +01:00
rpb->rpb_number.decompose(dbb->dbb_max_records, dbb->dbb_dp_per_pp,
line, slot, pp_sequence);
2009-12-28 12:46:55 +01:00
pointer_page* ppage =
get_pointer_page(tdbb, rpb->rpb_relation, relPages, window, pp_sequence, LCK_read);
if (!ppage)
return;
const UCHAR* bits = (UCHAR*) (ppage->ppg_page + dbb->dbb_dp_per_pp);
if (slot >= ppage->ppg_count || !ppage->ppg_page[slot] ||
2009-12-28 12:46:55 +01:00
PPG_DP_BIT_TEST(bits, slot, ppg_dp_secondary | ppg_dp_swept))
{
CCH_RELEASE(tdbb, window);
return;
}
2009-12-28 12:46:55 +01:00
data_page* dpage = (data_page*)
CCH_HANDOFF(tdbb, window, ppage->ppg_page[slot], LCK_write, pag_data);
for (int line = 0; line < dpage->dpg_count; ++line)
{
const data_page::dpg_repeat* index = &dpage->dpg_rpt[line];
2009-12-28 12:46:55 +01:00
if (index->dpg_offset)
{
rhd* header = (rhd*) ((SCHAR*) dpage + index->dpg_offset);
if (header->rhd_transaction > transaction->tra_oldest ||
header->rhd_flags & (rpb_blob | rpb_chained | rpb_fragment) ||
header->rhd_b_page)
{
CCH_RELEASE_TAIL(tdbb, window);
return;
}
}
}
CCH_MARK(tdbb, window);
dpage->dpg_header.pag_flags |= dpg_swept;
mark_full(tdbb, rpb);
}
2006-05-22 04:36:53 +02:00
static void delete_tail(thread_db* tdbb, rhdf* header, const USHORT page_space, USHORT length)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* d e l e t e _ t a i l
*
**************************************
*
* Functional description
* Delete the tail of a large object. This is called only from
* DPM_delete_relation.
*
**************************************/
SET_TDBB(tdbb);
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_WRITES)
2004-04-29 00:43:34 +02:00
printf("delete_tail (header, length)\n");
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf(" transaction %"SLONGFORMAT" flags %d fragment %"
2003-04-01 19:58:19 +02:00
SLONGFORMAT":%d back %"SLONGFORMAT":%d\n",
2001-05-23 15:26:42 +02:00
header->rhdf_transaction, header->rhdf_flags,
header->rhdf_f_page, header->rhdf_f_line,
header->rhdf_b_page, header->rhdf_b_line);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
2006-05-22 00:07:35 +02:00
WIN window(page_space, -1);
2001-05-23 15:26:42 +02:00
window.win_flags = WIN_large_scan;
window.win_scans = 1;
2009-08-20 13:18:57 +02:00
// If the object isn't a blob, things are a little simpler.
2001-05-23 15:26:42 +02:00
if (!(header->rhdf_flags & rhd_blob))
{
2003-12-31 06:36:12 +01:00
SLONG page_number = header->rhdf_f_page;
while (true)
{
2001-05-23 15:26:42 +02:00
window.win_page = page_number;
2004-02-02 12:02:12 +01:00
data_page* dpage = (data_page*) CCH_FETCH(tdbb, &window, LCK_read, pag_data);
header = (rhdf*) ((UCHAR *) dpage + dpage->dpg_rpt[0].dpg_offset);
2003-12-31 06:36:12 +01:00
const USHORT flags = header->rhdf_flags;
2001-05-23 15:26:42 +02:00
page_number = header->rhdf_f_page;
CCH_RELEASE_TAIL(tdbb, &window);
PAG_release_page(tdbb, window.win_page, ZERO_PAGE_NUMBER);
if (!(flags & rhd_incomplete)) {
2001-05-23 15:26:42 +02:00
break;
}
2001-05-23 15:26:42 +02:00
}
return;
}
2009-08-20 13:18:57 +02:00
// Object is a blob, and a big one at that
2001-05-23 15:26:42 +02:00
blh* blob = (blh*) header;
2003-12-31 06:36:12 +01:00
const SLONG* page1 = blob->blh_page;
const SLONG* const end1 = page1 + (length - BLH_SIZE) / sizeof(SLONG);
2001-05-23 15:26:42 +02:00
for (; page1 < end1; page1++)
{
if (blob->blh_level == 2)
{
2001-05-23 15:26:42 +02:00
window.win_page = *page1;
2004-02-02 12:02:12 +01:00
blob_page* bpage = (blob_page*) CCH_FETCH(tdbb, &window, LCK_read, pag_blob);
2003-12-31 06:36:12 +01:00
SLONG* page2 = bpage->blp_page;
2008-12-18 11:47:25 +01:00
const SLONG* const end2 = page2 + ((bpage->blp_length - BLP_SIZE) / sizeof(SLONG));
while (page2 < end2) {
PAG_release_page(tdbb, PageNumber(page_space, *page2++), ZERO_PAGE_NUMBER);
}
2001-05-23 15:26:42 +02:00
CCH_RELEASE_TAIL(tdbb, &window);
}
PAG_release_page(tdbb, PageNumber(page_space, *page1), ZERO_PAGE_NUMBER);
2001-05-23 15:26:42 +02:00
}
}
static void fragment(thread_db* tdbb,
record_param* rpb,
2001-05-23 15:26:42 +02:00
SSHORT available_space,
DataComprControl* dcc, SSHORT length, const jrd_tra* transaction)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* f r a g m e n t
*
**************************************
*
* Functional description
* DPM_update tried to replace a record on a page, but it doesn't
* fit. The record, obviously, needs to be fragmented. Stick as
* much as fits on the page and put the rest elsewhere (though not
* in that order.
*
* Doesn't sound so bad does it?
*
* Little do you know. The challenge is that we have to put the
* head of the record in the designated page:line space and put
* the tail on a different page. To maintain on-disk consistency
* we have to write the tail first. But we don't want anybody
* messing with the head until we get back, and, if possible, we'd
* like to keep as much space on the original page as we can get.
* Making matters worse, we may be storing a new version of the
* record or we may be backing out an old one and replacing it
2001-05-23 15:26:42 +02:00
* with one older still (replacing a dead rolled back record with
* the preceding version).
*
* Here is the order of battle. If we're creating a new version,
* and the previous version is not a delta, we compress the page
* and create a new record which claims all the available space
* (but which contains garbage). That record will have our transaction
* id (so it won't be committed for anybody) and it will have legitimate
* back pointers to the previous version.
*
* If we're creating a new version and the previous version is a delta,
* we leave the current version in place, putting our transaction id
* on it. We then point the back pointers to the delta version of the
* same generation of the record, which will have the correct transaction
* id. Applying deltas to the expanded form of the same version of the
* record is a no-op -- but it doesn't cost much and the case is rare.
*
* If we're backing out a rolled back version, we've got another
2001-05-23 15:26:42 +02:00
* problem. The rpb we've got is for record version n - 1, not version
* n + 1. (e.g. I'm transaction 32 removing the rolled back record
* created by transaction 28 and reinstating the committed version
* created by transaction 26. The rpb I've got is for the record
* written by transaction 26 -- the delta flag indicates whether the
* version written by transaction 24 was a delta, the transaction id
* is for a committed transaction, and the back pointers go back to
* transaction 24's version. So in that case, we slap our transaction
* id on transaction 28's version (the one we'll be eliminating) and
2001-05-23 15:26:42 +02:00
* don't change anything else.
*
* In all cases, once the tail is written and safe, we come back to
* the head, store the remaining data, fix up the record header and
* everything is done.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES)
{
2004-04-29 00:43:34 +02:00
printf
("fragment (record_param %"QUADFORMAT
"d, available_space %d, dcc, length %d, transaction %"
2003-04-01 23:59:31 +02:00
SLONGFORMAT")\n",
rpb->rpb_number.getValue(), available_space, length,
2001-05-23 15:26:42 +02:00
transaction ? transaction->tra_number : 0);
2003-12-31 06:36:12 +01:00
}
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf
2003-04-01 19:58:19 +02:00
(" record %"SLONGFORMAT":%d transaction %"SLONGFORMAT" back %"
SLONGFORMAT":%d fragment %"SLONGFORMAT":%d flags %d\n",
rpb->rpb_page, rpb->rpb_line, rpb->rpb_transaction_nr,
2001-05-23 15:26:42 +02:00
rpb->rpb_b_page, rpb->rpb_b_line, rpb->rpb_f_page,
rpb->rpb_f_line, rpb->rpb_flags);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
2009-06-23 06:12:59 +02:00
/* Start by claiming what space exists on the page. Note that
DPM_update has already set the length for this record to zero,
so we're all set for a compress. However, if the current
version was re-stored as a delta, or if we're restoring an
older version, we must preserve it here so
that others may reconstruct it during fragmentation of the
new version. */
2001-05-23 15:26:42 +02:00
2006-05-22 00:07:35 +02:00
WIN* window = &rpb->getWindow(tdbb);
2004-02-02 12:02:12 +01:00
data_page* page = (data_page*) window->win_buffer;
const SSHORT line = rpb->rpb_line;
2001-05-23 15:26:42 +02:00
2003-12-31 06:36:12 +01:00
rhdf* header;
if (transaction->tra_number != rpb->rpb_transaction_nr)
{
header = (rhdf*) ((SCHAR *) page + page->dpg_rpt[line].dpg_offset);
2001-05-23 15:26:42 +02:00
header->rhdf_transaction = transaction->tra_number;
header->rhdf_flags |= rhd_gc_active;
page->dpg_rpt[line].dpg_length = available_space = length;
}
else
{
2009-06-23 06:12:59 +02:00
if (rpb->rpb_flags & rpb_delta)
{
header = (rhdf*) ((SCHAR *) page + page->dpg_rpt[line].dpg_offset);
2001-05-23 15:26:42 +02:00
header->rhdf_flags |= rpb_delta;
page->dpg_rpt[line].dpg_length = available_space = length;
}
2009-06-23 06:12:59 +02:00
else
{
2003-12-31 06:36:12 +01:00
const SSHORT space = DPM_compress(tdbb, page) - available_space;
header = (rhdf*) ((SCHAR *) page + space);
2001-05-23 15:26:42 +02:00
header->rhdf_flags = rhd_deleted;
header->rhdf_f_page = header->rhdf_f_line = 0;
page->dpg_rpt[line].dpg_offset = space;
page->dpg_rpt[line].dpg_length = available_space;
}
header->rhdf_transaction = rpb->rpb_transaction_nr;
2001-05-23 15:26:42 +02:00
header->rhdf_b_page = rpb->rpb_b_page;
header->rhdf_b_line = rpb->rpb_b_line;
}
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf
2003-04-01 19:58:19 +02:00
(" rhdf_transaction %"SLONGFORMAT", window record %"SLONGFORMAT
":%d, available_space %d, rhdf_flags %d, rhdf_f record %"
SLONGFORMAT":%d, rhdf_b record %"SLONGFORMAT":%d\n",
2006-05-22 04:36:53 +02:00
header->rhdf_transaction, window->win_page.getPageNum(), line,
2001-05-23 15:26:42 +02:00
available_space, header->rhdf_flags, header->rhdf_f_page,
header->rhdf_f_line, header->rhdf_b_page, header->rhdf_b_line);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
CCH_RELEASE(tdbb, window);
2009-08-20 13:18:57 +02:00
// The next task is to store the tail where it fits. To do this, we
// next to compute the size (compressed) of the tail. This requires
// first figuring out how much of the original record fits on the
// original page.
2001-05-23 15:26:42 +02:00
2003-12-31 06:36:12 +01:00
const USHORT pre_header_length =
2008-12-18 11:47:25 +01:00
SQZ_compress_length(dcc, (SCHAR*) rpb->rpb_address, (int) (available_space - RHDF_SIZE));
2001-05-23 15:26:42 +02:00
record_param tail_rpb = *rpb;
2001-05-23 15:26:42 +02:00
tail_rpb.rpb_flags = rpb_fragment;
tail_rpb.rpb_b_page = 0;
tail_rpb.rpb_b_line = 0;
tail_rpb.rpb_address = rpb->rpb_address + pre_header_length;
tail_rpb.rpb_length = rpb->rpb_length - pre_header_length;
2006-05-22 00:07:35 +02:00
tail_rpb.getWindow(tdbb).win_flags = 0;
2001-05-23 15:26:42 +02:00
PageStack stack;
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_WRITES_INFO)
2004-04-29 00:43:34 +02:00
printf(" about to store tail\n");
2001-05-23 15:26:42 +02:00
#endif
DPM_store(tdbb, &tail_rpb, stack, DPM_other);
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// That was unreasonablly easy. Now re-fetch the original page and
// fill in the fragment pointer
2001-05-23 15:26:42 +02:00
2004-02-02 12:02:12 +01:00
page = (data_page*) CCH_FETCH(tdbb, window, LCK_write, pag_data);
CCH_precedence(tdbb, window, tail_rpb.rpb_page);
2001-05-23 15:26:42 +02:00
CCH_MARK(tdbb, window);
header = (rhdf*) ((SCHAR *) page + page->dpg_rpt[line].dpg_offset);
2001-05-23 15:26:42 +02:00
header->rhdf_flags = rhd_incomplete | rpb->rpb_flags;
header->rhdf_transaction = rpb->rpb_transaction_nr;
2001-05-23 15:26:42 +02:00
header->rhdf_format = rpb->rpb_format_number;
header->rhdf_f_page = tail_rpb.rpb_page;
header->rhdf_f_line = tail_rpb.rpb_line;
2001-05-23 15:26:42 +02:00
2009-06-23 06:12:59 +02:00
if (transaction->tra_number != rpb->rpb_transaction_nr)
{
2001-05-23 15:26:42 +02:00
header->rhdf_b_page = rpb->rpb_b_page;
header->rhdf_b_line = rpb->rpb_b_line;
}
2003-12-31 06:36:12 +01:00
const USHORT post_header_length =
SQZ_compress(dcc, (SCHAR*) rpb->rpb_address, (SCHAR*) header->rhdf_data,
2001-05-23 15:26:42 +02:00
(int) (available_space - RHDF_SIZE));
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf(" fragment head \n");
printf
2003-04-01 19:58:19 +02:00
(" rhdf_trans %"SLONGFORMAT", window record %"SLONGFORMAT
":%d, dpg_length %d\n\trhdf_flags %d, rhdf_f record %"SLONGFORMAT
":%d, rhdf_b record %"SLONGFORMAT":%d\n",
2006-05-22 04:36:53 +02:00
header->rhdf_transaction, window->win_page.getPageNum(), line,
2001-05-23 15:26:42 +02:00
page->dpg_rpt[line].dpg_length, header->rhdf_flags,
header->rhdf_f_page, header->rhdf_f_line, header->rhdf_b_page,
header->rhdf_b_line);
}
#endif
2009-06-23 06:12:59 +02:00
if (pre_header_length != post_header_length)
{
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, window);
2009-08-20 13:18:57 +02:00
BUGCHECK(252); // msg 252 header fragment length changed
2001-05-23 15:26:42 +02:00
}
if (page->dpg_header.pag_flags & dpg_swept)
{
page->dpg_header.pag_flags &= ~dpg_swept;
mark_full(tdbb, rpb);
}
else
2009-12-28 12:46:55 +01:00
CCH_RELEASE(tdbb, window);
2001-05-23 15:26:42 +02:00
}
static void extend_relation(thread_db* tdbb, jrd_rel* relation, WIN* window, USHORT type)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* e x t e n d _ r e l a t i o n
*
**************************************
*
* Functional description
* Extend a relation with a given page. The window points to an
* already allocated, fetched, and marked data page to be inserted
* into the pointer pages for a given relation.
2001-05-23 15:26:42 +02:00
* This routine returns a window on the datapage locked for write
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
2006-05-22 00:07:35 +02:00
RelationPages* relPages = relation->getPages(tdbb);
WIN pp_window(relPages->rel_pg_space_id, -1),
2006-05-22 00:07:35 +02:00
new_pp_window(relPages->rel_pg_space_id, -1);
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
2003-12-31 06:36:12 +01:00
if (debug_flag > DEBUG_WRITES_INFO) {
2009-06-23 06:12:59 +02:00
printf(" extend_relation (relation %d, window)\n", relation->rel_id);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
2009-08-20 13:18:57 +02:00
// Release faked page before fetching pointer page to prevent deadlocks. This is only
// a problem for multi-threaded servers using internal latches. The faked page may be
// dirty from its previous incarnation and involved in a precedence relationship. This
// special case may need a more general solution.
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, window);
2009-06-23 06:12:59 +02:00
/* Search pointer pages for an empty slot.
If we run out of pointer pages, allocate another one. Note that the code below
is not careful in preventing deadlocks when it allocates a new pointer page:
- the last already-existing pointer page is fetched with an exclusive latch,
- allocation of a new pointer page requires the fetching of the PIP page with
an exlusive latch.
This might cause a deadlock. Fortunately, pointer pages don't need to be
allocated often. */
2009-06-27 05:03:56 +02:00
pointer_page* ppage = NULL;
2003-12-31 06:36:12 +01:00
USHORT pp_sequence, slot = 0;
2009-06-27 05:03:56 +02:00
data_page* dpage = NULL;
2001-05-23 15:26:42 +02:00
2009-06-25 12:59:10 +02:00
for (;;)
{
2009-06-25 12:59:10 +02:00
for (pp_sequence = relPages->rel_slot_space;; pp_sequence++)
2009-06-23 06:12:59 +02:00
{
2009-06-25 12:59:10 +02:00
if (!(ppage = get_pointer_page(tdbb, relation, relPages, &pp_window,
2009-06-27 05:03:56 +02:00
pp_sequence, LCK_write)))
2009-06-25 12:59:10 +02:00
{
2009-08-20 13:18:57 +02:00
BUGCHECK(253); // msg 253 pointer page vanished from extend_relation
2009-06-25 12:59:10 +02:00
}
SLONG* slots = ppage->ppg_page;
for (slot = 0; slot < ppage->ppg_count; slot++, slots++)
{
if (*slots == 0) {
break;
}
}
if (slot < ppage->ppg_count) {
2001-05-23 15:26:42 +02:00
break;
}
2009-06-25 12:59:10 +02:00
if ((pp_sequence && ppage->ppg_count < dbb->dbb_dp_per_pp) ||
(ppage->ppg_count < dbb->dbb_dp_per_pp - 1))
2008-12-18 11:47:25 +01:00
{
2009-06-25 12:59:10 +02:00
slot = ppage->ppg_count;
break;
2008-12-18 11:47:25 +01:00
}
2009-06-25 12:59:10 +02:00
if (ppage->ppg_header.pag_flags & ppg_eof)
{
ppage = (pointer_page*) DPM_allocate(tdbb, &new_pp_window);
ppage->ppg_header.pag_type = pag_pointer;
ppage->ppg_header.pag_flags |= ppg_eof;
ppage->ppg_relation = relation->rel_id;
ppage->ppg_sequence = ++pp_sequence;
slot = 0;
CCH_must_write(&new_pp_window);
CCH_RELEASE(tdbb, &new_pp_window);
vcl* vector = relPages->rel_pages =
vcl::newVector(*dbb->dbb_permanent, relPages->rel_pages, pp_sequence + 1);
(*vector)[pp_sequence] = new_pp_window.win_page.getPageNum();
// hvlad: temporary tables don't save their pointer pages in RDB$PAGES
if (relation->rel_id && (relPages->rel_instance_id == 0))
{
DPM_pages(tdbb, relation->rel_id, pag_pointer,
(SLONG) pp_sequence, new_pp_window.win_page.getPageNum());
}
relPages->rel_slot_space = pp_sequence;
2001-05-23 15:26:42 +02:00
2009-06-25 12:59:10 +02:00
ppage = (pointer_page*) pp_window.win_buffer;
CCH_MARK(tdbb, &pp_window);
ppage->ppg_header.pag_flags &= ~ppg_eof;
ppage->ppg_next = new_pp_window.win_page.getPageNum();
--pp_sequence;
}
CCH_RELEASE(tdbb, &pp_window);
2001-05-23 15:26:42 +02:00
}
2009-08-20 13:18:57 +02:00
// We've found a slot. Stick in the pointer to the data page
2001-05-23 15:26:42 +02:00
2009-06-25 12:59:10 +02:00
if (ppage->ppg_page[slot])
{
CCH_RELEASE(tdbb, &pp_window);
2009-08-20 13:18:57 +02:00
CORRUPT(258); // msg 258 page slot not empty
2009-06-25 12:59:10 +02:00
}
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// Refetch newly allocated page that was released above.
// To prevent possible deadlocks (since we own already an exlusive latch and we
// are asking for another exclusive latch), time out on the latch after 1 second.
2001-05-23 15:26:42 +02:00
2009-06-25 12:59:10 +02:00
dpage = (data_page*) CCH_FETCH_TIMEOUT(tdbb, window, LCK_write, pag_undefined, -1);
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// In the case of a timeout, retry the whole thing.
2001-05-23 15:26:42 +02:00
2009-06-25 12:59:10 +02:00
if (!dpage) {
CCH_RELEASE(tdbb, &pp_window);
}
else
break;
2001-05-23 15:26:42 +02:00
}
CCH_MARK(tdbb, window);
2009-06-25 12:59:10 +02:00
dpage->dpg_sequence = (SLONG) pp_sequence * dbb->dbb_dp_per_pp + slot;
2001-05-23 15:26:42 +02:00
dpage->dpg_relation = relation->rel_id;
dpage->dpg_header.pag_type = pag_data;
2006-05-22 00:07:35 +02:00
relPages->rel_data_space = pp_sequence;
if (type != DPM_primary) {
2009-12-24 12:48:17 +01:00
dpage->dpg_header.pag_flags |= dpg_secondary; // Who checks it?
}
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, window);
CCH_precedence(tdbb, &pp_window, window->win_page);
CCH_MARK_SYSTEM(tdbb, &pp_window);
2006-05-22 00:07:35 +02:00
ppage->ppg_page[slot] = window->win_page.getPageNum();
2001-05-23 15:26:42 +02:00
ppage->ppg_min_space = MIN(ppage->ppg_min_space, slot);
ppage->ppg_count = MAX(ppage->ppg_count, slot + 1);
2003-12-31 06:36:12 +01:00
UCHAR* bits = (UCHAR *) (ppage->ppg_page + dbb->dbb_dp_per_pp);
// bits[slot >> 2] &= ~(1 << ((slot & 3) << 1));
PPG_DP_BIT_CLEAR(bits, slot, PPG_DP_ALL_BITS);
if (type != DPM_primary) {
PPG_DP_BIT_SET(bits, slot, ppg_dp_secondary);
}
2006-05-22 00:07:35 +02:00
if (relPages->rel_data_pages) {
++relPages->rel_data_pages;
2001-05-23 15:26:42 +02:00
}
*window = pp_window;
CCH_HANDOFF(tdbb, window, ppage->ppg_page[slot], LCK_write, pag_data);
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf(" extended_relation (relation %d, window_page %"SLONGFORMAT")\n",
2006-05-22 04:36:53 +02:00
relation->rel_id, window->win_page.getPageNum());
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
}
static UCHAR* find_space(thread_db* tdbb,
record_param* rpb,
2001-05-23 15:26:42 +02:00
SSHORT size,
PageStack& stack,
Record* record,
2001-05-23 15:26:42 +02:00
USHORT type)
{
/**************************************
*
* f i n d _ s p a c e
*
**************************************
*
* Functional description
* Find space of a given size on a data page. If no space, return
* null. If space is found, mark the page, set up the line field
* in the record parameter block, set up the offset/length on the
* data page, and return a pointer to the space.
*
* To maintain page precedence when objects point to objects, a stack
* of pages of high precedence may be passed in.
2001-05-23 15:26:42 +02:00
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
2003-12-31 06:36:12 +01:00
const SSHORT aligned_size = ROUNDUP(size, ODS_ALIGNMENT);
2006-05-22 00:07:35 +02:00
data_page* page = (data_page*) rpb->getWindow(tdbb).win_buffer;
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// Scan allocated lines looking for an empty slot, the high water mark,
// and the amount of space potentially available on the page
2001-05-23 15:26:42 +02:00
2003-12-31 06:36:12 +01:00
SSHORT space = dbb->dbb_page_size;
SSHORT slot = 0;
SSHORT used = HIGH_WATER(page->dpg_count);
2001-05-23 15:26:42 +02:00
2003-12-31 06:36:12 +01:00
{ // scope
2006-07-21 03:35:17 +02:00
const bool reserving = !(dbb->dbb_flags & DBB_no_reserve);
const data_page::dpg_repeat* index = page->dpg_rpt;
for (SSHORT i = 0; i < page->dpg_count; i++, index++)
{
if (index->dpg_offset)
{
2006-07-21 03:35:17 +02:00
space = MIN(space, index->dpg_offset);
used += ROUNDUP(index->dpg_length, ODS_ALIGNMENT);
if (type == DPM_primary && reserving)
{
2006-07-21 03:35:17 +02:00
const rhd* header = (rhd*) ((SCHAR *) page + index->dpg_offset);
if (!header->rhd_b_page &&
2008-12-18 11:47:25 +01:00
!(header->rhd_flags & (rhd_chain | rhd_blob | rhd_deleted | rhd_fragment)))
2006-07-21 03:35:17 +02:00
{
used += SPACE_FUDGE;
}
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
}
2006-07-21 03:35:17 +02:00
else if (!slot) {
slot = i;
}
2001-05-23 15:26:42 +02:00
}
2003-12-31 06:36:12 +01:00
} // scope
2001-05-23 15:26:42 +02:00
if (!slot) {
2004-02-02 12:02:12 +01:00
used += sizeof(data_page::dpg_repeat);
}
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// If there isn't space, give up
2001-05-23 15:26:42 +02:00
2009-06-23 06:12:59 +02:00
if (aligned_size > (int) dbb->dbb_page_size - used)
{
if (!(page->dpg_header.pag_flags & dpg_full))
{
2009-12-28 12:46:55 +01:00
CCH_MARK(tdbb, &rpb->getWindow(tdbb));
page->dpg_header.pag_flags |= dpg_full;
mark_full(tdbb, rpb);
}
else
{
CCH_RELEASE(tdbb, &rpb->getWindow(tdbb));
}
2001-05-23 15:26:42 +02:00
return NULL;
}
2009-08-20 13:18:57 +02:00
// There's space on page. If the line index needs expansion, do so.
// If the page need to be compressed, compress it.
2001-05-23 15:26:42 +02:00
while (stack.hasData()) {
2006-05-22 00:07:35 +02:00
CCH_precedence(tdbb, &rpb->getWindow(tdbb), stack.pop());
}
2006-05-22 00:07:35 +02:00
CCH_MARK(tdbb, &rpb->getWindow(tdbb));
2001-05-23 15:26:42 +02:00
2003-12-31 06:36:12 +01:00
{ // scope
2009-02-28 12:57:40 +01:00
const USHORT rec_segments = page->dpg_count + (slot ? 0 : 1);
2003-12-31 06:36:12 +01:00
fb_assert(rec_segments > 0); // zero is a disaster in macro HIGH_WATER
if (aligned_size > space - HIGH_WATER(rec_segments))
space = DPM_compress(tdbb, page);
} // scope
2001-05-23 15:26:42 +02:00
if (!slot) {
2001-05-23 15:26:42 +02:00
slot = page->dpg_count++;
}
2001-05-23 15:26:42 +02:00
space -= aligned_size;
2004-02-02 12:02:12 +01:00
data_page::dpg_repeat* index = &page->dpg_rpt[slot];
2001-05-23 15:26:42 +02:00
index->dpg_length = size;
index->dpg_offset = space;
rpb->rpb_page = rpb->getWindow(tdbb).win_page.getPageNum();
rpb->rpb_line = slot;
rpb->rpb_number.setValue(((SINT64) page->dpg_sequence) * dbb->dbb_max_records + slot);
2001-05-23 15:26:42 +02:00
if (record) {
record->rec_precedence.push(rpb->rpb_page);
}
2001-05-23 15:26:42 +02:00
2008-12-20 20:57:43 +01:00
return (UCHAR*) page + space;
2001-05-23 15:26:42 +02:00
}
2009-11-25 09:38:52 +01:00
static bool get_header(WIN* window, SSHORT line, record_param* rpb)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* g e t _ h e a d e r
*
**************************************
*
* Functional description
* Copy record header fields into a record parameter block. If
* the line is empty, return false;
*
**************************************/
2004-02-02 12:02:12 +01:00
const data_page* page = (data_page*) window->win_buffer;
if (line >= page->dpg_count) {
2004-02-02 12:02:12 +01:00
return false;
}
2001-05-23 15:26:42 +02:00
2004-02-02 12:02:12 +01:00
const data_page::dpg_repeat* index = &page->dpg_rpt[line];
if (index->dpg_offset == 0) {
2004-02-02 12:02:12 +01:00
return false;
}
2001-05-23 15:26:42 +02:00
rhdf* header = (rhdf*) ((SCHAR *) page + index->dpg_offset);
rpb->rpb_page = window->win_page.getPageNum();
rpb->rpb_line = line;
2001-05-23 15:26:42 +02:00
rpb->rpb_flags = header->rhdf_flags;
2009-06-23 06:12:59 +02:00
if (!(rpb->rpb_flags & rpb_fragment))
{
2001-05-23 15:26:42 +02:00
rpb->rpb_b_page = header->rhdf_b_page;
rpb->rpb_b_line = header->rhdf_b_line;
rpb->rpb_transaction_nr = header->rhdf_transaction;
2001-05-23 15:26:42 +02:00
rpb->rpb_format_number = header->rhdf_format;
}
2009-06-23 06:12:59 +02:00
if (rpb->rpb_flags & rpb_incomplete)
{
2001-05-23 15:26:42 +02:00
rpb->rpb_f_page = header->rhdf_f_page;
rpb->rpb_f_line = header->rhdf_f_line;
rpb->rpb_address = header->rhdf_data;
rpb->rpb_length = index->dpg_length - RHDF_SIZE;
}
2009-06-23 06:12:59 +02:00
else
{
rpb->rpb_address = ((rhd*) header)->rhd_data;
2001-05-23 15:26:42 +02:00
rpb->rpb_length = index->dpg_length - RHD_SIZE;
}
2004-02-02 12:02:12 +01:00
return true;
2001-05-23 15:26:42 +02:00
}
static pointer_page* get_pointer_page(thread_db* tdbb,
jrd_rel* relation, RelationPages* relPages,
2009-08-20 13:18:57 +02:00
WIN* window, USHORT sequence, USHORT lock)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* g e t _ p o i n t e r _ p a g e
*
**************************************
*
* Functional description
* Fetch a specific pointer page. If we don't know about it,
* do a re-scan of RDB$PAGES to find it. If that doesn't work,
* try the sibling pointer. If that doesn't work, just stop,
* return NULL, and let our caller think about it.
*
**************************************/
SET_TDBB(tdbb);
2006-05-22 00:07:35 +02:00
vcl* vector = relPages->rel_pages;
if (!vector || sequence >= vector->count())
{
for (;;)
{
2001-05-23 15:26:42 +02:00
DPM_scan_pages(tdbb);
2009-08-20 13:18:57 +02:00
// If the relation is gone, then we can't do anything anymore.
2009-07-13 12:00:43 +02:00
if (!relation || !(vector = relPages->rel_pages)) {
2001-05-23 15:26:42 +02:00
return NULL;
}
if (sequence < vector->count()) {
2009-08-20 13:18:57 +02:00
break; // we are in business again
}
2001-12-24 03:51:06 +01:00
window->win_page = (*vector)[vector->count() - 1];
2004-02-02 12:02:12 +01:00
const pointer_page* page = (pointer_page*) CCH_FETCH(tdbb, window, lock, pag_pointer);
2003-12-31 06:36:12 +01:00
const SLONG next_ppg = page->ppg_next;
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, window);
if (!next_ppg)
return NULL;
2006-05-22 00:07:35 +02:00
2006-05-25 10:40:23 +02:00
// hvlad: temporary tables don't save their pointer pages in RDB$PAGES
2006-05-22 00:07:35 +02:00
if (relPages->rel_instance_id == 0)
2008-12-18 11:47:25 +01:00
DPM_pages(tdbb, relation->rel_id, pag_pointer, vector->count(), next_ppg);
2001-05-23 15:26:42 +02:00
}
}
2001-12-24 03:51:06 +01:00
window->win_page = (*vector)[sequence];
2004-02-02 12:02:12 +01:00
pointer_page* page = (pointer_page*) CCH_FETCH(tdbb, window, lock, pag_pointer);
2001-05-23 15:26:42 +02:00
2008-12-18 11:47:25 +01:00
if (page->ppg_relation != relation->rel_id || page->ppg_sequence != sequence)
{
2009-08-20 13:18:57 +02:00
CORRUPT(259); // msg 259 bad pointer page
}
2001-05-23 15:26:42 +02:00
return page;
}
static rhd* locate_space(thread_db* tdbb,
record_param* rpb,
SSHORT size, PageStack& stack, Record* record, USHORT type)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* l o c a t e _ s p a c e
*
**************************************
*
* Functional description
* Find space in a relation for a record. Find a likely data page
* and call find_space to see if there really is space there. If
* we can't find any space, extend the relation.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
2003-12-31 06:36:12 +01:00
jrd_rel* relation = rpb->rpb_relation;
2006-05-22 00:07:35 +02:00
RelationPages* relPages = relation->getPages(tdbb, rpb->rpb_transaction_nr);
WIN* window = &rpb->getWindow(tdbb);
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// If there is a preferred page, try there first
2001-05-23 15:26:42 +02:00
PagesArray lowPages;
SLONG dp_primary = 0;
2009-10-15 04:15:31 +02:00
if (type == DPM_secondary)
{
USHORT pp_sequence;
SSHORT slot, line;
rpb->rpb_number.decompose(dbb->dbb_max_records, dbb->dbb_dp_per_pp, line, slot, pp_sequence);
2008-12-18 11:47:25 +01:00
const pointer_page* ppage =
get_pointer_page(tdbb, relation, relPages, window, pp_sequence, LCK_read);
2003-12-31 06:36:12 +01:00
if (ppage)
2003-09-13 14:03:11 +02:00
{
if (slot < ppage->ppg_count && ((dp_primary = ppage->ppg_page[slot])) )
2003-09-13 14:03:11 +02:00
{
CCH_HANDOFF(tdbb, window, dp_primary, LCK_write, pag_data);
2003-12-31 06:36:12 +01:00
UCHAR* space = find_space(tdbb, rpb, size, stack, record, type);
if (space)
return (rhd*) space;
if (!window->win_page.isTemporary()) {
CCH_get_related(tdbb, window->win_page, lowPages);
}
2001-05-23 15:26:42 +02:00
}
else {
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, window);
}
2003-09-13 14:03:11 +02:00
}
2001-05-23 15:26:42 +02:00
}
2009-08-20 13:18:57 +02:00
// Look for space anywhere
2001-05-23 15:26:42 +02:00
for (USHORT pp_sequence = relPages->rel_data_space;; pp_sequence++)
{
2006-05-22 00:07:35 +02:00
relPages->rel_data_space = pp_sequence;
2004-02-02 12:02:12 +01:00
const pointer_page* ppage =
2006-05-22 00:07:35 +02:00
get_pointer_page(tdbb, relation, relPages, window, pp_sequence, LCK_read);
2009-06-23 06:12:59 +02:00
if (!ppage)
{
BUGCHECK(254);
2009-08-20 13:18:57 +02:00
// msg 254 pointer page vanished from relation list in locate_space
}
2006-05-22 00:07:35 +02:00
const SLONG pp_number = window->win_page.getPageNum();
2003-12-31 06:36:12 +01:00
const UCHAR* bits = (UCHAR *) (ppage->ppg_page + dbb->dbb_dp_per_pp);
for (USHORT slot = ppage->ppg_min_space; slot < ppage->ppg_count; slot++)
{
2003-12-31 06:36:12 +01:00
const SLONG dp_number = ppage->ppg_page[slot];
// hvlad: avoid creating circle in precedence graph, if possible
if (type == DPM_secondary && lowPages.exist(dp_number))
continue;
const bool dp_is_secondary = PPG_DP_BIT_TEST(bits, slot, ppg_dp_secondary);
//if (dp_number && ~bits[slot >> 2] & (1 << ((slot & 3) << 1)))
if (dp_number && !PPG_DP_BIT_TEST(bits, slot, ppg_dp_full) &&
2009-12-18 00:59:07 +01:00
((type == DPM_primary && !dp_is_secondary) || (type != DPM_primary && dp_is_secondary)))
{
2001-05-23 15:26:42 +02:00
CCH_HANDOFF(tdbb, window, dp_number, LCK_write, pag_data);
2003-12-31 06:36:12 +01:00
UCHAR* space = find_space(tdbb, rpb, size, stack, record, type);
if (space)
return (rhd*) space;
2001-05-23 15:26:42 +02:00
window->win_page = pp_number;
2004-02-02 12:02:12 +01:00
ppage = (pointer_page*) CCH_FETCH(tdbb, window, LCK_read, pag_pointer);
2001-05-23 15:26:42 +02:00
}
2003-12-31 06:36:12 +01:00
}
const UCHAR flags = ppage->ppg_header.pag_flags;
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, window);
if (flags & ppg_eof) {
2001-05-23 15:26:42 +02:00
break;
}
2001-05-23 15:26:42 +02:00
}
2009-08-20 13:18:57 +02:00
// Sigh. No space. Extend relation. Try for a while in case someone grabs the page
// before we can get it locked, then give up on the assumption that things
// are really screwed up.
2003-12-31 06:36:12 +01:00
UCHAR* space = 0;
2008-01-16 09:46:02 +01:00
int i;
for (i = 0; i < 20; ++i)
{
2001-05-23 15:26:42 +02:00
DPM_allocate(tdbb, window);
extend_relation(tdbb, relation, window, type);
2003-12-31 06:36:12 +01:00
space = find_space(tdbb, rpb, size, stack, record, type);
if (space) {
2001-05-23 15:26:42 +02:00
break;
}
2001-05-23 15:26:42 +02:00
}
if (i == 20) {
2009-08-20 13:18:57 +02:00
BUGCHECK(255); // msg 255 cannot find free space
}
2001-05-23 15:26:42 +02:00
if (record) {
2006-05-22 00:07:35 +02:00
record->rec_precedence.push(window->win_page.getPageNum());
}
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf(" extended relation %d with page %"SLONGFORMAT" to get %d bytes\n",
2006-05-22 04:36:53 +02:00
relation->rel_id, window->win_page.getPageNum(), size);
2003-12-31 06:36:12 +01:00
}
2001-05-23 15:26:42 +02:00
#endif
return (rhd*) space;
2001-05-23 15:26:42 +02:00
}
2009-11-25 09:38:52 +01:00
static void mark_full(thread_db* tdbb, record_param* rpb)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* m a r k _ f u l l
*
**************************************
*
* Functional description
2009-12-28 12:46:55 +01:00
* Mark a fetched page and its pointer page to indicate the page
2001-05-23 15:26:42 +02:00
* is full.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_TRACE_ALL)
2004-04-29 00:43:34 +02:00
printf("mark_full ()\n");
2001-05-23 15:26:42 +02:00
#endif
2009-08-20 13:18:57 +02:00
// We need to access the pointer page for write. To avoid deadlocks,
// we need to release the data page, fetch the pointer page for write,
// and re-fetch the data page. If the data page is still empty, set
// it's "full" bit on the pointer page.
2001-05-23 15:26:42 +02:00
2006-05-22 00:07:35 +02:00
data_page* dpage = (data_page*) rpb->getWindow(tdbb).win_buffer;
2003-12-31 06:36:12 +01:00
const SLONG sequence = dpage->dpg_sequence;
2006-05-22 00:07:35 +02:00
CCH_RELEASE(tdbb, &rpb->getWindow(tdbb));
2001-05-23 15:26:42 +02:00
2003-12-31 06:36:12 +01:00
jrd_rel* relation = rpb->rpb_relation;
2006-05-22 00:07:35 +02:00
RelationPages* relPages = relation->getPages(tdbb);
WIN pp_window(relPages->rel_pg_space_id, -1);
2003-12-31 06:36:12 +01:00
USHORT slot, pp_sequence;
2001-05-23 15:26:42 +02:00
DECOMPOSE(sequence, dbb->dbb_dp_per_pp, pp_sequence, slot);
2009-08-20 13:18:57 +02:00
// Fetch the pointer page, then the data page. Since this is a case of
// fetching a second page after having fetched the first page with an
// exclusive latch, care has to be taken to prevent a deadlock. This
// is accomplished by timing out the second latch request and retrying
// the whole thing.
2001-05-23 15:26:42 +02:00
2004-02-02 12:02:12 +01:00
pointer_page* ppage = 0;
2001-05-23 15:26:42 +02:00
do {
2008-12-18 11:47:25 +01:00
ppage = get_pointer_page(tdbb, relation, relPages, &pp_window, pp_sequence, LCK_write);
2003-12-31 06:36:12 +01:00
if (!ppage)
2009-08-20 13:18:57 +02:00
BUGCHECK(256); // msg 256 pointer page vanished from mark_full
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// If data page has been deleted from relation then there's nothing left to do.
2001-05-23 15:26:42 +02:00
if (slot >= ppage->ppg_count ||
2006-05-22 00:07:35 +02:00
rpb->getWindow(tdbb).win_page.getPageNum() != ppage->ppg_page[slot])
2003-12-31 06:36:12 +01:00
{
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, &pp_window);
return;
}
2009-08-20 13:18:57 +02:00
// Fetch the data page, but timeout after 1 second to break a possible deadlock.
2008-12-18 11:47:25 +01:00
dpage = (data_page*) CCH_FETCH_TIMEOUT(tdbb, &rpb->getWindow(tdbb), LCK_read, pag_data, -1);
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// In case of a latch timeout, release the latch on the pointer page and retry.
if (!dpage) {
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, &pp_window);
}
} while (!dpage);
2001-05-23 15:26:42 +02:00
const UCHAR flags = dpage->dpg_header.pag_flags;
2006-05-22 00:07:35 +02:00
CCH_RELEASE(tdbb, &rpb->getWindow(tdbb));
2001-05-23 15:26:42 +02:00
// Check if PP flags already equal to the DP flags
UCHAR* byte = &PPG_DP_BITS_BYTE((UCHAR*) &ppage->ppg_page[dbb->dbb_dp_per_pp], slot);
const UCHAR bit_full_set = ((*byte & PPG_DP_BIT_MASK(slot, ppg_dp_full)) == 0) ? 0 : dpg_full;
const UCHAR bit_large_set = ((*byte & PPG_DP_BIT_MASK(slot, ppg_dp_large)) == 0) ? 0 : dpg_large;
const UCHAR bit_swept_set = ((*byte & PPG_DP_BIT_MASK(slot, ppg_dp_swept)) == 0) ? 0 : dpg_swept;
if ((flags & (dpg_full | dpg_large | dpg_swept)) == (bit_full_set | bit_large_set | bit_swept_set))
{
CCH_RELEASE(tdbb, &pp_window);
return;
}
2006-05-22 00:07:35 +02:00
CCH_precedence(tdbb, &pp_window, rpb->getWindow(tdbb).win_page);
2001-05-23 15:26:42 +02:00
CCH_MARK(tdbb, &pp_window);
2009-12-15 00:55:56 +01:00
//UCHAR bit = 1 << ((slot & 3) << 1);
//UCHAR* byte = (UCHAR *) (&ppage->ppg_page[dbb->dbb_dp_per_pp]) + (slot >> 2);
2001-05-23 15:26:42 +02:00
UCHAR bit = PPG_DP_BIT_MASK(slot, ppg_dp_full);
2009-06-23 06:12:59 +02:00
if (flags & dpg_full)
{
2001-05-23 15:26:42 +02:00
*byte |= bit;
ppage->ppg_min_space = MAX(slot + 1, ppage->ppg_min_space);
}
2009-06-23 06:12:59 +02:00
else
{
2001-05-23 15:26:42 +02:00
*byte &= ~bit;
ppage->ppg_min_space = MIN(slot, ppage->ppg_min_space);
2006-05-22 00:07:35 +02:00
relPages->rel_data_space = MIN(pp_sequence, relPages->rel_data_space);
2001-05-23 15:26:42 +02:00
}
2009-08-20 13:18:57 +02:00
// Next, handle the "large object" bit
2001-05-23 15:26:42 +02:00
2009-12-15 00:55:56 +01:00
//bit <<= 1;
bit = PPG_DP_BIT_MASK(slot, ppg_dp_large);
if (flags & dpg_large) {
2001-05-23 15:26:42 +02:00
*byte |= bit;
}
else {
2001-05-23 15:26:42 +02:00
*byte &= ~bit;
}
2001-05-23 15:26:42 +02:00
bit = PPG_DP_BIT_MASK(slot, ppg_dp_swept);
if (flags & dpg_swept) {
*byte |= bit;
}
else {
*byte &= ~bit;
}
2001-05-23 15:26:42 +02:00
CCH_RELEASE(tdbb, &pp_window);
}
static void store_big_record(thread_db* tdbb, record_param* rpb,
PageStack& stack,
DataComprControl* dcc, USHORT size, USHORT type)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* s t o r e _ b i g _ r e c o r d
*
**************************************
*
* Functional description
* Store a new record in a relation.
*
**************************************/
SET_TDBB(tdbb);
Database* dbb = tdbb->getDatabase();
2001-05-23 15:26:42 +02:00
CHECK_DBB(dbb);
#ifdef VIO_DEBUG
if (debug_flag > DEBUG_TRACE_ALL)
2004-04-29 00:43:34 +02:00
printf("store_big_record ()\n");
2001-05-23 15:26:42 +02:00
#endif
2009-08-20 13:18:57 +02:00
// Start compression from the end.
2001-05-23 15:26:42 +02:00
const SCHAR* control = dcc->end();
2003-12-31 06:36:12 +01:00
const SCHAR* in = (SCHAR *) rpb->rpb_address + rpb->rpb_length;
2006-05-22 00:07:35 +02:00
RelationPages* relPages = rpb->rpb_relation->getPages(tdbb);
PageNumber prior(relPages->rel_pg_space_id, 0);
2003-12-31 06:36:12 +01:00
SCHAR count = 0;
2004-02-02 12:02:12 +01:00
const USHORT max_data = dbb->dbb_page_size - (sizeof(data_page) + RHDF_SIZE);
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// Fill up data pages tail first until what's left fits on a single page.
2001-05-23 15:26:42 +02:00
while (size > max_data)
{
2009-08-20 13:18:57 +02:00
// Allocate and format data page and fragment header
2001-05-23 15:26:42 +02:00
2006-05-22 00:07:35 +02:00
data_page* page = (data_page*) DPM_allocate(tdbb, &rpb->getWindow(tdbb));
page->dpg_header.pag_type = pag_data;
page->dpg_header.pag_flags = dpg_orphan | dpg_full;
2001-05-23 15:26:42 +02:00
page->dpg_relation = rpb->rpb_relation->rel_id;
page->dpg_count = 1;
rhdf* header = (rhdf*) & page->dpg_rpt[1];
2001-05-23 15:26:42 +02:00
page->dpg_rpt[0].dpg_offset = (UCHAR *) header - (UCHAR *) page;
page->dpg_rpt[0].dpg_length = max_data + RHDF_SIZE;
2008-12-18 11:47:25 +01:00
header->rhdf_flags = (prior.getPageNum()) ? rhd_fragment | rhd_incomplete : rhd_fragment;
2006-05-22 00:07:35 +02:00
header->rhdf_f_page = prior.getPageNum();
2003-12-31 06:36:12 +01:00
USHORT length = max_data;
2001-05-23 15:26:42 +02:00
size -= length;
2003-12-31 06:36:12 +01:00
SCHAR* out = (SCHAR *) header->rhdf_data + length;
2001-05-23 15:26:42 +02:00
2009-08-20 13:18:57 +02:00
// Move compressed data onto page
2001-05-23 15:26:42 +02:00
while (length > 1)
{
2009-08-20 13:18:57 +02:00
// Handle residual count, if any
if (count > 0)
{
2003-12-31 06:36:12 +01:00
const USHORT l = MIN((USHORT) count, length - 1);
USHORT n = l;
do {
2001-05-23 15:26:42 +02:00
*--out = *--in;
2003-12-31 06:36:12 +01:00
} while (--n);
2001-05-23 15:26:42 +02:00
*--out = l;
2009-08-20 13:18:57 +02:00
length -= (SSHORT) (l + 1); // bytes remaining on page
count -= (SSHORT) l; // bytes remaining in run
2001-05-23 15:26:42 +02:00
continue;
}
2009-06-23 06:12:59 +02:00
if ((count = *--control) < 0)
{
2001-05-23 15:26:42 +02:00
*--out = in[-1];
*--out = count;
in += count;
length -= 2;
}
}
2009-08-20 13:18:57 +02:00
// Page is full. If there is an odd byte left, fudge it.
2001-05-23 15:26:42 +02:00
2009-06-23 06:12:59 +02:00
if (length)
{
2001-05-23 15:26:42 +02:00
*--out = 0;
++size;
}
else if (count > 0) {
2001-05-23 15:26:42 +02:00
++size;
}
2006-05-22 00:07:35 +02:00
if (prior.getPageNum()) {
CCH_precedence(tdbb, &rpb->getWindow(tdbb), prior);
}
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf(" back portion\n");
printf
2006-05-22 00:07:35 +02:00
(" getWindow(tdbb) page %"SLONGFORMAT
2003-04-01 19:58:19 +02:00
", max_data %d, \n\trhdf_flags %d, prior %"SLONGFORMAT"\n",
2006-05-22 04:36:53 +02:00
rpb->getWindow(tdbb).win_page.getPageNum(), max_data, header->rhdf_flags,
2006-05-22 19:41:10 +02:00
prior.getPageNum());
2001-05-23 15:26:42 +02:00
}
#endif
2006-05-22 00:07:35 +02:00
CCH_RELEASE(tdbb, &rpb->getWindow(tdbb));
prior = rpb->getWindow(tdbb).win_page;
2001-05-23 15:26:42 +02:00
}
2009-08-20 13:18:57 +02:00
// What's left fits on a page. Luckily, we don't have to store it ourselves.
2001-05-23 15:26:42 +02:00
2009-06-19 04:29:08 +02:00
size = SQZ_length((SCHAR*) rpb->rpb_address, in - (SCHAR*) rpb->rpb_address, dcc);
2009-12-18 00:59:07 +01:00
rhdf* header = (rhdf*) locate_space(tdbb, rpb, (SSHORT) (RHDF_SIZE + size), stack, NULL, type);
2001-05-23 15:26:42 +02:00
header->rhdf_flags = rhd_incomplete | rhd_large | rpb->rpb_flags;
header->rhdf_transaction = rpb->rpb_transaction_nr;
2001-05-23 15:26:42 +02:00
header->rhdf_format = rpb->rpb_format_number;
header->rhdf_b_page = rpb->rpb_b_page;
header->rhdf_b_line = rpb->rpb_b_line;
2006-05-22 00:07:35 +02:00
header->rhdf_f_page = prior.getPageNum();
2001-05-23 15:26:42 +02:00
header->rhdf_f_line = 0;
SQZ_fast(dcc, (SCHAR*) rpb->rpb_address, (SCHAR*) header->rhdf_data);
2001-05-23 15:26:42 +02:00
2006-05-22 00:07:35 +02:00
data_page* page = (data_page*) rpb->getWindow(tdbb).win_buffer;
2001-05-23 15:26:42 +02:00
#ifdef VIO_DEBUG
2009-06-23 06:12:59 +02:00
if (debug_flag > DEBUG_WRITES_INFO)
{
2004-04-29 00:43:34 +02:00
printf(" front part\n");
printf
2003-04-01 19:58:19 +02:00
(" rhdf_trans %"SLONGFORMAT", rpb_window record %"SLONGFORMAT
":%d, dpg_length %d \n\trhdf_flags %d, rhdf_f record %"SLONGFORMAT
":%d, rhdf_b record %"SLONGFORMAT":%d\n",
2006-05-22 04:36:53 +02:00
header->rhdf_transaction, rpb->getWindow(tdbb).win_page.getPageNum(),
rpb->rpb_line, page->dpg_rpt[rpb->rpb_line].dpg_length,
2001-05-23 15:26:42 +02:00
header->rhdf_flags, header->rhdf_f_page, header->rhdf_f_line,
header->rhdf_b_page, header->rhdf_b_line);
}
#endif
2009-06-23 06:12:59 +02:00
if (!(page->dpg_header.pag_flags & dpg_large))
{
page->dpg_header.pag_flags |= dpg_large;
2001-05-23 15:26:42 +02:00
mark_full(tdbb, rpb);
}
else {
2006-05-22 00:07:35 +02:00
CCH_RELEASE(tdbb, &rpb->getWindow(tdbb));
}
}