8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-30 19:23:03 +01:00
firebird-mirror/src/include/fb_exception.h

154 lines
4.9 KiB
C
Raw Normal View History

2001-12-24 03:51:06 +01:00
/*
* PROGRAM: Firebird exceptions classes
* MODULE: fb_exception.h
* DESCRIPTION: Specific exception classes derived from std::exception
*
* 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): ______________________________________.
2001-12-24 03:51:06 +01:00
*
*
*/
2001-12-24 03:51:06 +01:00
#ifndef FB_EXCEPTION_H
#define FB_EXCEPTION_H
#include <exception>
#include <stddef.h>
#include <stdarg.h>
2004-05-03 06:25:06 +02:00
#include <string.h>
2001-12-24 03:51:06 +01:00
#include "fb_types.h"
namespace Firebird {
class StringsBuffer {
2001-12-24 03:51:06 +01:00
public:
virtual char* alloc(const char* string, size_t length) = 0;
virtual ~StringsBuffer() {}
2001-12-24 03:51:06 +01:00
};
template <size_t BUFFER_SIZE>
class CircularStringsBuffer : public StringsBuffer {
2001-12-24 03:51:06 +01:00
public:
CircularStringsBuffer() 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;
}
virtual char* alloc(const char* string, size_t length) {
2004-12-08 06:58:41 +01:00
// fb_assert(length + 1 < BUFFER_SIZE);
// 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:
char buffer[BUFFER_SIZE];
char *buffer_ptr;
2001-12-24 03:51:06 +01:00
};
class status_exception : public std::exception
2001-12-24 03:51:06 +01:00
{
public:
// This version of constructor receives status vector pointing to permanent strings
// which may be returned to user in status vector directly without transfer of string ownership
explicit status_exception(const ISC_STATUS *status_vector) throw();
// These versions of constructor clone passed transient strings
status_exception(ISC_STATUS status, va_list status_args);
status_exception(ISC_STATUS status, ...);
// Create exception with undefined status vector, this constructor allows to use this
// class as jmpbuf replacement for transitional period
status_exception() throw();
virtual ~status_exception() throw();
virtual const char* what() const throw()
{ return "Firebird::status_exception"; }
const ISC_STATUS* value() const { 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 { return m_strings_permanent; }
// Returns true if exception class holds status vector for the error.
// Returns false when status vector is passed "by magic", probably
// somewhere in tdbb_status_vector
bool status_known() const { return m_status_known; }
2001-12-24 03:51:06 +01:00
// Takes permanent strings
static void raise(const ISC_STATUS *status_vector);
2002-11-11 19:06:01 +01:00
static void raise();
// Take transient strings
static void raise(ISC_STATUS status, ...);
private:
ISC_STATUS_ARRAY m_status_vector;
bool m_strings_permanent;
bool m_status_known;
void fill_status(ISC_STATUS status, va_list status_args);
2002-11-11 19:06:01 +01:00
};
class system_call_failed : public status_exception
2002-11-11 19:06:01 +01:00
{
public:
system_call_failed(const char* v_syscall, int v_error_code);
2002-11-11 19:06:01 +01:00
static void raise(const char* syscall, int error_code);
static void raise(const char* syscall);
2002-11-11 19:06:01 +01:00
};
// Moved what() here due to gpre. Didn't want to use macros for gpre_static.
class fatal_exception : public status_exception
2002-11-11 19:06:01 +01:00
{
public:
2003-12-31 06:36:12 +01:00
explicit fatal_exception(const char* message);
static void raiseFmt(const char* format, ...);
// Keep in sync with the constructor above, please; "message" becomes 4th element
// after status_exception's constructor invokes fill_status().
const char* what() const throw()
{
return reinterpret_cast<const char*>(value()[3]);
}
2003-12-31 06:36:12 +01:00
static void raise(const char* message);
2001-12-24 03:51:06 +01:00
};
// Serialize exception into status_vector, put transient strings from exception into given StringsBuffer
ISC_STATUS stuff_exception(ISC_STATUS *status_vector, const std::exception& ex, StringsBuffer* sb = NULL) throw();
// These routines put strings into process-level circular buffer
// They are obsolete, use transient version of status_exception::raise in combination with
// stuff_exception instead
const char* status_string(const char* string);
const char* status_nstring(const char* string, size_t length);
2001-12-24 03:51:06 +01:00
} // namespace Firebird
2001-12-24 03:51:06 +01:00
#endif // FB_EXCEPTION_H
2003-12-31 06:36:12 +01:00