8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-26 07:23:08 +01:00
firebird-mirror/src/include/fb_exception.h
asfernandes 28a7e4c037 Misc
2009-07-24 02:15:07 +00:00

210 lines
5.8 KiB
C++

/*
* PROGRAM: Firebird exceptions classes
* MODULE: fb_exception.h
* DESCRIPTION: Firebird's exception classes
*
* 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 Mike Nordell
* for the Firebird Open Source RDBMS project.
*
* Copyright (c) 2001 Mike Nordell <tamlin at algonet.se>
* and all contributors signed below.
*
* All Rights Reserved.
* Contributor(s): ______________________________________.
*
*
*/
#ifndef FB_EXCEPTION_H
#define FB_EXCEPTION_H
#include <stddef.h>
#include <string.h>
#include <new>
#include "fb_types.h"
#include "../common/StatusArg.h"
namespace Firebird {
class MemoryPool;
class StringsBuffer
{
public:
virtual const char* alloc(const char* string, size_t& length) = 0;
virtual ~StringsBuffer() {}
void makePermanentVector(ISC_STATUS* perm, const ISC_STATUS* temp);
static void makeEnginePermanentVector(ISC_STATUS* v);
};
template <size_t BUFFER_SIZE>
class CircularStringsBuffer : public StringsBuffer
{
public:
CircularStringsBuffer() throw() { init(); }
explicit CircularStringsBuffer(MemoryPool&) throw() { init(); }
virtual const char* alloc(const char* string, size_t& length)
{
// if string is already in our buffer - return it
// it was already saved in our buffer once
if (string >= buffer && string < &buffer[BUFFER_SIZE])
return string;
// if string too long, truncate it
if (length > BUFFER_SIZE / 4)
length = BUFFER_SIZE / 4;
// If there isn't any more room in the buffer, start at the beginning again
if (buffer_ptr + length + 1 > buffer + BUFFER_SIZE)
buffer_ptr = buffer;
char* new_string = buffer_ptr;
memcpy(new_string, string, length);
new_string[length] = 0;
buffer_ptr += length + 1;
return new_string;
}
private:
void init() throw()
{
// This is to ensure we have zero at the end of buffer in case of buffer overflow
memset(buffer, 0, BUFFER_SIZE);
buffer_ptr = buffer;
}
char buffer[BUFFER_SIZE];
char* buffer_ptr;
};
class Exception
{
protected:
Exception() throw() { }
public:
virtual ~Exception() throw();
virtual ISC_STATUS stuff_exception(ISC_STATUS* const status_vector, StringsBuffer* sb = NULL) const throw() = 0;
virtual const char* what() const throw() = 0;
};
// Used as jmpbuf to unwind when needed
class LongJump : public Exception
{
public:
virtual ISC_STATUS stuff_exception(ISC_STATUS* const status_vector, StringsBuffer* sb = NULL) const throw();
virtual const char* what() const throw();
static void raise();
LongJump() throw() : Exception() { }
};
// Used in MemoryPool
class BadAlloc : public std::bad_alloc, public Exception
{
public:
virtual ISC_STATUS stuff_exception(ISC_STATUS* const status_vector, StringsBuffer* sb = NULL) const throw();
virtual const char* what() const throw();
static void raise();
BadAlloc() throw() : std::bad_alloc(), Exception() { }
};
// Main exception class in firebird
class status_exception : public Exception
{
public:
// This version of constructor receives status vector pointing to permanent or
// temp strings, depending upon second parameter.
status_exception(const ISC_STATUS *status_vector, bool permanent) throw();
virtual ~status_exception() throw();
virtual ISC_STATUS stuff_exception(ISC_STATUS* const status_vector, StringsBuffer* sb = NULL) const throw();
virtual const char* what() const throw();
const ISC_STATUS* value() const throw() { return m_status_vector; }
// Returns true if strings contained in status vector are located in magical
// permanent circular buffer. False means that exception object owns strings
// and is about to deallocate them in its destructor
bool strings_permanent() const throw() { return m_strings_permanent; }
// Takes permanent strings
static void raise(const ISC_STATUS *status_vector);
// Takes transient strings
static void raise(const Arg::StatusVector& statusVector);
protected:
// Create exception with undefined status vector, this constructor allows
// derived classes create empty exception ...
status_exception() throw();
// .. and adjust it later using somehow created status vector.
void set_status(const ISC_STATUS *new_vector, bool permanent) throw();
private:
ISC_STATUS_ARRAY m_status_vector;
bool m_strings_permanent;
void release_vector() throw();
};
// use this class if exception can be handled
class system_error : public status_exception
{
private:
int errorCode;
public:
system_error(const char* syscall, int error_code);
static void raise(const char* syscall, int error_code);
static void raise(const char* syscall);
int getErrorCode() const
{
return errorCode;
}
static int getSystemError();
};
// use this class if exception can't be handled
// it will call abort() in DEV_BUILD to create core dump
class system_call_failed : public system_error
{
public:
system_call_failed(const char* syscall, int error_code);
static void raise(const char* syscall, int error_code);
static void raise(const char* syscall);
};
class fatal_exception : public status_exception
{
public:
explicit fatal_exception(const char* message);
static void raiseFmt(const char* format, ...);
const char* what() const throw();
static void raise(const char* message);
};
// Serialize exception into status_vector, put transient strings from exception into given StringsBuffer
ISC_STATUS stuff_exception(ISC_STATUS *status_vector, const Firebird::Exception& ex, StringsBuffer* sb = NULL) throw();
} // namespace Firebird
#endif // FB_EXCEPTION_H