2001-05-23 15:26:42 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: JRD Access Method
|
2003-11-05 10:02:33 +01:00
|
|
|
* MODULE: err.cpp
|
2001-05-23 15:26:42 +02:00
|
|
|
* DESCRIPTION: Bug check routine
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Interbase 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.Inprise.com/IPL.html
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an
|
|
|
|
* "AS IS" basis, 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 Inprise Corporation
|
|
|
|
* and its predecessors. Portions created by Inprise Corporation are
|
|
|
|
* Copyright (C) Inprise Corporation.
|
|
|
|
*
|
|
|
|
* All Rights Reserved.
|
|
|
|
* Contributor(s): ______________________________________.
|
2002-10-29 03:59:57 +01:00
|
|
|
*
|
|
|
|
* 2002.10.28 Sean Leyne - Code cleanup, removed obsolete "MPEXL" port
|
|
|
|
*
|
2002-10-30 07:40:58 +01:00
|
|
|
* 2002.10.29 Sean Leyne - Removed obsolete "Netware" port
|
|
|
|
*
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
|
|
|
|
2001-07-29 19:42:23 +02:00
|
|
|
#include "firebird.h"
|
2004-04-29 00:36:29 +02:00
|
|
|
#include <stdio.h>
|
2001-05-23 15:26:42 +02:00
|
|
|
#include <string.h>
|
|
|
|
#include "../jrd/common.h"
|
|
|
|
#include <stdarg.h>
|
2003-11-11 13:19:20 +01:00
|
|
|
#include "gen/iberror.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/iberr.h"
|
|
|
|
|
|
|
|
#if ( !defined( REQUESTER) && !defined( SUPERCLIENT))
|
|
|
|
#include "../jrd/jrd.h"
|
2003-07-14 12:35:49 +02:00
|
|
|
#include "../jrd/os/pio.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/val.h"
|
|
|
|
#include "../jrd/ods.h"
|
|
|
|
#include "../jrd/btr.h"
|
|
|
|
#include "../jrd/req.h"
|
2003-10-20 12:53:52 +02:00
|
|
|
#include "../jrd/rse.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/tra.h"
|
|
|
|
#include "../jrd/all_proto.h"
|
|
|
|
#include "../jrd/cch_proto.h"
|
|
|
|
#include "../jrd/met_proto.h"
|
|
|
|
#endif
|
2004-06-08 15:41:08 +02:00
|
|
|
#include "../jrd/thd.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/dbg_proto.h"
|
|
|
|
#include "../jrd/err_proto.h"
|
|
|
|
#include "../jrd/gds_proto.h"
|
2004-04-06 09:25:45 +02:00
|
|
|
#include "../common/config/config.h"
|
2004-10-09 05:58:45 +02:00
|
|
|
#include "../common/utils_proto.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-20 15:57:40 +01:00
|
|
|
using namespace Jrd;
|
|
|
|
|
2004-05-03 19:47:44 +02:00
|
|
|
//#define JRD_FAILURE_SPACE 2048
|
|
|
|
//#define JRD_FAILURE_UNKNOWN "<UNKNOWN>" /* Used when buffer fails */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
2004-09-24 02:11:32 +02:00
|
|
|
static void internal_error(ISC_STATUS status, int number,
|
|
|
|
const TEXT* file = NULL, int line = 0);
|
2004-08-06 17:26:55 +02:00
|
|
|
static void internal_post(ISC_STATUS status, va_list args);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
|
|
|
#if ( !defined( REQUESTER) && !defined( SUPERCLIENT))
|
2004-09-24 02:11:32 +02:00
|
|
|
void ERR_bugcheck(int number, const TEXT* file, int line)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E R R _ b u g c h e c k
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Things seem to be going poorly today.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-05-23 05:18:10 +02:00
|
|
|
Database* dbb = GET_DBB();
|
2001-05-23 15:26:42 +02:00
|
|
|
dbb->dbb_flags |= DBB_bugcheck;
|
|
|
|
|
|
|
|
CCH_shutdown_database(dbb);
|
|
|
|
|
2004-09-24 02:11:32 +02:00
|
|
|
internal_error(isc_bug_check, number, file, line);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#if ( !defined( REQUESTER) && !defined( SUPERCLIENT))
|
2003-10-03 03:53:34 +02:00
|
|
|
void ERR_bugcheck_msg(const TEXT* msg)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E R R _ b u g c h e c k _ m s g
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Things seem to be going poorly today.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-05-23 05:18:10 +02:00
|
|
|
Database* dbb = GET_DBB();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
dbb->dbb_flags |= DBB_bugcheck;
|
|
|
|
DEBUG;
|
|
|
|
|
|
|
|
CCH_shutdown_database(dbb);
|
|
|
|
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_bug_check, isc_arg_string, ERR_cstring(msg), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#if ( !defined( REQUESTER) && !defined( SUPERCLIENT))
|
2003-10-03 03:53:34 +02:00
|
|
|
void ERR_corrupt(int number)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E R R _ c o r r u p t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Things seem to be going poorly today.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2003-11-11 13:19:20 +01:00
|
|
|
internal_error(isc_db_corrupt, number);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2004-07-07 05:43:20 +02:00
|
|
|
|
2003-10-03 03:53:34 +02:00
|
|
|
const TEXT* ERR_cstring(const TEXT* in_string)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E R R _ c s t r i n g
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Copy a stack local string into an area that is
|
|
|
|
* guaranteed to exist for a reasonable length of time.
|
|
|
|
* Only strings that are to be placed in status vectors
|
|
|
|
* should use this routine.
|
|
|
|
* A circular buffer is used to hold these strings. It
|
|
|
|
* is independent of the JRD allocator mechanism.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-01 04:35:23 +01:00
|
|
|
return Firebird::status_string(in_string);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#if ( !defined( REQUESTER) && !defined( SUPERCLIENT))
|
2003-10-03 03:53:34 +02:00
|
|
|
void ERR_duplicate_error(IDX_E code,
|
2004-07-07 05:43:20 +02:00
|
|
|
const jrd_rel* relation,
|
2003-10-03 03:53:34 +02:00
|
|
|
USHORT index_number)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E R R _ d u p l i c a t e _ e r r o r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Duplicate error during index update.
|
|
|
|
*
|
|
|
|
**************************************/
|
2005-05-12 20:28:04 +02:00
|
|
|
Firebird::MetaName index, constraint;
|
2002-10-29 17:27:47 +01:00
|
|
|
const TEXT* index_name;
|
|
|
|
const TEXT* constraint_name;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-22 16:28:54 +02:00
|
|
|
thread_db* tdbb = JRD_get_thread_data();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
MET_lookup_index(tdbb, index, relation->rel_name, index_number + 1);
|
2005-05-12 20:28:04 +02:00
|
|
|
if (index.length()) {
|
2001-05-23 15:26:42 +02:00
|
|
|
index_name = ERR_cstring(index);
|
2005-05-12 20:28:04 +02:00
|
|
|
MET_lookup_cnstrt_for_index(tdbb, constraint, index);
|
|
|
|
if (constraint.length() > 0)
|
2001-05-23 15:26:42 +02:00
|
|
|
constraint_name = ERR_cstring(constraint);
|
2004-07-07 05:43:20 +02:00
|
|
|
else
|
|
|
|
constraint_name = "***unknown***";
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
index_name = "***unknown***";
|
|
|
|
constraint_name = "***unknown***";
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (code) {
|
|
|
|
case idx_e_keytoobig:
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_imp_exc, isc_arg_gds, isc_keytoobig,
|
2004-10-09 05:55:43 +02:00
|
|
|
isc_arg_string, index_name, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case idx_e_conversion:
|
|
|
|
ERR_punt();
|
|
|
|
break;
|
|
|
|
|
2004-09-01 00:29:59 +02:00
|
|
|
case idx_e_foreign_target_doesnt_exist:
|
2004-10-09 05:55:43 +02:00
|
|
|
ERR_post(isc_foreign_key, isc_arg_string, constraint_name,
|
2005-05-12 20:28:04 +02:00
|
|
|
isc_arg_string, relation->rel_name.c_str(),
|
2004-09-01 00:29:59 +02:00
|
|
|
isc_arg_gds, isc_foreign_key_target_doesnt_exist, 0);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case idx_e_foreign_references_present:
|
2004-10-09 05:55:43 +02:00
|
|
|
ERR_post(isc_foreign_key, isc_arg_string, constraint_name,
|
2005-05-12 20:28:04 +02:00
|
|
|
isc_arg_string, relation->rel_name.c_str(),
|
2004-09-01 00:29:59 +02:00
|
|
|
isc_arg_gds, isc_foreign_key_references_present, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2005-05-12 20:28:04 +02:00
|
|
|
if (constraint.length() > 0)
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_unique_key_violation,
|
2004-10-09 05:55:43 +02:00
|
|
|
isc_arg_string, constraint_name,
|
2005-05-12 20:28:04 +02:00
|
|
|
isc_arg_string, relation->rel_name.c_str(), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
2004-10-09 05:55:43 +02:00
|
|
|
ERR_post(isc_no_dup, isc_arg_string, index_name, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#if ( !defined( REQUESTER) && !defined( SUPERCLIENT))
|
2003-10-03 03:53:34 +02:00
|
|
|
void ERR_error(int number)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E R R _ e r r o r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Post a user-level error. This is a temporary mechanism
|
|
|
|
* that will eventually disappear.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
TEXT errmsg[MAX_ERRMSG_LEN + 1];
|
|
|
|
|
|
|
|
DEBUG;
|
2004-07-07 05:43:20 +02:00
|
|
|
if (gds__msg_lookup(0, JRD_BUGCHK, number, sizeof(errmsg), errmsg, NULL) < 1)
|
|
|
|
sprintf(errmsg, "error code %d", number);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_random, isc_arg_string, ERR_cstring(errmsg), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#if ( !defined( REQUESTER) && !defined( SUPERCLIENT))
|
2003-10-03 03:53:34 +02:00
|
|
|
void ERR_error_msg(const TEXT* msg)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E R R _ e r r o r _ m s g
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Post a user-level error. This is a temporary mechanism
|
|
|
|
* that will eventually disappear.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
DEBUG;
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(isc_random, isc_arg_string, ERR_cstring(msg), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#if ( !defined( REQUESTER) && !defined( SUPERCLIENT))
|
2003-10-03 03:53:34 +02:00
|
|
|
void ERR_log(int facility, int number, const TEXT* message)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E R R _ l o g
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2002-09-20 12:23:15 +02:00
|
|
|
* Log a message to the firebird.log
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
TEXT errmsg[MAX_ERRMSG_LEN + 1];
|
2004-05-22 16:28:54 +02:00
|
|
|
thread_db* tdbb = JRD_get_thread_data();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
DEBUG;
|
2004-10-09 05:58:45 +02:00
|
|
|
if (message) {
|
|
|
|
strncpy(errmsg, message, sizeof(errmsg));
|
|
|
|
errmsg[sizeof(errmsg) - 1] = 0;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
2004-07-07 05:43:20 +02:00
|
|
|
if (gds__msg_lookup(0, facility, number, sizeof(errmsg), errmsg, NULL) < 1)
|
|
|
|
strcpy(errmsg, "Internal error code");
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-10-09 05:58:45 +02:00
|
|
|
const size_t len = strlen(errmsg);
|
|
|
|
fb_utils::snprintf(errmsg + len, sizeof(errmsg) - len, " (%d)", number);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-14 14:40:14 +01:00
|
|
|
gds__log("Database: %s\n\t%s", (tdbb && tdbb->tdbb_attachment) ?
|
|
|
|
tdbb->tdbb_attachment->att_filename.c_str() : "",
|
2004-03-07 08:58:55 +01:00
|
|
|
errmsg, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#if ( !defined( REQUESTER) && !defined( SUPERCLIENT))
|
2004-02-20 07:43:27 +01:00
|
|
|
bool ERR_post_warning(ISC_STATUS status, ...)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E R R _ p o s t _ w a r n i n g
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Post a warning to the current status vector.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
va_list args;
|
|
|
|
int type, len;
|
2004-02-20 07:43:27 +01:00
|
|
|
ISC_STATUS* q;
|
2001-05-23 15:26:42 +02:00
|
|
|
int indx = 0, warning_indx = 0;
|
2004-02-20 07:43:27 +01:00
|
|
|
ISC_STATUS* status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-24 01:28:06 +02:00
|
|
|
va_start(args, status);
|
2004-05-22 16:28:54 +02:00
|
|
|
status_vector = ((thread_db*) JRD_get_thread_data())->tdbb_status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-11-11 13:19:20 +01:00
|
|
|
if (status_vector[0] != isc_arg_gds ||
|
|
|
|
(status_vector[0] == isc_arg_gds && status_vector[1] == 0 &&
|
2004-07-07 05:43:20 +02:00
|
|
|
status_vector[2] != isc_arg_warning))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
/* this is a blank status vector */
|
2003-11-11 13:19:20 +01:00
|
|
|
status_vector[0] = isc_arg_gds;
|
2001-05-23 15:26:42 +02:00
|
|
|
status_vector[1] = 0;
|
2003-11-11 13:19:20 +01:00
|
|
|
status_vector[2] = isc_arg_end;
|
2001-05-23 15:26:42 +02:00
|
|
|
indx = 2;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* find end of a status vector */
|
|
|
|
PARSE_STATUS(status_vector, indx, warning_indx);
|
|
|
|
if (indx)
|
|
|
|
--indx;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* stuff the warning */
|
|
|
|
if (indx + 3 < ISC_STATUS_LENGTH) {
|
2003-11-11 13:19:20 +01:00
|
|
|
status_vector[indx++] = isc_arg_warning;
|
2001-05-23 15:26:42 +02:00
|
|
|
status_vector[indx++] = status;
|
|
|
|
while ((type = va_arg(args, int)) && (indx + 3 < ISC_STATUS_LENGTH))
|
|
|
|
switch (status_vector[indx++] = type) {
|
2003-11-11 13:19:20 +01:00
|
|
|
case isc_arg_warning:
|
2003-04-10 08:49:16 +02:00
|
|
|
status_vector[indx++] = (ISC_STATUS) va_arg(args, ISC_STATUS);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
2003-11-11 13:19:20 +01:00
|
|
|
case isc_arg_string:
|
2004-02-20 07:43:27 +01:00
|
|
|
q = reinterpret_cast<ISC_STATUS*>(va_arg(args, TEXT*));
|
2004-05-06 20:09:24 +02:00
|
|
|
if (strlen((TEXT *) q) >= (size_t)MAX_ERRSTR_LEN) {
|
2003-11-11 13:19:20 +01:00
|
|
|
status_vector[(indx - 1)] = isc_arg_cstring;
|
2001-05-23 15:26:42 +02:00
|
|
|
status_vector[indx++] = MAX_ERRSTR_LEN;
|
|
|
|
}
|
2004-02-20 07:43:27 +01:00
|
|
|
// TEXT* forced to platform's int and stored as ISC_STATUS !!!
|
|
|
|
status_vector[indx++] = (ISC_STATUS)(U_IPTR) q;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
2003-11-11 13:19:20 +01:00
|
|
|
case isc_arg_interpreted:
|
2003-04-10 08:49:16 +02:00
|
|
|
status_vector[indx++] = va_arg(args, ISC_STATUS);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
2003-11-11 13:19:20 +01:00
|
|
|
case isc_arg_cstring:
|
2001-05-23 15:26:42 +02:00
|
|
|
len = va_arg(args, int);
|
|
|
|
status_vector[indx++] =
|
2003-04-10 08:49:16 +02:00
|
|
|
(ISC_STATUS) (len >= MAX_ERRSTR_LEN) ? MAX_ERRSTR_LEN : len;
|
2004-02-20 07:43:27 +01:00
|
|
|
// TEXT* forced to platform's int and stored as ISC_STATUS !!!
|
|
|
|
status_vector[indx++] = (ISC_STATUS) (U_IPTR) va_arg(args, TEXT*);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
2003-11-11 13:19:20 +01:00
|
|
|
case isc_arg_number:
|
2003-04-10 08:49:16 +02:00
|
|
|
status_vector[indx++] = (ISC_STATUS) va_arg(args, SLONG);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
2003-11-11 13:19:20 +01:00
|
|
|
case isc_arg_vms:
|
|
|
|
case isc_arg_unix:
|
|
|
|
case isc_arg_win32:
|
2001-05-23 15:26:42 +02:00
|
|
|
default:
|
2003-04-10 08:49:16 +02:00
|
|
|
status_vector[indx++] = (ISC_STATUS) va_arg(args, int);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
2003-11-11 13:19:20 +01:00
|
|
|
status_vector[indx] = isc_arg_end;
|
2004-09-25 12:28:09 +02:00
|
|
|
va_end(args);
|
2004-02-20 07:43:27 +01:00
|
|
|
return true;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* not enough free space */
|
2004-09-25 12:28:09 +02:00
|
|
|
va_end(args);
|
2004-02-20 07:43:27 +01:00
|
|
|
return false;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#if ( !defined( REQUESTER) && !defined( SUPERCLIENT))
|
2004-08-06 17:26:55 +02:00
|
|
|
void ERR_post_nothrow(ISC_STATUS status, ...)
|
2004-08-10 15:40:10 +02:00
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E R R _ p o s t _ n o t h r o w
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Create a status vector.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-08-06 17:26:55 +02:00
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
va_start(args, status);
|
|
|
|
internal_post(status, args);
|
2004-09-25 12:28:09 +02:00
|
|
|
va_end(args);
|
2004-08-21 11:29:46 +02:00
|
|
|
}
|
|
|
|
|
2004-08-06 17:26:55 +02:00
|
|
|
|
2003-10-03 03:53:34 +02:00
|
|
|
void ERR_post(ISC_STATUS status, ...)
|
2004-08-10 15:40:10 +02:00
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E R R _ p o s t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Create a status vector and return to
|
|
|
|
* the user.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-08-06 17:26:55 +02:00
|
|
|
{
|
|
|
|
va_list args;
|
|
|
|
va_start(args, status);
|
|
|
|
internal_post(status, args);
|
2004-09-25 12:28:09 +02:00
|
|
|
va_end(args);
|
|
|
|
|
2004-08-06 17:26:55 +02:00
|
|
|
DEBUG;
|
|
|
|
ERR_punt();
|
2004-08-21 11:29:46 +02:00
|
|
|
}
|
|
|
|
|
2004-08-06 17:26:55 +02:00
|
|
|
|
|
|
|
static void internal_post(ISC_STATUS status, va_list args)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2004-08-10 15:40:10 +02:00
|
|
|
* i n t e r n a l _ p o s t
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2004-08-10 15:40:10 +02:00
|
|
|
* Append status vector with new values.
|
|
|
|
* Used in ERR_post and ERR_post_nothrow
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY tmp_status, warning_status;
|
2001-05-23 15:26:42 +02:00
|
|
|
int i, tmp_status_len = 0, status_len = 0, err_status_len = 0;
|
|
|
|
int warning_count = 0, warning_indx = 0;
|
|
|
|
|
2004-05-22 16:28:54 +02:00
|
|
|
ISC_STATUS* status_vector = ((thread_db*) JRD_get_thread_data())->tdbb_status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* stuff the status into temp buffer */
|
|
|
|
MOVE_CLEAR(tmp_status, sizeof(tmp_status));
|
2004-08-06 17:26:55 +02:00
|
|
|
STUFF_STATUS_function(tmp_status, status, args);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* calculate length of the status */
|
|
|
|
PARSE_STATUS(tmp_status, tmp_status_len, warning_indx);
|
2003-11-04 00:59:24 +01:00
|
|
|
fb_assert(warning_indx == 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-11-11 13:19:20 +01:00
|
|
|
if (status_vector[0] != isc_arg_gds ||
|
|
|
|
(status_vector[0] == isc_arg_gds && status_vector[1] == 0 &&
|
2004-07-07 05:43:20 +02:00
|
|
|
status_vector[2] != isc_arg_warning))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
/* this is a blank status vector just stuff the status */
|
|
|
|
MOVE_FASTER(tmp_status, status_vector,
|
2003-04-10 08:49:16 +02:00
|
|
|
sizeof(ISC_STATUS) * tmp_status_len);
|
2004-08-06 17:26:55 +02:00
|
|
|
return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
PARSE_STATUS(status_vector, status_len, warning_indx);
|
|
|
|
if (status_len)
|
|
|
|
--status_len;
|
|
|
|
|
|
|
|
/* check for duplicated error code */
|
|
|
|
for (i = 0; i < ISC_STATUS_LENGTH; i++) {
|
2003-11-11 13:19:20 +01:00
|
|
|
if (status_vector[i] == isc_arg_end && i == status_len)
|
2001-05-23 15:26:42 +02:00
|
|
|
break; /* end of argument list */
|
|
|
|
|
|
|
|
if (i && i == warning_indx)
|
|
|
|
break; /* vector has no more errors */
|
|
|
|
|
|
|
|
if (status_vector[i] == tmp_status[1] && i &&
|
2003-11-11 13:19:20 +01:00
|
|
|
status_vector[i - 1] != isc_arg_warning &&
|
2001-05-23 15:26:42 +02:00
|
|
|
i + tmp_status_len - 2 < ISC_STATUS_LENGTH &&
|
|
|
|
(memcmp(&status_vector[i], &tmp_status[1],
|
2004-07-07 05:43:20 +02:00
|
|
|
sizeof(ISC_STATUS) * (tmp_status_len - 2)) == 0))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
/* duplicate found */
|
2004-08-06 17:26:55 +02:00
|
|
|
return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* if the status_vector has only warnings then adjust err_status_len */
|
|
|
|
if ((err_status_len = i) == 2 && warning_indx)
|
|
|
|
err_status_len = 0;
|
|
|
|
|
|
|
|
if (warning_indx) {
|
|
|
|
/* copy current warning(s) to a temp buffer */
|
|
|
|
MOVE_CLEAR(warning_status, sizeof(warning_status));
|
|
|
|
MOVE_FASTER(&status_vector[warning_indx], warning_status,
|
2003-04-10 08:49:16 +02:00
|
|
|
sizeof(ISC_STATUS) * (ISC_STATUS_LENGTH - warning_indx));
|
2001-05-23 15:26:42 +02:00
|
|
|
PARSE_STATUS(warning_status, warning_count, warning_indx);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* add the status into a real buffer right in between last error
|
|
|
|
and first warning */
|
|
|
|
|
|
|
|
if ((i = err_status_len + tmp_status_len) < ISC_STATUS_LENGTH) {
|
|
|
|
MOVE_FASTER(tmp_status, &status_vector[err_status_len],
|
2003-04-10 08:49:16 +02:00
|
|
|
sizeof(ISC_STATUS) * tmp_status_len);
|
2001-05-23 15:26:42 +02:00
|
|
|
/* copy current warning(s) to the status_vector */
|
|
|
|
if (warning_count && i + warning_count - 1 < ISC_STATUS_LENGTH) {
|
|
|
|
MOVE_FASTER(warning_status, &status_vector[i - 1],
|
2003-04-10 08:49:16 +02:00
|
|
|
sizeof(ISC_STATUS) * warning_count);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
2004-08-06 17:26:55 +02:00
|
|
|
return;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
#if ( !defined( REQUESTER) && !defined(SUPERCLIENT))
|
2003-10-03 03:53:34 +02:00
|
|
|
void ERR_punt(void)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E R R _ p u n t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Error stuff has been copied to status vector. Now punt.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-05-22 16:28:54 +02:00
|
|
|
thread_db* tdbb = JRD_get_thread_data();
|
2004-03-07 08:58:55 +01:00
|
|
|
Database* dbb = tdbb->tdbb_database;
|
2001-12-24 03:51:06 +01:00
|
|
|
|
|
|
|
if (dbb && (dbb->dbb_flags & DBB_bugcheck))
|
|
|
|
{
|
2004-05-27 18:26:52 +02:00
|
|
|
gds__log_status(tdbb->tdbb_attachment->att_filename.hasData() ?
|
2006-03-03 17:06:25 +01:00
|
|
|
tdbb->tdbb_attachment->att_filename.c_str() : "Database unknown in ERR_punt on bugcheck",
|
2003-10-29 11:53:47 +01:00
|
|
|
tdbb->tdbb_status_vector);
|
2004-04-10 02:25:22 +02:00
|
|
|
if (Config::getBugcheckAbort())
|
|
|
|
abort();
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-03-01 04:35:23 +01:00
|
|
|
Firebird::status_exception::raise(tdbb->tdbb_status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2003-10-03 03:53:34 +02:00
|
|
|
const TEXT* ERR_string(const TEXT* in_string, int length)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E R R _ s t r i n g
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Copy a stack local string into an area that is
|
|
|
|
* guaranteed to exist for a reasonable length of time.
|
|
|
|
* Only strings that are to be placed in status vectors
|
|
|
|
* should use this routine.
|
|
|
|
* A circular buffer is used to hold these strings. It
|
|
|
|
* is independent of the JRD allocator mechanism.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-03-01 04:35:23 +01:00
|
|
|
return Firebird::status_nstring(in_string, length);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#if ( !defined( REQUESTER) && !defined( SUPERCLIENT))
|
2003-10-03 03:53:34 +02:00
|
|
|
void ERR_warning(ISC_STATUS status, ...)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* E R R _ w a r n i n g
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Write an error out to the status vector but
|
2001-12-29 12:41:29 +01:00
|
|
|
* don't throw an exception. This allows
|
2001-05-23 15:26:42 +02:00
|
|
|
* sending a warning message back to the user
|
|
|
|
* without stopping execution of a request. Note
|
|
|
|
* that subsequent errors can supersede this one.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-05-22 16:28:54 +02:00
|
|
|
thread_db* tdbb = JRD_get_thread_data();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
STUFF_STATUS(tdbb->tdbb_status_vector, status);
|
|
|
|
DEBUG;
|
|
|
|
tdbb->tdbb_request->req_flags |= req_warning;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#if ( !defined( REQUESTER) && !defined( SUPERCLIENT))
|
2004-09-24 02:11:32 +02:00
|
|
|
static void internal_error(ISC_STATUS status, int number,
|
|
|
|
const TEXT* file, int line)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n t e r n a l _ e r r o r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Things seem to be going poorly today.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
TEXT errmsg[MAX_ERRMSG_LEN + 1];
|
|
|
|
|
|
|
|
DEBUG;
|
2004-07-07 05:43:20 +02:00
|
|
|
if (gds__msg_lookup(0, JRD_BUGCHK, number, sizeof(errmsg), errmsg, NULL) < 1)
|
|
|
|
strcpy(errmsg, "Internal error code");
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-10-09 05:58:45 +02:00
|
|
|
const size_t len = strlen(errmsg);
|
|
|
|
|
2004-09-24 02:11:32 +02:00
|
|
|
if (file) {
|
|
|
|
// Remove path information
|
2004-10-09 05:58:45 +02:00
|
|
|
const TEXT* ptr = (TEXT*)file + strlen(file);
|
2004-09-24 02:11:32 +02:00
|
|
|
for (; ptr > file; ptr--) {
|
|
|
|
if ((*ptr == '/') || (*ptr == '\\')) {
|
|
|
|
ptr++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2004-10-09 05:58:45 +02:00
|
|
|
fb_utils::snprintf(errmsg + len, sizeof(errmsg) - len,
|
|
|
|
" (%d), file: %s line: %d", number, ptr, line);
|
2004-09-24 02:11:32 +02:00
|
|
|
}
|
|
|
|
else {
|
2004-10-09 05:58:45 +02:00
|
|
|
fb_utils::snprintf(errmsg + len, sizeof(errmsg) - len, " (%d)", number);
|
2004-09-24 02:11:32 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-11-11 13:19:20 +01:00
|
|
|
ERR_post(status, isc_arg_string, ERR_cstring(errmsg), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
2004-03-07 08:58:55 +01:00
|
|
|
|