mirror of
https://github.com/FirebirdSQL/firebird.git
synced 2025-01-27 18:03:04 +01:00
616 lines
14 KiB
C++
616 lines
14 KiB
C++
/*
|
|
* PROGRAM: Firebird Trace Services
|
|
* MODULE: TraceConfigStorage.cpp
|
|
* DESCRIPTION: Trace API shared configurations storage
|
|
*
|
|
* 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 Khorsun Vladyslav
|
|
* for the Firebird Open Source RDBMS project.
|
|
*
|
|
* Copyright (c) 2008 Khorsun Vladyslav <hvlad@users.sourceforge.net>
|
|
* and all contributors signed below.
|
|
*
|
|
* All Rights Reserved.
|
|
* Contributor(s): ______________________________________.
|
|
*
|
|
*/
|
|
|
|
#include "firebird.h"
|
|
|
|
#include "../../common/classes/TempFile.h"
|
|
#include "../../common/StatusArg.h"
|
|
#include "../../common/utils_proto.h"
|
|
#include "../../jrd/err_proto.h"
|
|
#include "../../common/isc_proto.h"
|
|
#include "../../common/isc_s_proto.h"
|
|
#include "../../jrd/jrd.h"
|
|
#include "../../common/os/path_utils.h"
|
|
#include "../../common/config/os/config_root.h"
|
|
#include "../../common/os/os_utils.h"
|
|
#include "../../jrd/trace/TraceConfigStorage.h"
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
#include <unistd.h>
|
|
#endif
|
|
#ifdef HAVE_IO_H
|
|
#include <io.h>
|
|
#endif
|
|
#include <fcntl.h>
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#ifndef O_BINARY
|
|
#define O_BINARY 0
|
|
#endif
|
|
|
|
using namespace Firebird;
|
|
|
|
namespace Jrd {
|
|
|
|
static const FB_UINT64 TOUCH_INTERVAL = 60 * 60; // in seconds, one hour should be enough
|
|
|
|
void checkFileError(const char* filename, const char* operation, ISC_STATUS iscError)
|
|
{
|
|
if (errno == 0)
|
|
return;
|
|
|
|
#ifdef WIN_NT
|
|
// we can't use SYS_ERR(errno) on Windows as errno codes is not
|
|
// the same as GetLastError() codes
|
|
const char* strErr = strerror(errno);
|
|
|
|
(Arg::Gds(isc_io_error) << Arg::Str(operation) << Arg::Str(filename) <<
|
|
Arg::Gds(iscError) << Arg::Str(strErr)).raise();
|
|
#else
|
|
(Arg::Gds(isc_io_error) << Arg::Str(operation) << Arg::Str(filename) <<
|
|
Arg::Gds(iscError) << SYS_ERR(errno)).raise();
|
|
#endif
|
|
}
|
|
|
|
ConfigStorage::ConfigStorage()
|
|
: m_timer(new TouchFile),
|
|
m_sharedMemory(NULL),
|
|
m_recursive(0),
|
|
m_mutexTID(0)
|
|
{
|
|
m_cfg_file = -1;
|
|
m_dirty = false;
|
|
|
|
PathName filename;
|
|
#ifdef WIN_NT
|
|
DWORD sesID = 0;
|
|
|
|
typedef BOOL (WINAPI *PFnProcessIdToSessionId) (DWORD, DWORD *);
|
|
|
|
HMODULE hmodKernel32 = GetModuleHandle("kernel32.dll");
|
|
|
|
PFnProcessIdToSessionId pfnProcessIdToSessionId =
|
|
(PFnProcessIdToSessionId) GetProcAddress(hmodKernel32, "ProcessIdToSessionId");
|
|
|
|
if (fb_utils::isGlobalKernelPrefix() ||
|
|
!pfnProcessIdToSessionId ||
|
|
pfnProcessIdToSessionId(GetCurrentProcessId(), &sesID) == 0 ||
|
|
sesID == 0)
|
|
{
|
|
filename.printf(TRACE_FILE); // TODO: it must be per engine instance
|
|
}
|
|
else
|
|
{
|
|
filename.printf("%s.%u", TRACE_FILE, sesID);
|
|
}
|
|
#else
|
|
filename.printf(TRACE_FILE); // TODO: it must be per engine instance
|
|
#endif
|
|
|
|
try
|
|
{
|
|
m_sharedMemory.reset(FB_NEW(getPool())
|
|
SharedMemory<TraceCSHeader>(filename.c_str(), sizeof(TraceCSHeader), this));
|
|
}
|
|
catch (const Exception& ex)
|
|
{
|
|
iscLogException("ConfigStorage: Cannot initialize the shared memory region", ex);
|
|
throw;
|
|
}
|
|
|
|
fb_assert(m_sharedMemory->getHeader());
|
|
fb_assert(m_sharedMemory->getHeader()->mhb_version == 1);
|
|
|
|
StorageGuard guard(this);
|
|
checkFile();
|
|
m_timer->start(m_sharedMemory->getHeader()->cfg_file_name);
|
|
|
|
++(m_sharedMemory->getHeader()->cnt_uses);
|
|
}
|
|
|
|
ConfigStorage::~ConfigStorage()
|
|
{
|
|
m_timer->stop();
|
|
|
|
::close(m_cfg_file);
|
|
m_cfg_file = -1;
|
|
|
|
{
|
|
StorageGuard guard(this);
|
|
--(m_sharedMemory->getHeader()->cnt_uses);
|
|
if (m_sharedMemory->getHeader()->cnt_uses == 0)
|
|
{
|
|
unlink(m_sharedMemory->getHeader()->cfg_file_name);
|
|
memset(m_sharedMemory->getHeader()->cfg_file_name, 0,
|
|
sizeof(m_sharedMemory->getHeader()->cfg_file_name));
|
|
|
|
m_sharedMemory->removeMapFile();
|
|
}
|
|
}
|
|
}
|
|
|
|
void ConfigStorage::mutexBug(int state, const TEXT* string)
|
|
{
|
|
TEXT msg[BUFFER_TINY];
|
|
|
|
// While string is kept below length 70, all is well.
|
|
sprintf(msg, "ConfigStorage: mutex %s error, status = %d", string, state);
|
|
fb_utils::logAndDie(msg);
|
|
}
|
|
|
|
bool ConfigStorage::initialize(SharedMemoryBase* sm, bool init)
|
|
{
|
|
TraceCSHeader* header = reinterpret_cast<TraceCSHeader*>(sm->sh_mem_header);
|
|
|
|
// Initialize the shared data header
|
|
if (init)
|
|
{
|
|
header->mhb_type = SharedMemoryBase::SRAM_TRACE_CONFIG;
|
|
header->mhb_version = 1;
|
|
header->mhb_timestamp = TimeStamp::getCurrentTimeStamp().value();
|
|
header->change_number = 0;
|
|
header->session_number = 1;
|
|
header->cnt_uses = 0;
|
|
memset(header->cfg_file_name, 0, sizeof(header->cfg_file_name));
|
|
}
|
|
else
|
|
{
|
|
fb_assert(header->mhb_type == SharedMemoryBase::SRAM_TRACE_CONFIG);
|
|
fb_assert(header->mhb_version == 1);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void ConfigStorage::checkFile()
|
|
{
|
|
if (m_cfg_file >= 0)
|
|
return;
|
|
|
|
char* cfg_file_name = m_sharedMemory->getHeader()->cfg_file_name;
|
|
|
|
if (!(*cfg_file_name))
|
|
{
|
|
fb_assert(m_sharedMemory->getHeader()->cnt_uses == 0);
|
|
|
|
char dir[MAXPATHLEN];
|
|
iscPrefixLock(dir, "", true);
|
|
|
|
PathName filename = TempFile::create("fb_trace_", dir);
|
|
filename.copyTo(cfg_file_name, sizeof(m_sharedMemory->getHeader()->cfg_file_name));
|
|
m_cfg_file = os_utils::openCreateSharedFile(cfg_file_name, O_BINARY);
|
|
}
|
|
else
|
|
{
|
|
m_cfg_file = ::open(cfg_file_name, O_RDWR | O_BINARY);
|
|
|
|
if (m_cfg_file < 0)
|
|
checkFileError(cfg_file_name, "open", isc_io_open_err);
|
|
}
|
|
|
|
// put default (audit) trace file contents into storage
|
|
if (m_sharedMemory->getHeader()->change_number == 0)
|
|
{
|
|
FILE* cfgFile = NULL;
|
|
|
|
try
|
|
{
|
|
PathName configFileName(Config::getAuditTraceConfigFile());
|
|
|
|
// remove quotes around path if present
|
|
{ // scope
|
|
const FB_SIZE_T pathLen = configFileName.length();
|
|
if (pathLen > 1 && configFileName[0] == '"' &&
|
|
configFileName[pathLen - 1] == '"')
|
|
{
|
|
configFileName.erase(0, 1);
|
|
configFileName.erase(pathLen - 2, 1);
|
|
}
|
|
}
|
|
|
|
if (configFileName.empty())
|
|
return;
|
|
|
|
if (PathUtils::isRelative(configFileName))
|
|
{
|
|
PathName root(Config::getRootDirectory());
|
|
PathUtils::ensureSeparator(root);
|
|
configFileName.insert(0, root);
|
|
}
|
|
|
|
cfgFile = fopen(configFileName.c_str(), "rb");
|
|
if (!cfgFile) {
|
|
checkFileError(configFileName.c_str(), "fopen", isc_io_open_err);
|
|
}
|
|
|
|
TraceSession session(*getDefaultMemoryPool());
|
|
|
|
fseek(cfgFile, 0, SEEK_END);
|
|
const long len = ftell(cfgFile);
|
|
if (len)
|
|
{
|
|
fseek(cfgFile, 0, SEEK_SET);
|
|
char* p = session.ses_config.getBuffer(len + 1);
|
|
|
|
if (fread(p, 1, len, cfgFile) != size_t(len)) {
|
|
checkFileError(configFileName.c_str(), "fread", isc_io_read_err);
|
|
}
|
|
p[len] = 0;
|
|
}
|
|
else {
|
|
gds__log("Audit configuration file \"%s\" is empty", configFileName.c_str());
|
|
}
|
|
|
|
session.ses_user = SYSDBA_USER_NAME;
|
|
session.ses_name = "Firebird Audit";
|
|
session.ses_flags = trs_admin | trs_system;
|
|
|
|
addSession(session);
|
|
}
|
|
catch(const Exception& ex)
|
|
{
|
|
ISC_STATUS_ARRAY temp;
|
|
ex.stuff_exception(temp);
|
|
iscLogStatus("Cannot open audit configuration file", temp);
|
|
}
|
|
|
|
if (cfgFile) {
|
|
fclose(cfgFile);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void ConfigStorage::acquire()
|
|
{
|
|
fb_assert(m_recursive >= 0);
|
|
const ThreadId currTID = getThreadId();
|
|
|
|
if (m_mutexTID == currTID)
|
|
m_recursive++;
|
|
else
|
|
{
|
|
m_sharedMemory->mutexLock();
|
|
|
|
fb_assert(m_recursive == 0);
|
|
m_recursive = 1;
|
|
|
|
fb_assert(m_mutexTID == 0);
|
|
m_mutexTID = currTID;
|
|
}
|
|
}
|
|
|
|
void ConfigStorage::release()
|
|
{
|
|
fb_assert(m_recursive > 0);
|
|
fb_assert(m_mutexTID == getThreadId());
|
|
|
|
if (--m_recursive == 0)
|
|
{
|
|
checkDirty();
|
|
m_mutexTID = 0;
|
|
m_sharedMemory->mutexUnlock();
|
|
}
|
|
}
|
|
|
|
void ConfigStorage::addSession(TraceSession& session)
|
|
{
|
|
setDirty();
|
|
session.ses_id = (m_sharedMemory->getHeader()->session_number)++;
|
|
session.ses_flags |= trs_active;
|
|
time(&session.ses_start);
|
|
|
|
const long pos1 = lseek(m_cfg_file, 0, SEEK_END);
|
|
if (pos1 < 0)
|
|
{
|
|
const char* fn = m_sharedMemory->getHeader()->cfg_file_name;
|
|
ERR_post(Arg::Gds(isc_io_error) << Arg::Str("lseek") << Arg::Str(fn) <<
|
|
Arg::Gds(isc_io_read_err) << SYS_ERR(errno));
|
|
}
|
|
|
|
putItem(tagID, sizeof(session.ses_id), &session.ses_id);
|
|
if (!session.ses_name.empty()) {
|
|
putItem(tagName, session.ses_name.length(), session.ses_name.c_str());
|
|
}
|
|
putItem(tagUserName, session.ses_user.length(), session.ses_user.c_str());
|
|
putItem(tagFlags, sizeof(session.ses_flags), &session.ses_flags);
|
|
putItem(tagConfig, session.ses_config.length(), session.ses_config.c_str());
|
|
putItem(tagStartTS, sizeof(session.ses_start), &session.ses_start);
|
|
if (!session.ses_logfile.empty()) {
|
|
putItem(tagLogFile, session.ses_logfile.length(), session.ses_logfile.c_str());
|
|
}
|
|
putItem(tagEnd, 0, NULL);
|
|
|
|
// const long pos2 = lseek(m_cfg_file, 0, SEEK_END);
|
|
// m_sharedMemory->getHeader()->used_space += pos2 - pos1;
|
|
}
|
|
|
|
bool ConfigStorage::getNextSession(TraceSession& session)
|
|
{
|
|
ITEM tag = tagID;
|
|
ULONG len;
|
|
session.clear();
|
|
|
|
while (true)
|
|
{
|
|
if (!getItemLength(tag, len))
|
|
return false;
|
|
|
|
if (tag == tagEnd)
|
|
{
|
|
if (session.ses_id != 0)
|
|
return true;
|
|
|
|
continue;
|
|
}
|
|
|
|
void* p = NULL;
|
|
|
|
switch (tag)
|
|
{
|
|
case tagID:
|
|
fb_assert(len == sizeof(session.ses_id));
|
|
p = &session.ses_id;
|
|
break;
|
|
|
|
case tagName:
|
|
if (session.ses_id)
|
|
p = session.ses_name.getBuffer(len);
|
|
break;
|
|
|
|
case tagUserName:
|
|
if (session.ses_id)
|
|
p = session.ses_user.getBuffer(len);
|
|
break;
|
|
|
|
case tagFlags:
|
|
fb_assert(len == sizeof(session.ses_flags));
|
|
if (session.ses_id)
|
|
p = &session.ses_flags;
|
|
break;
|
|
|
|
case tagConfig:
|
|
if (session.ses_id)
|
|
p = session.ses_config.getBuffer(len);
|
|
break;
|
|
|
|
case tagStartTS:
|
|
fb_assert(len == sizeof(session.ses_start));
|
|
if (session.ses_id)
|
|
p = &session.ses_start;
|
|
break;
|
|
|
|
case tagLogFile:
|
|
if (session.ses_id)
|
|
p = session.ses_logfile.getBuffer(len);
|
|
break;
|
|
|
|
default:
|
|
fb_assert(false);
|
|
}
|
|
|
|
if (p)
|
|
{
|
|
if (::read(m_cfg_file, p, len) != len)
|
|
checkFileError(m_sharedMemory->getHeader()->cfg_file_name, "read", isc_io_read_err);
|
|
}
|
|
else
|
|
{
|
|
if (lseek(m_cfg_file, len, SEEK_CUR) < 0)
|
|
checkFileError(m_sharedMemory->getHeader()->cfg_file_name, "lseek", isc_io_read_err);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void ConfigStorage::removeSession(ULONG id)
|
|
{
|
|
ITEM tag = tagID;
|
|
ULONG len;
|
|
|
|
restart();
|
|
while (true)
|
|
{
|
|
if (!getItemLength(tag, len))
|
|
return;
|
|
|
|
if (tag == tagID)
|
|
{
|
|
ULONG currID;
|
|
fb_assert(len == sizeof(currID));
|
|
|
|
bool err = (::read(m_cfg_file, &currID, len) != len);
|
|
if (!err && currID == id)
|
|
{
|
|
setDirty();
|
|
|
|
currID = 0;
|
|
// Do not delete this temporary signed var, otherwise we get
|
|
// warning C4146: unary minus operator applied to unsigned type, result still unsigned
|
|
// but we need a negative offset here.
|
|
const long local_len = len;
|
|
if (lseek(m_cfg_file, -local_len, SEEK_CUR) < 0)
|
|
checkFileError(m_sharedMemory->getHeader()->cfg_file_name, "lseek", isc_io_read_err);
|
|
|
|
if (write(m_cfg_file, &currID, len) != len)
|
|
checkFileError(m_sharedMemory->getHeader()->cfg_file_name, "write", isc_io_write_err);
|
|
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (lseek(m_cfg_file, len, SEEK_CUR) < 0)
|
|
checkFileError(m_sharedMemory->getHeader()->cfg_file_name, "lseek", isc_io_read_err);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void ConfigStorage::restart()
|
|
{
|
|
checkDirty();
|
|
|
|
if (lseek(m_cfg_file, 0, SEEK_SET) < 0)
|
|
checkFileError(m_sharedMemory->getHeader()->cfg_file_name, "lseek", isc_io_read_err);
|
|
}
|
|
|
|
|
|
void ConfigStorage::updateSession(TraceSession& session)
|
|
{
|
|
restart();
|
|
|
|
ITEM tag;
|
|
ULONG len;
|
|
ULONG currID = 0;
|
|
|
|
while (true)
|
|
{
|
|
if (!getItemLength(tag, len))
|
|
return;
|
|
|
|
void* p = NULL;
|
|
switch (tag)
|
|
{
|
|
case tagID:
|
|
fb_assert(len == sizeof(currID));
|
|
FB_UNUSED(read(m_cfg_file, &currID, len));
|
|
continue;
|
|
|
|
case tagFlags:
|
|
fb_assert(len == sizeof(session.ses_flags));
|
|
if (currID == session.ses_id)
|
|
p = &session.ses_flags;
|
|
break;
|
|
|
|
case tagEnd:
|
|
if (currID == session.ses_id)
|
|
return;
|
|
len = 0;
|
|
break;
|
|
}
|
|
|
|
if (p)
|
|
{
|
|
setDirty();
|
|
if (write(m_cfg_file, p, len) != len)
|
|
checkFileError(m_sharedMemory->getHeader()->cfg_file_name, "write", isc_io_write_err);
|
|
}
|
|
else if (len)
|
|
{
|
|
if (lseek(m_cfg_file, len, SEEK_CUR) < 0)
|
|
checkFileError(m_sharedMemory->getHeader()->cfg_file_name, "lseek", isc_io_read_err);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void ConfigStorage::putItem(ITEM tag, ULONG len, const void* data)
|
|
{
|
|
const char tag_data = (char) tag;
|
|
ULONG to_write = sizeof(tag_data);
|
|
if (write(m_cfg_file, &tag_data, to_write) != to_write)
|
|
checkFileError(m_sharedMemory->getHeader()->cfg_file_name, "write", isc_io_write_err);
|
|
|
|
if (tag == tagEnd)
|
|
return;
|
|
|
|
to_write = sizeof(len);
|
|
if (write(m_cfg_file, &len, to_write) != to_write)
|
|
checkFileError(m_sharedMemory->getHeader()->cfg_file_name, "write", isc_io_write_err);
|
|
|
|
if (len)
|
|
{
|
|
if (write(m_cfg_file, data, len) != len)
|
|
checkFileError(m_sharedMemory->getHeader()->cfg_file_name, "write", isc_io_write_err);
|
|
}
|
|
}
|
|
|
|
bool ConfigStorage::getItemLength(ITEM& tag, ULONG& len)
|
|
{
|
|
char data;
|
|
const int cnt_read = read(m_cfg_file, &data, sizeof(data));
|
|
|
|
if (cnt_read == 0)
|
|
return false;
|
|
|
|
if (cnt_read < 0)
|
|
checkFileError(m_sharedMemory->getHeader()->cfg_file_name, "read", isc_io_read_err);
|
|
|
|
tag = (ITEM) data;
|
|
|
|
if (tag == tagEnd)
|
|
len = 0;
|
|
else
|
|
{
|
|
if (read(m_cfg_file, &len, sizeof(ULONG)) != sizeof(ULONG))
|
|
checkFileError(m_sharedMemory->getHeader()->cfg_file_name, "read", isc_io_read_err);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void ConfigStorage::TouchFile::handler()
|
|
{
|
|
os_utils::touchFile(fileName);
|
|
LocalStatus s;
|
|
TimerInterfacePtr()->start(&s, this, TOUCH_INTERVAL * 1000 * 1000);
|
|
// ignore error in handler
|
|
}
|
|
|
|
void ConfigStorage::TouchFile::start(const char* fName)
|
|
{
|
|
fileName = fName;
|
|
LocalStatus s;
|
|
TimerInterfacePtr()->start(&s, this, TOUCH_INTERVAL * 1000 * 1000);
|
|
check(&s);
|
|
}
|
|
|
|
void ConfigStorage::TouchFile::stop()
|
|
{
|
|
LocalStatus s;
|
|
TimerInterfacePtr()->stop(&s, this);
|
|
// ignore error in stop timer
|
|
}
|
|
|
|
int ConfigStorage::TouchFile::release()
|
|
{
|
|
if (--refCounter == 0)
|
|
{
|
|
delete this;
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
} // namespace Jrd
|