2001-05-23 15:26:42 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: JRD Access Method
|
2003-12-05 11:35:47 +01:00
|
|
|
* MODULE: cch.cpp
|
2001-07-10 19:35:13 +02:00
|
|
|
* DESCRIPTION: Disk cache manager
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
* 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): ______________________________________.
|
2001-07-10 19:35:13 +02:00
|
|
|
* 2001.07.06 Sean Leyne - Code Cleanup, removed "#ifdef READONLY_DATABASE"
|
|
|
|
* conditionals, as the engine now fully supports
|
|
|
|
* readonly databases.
|
2002-10-30 07:40:58 +01:00
|
|
|
*
|
|
|
|
* 2002.10.29 Sean Leyne - Removed obsolete "Netware" port
|
|
|
|
*
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
2001-07-29 19:42:23 +02:00
|
|
|
#include "firebird.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/ib_stdio.h"
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include "../jrd/jrd.h"
|
|
|
|
#include "../jrd/que.h"
|
|
|
|
#include "../jrd/lck.h"
|
|
|
|
#include "../jrd/ods.h"
|
2003-07-14 12:35:49 +02:00
|
|
|
#include "../jrd/os/pio.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/cch.h"
|
2003-11-11 13:19:20 +01:00
|
|
|
#include "gen/iberror.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/lls.h"
|
|
|
|
#include "../jrd/req.h"
|
|
|
|
#include "../jrd/sdw.h"
|
|
|
|
#include "../jrd/tra.h"
|
|
|
|
#include "../jrd/sbm.h"
|
|
|
|
#include "../jrd/iberr.h"
|
|
|
|
#include "../jrd/rse.h"
|
2001-12-24 03:51:06 +01:00
|
|
|
#include "../jrd/btr.h"
|
2003-12-01 03:37:25 +01:00
|
|
|
#include "../jrd/btn.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/gdsassert.h"
|
|
|
|
#include "../jrd/all_proto.h"
|
|
|
|
#include "../jrd/cch_proto.h"
|
|
|
|
#include "../jrd/err_proto.h"
|
|
|
|
#include "../jrd/gds_proto.h"
|
|
|
|
#include "../jrd/iberr_proto.h"
|
|
|
|
#include "../jrd/isc_proto.h"
|
|
|
|
#include "../jrd/isc_s_proto.h"
|
|
|
|
#include "../jrd/jrd_proto.h"
|
|
|
|
|
|
|
|
#include "../jrd/lck_proto.h"
|
|
|
|
#include "../jrd/misc_proto.h"
|
|
|
|
#include "../jrd/mov_proto.h"
|
|
|
|
#include "../jrd/pag_proto.h"
|
2003-07-14 12:35:49 +02:00
|
|
|
#include "../jrd/os/pio_proto.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/sbm_proto.h"
|
|
|
|
#include "../jrd/sch_proto.h"
|
|
|
|
#include "../jrd/sdw_proto.h"
|
|
|
|
#include "../jrd/shut_proto.h"
|
|
|
|
#include "../jrd/thd_proto.h"
|
|
|
|
#include "../jrd/tra_proto.h"
|
2003-02-05 21:43:01 +01:00
|
|
|
#include "../common/config/config.h"
|
|
|
|
#include "../jrd/jrd_time.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-03-01 19:10:10 +01:00
|
|
|
/* In the superserver mode, no page locks are acquired through the lock manager.
|
|
|
|
Instead, a latching mechanism is used. So the calls to lock subsystem for
|
|
|
|
database pages in the original code should not be made, lest they should cause
|
|
|
|
any undesirable side-effects. The following defines help us achieve that. */
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
#define CACHE_WRITER
|
|
|
|
#define PAGE_LATCHING 1
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER_V2
|
|
|
|
#define CACHE_READER
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef PAGE_LATCHING
|
|
|
|
#define PAGE_LOCK(lock, lock_type, wait)
|
2003-08-09 22:58:34 +02:00
|
|
|
#define PAGE_LOCK_RELEASE(lock)
|
2003-03-01 19:10:10 +01:00
|
|
|
#define PAGE_LOCK_ASSERT(lock)
|
|
|
|
#define PAGE_LOCK_OPT(lock, lock_type, wait)
|
2003-08-09 22:58:34 +02:00
|
|
|
#define PAGE_LOCK_RE_POST(lock)
|
2004-03-11 06:04:26 +01:00
|
|
|
#define PAGE_OVERHEAD (sizeof (bcb_repeat) + sizeof(Buffer_desc) + \
|
2003-03-01 19:10:10 +01:00
|
|
|
(int) dbb->dbb_page_size)
|
|
|
|
#else
|
|
|
|
#define PAGE_LOCK(lock, lock_type, wait) LCK_lock (tdbb, lock, lock_type, wait)
|
|
|
|
#define PAGE_LOCK_RELEASE(lock) LCK_release (tdbb, lock)
|
|
|
|
#define PAGE_LOCK_ASSERT(lock) LCK_assert (tdbb, lock)
|
|
|
|
#define PAGE_LOCK_OPT(lock, lock_type, wait) LCK_lock_opt (tdbb, lock, lock_type, wait)
|
|
|
|
#define PAGE_LOCK_RE_POST(lock) LCK_re_post (lock)
|
2004-03-11 06:04:26 +01:00
|
|
|
#define PAGE_OVERHEAD (sizeof (bcb_repeat) + sizeof(Buffer_desc) + \
|
2003-03-01 19:10:10 +01:00
|
|
|
sizeof (struct lck) + (int) dbb->dbb_page_size)
|
|
|
|
#endif
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static Buffer_desc* alloc_bdb(thread_db*, BCB, UCHAR **);
|
2003-03-01 19:10:10 +01:00
|
|
|
#ifndef PAGE_LATCHING
|
2003-05-16 22:35:19 +02:00
|
|
|
static int blocking_ast_bdb(void *);
|
2003-03-01 19:10:10 +01:00
|
|
|
#endif
|
2004-03-11 06:04:26 +01:00
|
|
|
static void btc_flush(thread_db*, SLONG, const bool, ISC_STATUS*);
|
|
|
|
static void btc_insert(Database*, Buffer_desc*);
|
|
|
|
static void btc_remove(Buffer_desc*);
|
2001-05-23 15:26:42 +02:00
|
|
|
static void cache_bugcheck(int);
|
2003-03-01 19:10:10 +01:00
|
|
|
#ifdef CACHE_READER
|
2004-03-07 08:58:55 +01:00
|
|
|
static void THREAD_ROUTINE cache_reader(Database*);
|
2003-03-01 19:10:10 +01:00
|
|
|
#endif
|
|
|
|
#ifdef CACHE_WRITER
|
2004-03-07 08:58:55 +01:00
|
|
|
static void THREAD_ROUTINE cache_writer(Database*);
|
2003-03-01 19:10:10 +01:00
|
|
|
#endif
|
2004-03-11 06:04:26 +01:00
|
|
|
static void check_precedence(thread_db*, WIN *, SLONG);
|
|
|
|
static void clear_precedence(Database*, Buffer_desc*);
|
|
|
|
static Buffer_desc* dealloc_bdb(Buffer_desc*);
|
2003-03-01 19:10:10 +01:00
|
|
|
#ifndef PAGE_LATCHING
|
2004-03-11 06:04:26 +01:00
|
|
|
static void down_grade(thread_db*, Buffer_desc*);
|
2003-03-01 19:10:10 +01:00
|
|
|
#endif
|
2004-03-11 06:04:26 +01:00
|
|
|
static void expand_buffers(thread_db*, ULONG);
|
|
|
|
static Buffer_desc* get_buffer(thread_db*, SLONG, LATCH, SSHORT);
|
|
|
|
static SSHORT latch_bdb(thread_db*, LATCH, Buffer_desc*, SLONG, SSHORT);
|
|
|
|
static SSHORT lock_buffer(thread_db*, Buffer_desc*, SSHORT, SSHORT);
|
|
|
|
static ULONG memory_init(thread_db*, BCB, ULONG);
|
|
|
|
static void page_validation_error(thread_db*, win*, SSHORT);
|
2003-03-01 19:10:10 +01:00
|
|
|
#ifdef CACHE_READER
|
2004-02-20 07:43:27 +01:00
|
|
|
static void prefetch_epilogue(Prefetch*, ISC_STATUS *);
|
2004-03-11 06:04:26 +01:00
|
|
|
static void prefetch_init(Prefetch*, thread_db*);
|
2004-02-20 07:43:27 +01:00
|
|
|
static void prefetch_io(Prefetch*, ISC_STATUS *);
|
|
|
|
static void prefetch_prologue(Prefetch*, SLONG *);
|
2003-03-01 19:10:10 +01:00
|
|
|
#endif
|
2004-03-11 06:04:26 +01:00
|
|
|
static SSHORT related(const Buffer_desc*, const Buffer_desc*, SSHORT);
|
|
|
|
static void release_bdb(thread_db*, Buffer_desc*, const bool, const bool, const bool);
|
|
|
|
static bool writeable(const Buffer_desc*);
|
|
|
|
static int write_buffer(thread_db*, Buffer_desc*, SLONG, const bool, ISC_STATUS*, const bool);
|
|
|
|
static bool write_page(thread_db*, Buffer_desc*, const bool, ISC_STATUS*, const bool);
|
|
|
|
static void unmark(thread_db*, WIN *);
|
|
|
|
static void update_write_direction(thread_db*, Buffer_desc*);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#define MIN_BUFFER_SEGMENT 65536L
|
|
|
|
|
|
|
|
#ifndef DEBUG_PRINTF
|
|
|
|
#define DEBUG_PRINTF(msg) ib_fprintf (ib_stderr, (msg))
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Given pointer a field in the block, find the block */
|
|
|
|
|
|
|
|
#define BLOCK(fld_ptr, type, fld) (type)((SCHAR*) fld_ptr - OFFSET (type, fld))
|
|
|
|
|
|
|
|
#ifdef MULTI_THREAD
|
|
|
|
#ifndef VMS
|
|
|
|
#define INTERLOCK_CACHE
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#define BCB_MUTEX_ACQUIRE
|
|
|
|
#define BCB_MUTEX_RELEASE
|
|
|
|
|
2003-12-01 03:37:25 +01:00
|
|
|
#define PRE_MUTEX_ACQUIRE
|
|
|
|
#define PRE_MUTEX_RELEASE
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-01 03:37:25 +01:00
|
|
|
#define BTC_MUTEX_ACQUIRE
|
|
|
|
#define BTC_MUTEX_RELEASE
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#define LATCH_MUTEX_ACQUIRE
|
|
|
|
#define LATCH_MUTEX_RELEASE
|
|
|
|
|
|
|
|
#define JOURNAL_PAGE -1
|
|
|
|
#define SHADOW_PAGE -2
|
|
|
|
#define FREE_PAGE -3
|
|
|
|
#define CHECKPOINT_PAGE -4
|
|
|
|
#define MIN_PAGE_NUMBER -5
|
|
|
|
|
|
|
|
#define PRE_SEARCH_LIMIT 100
|
|
|
|
#define PRE_EXISTS -1
|
|
|
|
#define PRE_UNKNOWN -2
|
|
|
|
|
|
|
|
#define DUMMY_CHECKSUM 12345
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
bool set_write_direction(Database* dbb, Buffer_desc* bdb, SSHORT direction)
|
2004-02-20 07:43:27 +01:00
|
|
|
{
|
2003-09-08 22:23:46 +02:00
|
|
|
#ifdef SUPERSERVER
|
2003-09-13 20:35:39 +02:00
|
|
|
NBAK_TRACE(("set_write_direction page=%d old=%d new=%d", bdb->bdb_page,
|
|
|
|
bdb->bdb_write_direction, direction));
|
2003-09-08 22:23:46 +02:00
|
|
|
if (bdb->bdb_write_direction == BDB_write_normal ||
|
|
|
|
bdb->bdb_write_direction == BDB_write_both)
|
|
|
|
{
|
|
|
|
if (direction != BDB_write_normal && direction != BDB_write_both)
|
|
|
|
dbb->backup_manager->release_sw_database_lock();
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
|
|
|
else {
|
2003-09-08 22:23:46 +02:00
|
|
|
if (direction == BDB_write_normal || direction == BDB_write_both)
|
2003-09-13 20:35:39 +02:00
|
|
|
dbb->backup_manager->get_sw_database_lock(true);
|
2003-09-08 22:23:46 +02:00
|
|
|
}
|
|
|
|
bdb->bdb_write_direction = direction;
|
|
|
|
#else
|
|
|
|
LCK_ast_inhibit();
|
2004-02-20 07:43:27 +01:00
|
|
|
switch (bdb->bdb_write_direction) {
|
2003-09-08 22:23:46 +02:00
|
|
|
case BDB_write_normal:
|
|
|
|
case BDB_write_both:
|
|
|
|
switch (direction) {
|
|
|
|
case BDB_write_diff:
|
|
|
|
dbb->backup_manager->increment_diff_use_count();
|
|
|
|
dbb->backup_manager->release_sw_database_lock();
|
|
|
|
break;
|
|
|
|
case BDB_write_undefined:
|
|
|
|
dbb->backup_manager->release_sw_database_lock();
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case BDB_write_diff:
|
2004-02-20 07:43:27 +01:00
|
|
|
switch (direction) {
|
2003-09-08 22:23:46 +02:00
|
|
|
case BDB_write_normal:
|
|
|
|
case BDB_write_both:
|
|
|
|
dbb->backup_manager->decrement_diff_use_count();
|
|
|
|
bdb->bdb_write_direction = direction;
|
|
|
|
// We ask this function to enable signals
|
2003-09-13 20:35:39 +02:00
|
|
|
if (!dbb->backup_manager->get_sw_database_lock(true)) {
|
2003-09-08 22:23:46 +02:00
|
|
|
bdb->bdb_write_direction = BDB_write_undefined;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
case BDB_write_undefined:
|
|
|
|
dbb->backup_manager->decrement_diff_use_count();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case BDB_write_undefined:
|
|
|
|
switch (direction) {
|
|
|
|
case BDB_write_diff:
|
|
|
|
dbb->backup_manager->increment_diff_use_count();
|
|
|
|
break;
|
|
|
|
case BDB_write_normal:
|
|
|
|
case BDB_write_both:
|
|
|
|
bdb->bdb_write_direction = direction;
|
|
|
|
// We ask this function to enable signals
|
2003-09-13 20:35:39 +02:00
|
|
|
if (!dbb->backup_manager->get_sw_database_lock(true)) {
|
2003-09-08 22:23:46 +02:00
|
|
|
bdb->bdb_write_direction = BDB_write_undefined;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
bdb->bdb_write_direction = direction;
|
|
|
|
LCK_ast_enable();
|
|
|
|
#endif
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void CCH_flush_database(thread_db* tdbb)
|
2004-02-20 07:43:27 +01:00
|
|
|
{
|
2003-09-08 22:23:46 +02:00
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ f l u s h _ d a t a b a s e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Flush all buffers coming from database file.
|
|
|
|
* May be called from AST
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2003-09-08 22:23:46 +02:00
|
|
|
|
|
|
|
BCB bcb = dbb->dbb_bcb;
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
// This is called on architectures with shared buffer cache (like SuperServer)
|
|
|
|
|
2003-09-13 20:35:39 +02:00
|
|
|
// Redirect pages to the difference file
|
|
|
|
|
|
|
|
// Need to reconsider this protection for possible deadlocks when MT-safety is implemented
|
|
|
|
BCB_MUTEX_ACQUIRE;
|
|
|
|
for (ULONG i = 0; (bcb = dbb->dbb_bcb) && i < bcb->bcb_count; i++) {
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = bcb->bcb_rpt[i].bcb_bdb;
|
2003-09-13 20:35:39 +02:00
|
|
|
if (bdb->bdb_write_direction != BDB_write_normal &&
|
|
|
|
bdb->bdb_write_direction != BDB_write_both)
|
|
|
|
{
|
|
|
|
continue;
|
2003-09-08 22:23:46 +02:00
|
|
|
}
|
2003-09-13 20:35:39 +02:00
|
|
|
NBAK_TRACE(("Redirect page=%d use=%d flags=%d", bdb->bdb_page, bdb->bdb_use_count, bdb->bdb_flags));
|
|
|
|
update_write_direction(tdbb, bdb);
|
2003-09-08 22:23:46 +02:00
|
|
|
}
|
|
|
|
BCB_MUTEX_RELEASE;
|
|
|
|
#else
|
|
|
|
/* Do some fancy footwork to make sure that pages are
|
|
|
|
not removed from the btc tree at AST level. Then
|
|
|
|
restore the flag to whatever it was before. */
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
const bool keep_pages = bcb->bcb_flags & BCB_keep_pages;
|
2003-09-08 22:23:46 +02:00
|
|
|
dbb->dbb_bcb->bcb_flags |= BCB_keep_pages;
|
|
|
|
for (ULONG i = 0; (bcb = dbb->dbb_bcb) && i < bcb->bcb_count; i++) {
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = bcb->bcb_rpt[i].bcb_bdb;
|
2004-01-06 11:33:18 +01:00
|
|
|
if (!(bdb->bdb_flags & (BDB_dirty | BDB_db_dirty)) ||
|
2003-12-22 11:00:59 +01:00
|
|
|
bdb->bdb_write_direction == BDB_write_diff)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
2003-09-08 22:23:46 +02:00
|
|
|
down_grade(tdbb, bdb);
|
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!keep_pages) {
|
2003-09-08 22:23:46 +02:00
|
|
|
bcb->bcb_flags &= ~BCB_keep_pages;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2003-09-08 22:23:46 +02:00
|
|
|
#endif
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
USHORT CCH_checksum(Buffer_desc* bdb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ c h e c k s u m
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Compute the checksum of a page.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
#ifdef NO_CHECKSUM
|
|
|
|
return DUMMY_CHECKSUM;
|
|
|
|
#else
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = bdb->bdb_dbb;
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef WIN_NT
|
|
|
|
/* ODS_VERSION8 for NT was shipped before page checksums
|
|
|
|
were disabled on other platforms. Continue to compute
|
|
|
|
checksums for ODS_VERSION8 databases but eliminate them
|
|
|
|
for ODS_VERSION9 databases. The following code can be
|
|
|
|
deleted when development on ODS_VERSION10 begins and
|
|
|
|
NO_CHECKSUM is defined for all platforms. */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (dbb->dbb_ods_version >= ODS_VERSION9) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return DUMMY_CHECKSUM;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2003-12-22 11:00:59 +01:00
|
|
|
pag* page = bdb->bdb_buffer;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
const ULONG* const end = (ULONG *) ((SCHAR *) page + dbb->dbb_page_size);
|
|
|
|
const USHORT old_checksum = page->pag_checksum;
|
2001-05-23 15:26:42 +02:00
|
|
|
page->pag_checksum = 0;
|
2003-12-22 11:00:59 +01:00
|
|
|
const ULONG* p = (ULONG *) page;
|
|
|
|
ULONG checksum = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
do {
|
|
|
|
checksum += *p++;
|
|
|
|
checksum += *p++;
|
|
|
|
checksum += *p++;
|
|
|
|
checksum += *p++;
|
|
|
|
checksum += *p++;
|
|
|
|
checksum += *p++;
|
|
|
|
checksum += *p++;
|
|
|
|
checksum += *p++;
|
2004-02-20 07:43:27 +01:00
|
|
|
} while (p < end);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
page->pag_checksum = old_checksum;
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (checksum) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return (USHORT) checksum;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If the page is all zeros, return an artificial checksum */
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
for (p = (ULONG *) page; p < end;) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (*p++)
|
|
|
|
return (USHORT) checksum;
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Page is all zeros -- invent a checksum */
|
|
|
|
|
|
|
|
return 12345;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
int CCH_down_grade_dbb(void* ast_object)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ d o w n _ g r a d e _ d b b
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Down grade the lock on the database in response to a blocking
|
|
|
|
* AST.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = static_cast<Database*>(ast_object);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Ignore the request if the database or lock block does not appear
|
|
|
|
to be valid . */
|
2004-02-20 07:43:27 +01:00
|
|
|
lck* lock;
|
2001-12-24 03:51:06 +01:00
|
|
|
if ((MemoryPool::blk_type(dbb) != type_dbb) ||
|
2001-05-23 15:26:42 +02:00
|
|
|
!(lock = dbb->dbb_lock) ||
|
2001-12-24 03:51:06 +01:00
|
|
|
(MemoryPool::blk_type(lock) != type_lck) || !(lock->lck_id))
|
2003-12-22 11:00:59 +01:00
|
|
|
{
|
2003-05-16 22:35:19 +02:00
|
|
|
return 0;
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Since this routine will be called asynchronously, we must establish
|
|
|
|
a thread context. */
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db thd_context, *tdbb;
|
2001-05-23 15:26:42 +02:00
|
|
|
SET_THREAD_DATA;
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
ISC_STATUS_ARRAY ast_status;
|
2001-05-23 15:26:42 +02:00
|
|
|
tdbb->tdbb_database = dbb;
|
|
|
|
tdbb->tdbb_attachment = lock->lck_attachment;
|
|
|
|
tdbb->tdbb_quantum = QUANTUM;
|
|
|
|
tdbb->tdbb_request = NULL;
|
|
|
|
tdbb->tdbb_transaction = NULL;
|
|
|
|
tdbb->tdbb_status_vector = ast_status;
|
|
|
|
|
|
|
|
dbb->dbb_ast_flags |= DBB_blocking;
|
|
|
|
|
|
|
|
/* Database shutdown will release the database lock; just return. */
|
|
|
|
|
|
|
|
if (SHUT_blocking_ast(dbb)) {
|
|
|
|
dbb->dbb_ast_flags &= ~DBB_blocking;
|
|
|
|
RESTORE_THREAD_DATA;
|
2003-05-16 22:35:19 +02:00
|
|
|
return 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
if (dbb->dbb_use_count)
|
|
|
|
{
|
|
|
|
RESTORE_THREAD_DATA;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* If we are already shared, there is nothing more we can do.
|
|
|
|
If any case, the other guy probably wants exclusive access,
|
|
|
|
and we can't give it anyway */
|
|
|
|
|
|
|
|
if ((lock->lck_logical == LCK_SW) || (lock->lck_logical == LCK_SR)) {
|
|
|
|
RESTORE_THREAD_DATA;
|
2003-05-16 22:35:19 +02:00
|
|
|
return 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (dbb->dbb_flags & DBB_bugcheck) {
|
|
|
|
LCK_convert(tdbb, lock, LCK_SW, LCK_WAIT);
|
|
|
|
dbb->dbb_ast_flags &= ~DBB_blocking;
|
|
|
|
RESTORE_THREAD_DATA;
|
2003-05-16 22:35:19 +02:00
|
|
|
return 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* If we are supposed to be exclusive, stay exclusive */
|
|
|
|
|
2004-02-25 02:50:40 +01:00
|
|
|
if ((dbb->dbb_flags & DBB_exclusive) || (dbb->dbb_ast_flags & DBB_shutdown_single)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
RESTORE_THREAD_DATA;
|
2003-05-16 22:35:19 +02:00
|
|
|
return 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Assert any page locks that have been requested, but not asserted */
|
|
|
|
|
|
|
|
ISC_ast_enter();
|
|
|
|
|
|
|
|
dbb->dbb_ast_flags |= DBB_assert_locks;
|
2003-12-22 11:00:59 +01:00
|
|
|
BCB bcb = dbb->dbb_bcb;
|
|
|
|
if (bcb) {
|
|
|
|
if (bcb->bcb_count) {
|
|
|
|
const bcb_repeat* tail = bcb->bcb_rpt;
|
|
|
|
for (const bcb_repeat* const end = tail + bcb->bcb_count;
|
|
|
|
tail < end; tail++)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
PAGE_LOCK_ASSERT(tail->bcb_bdb->bdb_lock);
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Down grade the lock on the database itself */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (lock->lck_physical == LCK_EX) {
|
2001-05-23 15:26:42 +02:00
|
|
|
LCK_convert(tdbb, lock, LCK_PW, LCK_WAIT); /* This lets waiting cache manager in first */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
LCK_convert(tdbb, lock,
|
|
|
|
(dbb->dbb_flags & DBB_cache_manager) ? LCK_SR : LCK_SW,
|
|
|
|
LCK_WAIT);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
dbb->dbb_ast_flags &= ~DBB_blocking;
|
|
|
|
ISC_ast_exit();
|
|
|
|
|
|
|
|
/* Restore the prior thread context */
|
|
|
|
|
|
|
|
RESTORE_THREAD_DATA;
|
2003-05-16 22:35:19 +02:00
|
|
|
return 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
bool CCH_exclusive(thread_db* tdbb, USHORT level, SSHORT wait_flag)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ e x c l u s i v e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2004-02-20 07:43:27 +01:00
|
|
|
* Get exclusive access to a database. If we get it, return true.
|
2001-07-10 19:35:13 +02:00
|
|
|
* If the wait flag is FALSE, and we can't get it, give up and
|
2004-02-20 07:43:27 +01:00
|
|
|
* return false. There are two levels of database exclusivity: LCK_PW
|
2001-05-23 15:26:42 +02:00
|
|
|
* guarantees there are no normal users in the database while LCK_EX
|
|
|
|
* additionally guarantes background database processes like the
|
|
|
|
* shared cache manager have detached.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef SUPERSERVER
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!CCH_exclusive_attachment(tdbb, level, wait_flag)) {
|
2004-02-20 07:43:27 +01:00
|
|
|
return false;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2004-02-20 07:43:27 +01:00
|
|
|
lck* lock = dbb->dbb_lock;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!lock) {
|
2004-02-20 07:43:27 +01:00
|
|
|
return false;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
dbb->dbb_flags |= DBB_exclusive;
|
|
|
|
|
|
|
|
switch (level) {
|
|
|
|
case LCK_PW:
|
|
|
|
if ((lock->lck_physical >= LCK_PW)
|
|
|
|
|| LCK_convert(tdbb, lock, LCK_PW, wait_flag))
|
2003-12-22 11:00:59 +01:00
|
|
|
{
|
2004-02-20 07:43:27 +01:00
|
|
|
return true;
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case LCK_EX:
|
|
|
|
if (lock->lck_physical == LCK_EX ||
|
2003-12-22 11:00:59 +01:00
|
|
|
LCK_convert(tdbb, lock, LCK_EX, wait_flag))
|
|
|
|
{
|
2004-02-20 07:43:27 +01:00
|
|
|
return true;
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If we are supposed to wait (presumably patiently),
|
|
|
|
but can't get the lock, generate an error */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (wait_flag == LCK_WAIT) {
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_deadlock, 0);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
dbb->dbb_flags &= ~DBB_exclusive;
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
return false;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
bool CCH_exclusive_attachment(thread_db* tdbb, USHORT level, SSHORT wait_flag)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ e x c l u s i v e _ a t t a c h m e n t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2004-02-20 07:43:27 +01:00
|
|
|
* Get exclusive access to a database. If we get it, return true.
|
2001-07-10 19:35:13 +02:00
|
|
|
* If the wait flag is FALSE, and we can't get it, give up and
|
2004-02-20 07:43:27 +01:00
|
|
|
* return false.
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
#define CCH_EXCLUSIVE_RETRY_INTERVAL 1 /* retry interval in seconds */
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2003-12-22 11:00:59 +01:00
|
|
|
att* attachment = tdbb->tdbb_attachment;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (attachment->att_flags & ATT_exclusive) {
|
2004-02-20 07:43:27 +01:00
|
|
|
return true;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
attachment->att_flags |=
|
|
|
|
(level == LCK_none) ? ATT_attach_pending : ATT_exclusive_pending;
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
const SLONG timeout =
|
|
|
|
(SLONG) (wait_flag < 0) ? -wait_flag :
|
|
|
|
((wait_flag == LCK_WAIT) ? 1L << 30 : CCH_EXCLUSIVE_RETRY_INTERVAL);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If requesting exclusive database access, then re-position attachment as the
|
|
|
|
youngest so that pending attachments may pass. */
|
|
|
|
|
|
|
|
if (level != LCK_none) {
|
2003-12-22 11:00:59 +01:00
|
|
|
for (att** ptr = &dbb->dbb_attachments; *ptr; ptr = &(*ptr)->att_next) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (*ptr == attachment) {
|
|
|
|
*ptr = attachment->att_next;
|
|
|
|
break;
|
|
|
|
}
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
attachment->att_next = dbb->dbb_attachments;
|
|
|
|
dbb->dbb_attachments = attachment;
|
|
|
|
}
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
for (SLONG remaining = timeout; remaining > 0;
|
|
|
|
remaining -= CCH_EXCLUSIVE_RETRY_INTERVAL)
|
|
|
|
{
|
2004-01-03 11:59:52 +01:00
|
|
|
if (tdbb->tdbb_attachment->att_flags & ATT_shutdown) {
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
bool found = false;
|
2001-05-23 15:26:42 +02:00
|
|
|
for (attachment = tdbb->tdbb_attachment->att_next; attachment;
|
2003-12-22 11:00:59 +01:00
|
|
|
attachment = attachment->att_next)
|
|
|
|
{
|
2004-01-03 11:59:52 +01:00
|
|
|
if (attachment->att_flags & ATT_shutdown) {
|
2001-05-23 15:26:42 +02:00
|
|
|
continue;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (level == LCK_none) { /* Wait for other attachments requesting exclusive access */
|
|
|
|
if (attachment->att_flags &
|
2003-12-22 11:00:59 +01:00
|
|
|
(ATT_exclusive | ATT_exclusive_pending))
|
|
|
|
{
|
|
|
|
found = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else { /* Requesting exclusive database access */
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
found = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (attachment->att_flags & ATT_exclusive_pending) {
|
|
|
|
tdbb->tdbb_attachment->att_flags &=
|
|
|
|
~ATT_exclusive_pending;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (wait_flag == LCK_WAIT) {
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_deadlock, 0);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else {
|
2004-02-20 07:43:27 +01:00
|
|
|
return false;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!found) {
|
|
|
|
tdbb->tdbb_attachment->att_flags &=
|
|
|
|
~(ATT_exclusive_pending | ATT_attach_pending);
|
2004-01-03 11:59:52 +01:00
|
|
|
if (level != LCK_none) {
|
2001-05-23 15:26:42 +02:00
|
|
|
tdbb->tdbb_attachment->att_flags |= ATT_exclusive;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2004-02-20 07:43:27 +01:00
|
|
|
return true;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Our thread needs to sleep for CCH_EXCLUSIVE_RETRY_INTERVAL seconds. */
|
|
|
|
|
2002-04-29 13:14:46 +02:00
|
|
|
if (remaining > CCH_EXCLUSIVE_RETRY_INTERVAL)
|
|
|
|
{
|
|
|
|
THREAD_EXIT;
|
|
|
|
THREAD_SLEEP(CCH_EXCLUSIVE_RETRY_INTERVAL * 1000);
|
|
|
|
THREAD_ENTER;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef CANCEL_OPERATION
|
2003-12-22 11:00:59 +01:00
|
|
|
if (tdbb->tdbb_attachment->att_flags & ATT_cancel_raise) {
|
|
|
|
if (JRD_reschedule(tdbb, 0, false)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
tdbb->tdbb_attachment->att_flags &=
|
|
|
|
~(ATT_exclusive_pending | ATT_attach_pending);
|
|
|
|
ERR_punt();
|
|
|
|
}
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
tdbb->tdbb_attachment->att_flags &=
|
|
|
|
~(ATT_exclusive_pending | ATT_attach_pending);
|
2004-02-20 07:43:27 +01:00
|
|
|
return false;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void CCH_expand(thread_db* tdbb, ULONG number)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ e x p a n d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Expand the cache to at least a given number of buffers. If
|
|
|
|
* it's already that big, don't do anything.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
BCB_MUTEX_ACQUIRE;
|
|
|
|
expand_buffers(tdbb, number);
|
|
|
|
BCB_MUTEX_RELEASE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
PAG CCH_fake(thread_db* tdbb, WIN * window, SSHORT latch_wait)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ f a k e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Fake a fetch to a page. Rather than reading it, however,
|
|
|
|
* zero it in memory. This is used when allocating a new page.
|
|
|
|
*
|
|
|
|
* 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,
|
|
|
|
* or an IO would be necessary, then give
|
|
|
|
* up and return 0.
|
|
|
|
* <negative number> => Latch timeout interval in seconds.
|
|
|
|
*
|
|
|
|
* return
|
|
|
|
* pag pointer if successful.
|
|
|
|
* NULL pointer if timeout occurred (only possible if latch_wait <> 1).
|
|
|
|
* NULL pointer if latch_wait=0 and the faked page would have to be
|
|
|
|
* before reuse.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* if there has been a shadow added recently, go out and
|
|
|
|
find it before we grant any more write locks */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (dbb->dbb_ast_flags & DBB_get_shadows) {
|
2001-05-23 15:26:42 +02:00
|
|
|
SDW_get_shadows();
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = get_buffer(tdbb, window->win_page, LATCH_exclusive, latch_wait);
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!bdb) {
|
2003-09-01 09:58:04 +02:00
|
|
|
return NULL; /* latch timeout occurred */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If a dirty orphaned page is being reused - better write it first
|
|
|
|
to clear current precedences and checkpoint state. This would also
|
|
|
|
update the bcb_free_pages field appropriately. */
|
|
|
|
|
|
|
|
if (bdb->bdb_flags & (BDB_dirty | BDB_db_dirty)) {
|
|
|
|
/* If the caller didn't want to wait at all, then
|
|
|
|
return 'try to fake an other page' to the caller. */
|
|
|
|
|
|
|
|
if (!latch_wait) {
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, false, false, false);
|
2003-09-01 09:58:04 +02:00
|
|
|
return NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!write_buffer
|
2003-12-22 11:00:59 +01:00
|
|
|
(tdbb, bdb, bdb->bdb_page, true, tdbb->tdbb_status_vector, true))
|
|
|
|
{
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else if (QUE_NOT_EMPTY(bdb->bdb_lower)) {
|
|
|
|
/* Clear residual precedence left over from AST-level I/O. */
|
|
|
|
clear_precedence(dbb, bdb);
|
|
|
|
}
|
|
|
|
|
|
|
|
bdb->bdb_flags = (BDB_writer | BDB_faked);
|
|
|
|
bdb->bdb_scan_count = 0;
|
|
|
|
|
|
|
|
lock_buffer(tdbb, bdb, LCK_WAIT, pag_undefined);
|
|
|
|
|
|
|
|
MOVE_CLEAR(bdb->bdb_buffer, (SLONG) dbb->dbb_page_size);
|
|
|
|
window->win_buffer = bdb->bdb_buffer;
|
|
|
|
window->win_expanded_buffer = NULL;
|
|
|
|
window->win_bdb = bdb;
|
|
|
|
window->win_flags = 0;
|
|
|
|
CCH_MARK(tdbb, window);
|
|
|
|
|
|
|
|
return bdb->bdb_buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
PAG CCH_fetch(thread_db* tdbb,
|
2004-02-20 07:43:27 +01:00
|
|
|
WIN* window,
|
2001-05-23 15:26:42 +02:00
|
|
|
USHORT lock_type,
|
|
|
|
SSHORT page_type,
|
2004-02-25 02:50:40 +01:00
|
|
|
SSHORT checksum, SSHORT latch_wait, bool read_shadow)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ f e t c h
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Fetch a specific page. If it's already in cache,
|
|
|
|
* so much the better.
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
* PAG if successful.
|
|
|
|
* NULL pointer if timeout occurred (only possible if latch_wait <> 1).
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SSHORT fetch_lock_return;
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
/* FETCH_LOCK will return 0, 1, -1 or -2 */
|
|
|
|
|
|
|
|
fetch_lock_return =
|
|
|
|
CCH_FETCH_LOCK(tdbb, window, lock_type, LCK_WAIT, latch_wait,
|
|
|
|
page_type);
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (fetch_lock_return == 1) {
|
2001-05-23 15:26:42 +02:00
|
|
|
CCH_FETCH_PAGE(tdbb, window, checksum, read_shadow); /* must read page from disk */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else if (fetch_lock_return == -2 || fetch_lock_return == -1) {
|
2003-09-01 09:58:04 +02:00
|
|
|
return NULL; /* latch or lock timeout */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = window->win_bdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If a page was read or prefetched on behalf of a large scan
|
|
|
|
then load the window scan count into the buffer descriptor.
|
|
|
|
This buffer scan count is decremented by releasing a buffer
|
|
|
|
with CCH_RELEASE_TAIL.
|
|
|
|
|
|
|
|
Otherwise zero the buffer scan count to prevent the buffer
|
|
|
|
from being queued to the LRU tail. */
|
|
|
|
|
|
|
|
if (window->win_flags & WIN_large_scan) {
|
|
|
|
if (fetch_lock_return == 1 ||
|
|
|
|
bdb->bdb_flags & BDB_prefetch || bdb->bdb_scan_count < 0)
|
2004-01-03 11:59:52 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_scan_count = window->win_scans;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else if (window->win_flags & WIN_garbage_collector) {
|
2004-01-03 11:59:52 +01:00
|
|
|
if (fetch_lock_return == 1) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_scan_count = -1;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
if (bdb->bdb_flags & BDB_garbage_collect) {
|
2001-05-23 15:26:42 +02:00
|
|
|
window->win_flags |= WIN_garbage_collect;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else if (window->win_flags & WIN_secondary) {
|
2004-01-03 11:59:52 +01:00
|
|
|
if (fetch_lock_return == 1) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_scan_count = -1;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
bdb->bdb_scan_count = 0;
|
2003-09-13 20:35:39 +02:00
|
|
|
if (bdb->bdb_flags & BDB_garbage_collect) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_flags &= ~BDB_garbage_collect;
|
2003-09-13 20:35:39 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Validate the fetched page matches the expected type */
|
|
|
|
|
|
|
|
if (bdb->bdb_buffer->pag_type != (SCHAR) page_type &&
|
|
|
|
page_type != pag_undefined)
|
2004-01-03 11:59:52 +01:00
|
|
|
{
|
|
|
|
page_validation_error(tdbb, window, page_type);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return window->win_buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
SSHORT CCH_fetch_lock(thread_db* tdbb,
|
2003-02-10 14:28:35 +01:00
|
|
|
WIN * window,
|
2001-05-23 15:26:42 +02:00
|
|
|
USHORT lock_type,
|
|
|
|
SSHORT wait, SSHORT latch_wait, SSHORT page_type)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ f e t c h _ l o c k
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Fetch a lock for a specific page.
|
|
|
|
* Return TRUE if the page needs to be
|
|
|
|
* read and FALSE if not. If a timeout
|
|
|
|
* was passed (wait < 0) and the lock
|
|
|
|
* could not be granted return wait.
|
|
|
|
*
|
|
|
|
* input
|
|
|
|
*
|
|
|
|
* wait: LCK_WAIT = TRUE = 1 => Wait as long a necessary to get the lock.
|
|
|
|
* LCK_NO_WAIT = FALSE = 0 => If the lock can't be acquired immediately,
|
|
|
|
* give up and return -1.
|
|
|
|
* <negative number> => Lock timeout interval in seconds.
|
|
|
|
*
|
|
|
|
* 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 -2.
|
|
|
|
* <negative number> => Latch timeout interval in seconds.
|
|
|
|
*
|
|
|
|
* return
|
|
|
|
* 0: fetch & lock were successful, page doesn't need to be read.
|
|
|
|
* 1: fetch & lock were successful, page needs to be read from disk.
|
|
|
|
* -1: lock timed out, fetch failed.
|
|
|
|
* -2: latch timed out, fetch failed, lock not attempted.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* if there has been a shadow added recently, go out and
|
|
|
|
find it before we grant any more write locks */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (dbb->dbb_ast_flags & DBB_get_shadows) {
|
2001-05-23 15:26:42 +02:00
|
|
|
SDW_get_shadows();
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Look for the page in the cache. */
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = get_buffer(tdbb, window->win_page,
|
2001-05-23 15:26:42 +02:00
|
|
|
((lock_type >= LCK_write) ? LATCH_exclusive :
|
|
|
|
LATCH_shared), latch_wait);
|
2004-01-03 11:59:52 +01:00
|
|
|
if ((latch_wait != 1) && (bdb == 0)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return -2; /* latch timeout */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (lock_type >= LCK_write) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_flags |= BDB_writer;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-07-10 19:35:13 +02:00
|
|
|
/* the expanded index buffer is only good when the page is
|
2001-05-23 15:26:42 +02:00
|
|
|
fetched for read; if it is ever fetched for write, it must
|
|
|
|
be discarded */
|
|
|
|
|
|
|
|
if (bdb->bdb_expanded_buffer && (lock_type > LCK_read)) {
|
2001-12-24 03:51:06 +01:00
|
|
|
delete bdb->bdb_expanded_buffer;
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_expanded_buffer = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
window->win_bdb = bdb;
|
|
|
|
window->win_buffer = bdb->bdb_buffer;
|
|
|
|
window->win_expanded_buffer = bdb->bdb_expanded_buffer;
|
|
|
|
|
|
|
|
/* lock_buffer returns 0 or 1 or -1. */
|
|
|
|
return lock_buffer(tdbb, bdb, wait, page_type);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void CCH_fetch_page(
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb,
|
2003-02-10 14:28:35 +01:00
|
|
|
WIN * window,
|
2004-02-20 07:43:27 +01:00
|
|
|
SSHORT compute_checksum, bool read_shadow)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ f e t c h _ p a g e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Fetch a specific page. If it's already in cache,
|
|
|
|
* so much the better. When "compute_checksum" is 1, compute
|
|
|
|
* the checksum of the page. When it is 2, compute
|
|
|
|
* the checksum only when the page type is nonzero.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = window->win_bdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
ISC_STATUS* status = tdbb->tdbb_status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
pag* page = bdb->bdb_buffer;
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_incarnation = ++dbb->dbb_page_incarnation;
|
|
|
|
|
|
|
|
AST_CHECK;
|
|
|
|
++dbb->dbb_reads;
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
THREAD_EXIT;
|
|
|
|
#endif
|
|
|
|
page = bdb->bdb_buffer;
|
2004-02-20 07:43:27 +01:00
|
|
|
jrd_file* file = dbb->dbb_file;
|
2004-01-03 11:59:52 +01:00
|
|
|
SSHORT retryCount = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-07-10 19:35:13 +02:00
|
|
|
/* We will read a page, and if there is an I/O error we will try to
|
2001-05-23 15:26:42 +02:00
|
|
|
use the shadow file, and try reading again, for a maximum of
|
|
|
|
3 tries, before it gives up.
|
|
|
|
|
|
|
|
The read_shadow flag is set to false only in the call to
|
|
|
|
FETCH_NO_SHADOW, which is only called from validate
|
2001-07-10 19:35:13 +02:00
|
|
|
code.
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
read_shadow = false -> IF an I/O error occurs give up (exit
|
2001-05-23 15:26:42 +02:00
|
|
|
the loop, clean up, and return). So the caller,
|
2001-07-10 19:35:13 +02:00
|
|
|
validate in most cases, can know about it and attempt
|
|
|
|
to remedy the situation.
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
read_shadow = true -> IF an I/O error occurs attempt
|
2001-05-23 15:26:42 +02:00
|
|
|
to rollover to the shadow file. If the I/O error is
|
2001-07-10 19:35:13 +02:00
|
|
|
persistant (more than 3 times) error out of the routine by
|
2001-05-23 15:26:42 +02:00
|
|
|
calling CCH_unwind, and eventually punting out. */
|
2003-08-06 18:30:49 +02:00
|
|
|
|
2003-09-08 22:23:46 +02:00
|
|
|
if (!dbb->backup_manager->lock_state(false))
|
2003-08-06 18:30:49 +02:00
|
|
|
{
|
|
|
|
PAGE_LOCK_RELEASE(bdb->bdb_lock);
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
2004-02-20 07:43:27 +01:00
|
|
|
const int bak_state = dbb->backup_manager->get_state();
|
2003-08-06 20:06:22 +02:00
|
|
|
ULONG diff_page = 0;
|
2003-08-06 18:30:49 +02:00
|
|
|
if (bak_state == nbak_state_stalled || bak_state == nbak_state_merge) {
|
2003-09-08 22:23:46 +02:00
|
|
|
if (!dbb->backup_manager->lock_alloc(false))
|
2003-08-06 18:30:49 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
PAGE_LOCK_RELEASE(bdb->bdb_lock);
|
2003-08-06 18:30:49 +02:00
|
|
|
dbb->backup_manager->unlock_state();
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2003-08-06 18:30:49 +02:00
|
|
|
diff_page = dbb->backup_manager->get_page_index(bdb->bdb_page);
|
|
|
|
dbb->backup_manager->unlock_alloc();
|
2003-09-08 22:23:46 +02:00
|
|
|
NBAK_TRACE(("Reading page %d, state=%d, diff page=%d", bdb->bdb_page, bak_state, diff_page));
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (bak_state == nbak_state_normal ||
|
|
|
|
(bak_state == nbak_state_stalled && !diff_page) ||
|
|
|
|
// In merge mode, if we are reading past beyond old end of file and page is in .delta file
|
|
|
|
// then we maintain actual page in difference file. Always read it from there.
|
|
|
|
(bak_state == nbak_state_merge &&
|
|
|
|
(!diff_page || (bdb->bdb_page < dbb->backup_manager->get_backup_pages())))
|
|
|
|
)
|
|
|
|
{
|
2003-09-08 22:23:46 +02:00
|
|
|
NBAK_TRACE(("Reading page %d, state=%d, diff page=%d from DISK",
|
|
|
|
bdb->bdb_page, bak_state, diff_page));
|
2003-08-06 18:30:49 +02:00
|
|
|
// Read page from disk as normal
|
|
|
|
while (!PIO_read(file, bdb, page, status)) {
|
|
|
|
if (!read_shadow) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
THREAD_ENTER;
|
|
|
|
#endif
|
2003-12-22 11:00:59 +01:00
|
|
|
if (!CCH_rollover_to_shadow(dbb, file, false)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
PAGE_LOCK_RELEASE(bdb->bdb_lock);
|
2003-08-06 18:30:49 +02:00
|
|
|
dbb->backup_manager->unlock_state();
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
if (file != dbb->dbb_file) {
|
2003-08-06 18:30:49 +02:00
|
|
|
file = dbb->dbb_file;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2003-08-06 18:30:49 +02:00
|
|
|
else {
|
|
|
|
if (retryCount++ == 3) {
|
|
|
|
ib_fprintf(ib_stderr,
|
|
|
|
"IO error loop Unwind to avoid a hang\n");
|
|
|
|
PAGE_LOCK_RELEASE(bdb->bdb_lock);
|
|
|
|
dbb->backup_manager->unlock_state();
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef SUPERSERVER
|
2003-08-06 18:30:49 +02:00
|
|
|
THREAD_EXIT;
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (diff_page && (
|
|
|
|
bak_state == nbak_state_stalled ||
|
|
|
|
(bak_state == nbak_state_merge &&
|
|
|
|
(bdb->bdb_page >= dbb->backup_manager->get_backup_pages() ||
|
2004-02-02 12:02:12 +01:00
|
|
|
page->pag_scn < dbb->backup_manager->get_current_scn())
|
2003-08-06 18:30:49 +02:00
|
|
|
)))
|
|
|
|
{
|
2003-09-08 22:23:46 +02:00
|
|
|
NBAK_TRACE(("Reading page %d, state=%d, diff page=%d from DIFFERENCE",
|
|
|
|
bdb->bdb_page, bak_state, diff_page));
|
|
|
|
if (!dbb->backup_manager->read_difference(diff_page, page)) {
|
2003-08-06 18:30:49 +02:00
|
|
|
PAGE_LOCK_RELEASE(bdb->bdb_lock);
|
|
|
|
dbb->backup_manager->unlock_state();
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
2003-09-08 22:23:46 +02:00
|
|
|
if (page->pag_type == pag_undefined) {
|
|
|
|
// Page was marked as allocated inside the difference file, but not really used
|
|
|
|
// this is very rare, but possible case (after certain errors).
|
|
|
|
// Read (or re-read) page from database
|
|
|
|
NBAK_TRACE(("Re-reading page %d, state=%d, diff page=%d from DISK",
|
|
|
|
bdb->bdb_page, bak_state, diff_page));
|
|
|
|
while (!PIO_read(file, bdb, page, status)) {
|
|
|
|
if (!read_shadow) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
THREAD_ENTER;
|
|
|
|
#endif
|
2003-12-22 11:00:59 +01:00
|
|
|
if (!CCH_rollover_to_shadow(dbb, file, false)) {
|
2003-09-08 22:23:46 +02:00
|
|
|
PAGE_LOCK_RELEASE(bdb->bdb_lock);
|
|
|
|
dbb->backup_manager->unlock_state();
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2003-09-08 22:23:46 +02:00
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
if (file != dbb->dbb_file) {
|
2003-09-08 22:23:46 +02:00
|
|
|
file = dbb->dbb_file;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2003-09-08 22:23:46 +02:00
|
|
|
else {
|
|
|
|
if (retryCount++ == 3) {
|
|
|
|
ib_fprintf(ib_stderr,
|
|
|
|
"IO error loop Unwind to avoid a hang\n");
|
|
|
|
PAGE_LOCK_RELEASE(bdb->bdb_lock);
|
|
|
|
dbb->backup_manager->unlock_state();
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2003-09-08 22:23:46 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
THREAD_EXIT;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2003-08-06 18:30:49 +02:00
|
|
|
|
|
|
|
dbb->backup_manager->unlock_state();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifndef NO_CHECKSUM
|
|
|
|
if (((compute_checksum == 1)
|
|
|
|
|| ((compute_checksum == 2) && page->pag_type))
|
|
|
|
&& ((page->pag_checksum != CCH_checksum(bdb))
|
2004-01-03 11:59:52 +01:00
|
|
|
&& !(dbb->dbb_flags & DBB_damaged)))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef SUPERSERVER
|
|
|
|
THREAD_ENTER;
|
|
|
|
#endif
|
|
|
|
IBERR_build_status(tdbb->tdbb_status_vector,
|
2003-11-11 13:19:20 +01:00
|
|
|
isc_db_corrupt,
|
|
|
|
isc_arg_string, "",
|
|
|
|
isc_arg_gds, isc_bad_checksum,
|
|
|
|
isc_arg_gds, isc_badpage,
|
|
|
|
isc_arg_number, (SLONG) bdb->bdb_page, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
/* We should invalidate this bad buffer. */
|
|
|
|
|
|
|
|
PAGE_LOCK_RELEASE(bdb->bdb_lock);
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif /* NO_CHECKSUM */
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
THREAD_ENTER;
|
|
|
|
#endif
|
|
|
|
AST_CHECK;
|
|
|
|
|
|
|
|
bdb->bdb_flags &= ~(BDB_not_valid | BDB_read_pending);
|
|
|
|
window->win_buffer = bdb->bdb_buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void CCH_fini(thread_db* tdbb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ f i n i
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Shut down buffer operation.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2004-01-13 10:52:19 +01:00
|
|
|
bool flush_error = false;
|
|
|
|
|
|
|
|
// CVC: Patching a conversion error FB1->FB2 with crude logic
|
|
|
|
for (int i = 0; i < 2; ++i)
|
|
|
|
{
|
2004-01-09 02:06:12 +01:00
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
try {
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
/* If we've been initialized, either flush buffers
|
|
|
|
or release locks, depending on where we've been
|
|
|
|
bug-checked; as a defensive programming measure,
|
|
|
|
make sure that the buffers were actually allocated */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
bcb_repeat* tail;
|
|
|
|
BCB bcb = dbb->dbb_bcb;
|
|
|
|
if (bcb && (tail = bcb->bcb_rpt) && (tail->bcb_bdb)) {
|
|
|
|
if (dbb->dbb_flags & DBB_bugcheck || flush_error) {
|
|
|
|
for (const bcb_repeat* const end = bcb->bcb_rpt + bcb->bcb_count;
|
|
|
|
tail < end; tail++)
|
|
|
|
{
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = tail->bcb_bdb;
|
2004-01-13 10:52:19 +01:00
|
|
|
if (bdb->bdb_expanded_buffer) {
|
|
|
|
delete bdb->bdb_expanded_buffer;
|
|
|
|
bdb->bdb_expanded_buffer = NULL;
|
|
|
|
}
|
|
|
|
PAGE_LOCK_RELEASE(bdb->bdb_lock);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-01-13 10:52:19 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
CCH_flush(tdbb, (USHORT) FLUSH_FINI, (SLONG) 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
|
|
|
#ifdef CACHE_READER
|
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
/* Shutdown the dedicated cache reader for this database. */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
if ((bcb = dbb->dbb_bcb) && (bcb->bcb_flags & BCB_cache_reader)) {
|
2004-02-02 12:02:12 +01:00
|
|
|
event_t* event = dbb->dbb_reader_event;
|
2004-01-13 10:52:19 +01:00
|
|
|
bcb->bcb_flags &= ~BCB_cache_reader;
|
|
|
|
ISC_event_post(event);
|
|
|
|
SLONG count = ISC_event_clear(event);
|
|
|
|
THREAD_EXIT;
|
|
|
|
ISC_event_wait(1, &event, &count, 0, NULL, 0);
|
|
|
|
THREAD_ENTER;
|
|
|
|
/* Now dispose off the cache reader associated semaphore */
|
|
|
|
ISC_event_fini(event);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CACHE_WRITER
|
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
/* Shutdown the dedicated cache writer for this database. */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
if ((bcb = dbb->dbb_bcb) && (bcb->bcb_flags & BCB_cache_writer)) {
|
2004-02-02 12:02:12 +01:00
|
|
|
event_t* event = dbb->dbb_writer_event_fini;
|
2004-01-13 10:52:19 +01:00
|
|
|
/* initialize initialization event */
|
|
|
|
ISC_event_init(event, 0, 0);
|
|
|
|
SLONG count = ISC_event_clear(event);
|
2003-02-12 23:57:57 +01:00
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
bcb->bcb_flags &= ~BCB_cache_writer;
|
|
|
|
ISC_event_post(dbb->dbb_writer_event); /* Wake up running thread */
|
|
|
|
THREAD_EXIT;
|
|
|
|
ISC_event_wait(1, &event, &count, 0, NULL, 0);
|
|
|
|
THREAD_ENTER;
|
|
|
|
/* Cleanup initialization event */
|
|
|
|
ISC_event_fini(event);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
/* close the database file and all associated shadow files */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
PIO_close(dbb->dbb_file);
|
|
|
|
SDW_close();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
if ( (bcb = dbb->dbb_bcb) ) {
|
|
|
|
while (bcb->bcb_memory) {
|
|
|
|
gds__free(LLS_POP(&bcb->bcb_memory));
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef CACHE_WRITER
|
2004-01-13 10:52:19 +01:00
|
|
|
/* Dispose off any associated latching semaphores */
|
|
|
|
while (QUE_NOT_EMPTY(bcb->bcb_free_lwt)) {
|
|
|
|
QUE que = bcb->bcb_free_lwt.que_forward;
|
|
|
|
QUE_DELETE((*que));
|
2004-02-20 07:43:27 +01:00
|
|
|
Latch_wait* lwt = (Latch_wait*) BLOCK(que, Latch_wait*, lwt_waiters);
|
|
|
|
ISC_event_fini(&lwt->lwt_event);
|
2004-01-13 10:52:19 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2004-01-13 10:52:19 +01:00
|
|
|
}
|
2004-01-09 02:06:12 +01:00
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
} // try
|
2004-03-01 04:35:23 +01:00
|
|
|
catch (const std::exception& ex)
|
2004-01-13 10:52:19 +01:00
|
|
|
{
|
2004-03-01 04:35:23 +01:00
|
|
|
Firebird::stuff_exception(tdbb->tdbb_status_vector, ex);
|
2004-01-13 10:52:19 +01:00
|
|
|
if (!flush_error) {
|
|
|
|
flush_error = true;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
ERR_punt();
|
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2004-01-13 10:52:19 +01:00
|
|
|
|
|
|
|
if (!flush_error) { // wasn't set in the catch => no failure, just exit
|
|
|
|
break;
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
|
|
|
|
2004-01-13 10:52:19 +01:00
|
|
|
} // for
|
2004-01-09 02:06:12 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void CCH_flush(thread_db* tdbb, USHORT flush_flag, SLONG tra_number)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ f l u s h
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2001-07-10 19:35:13 +02:00
|
|
|
* Flush all buffers. If the release flag is set,
|
2001-05-23 15:26:42 +02:00
|
|
|
* release all locks.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
BCB bcb = dbb->dbb_bcb;
|
2004-01-03 11:59:52 +01:00
|
|
|
ISC_STATUS* status = tdbb->tdbb_status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* note that some of the code for btc_flush()
|
|
|
|
replicates code in the for loop
|
|
|
|
to minimize call overhead -- changes should
|
|
|
|
be made in both places */
|
|
|
|
|
|
|
|
if (flush_flag & (FLUSH_TRAN | FLUSH_SYSTEM)) {
|
2003-12-22 11:00:59 +01:00
|
|
|
const SLONG transaction_mask =
|
2001-05-23 15:26:42 +02:00
|
|
|
(tra_number) ? 1L << (tra_number & (BITS_PER_LONG - 1)) : 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
bool sys_only = false;
|
|
|
|
if (!transaction_mask && (flush_flag & FLUSH_SYSTEM)) {
|
2003-12-22 11:00:59 +01:00
|
|
|
sys_only = true;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef SUPERSERVER_V2
|
2004-01-06 11:33:18 +01:00
|
|
|
// if (!dbb->dbb_wal && A && B) becomes
|
|
|
|
// if (true && A && B) then finally (A && B)
|
|
|
|
if (!(dbb->dbb_flags & DBB_force_write) && transaction_mask) {
|
2001-05-23 15:26:42 +02:00
|
|
|
dbb->dbb_flush_cycle |= transaction_mask;
|
|
|
|
if (!(bcb->bcb_flags & BCB_writer_active))
|
|
|
|
ISC_event_post(dbb->dbb_writer_event);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
btc_flush(tdbb, transaction_mask, sys_only, status);
|
|
|
|
}
|
|
|
|
else {
|
2003-12-22 11:00:59 +01:00
|
|
|
const bool all_flag = (flush_flag & FLUSH_ALL) != 0;
|
|
|
|
const bool release_flag = (flush_flag & FLUSH_RLSE) != 0;
|
|
|
|
const bool write_thru = release_flag;
|
|
|
|
const bool sweep_flag = (flush_flag & FLUSH_SWEEP) != 0;
|
|
|
|
LATCH latch = (release_flag) ? LATCH_exclusive : LATCH_none;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
for (ULONG i = 0; (bcb = dbb->dbb_bcb) && i < bcb->bcb_count; i++) {
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = bcb->bcb_rpt[i].bcb_bdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!release_flag
|
2003-12-22 11:00:59 +01:00
|
|
|
&& !(bdb->bdb_flags & (BDB_dirty | BDB_db_dirty)))
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
if (latch == LATCH_exclusive
|
|
|
|
&& latch_bdb(tdbb, latch, bdb, bdb->bdb_page, 1) == -1)
|
2003-12-22 11:00:59 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
cache_bugcheck(302); /* msg 302 unexpected page change */
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
if (latch == LATCH_exclusive && bdb->bdb_use_count > 1)
|
|
|
|
cache_bugcheck(210); /* msg 210 page in use during flush */
|
|
|
|
#ifdef SUPERSERVER
|
2003-12-22 11:00:59 +01:00
|
|
|
if (bdb->bdb_flags & BDB_db_dirty) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (all_flag
|
|
|
|
|| (sweep_flag
|
2003-12-22 11:00:59 +01:00
|
|
|
&& (!bdb->bdb_parent && bdb != bcb->bcb_btree)))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!write_buffer
|
2003-12-22 11:00:59 +01:00
|
|
|
(tdbb, bdb, bdb->bdb_page, write_thru, status, true))
|
|
|
|
{
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#else
|
2003-12-22 11:00:59 +01:00
|
|
|
if (bdb->bdb_flags & BDB_dirty) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!write_buffer
|
2003-12-22 11:00:59 +01:00
|
|
|
(tdbb, bdb, bdb->bdb_page, false, status, true))
|
|
|
|
{
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
if (release_flag)
|
2003-08-13 13:11:12 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
PAGE_LOCK_RELEASE(bdb->bdb_lock);
|
2003-08-13 13:11:12 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
if (latch == LATCH_exclusive)
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, false, false, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-02-05 21:43:01 +01:00
|
|
|
//
|
|
|
|
// Check if flush needed
|
|
|
|
//
|
2004-01-13 10:52:19 +01:00
|
|
|
const int max_unflushed_writes = Config::getMaxUnflushedWrites();
|
|
|
|
const time_t max_unflushed_write_time = Config::getMaxUnflushedWriteTime();
|
2003-02-16 14:25:44 +01:00
|
|
|
bool max_num = (max_unflushed_writes >= 0);
|
|
|
|
bool max_time = (max_unflushed_write_time >= 0);
|
|
|
|
|
|
|
|
bool doFlush = false;
|
|
|
|
|
|
|
|
if (!(dbb->dbb_file->fil_flags & FIL_force_write) && (max_num || max_time))
|
2003-02-05 21:43:01 +01:00
|
|
|
{
|
2003-10-08 10:42:48 +02:00
|
|
|
const time_t now = time(0);
|
2003-02-16 14:25:44 +01:00
|
|
|
|
2003-02-05 21:43:01 +01:00
|
|
|
THD_MUTEX_LOCK(dbb->dbb_mutexes + DBB_MUTX_flush_count);
|
2003-02-16 14:25:44 +01:00
|
|
|
|
2003-02-06 00:32:32 +01:00
|
|
|
// If this is the first commit set last_flushed_write to now
|
|
|
|
if (!dbb->last_flushed_write)
|
|
|
|
{
|
|
|
|
dbb->last_flushed_write = now;
|
|
|
|
}
|
2003-02-16 14:25:44 +01:00
|
|
|
|
2003-02-06 00:32:32 +01:00
|
|
|
// test max_num condition and max_time condition
|
|
|
|
max_num = max_num
|
|
|
|
&& (dbb->unflushed_writes == max_unflushed_writes);
|
|
|
|
max_time = max_time
|
|
|
|
&& (now - dbb->last_flushed_write > max_unflushed_write_time);
|
2003-02-16 14:25:44 +01:00
|
|
|
|
2003-02-06 00:32:32 +01:00
|
|
|
if (max_num || max_time)
|
|
|
|
{
|
2003-02-16 14:25:44 +01:00
|
|
|
doFlush = true;
|
2003-02-05 21:43:01 +01:00
|
|
|
dbb->unflushed_writes = 0;
|
2003-02-06 00:32:32 +01:00
|
|
|
dbb->last_flushed_write = now;
|
2003-02-05 21:43:01 +01:00
|
|
|
}
|
2003-02-16 14:25:44 +01:00
|
|
|
else
|
|
|
|
{
|
2003-02-05 21:43:01 +01:00
|
|
|
dbb->unflushed_writes++;
|
|
|
|
}
|
2003-02-16 14:25:44 +01:00
|
|
|
|
2003-02-05 21:43:01 +01:00
|
|
|
THD_MUTEX_UNLOCK(dbb->dbb_mutexes + DBB_MUTX_flush_count);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-02-05 21:43:01 +01:00
|
|
|
if (doFlush)
|
|
|
|
{
|
2003-02-16 14:25:44 +01:00
|
|
|
PIO_flush(dbb->dbb_file);
|
2003-02-05 21:43:01 +01:00
|
|
|
if (dbb->dbb_shadow)
|
2003-02-16 14:25:44 +01:00
|
|
|
{
|
|
|
|
PIO_flush(dbb->dbb_shadow->sdw_file);
|
|
|
|
}
|
2003-02-05 21:43:01 +01:00
|
|
|
}
|
2003-02-16 14:25:44 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* take the opportunity when we know there are no pages
|
2001-07-10 19:35:13 +02:00
|
|
|
in cache to check that the shadow(s) have not been
|
2001-05-23 15:26:42 +02:00
|
|
|
scheduled for shutdown or deletion */
|
|
|
|
|
|
|
|
SDW_check();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
bool CCH_free_page(thread_db* tdbb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ f r e e _ p a g e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Check if the cache is below its free pages
|
|
|
|
* threshold and write a page on the LRU tail.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
/* Called by VIO/garbage_collector() when it is idle to
|
|
|
|
help quench the thirst for free pages. */
|
|
|
|
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2004-01-03 11:59:52 +01:00
|
|
|
BCB bcb = dbb->dbb_bcb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (dbb->dbb_flags & DBB_read_only) {
|
2004-02-20 07:43:27 +01:00
|
|
|
return false;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (bcb->bcb_flags & BCB_free_pending &&
|
2003-12-22 11:00:59 +01:00
|
|
|
(bdb = get_buffer(tdbb, FREE_PAGE, LATCH_none, 1)))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!write_buffer
|
2003-12-22 11:00:59 +01:00
|
|
|
(tdbb, bdb, bdb->bdb_page, true, tdbb->tdbb_status_vector, true))
|
|
|
|
{
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, false);
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
2004-02-20 07:43:27 +01:00
|
|
|
return true;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
return false;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SLONG CCH_get_incarnation(WIN * window)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ g e t _ i n c a r n a t i o n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Get page incarnation associated with buffer.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
return window->win_bdb->bdb_incarnation;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
PAG CCH_handoff(thread_db* tdbb,
|
2003-02-10 14:28:35 +01:00
|
|
|
WIN* window,
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG page,
|
|
|
|
SSHORT lock,
|
|
|
|
SSHORT page_type,
|
|
|
|
SSHORT latch_wait,
|
|
|
|
SSHORT release_tail)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ h a n d o f f
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Follow a pointer handing off the lock. Fetch the new page
|
|
|
|
* before retiring the old page lock.
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
* PAG if successful.
|
|
|
|
* 0 if a latch timeout occurred (only possible if latch_wait <> 1).
|
|
|
|
* The latch on the fetched page is downgraded to shared.
|
|
|
|
* The fetched page is unmarked.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
/* The update, if there was one, of the input page is complete.
|
|
|
|
The cache buffer can be 'unmarked'. It is important to
|
|
|
|
unmark before CCH_unwind is (might be) called. */
|
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
unmark(tdbb, window);
|
|
|
|
|
|
|
|
/* If the 'from-page' and 'to-page' of the handoff are the
|
|
|
|
same and the latch requested is shared then downgrade it. */
|
|
|
|
|
|
|
|
if ((window->win_page == page) && (lock == LCK_read)) {
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, window->win_bdb, false, true, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
return window->win_buffer;
|
|
|
|
}
|
|
|
|
|
2003-12-11 11:33:30 +01:00
|
|
|
WIN temp = *window;
|
2001-05-23 15:26:42 +02:00
|
|
|
window->win_page = page;
|
2004-01-03 11:59:52 +01:00
|
|
|
const SSHORT must_read =
|
2001-05-23 15:26:42 +02:00
|
|
|
CCH_FETCH_LOCK(tdbb, window, lock, LCK_WAIT, latch_wait, page_type);
|
|
|
|
|
|
|
|
/* Latch or lock timeout, return failure. */
|
|
|
|
|
|
|
|
if (must_read == -2 || must_read == -1) {
|
|
|
|
*window = temp;
|
|
|
|
CCH_RELEASE(tdbb, window);
|
2003-09-01 09:58:04 +02:00
|
|
|
return NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (release_tail)
|
|
|
|
CCH_RELEASE_TAIL(tdbb, &temp);
|
|
|
|
else
|
|
|
|
CCH_RELEASE(tdbb, &temp);
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (must_read) {
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_FETCH_PAGE(tdbb, window, TRUE, true);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = window->win_bdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If a page was read or prefetched on behalf of a large scan
|
|
|
|
then load the window scan count into the buffer descriptor.
|
|
|
|
This buffer scan count is decremented by releasing a buffer
|
|
|
|
with CCH_RELEASE_TAIL.
|
|
|
|
|
|
|
|
Otherwise zero the buffer scan count to prevent the buffer
|
|
|
|
from being queued to the LRU tail. */
|
|
|
|
|
|
|
|
if (window->win_flags & WIN_large_scan) {
|
|
|
|
if (must_read == 1 ||
|
|
|
|
bdb->bdb_flags & BDB_prefetch || bdb->bdb_scan_count < 0)
|
2004-01-03 11:59:52 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_scan_count = window->win_scans;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else if (window->win_flags & WIN_garbage_collector) {
|
2004-01-03 11:59:52 +01:00
|
|
|
if (must_read == 1) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_scan_count = -1;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
if (bdb->bdb_flags & BDB_garbage_collect) {
|
2001-05-23 15:26:42 +02:00
|
|
|
window->win_flags |= WIN_garbage_collect;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else if (window->win_flags & WIN_secondary) {
|
2004-01-03 11:59:52 +01:00
|
|
|
if (must_read == 1) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_scan_count = -1;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
bdb->bdb_scan_count = 0;
|
2003-09-13 20:35:39 +02:00
|
|
|
if (bdb->bdb_flags & BDB_garbage_collect) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_flags &= ~BDB_garbage_collect;
|
2003-09-13 20:35:39 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Validate the fetched page matches the expected type */
|
|
|
|
|
|
|
|
if (bdb->bdb_buffer->pag_type != (SCHAR) page_type &&
|
|
|
|
page_type != pag_undefined)
|
2004-01-03 11:59:52 +01:00
|
|
|
{
|
|
|
|
page_validation_error(tdbb, window, page_type);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return window->win_buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void CCH_init(thread_db* tdbb, ULONG number)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ i n i t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Initialize the cache. Allocate buffers control block,
|
|
|
|
* buffer descriptors, and actual buffers.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Check for database-specific page buffers */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (dbb->dbb_page_buffers) {
|
2001-05-23 15:26:42 +02:00
|
|
|
number = dbb->dbb_page_buffers;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Enforce page buffer cache constraints */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (number < MIN_PAGE_BUFFERS) {
|
2001-05-23 15:26:42 +02:00
|
|
|
number = MIN_PAGE_BUFFERS;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
if (number > MAX_PAGE_BUFFERS) {
|
2001-05-23 15:26:42 +02:00
|
|
|
number = MAX_PAGE_BUFFERS;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
SLONG count = number;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Allocate and initialize buffers control block */
|
2004-01-03 11:59:52 +01:00
|
|
|
BCB bcb_ = 0;
|
2001-12-24 03:51:06 +01:00
|
|
|
while (!bcb_) {
|
|
|
|
try {
|
2002-09-25 19:12:16 +02:00
|
|
|
bcb_ = FB_NEW_RPT(*dbb->dbb_bufferpool, number) bcb;
|
2004-03-01 04:35:23 +01:00
|
|
|
} catch(const std::exception& ex) {
|
|
|
|
Firebird::stuff_exception(tdbb->tdbb_status_vector, ex);
|
2001-12-24 03:51:06 +01:00
|
|
|
/* If the buffer control block can't be allocated, memory is
|
|
|
|
very low. Recalculate the number of buffers to account for
|
|
|
|
page buffer overhead and reduce that number by a 25% fudge
|
|
|
|
factor. */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
number = (sizeof(bcb_repeat) * number) / PAGE_OVERHEAD;
|
|
|
|
number -= number >> 2;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (number < MIN_PAGE_BUFFERS) {
|
2001-12-24 03:51:06 +01:00
|
|
|
ERR_post(isc_cache_too_small, 0);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
dbb->dbb_bcb = bcb_;
|
|
|
|
QUE_INIT(bcb_->bcb_in_use);
|
|
|
|
QUE_INIT(bcb_->bcb_empty);
|
|
|
|
QUE_INIT(bcb_->bcb_free_lwt);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* initialization of memory is system-specific */
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
bcb_->bcb_count = memory_init(tdbb, bcb_, number);
|
|
|
|
bcb_->bcb_free_minimum = (SSHORT) MIN(bcb_->bcb_count / 4, 128);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bcb_->bcb_count < MIN_PAGE_BUFFERS) {
|
2001-05-23 15:26:42 +02:00
|
|
|
ERR_post(isc_cache_too_small, 0);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Log if requested number of page buffers could not be allocated. */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (count != (SLONG) bcb_->bcb_count) {
|
2001-05-23 15:26:42 +02:00
|
|
|
gds__log
|
|
|
|
("Database: %s\n\tAllocated %ld page buffers of %ld requested",
|
2004-03-14 14:40:14 +01:00
|
|
|
tdbb->tdbb_attachment->att_filename.c_str(), bcb_->bcb_count, count);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (dbb->dbb_lock->lck_logical != LCK_EX) {
|
2001-05-23 15:26:42 +02:00
|
|
|
dbb->dbb_ast_flags |= DBB_assert_locks;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef CACHE_READER
|
2004-02-02 12:02:12 +01:00
|
|
|
event_t* event = dbb->dbb_reader_event;
|
2001-05-23 15:26:42 +02:00
|
|
|
ISC_event_init(event, 0, 0);
|
|
|
|
count = ISC_event_clear(event);
|
2002-04-29 17:05:11 +02:00
|
|
|
if (gds__thread_start
|
2004-02-20 07:43:27 +01:00
|
|
|
(reinterpret_cast<FPTR_INT_VOID_PTR>(cache_reader), dbb,
|
2002-04-29 17:05:11 +02:00
|
|
|
THREAD_high, 0, 0))
|
2004-01-03 11:59:52 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
ERR_bugcheck_msg("cannot start thread");
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
THREAD_EXIT;
|
2003-09-01 09:58:04 +02:00
|
|
|
ISC_event_wait(1, &event, &count, 5 * 1000000, NULL, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
THREAD_ENTER;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef CACHE_WRITER
|
2003-02-12 23:57:57 +01:00
|
|
|
if (!(dbb->dbb_flags & DBB_read_only)) {
|
2004-02-02 12:02:12 +01:00
|
|
|
event_t* event = dbb->dbb_writer_event_init;
|
2003-02-12 23:57:57 +01:00
|
|
|
/* Initialize initialization event */
|
2001-05-23 15:26:42 +02:00
|
|
|
ISC_event_init(event, 0, 0);
|
|
|
|
count = ISC_event_clear(event);
|
2003-02-12 23:57:57 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
if (gds__thread_start
|
2004-02-20 07:43:27 +01:00
|
|
|
(reinterpret_cast<FPTR_INT_VOID_PTR>(cache_writer), dbb,
|
2004-01-03 11:59:52 +01:00
|
|
|
THREAD_high, 0, 0))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
ERR_bugcheck_msg("cannot start thread");
|
2003-02-12 23:57:57 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
THREAD_EXIT;
|
2003-09-01 09:58:04 +02:00
|
|
|
ISC_event_wait(1, &event, &count, 5 * 1000000, NULL, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
THREAD_ENTER;
|
2003-02-12 23:57:57 +01:00
|
|
|
/* Clean up initialization event */
|
|
|
|
ISC_event_fini(event);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void CCH_mark(thread_db* tdbb, WIN * window, USHORT mark_system)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ m a r k
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Mark a window as dirty.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
dbb->dbb_marks++;
|
2004-01-03 11:59:52 +01:00
|
|
|
BCB bcb = dbb->dbb_bcb;
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = window->win_bdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(bdb, type_bdb);
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
/* A LATCH_mark is needed before the Buffer_desc can be marked.
|
2001-05-23 15:26:42 +02:00
|
|
|
This prevents a write while the page is being modified. */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (latch_bdb(tdbb, LATCH_mark, bdb, bdb->bdb_page, 1) == -1) {
|
2001-05-23 15:26:42 +02:00
|
|
|
cache_bugcheck(302); /* msg 302 unexpected page change */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
bdb->bdb_incarnation = ++dbb->dbb_page_incarnation;
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!(bdb->bdb_flags & BDB_writer)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
BUGCHECK(208); /* msg 208 page not accessed for write */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* mark the dirty bit vector for this specific transaction,
|
|
|
|
if it exists; otherwise mark that the system transaction
|
|
|
|
has updated this page */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
SLONG number;
|
|
|
|
jrd_tra* transaction = tdbb->tdbb_transaction;
|
|
|
|
if (transaction && (number = transaction->tra_number)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!(tdbb->tdbb_flags & TDBB_sweeper)) {
|
2004-01-03 11:59:52 +01:00
|
|
|
const ULONG trans_bucket = number & (BITS_PER_LONG - 1);
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_transactions |= (1L << trans_bucket);
|
2004-01-03 11:59:52 +01:00
|
|
|
if (number > bdb->bdb_mark_transaction) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_mark_transaction = number;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_flags |= BDB_system_dirty;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (mark_system) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_flags |= BDB_system_dirty;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!(tdbb->tdbb_flags & TDBB_sweeper) ||
|
2004-01-03 11:59:52 +01:00
|
|
|
bdb->bdb_flags & BDB_system_dirty)
|
|
|
|
{
|
|
|
|
if (!bdb->bdb_parent && bdb != bcb->bcb_btree) {
|
2001-05-23 15:26:42 +02:00
|
|
|
btc_insert(dbb, bdb);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
bdb->bdb_flags |= BDB_db_dirty;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
bdb->bdb_flags |= (BDB_dirty | BDB_marked);
|
2003-09-13 20:35:39 +02:00
|
|
|
update_write_direction(tdbb, bdb);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void CCH_mark_must_write(thread_db* tdbb, WIN * window)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ m a r k _ m u s t _ w r i t e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Mark a window as dirty and must_write.
|
|
|
|
* This will prevent the page from being
|
|
|
|
* inserted in the dirty binary tree when
|
|
|
|
* the intention is to write the page
|
|
|
|
* immediately any way.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = window->win_bdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(bdb, type_bdb);
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!(bdb->bdb_flags & BDB_writer)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
BUGCHECK(208); /* msg 208 page not accessed for write */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
bdb->bdb_flags |= (BDB_dirty | BDB_must_write);
|
|
|
|
|
|
|
|
CCH_MARK(tdbb, window);
|
2003-09-13 20:35:39 +02:00
|
|
|
update_write_direction(tdbb, bdb);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void CCH_must_write(WIN * window)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ m u s t _ w r i t e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Mark a window as "must write".
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = window->win_bdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(bdb, type_bdb);
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!(bdb->bdb_flags & BDB_marked)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
BUGCHECK(208); /* msg 208 page not accessed for write */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
bdb->bdb_flags |= (BDB_dirty | BDB_must_write);
|
2003-09-13 20:35:39 +02:00
|
|
|
update_write_direction(GET_THREAD_DATA, bdb);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
lck* CCH_page_lock(thread_db* tdbb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ p a g e _ l o c k
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Allocate a page-type lock.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
lck* lock = FB_NEW_RPT(*dbb->dbb_bufferpool, sizeof(SLONG)) lck;
|
2001-05-23 15:26:42 +02:00
|
|
|
lock->lck_type = LCK_bdb;
|
|
|
|
lock->lck_owner_handle = LCK_get_owner_handle(tdbb, lock->lck_type);
|
|
|
|
lock->lck_length = sizeof(SLONG);
|
|
|
|
|
|
|
|
lock->lck_dbb = dbb;
|
|
|
|
lock->lck_parent = dbb->dbb_lock;
|
|
|
|
|
|
|
|
return lock;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void CCH_precedence(thread_db* tdbb, WIN * window, SLONG page)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ p r e c e d e n c e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Given a window accessed for write and a page number,
|
|
|
|
* establish a precedence relationship such that the
|
|
|
|
* specified page will always be written before the page
|
|
|
|
* associated with the window.
|
|
|
|
*
|
|
|
|
* If the "page number" is negative, it is really a transaction
|
|
|
|
* id. In this case, the precedence relationship is to the
|
|
|
|
* database header page from which the transaction id was
|
|
|
|
* obtained. If the header page has been written since the
|
|
|
|
* transaction id was assigned, no precedence relationship
|
|
|
|
* is required.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
/* If the page is zero, the caller isn't really serious */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (page == 0) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
check_precedence(tdbb, window, page);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void CCH_prefetch(thread_db* tdbb, SLONG * pages, SSHORT count)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ p r e f e t c h
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Given a vector of pages, set corresponding bits
|
|
|
|
* in global prefetch bitmap. Initiate an asynchronous
|
|
|
|
* I/O and get the cache reader reading in our behalf
|
|
|
|
* as well.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
#ifdef CACHE_READER
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2004-01-03 11:59:52 +01:00
|
|
|
BCB bcb = dbb->dbb_bcb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!count || !(bcb->bcb_flags & BCB_cache_reader)) {
|
|
|
|
/* Caller isn't really serious. */
|
2001-05-23 15:26:42 +02:00
|
|
|
return;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Switch default pool to permanent pool for setting bits in
|
|
|
|
prefetch bitmap. */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
JrdMemoryPool* old_pool = tdbb->tdbb_default;
|
2001-05-23 15:26:42 +02:00
|
|
|
tdbb->tdbb_default = dbb->dbb_bufferpool;
|
|
|
|
|
|
|
|
/* The global prefetch bitmap is the key to the I/O coalescense
|
|
|
|
mechanism which dovetails all thread prefetch requests to
|
|
|
|
minimize sequential I/O requests. It also enables multipage
|
|
|
|
I/O by implicitly sorting page vector requests. */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
SLONG first_page = 0;
|
|
|
|
for (const SLONG* const end = pages + count; pages < end;) {
|
|
|
|
const SLONG page = *pages++;
|
|
|
|
if (page) {
|
2001-05-23 15:26:42 +02:00
|
|
|
SBM_set(tdbb, &bcb->bcb_prefetch, page);
|
|
|
|
if (!first_page)
|
|
|
|
first_page = page;
|
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Not likely that the caller's page vector was
|
|
|
|
empty but check anyway. */
|
|
|
|
|
|
|
|
if (first_page) {
|
2003-10-20 12:53:52 +02:00
|
|
|
prf prefetch;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
prefetch_init(&prefetch, tdbb);
|
|
|
|
prefetch_prologue(&prefetch, &first_page);
|
|
|
|
prefetch_io(&prefetch, tdbb->tdbb_status_vector);
|
|
|
|
prefetch_epilogue(&prefetch, tdbb->tdbb_status_vector);
|
|
|
|
}
|
|
|
|
|
|
|
|
tdbb->tdbb_default = old_pool;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
bool CCH_prefetch_pages(thread_db* tdbb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ p r e f e t c h _ p a g e s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Check the prefetch bitmap for a set
|
2001-07-10 19:35:13 +02:00
|
|
|
* of pages and read them into the cache.
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
/* Place holder to be implemented when predictive prefetch is
|
|
|
|
enabled. This is called by VIO/garbage_collector() when it
|
|
|
|
is idle to help satisfy the prefetch demand. */
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
return false;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void invalidate_and_release_buffer(thread_db* tdbb, Buffer_desc* bdb)
|
2004-01-03 11:59:52 +01:00
|
|
|
{
|
2003-09-08 22:23:46 +02:00
|
|
|
// This function should be called before difference processing is done.
|
|
|
|
// So there should be no need to no need to release difference locks though
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2003-09-08 22:23:46 +02:00
|
|
|
bdb->bdb_flags |= BDB_not_valid;
|
|
|
|
bdb->bdb_flags &= ~BDB_dirty;
|
|
|
|
set_write_direction(dbb, bdb, BDB_write_undefined);
|
|
|
|
TRA_invalidate(dbb, bdb->bdb_transactions);
|
|
|
|
bdb->bdb_transactions = 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, false, false, false);
|
2003-09-08 22:23:46 +02:00
|
|
|
}
|
|
|
|
|
2003-09-13 20:35:39 +02:00
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void update_write_direction(thread_db* tdbb, Buffer_desc* bdb)
|
2004-01-03 11:59:52 +01:00
|
|
|
{
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2004-01-03 11:59:52 +01:00
|
|
|
// Determine location of the page in difference file and write destination
|
2004-03-11 06:04:26 +01:00
|
|
|
// so Buffer_desc AST handlers and write_page routine can safely use this information
|
2003-09-13 20:35:39 +02:00
|
|
|
if (!dbb->backup_manager->lock_state(true)) {
|
|
|
|
invalidate_and_release_buffer(tdbb, bdb);
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2003-09-13 20:35:39 +02:00
|
|
|
}
|
|
|
|
#ifndef SUPERSERVER
|
|
|
|
bdb->bdb_diff_generation = dbb->backup_manager->get_current_generation();
|
|
|
|
#endif
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_page != HEADER_PAGE)
|
|
|
|
{
|
|
|
|
// SCN of header page is adjusted in nbak.cpp
|
2004-02-02 12:02:12 +01:00
|
|
|
bdb->bdb_buffer->pag_scn = dbb->backup_manager->get_current_scn(); // Set SCN for the page
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
SSHORT write_direction;
|
|
|
|
const int backup_state = dbb->backup_manager->get_state();
|
2003-09-13 20:35:39 +02:00
|
|
|
switch (backup_state) {
|
|
|
|
case nbak_state_normal:
|
|
|
|
write_direction = BDB_write_normal;
|
|
|
|
break;
|
|
|
|
case nbak_state_stalled:
|
|
|
|
write_direction = BDB_write_diff;
|
|
|
|
break;
|
|
|
|
case nbak_state_merge:
|
|
|
|
if (tdbb->tdbb_flags & TDBB_backup_merge ||
|
|
|
|
bdb->bdb_page < dbb->backup_manager->get_backup_pages())
|
|
|
|
{
|
|
|
|
write_direction = BDB_write_normal;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else {
|
2003-09-13 20:35:39 +02:00
|
|
|
write_direction = BDB_write_both;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2003-09-13 20:35:39 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
switch (write_direction) {
|
|
|
|
case BDB_write_diff:
|
|
|
|
if (!dbb->backup_manager->lock_alloc(true)) {
|
|
|
|
dbb->backup_manager->unlock_state();
|
|
|
|
invalidate_and_release_buffer(tdbb, bdb);
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2003-09-13 20:35:39 +02:00
|
|
|
}
|
|
|
|
bdb->bdb_difference_page = dbb->backup_manager->get_page_index(bdb->bdb_page);
|
|
|
|
dbb->backup_manager->unlock_alloc();
|
|
|
|
if (!bdb->bdb_difference_page) {
|
|
|
|
if (!dbb->backup_manager->lock_alloc_write(true)) {
|
|
|
|
dbb->backup_manager->unlock_state();
|
|
|
|
invalidate_and_release_buffer(tdbb, bdb);
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2003-09-13 20:35:39 +02:00
|
|
|
}
|
|
|
|
bdb->bdb_difference_page = dbb->backup_manager->allocate_difference_page(bdb->bdb_page);
|
|
|
|
dbb->backup_manager->unlock_alloc_write();
|
|
|
|
if (!bdb->bdb_difference_page) {
|
|
|
|
dbb->backup_manager->unlock_state();
|
|
|
|
invalidate_and_release_buffer(tdbb, bdb);
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2003-09-13 20:35:39 +02:00
|
|
|
}
|
|
|
|
NBAK_TRACE(("Allocate difference page %d for database page %d",
|
|
|
|
bdb->bdb_difference_page, bdb->bdb_page));
|
2004-01-28 08:50:41 +01:00
|
|
|
}
|
|
|
|
else {
|
2003-09-13 20:35:39 +02:00
|
|
|
NBAK_TRACE(("Map existing difference page %d to database page %d",
|
|
|
|
bdb->bdb_difference_page, bdb->bdb_page));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case BDB_write_both:
|
|
|
|
if (!dbb->backup_manager->lock_alloc(true)) {
|
|
|
|
dbb->backup_manager->unlock_state();
|
|
|
|
invalidate_and_release_buffer(tdbb, bdb);
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2003-09-13 20:35:39 +02:00
|
|
|
}
|
|
|
|
bdb->bdb_difference_page = dbb->backup_manager->get_page_index(bdb->bdb_page);
|
|
|
|
dbb->backup_manager->unlock_alloc();
|
|
|
|
if (bdb->bdb_difference_page) {
|
|
|
|
NBAK_TRACE(("Map existing difference page %d to database page %d (write_both)",
|
|
|
|
bdb->bdb_difference_page, bdb->bdb_page));
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else {
|
2003-09-13 20:35:39 +02:00
|
|
|
// This may really happen. Database file can grow while in merge mode
|
|
|
|
write_direction = BDB_write_normal;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (!set_write_direction(dbb, bdb, write_direction)) {
|
|
|
|
dbb->backup_manager->unlock_state();
|
|
|
|
invalidate_and_release_buffer(tdbb, bdb);
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2003-09-13 20:35:39 +02:00
|
|
|
}
|
|
|
|
dbb->backup_manager->unlock_state();
|
|
|
|
}
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void CCH_release(thread_db* tdbb, WIN * window, bool release_tail)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ r e l e a s e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Release a window. If the release_tail
|
2004-02-20 07:43:27 +01:00
|
|
|
* flag is true then make the buffer
|
2001-05-23 15:26:42 +02:00
|
|
|
* least-recently-used.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = window->win_bdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(bdb, type_bdb);
|
|
|
|
|
2001-07-10 19:35:13 +02:00
|
|
|
/* if an expanded buffer has been created, retain it
|
2001-05-23 15:26:42 +02:00
|
|
|
for possible future use */
|
|
|
|
|
|
|
|
bdb->bdb_expanded_buffer = window->win_expanded_buffer;
|
|
|
|
window->win_expanded_buffer = NULL;
|
|
|
|
|
|
|
|
/* A large sequential scan has requested that the garbage
|
|
|
|
collector garbage collect. Mark the buffer so that the
|
|
|
|
page isn't released to the LRU tail before the garbage
|
|
|
|
collector can process the page. */
|
|
|
|
|
|
|
|
if (window->win_flags & WIN_large_scan &&
|
|
|
|
window->win_flags & WIN_garbage_collect)
|
|
|
|
{
|
|
|
|
bdb->bdb_flags |= BDB_garbage_collect;
|
|
|
|
window->win_flags &= ~WIN_garbage_collect;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (bdb->bdb_use_count == 1)
|
|
|
|
{
|
2003-09-08 22:23:46 +02:00
|
|
|
bool marked = bdb->bdb_flags & BDB_marked;
|
|
|
|
bdb->bdb_flags &= ~(BDB_writer | BDB_marked | BDB_faked);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (marked)
|
|
|
|
{
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, false, false, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
if (bdb->bdb_flags & BDB_must_write)
|
|
|
|
{
|
|
|
|
/* Downgrade exclusive latch to shared to allow concurrent share access
|
|
|
|
to page during I/O. */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, false, true, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!write_buffer( tdbb,
|
|
|
|
bdb,
|
|
|
|
bdb->bdb_page,
|
2003-12-22 11:00:59 +01:00
|
|
|
false,
|
2001-05-23 15:26:42 +02:00
|
|
|
tdbb->tdbb_status_vector,
|
2003-12-22 11:00:59 +01:00
|
|
|
true))
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
btc_insert(dbb, bdb); /* Don't lose track of must_write */
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#ifndef PAGE_LATCHING
|
|
|
|
if (bdb->bdb_flags & BDB_no_blocking_ast)
|
|
|
|
{
|
|
|
|
if (bdb->bdb_flags & (BDB_db_dirty | BDB_dirty))
|
|
|
|
{
|
|
|
|
if (!write_buffer( tdbb,
|
|
|
|
bdb,
|
|
|
|
bdb->bdb_page,
|
2003-12-22 11:00:59 +01:00
|
|
|
false,
|
2001-05-23 15:26:42 +02:00
|
|
|
tdbb->tdbb_status_vector,
|
2003-12-22 11:00:59 +01:00
|
|
|
true))
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/* Reassert blocking AST after write failure with dummy lock convert
|
|
|
|
to same level. This will re-enable blocking AST notification. */
|
|
|
|
|
|
|
|
LCK_convert_opt(tdbb, bdb->bdb_lock,
|
|
|
|
bdb->bdb_lock->lck_logical);
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
PAGE_LOCK_RELEASE(bdb->bdb_lock);
|
|
|
|
bdb->bdb_flags &= ~BDB_no_blocking_ast;
|
|
|
|
bdb->bdb_ast_flags &= ~BDB_blocking;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Make buffer the least-recently-used by queueing it
|
|
|
|
to the LRU tail. */
|
|
|
|
|
|
|
|
if (release_tail)
|
|
|
|
{
|
|
|
|
if ((window->win_flags & WIN_large_scan &&
|
|
|
|
bdb->bdb_scan_count > 0 &&
|
|
|
|
!(--bdb->bdb_scan_count) &&
|
|
|
|
!(bdb->bdb_flags & BDB_garbage_collect)) ||
|
|
|
|
(window->win_flags & WIN_garbage_collector &&
|
|
|
|
bdb->bdb_flags & BDB_garbage_collect &&
|
|
|
|
!(bdb->bdb_scan_count)))
|
|
|
|
{
|
|
|
|
if (window->win_flags & WIN_garbage_collector)
|
|
|
|
{
|
|
|
|
bdb->bdb_flags &= ~BDB_garbage_collect;
|
|
|
|
}
|
2004-02-20 07:43:27 +01:00
|
|
|
BCB bcb = dbb->dbb_bcb;
|
2001-05-23 15:26:42 +02:00
|
|
|
QUE_LEAST_RECENTLY_USED(bdb->bdb_in_use);
|
|
|
|
bdb->bdb_sequence = 0;
|
|
|
|
#ifdef CACHE_WRITER
|
|
|
|
if (bdb->bdb_flags & (BDB_dirty | BDB_db_dirty))
|
|
|
|
{
|
|
|
|
bcb->bcb_flags |= BCB_free_pending;
|
|
|
|
if (bcb->bcb_flags & BCB_cache_writer &&
|
|
|
|
!(bcb->bcb_flags & BCB_writer_active))
|
|
|
|
{
|
|
|
|
ISC_event_post(dbb->dbb_writer_event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, false, false, false);
|
|
|
|
const SSHORT use_count = bdb->bdb_use_count;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (use_count < 0) {
|
2001-05-23 15:26:42 +02:00
|
|
|
BUGCHECK(209); /* msg 209 attempt to release page not acquired */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!use_count && (bdb->bdb_ast_flags & BDB_blocking))
|
2003-08-13 13:11:12 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
PAGE_LOCK_RE_POST(bdb->bdb_lock);
|
2003-08-13 13:11:12 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
window->win_bdb = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void CCH_release_exclusive(thread_db* tdbb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ r e l e a s e _ e x c l u s i v e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Release exclusive access to database.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-02-25 02:50:40 +01:00
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
dbb->dbb_flags &= ~DBB_exclusive;
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
att* attachment = tdbb->tdbb_attachment;
|
|
|
|
if (attachment) {
|
2001-05-23 15:26:42 +02:00
|
|
|
attachment->att_flags &= ~ATT_exclusive;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (dbb->dbb_ast_flags & DBB_blocking) {
|
2001-05-23 15:26:42 +02:00
|
|
|
LCK_re_post(dbb->dbb_lock);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-07 08:58:55 +01:00
|
|
|
bool CCH_rollover_to_shadow(Database* dbb, jrd_file* file, const bool inAst)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ r o l l o v e r _ t o _ s h a d o w
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2001-07-10 19:35:13 +02:00
|
|
|
* An I/O error has been detected on the
|
2001-05-23 15:26:42 +02:00
|
|
|
* main database file. Roll over to use
|
|
|
|
* the shadow file.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
/* Is the shadow subsystem yet initialized */
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!dbb->dbb_shadow_lock) {
|
2003-12-22 11:00:59 +01:00
|
|
|
return false;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
/* notify other process immediately to ensure all read from sdw
|
|
|
|
file instead of db file */
|
2004-03-07 08:58:55 +01:00
|
|
|
return SDW_rollover_to_shadow(file, inAst);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-07 08:58:55 +01:00
|
|
|
void CCH_shutdown_database(Database* dbb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ s h u t d o w n _ d a t a b a s e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Shutdown database physical page locks.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb = GET_THREAD_DATA;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
bcb_repeat* tail;
|
|
|
|
BCB bcb = dbb->dbb_bcb;
|
|
|
|
if (bcb && (tail = bcb->bcb_rpt) && (tail->bcb_bdb)) {
|
|
|
|
for (const bcb_repeat* const end = tail + bcb->bcb_count;
|
|
|
|
tail < end; tail++)
|
|
|
|
{
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = tail->bcb_bdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_flags &= ~(BDB_dirty | BDB_db_dirty);
|
|
|
|
PAGE_LOCK_RELEASE(bdb->bdb_lock);
|
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifndef SUPERSERVER
|
|
|
|
PIO_close(dbb->dbb_file);
|
|
|
|
SDW_close();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
void CCH_unwind(thread_db* tdbb, bool punt)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ u n w i n d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Synchronously unwind cache after I/O or lock error.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* CCH_unwind is called when any of the following occurs:
|
|
|
|
- IO error
|
2004-01-06 11:33:18 +01:00
|
|
|
- journaling error => obsolete
|
2001-05-23 15:26:42 +02:00
|
|
|
- bad page checksum
|
|
|
|
- wrong page type
|
|
|
|
- page locking (not latching) deadlock (doesn't happen on Netware) */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
BCB bcb = dbb->dbb_bcb;
|
|
|
|
if (!bcb || (tdbb->tdbb_flags & TDBB_no_cache_unwind)) {
|
2001-12-24 03:51:06 +01:00
|
|
|
if (punt) {
|
2001-05-23 15:26:42 +02:00
|
|
|
ERR_punt();
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
return;
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* A cache error has occurred. Scan the cache for buffers
|
|
|
|
which may be in use and release them. */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
bcb_repeat* tail = bcb->bcb_rpt;
|
|
|
|
for (const bcb_repeat* const end = tail + bcb->bcb_count; tail < end; tail++)
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = tail->bcb_bdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifndef SUPERSERVER
|
2004-01-06 11:33:18 +01:00
|
|
|
// if (bdb->bdb_length || !bdb->bdb_use_count) becomes (false || expr)
|
|
|
|
if (!bdb->bdb_use_count) {
|
2001-05-23 15:26:42 +02:00
|
|
|
continue;
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
|
|
|
if (bdb->bdb_flags & BDB_marked) {
|
2001-05-23 15:26:42 +02:00
|
|
|
cache_bugcheck(268); /* msg 268 buffer marked during cache unwind */
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_flags &= ~BDB_writer;
|
2001-12-24 03:51:06 +01:00
|
|
|
while (bdb->bdb_use_count) {
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, true, false, false);
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
|
|
|
PAG page = bdb->bdb_buffer;
|
2001-05-23 15:26:42 +02:00
|
|
|
if ((page->pag_type == pag_header) ||
|
2001-12-24 03:51:06 +01:00
|
|
|
(page->pag_type == pag_transactions))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
++bdb->bdb_use_count;
|
2003-09-08 22:23:46 +02:00
|
|
|
// Adjust backup page locks
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_flags & BDB_dirty) {
|
2003-09-08 22:23:46 +02:00
|
|
|
set_write_direction(dbb, bdb, BDB_write_undefined);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2003-09-08 22:23:46 +02:00
|
|
|
bdb->bdb_flags &= ~(BDB_dirty |
|
|
|
|
BDB_writer | BDB_marked | BDB_faked | BDB_db_dirty);
|
2001-05-23 15:26:42 +02:00
|
|
|
PAGE_LOCK_RELEASE(bdb->bdb_lock);
|
|
|
|
--bdb->bdb_use_count;
|
|
|
|
}
|
|
|
|
#else
|
2001-12-24 03:51:06 +01:00
|
|
|
if (!bdb->bdb_use_count) {
|
2001-05-23 15:26:42 +02:00
|
|
|
continue;
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
|
|
|
if (bdb->bdb_io == tdbb) {
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, true, false, false);
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
if (bdb->bdb_exclusive == tdbb) {
|
|
|
|
/* The sanity check below can't be enforced for the log page because
|
|
|
|
the ail.c code does IO while having the log page marked (should be
|
|
|
|
fixed in the future). */
|
2004-01-03 11:59:52 +01:00
|
|
|
if ((bdb->bdb_flags & BDB_marked) && (bdb->bdb_page != LOG_PAGE)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
cache_bugcheck(268); /* msg 268 buffer marked during cache unwind */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_flags &= ~(BDB_writer | BDB_faked | BDB_must_write);
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, true, false, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2001-12-24 03:51:06 +01:00
|
|
|
for (int i = 0; i < BDB_max_shared; ++i) {
|
|
|
|
if (bdb->bdb_shared[i] == tdbb) {
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, true, false, false);
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (punt) {
|
2001-05-23 15:26:42 +02:00
|
|
|
ERR_punt();
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
bool CCH_validate(WIN * window)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ v a l i d a t e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Give a page a quick once over looking for unhealthyness.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = window->win_bdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If page is marked for write, checksum is questionable */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if ((bdb->bdb_flags & (BDB_dirty | BDB_db_dirty))) {
|
2004-02-20 07:43:27 +01:00
|
|
|
return true;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
pag* page = window->win_buffer;
|
|
|
|
const USHORT sum = CCH_checksum(bdb);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (sum == page->pag_checksum) {
|
2004-02-20 07:43:27 +01:00
|
|
|
return true;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
return false;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
bool CCH_write_all_shadows(thread_db* tdbb,
|
2004-02-20 07:43:27 +01:00
|
|
|
Shadow* shadow,
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb,
|
2003-12-22 11:00:59 +01:00
|
|
|
ISC_STATUS* status, USHORT checksum,
|
|
|
|
const bool inAst)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* C C H _ w r i t e _ a l l _ s h a d o w s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Compute a checksum and write a page out to all shadows
|
|
|
|
* detecting failure on write.
|
|
|
|
* If shadow is null, write to all shadows, otherwise only to specified
|
|
|
|
* shadow.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
Shadow* sdw = shadow ? shadow : dbb->dbb_shadow;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
if (!sdw) {
|
2003-12-22 11:00:59 +01:00
|
|
|
return true;
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
bool result = true;
|
2004-01-03 11:59:52 +01:00
|
|
|
SLONG* spare_buffer = NULL;
|
2003-12-22 11:00:59 +01:00
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
try {
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
pag* page;
|
|
|
|
pag* old_buffer = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (bdb->bdb_page == HEADER_PAGE) {
|
|
|
|
/* allocate a spare buffer which is large enough,
|
|
|
|
and set up to release it in case of error */
|
|
|
|
|
|
|
|
spare_buffer =
|
2002-09-25 19:12:16 +02:00
|
|
|
(SLONG *) dbb->dbb_bufferpool->allocate((SLONG) dbb->dbb_page_size,0
|
|
|
|
#ifdef DEBUG_GDS_ALLOC
|
|
|
|
,__FILE__,__LINE__
|
|
|
|
#endif
|
|
|
|
);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
page = (PAG) spare_buffer;
|
2004-03-11 06:04:26 +01:00
|
|
|
MOVE_FAST((const UCHAR*) bdb->bdb_buffer, (UCHAR*) page, HDR_SIZE);
|
2001-05-23 15:26:42 +02:00
|
|
|
old_buffer = bdb->bdb_buffer;
|
|
|
|
bdb->bdb_buffer = page;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
page = bdb->bdb_buffer;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (checksum) {
|
2001-05-23 15:26:42 +02:00
|
|
|
page->pag_checksum = CCH_checksum(bdb);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for (; sdw; sdw = sdw->sdw_next) {
|
|
|
|
/* don't bother to write to the shadow if it is no longer viable */
|
|
|
|
|
|
|
|
/* Fix for bug 7925. drop_gdb fails to remove secondary file if
|
|
|
|
the shadow is conditional. Reason being the header page not
|
2001-07-10 19:35:13 +02:00
|
|
|
being correctly initialized.
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
The following block was not being performed for a conditional
|
|
|
|
shadow since SDW_INVALID expanded to include conditional shadow
|
|
|
|
|
2001-07-10 19:35:13 +02:00
|
|
|
-Sudesh 07/10/95
|
2001-05-23 15:26:42 +02:00
|
|
|
old code --> if (sdw->sdw_flags & SDW_INVALID)
|
|
|
|
*/
|
|
|
|
|
|
|
|
if ((sdw->sdw_flags & SDW_INVALID) &&
|
2004-01-03 11:59:52 +01:00
|
|
|
!(sdw->sdw_flags & SDW_conditional))
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (bdb->bdb_page == HEADER_PAGE) {
|
|
|
|
/* fixup header for shadow file */
|
2004-02-20 07:43:27 +01:00
|
|
|
jrd_file* shadow_file = sdw->sdw_file;
|
|
|
|
header_page* header = (header_page*) page;
|
2004-01-03 11:59:52 +01:00
|
|
|
const UCHAR* q = (UCHAR *) dbb->dbb_file->fil_string;
|
2001-05-23 15:26:42 +02:00
|
|
|
header->hdr_data[0] = HDR_end;
|
|
|
|
header->hdr_end = HDR_SIZE;
|
|
|
|
header->hdr_next_page = 0;
|
|
|
|
|
|
|
|
PAG_add_header_entry(header, HDR_root_file_name,
|
2004-01-03 11:59:52 +01:00
|
|
|
(USHORT) strlen((const char*) q), q);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
jrd_file* next_file = shadow_file->fil_next;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (next_file) {
|
2001-05-23 15:26:42 +02:00
|
|
|
q = (UCHAR *) next_file->fil_string;
|
2004-01-03 11:59:52 +01:00
|
|
|
const SLONG last = next_file->fil_min_page - 1;
|
2001-05-23 15:26:42 +02:00
|
|
|
PAG_add_header_entry(header, HDR_file,
|
2004-01-03 11:59:52 +01:00
|
|
|
(USHORT) strlen((const char*) q), q);
|
2001-05-23 15:26:42 +02:00
|
|
|
PAG_add_header_entry(header, HDR_last_page, sizeof(last),
|
2004-01-03 11:59:52 +01:00
|
|
|
(const UCHAR*) & last);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
header->hdr_flags |= hdr_active_shadow;
|
|
|
|
header->hdr_header.pag_checksum = CCH_checksum(bdb);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* This condition makes sure that PIO_write is performed in case of
|
|
|
|
conditional shadow only if the page is Header page
|
|
|
|
|
|
|
|
-Sudesh 07/10/95
|
|
|
|
*/
|
|
|
|
if ((sdw->sdw_flags & SDW_conditional) &&
|
2004-01-03 11:59:52 +01:00
|
|
|
(bdb->bdb_page != HEADER_PAGE))
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-07-10 19:35:13 +02:00
|
|
|
/* if a write failure happens on an AUTO shadow, mark the
|
|
|
|
shadow to be deleted at the next available opportunity when we
|
2001-05-23 15:26:42 +02:00
|
|
|
know we don't have a page fetched */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!PIO_write(sdw->sdw_file, bdb, page, status)) {
|
|
|
|
if (sdw->sdw_flags & SDW_manual) {
|
2003-12-22 11:00:59 +01:00
|
|
|
result = false;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else {
|
|
|
|
sdw->sdw_flags |= SDW_delete;
|
|
|
|
if (!inAst && SDW_check_conditional()) {
|
|
|
|
if (SDW_lck_update((SLONG) 0)) {
|
|
|
|
SDW_notify();
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
SDW_dump_pages();
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_deadlock, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
/* If shadow was specified, break out of loop */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (shadow) {
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_page == HEADER_PAGE) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_buffer = old_buffer;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (spare_buffer) {
|
2003-01-16 18:47:10 +01:00
|
|
|
dbb->dbb_bufferpool->deallocate(spare_buffer);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-12-24 03:51:06 +01:00
|
|
|
|
|
|
|
} // try
|
2004-03-01 04:35:23 +01:00
|
|
|
catch (const std::exception& ex) {
|
|
|
|
Firebird::stuff_exception(tdbb->tdbb_status_vector, ex);
|
2001-12-24 03:51:06 +01:00
|
|
|
if (spare_buffer) {
|
2003-01-16 18:47:10 +01:00
|
|
|
dbb->dbb_bufferpool->deallocate(spare_buffer);
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
|
|
|
ERR_punt();
|
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static Buffer_desc* alloc_bdb(thread_db* tdbb, BCB bcb, UCHAR** memory)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* a l l o c _ b d b
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Allocate buffer descriptor block.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb_ = FB_NEW(*dbb->dbb_bufferpool) Buffer_desc;
|
2001-12-24 03:51:06 +01:00
|
|
|
bdb_->bdb_dbb = dbb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifndef PAGE_LATCHING
|
2004-01-03 11:59:52 +01:00
|
|
|
lck* lock;
|
2001-12-24 03:51:06 +01:00
|
|
|
try {
|
|
|
|
bdb_->bdb_lock = lock = CCH_page_lock(tdbb);
|
|
|
|
}
|
2003-02-13 14:33:57 +01:00
|
|
|
catch (const std::exception&) {
|
2001-12-24 03:51:06 +01:00
|
|
|
delete bdb_;
|
|
|
|
throw;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2003-05-16 22:35:19 +02:00
|
|
|
lock->lck_ast = blocking_ast_bdb;
|
2004-02-20 07:43:27 +01:00
|
|
|
lock->lck_object = bdb_;
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
bdb_->bdb_buffer = (pag*) *memory;
|
2001-05-23 15:26:42 +02:00
|
|
|
*memory += dbb->dbb_page_size;
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
QUE_INIT(bdb_->bdb_higher);
|
|
|
|
QUE_INIT(bdb_->bdb_lower);
|
|
|
|
QUE_INIT(bdb_->bdb_waiters);
|
|
|
|
QUE_INSERT(bcb->bcb_empty, bdb_->bdb_que);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
return bdb_;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef PAGE_LATCHING
|
2003-05-16 22:35:19 +02:00
|
|
|
static int blocking_ast_bdb(void* ast_object)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* b l o c k i n g _ a s t _ b d b
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Blocking AST for buffer control blocks. This is called at
|
|
|
|
* AST (signal) level to indicate that a lock is blocking another
|
2004-03-11 06:04:26 +01:00
|
|
|
* process. If the Buffer_desc* is in use, set a flag indicating that the
|
|
|
|
* lock is blocking and return. When the Buffer_desc is released at normal
|
|
|
|
* level the lock will be down graded. If the Buffer_desc* is not in use,
|
2001-05-23 15:26:42 +02:00
|
|
|
* write the page if dirty then downgrade lock. Things would be
|
|
|
|
* much hairier if UNIX supported asynchronous IO, but it doesn't.
|
|
|
|
* WHEW!
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = static_cast<Buffer_desc*>(ast_object);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
ISC_ast_enter();
|
|
|
|
|
|
|
|
/* Since this routine will be called asynchronously, we must establish
|
|
|
|
a thread context. */
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db thd_context, *tdbb;
|
2001-05-23 15:26:42 +02:00
|
|
|
SET_THREAD_DATA;
|
|
|
|
|
|
|
|
BLKCHK(bdb, type_bdb);
|
2004-01-03 11:59:52 +01:00
|
|
|
|
|
|
|
ISC_STATUS_ARRAY ast_status;
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = bdb->bdb_dbb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
tdbb->tdbb_database = dbb;
|
|
|
|
tdbb->tdbb_attachment = NULL;
|
|
|
|
tdbb->tdbb_quantum = QUANTUM;
|
|
|
|
tdbb->tdbb_request = NULL;
|
|
|
|
tdbb->tdbb_transaction = NULL;
|
|
|
|
tdbb->tdbb_status_vector = ast_status;
|
|
|
|
|
|
|
|
/* Do some fancy footwork to make sure that pages are
|
|
|
|
not removed from the btc tree at AST level. Then
|
|
|
|
restore the flag to whatever it was before. */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
const bool keep_pages = (dbb->dbb_bcb->bcb_flags & BCB_keep_pages) != 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
dbb->dbb_bcb->bcb_flags |= BCB_keep_pages;
|
|
|
|
ast_status[1] = 0;
|
|
|
|
|
|
|
|
down_grade(tdbb, bdb);
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!keep_pages) {
|
2001-05-23 15:26:42 +02:00
|
|
|
dbb->dbb_bcb->bcb_flags &= ~BCB_keep_pages;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (ast_status[1]) {
|
2001-05-23 15:26:42 +02:00
|
|
|
gds__log_status(dbb->dbb_file->fil_string, ast_status);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Restore the prior thread context */
|
|
|
|
|
|
|
|
RESTORE_THREAD_DATA;
|
|
|
|
|
|
|
|
ISC_ast_exit();
|
2001-12-24 03:51:06 +01:00
|
|
|
|
2003-05-16 22:35:19 +02:00
|
|
|
return 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
static void btc_flush(
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb,
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG transaction_mask,
|
2003-12-22 11:00:59 +01:00
|
|
|
const bool sys_only, ISC_STATUS* status)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* b t c _ f l u s h
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Walk the dirty page binary tree, flushing all buffers
|
2001-07-10 19:35:13 +02:00
|
|
|
* that could have been modified by this transaction.
|
|
|
|
* The pages are flushed in page order to roughly
|
|
|
|
* emulate an elevator-type disk controller. Iteration
|
2001-05-23 15:26:42 +02:00
|
|
|
* is used to minimize call overhead.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* traverse the tree, flagging to prevent pages
|
|
|
|
from being removed from the tree during write_page() --
|
|
|
|
this simplifies worrying about random pages dropping
|
|
|
|
out when dependencies have been set up */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
SLONG max_seen = MIN_PAGE_NUMBER;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Pick starting place at leftmost node */
|
|
|
|
|
|
|
|
BTC_MUTEX_ACQUIRE;
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* next = dbb->dbb_bcb->bcb_btree;
|
2004-01-03 11:59:52 +01:00
|
|
|
while (next && next->bdb_left) {
|
|
|
|
next = next->bdb_left;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
SLONG next_page = 0;
|
|
|
|
if (next) {
|
2001-05-23 15:26:42 +02:00
|
|
|
next_page = next->bdb_page;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Walk tree. If we get lost, reposition and continue */
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb;
|
2001-12-24 03:51:06 +01:00
|
|
|
while ( (bdb = next) ) {
|
2001-05-23 15:26:42 +02:00
|
|
|
/* If we're lost, reposition */
|
|
|
|
|
|
|
|
if ((bdb->bdb_page != next_page) ||
|
2004-01-03 11:59:52 +01:00
|
|
|
(!bdb->bdb_parent && (bdb != dbb->dbb_bcb->bcb_btree)))
|
|
|
|
{
|
|
|
|
for (bdb = dbb->dbb_bcb->bcb_btree; bdb;) {
|
|
|
|
if (bdb->bdb_left && (max_seen < bdb->bdb_page)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb = bdb->bdb_left;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else if (bdb->bdb_right && (max_seen > bdb->bdb_page)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb = bdb->bdb_right;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!bdb) {
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Decide where to go next. The options are (right, then down to the left)
|
|
|
|
or up */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_right && (max_seen < bdb->bdb_right->bdb_page)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
for (next = bdb->bdb_right; next->bdb_left;
|
|
|
|
next = next->bdb_left);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
next = bdb->bdb_parent;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (next) {
|
2001-05-23 15:26:42 +02:00
|
|
|
next_page = next->bdb_page;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (max_seen >= bdb->bdb_page) {
|
2001-05-23 15:26:42 +02:00
|
|
|
continue;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
max_seen = bdb->bdb_page;
|
|
|
|
|
|
|
|
/* forget about this page if it was written out
|
|
|
|
as a dependency while we were walking the tree */
|
|
|
|
|
|
|
|
if (!(bdb->bdb_flags & BDB_dirty)) {
|
|
|
|
BTC_MUTEX_RELEASE;
|
|
|
|
btc_remove(bdb);
|
|
|
|
BTC_MUTEX_ACQUIRE;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* this code replicates code in CCH_flush() --
|
|
|
|
changes should be made in both places */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
const SLONG page = bdb->bdb_page;
|
2001-05-23 15:26:42 +02:00
|
|
|
BTC_MUTEX_RELEASE;
|
|
|
|
|
|
|
|
#ifndef SUPERSERVER
|
|
|
|
if (bdb->bdb_use_count)
|
|
|
|
cache_bugcheck(210); /* msg 210 page in use during flush */
|
|
|
|
#endif
|
|
|
|
/* if any transaction has dirtied this page,
|
|
|
|
check to see if it could have been this one */
|
|
|
|
|
|
|
|
if ((transaction_mask & bdb->bdb_transactions) ||
|
|
|
|
(bdb->bdb_flags & BDB_system_dirty) ||
|
2003-12-22 11:00:59 +01:00
|
|
|
(!transaction_mask && !sys_only) || (!bdb->bdb_transactions))
|
|
|
|
{
|
|
|
|
if (!write_buffer(tdbb, bdb, page, false, status, true)) {
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* re-post the lock only if it was really written */
|
|
|
|
|
|
|
|
if ((bdb->bdb_ast_flags & BDB_blocking) &&
|
2003-12-22 11:00:59 +01:00
|
|
|
!(bdb->bdb_flags & BDB_dirty))
|
|
|
|
{
|
2003-08-13 13:11:12 +02:00
|
|
|
PAGE_LOCK_RE_POST(bdb->bdb_lock);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
BTC_MUTEX_ACQUIRE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BTC_MUTEX_RELEASE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static void btc_insert(Database* dbb, Buffer_desc* bdb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* b t c _ i n s e r t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2001-07-10 19:35:13 +02:00
|
|
|
* Insert a buffer into the dirty page
|
2001-05-23 15:26:42 +02:00
|
|
|
* binary tree.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
/* if the page is already in the tree (as in when it is
|
|
|
|
written out as a dependency while walking the tree),
|
2001-07-10 19:35:13 +02:00
|
|
|
just leave well enough alone -- this won't check if
|
2001-05-23 15:26:42 +02:00
|
|
|
it's at the root but who cares then */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_parent) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_DBB(dbb);
|
|
|
|
|
|
|
|
/* if the tree is empty, this is now the tree */
|
|
|
|
|
|
|
|
BTC_MUTEX_ACQUIRE;
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* node = dbb->dbb_bcb->bcb_btree;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!node) {
|
2001-05-23 15:26:42 +02:00
|
|
|
dbb->dbb_bcb->bcb_btree = bdb;
|
|
|
|
BTC_MUTEX_RELEASE;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* insert the page sorted by page number;
|
|
|
|
do this iteratively to minimize call overhead */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
const SLONG page = bdb->bdb_page;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-11 11:33:30 +01:00
|
|
|
while (true) {
|
2004-01-03 11:59:52 +01:00
|
|
|
if (page == node->bdb_page) {
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (page < node->bdb_page) {
|
|
|
|
if (!node->bdb_left) {
|
|
|
|
node->bdb_left = bdb;
|
|
|
|
bdb->bdb_parent = node;
|
|
|
|
break;
|
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
node = node->bdb_left;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (!node->bdb_right) {
|
|
|
|
node->bdb_right = bdb;
|
|
|
|
bdb->bdb_parent = node;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
node = node->bdb_right;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
BTC_MUTEX_RELEASE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static void btc_remove(Buffer_desc* bdb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* b t c _ r e m o v e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Remove a page from the dirty page binary tree.
|
2001-07-10 19:35:13 +02:00
|
|
|
* The idea is to place the left child of this
|
|
|
|
* page in this page's place, then make the
|
|
|
|
* right child of this page a child of the left
|
|
|
|
* child -- this removal mechanism won't promote
|
2001-05-23 15:26:42 +02:00
|
|
|
* a balanced tree but that isn't of primary
|
|
|
|
* importance.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = bdb->bdb_dbb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* engage in a little defensive programming to make
|
|
|
|
sure the node is actually in the tree */
|
|
|
|
|
|
|
|
BTC_MUTEX_ACQUIRE;
|
2004-01-03 11:59:52 +01:00
|
|
|
BCB bcb = dbb->dbb_bcb;
|
2001-05-23 15:26:42 +02:00
|
|
|
if ((!bcb->bcb_btree) ||
|
|
|
|
(!bdb->bdb_parent &&
|
2004-01-03 11:59:52 +01:00
|
|
|
!bdb->bdb_left && !bdb->bdb_right && (bcb->bcb_btree != bdb)))
|
|
|
|
{
|
|
|
|
if ((bdb->bdb_flags & BDB_must_write) || !(bdb->bdb_flags & BDB_dirty))
|
|
|
|
{
|
|
|
|
/* Must writes aren't worth the effort */
|
2001-05-23 15:26:42 +02:00
|
|
|
BTC_MUTEX_RELEASE;
|
|
|
|
return;
|
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
else {
|
|
|
|
cache_bugcheck(211);
|
|
|
|
/* msg 211 attempt to remove page from dirty page list when not there */
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* make a new child out of the left and right children */
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* new_child = bdb->bdb_left;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (new_child) {
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* ptr = new_child;
|
2004-01-03 11:59:52 +01:00
|
|
|
while (ptr->bdb_right) {
|
|
|
|
ptr = ptr->bdb_right;
|
|
|
|
}
|
|
|
|
if ( (ptr->bdb_right = bdb->bdb_right) ) {
|
2001-05-23 15:26:42 +02:00
|
|
|
ptr->bdb_right->bdb_parent = ptr;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
new_child = bdb->bdb_right;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-07-10 19:35:13 +02:00
|
|
|
/* link the parent with the child node --
|
2001-05-23 15:26:42 +02:00
|
|
|
if no parent place it at the root */
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb_parent = bdb->bdb_parent;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!bdb_parent) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bcb->bcb_btree = new_child;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else if (bdb_parent->bdb_left == bdb) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb_parent->bdb_left = new_child;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb_parent->bdb_right = new_child;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (new_child) {
|
2001-05-23 15:26:42 +02:00
|
|
|
new_child->bdb_parent = bdb_parent;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* initialize the node for next usage */
|
|
|
|
|
2003-08-28 15:16:03 +02:00
|
|
|
bdb->bdb_left = bdb->bdb_right = bdb->bdb_parent = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
BTC_MUTEX_RELEASE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void cache_bugcheck(int number)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c a c h e _ b u g c h e c k
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* There has been a bugcheck during a cache operation. Release
|
|
|
|
* the cache mutex and post the bugcheck.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
BUGCHECK(number);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef CACHE_READER
|
2004-03-07 08:58:55 +01:00
|
|
|
static void THREAD_ROUTINE cache_reader(Database* dbb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c a c h e _ r e a d e r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Prefetch pages into cache for sequential scans.
|
|
|
|
* Use asynchronous I/O to keep two prefetch requests
|
|
|
|
* busy at a time.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
THREAD_ENTER;
|
|
|
|
|
|
|
|
/* Establish a thread context. */
|
|
|
|
/* Note: Since this function operates as its own thread,
|
|
|
|
we have no need to restore the THREAD CONTEXT on exit.
|
|
|
|
Once we reach the end, the thread will die, thus implicitly
|
|
|
|
killing all its contexts. */
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db thd_context, *tdbb;
|
2001-05-23 15:26:42 +02:00
|
|
|
SET_THREAD_DATA;
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Dummy attachment needed for lock owner identification. */
|
|
|
|
tdbb->tdbb_database = dbb;
|
|
|
|
tdbb->tdbb_default = dbb->dbb_bufferpool;
|
|
|
|
tdbb->tdbb_status_vector = status_vector;
|
|
|
|
tdbb->tdbb_quantum = QUANTUM;
|
2002-09-25 19:12:16 +02:00
|
|
|
tdbb->tdbb_attachment = FB_NEW(*dbb->dbb_bufferpool) att();
|
2001-05-23 15:26:42 +02:00
|
|
|
tdbb->tdbb_attachment->att_database = dbb;
|
2003-01-18 17:44:01 +01:00
|
|
|
tdbb->tdbb_attachment->att_filename = dbb->dbb_filename;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-04-04 09:10:40 +02:00
|
|
|
/* This try block is specifically to protect the LCK_init call: if
|
2001-05-23 15:26:42 +02:00
|
|
|
LCK_init fails we won't be able to accomplish anything anyway, so
|
2002-04-04 09:10:40 +02:00
|
|
|
return, unlike the other try blocks further down the page. */
|
2004-01-03 11:59:52 +01:00
|
|
|
|
2004-02-02 12:02:12 +01:00
|
|
|
event_t* reader_event = 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
BCB bcb = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
try {
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
LCK_init(tdbb, LCK_OWNER_attachment);
|
|
|
|
reader_event = dbb->dbb_reader_event;
|
2001-05-23 15:26:42 +02:00
|
|
|
bcb = dbb->dbb_bcb;
|
2001-12-24 03:51:06 +01:00
|
|
|
bcb->bcb_flags |= BCB_cache_reader;
|
|
|
|
ISC_event_post(reader_event); /* Notify our creator that we have started */
|
|
|
|
}
|
2004-03-01 04:35:23 +01:00
|
|
|
catch (const std::exception& ex) {
|
|
|
|
Firebird::stuff_exception(status_vector, ex);
|
2001-05-23 15:26:42 +02:00
|
|
|
gds__log_status(dbb->dbb_file->fil_string, status_vector);
|
2001-12-24 03:51:06 +01:00
|
|
|
THREAD_EXIT;
|
|
|
|
return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
try {
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Set up dual prefetch control blocks to keep multiple prefetch
|
|
|
|
requests active at a time. Also helps to prevent the cache reader
|
2004-01-03 11:59:52 +01:00
|
|
|
from becoming dedicated or locked into a single request's prefetch
|
2001-05-23 15:26:42 +02:00
|
|
|
demands. */
|
2004-01-03 11:59:52 +01:00
|
|
|
prf prefetch1, prefetch2;
|
2001-05-23 15:26:42 +02:00
|
|
|
prefetch_init(&prefetch1, tdbb);
|
|
|
|
prefetch_init(&prefetch2, tdbb);
|
|
|
|
|
|
|
|
while (bcb->bcb_flags & BCB_cache_reader) {
|
2004-01-03 11:59:52 +01:00
|
|
|
SLONG count = ISC_event_clear(reader_event);
|
2001-05-23 15:26:42 +02:00
|
|
|
bcb->bcb_flags |= BCB_reader_active;
|
2004-01-03 11:59:52 +01:00
|
|
|
bool found = false;
|
|
|
|
SLONG starting_page = -1;
|
|
|
|
prf* next_prefetch = &prefetch1;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (dbb->dbb_flags & DBB_suspend_bgio) {
|
|
|
|
THREAD_EXIT;
|
2003-09-01 09:58:04 +02:00
|
|
|
ISC_event_wait(1, &reader_event, &count, 10 * 1000000, NULL, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
THREAD_ENTER;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Make a pass thru the global prefetch bitmap looking for something
|
|
|
|
to read. When the entire bitmap has been scanned and found to be
|
|
|
|
empty then wait for new requests. */
|
|
|
|
|
|
|
|
do {
|
|
|
|
if (!(prefetch1.prf_flags & PRF_active) &&
|
2003-12-31 06:36:12 +01:00
|
|
|
SBM_next(bcb->bcb_prefetch, &starting_page, RSE_get_forward))
|
|
|
|
{
|
2004-01-03 11:59:52 +01:00
|
|
|
found = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
prefetch_prologue(&prefetch1, &starting_page);
|
|
|
|
prefetch_io(&prefetch1, status_vector);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(prefetch2.prf_flags & PRF_active) &&
|
2003-12-31 06:36:12 +01:00
|
|
|
SBM_next(bcb->bcb_prefetch, &starting_page, RSE_get_forward))
|
|
|
|
{
|
2004-01-03 11:59:52 +01:00
|
|
|
found = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
prefetch_prologue(&prefetch2, &starting_page);
|
|
|
|
prefetch_io(&prefetch2, status_vector);
|
|
|
|
}
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
prf* post_prefetch = next_prefetch;
|
2001-05-23 15:26:42 +02:00
|
|
|
next_prefetch =
|
|
|
|
(post_prefetch == &prefetch1) ? &prefetch2 : &prefetch1;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (post_prefetch->prf_flags & PRF_active) {
|
2001-05-23 15:26:42 +02:00
|
|
|
prefetch_epilogue(post_prefetch, status_vector);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (found) {
|
|
|
|
/* If the cache writer or garbage collector is idle, put
|
|
|
|
them to work prefetching pages. */
|
|
|
|
#ifdef CACHE_WRITER
|
|
|
|
if (bcb->bcb_flags & BCB_cache_writer &&
|
|
|
|
!(bcb->bcb_flags & BCB_writer_active))
|
2003-12-31 06:36:12 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
ISC_event_post(dbb->dbb_writer_event);
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
#ifdef GARBAGE_THREAD
|
|
|
|
if (dbb->dbb_flags & DBB_garbage_collector &&
|
|
|
|
!(dbb->dbb_flags & DBB_gc_active))
|
2003-12-31 06:36:12 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
ISC_event_post(dbb->dbb_gc_event);
|
2003-12-31 06:36:12 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
} while (prefetch1.prf_flags & PRF_active
|
|
|
|
|| prefetch2.prf_flags & PRF_active);
|
|
|
|
|
|
|
|
/* If there's more work to do voluntarily ask to be rescheduled.
|
|
|
|
Otherwise, wait for event notification. */
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (found) {
|
2003-12-22 11:00:59 +01:00
|
|
|
JRD_reschedule(tdbb, 0, true);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else if (bcb->bcb_flags & BCB_free_pending &&
|
2003-12-22 11:00:59 +01:00
|
|
|
(bdb = get_buffer(tdbb, FREE_PAGE, LATCH_none, 1)))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
/* In our spare time, help writer clean the cache. */
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
write_buffer(tdbb, bdb, bdb->bdb_page, true, status_vector, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
bcb->bcb_flags &= ~BCB_reader_active;
|
|
|
|
THREAD_EXIT;
|
2003-09-01 09:58:04 +02:00
|
|
|
ISC_event_wait(1, &reader_event, &count, 10 * 1000000, NULL, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
THREAD_ENTER;
|
|
|
|
}
|
|
|
|
bcb = dbb->dbb_bcb;
|
|
|
|
}
|
|
|
|
|
|
|
|
LCK_fini(tdbb, LCK_OWNER_attachment);
|
2001-12-24 03:51:06 +01:00
|
|
|
delete tdbb->tdbb_attachment;
|
2001-05-23 15:26:42 +02:00
|
|
|
bcb->bcb_flags &= ~BCB_cache_reader;
|
|
|
|
ISC_event_post(reader_event);
|
|
|
|
THREAD_EXIT;
|
2001-12-24 03:51:06 +01:00
|
|
|
|
|
|
|
} // try
|
2004-03-01 04:35:23 +01:00
|
|
|
catch (const std::exception& ex) {
|
|
|
|
Firebird::stuff_exception(status_vector, ex);
|
2001-12-24 03:51:06 +01:00
|
|
|
bcb = dbb->dbb_bcb;
|
|
|
|
gds__log_status(dbb->dbb_file->fil_string, status_vector);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef CACHE_WRITER
|
2004-03-07 08:58:55 +01:00
|
|
|
static void THREAD_ROUTINE cache_writer(Database* dbb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c a c h e _ w r i t e r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Write dirty pages to database to maintain an
|
|
|
|
* adequate supply of free pages. If WAL is enabled,
|
|
|
|
* perform database checkpoint when WAL subsystem
|
|
|
|
* deems it necessary.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
THREAD_ENTER;
|
|
|
|
|
|
|
|
/* Establish a thread context. */
|
|
|
|
/* Note: Since this function operates as its own thread,
|
|
|
|
we have no need to restore the THREAD CONTEXT on exit.
|
|
|
|
Once we reach the end, the thread will die, thus implicitly
|
|
|
|
killing all its contexts. */
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db thd_context, *tdbb;
|
2001-05-23 15:26:42 +02:00
|
|
|
SET_THREAD_DATA;
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Dummy attachment needed for lock owner identification. */
|
|
|
|
|
|
|
|
tdbb->tdbb_database = dbb;
|
|
|
|
tdbb->tdbb_default = dbb->dbb_bufferpool;
|
|
|
|
tdbb->tdbb_status_vector = status_vector;
|
|
|
|
tdbb->tdbb_quantum = QUANTUM;
|
2004-03-14 14:40:14 +01:00
|
|
|
tdbb->tdbb_attachment = FB_NEW(*dbb->dbb_bufferpool) att(dbb);
|
2003-01-18 17:44:01 +01:00
|
|
|
tdbb->tdbb_attachment->att_filename = dbb->dbb_filename;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-04-04 09:10:40 +02:00
|
|
|
/* This try block is specifically to protect the LCK_init call: if
|
2001-05-23 15:26:42 +02:00
|
|
|
LCK_init fails we won't be able to accomplish anything anyway, so
|
2002-04-04 09:10:40 +02:00
|
|
|
return, unlike the other try blocks further down the page. */
|
2004-02-02 12:02:12 +01:00
|
|
|
event_t* writer_event = 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
BCB bcb = 0;
|
2001-12-24 03:51:06 +01:00
|
|
|
|
|
|
|
try {
|
|
|
|
writer_event = dbb->dbb_writer_event;
|
2003-02-12 23:57:57 +01:00
|
|
|
ISC_event_init(writer_event, 0, 0);
|
|
|
|
LCK_init(tdbb, LCK_OWNER_attachment);
|
2001-12-24 03:51:06 +01:00
|
|
|
bcb = dbb->dbb_bcb;
|
|
|
|
bcb->bcb_flags |= BCB_cache_writer;
|
2003-02-12 23:57:57 +01:00
|
|
|
|
|
|
|
/* Notify our creator that we have started */
|
|
|
|
ISC_event_post(dbb->dbb_writer_event_init);
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2004-03-01 04:35:23 +01:00
|
|
|
catch (const std::exception& ex) {
|
|
|
|
Firebird::stuff_exception(status_vector, ex);
|
2001-05-23 15:26:42 +02:00
|
|
|
gds__log_status(dbb->dbb_file->fil_string, status_vector);
|
2003-02-12 23:57:57 +01:00
|
|
|
ISC_event_fini(writer_event);
|
2001-05-23 15:26:42 +02:00
|
|
|
THREAD_EXIT;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
try {
|
2003-02-12 23:57:57 +01:00
|
|
|
while (bcb->bcb_flags & BCB_cache_writer)
|
|
|
|
{
|
2004-01-03 11:59:52 +01:00
|
|
|
SLONG count = ISC_event_clear(writer_event);
|
2003-02-12 23:57:57 +01:00
|
|
|
bcb->bcb_flags |= BCB_writer_active;
|
2004-01-03 11:59:52 +01:00
|
|
|
SLONG starting_page = -1;
|
2003-02-12 23:57:57 +01:00
|
|
|
|
|
|
|
if (dbb->dbb_flags & DBB_suspend_bgio) {
|
|
|
|
THREAD_EXIT;
|
2003-09-01 09:58:04 +02:00
|
|
|
ISC_event_wait(1, &writer_event, &count, 10 * 1000000, NULL, 0);
|
2003-02-12 23:57:57 +01:00
|
|
|
THREAD_ENTER;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef SUPERSERVER_V2
|
2003-02-12 23:57:57 +01:00
|
|
|
/* Flush buffers for lazy commit */
|
2004-01-03 11:59:52 +01:00
|
|
|
SLONG commit_mask;
|
2003-02-12 23:57:57 +01:00
|
|
|
if (!(dbb->dbb_flags & DBB_force_write) &&
|
2003-12-22 11:00:59 +01:00
|
|
|
(commit_mask = dbb->dbb_flush_cycle))
|
|
|
|
{
|
2003-02-12 23:57:57 +01:00
|
|
|
dbb->dbb_flush_cycle = 0;
|
2003-12-22 11:00:59 +01:00
|
|
|
btc_flush(tdbb, commit_mask, false, status_vector);
|
2003-02-12 23:57:57 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
2003-12-01 03:37:25 +01:00
|
|
|
THREAD_EXIT;
|
|
|
|
THREAD_YIELD;
|
|
|
|
THREAD_ENTER;
|
|
|
|
|
2003-02-12 23:57:57 +01:00
|
|
|
if (bcb->bcb_flags & BCB_free_pending) {
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = get_buffer(tdbb, FREE_PAGE, LATCH_none, 1);
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb) {
|
2003-12-22 11:00:59 +01:00
|
|
|
write_buffer(tdbb, bdb, bdb->bdb_page, true, status_vector, true);
|
2003-02-12 23:57:57 +01:00
|
|
|
bcb = dbb->dbb_bcb;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-02-12 23:57:57 +01:00
|
|
|
/* If the cache reader or garbage collector is idle, put
|
|
|
|
them to work freeing pages. */
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef CACHE_READER
|
2003-02-12 23:57:57 +01:00
|
|
|
if (bcb->bcb_flags & BCB_cache_reader &&
|
2004-01-03 11:59:52 +01:00
|
|
|
!(bcb->bcb_flags & BCB_reader_active))
|
|
|
|
{
|
2003-02-12 23:57:57 +01:00
|
|
|
ISC_event_post(dbb->dbb_reader_event);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
#ifdef GARBAGE_THREAD
|
2003-02-12 23:57:57 +01:00
|
|
|
if (dbb->dbb_flags & DBB_garbage_collector &&
|
2004-01-03 11:59:52 +01:00
|
|
|
!(dbb->dbb_flags & DBB_gc_active))
|
|
|
|
{
|
2003-02-12 23:57:57 +01:00
|
|
|
ISC_event_post(dbb->dbb_gc_event);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2003-02-12 23:57:57 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-02-12 23:57:57 +01:00
|
|
|
/* If there's more work to do voluntarily ask to be rescheduled.
|
|
|
|
Otherwise, wait for event notification. */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-02-12 23:57:57 +01:00
|
|
|
if ((bcb->bcb_flags & BCB_free_pending) ||
|
2004-01-03 11:59:52 +01:00
|
|
|
bcb->bcb_checkpoint || dbb->dbb_flush_cycle)
|
|
|
|
{
|
2003-12-22 11:00:59 +01:00
|
|
|
JRD_reschedule(tdbb, 0, true);
|
2003-02-12 23:57:57 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef CACHE_READER
|
2004-02-20 07:43:27 +01:00
|
|
|
else if (SBM_next(bcb->bcb_prefetch, &starting_page, RSE_get_forward))
|
|
|
|
{
|
2003-02-12 23:57:57 +01:00
|
|
|
/* Prefetch some pages in our spare time and in the process
|
|
|
|
garbage collect the prefetch bitmap. */
|
2003-10-20 12:53:52 +02:00
|
|
|
prf prefetch;
|
2003-02-12 23:57:57 +01:00
|
|
|
|
|
|
|
prefetch_init(&prefetch, tdbb);
|
|
|
|
prefetch_prologue(&prefetch, &starting_page);
|
|
|
|
prefetch_io(&prefetch, status_vector);
|
|
|
|
prefetch_epilogue(&prefetch, status_vector);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2003-02-12 23:57:57 +01:00
|
|
|
else {
|
|
|
|
bcb->bcb_flags &= ~BCB_writer_active;
|
|
|
|
THREAD_EXIT;
|
2003-09-01 09:58:04 +02:00
|
|
|
ISC_event_wait(1, &writer_event, &count, 10 * 1000000, NULL, 0);
|
2003-02-12 23:57:57 +01:00
|
|
|
THREAD_ENTER;
|
|
|
|
}
|
|
|
|
bcb = dbb->dbb_bcb;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2003-02-12 23:57:57 +01:00
|
|
|
LCK_fini(tdbb, LCK_OWNER_attachment);
|
|
|
|
delete tdbb->tdbb_attachment;
|
|
|
|
bcb->bcb_flags &= ~BCB_cache_writer;
|
|
|
|
/* Notify the finalization caller that we're finishing. */
|
|
|
|
ISC_event_post(dbb->dbb_writer_event_fini);
|
|
|
|
ISC_event_fini(writer_event);
|
|
|
|
THREAD_EXIT;
|
2001-12-24 03:51:06 +01:00
|
|
|
|
|
|
|
} // try
|
2004-03-01 04:35:23 +01:00
|
|
|
catch (const std::exception& ex) {
|
|
|
|
Firebird::stuff_exception(status_vector, ex);
|
2001-12-24 03:51:06 +01:00
|
|
|
bcb = dbb->dbb_bcb;
|
|
|
|
gds__log_status(dbb->dbb_file->fil_string, status_vector);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static void check_precedence(thread_db* tdbb, WIN * window, SLONG page)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c h e c k _ p r e c e d e n c e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Given a window accessed for write and a page number,
|
|
|
|
* establish a precedence relationship such that the
|
|
|
|
* specified page will always be written before the page
|
|
|
|
* associated with the window.
|
|
|
|
*
|
|
|
|
* If the "page number" is negative, it is really a transaction
|
|
|
|
* id. In this case, the precedence relationship is to the
|
|
|
|
* database header page from which the transaction id was
|
|
|
|
* obtained. If the header page has been written since the
|
|
|
|
* transaction id was assigned, no precedence relationship
|
|
|
|
* is required.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If this is really a transaction id, sort things out */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (page < 0) {
|
|
|
|
if (-page <= dbb->dbb_last_header_write) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
page = 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Start by finding the buffer containing the high priority page */
|
|
|
|
|
|
|
|
BCB_MUTEX_ACQUIRE;
|
|
|
|
PRE_MUTEX_ACQUIRE;
|
2004-01-03 11:59:52 +01:00
|
|
|
BCB bcb = dbb->dbb_bcb;
|
|
|
|
QUE mod_que = &bcb->bcb_rpt[page % bcb->bcb_count].bcb_page_mod;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* high = 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
QUE que;
|
|
|
|
for (que = mod_que->que_forward; que != mod_que; que = que->que_forward) {
|
2004-03-11 06:04:26 +01:00
|
|
|
if ((high = BLOCK(que, Buffer_desc*, bdb_que))->bdb_page == page) {
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
BCB_MUTEX_RELEASE;
|
|
|
|
if (que == mod_que) {
|
|
|
|
PRE_MUTEX_RELEASE;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Found the higher precedence buffer. If it's not dirty, don't sweat it.
|
|
|
|
If it's the same page, ditto. */
|
|
|
|
|
|
|
|
if (!(high->bdb_flags & BDB_dirty)
|
2004-01-03 11:59:52 +01:00
|
|
|
|| (high->bdb_page == window->win_page))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
PRE_MUTEX_RELEASE;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* low = window->win_bdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if ((low->bdb_flags & BDB_marked) && !(low->bdb_flags & BDB_faked))
|
|
|
|
cache_bugcheck(212); /* msg 212 CCH_precedence: block marked */
|
|
|
|
|
|
|
|
/* If already related, there's nothing more to do. If the precedence
|
|
|
|
search was too complex to complete, just write the high page and
|
|
|
|
forget about about establishing the relationship. */
|
|
|
|
|
|
|
|
if (QUE_NOT_EMPTY(high->bdb_lower)) {
|
2004-01-03 11:59:52 +01:00
|
|
|
const SSHORT relationship = related(low, high, PRE_SEARCH_LIMIT);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (relationship == PRE_EXISTS) {
|
|
|
|
PRE_MUTEX_RELEASE;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else if (relationship == PRE_UNKNOWN) {
|
2004-01-03 11:59:52 +01:00
|
|
|
const SLONG high_page = high->bdb_page;
|
2001-05-23 15:26:42 +02:00
|
|
|
PRE_MUTEX_RELEASE;
|
|
|
|
if (!write_buffer
|
2003-12-22 11:00:59 +01:00
|
|
|
(tdbb, high, high_page, false, tdbb->tdbb_status_vector, true))
|
|
|
|
{
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check to see if we're going to create a cycle or the precedence search
|
|
|
|
was too complex to complete. If so, force a write of the "after"
|
|
|
|
(currently fetched) page. Assuming everyone obeys the rules and calls
|
|
|
|
precedence before marking the buffer, everything should be ok */
|
|
|
|
|
|
|
|
if (QUE_NOT_EMPTY(low->bdb_lower)) {
|
2004-01-03 11:59:52 +01:00
|
|
|
const SSHORT relationship = related(high, low, PRE_SEARCH_LIMIT);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (relationship == PRE_EXISTS || relationship == PRE_UNKNOWN) {
|
2004-01-03 11:59:52 +01:00
|
|
|
const SLONG low_page = low->bdb_page;
|
2001-05-23 15:26:42 +02:00
|
|
|
PRE_MUTEX_RELEASE;
|
|
|
|
if (!write_buffer
|
2003-12-22 11:00:59 +01:00
|
|
|
(tdbb, low, low_page, false, tdbb->tdbb_status_vector, true))
|
|
|
|
{
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
PRE_MUTEX_ACQUIRE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We're going to establish a new precedence relationship. Get a block,
|
|
|
|
fill in the appropriate fields, and insert it into the various ques */
|
|
|
|
|
|
|
|
bcb = dbb->dbb_bcb; /* Re-initialize */
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
Precedence* precedence = bcb->bcb_free;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (precedence) {
|
2004-02-20 07:43:27 +01:00
|
|
|
bcb->bcb_free = (Precedence*) precedence->pre_hi;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else {
|
2004-02-20 07:43:27 +01:00
|
|
|
precedence = FB_NEW(*dbb->dbb_bufferpool) Precedence;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
precedence->pre_low = low;
|
|
|
|
precedence->pre_hi = high;
|
|
|
|
precedence->pre_flags = 0;
|
|
|
|
QUE_INSERT(low->bdb_higher, precedence->pre_higher);
|
|
|
|
QUE_INSERT(high->bdb_lower, precedence->pre_lower);
|
|
|
|
PRE_MUTEX_RELEASE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static void clear_precedence(Database* dbb, Buffer_desc* bdb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c l e a r _ p r e c e d e n c e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Clear precedence relationships to lower precedence block.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_DBB(dbb);
|
|
|
|
|
|
|
|
PRE_MUTEX_ACQUIRE;
|
2004-01-03 11:59:52 +01:00
|
|
|
BCB bcb = dbb->dbb_bcb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Loop thru lower precedence buffers. If any can be downgraded,
|
|
|
|
by all means down grade them. */
|
|
|
|
|
|
|
|
while (QUE_NOT_EMPTY(bdb->bdb_lower)) {
|
2004-01-03 11:59:52 +01:00
|
|
|
QUE que = bdb->bdb_lower.que_forward;
|
2004-02-20 07:43:27 +01:00
|
|
|
Precedence* precedence = BLOCK(que, Precedence*, pre_lower);
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* low_bdb = precedence->pre_low;
|
2001-05-23 15:26:42 +02:00
|
|
|
QUE_DELETE(precedence->pre_higher);
|
|
|
|
QUE_DELETE(precedence->pre_lower);
|
2004-03-11 06:04:26 +01:00
|
|
|
precedence->pre_hi = (Buffer_desc*) bcb->bcb_free;
|
2001-05-23 15:26:42 +02:00
|
|
|
bcb->bcb_free = precedence;
|
|
|
|
if (!(precedence->pre_flags & PRE_cleared)) {
|
|
|
|
if (low_bdb->bdb_ast_flags & BDB_blocking)
|
2003-08-13 13:11:12 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
PAGE_LOCK_RE_POST(low_bdb->bdb_lock);
|
2003-08-13 13:11:12 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
PRE_MUTEX_RELEASE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static Buffer_desc* dealloc_bdb(Buffer_desc* bdb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* d e a l l o c _ b d b
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Deallocate buffer descriptor block.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
if (bdb) {
|
|
|
|
#ifndef PAGE_LATCHING
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_lock) {
|
2001-12-24 03:51:06 +01:00
|
|
|
delete bdb->bdb_lock;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
QUE_DELETE(bdb->bdb_que);
|
2001-12-24 03:51:06 +01:00
|
|
|
delete bdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifndef PAGE_LATCHING
|
2003-12-22 11:00:59 +01:00
|
|
|
// CVC: Nobody was interested in the result from this function, so I made it
|
|
|
|
// void instead of bool, but preserved the returned values in comments.
|
2004-03-11 06:04:26 +01:00
|
|
|
static void down_grade(thread_db* tdbb, Buffer_desc* bdb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* d o w n _ g r a d e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* A lock on a page is blocking another process. If possible, downgrade
|
|
|
|
* the lock on the buffer. This may be called from either AST or
|
|
|
|
* regular level. Return TRUE if the down grade was successful. If the
|
|
|
|
* down grade was deferred for any reason, return FALSE.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
|
|
|
|
bdb->bdb_ast_flags |= BDB_blocking;
|
2004-01-03 11:59:52 +01:00
|
|
|
lck* lock = bdb->bdb_lock;
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = bdb->bdb_dbb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (dbb->dbb_flags & DBB_bugcheck) {
|
|
|
|
PAGE_LOCK_RELEASE(bdb->bdb_lock);
|
|
|
|
bdb->bdb_ast_flags &= ~BDB_blocking;
|
2003-09-08 22:23:46 +02:00
|
|
|
// Release backup pages lock as buffer is no longer dirty
|
|
|
|
if (bdb->bdb_flags & BDB_dirty) {
|
|
|
|
bdb->bdb_flags &= ~BDB_dirty;
|
|
|
|
set_write_direction(dbb, bdb, BDB_write_undefined);
|
|
|
|
}
|
2003-12-22 11:00:59 +01:00
|
|
|
return; // TRUE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
/* If the Buffer_desc is in use and, being written or already
|
2001-05-23 15:26:42 +02:00
|
|
|
downgraded to read, mark it as blocking and exit. */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_use_count) {
|
2003-12-22 11:00:59 +01:00
|
|
|
return; // FALSE;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
latch_bdb(tdbb, LATCH_io, bdb, bdb->bdb_page, 1);
|
|
|
|
|
|
|
|
/* If the page isn't dirty, the lock can be quietly downgraded. */
|
|
|
|
|
|
|
|
if (!(bdb->bdb_flags & BDB_dirty)) {
|
|
|
|
bdb->bdb_ast_flags &= ~BDB_blocking;
|
|
|
|
#ifdef VMS
|
2004-01-03 11:59:52 +01:00
|
|
|
if (lock->lck_logical == LCK_write) {
|
2001-05-23 15:26:42 +02:00
|
|
|
LCK_convert(tdbb, lock, LCK_read, LCK_WAIT);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
PAGE_LOCK_RELEASE(bdb->bdb_lock);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#else
|
|
|
|
LCK_downgrade(tdbb, lock);
|
|
|
|
#endif
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, false, false, false);
|
2003-12-22 11:00:59 +01:00
|
|
|
return; // TRUE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
bool in_use = false, invalid = false;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_flags & BDB_not_valid) {
|
|
|
|
invalid = true;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If there are higher precedence guys, see if they can be written. */
|
2004-01-03 11:59:52 +01:00
|
|
|
QUE que;
|
2001-05-23 15:26:42 +02:00
|
|
|
for (que = bdb->bdb_higher.que_forward; que != &bdb->bdb_higher;
|
2003-12-22 11:00:59 +01:00
|
|
|
que = que->que_forward)
|
|
|
|
{
|
2004-02-20 07:43:27 +01:00
|
|
|
Precedence* precedence = BLOCK(que, Precedence*, pre_higher);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (precedence->pre_flags & PRE_cleared)
|
|
|
|
continue;
|
|
|
|
if (invalid) {
|
|
|
|
precedence->pre_flags |= PRE_cleared;
|
|
|
|
continue;
|
|
|
|
}
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* blocking_bdb = precedence->pre_hi;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (blocking_bdb->bdb_flags & BDB_dirty) {
|
|
|
|
down_grade(tdbb, blocking_bdb);
|
2004-01-03 11:59:52 +01:00
|
|
|
if (blocking_bdb->bdb_flags & BDB_dirty) {
|
|
|
|
in_use = true;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
if (blocking_bdb->bdb_flags & BDB_not_valid) {
|
2004-01-03 11:59:52 +01:00
|
|
|
invalid = true;
|
|
|
|
in_use = false;
|
2001-05-23 15:26:42 +02:00
|
|
|
que = bdb->bdb_higher.que_forward;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If any higher precedence buffer can't be written, mark this buffer
|
|
|
|
as blocking and exit. */
|
|
|
|
|
|
|
|
if (in_use) {
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, false, false, false);
|
2003-12-22 11:00:59 +01:00
|
|
|
return; // FALSE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Everything is clear to write this buffer. Do so and reduce the lock */
|
|
|
|
|
|
|
|
if (invalid
|
2003-12-22 11:00:59 +01:00
|
|
|
|| !write_page(tdbb, bdb, false, tdbb->tdbb_status_vector, true))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_flags |= BDB_not_valid;
|
2003-09-08 22:23:46 +02:00
|
|
|
// Release backup pages lock
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_flags &= ~BDB_dirty;
|
2003-09-08 22:23:46 +02:00
|
|
|
set_write_direction(dbb, bdb, BDB_write_undefined);
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_ast_flags &= ~BDB_blocking;
|
|
|
|
TRA_invalidate(dbb, bdb->bdb_transactions);
|
|
|
|
bdb->bdb_transactions = 0;
|
|
|
|
PAGE_LOCK_RELEASE(bdb->bdb_lock);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
bdb->bdb_ast_flags &= ~BDB_blocking;
|
|
|
|
#ifdef VMS
|
|
|
|
LCK_convert(tdbb, lock, LCK_read, LCK_WAIT);
|
|
|
|
#else
|
|
|
|
LCK_downgrade(tdbb, lock);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Clear precedence relationships to lower precedence buffers. Since it
|
|
|
|
isn't safe to tweak the que pointers from AST level, just mark the
|
|
|
|
precedence links as cleared. Somebody else will clean up the precedence
|
|
|
|
blocks. */
|
|
|
|
|
|
|
|
for (que = bdb->bdb_lower.que_forward; que != &bdb->bdb_lower;
|
2003-12-22 11:00:59 +01:00
|
|
|
que = que->que_forward)
|
|
|
|
{
|
2004-02-20 07:43:27 +01:00
|
|
|
Precedence* precedence = BLOCK(que, Precedence*, pre_lower);
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* blocking_bdb = precedence->pre_low;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (bdb->bdb_flags & BDB_not_valid)
|
|
|
|
blocking_bdb->bdb_flags |= BDB_not_valid;
|
|
|
|
precedence->pre_flags |= PRE_cleared;
|
|
|
|
if (blocking_bdb->bdb_flags & BDB_not_valid ||
|
|
|
|
blocking_bdb->bdb_ast_flags & BDB_blocking)
|
2003-12-22 11:00:59 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
down_grade(tdbb, blocking_bdb);
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bdb->bdb_flags &= ~BDB_not_valid;
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, false, false, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
return; // TRUE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static void expand_buffers(thread_db* tdbb, ULONG number)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* e x p a n d _ b u f f e r s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Expand the cache to at least a given number of buffers. If
|
|
|
|
* it's already that big, don't do anything.
|
|
|
|
*
|
2004-03-09 01:17:07 +01:00
|
|
|
* Nickolay Samofatov, 08-Mar-2004.
|
|
|
|
* This function does not handle exceptions correctly,
|
|
|
|
* it looks like good handling requires rewrite.
|
|
|
|
*
|
2001-05-23 15:26:42 +02:00
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2004-01-03 11:59:52 +01:00
|
|
|
BCB old = dbb->dbb_bcb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (number <= old->bcb_count || number > MAX_PAGE_BUFFERS) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* for Win16 platform, we want to ensure that no cache buffer ever
|
|
|
|
ends on a segment boundary */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
ULONG num_per_seg = number - old->bcb_count;
|
|
|
|
ULONG left_to_do = num_per_seg;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Allocate and initialize buffers control block */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
JrdMemoryPool* old_pool = tdbb->tdbb_default;
|
2001-05-23 15:26:42 +02:00
|
|
|
tdbb->tdbb_default = dbb->dbb_bufferpool;
|
|
|
|
|
|
|
|
old = dbb->dbb_bcb;
|
2004-01-03 11:59:52 +01:00
|
|
|
const bcb_repeat* const old_end = old->bcb_rpt + old->bcb_count;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-02 12:02:12 +01:00
|
|
|
BCB new_block = FB_NEW_RPT(*dbb->dbb_bufferpool, number) bcb;
|
|
|
|
new_block->bcb_count = number;
|
|
|
|
new_block->bcb_free_minimum = (SSHORT) MIN(number / 4, 128); /* 25% clean page reserve */
|
|
|
|
new_block->bcb_checkpoint = old->bcb_checkpoint;
|
|
|
|
new_block->bcb_flags = old->bcb_flags;
|
|
|
|
const bcb_repeat* const new_end = new_block->bcb_rpt + number;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* point at the dirty page binary tree */
|
|
|
|
|
2004-02-02 12:02:12 +01:00
|
|
|
new_block->bcb_btree = old->bcb_btree;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* point at the free precedence blocks */
|
|
|
|
|
2004-02-02 12:02:12 +01:00
|
|
|
new_block->bcb_free = old->bcb_free;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* position the new bcb in the in use, empty and latch queues */
|
|
|
|
|
2004-02-02 12:02:12 +01:00
|
|
|
QUE_INSERT(old->bcb_in_use, new_block->bcb_in_use);
|
2001-05-23 15:26:42 +02:00
|
|
|
QUE_DELETE(old->bcb_in_use);
|
2004-02-02 12:02:12 +01:00
|
|
|
QUE_INSERT(old->bcb_empty, new_block->bcb_empty);
|
2001-05-23 15:26:42 +02:00
|
|
|
QUE_DELETE(old->bcb_empty);
|
2004-02-02 12:02:12 +01:00
|
|
|
QUE_INSERT(old->bcb_free_lwt, new_block->bcb_free_lwt);
|
2001-05-23 15:26:42 +02:00
|
|
|
QUE_DELETE(old->bcb_free_lwt);
|
|
|
|
|
|
|
|
/* Copy addresses of previously allocated buffer space to new block */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
for (const lls* stack = old->bcb_memory; stack; stack = stack->lls_next) {
|
2004-02-02 12:02:12 +01:00
|
|
|
LLS_PUSH(stack->lls_object, &new_block->bcb_memory);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Initialize tail of new buffer control block */
|
2004-01-03 11:59:52 +01:00
|
|
|
bcb_repeat* new_tail;
|
2004-02-02 12:02:12 +01:00
|
|
|
for (new_tail = new_block->bcb_rpt; new_tail < new_end; new_tail++) {
|
2001-05-23 15:26:42 +02:00
|
|
|
QUE_INIT(new_tail->bcb_page_mod);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Move any active buffers from old block to new */
|
|
|
|
|
2004-02-02 12:02:12 +01:00
|
|
|
new_tail = new_block->bcb_rpt;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
for (bcb_repeat* old_tail = old->bcb_rpt; old_tail < old_end;
|
|
|
|
old_tail++, new_tail++)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
new_tail->bcb_bdb = old_tail->bcb_bdb;
|
|
|
|
while (QUE_NOT_EMPTY(old_tail->bcb_page_mod)) {
|
2004-01-03 11:59:52 +01:00
|
|
|
QUE que = old_tail->bcb_page_mod.que_forward;
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = BLOCK(que, Buffer_desc*, bdb_que);
|
2001-05-23 15:26:42 +02:00
|
|
|
QUE_DELETE((*que));
|
2004-01-03 11:59:52 +01:00
|
|
|
QUE mod_que =
|
2004-02-02 12:02:12 +01:00
|
|
|
&new_block->bcb_rpt[bdb->bdb_page % new_block->bcb_count].bcb_page_mod;
|
2001-05-23 15:26:42 +02:00
|
|
|
QUE_INSERT((*mod_que), (*que));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Allocate new buffer descriptor blocks */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
ULONG num_in_seg = 0;
|
|
|
|
UCHAR* memory = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
for (; new_tail < new_end; new_tail++) {
|
|
|
|
/* if current segment is exhausted, allocate another */
|
|
|
|
|
|
|
|
if (!num_in_seg) {
|
2003-01-18 19:02:12 +01:00
|
|
|
memory = (UCHAR *)gds__alloc((SLONG) dbb->dbb_page_size *
|
|
|
|
(num_per_seg + 1));
|
2004-03-09 01:17:07 +01:00
|
|
|
// NOMEM: crash!
|
2004-02-02 12:02:12 +01:00
|
|
|
LLS_PUSH(memory, &new_block->bcb_memory);
|
2001-05-23 15:26:42 +02:00
|
|
|
memory = (UCHAR *) (((U_IPTR) memory + dbb->dbb_page_size - 1) &
|
|
|
|
~((int) dbb->dbb_page_size - 1));
|
|
|
|
num_in_seg = num_per_seg;
|
|
|
|
left_to_do -= num_per_seg;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (num_per_seg > left_to_do) {
|
2001-05-23 15:26:42 +02:00
|
|
|
num_per_seg = left_to_do;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-02-02 12:02:12 +01:00
|
|
|
new_tail->bcb_bdb = alloc_bdb(tdbb, new_block, &memory);
|
2001-05-23 15:26:42 +02:00
|
|
|
num_in_seg--;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set up new buffer control, release old buffer control, and clean up */
|
|
|
|
|
2004-02-02 12:02:12 +01:00
|
|
|
dbb->dbb_bcb = new_block;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
delete old;
|
2001-05-23 15:26:42 +02:00
|
|
|
tdbb->tdbb_default = old_pool;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static Buffer_desc* get_buffer(thread_db* tdbb, SLONG page, LATCH latch, SSHORT latch_wait)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* g e t _ b u f f e r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Get a buffer. If possible, get a buffer already assigned
|
|
|
|
* to the page. Otherwise get one from the free list or pick
|
|
|
|
* the least recently used buffer to be reused.
|
|
|
|
* Note the following special page numbers:
|
2004-01-06 11:33:18 +01:00
|
|
|
* -1 indicates that a buffer is required for journaling => obsolete
|
2001-05-23 15:26:42 +02:00
|
|
|
* -2 indicates a special scratch buffer for shadowing
|
|
|
|
*
|
|
|
|
* input
|
|
|
|
* page: page to get
|
|
|
|
* latch: type of latch to acquire on the page.
|
|
|
|
* 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
|
2004-03-11 06:04:26 +01:00
|
|
|
* Buffer_desc pointer if successful.
|
2001-05-23 15:26:42 +02:00
|
|
|
* NULL pointer if timeout occurred (only possible is latch_wait <> 1).
|
|
|
|
* if cache manager doesn't have any pages to write anymore.
|
2001-07-10 19:35:13 +02:00
|
|
|
*
|
2001-05-23 15:26:42 +02:00
|
|
|
**************************************/
|
2004-01-03 11:59:52 +01:00
|
|
|
// CVC: Those two vars are tricky or nonsense to put in minimal scope.
|
2003-02-10 14:28:35 +01:00
|
|
|
QUE que;
|
|
|
|
BCB bcb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2004-01-03 11:59:52 +01:00
|
|
|
SSHORT walk = dbb->dbb_bcb->bcb_free_minimum;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
BCB_MUTEX_ACQUIRE;
|
|
|
|
|
2003-12-11 11:33:30 +01:00
|
|
|
while (true) {
|
2001-05-23 15:26:42 +02:00
|
|
|
find_page:
|
|
|
|
|
|
|
|
bcb = dbb->dbb_bcb;
|
|
|
|
if (page >= 0) {
|
|
|
|
/* Check to see if buffer has already been assigned to page */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
QUE mod_que = &bcb->bcb_rpt[page % bcb->bcb_count].bcb_page_mod;
|
2001-05-23 15:26:42 +02:00
|
|
|
for (que = mod_que->que_forward; que != mod_que;
|
2003-08-09 22:58:34 +02:00
|
|
|
que = que->que_forward)
|
2004-01-03 11:59:52 +01:00
|
|
|
{
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = BLOCK(que, Buffer_desc*, bdb_que);
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_page == page) {
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef SUPERSERVER_V2
|
|
|
|
if (page != HEADER_PAGE)
|
|
|
|
#endif
|
|
|
|
QUE_MOST_RECENTLY_USED(bdb->bdb_in_use);
|
|
|
|
BCB_MUTEX_RELEASE;
|
2004-01-03 11:59:52 +01:00
|
|
|
const SSHORT latch_return =
|
|
|
|
latch_bdb(tdbb, latch, bdb, page, latch_wait);
|
|
|
|
if (latch_return) {
|
|
|
|
if (latch_return == 1) {
|
2003-09-01 09:58:04 +02:00
|
|
|
return NULL; /* permitted timeout happened */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
BCB_MUTEX_ACQUIRE;
|
|
|
|
goto find_page;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
bdb->bdb_flags &= ~(BDB_faked | BDB_prefetch);
|
|
|
|
bdb->bdb_sequence = dbb->dbb_fetches++;
|
|
|
|
return bdb;
|
|
|
|
}
|
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#ifdef CACHE_WRITER
|
|
|
|
else if ((page == FREE_PAGE) || (page == CHECKPOINT_PAGE)) {
|
|
|
|
/* This code is only used by the background I/O threads:
|
|
|
|
cache writer, cache reader and garbage collector. */
|
|
|
|
|
2003-12-01 03:37:25 +01:00
|
|
|
THREAD_EXIT;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
for (que = bcb->bcb_in_use.que_backward;
|
2003-12-22 11:00:59 +01:00
|
|
|
que != &bcb->bcb_in_use; que = que->que_backward)
|
|
|
|
{
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = BLOCK(que, Buffer_desc*, bdb_in_use);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (page == FREE_PAGE) {
|
|
|
|
if (bdb->bdb_use_count ||
|
2004-01-03 11:59:52 +01:00
|
|
|
bdb->bdb_flags & BDB_free_pending)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
if (bdb->bdb_flags & BDB_db_dirty) {
|
2003-12-01 03:37:25 +01:00
|
|
|
THREAD_ENTER;
|
2001-05-23 15:26:42 +02:00
|
|
|
BCB_MUTEX_RELEASE;
|
|
|
|
return bdb;
|
|
|
|
}
|
|
|
|
if (!--walk) {
|
|
|
|
bcb->bcb_flags &= ~BCB_free_pending;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else { /* if (page == CHECKPOINT_PAGE) */
|
|
|
|
|
|
|
|
if (bdb->bdb_flags & BDB_checkpoint) {
|
2003-12-01 03:37:25 +01:00
|
|
|
THREAD_ENTER;
|
2001-05-23 15:26:42 +02:00
|
|
|
BCB_MUTEX_RELEASE;
|
|
|
|
return bdb;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-12-01 03:37:25 +01:00
|
|
|
THREAD_ENTER;
|
2001-05-23 15:26:42 +02:00
|
|
|
BCB_MUTEX_RELEASE;
|
2003-09-01 09:58:04 +02:00
|
|
|
return NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
for (que = bcb->bcb_in_use.que_backward;
|
|
|
|
que != &bcb->bcb_in_use || QUE_NOT_EMPTY(bcb->bcb_empty);
|
2004-01-03 11:59:52 +01:00
|
|
|
que = que->que_backward)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
bcb = dbb->dbb_bcb; /* Re-initialize in the loop */
|
2004-01-03 11:59:52 +01:00
|
|
|
QUE mod_que = &bcb->bcb_rpt[page % bcb->bcb_count].bcb_page_mod;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If there is an empty buffer sitting around, allocate it */
|
|
|
|
|
|
|
|
if (QUE_NOT_EMPTY(bcb->bcb_empty)) {
|
|
|
|
que = bcb->bcb_empty.que_forward;
|
|
|
|
QUE_DELETE((*que));
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = BLOCK(que, Buffer_desc*, bdb_que);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (page >= 0) {
|
|
|
|
QUE_INSERT((*mod_que), (*que));
|
|
|
|
#ifdef SUPERSERVER_V2
|
|
|
|
/* Reserve a buffer for header page with deferred header
|
|
|
|
page write mechanism. Otherwise, a deadlock will occur
|
|
|
|
if all dirty pages in the cache must force header page
|
|
|
|
to disk before they can be written but there is no free
|
|
|
|
buffer to read the header page into. */
|
|
|
|
|
|
|
|
if (page != HEADER_PAGE)
|
|
|
|
#endif
|
|
|
|
QUE_INSERT(bcb->bcb_in_use, bdb->bdb_in_use);
|
|
|
|
}
|
|
|
|
|
2001-07-10 19:35:13 +02:00
|
|
|
/* This correction for bdb_use_count below is needed to
|
2001-05-23 15:26:42 +02:00
|
|
|
avoid a deadlock situation in latching code. It's not
|
|
|
|
clear though how the bdb_use_count can get < 0 for a bdb
|
|
|
|
in bcb_empty queue */
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_use_count < 0) {
|
2001-05-23 15:26:42 +02:00
|
|
|
cache_bugcheck(301); /* msg 301 Non-zero use_count of a buffer in the empty que */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
bdb->bdb_page = page;
|
|
|
|
bdb->bdb_flags = BDB_read_pending;
|
|
|
|
bdb->bdb_scan_count = 0;
|
|
|
|
/* The following latch should never fail because the buffer is 'empty'
|
|
|
|
and the page is not in cache. */
|
2004-01-03 11:59:52 +01:00
|
|
|
if (latch_bdb(tdbb, latch, bdb, page, -100) == -1) {
|
2001-05-23 15:26:42 +02:00
|
|
|
cache_bugcheck(302); /* msg 302 unexpected page change */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifndef PAGE_LATCHING
|
2004-01-03 11:59:52 +01:00
|
|
|
if (page >= 0) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_lock->lck_logical = LCK_none;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
PAGE_LOCK_RELEASE(bdb->bdb_lock);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
bdb->bdb_sequence = dbb->dbb_fetches++;
|
|
|
|
BCB_MUTEX_RELEASE;
|
|
|
|
return bdb;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* get the oldest buffer as the least recently used -- note
|
|
|
|
that since there are no empty buffers this queue cannot be empty */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bcb->bcb_in_use.que_forward == &bcb->bcb_in_use) {
|
2001-05-23 15:26:42 +02:00
|
|
|
cache_bugcheck(213); /* msg 213 insufficient cache size */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* oldest = BLOCK(que, Buffer_desc*, bdb_in_use);
|
2001-05-23 15:26:42 +02:00
|
|
|
LATCH_MUTEX_ACQUIRE;
|
|
|
|
if (oldest->bdb_use_count
|
|
|
|
|| (oldest->bdb_flags & BDB_free_pending)
|
2003-12-22 11:00:59 +01:00
|
|
|
|| !writeable(oldest))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER_V2
|
|
|
|
/* If page has been prefetched but not yet fetched, let
|
|
|
|
it cycle once more thru LRU queue before re-using it. */
|
|
|
|
|
|
|
|
if (oldest->bdb_flags & BDB_prefetch) {
|
|
|
|
oldest->bdb_flags &= ~BDB_prefetch;
|
|
|
|
que = que->que_forward;
|
|
|
|
QUE_MOST_RECENTLY_USED(oldest->bdb_in_use);
|
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
#ifdef CACHE_WRITER
|
|
|
|
if (oldest->bdb_flags & (BDB_dirty | BDB_db_dirty)) {
|
|
|
|
bcb->bcb_flags |= BCB_free_pending;
|
|
|
|
if (bcb->bcb_flags & BCB_cache_writer &&
|
|
|
|
!(bcb->bcb_flags & BCB_writer_active))
|
2004-01-03 11:59:52 +01:00
|
|
|
{
|
|
|
|
ISC_event_post(dbb->dbb_writer_event);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
if (walk) {
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!--walk) {
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
continue;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = oldest;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (latch_bdb(tdbb, LATCH_exclusive, bdb, bdb->bdb_page, 0)) {
|
2004-03-11 06:04:26 +01:00
|
|
|
continue; /* Buffer_desc changed, continue looking for a buffer */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
QUE_MOST_RECENTLY_USED(bdb->bdb_in_use);
|
|
|
|
bdb->bdb_flags |= BDB_free_pending;
|
|
|
|
|
|
|
|
/* If the buffer selected is dirty, arrange to have it written. */
|
|
|
|
|
|
|
|
if (bdb->bdb_flags & (BDB_dirty | BDB_db_dirty)) {
|
|
|
|
BCB_MUTEX_RELEASE;
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
if (!write_buffer
|
2003-12-22 11:00:59 +01:00
|
|
|
(tdbb, bdb, bdb->bdb_page, true, tdbb->tdbb_status_vector,
|
|
|
|
true))
|
2001-05-23 15:26:42 +02:00
|
|
|
#else
|
|
|
|
if (!write_buffer
|
2003-12-22 11:00:59 +01:00
|
|
|
(tdbb, bdb, bdb->bdb_page, false,
|
|
|
|
tdbb->tdbb_status_vector, true))
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
{
|
|
|
|
bdb->bdb_flags &= ~BDB_free_pending;
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, false, false, false);
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bcb = dbb->dbb_bcb; /* Re-initialize */
|
|
|
|
|
|
|
|
/* If the buffer is still in the dirty tree, remove it.
|
|
|
|
In any case, release any lock it may have. */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_parent || (bdb == bcb->bcb_btree)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
btc_remove(bdb);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* if the page has an expanded index buffer, release it */
|
|
|
|
|
|
|
|
if (bdb->bdb_expanded_buffer) {
|
2001-12-24 03:51:06 +01:00
|
|
|
delete bdb->bdb_expanded_buffer;
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_expanded_buffer = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Cleanup any residual precedence blocks. Unless something is
|
|
|
|
screwed up, the only precedence blocks that can still be hanging
|
|
|
|
around are ones cleared at AST level. */
|
|
|
|
|
|
|
|
PRE_MUTEX_ACQUIRE;
|
|
|
|
while (QUE_NOT_EMPTY(bdb->bdb_higher)) {
|
2004-01-03 11:59:52 +01:00
|
|
|
QUE que2 = bdb->bdb_higher.que_forward;
|
2004-02-20 07:43:27 +01:00
|
|
|
Precedence* precedence = BLOCK(que2, Precedence*, pre_higher);
|
2001-05-23 15:26:42 +02:00
|
|
|
QUE_DELETE(precedence->pre_higher);
|
|
|
|
QUE_DELETE(precedence->pre_lower);
|
2004-03-11 06:04:26 +01:00
|
|
|
precedence->pre_hi = (Buffer_desc*) bcb->bcb_free;
|
2001-05-23 15:26:42 +02:00
|
|
|
bcb->bcb_free = precedence;
|
|
|
|
}
|
|
|
|
PRE_MUTEX_RELEASE;
|
|
|
|
|
|
|
|
clear_precedence(dbb, bdb);
|
|
|
|
|
|
|
|
/* remove the buffer from the "mod" queue and place it
|
|
|
|
in it's new spot, provided it's not a negative (scratch) page */
|
|
|
|
|
|
|
|
BCB_MUTEX_ACQUIRE;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_page >= 0) {
|
2001-05-23 15:26:42 +02:00
|
|
|
QUE_DELETE(bdb->bdb_que);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
QUE_INSERT(bcb->bcb_empty, bdb->bdb_que);
|
|
|
|
QUE_DELETE(bdb->bdb_in_use)
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
bdb->bdb_page = JOURNAL_PAGE;
|
|
|
|
release_bdb(tdbb, bdb, false, false, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (que == &bcb->bcb_in_use) {
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef SUPERSERVER
|
|
|
|
expand_buffers(tdbb, bcb->bcb_count + 75);
|
|
|
|
#else
|
|
|
|
cache_bugcheck(214); /* msg 214 no cache buffers available for reuse */
|
|
|
|
#endif
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef PAGE_LATCHING
|
|
|
|
static SSHORT latch_bdb(
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb,
|
|
|
|
LATCH type, Buffer_desc* bdb, SLONG page, SSHORT latch_wait)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* l a t c h _ b d b
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
* input
|
|
|
|
* type: LATCH_none, LATCH_exclusive, LATCH_io, LATCH_shared, or LATCH_mark.
|
|
|
|
* bdb: object to acquire latch on.
|
|
|
|
* page: page of bdb, for verification.
|
|
|
|
* 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 1.
|
|
|
|
* <negative number> => Latch timeout interval in seconds.
|
|
|
|
*
|
|
|
|
* return
|
|
|
|
* 0: latch successfully acquired.
|
|
|
|
* 1: latch not acquired due to a (permitted) timeout.
|
|
|
|
* -1: latch not acquired, bdb doesn't corresponds to page.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
/* If the buffer has been reassigned to another page
|
|
|
|
make the caller deal with it. */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_page != page) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return -1;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
LATCH_MUTEX_ACQUIRE;
|
|
|
|
|
|
|
|
/* Handle the easy case first, no users of the buffer. */
|
|
|
|
|
|
|
|
if (!bdb->bdb_use_count) {
|
|
|
|
switch (type) {
|
|
|
|
case LATCH_shared:
|
|
|
|
++bdb->bdb_use_count;
|
|
|
|
bdb->bdb_shared[0] = tdbb;
|
|
|
|
break;
|
|
|
|
case LATCH_exclusive:
|
|
|
|
++bdb->bdb_use_count;
|
|
|
|
bdb->bdb_exclusive = tdbb;
|
|
|
|
break;
|
|
|
|
case LATCH_io:
|
|
|
|
++bdb->bdb_use_count;
|
|
|
|
bdb->bdb_io = tdbb;
|
|
|
|
break;
|
|
|
|
case LATCH_mark:
|
|
|
|
cache_bugcheck(295); /* inconsistent LATCH_mark call */
|
|
|
|
break;
|
|
|
|
case LATCH_none:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Grant the latch request if it is compatible with existing
|
|
|
|
latch holders. Pending latches are queued in the order in
|
|
|
|
which they are requested except that io/mark latches are queued
|
|
|
|
ahead of all other latch requests (to avoid deadlocks and
|
|
|
|
this does not cause starvation). Also, shared latches are granted
|
|
|
|
immediately if a disk write is in progress.
|
2001-07-10 19:35:13 +02:00
|
|
|
Note that asking for a higher mode latch when already holding a
|
2001-05-23 15:26:42 +02:00
|
|
|
share latch results in deadlock. CCH_handoff routinely acquires a
|
|
|
|
shared latch while owning already a shared latch on the page
|
|
|
|
(the case of handing off to the same page). If the BDB_must_write
|
2001-07-10 19:35:13 +02:00
|
|
|
flag is set, then an exclusive latch request will be followed by
|
2001-05-23 15:26:42 +02:00
|
|
|
an io latch request. */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
SSHORT i;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
switch (type) {
|
|
|
|
|
|
|
|
case LATCH_none:
|
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
case LATCH_shared:
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_flags & BDB_read_pending) {
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
if (bdb->bdb_exclusive) {
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_exclusive != tdbb) {
|
2001-05-23 15:26:42 +02:00
|
|
|
break; /* someone else owns exclusive latch */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* Note that InterBase often 'hands-off' to the same page, for both
|
|
|
|
shared and exlusive latches. */
|
|
|
|
/* Check is we own already an exclusive latch. */
|
|
|
|
for (i = 0; (i < BDB_max_shared) && (bdb->bdb_shared[i] != tdbb);
|
|
|
|
i++);
|
|
|
|
if (i >= BDB_max_shared) { /* we don't own a shared latch yet */
|
|
|
|
/* If there are latch-waiters, and they are not waiting for an
|
|
|
|
io_latch, then we have to wait also (there must be a exclusive
|
|
|
|
latch waiter). If there is an IO in progress, the violate the
|
|
|
|
fairness and sneak ahead of the exclusive (or io) waiters. */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if ((QUE_NOT_EMPTY(bdb->bdb_waiters)) && !bdb->bdb_io) {
|
2001-05-23 15:26:42 +02:00
|
|
|
break; /* be fair and wait behind exclusive latch requests */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/* Nobody owns an exlusive latch, or sneak ahead of exclusive latch
|
|
|
|
waiters while an io is in progress. */
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
for (i = 0; (i < BDB_max_shared) && bdb->bdb_shared[i]; i++); // empty loop body
|
2004-01-03 11:59:52 +01:00
|
|
|
if (i >= BDB_max_shared) {
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
++bdb->bdb_use_count;
|
|
|
|
bdb->bdb_shared[i] = tdbb;
|
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
case LATCH_io:
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_flags & BDB_read_pending) {
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
if (bdb->bdb_io) {
|
2001-05-23 15:26:42 +02:00
|
|
|
break; /* someone else owns the io latch */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
++bdb->bdb_use_count;
|
|
|
|
bdb->bdb_io = tdbb;
|
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
case LATCH_exclusive:
|
|
|
|
/* Exclusive latches wait for existing shared latches and
|
|
|
|
(unfortunately) for existing io latches. This is not as
|
2001-07-10 19:35:13 +02:00
|
|
|
bad as it sounds because an exclusive latch is typically followed
|
2001-05-23 15:26:42 +02:00
|
|
|
by a mark latch, which then would wait behind the io latch. */
|
|
|
|
/* Note that the ail-code latches the same buffer multiple times
|
|
|
|
in shared and exclusive */
|
|
|
|
/* Note that InterBase often 'hands-off' to the same page, for both
|
|
|
|
shared and exlusive latches. */
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_use_count && (bdb->bdb_exclusive != tdbb)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
++bdb->bdb_use_count;
|
|
|
|
bdb->bdb_exclusive = tdbb;
|
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
case LATCH_mark:
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_exclusive != tdbb) {
|
2001-05-23 15:26:42 +02:00
|
|
|
cache_bugcheck(295); /* inconsistent LATCH_mark call */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Some InterBase code marks a buffer more than once. */
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_io && (bdb->bdb_io != tdbb)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_io = tdbb;
|
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If the caller doesn't want to wait for this latch, then return now. */
|
|
|
|
if (latch_wait == 0) {
|
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get or create a latch wait block and wait for someone to grant
|
|
|
|
the latch. */
|
|
|
|
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2004-01-03 11:59:52 +01:00
|
|
|
BCB bcb = dbb->dbb_bcb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
Latch_wait* lwt;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (QUE_NOT_EMPTY(bcb->bcb_free_lwt)) {
|
2004-01-03 11:59:52 +01:00
|
|
|
QUE que = bcb->bcb_free_lwt.que_forward;
|
2001-05-23 15:26:42 +02:00
|
|
|
QUE_DELETE((*que));
|
2004-02-20 07:43:27 +01:00
|
|
|
lwt = (Latch_wait*) BLOCK(que, Latch_wait*, lwt_waiters);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
2004-02-20 07:43:27 +01:00
|
|
|
lwt = FB_NEW(*dbb->dbb_bufferpool) Latch_wait;
|
|
|
|
QUE_INIT(lwt->lwt_waiters);
|
|
|
|
ISC_event_init(&lwt->lwt_event, 0, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
lwt->lwt_flags |= LWT_pending;
|
|
|
|
lwt->lwt_latch = type;
|
|
|
|
lwt->lwt_tdbb = tdbb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Give priority to IO. This might prevent deadlocks while performing
|
|
|
|
precedence writes. This does not cause starvation because an
|
|
|
|
exclusive latch is needed to dirty the page again. */
|
2004-01-03 11:59:52 +01:00
|
|
|
if ((type == LATCH_io) || (type == LATCH_mark)) {
|
2004-02-20 07:43:27 +01:00
|
|
|
QUE_INSERT(bdb->bdb_waiters, lwt->lwt_waiters)
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else {
|
2004-02-20 07:43:27 +01:00
|
|
|
QUE_APPEND(bdb->bdb_waiters, lwt->lwt_waiters);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
event_t* event = &lwt->lwt_event;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
int timeout_occurred = FALSE;
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Loop until the latch is granted or until a timeout occurrs. */
|
2004-01-03 11:59:52 +01:00
|
|
|
for (SLONG count = ISC_event_clear(event);
|
2004-02-20 07:43:27 +01:00
|
|
|
((lwt->lwt_flags & LWT_pending) && !timeout_occurred);
|
2004-01-03 11:59:52 +01:00
|
|
|
count = ISC_event_clear(event))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
THREAD_EXIT;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (latch_wait == 1) {
|
2001-05-23 15:26:42 +02:00
|
|
|
timeout_occurred =
|
2003-09-01 09:58:04 +02:00
|
|
|
ISC_event_wait(1, &event, &count, 120 * 1000000, NULL, event);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
timeout_occurred =
|
|
|
|
ISC_event_wait(1, &event, &count, -latch_wait * 1000000,
|
2003-09-01 09:58:04 +02:00
|
|
|
NULL, event);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
THREAD_ENTER;
|
|
|
|
LATCH_MUTEX_ACQUIRE;
|
|
|
|
}
|
|
|
|
|
|
|
|
bcb = dbb->dbb_bcb; /* Re-initialize */
|
2004-02-20 07:43:27 +01:00
|
|
|
QUE_DELETE(lwt->lwt_waiters);
|
|
|
|
QUE_INSERT(bcb->bcb_free_lwt, lwt->lwt_waiters);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If the latch is not granted then a timeout must have occurred. */
|
2004-02-20 07:43:27 +01:00
|
|
|
if ((lwt->lwt_flags & LWT_pending) && timeout_occurred) {
|
2001-05-23 15:26:42 +02:00
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
if (latch_wait == 1) {
|
2003-11-11 13:19:20 +01:00
|
|
|
IBERR_build_status(tdbb->tdbb_status_vector, isc_deadlock, 0);
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
return 1;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (bdb->bdb_page != page) {
|
|
|
|
LATCH_MUTEX_RELEASE;
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, true, false, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#else
|
|
|
|
static SSHORT latch_bdb(
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb,
|
|
|
|
LATCH type, Buffer_desc* bdb, SLONG page, SSHORT latch_wait)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* l a t c h _ b d b
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Simple optimized latching for single-threaded
|
|
|
|
* non-SUPERSERVER platforms.
|
2001-07-10 19:35:13 +02:00
|
|
|
*
|
2001-05-23 15:26:42 +02:00
|
|
|
**************************************/
|
|
|
|
|
|
|
|
++bdb->bdb_use_count;
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case LATCH_shared:
|
|
|
|
break;
|
|
|
|
case LATCH_exclusive:
|
|
|
|
bdb->bdb_exclusive = tdbb;
|
|
|
|
break;
|
|
|
|
case LATCH_io:
|
|
|
|
bdb->bdb_io = tdbb;
|
|
|
|
break;
|
|
|
|
case LATCH_mark:
|
|
|
|
bdb->bdb_io = tdbb;
|
|
|
|
case LATCH_none:
|
|
|
|
--bdb->bdb_use_count;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
static SSHORT lock_buffer(
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb,
|
|
|
|
Buffer_desc* bdb, SSHORT wait, SSHORT page_type)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* l o c k _ b u f f e r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Get a lock on page for a buffer. If the lock ever slipped
|
|
|
|
* below READ, indicate that the page must be read.
|
|
|
|
*
|
|
|
|
* input:
|
|
|
|
* wait: LCK_WAIT = TRUE = 1 => Wait as long a necessary to get the lock.
|
|
|
|
* LCK_NO_WAIT = FALSE = 0 => If the lock can't be acquired immediately,
|
|
|
|
* give up and return -1.
|
|
|
|
* <negative number> => Lock timeout interval in seconds.
|
2001-07-10 19:35:13 +02:00
|
|
|
*
|
2001-05-23 15:26:42 +02:00
|
|
|
* return: 0 => buffer locked, page is already in memroy.
|
|
|
|
* 1 => buffer locked, page needs to be read from disk.
|
|
|
|
* -1 => timeout on lock occurred, see input parameter 'wait'.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
|
|
|
#ifdef PAGE_LATCHING
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (dbb->dbb_refresh_ranges && bdb->bdb_flags & BDB_writer &&
|
2004-02-20 07:43:27 +01:00
|
|
|
(page_type == pag_data || page_type == pag_index))
|
|
|
|
{
|
2001-07-10 19:35:13 +02:00
|
|
|
/* This gives refresh cache ranges the potential to work with page
|
|
|
|
* latching by taking out a temporary page lock for notification
|
2001-05-23 15:26:42 +02:00
|
|
|
* purposes.
|
|
|
|
* Fix cache ranges on NetWare due to latching (bug 7199)
|
|
|
|
* Marion change # 18270, 13-Oct-1994
|
|
|
|
*/
|
|
|
|
|
2003-10-20 12:53:52 +02:00
|
|
|
lck refresh;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
refresh.lck_dbb = dbb;
|
|
|
|
refresh.lck_type = LCK_bdb;
|
|
|
|
refresh.lck_owner_handle =
|
|
|
|
LCK_get_owner_handle(tdbb, refresh.lck_type);
|
|
|
|
refresh.lck_parent = dbb->dbb_lock;
|
|
|
|
refresh.lck_length = sizeof(SLONG);
|
|
|
|
refresh.lck_key.lck_long = bdb->bdb_page;
|
|
|
|
|
|
|
|
if (LCK_lock_non_blocking(tdbb, &refresh, LCK_write, -1))
|
|
|
|
LCK_release(tdbb, &refresh);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ((bdb->bdb_flags & BDB_read_pending) ? 1 : 0);
|
|
|
|
#else
|
2004-01-03 11:59:52 +01:00
|
|
|
const USHORT lock_type =
|
2001-05-23 15:26:42 +02:00
|
|
|
(bdb->bdb_flags & (BDB_dirty | BDB_writer)) ? LCK_write : LCK_read;
|
2004-01-03 11:59:52 +01:00
|
|
|
lck* lock = bdb->bdb_lock;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (lock->lck_logical >= lock_type) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
|
|
|
|
TEXT errmsg[MAX_ERRMSG_LEN + 1];
|
|
|
|
ISC_STATUS* status = tdbb->tdbb_status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (lock->lck_logical == LCK_none) {
|
|
|
|
/* Prevent header and TIP pages from generating blocking AST
|
|
|
|
overhead. The promise is that the lock will unconditionally
|
|
|
|
be released when the buffer use count indicates it is safe
|
|
|
|
to do so. */
|
|
|
|
|
|
|
|
if (page_type == pag_header || page_type == pag_transactions) {
|
2003-11-04 00:59:24 +01:00
|
|
|
fb_assert(lock->lck_ast == blocking_ast_bdb);
|
2004-02-20 07:43:27 +01:00
|
|
|
fb_assert(lock->lck_object == bdb);
|
2001-05-23 15:26:42 +02:00
|
|
|
lock->lck_ast = 0;
|
|
|
|
lock->lck_object = NULL;
|
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
else {
|
2003-11-04 00:59:24 +01:00
|
|
|
fb_assert(lock->lck_ast != NULL);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
lock->lck_key.lck_long = bdb->bdb_page;
|
|
|
|
if (PAGE_LOCK_OPT(lock, lock_type, wait)) {
|
|
|
|
if (!lock->lck_ast) {
|
|
|
|
/* Restore blocking AST to lock block if it was swapped
|
2004-03-11 06:04:26 +01:00
|
|
|
out. Flag the Buffer_desc so that the lock is released when
|
2001-05-23 15:26:42 +02:00
|
|
|
the buffer is released. */
|
|
|
|
|
2003-11-04 00:59:24 +01:00
|
|
|
fb_assert(page_type == pag_header
|
2001-05-23 15:26:42 +02:00
|
|
|
|| page_type == pag_transactions);
|
2003-05-16 22:35:19 +02:00
|
|
|
lock->lck_ast = blocking_ast_bdb;
|
2004-02-20 07:43:27 +01:00
|
|
|
lock->lck_object = bdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_flags |= BDB_no_blocking_ast;
|
|
|
|
}
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!lock->lck_ast) {
|
2003-11-04 00:59:24 +01:00
|
|
|
fb_assert(page_type == pag_header || page_type == pag_transactions);
|
2003-05-16 22:35:19 +02:00
|
|
|
lock->lck_ast = blocking_ast_bdb;
|
2004-03-11 06:04:26 +01:00
|
|
|
lock->lck_object = bdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2001-07-10 19:35:13 +02:00
|
|
|
/* Case: a timeout was specified, or the caller didn't want to wait,
|
2001-05-23 15:26:42 +02:00
|
|
|
return the error. */
|
|
|
|
|
|
|
|
if ((wait == LCK_NO_WAIT)
|
2004-01-03 11:59:52 +01:00
|
|
|
|| ((wait < 0) && (status[1] == isc_lock_timeout)))
|
|
|
|
{
|
|
|
|
release_bdb(tdbb, bdb, false, false, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2001-07-10 19:35:13 +02:00
|
|
|
/* Case: lock manager detected a deadlock, probably caused by locking the
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc's in an unfortunate order. Nothing we can do about it, return the
|
2002-09-20 12:23:15 +02:00
|
|
|
error, and log it to firebird.log. */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
gds__msg_format(0, JRD_BUGCHK, 215, sizeof(errmsg), errmsg,
|
2004-01-21 08:18:30 +01:00
|
|
|
(TEXT *) (IPTR) bdb->bdb_page,
|
|
|
|
(TEXT *) (IPTR) page_type, 0, 0, 0);
|
2003-11-11 13:19:20 +01:00
|
|
|
IBERR_append_status(status, isc_random, isc_arg_string,
|
2001-05-23 15:26:42 +02:00
|
|
|
ERR_cstring(errmsg), 0);
|
|
|
|
ERR_log(JRD_BUGCHK, 215, errmsg); /* msg 215 page %ld, page type %ld lock conversion denied */
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
/* CCH_unwind releases all the Buffer_desc's and calls ERR_punt()
|
2001-05-23 15:26:42 +02:00
|
|
|
ERR_punt will longjump. */
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Lock requires an upward conversion. Sigh. Try to get the conversion.
|
|
|
|
If it fails, release the lock and re-seize. Save the contents of the
|
|
|
|
status vector just in case */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
const USHORT must_read = (lock->lck_logical < LCK_read) ? 1 : 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
ISC_STATUS_ARRAY alt_status;
|
2001-05-23 15:26:42 +02:00
|
|
|
memcpy(alt_status, tdbb->tdbb_status_vector, sizeof(alt_status));
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (LCK_convert_opt(tdbb, lock, lock_type)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return must_read;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else if (wait == LCK_NO_WAIT) {
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, true, false, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
memcpy(tdbb->tdbb_status_vector, alt_status, sizeof(alt_status));
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (PAGE_LOCK(lock, lock_type, wait)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return 1;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-07-10 19:35:13 +02:00
|
|
|
/* Case: a timeout was specified, or the caller didn't want to wait,
|
2001-05-23 15:26:42 +02:00
|
|
|
return the error. */
|
|
|
|
|
2003-11-11 13:19:20 +01:00
|
|
|
if ((wait < 0) && (status[1] == isc_lock_timeout))
|
2003-05-16 22:35:19 +02:00
|
|
|
{
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, false, false, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2001-07-10 19:35:13 +02:00
|
|
|
/* Case: lock manager detected a deadlock, probably caused by locking the
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc's in an unfortunate order. Nothing we can do about it, return the
|
2002-09-20 12:23:15 +02:00
|
|
|
error, and log it to firebird.log. */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
gds__msg_format(0, JRD_BUGCHK, 216, sizeof(errmsg), errmsg,
|
2004-01-21 08:18:30 +01:00
|
|
|
(TEXT*) (IPTR) bdb->bdb_page,
|
|
|
|
(TEXT*) (IPTR) page_type, 0, 0, 0);
|
2003-11-11 13:19:20 +01:00
|
|
|
IBERR_append_status(status, isc_random, isc_arg_string,
|
2001-05-23 15:26:42 +02:00
|
|
|
ERR_cstring(errmsg), 0);
|
|
|
|
ERR_log(JRD_BUGCHK, 216, errmsg); /* msg 216 page %ld, page type %ld lock denied */
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
return 0; /* Added to get rid of Compiler Warning */
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static ULONG memory_init(thread_db* tdbb, BCB bcb, ULONG number)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* m e m o r y _ i n i t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Initialize memory for the cache.
|
|
|
|
* Return number of buffers allocated.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
UCHAR* memory = 0;
|
|
|
|
SLONG buffers = 0;
|
|
|
|
const SLONG page_size = (SLONG) dbb->dbb_page_size;
|
|
|
|
SLONG memory_size = page_size * (number + 1);
|
|
|
|
|
|
|
|
SLONG old_buffers = 0;
|
|
|
|
bcb_repeat* old_tail = 0;
|
|
|
|
const UCHAR* memory_end = 0;
|
|
|
|
bcb_repeat* tail = bcb->bcb_rpt;
|
|
|
|
// "end" is changed inside the loop
|
|
|
|
for (const bcb_repeat* end = tail + number; tail < end; tail++) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!memory) {
|
|
|
|
/* Allocate only what is required for remaining buffers. */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (memory_size > (SLONG) (page_size * (number + 1))) {
|
2001-05-23 15:26:42 +02:00
|
|
|
memory_size = page_size * (number + 1);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-01-16 18:47:10 +01:00
|
|
|
do {
|
|
|
|
try {
|
2003-01-18 19:02:12 +01:00
|
|
|
memory = (UCHAR *)gds__alloc(memory_size);
|
2003-01-16 18:47:10 +01:00
|
|
|
break;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
catch(const std::exception&) {
|
2003-01-16 18:47:10 +01:00
|
|
|
/* Either there's not enough virtual memory or there is
|
|
|
|
but it's not virtually contiguous. Let's find out by
|
|
|
|
cutting the size in half to see if the buffers can be
|
|
|
|
scattered over the remaining virtual address space. */
|
|
|
|
memory_size >>= 1;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (memory_size < MIN_BUFFER_SEGMENT) {
|
|
|
|
/* Diminishing returns */
|
2003-01-16 18:47:10 +01:00
|
|
|
return buffers;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2003-01-16 18:47:10 +01:00
|
|
|
}
|
2003-12-11 11:33:30 +01:00
|
|
|
} while (true);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
LLS_PUSH(memory, &bcb->bcb_memory);
|
|
|
|
memory_end = memory + memory_size;
|
|
|
|
|
|
|
|
/* Allocate buffers on an address that is an even multiple
|
|
|
|
of the page size (rather the physical sector size.) This
|
|
|
|
is a necessary condition to support raw I/O interfaces. */
|
|
|
|
memory = (UCHAR *) (((U_IPTR) memory + page_size - 1) &
|
|
|
|
~((int) page_size - 1));
|
|
|
|
old_tail = tail;
|
|
|
|
old_buffers = buffers;
|
|
|
|
}
|
|
|
|
|
|
|
|
QUE_INIT(tail->bcb_page_mod);
|
|
|
|
|
|
|
|
if (!(tail->bcb_bdb = alloc_bdb(tdbb, bcb, &memory))) {
|
|
|
|
/* Whoops! Time to reset our expectations. Release the
|
|
|
|
buffer memory but use that memory size to calculate
|
|
|
|
a new number that takes into account the page buffer
|
|
|
|
overhead. Reduce this number by a 25% fudge factor to
|
|
|
|
leave some memory for useful work. */
|
|
|
|
|
2003-01-18 19:02:12 +01:00
|
|
|
gds__free(LLS_POP(&bcb->bcb_memory));
|
2001-05-23 15:26:42 +02:00
|
|
|
memory = 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
for (bcb_repeat* tail2 = old_tail; tail2 < tail; tail2++) {
|
2001-05-23 15:26:42 +02:00
|
|
|
tail2->bcb_bdb = dealloc_bdb(tail2->bcb_bdb);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
number = memory_size / PAGE_OVERHEAD;
|
|
|
|
number -= number >> 2;
|
|
|
|
end = old_tail + number;
|
|
|
|
tail = --old_tail; /* For loop continue pops tail above */
|
|
|
|
buffers = old_buffers;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
buffers++; /* Allocated buffers */
|
|
|
|
number--; /* Remaining buffers */
|
|
|
|
|
|
|
|
/* Check if memory segment has been exhausted. */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (memory + page_size > memory_end) {
|
2001-05-23 15:26:42 +02:00
|
|
|
memory = 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return buffers;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static void page_validation_error(thread_db* tdbb, WIN * window, SSHORT type)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a g e _ v a l i d a t i o n _ e r r o r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* We've detected a validation error on fetch. Generally
|
|
|
|
* we've detected that the type of page fetched didn't match the
|
|
|
|
* type of page we were expecting. Report an error and
|
|
|
|
* get out.
|
|
|
|
* This function will only be called rarely, as a page validation
|
|
|
|
* error is an indication of on-disk database corruption.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = window->win_bdb;
|
2004-01-03 11:59:52 +01:00
|
|
|
const pag* page = bdb->bdb_buffer;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
IBERR_build_status(tdbb->tdbb_status_vector,
|
2003-11-11 13:19:20 +01:00
|
|
|
isc_db_corrupt,
|
|
|
|
isc_arg_string, "",
|
|
|
|
isc_arg_gds, isc_page_type_err,
|
|
|
|
isc_arg_gds, isc_badpagtyp,
|
|
|
|
isc_arg_number, (SLONG) bdb->bdb_page,
|
|
|
|
isc_arg_number, (SLONG) type,
|
|
|
|
isc_arg_number, (SLONG) page->pag_type, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
/* We should invalidate this bad buffer. */
|
2004-02-20 07:43:27 +01:00
|
|
|
CCH_unwind(tdbb, true);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef CACHE_READER
|
2004-02-20 07:43:27 +01:00
|
|
|
static void prefetch_epilogue(Prefetch* prefetch, ISC_STATUS* status_vector)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p r e f e t c h _ e p i l o g u e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Stall on asynchronous I/O completion.
|
2001-07-10 19:35:13 +02:00
|
|
|
* Move data from prefetch buffer to database
|
2001-05-23 15:26:42 +02:00
|
|
|
* buffers, compute the checksum, and release
|
|
|
|
* the latch.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!(prefetch->prf_flags & PRF_active)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
THREAD_EXIT;
|
|
|
|
prefetch->prf_piob.piob_wait = TRUE;
|
2004-02-20 07:43:27 +01:00
|
|
|
const bool async_status = PIO_status(&prefetch->prf_piob, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
THREAD_ENTER;
|
|
|
|
|
|
|
|
/* If there was an I/O error release all buffer latches acquired
|
|
|
|
for the prefetch request. */
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb = prefetch->prf_tdbb;
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
if (!async_status) {
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc** next_bdb = prefetch->prf_bdbs;
|
2004-01-03 11:59:52 +01:00
|
|
|
for (USHORT i = 0; i < prefetch->prf_max_prefetch; i++) {
|
|
|
|
if (*next_bdb) {
|
|
|
|
release_bdb(tdbb, *next_bdb, true, false, false);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
next_bdb++;
|
|
|
|
}
|
|
|
|
prefetch->prf_flags &= ~PRF_active;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
SCHAR* next_buffer = prefetch->prf_io_buffer;
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc** next_bdb = prefetch->prf_bdbs;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
for (USHORT i = 0; i < prefetch->prf_max_prefetch; i++) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (*next_bdb) {
|
2004-01-03 11:59:52 +01:00
|
|
|
pag* page = (*next_bdb)->bdb_buffer;
|
|
|
|
if (next_buffer != reinterpret_cast<char*>(page)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
MOVE_FASTER(next_buffer, (SCHAR *) page,
|
|
|
|
(ULONG) dbb->dbb_page_size);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
if (page->pag_checksum == CCH_checksum(*next_bdb)) {
|
|
|
|
(*next_bdb)->bdb_flags &= ~(BDB_read_pending | BDB_not_valid);
|
|
|
|
(*next_bdb)->bdb_flags |= BDB_prefetch;
|
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, *next_bdb, true, false, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
next_buffer += dbb->dbb_page_size;
|
|
|
|
next_bdb++;
|
|
|
|
}
|
|
|
|
|
|
|
|
prefetch->prf_flags &= ~PRF_active;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static void prefetch_init(Prefetch* prefetch, thread_db* tdbb)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p r e f e t c h _ i n i t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Initialize prefetch data structure.
|
|
|
|
* Most systems that allow access to "raw" I/O
|
|
|
|
* interfaces want the buffer address aligned.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
prefetch->prf_tdbb = tdbb;
|
|
|
|
prefetch->prf_flags = 0;
|
|
|
|
prefetch->prf_max_prefetch = PREFETCH_MAX_TRANSFER / dbb->dbb_page_size;
|
|
|
|
prefetch->prf_aligned_buffer =
|
|
|
|
(SCHAR
|
|
|
|
*) (((U_IPTR) & prefetch->prf_unaligned_buffer + MIN_PAGE_SIZE -
|
|
|
|
1) & ~((U_IPTR) MIN_PAGE_SIZE - 1));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
static void prefetch_io(Prefetch* prefetch, ISC_STATUS* status_vector)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p r e f e t c h _ i o
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Queue an asynchronous I/O to read
|
|
|
|
* multiple pages into prefetch buffer.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb = prefetch->prf_tdbb;
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!prefetch->prf_page_count) {
|
2001-05-23 15:26:42 +02:00
|
|
|
prefetch->prf_flags &= ~PRF_active;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else {
|
|
|
|
/* Get the cache reader working on our behalf too */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!(dbb->dbb_bcb->bcb_flags & BCB_reader_active)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
ISC_event_post(dbb->dbb_reader_event);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
THREAD_EXIT;
|
2004-02-20 07:43:27 +01:00
|
|
|
const bool async_status =
|
2001-05-23 15:26:42 +02:00
|
|
|
PIO_read_ahead(dbb, prefetch->prf_start_page,
|
|
|
|
prefetch->prf_io_buffer, prefetch->prf_page_count,
|
|
|
|
&prefetch->prf_piob, status_vector);
|
|
|
|
THREAD_ENTER;
|
2004-02-20 07:43:27 +01:00
|
|
|
if (!async_status) {
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc** next_bdb = prefetch->prf_bdbs;
|
2004-01-03 11:59:52 +01:00
|
|
|
for (USHORT i = 0; i < prefetch->prf_max_prefetch; i++) {
|
|
|
|
if (*next_bdb) {
|
|
|
|
release_bdb(tdbb, *next_bdb, true, false, false);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
next_bdb++;
|
|
|
|
}
|
|
|
|
prefetch->prf_flags &= ~PRF_active;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
static void prefetch_prologue(Prefetch* prefetch, SLONG* start_page)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p r e f e t c h _ p r o l o g u e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Search for consecutive pages to be prefetched
|
|
|
|
* and latch them for I/O.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb = prefetch->prf_tdbb;
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2004-01-03 11:59:52 +01:00
|
|
|
BCB bcb = dbb->dbb_bcb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
prefetch->prf_start_page = *start_page;
|
|
|
|
prefetch->prf_page_count = 0;
|
|
|
|
prefetch->prf_flags |= PRF_active;
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc** next_bdb = prefetch->prf_bdbs;
|
2004-01-03 11:59:52 +01:00
|
|
|
for (USHORT i = 0; i < prefetch->prf_max_prefetch; i++) {
|
2001-05-23 15:26:42 +02:00
|
|
|
*next_bdb = 0;
|
|
|
|
if (SBM_clear(bcb->bcb_prefetch, *start_page) &&
|
2004-01-03 11:59:52 +01:00
|
|
|
(*next_bdb = get_buffer(tdbb, *start_page, LATCH_shared, 0)))
|
|
|
|
{
|
|
|
|
if ((*next_bdb)->bdb_flags & BDB_read_pending) {
|
2001-05-23 15:26:42 +02:00
|
|
|
prefetch->prf_page_count = i + 1;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else {
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, *next_bdb, true, false, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
*next_bdb = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
next_bdb++;
|
|
|
|
(*start_page)++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Optimize non-sequential list prefetching to transfer
|
|
|
|
directly to database buffers. */
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (prefetch->prf_page_count == 1 && (bdb = prefetch->prf_bdbs[0])) {
|
2002-04-29 17:05:11 +02:00
|
|
|
prefetch->prf_io_buffer = reinterpret_cast<char*>(bdb->bdb_buffer);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
prefetch->prf_io_buffer = prefetch->prf_aligned_buffer;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Reset starting page for next bitmap walk */
|
|
|
|
|
|
|
|
--(*start_page);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static SSHORT related(const Buffer_desc* low, const Buffer_desc* high, SSHORT limit)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e l a t e d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* See if there are precedence relationships linking two buffers.
|
|
|
|
* Since precedence graphs can become very complex, limit search for
|
|
|
|
* precedence relationship by visiting a presribed limit of higher
|
|
|
|
* precedence blocks.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-11 06:04:26 +01:00
|
|
|
const struct que* base = &low->bdb_higher;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
for (const struct que* que = base->que_forward; que != base; que = que->que_forward)
|
|
|
|
{
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!--limit) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return PRE_UNKNOWN;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2004-03-11 06:04:26 +01:00
|
|
|
const Precedence* precedence = BLOCK(que, Precedence*, pre_higher);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!(precedence->pre_flags & PRE_cleared)) {
|
2004-01-03 11:59:52 +01:00
|
|
|
if (precedence->pre_hi == high) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return PRE_EXISTS;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
limit = related(precedence->pre_hi, high, limit);
|
2004-01-03 11:59:52 +01:00
|
|
|
if (limit == PRE_EXISTS || limit == PRE_UNKNOWN) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return limit;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return limit;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef PAGE_LATCHING
|
|
|
|
static void release_bdb(
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb,
|
|
|
|
Buffer_desc* bdb,
|
2004-01-03 11:59:52 +01:00
|
|
|
const bool repost,
|
|
|
|
const bool downgrade_latch, const bool rel_mark_latch)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e l e a s e _ b d b
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2004-03-11 06:04:26 +01:00
|
|
|
* Decrement the use count of a Buffer_desc, reposting
|
2001-05-23 15:26:42 +02:00
|
|
|
* blocking AST if required.
|
2004-01-03 11:59:52 +01:00
|
|
|
* If rel_mark_latch is true, the the value of downgrade_latch is ignored.
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
2004-01-03 11:59:52 +01:00
|
|
|
SSHORT i;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
LATCH_MUTEX_ACQUIRE;
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
QUE wait_que = &bdb->bdb_waiters;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Releasing a LATCH_mark. */
|
|
|
|
if (rel_mark_latch) {
|
2004-01-03 11:59:52 +01:00
|
|
|
if ((bdb->bdb_io != tdbb) || (bdb->bdb_exclusive != tdbb)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
cache_bugcheck(294); /* inconsistent LATCH_mark release */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_io = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
/* Downgrading from an exlusive to a shared latch. */
|
|
|
|
if (downgrade_latch) {
|
|
|
|
/* Only the transition from exclusive to shared is supported.
|
|
|
|
If an actual state changed, then we need to check if waiters
|
|
|
|
can be granted. Otherwise, there is nothing further to do. */
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_io == tdbb) {
|
2001-05-23 15:26:42 +02:00
|
|
|
cache_bugcheck(296); /* inconsistent latch downgrade call */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
if (bdb->bdb_exclusive == tdbb) {
|
|
|
|
bdb->bdb_exclusive = 0;
|
2004-03-11 06:04:26 +01:00
|
|
|
for (i = 0; bdb->bdb_shared[i]; i++); // empty loop body
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_shared[i] = tdbb;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
/* If the exclusive latch is held, then certain code does funny things:
|
|
|
|
ail.c does: exclusive - mark - exclusive */
|
|
|
|
if (bdb->bdb_exclusive == tdbb) {
|
|
|
|
--bdb->bdb_use_count;
|
|
|
|
if (!bdb->bdb_use_count) { /* All latches are released */
|
|
|
|
bdb->bdb_exclusive = bdb->bdb_io = 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
for (i = 0; i < BDB_max_shared; bdb->bdb_shared[i++] = 0); // null loop body
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else if (bdb->bdb_io) { /* This is a release for an io or an exclusive latch */
|
|
|
|
if (bdb->bdb_io == tdbb) { /* We have an io latch */
|
|
|
|
|
|
|
|
/* The BDB_must_write flag causes the system to latch for io, in addition
|
|
|
|
to the already owned latches. Make sure not to disturb an already existing
|
|
|
|
exclusive latch. */
|
|
|
|
/* ail.c does: EX => MARK => SHARED => release => EX => MARK => RELEASE => EX */
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!(bdb->bdb_flags & BDB_marked)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_io = 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
else if (bdb->bdb_use_count == 1) {
|
2001-05-23 15:26:42 +02:00
|
|
|
/* This must be a release for our exclusive latch */
|
|
|
|
bdb->bdb_exclusive = 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else { /* This is a release for a shared latch */
|
|
|
|
for (i = 0; (i < BDB_max_shared) && (bdb->bdb_shared[i] != tdbb);
|
2004-01-03 11:59:52 +01:00
|
|
|
i++); // null loop body
|
|
|
|
if (i < BDB_max_shared) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_shared[i] = 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
/* If the exclusive latch is not held, then things have to behave much nicer. */
|
|
|
|
{
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_flags & BDB_marked) {
|
2001-05-23 15:26:42 +02:00
|
|
|
cache_bugcheck(297); /* bdb is unexpectedly marked */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
--bdb->bdb_use_count;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_io == tdbb) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_io = 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else {
|
|
|
|
for (i = 0; (i < BDB_max_shared) && (bdb->bdb_shared[i] != tdbb);
|
2004-01-03 11:59:52 +01:00
|
|
|
i++); // null loop body
|
|
|
|
if (i >= BDB_max_shared) {
|
2001-05-23 15:26:42 +02:00
|
|
|
cache_bugcheck(300); /* can't find shared latch */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_shared[i] = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
bool granted = false;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
for (QUE que = wait_que->que_forward; que != wait_que; que = que->que_forward)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Note that this loop assumes that requests for LATCH_io and LATCH_mark
|
|
|
|
are queued before LATCH_shared and LATCH_exclusive. */
|
2004-02-20 07:43:27 +01:00
|
|
|
Latch_wait* lwt = BLOCK(que, Latch_wait*, lwt_waiters);
|
|
|
|
if (lwt->lwt_flags & LWT_pending) {
|
|
|
|
switch (lwt->lwt_latch) {
|
2004-01-03 11:59:52 +01:00
|
|
|
case LATCH_exclusive:
|
2001-05-23 15:26:42 +02:00
|
|
|
if (bdb->bdb_use_count) {
|
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
++bdb->bdb_use_count;
|
2004-02-20 07:43:27 +01:00
|
|
|
bdb->bdb_exclusive = lwt->lwt_tdbb;
|
|
|
|
lwt->lwt_flags &= ~LWT_pending;
|
|
|
|
ISC_event_post(&lwt->lwt_event);
|
2001-05-23 15:26:42 +02:00
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
case LATCH_io:
|
|
|
|
if (bdb->bdb_io) {
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else {
|
|
|
|
++bdb->bdb_use_count;
|
2004-02-20 07:43:27 +01:00
|
|
|
bdb->bdb_io = lwt->lwt_tdbb;
|
|
|
|
lwt->lwt_flags &= ~LWT_pending;
|
|
|
|
ISC_event_post(&lwt->lwt_event);
|
2004-01-03 11:59:52 +01:00
|
|
|
granted = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
case LATCH_mark:
|
2004-02-20 07:43:27 +01:00
|
|
|
if (bdb->bdb_exclusive != lwt->lwt_tdbb) {
|
2001-05-23 15:26:42 +02:00
|
|
|
cache_bugcheck(298); /* missing exclusive latch */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
if (bdb->bdb_io) {
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else {
|
2004-02-20 07:43:27 +01:00
|
|
|
bdb->bdb_io = lwt->lwt_tdbb;
|
|
|
|
lwt->lwt_flags &= ~LWT_pending;
|
|
|
|
ISC_event_post(&lwt->lwt_event);
|
2004-01-03 11:59:52 +01:00
|
|
|
granted = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
case LATCH_shared:
|
2001-05-23 15:26:42 +02:00
|
|
|
if (bdb->bdb_exclusive) {
|
|
|
|
break; /* defensive programming */
|
2004-01-03 11:59:52 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* correct programming
|
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
return; */
|
|
|
|
}
|
|
|
|
for (i = 0; (i < BDB_max_shared) && bdb->bdb_shared[i]; i++);
|
2004-01-03 11:59:52 +01:00
|
|
|
if (i >= BDB_max_shared) {
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
++bdb->bdb_use_count;
|
2004-02-20 07:43:27 +01:00
|
|
|
bdb->bdb_shared[i] = lwt->lwt_tdbb;
|
|
|
|
lwt->lwt_flags &= ~LWT_pending;
|
|
|
|
ISC_event_post(&lwt->lwt_event);
|
2004-01-03 11:59:52 +01:00
|
|
|
granted = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (granted && (bdb->bdb_flags & BDB_read_pending)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Allow only one reader to proceed */
|
|
|
|
break;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (bdb->bdb_use_count || !repost) {
|
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
LATCH_MUTEX_RELEASE;
|
|
|
|
|
|
|
|
if (bdb->bdb_ast_flags & BDB_blocking)
|
2003-08-13 13:11:12 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
PAGE_LOCK_RE_POST(bdb->bdb_lock);
|
2003-08-13 13:11:12 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#else
|
|
|
|
static void release_bdb(
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb,
|
|
|
|
Buffer_desc* bdb,
|
2004-01-03 11:59:52 +01:00
|
|
|
const bool repost,
|
|
|
|
const bool downgrade_latch, const bool rel_mark_latch)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e l e a s e _ b d b
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2004-03-11 06:04:26 +01:00
|
|
|
* Decrement the use count of a Buffer_desc, reposting
|
2001-05-23 15:26:42 +02:00
|
|
|
* blocking AST if required.
|
|
|
|
* If rel_mark_latch is TRUE, the the value of downgrade_latch is ignored.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
/* Releasing a LATCH_mark. */
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (rel_mark_latch) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_io = 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
|
|
|
/* Downgrading from an exlusive to a shared latch. */
|
2004-01-03 11:59:52 +01:00
|
|
|
if (downgrade_latch) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_exclusive = 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
|
|
|
/* If the exclusive latch is held, then certain code does funny things:
|
|
|
|
ail.c does: exclusive - mark - exclusive */
|
|
|
|
if (bdb->bdb_exclusive == tdbb) {
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!--bdb->bdb_use_count) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_exclusive = bdb->bdb_io = 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else if (bdb->bdb_io) { /* This is a release for an io or an exclusive latch */
|
|
|
|
if (bdb->bdb_io == tdbb) { /* We have an io latch */
|
|
|
|
|
|
|
|
/* The BDB_must_write flag causes the system to latch for io, in addition
|
|
|
|
to the already owned latches. Make sure not to disturb an already existing
|
|
|
|
exclusive latch. */
|
|
|
|
/* ail.c does: EX => MARK => SHARED => release => EX => MARK => RELEASE => EX */
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!(bdb->bdb_flags & BDB_marked)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_io = 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
else if (bdb->bdb_use_count == 1) {
|
2001-05-23 15:26:42 +02:00
|
|
|
/* This must be a release for our exclusive latch */
|
|
|
|
bdb->bdb_exclusive = 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
/* If the exclusive latch is not held, then things have to behave much nicer. */
|
|
|
|
{
|
|
|
|
--bdb->bdb_use_count;
|
|
|
|
bdb->bdb_io = 0;
|
|
|
|
}
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_use_count || !repost) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_ast_flags & BDB_blocking) {
|
2001-05-23 15:26:42 +02:00
|
|
|
PAGE_LOCK_RE_POST(bdb->bdb_lock);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static bool writeable(const Buffer_desc* bdblock)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* w r i t e a b l e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* See if a buffer is writeable. A buffer is writeable if
|
|
|
|
* neither it nor any of it's higher precedence cousins are
|
|
|
|
* marked for write.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdblock->bdb_flags & BDB_marked) {
|
2003-12-22 11:00:59 +01:00
|
|
|
return false;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If there are buffers that must be written first, check them, too. */
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
for (const que* queue = bdblock->bdb_higher.que_forward;
|
|
|
|
queue != &bdblock->bdb_higher; queue = queue->que_forward)
|
|
|
|
{
|
2004-02-20 07:43:27 +01:00
|
|
|
const Precedence* precedence = BLOCK(queue, Precedence*, pre_higher);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!(precedence->pre_flags & PRE_cleared) &&
|
2003-12-22 11:00:59 +01:00
|
|
|
!writeable(precedence->pre_hi))
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
return true;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int write_buffer(
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb,
|
|
|
|
Buffer_desc* bdb,
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG page,
|
2003-12-22 11:00:59 +01:00
|
|
|
const bool write_thru,
|
2004-02-20 07:43:27 +01:00
|
|
|
ISC_STATUS* status, const bool write_this_page)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* w r i t e _ b u f f e r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Write a dirty buffer. This may recurse due to
|
|
|
|
* precedence problems.
|
|
|
|
*
|
2001-07-10 19:35:13 +02:00
|
|
|
* input: write_this_page
|
2001-05-23 15:26:42 +02:00
|
|
|
* = true if the input page needs to be written
|
|
|
|
* before returning. (normal case)
|
2001-07-10 19:35:13 +02:00
|
|
|
* = false if the input page is being written
|
2001-05-23 15:26:42 +02:00
|
|
|
* because of precedence. Only write
|
|
|
|
* one page and return so that the caller
|
|
|
|
* can re-establish the need to write this
|
|
|
|
* page.
|
2001-07-10 19:35:13 +02:00
|
|
|
*
|
2001-05-23 15:26:42 +02:00
|
|
|
* return: 0 = Write failed.
|
|
|
|
* 1 = Page is written. Page was written by this
|
|
|
|
* call, or was written by someone else, or the
|
|
|
|
* cache buffer is already reassigned.
|
2003-12-22 11:00:59 +01:00
|
|
|
* 2 = Only possible if write_this_page is false.
|
2001-07-10 19:35:13 +02:00
|
|
|
* This input page is not written. One
|
2001-05-23 15:26:42 +02:00
|
|
|
* page higher in precedence is written
|
|
|
|
* though. Probable action: re-establich the
|
|
|
|
* need to write this page and retry write.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (latch_bdb(tdbb, LATCH_io, bdb, page, 1) == -1) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return 1;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if ((bdb->bdb_flags & BDB_marked) && !(bdb->bdb_flags & BDB_faked)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
cache_bugcheck(217); /* msg 217 buffer marked for update */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!(bdb->bdb_flags & BDB_dirty) &&
|
2003-12-22 11:00:59 +01:00
|
|
|
!(write_thru && bdb->bdb_flags & BDB_db_dirty))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
clear_precedence(dbb, bdb);
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, true, false, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If there are buffers that must be written first, write them now. */
|
|
|
|
|
|
|
|
PRE_MUTEX_ACQUIRE;
|
|
|
|
|
|
|
|
while (QUE_NOT_EMPTY(bdb->bdb_higher)) {
|
2004-01-03 11:59:52 +01:00
|
|
|
BCB bcb = dbb->dbb_bcb; /* Re-initialize in the loop */
|
|
|
|
QUE que = bdb->bdb_higher.que_forward;
|
2004-02-20 07:43:27 +01:00
|
|
|
Precedence* precedence = BLOCK(que, Precedence*, pre_higher);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (precedence->pre_flags & PRE_cleared) {
|
|
|
|
QUE_DELETE(precedence->pre_higher);
|
|
|
|
QUE_DELETE(precedence->pre_lower);
|
2004-03-11 06:04:26 +01:00
|
|
|
precedence->pre_hi = (Buffer_desc*) bcb->bcb_free;
|
2001-05-23 15:26:42 +02:00
|
|
|
bcb->bcb_free = precedence;
|
|
|
|
}
|
|
|
|
else {
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* hi_bdb = precedence->pre_hi;
|
2004-01-03 11:59:52 +01:00
|
|
|
const SLONG hi_page = hi_bdb->bdb_page;
|
2001-05-23 15:26:42 +02:00
|
|
|
PRE_MUTEX_RELEASE;
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, false, false, false);
|
|
|
|
const int write_status =
|
2001-05-23 15:26:42 +02:00
|
|
|
write_buffer(tdbb, hi_bdb, hi_page, write_thru, status,
|
2003-12-22 11:00:59 +01:00
|
|
|
false);
|
2004-01-03 11:59:52 +01:00
|
|
|
if (write_status == 0) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return 0; /* return IO error */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef SUPERSERVER
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!write_this_page) {
|
|
|
|
return 2;
|
|
|
|
/* caller wants to re-establish the need for this write after one precedence write */
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2004-01-03 11:59:52 +01:00
|
|
|
if (latch_bdb(tdbb, LATCH_io, bdb, page, 1) == -1) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return 1; /* cache buffer reassigned, return 'write successful' */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
PRE_MUTEX_ACQUIRE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
PRE_MUTEX_RELEASE;
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER_V2
|
|
|
|
/* Header page I/O is deferred until a dirty page, which was modified by a
|
|
|
|
transaction not updated on the header page, needs to be written. Note
|
|
|
|
that the header page needs to be written with the target page latched to
|
|
|
|
prevent younger transactions from modifying the target page. */
|
|
|
|
|
|
|
|
if (page != HEADER_PAGE
|
2003-12-22 11:00:59 +01:00
|
|
|
&& bdb->bdb_mark_transaction > dbb->dbb_last_header_write)
|
|
|
|
{
|
|
|
|
TRA_header_write(tdbb, dbb, bdb->bdb_mark_transaction);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Unless the buffer has been faked (recently re-allocated), write
|
|
|
|
out the page */
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
bool result = true;
|
|
|
|
if ((bdb->bdb_flags & BDB_dirty
|
2001-05-23 15:26:42 +02:00
|
|
|
|| (write_thru && bdb->bdb_flags & BDB_db_dirty))
|
2003-12-22 11:00:59 +01:00
|
|
|
&& !(bdb->bdb_flags & BDB_marked))
|
|
|
|
{
|
2004-01-03 11:59:52 +01:00
|
|
|
if ( (result = write_page(tdbb, bdb, write_thru, status, false)) ) {
|
2001-05-23 15:26:42 +02:00
|
|
|
clear_precedence(dbb, bdb);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-01-03 11:59:52 +01:00
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
clear_precedence(dbb, bdb);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
release_bdb(tdbb, bdb, true, false, false);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!result) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return 0;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
|
|
|
#ifdef SUPERSERVER
|
2004-01-03 11:59:52 +01:00
|
|
|
if (!write_this_page) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return 2;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
|
|
|
#endif
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
static bool write_page(
|
2004-03-11 06:04:26 +01:00
|
|
|
thread_db* tdbb,
|
|
|
|
Buffer_desc* bdb,
|
2003-12-22 11:00:59 +01:00
|
|
|
const bool write_thru, ISC_STATUS* status,
|
|
|
|
const bool inAst)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* w r i t e _ p a g e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Do actions required when writing a database page,
|
|
|
|
* including journaling, shadowing.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
if (bdb->bdb_flags & BDB_not_valid) {
|
2003-11-11 13:19:20 +01:00
|
|
|
*status++ = isc_arg_gds;
|
|
|
|
*status++ = isc_buf_invalid;
|
|
|
|
*status++ = isc_arg_number;
|
2001-05-23 15:26:42 +02:00
|
|
|
*status++ = bdb->bdb_page;
|
2003-11-11 13:19:20 +01:00
|
|
|
*status++ = isc_arg_end;
|
2003-12-22 11:00:59 +01:00
|
|
|
return false;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
bool result = true;
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = bdb->bdb_dbb;
|
2003-12-22 11:00:59 +01:00
|
|
|
pag* page = bdb->bdb_buffer;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Before writing db header page, make sure that the next_transaction > oldest_active
|
|
|
|
transaction */
|
|
|
|
if (bdb->bdb_page == HEADER_PAGE) {
|
2004-02-20 07:43:27 +01:00
|
|
|
header_page* header = (header_page*) page;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (header->hdr_next_transaction) {
|
2004-01-03 11:59:52 +01:00
|
|
|
if (header->hdr_oldest_active > header->hdr_next_transaction) {
|
2001-05-23 15:26:42 +02:00
|
|
|
BUGCHECK(266); /*next transaction older than oldest active */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (header->hdr_oldest_transaction > header->hdr_next_transaction) {
|
2001-05-23 15:26:42 +02:00
|
|
|
BUGCHECK(267); /* next transaction older than oldest transaction */
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
page->pag_generation++;
|
|
|
|
|
2004-01-06 11:33:18 +01:00
|
|
|
// if (!dbb->dbb_wal || write_thru) becomes
|
|
|
|
// if (true || write_thru) then finally if (true)
|
|
|
|
// I won't wipe out the if() itself to allow my changes be verified easily by others
|
|
|
|
if (true) {
|
2001-05-23 15:26:42 +02:00
|
|
|
AST_CHECK;
|
|
|
|
dbb->dbb_writes++;
|
|
|
|
|
2001-07-10 19:35:13 +02:00
|
|
|
/* write out page to main database file, and to any
|
2001-05-23 15:26:42 +02:00
|
|
|
shadows, making a special case of the header page */
|
|
|
|
|
2003-09-08 22:23:46 +02:00
|
|
|
if (bdb->bdb_page >= 0) {
|
|
|
|
if (bdb->bdb_write_direction == BDB_write_undefined) {
|
|
|
|
dbb->dbb_flags |= DBB_bugcheck;
|
|
|
|
status[0] = isc_arg_gds;
|
2003-11-11 13:19:20 +01:00
|
|
|
status[1] = isc_bug_check;
|
|
|
|
status[2] = isc_arg_string;
|
2003-09-08 22:23:46 +02:00
|
|
|
status[3] = (ISC_STATUS)ERR_cstring("Undefined page write direction");
|
2003-11-11 13:19:20 +01:00
|
|
|
status[4] = isc_arg_end;
|
2003-12-22 11:00:59 +01:00
|
|
|
return false;
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
page->pag_checksum = CCH_checksum(bdb);
|
2003-09-08 22:23:46 +02:00
|
|
|
if (bdb->bdb_write_direction == BDB_write_diff ||
|
|
|
|
(bdb->bdb_write_direction == BDB_write_both
|
|
|
|
#ifndef SUPERSERVER
|
|
|
|
&& bdb->bdb_diff_generation == dbb->backup_manager->get_current_generation()
|
|
|
|
#endif
|
|
|
|
))
|
2003-08-06 18:30:49 +02:00
|
|
|
{
|
2003-09-08 22:23:46 +02:00
|
|
|
#ifdef NBAK_DEBUG
|
|
|
|
// We cannot call normal trace functions here as they are signal-unsafe
|
|
|
|
char buffer[1000], *ptr = buffer;
|
2004-03-11 06:04:26 +01:00
|
|
|
strcpy(ptr, "NBAK,Write page ");
|
|
|
|
ptr += strlen(ptr);
|
|
|
|
gds__ulstr(ptr, bdb->bdb_page, 0, 0);
|
|
|
|
ptr += strlen(ptr);
|
|
|
|
strcpy(ptr, " at offset ");
|
|
|
|
ptr += strlen(ptr);
|
|
|
|
gds__ulstr(ptr, bdb->bdb_difference_page, 0, 0);
|
|
|
|
ptr += strlen(ptr);
|
2003-09-08 22:23:46 +02:00
|
|
|
strcpy(ptr, " in difference file");
|
|
|
|
gds__trace(buffer);
|
|
|
|
#endif
|
|
|
|
if (!dbb->backup_manager->write_difference(
|
|
|
|
status, bdb->bdb_difference_page, bdb->bdb_buffer))
|
2003-08-06 18:30:49 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_flags |= BDB_io_error;
|
|
|
|
dbb->dbb_flags |= DBB_suspend_bgio;
|
2003-12-22 11:00:59 +01:00
|
|
|
return false;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
2003-09-08 22:23:46 +02:00
|
|
|
if (bdb->bdb_write_direction == BDB_write_diff) {
|
2003-08-06 18:30:49 +02:00
|
|
|
// We finished. Adjust transaction accounting and get ready for exit
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_page == HEADER_PAGE) {
|
2003-08-06 18:30:49 +02:00
|
|
|
dbb->dbb_last_header_write =
|
2004-02-20 07:43:27 +01:00
|
|
|
((header_page*) page)->hdr_next_transaction;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2003-09-08 22:23:46 +02:00
|
|
|
set_write_direction(dbb, bdb, BDB_write_undefined);
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
|
|
|
else {
|
2003-08-06 18:30:49 +02:00
|
|
|
// We need to write our pages to main database files
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef SUPERSERVER
|
|
|
|
THREAD_EXIT;
|
|
|
|
#endif
|
2004-02-20 07:43:27 +01:00
|
|
|
jrd_file* file = dbb->dbb_file;
|
2003-08-06 18:30:49 +02:00
|
|
|
while (!PIO_write(file, bdb, page, status)) {
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
THREAD_ENTER;
|
|
|
|
#endif
|
|
|
|
if (!CCH_rollover_to_shadow(dbb, file, inAst)) {
|
|
|
|
bdb->bdb_flags |= BDB_io_error;
|
|
|
|
dbb->dbb_flags |= DBB_suspend_bgio;
|
2003-12-22 11:00:59 +01:00
|
|
|
return false;
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
THREAD_EXIT;
|
|
|
|
#endif
|
|
|
|
file = dbb->dbb_file;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef SUPERSERVER
|
2003-08-06 18:30:49 +02:00
|
|
|
THREAD_ENTER;
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_page == HEADER_PAGE) {
|
2003-08-06 18:30:49 +02:00
|
|
|
dbb->dbb_last_header_write =
|
2004-02-20 07:43:27 +01:00
|
|
|
((header_page*) page)->hdr_next_transaction;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2003-12-22 11:00:59 +01:00
|
|
|
if (dbb->dbb_shadow) {
|
2003-08-06 18:30:49 +02:00
|
|
|
result =
|
|
|
|
CCH_write_all_shadows(tdbb, 0, bdb, status, 0, inAst);
|
2003-12-22 11:00:59 +01:00
|
|
|
}
|
2003-09-08 22:23:46 +02:00
|
|
|
set_write_direction(dbb, bdb, BDB_write_undefined);
|
2003-08-06 18:30:49 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
if (result) {
|
|
|
|
#ifdef CACHE_WRITER
|
2004-01-03 11:59:52 +01:00
|
|
|
if (bdb->bdb_flags & BDB_checkpoint) {
|
2001-05-23 15:26:42 +02:00
|
|
|
--dbb->dbb_bcb->bcb_checkpoint;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
bdb->bdb_flags &= ~(BDB_db_dirty | BDB_checkpoint);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
AST_CHECK;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!result) {
|
|
|
|
/* If there was a write error then idle background threads
|
|
|
|
so that they don't spin trying to write these pages. This
|
|
|
|
usually results from device full errors which can be fixed
|
|
|
|
by someone freeing disk space. */
|
|
|
|
|
|
|
|
bdb->bdb_flags |= BDB_io_error;
|
|
|
|
dbb->dbb_flags |= DBB_suspend_bgio;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* clear the dirty bit vector, since the buffer is now
|
|
|
|
clean regardless of which transactions have modified it */
|
|
|
|
|
|
|
|
bdb->bdb_transactions = bdb->bdb_mark_transaction = 0;
|
|
|
|
if (!(dbb->dbb_bcb->bcb_flags & BCB_keep_pages) &&
|
|
|
|
(bdb->bdb_parent || bdb == dbb->dbb_bcb->bcb_btree))
|
2004-01-03 11:59:52 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
btc_remove(bdb);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_flags &= ~(BDB_dirty | BDB_must_write | BDB_system_dirty);
|
|
|
|
if (bdb->bdb_flags & BDB_io_error) {
|
|
|
|
/* If a write error has cleared, signal background threads
|
|
|
|
to resume their regular duties. If someone has freed up
|
|
|
|
disk space these errors will spontaneously go away. */
|
|
|
|
|
|
|
|
bdb->bdb_flags &= ~BDB_io_error;
|
|
|
|
dbb->dbb_flags &= ~DBB_suspend_bgio;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-11 06:04:26 +01:00
|
|
|
static void unmark(thread_db* tdbb, WIN * window)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* u n m a r k
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2004-03-11 06:04:26 +01:00
|
|
|
* Unmark a Buffer_desc. Called when the update of a page is
|
2001-05-23 15:26:42 +02:00
|
|
|
* complete and delaying the 'unmarking' could cause
|
|
|
|
* problems.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SET_TDBB(tdbb);
|
2004-03-11 06:04:26 +01:00
|
|
|
Buffer_desc* bdb = window->win_bdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
BLKCHK(bdb, type_bdb);
|
|
|
|
|
|
|
|
if (bdb->bdb_use_count == 1) {
|
2004-01-03 11:59:52 +01:00
|
|
|
const bool marked = (bdb->bdb_flags & BDB_marked) != 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
bdb->bdb_flags &= ~BDB_marked;
|
2004-01-03 11:59:52 +01:00
|
|
|
if (marked) {
|
|
|
|
release_bdb(tdbb, bdb, false, false, true);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|