8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-24 12:03:02 +01:00
firebird-mirror/src/jrd/status.cpp

150 lines
3.5 KiB
C++

/*
* PROGRAM: Firebird exceptions classes
* MODULE: status.cpp
* DESCRIPTION: Status vector filling and parsing.
*
* 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): ______________________________________.
*
* 2002.10.28 Sean Leyne - Code cleanup, removed obsolete "MPEXL" port
*
* 2002.10.29 Sean Leyne - Removed obsolete "Netware" port
*
*/
#include "firebird.h"
#include <stdlib.h>
#include <stdarg.h>
#include <string.h>
#include "../jrd/status.h"
#include "../jrd/gdsassert.h"
#include "gen/iberror.h"
#include "../jrd/gds_proto.h"
/* The following function is used to stuff variable number of error message
arguments from stack to the status_vector. This macro should be the
first statement in a routine where it is invoked. */
/* Get the addresses of the argument vector and the status vector, and do
word-wise copy. */
void STUFF_STATUS_function(ISC_STATUS* status_vector, ISC_STATUS status, va_list args)
{
int type, len;
ISC_STATUS* p = status_vector;
*p++ = isc_arg_gds;
*p++ = status;
while ((type = va_arg(args, int)) && ((p - status_vector) < 17))
{
switch (*p++ = type)
{
case isc_arg_gds:
*p++ = va_arg(args, ISC_STATUS);
break;
case isc_arg_string:
{
ISC_STATUS* q = va_arg(args, ISC_STATUS*);
if (strlen((TEXT *) q) >= (size_t) MAX_ERRSTR_LEN)
{
*(p - 1) = isc_arg_cstring;
*p++ = (ISC_STATUS) MAX_ERRSTR_LEN;
}
*p++ = (ISC_STATUS) q;
}
break;
case isc_arg_interpreted:
*p++ = (ISC_STATUS) va_arg(args, TEXT *);
break;
case isc_arg_cstring:
len = (int) va_arg(args, int);
*p++ = (ISC_STATUS) (len >= MAX_ERRSTR_LEN) ? MAX_ERRSTR_LEN : len;
*p++ = (ISC_STATUS) va_arg(args, TEXT *);
break;
case isc_arg_number:
*p++ = (ISC_STATUS) va_arg(args, SLONG);
break;
case isc_arg_vms:
case isc_arg_unix:
case isc_arg_win32:
default:
*p++ = (ISC_STATUS) va_arg(args, int);
break;
}
}
*p = isc_arg_end;
}
/** Check that we never overrun the status vector. The status
* vector is 20 elements. The maximum is 3 entries for a
* type. So check for 17 or less
*/
void PARSE_STATUS(const ISC_STATUS* status_vector, int &length, int &warning)
{
warning = 0;
length = 0;
int i = 0;
for (; status_vector[i] != isc_arg_end; i++, length++)
{
switch (status_vector[i])
{
case isc_arg_warning:
if (!warning)
warning = i; // find the very first
// fallthrought intended
case isc_arg_gds:
case isc_arg_string:
case isc_arg_number:
case isc_arg_interpreted:
case isc_arg_vms:
case isc_arg_unix:
case isc_arg_win32:
i++;
length++;
break;
case isc_arg_cstring:
i += 2;
length += 2;
break;
default:
fb_assert(FALSE);
break;
}
}
if (i) {
length++; // isc_arg_end is counted
}
}