mirror of
https://github.com/FirebirdSQL/firebird.git
synced 2025-01-27 18:03:04 +01:00
354 lines
7.3 KiB
C++
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
|