8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-27 06:43:04 +01:00
firebird-mirror/src/jrd/DatabaseSnapshot.h
2009-08-13 15:08:50 +00:00

354 lines
7.3 KiB
C++

/*
* The contents of this file are subject to the Initial
* Developer's 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.ibphoenix.com/main.nfs?a=ibphoenix&page=ibp_idpl.
*
* Software distributed under the License is distributed AS IS,
* 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 Dmitry Yemanov
* for the Firebird Open Source RDBMS project.
*
* Copyright (c) 2006 Dmitry Yemanov <dimitr@users.sf.net>
* and all contributors signed below.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
*/
#ifndef JRD_DATABASE_SNAPSHOT_H
#define JRD_DATABASE_SNAPSHOT_H
#include "../common/classes/array.h"
#include "../common/classes/init.h"
namespace Jrd {
// forward declarations
class jrd_rel;
class RecordBuffer;
class RuntimeStatistics;
class DatabaseSnapshot
{
enum ValueType {VALUE_GLOBAL_ID, VALUE_INTEGER, VALUE_TIMESTAMP, VALUE_STRING};
struct DumpField
{
USHORT id;
ValueType type;
USHORT length;
void* data;
};
class DumpRecord
{
public:
DumpRecord() : offset(0), sizeLimit(0)
{}
explicit DumpRecord(int rel_id)
{
reset(rel_id);
}
void reset(int rel_id)
{
offset = 0;
sizeLimit = sizeof(buffer);
fb_assert(rel_id > 0 && rel_id <= int(MAX_UCHAR));
buffer[offset++] = (UCHAR) rel_id;
}
void assign(USHORT length, const UCHAR* ptr)
{
// CVC: While length is USHORT, this assertion is redundant.
// fb_assert(length <= MAX_FORMAT_SIZE); // commented - AP - avoid gcc warning
offset = 0;
sizeLimit = length;
memcpy(buffer, ptr, length);
}
ULONG getLength() const
{
return offset;
}
const UCHAR* getData() const
{
return buffer;
}
void storeGlobalId(int field_id, SINT64 value)
{
storeField(field_id, VALUE_GLOBAL_ID, sizeof(SINT64), &value);
}
void storeInteger(int field_id, SINT64 value)
{
storeField(field_id, VALUE_INTEGER, sizeof(SINT64), &value);
}
void storeTimestamp(int field_id, const Firebird::TimeStamp& value)
{
if (!value.isEmpty())
{
storeField(field_id, VALUE_TIMESTAMP, sizeof(ISC_TIMESTAMP), &value.value());
}
}
void storeString(int field_id, const Firebird::string& value)
{
if (value.length())
{
storeField(field_id, VALUE_STRING, value.length(), value.c_str());
}
}
void storeString(int field_id, const Firebird::PathName& value)
{
if (value.length())
{
storeField(field_id, VALUE_STRING, value.length(), value.c_str());
}
}
void storeString(int field_id, const Firebird::MetaName& value)
{
if (value.length())
{
storeField(field_id, VALUE_STRING, value.length(), value.c_str());
}
}
int getRelationId()
{
fb_assert(!offset);
return (ULONG) buffer[offset++];
}
bool getField(DumpField& field)
{
fb_assert(offset);
if (offset < sizeLimit)
{
field.id = (USHORT) buffer[offset++];
field.type = (ValueType) buffer[offset++];
fb_assert(field.type >= VALUE_GLOBAL_ID && field.type <= VALUE_STRING);
memcpy(&field.length, buffer + offset, sizeof(USHORT));
offset += sizeof(USHORT);
field.data = buffer + offset;
offset += field.length;
return true;
}
return false;
}
private:
void storeField(int field_id, ValueType type, size_t length, const void* value)
{
const size_t delta = sizeof(UCHAR) + sizeof(UCHAR) + sizeof(USHORT) + length;
if (offset + delta > sizeLimit)
{
fb_assert(false);
return;
}
UCHAR* ptr = buffer + offset;
fb_assert(field_id <= int(MAX_UCHAR));
*ptr++ = (UCHAR) field_id;
*ptr++ = (UCHAR) type;
const USHORT adjusted_length = (USHORT) length;
memcpy(ptr, &adjusted_length, sizeof(adjusted_length));
ptr += sizeof(USHORT);
memcpy(ptr, value, length);
offset += (ULONG) delta;
}
UCHAR buffer[MAX_FORMAT_SIZE];
ULONG offset;
ULONG sizeLimit;
};
struct RelationData
{
int rel_id;
RecordBuffer* data;
};
public:
class SharedData
{
static const ULONG MONITOR_VERSION = 3;
static const ULONG DEFAULT_SIZE = 1048576;
struct Header
{
ULONG version;
ULONG used;
ULONG allocated;
#ifndef WIN_NT
struct mtx mutex;
#endif
};
struct Element
{
SLONG processId;
SLONG localId;
ULONG length;
};
static ULONG alignOffset(ULONG absoluteOffset);
public:
explicit SharedData(const Database*);
~SharedData();
void acquire();
void release();
UCHAR* read(MemoryPool&, ULONG&);
ULONG setup();
void write(ULONG, ULONG, const void*);
void cleanup();
private:
// copying is prohibited
SharedData(const SharedData&);
SharedData& operator =(const SharedData&);
void ensureSpace(ULONG);
static void checkMutex(const TEXT*, int);
static void init(void*, sh_mem*, bool);
sh_mem handle;
#ifdef WIN_NT
struct mtx mutex;
#endif
Header* base;
const SLONG process_id;
const SLONG local_id;
};
private:
class DumpGuard
{
public:
explicit DumpGuard(SharedData* ptr)
: data(ptr)
{
data->acquire();
}
~DumpGuard()
{
data->release();
}
private:
DumpGuard(const DumpGuard&);
DumpGuard& operator=(const DumpGuard&);
SharedData* const data;
};
class Writer
{
public:
explicit Writer(SharedData* data)
: dump(data)
{
fb_assert(dump);
offset = dump->setup();
fb_assert(offset);
}
void putRecord(const DumpRecord& record)
{
const USHORT length = (USHORT) record.getLength();
dump->write(offset, sizeof(USHORT), &length);
dump->write(offset, length, record.getData());
}
private:
SharedData* dump;
ULONG offset;
};
class Reader
{
public:
Reader(ULONG length, UCHAR* ptr)
: sizeLimit(length), buffer(ptr), offset(0)
{}
bool getRecord(DumpRecord& record)
{
if (offset < sizeLimit)
{
USHORT length;
memcpy(&length, buffer + offset, sizeof(USHORT));
offset += sizeof(USHORT);
record.assign(length, buffer + offset);
offset += length;
return true;
}
return false;
}
private:
ULONG sizeLimit;
const UCHAR* buffer;
ULONG offset;
};
public:
~DatabaseSnapshot();
RecordBuffer* getData(const jrd_rel*) const;
static DatabaseSnapshot* create(thread_db*);
static int blockingAst(void*);
protected:
DatabaseSnapshot(thread_db*, MemoryPool&);
private:
RecordBuffer* allocBuffer(thread_db*, MemoryPool&, int);
void clearRecord(Record*);
void putField(thread_db*, Record*, const DumpField&, int&, bool = false);
static void dumpData(thread_db*);
static SINT64 getGlobalId(int);
static void putDatabase(const Database*, Writer&, int);
static bool putAttachment(const Attachment*, Writer&, int);
static void putTransaction(const jrd_tra*, Writer&, int);
static void putRequest(const jrd_req*, Writer&, int);
static void putCall(const jrd_req*, Writer&, int);
static void putStatistics(const RuntimeStatistics&, Writer&, int, int);
static void putContextVars(const Firebird::StringMap&, Writer&, int, bool);
static void putMemoryUsage(const Firebird::MemoryStats&, Writer&, int, int);
Firebird::Array<RelationData> snapshot;
Firebird::GenericMap<Firebird::Pair<Firebird::NonPooled<SINT64, SLONG> > > idMap;
int idCounter;
};
} // namespace
#endif // JRD_DATABASE_SNAPSHOT_H