8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-27 18:03:04 +01:00
firebird-mirror/src/jrd/gds.cpp
2003-02-14 02:50:10 +00:00

4780 lines
107 KiB
C++

/*
* PROGRAM: JRD Access Method
* MODULE: gds.c
* DESCRIPTION: User callable routines
*
* 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.02.15 Sean Leyne - Code Cleanup, removed obsolete "IMP" port
* 2002.02.15 Sean Leyne - Code Cleanup, removed obsolete "M88K" port
*
* 2002.10.29 Sean Leyne - Removed obsolete "Netware" port
*
* 2002.10.30 Sean Leyne - Removed support for obsolete "PC_PLATFORM" define
*
*/
#define IO_RETRY 20
#ifdef SHLIB_DEFS
#define LOCAL_SHLIB_DEFS
#endif
// 11 Sept 2002 Nickolay Samofatov
// this defined in included dsc.h
//#define ISC_TIME_SECONDS_PRECISION 10000L
//#define ISC_TIME_SECONDS_PRECISION_SCALE -4
#include "firebird.h"
#include "../jrd/ib_stdio.h"
#include <stdlib.h>
#include <string.h>
#include "../jrd/common.h"
#include "../jrd/gdsassert.h"
#include "../jrd/file_params.h"
#include "../jrd/msg_encode.h"
#include "../jrd/iberr.h"
#include "../jrd/gds_proto.h"
#include "../jrd/os/path_utils.h"
#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif
#ifndef WIN_NT
#include <sys/param.h>
#endif
#include <stdarg.h>
#include "../jrd/jrd_time.h"
#include "../jrd/misc.h"
#ifdef DARWIN
#include <CoreFoundation/CFBundle.h>
#include <CoreFoundation/CFURL.h>
#endif
#if defined(WIN_NT)
#include <io.h>
#include <sys/types.h>
#include <sys/timeb.h>
#endif
#ifdef VMS
#include <file.h>
#include <ib_perror.h>
#include <descrip.h>
#include <types.h>
#include <stat.h>
#include <rmsdef.h>
#else /* !VMS */
#include <errno.h>
#include <sys/types.h>
#include <sys/stat.h>
#ifndef WIN_NT
#include <sys/file.h>
#endif
#ifndef O_RDWR
#include <fcntl.h>
#endif
#ifdef UNIX
#if (defined SUPERSERVER) || (defined SOLARIS)
#include <sys/mman.h>
#include <sys/resource.h>
#include "../jrd/err_proto.h"
#endif
#endif
#endif /* VMS */
#include "../common/config/config.h"
/* Turn on V4 mutex protection for gds__alloc/free */
#ifdef WIN_NT
#define V4_THREADING
#endif
#ifdef SOLARIS_MT
#define V4_THREADING
#endif
#ifdef SUPERSERVER
#define V4_THREADING /* RFM: 9/22/2000 fix from Inprise tree,
Inprise bug 114840 */
#endif
/* The following ifdef was added to build thread safe gds shared
library on linux platform. It seems the gdslib works now (20020220)
with thread enabled applications. Anyway, more tests should be
done as I don't have deep knowledge of the interbase/firebird
engine and this change may imply side effect I haven't known
about yet. Tomas Nejedlik (tomas@nejedlik.cz)
*/
#if ((defined(LINUX) || defined(FREEBSD)) && defined(SUPERCLIENT))
#define V4_THREADING
#endif
#ifdef SUPERSERVER
static const TEXT gdslogid[] = " (Server)";
#else
#ifdef SUPERCLIENT
static const TEXT gdslogid[] = " (Client)";
#else
static const TEXT gdslogid[] = "";
#endif
#endif
#include "gen/sql_code.h"
#include "../jrd/thd.h"
#include "../jrd/gdsold.h"
//#include "../jrd/gen/codes.h" // is already included in gdsold.h
#include "../jrd/blr.h"
#include "../jrd/msg.h"
#include "../jrd/fil.h"
#include "../jrd/gds_proto.h"
#include "../jrd/isc_proto.h"
#ifndef REQUESTER
#include "../jrd/isc_i_proto.h"
#endif
#ifdef WIN_NT
#define _WINSOCKAPI_
#include <windows.h>
#include "thd_proto.h"
#include "err_proto.h"
#undef leave
#ifdef TEXT
#undef TEXT
#endif
#define TEXT SCHAR
#endif /* WIN_NT */
extern "C" {
#ifdef sparc
#ifndef SOLARIS
extern int ib_printf();
#endif
#endif
#if !(defined VMS || defined WIN_NT || defined linux || defined FREEBSD || defined NETBSD || defined DARWIN )
extern int errno;
extern SCHAR *sys_errlist[];
extern int sys_nerr;
#endif
#ifndef PRINTF
#define PRINTF ib_printf
#endif
static char *ib_prefix = 0;
static char *ib_prefix_lock = 0;
static char *ib_prefix_msg = 0;
static const SCHAR *FAR_VARIABLE const messages[] = {
#include "gen/msgs.h"
0 /* Null entry to terminate list */
};
#if (ALIGNMENT == 8)
#define GDS_ALLOC_ALIGNMENT 8
#else
#define GDS_ALLOC_ALIGNMENT 4
#endif
#define ALLOC_ROUNDUP(n) ROUNDUP ((n),GDS_ALLOC_ALIGNMENT)
typedef struct free {
struct free *free_next;
SLONG free_length;
#ifdef DEBUG_GDS_ALLOC
SLONG free_count;
#endif
} *FREE;
#ifdef DEBUG_GDS_ALLOC
#define DBG_MEM_FILE "memory.dbg"
#ifndef NEWLINE
#define NEWLINE "\n"
#endif
typedef struct alloc
{
union
{
ULONG alloc_length; /* aka block[0] */
struct free alloc_freed;
} alloc_status;
TEXT alloc_magic1[6]; /* set to ALLOC_HEADER1_MAGIC */
TEXT alloc_filename[30]; /* filename of caller */
ULONG alloc_lineno; /* lineno of caller */
ULONG alloc_requested_length; /* requested len from caller */
ULONG alloc_flags; /* Internal flags */
ULONG alloc_callno; /* Which call to gds__alloc() */
struct alloc *alloc_next; /* Next buffer in the chain */
TEXT alloc_magic2[8]; /* set to ALLOC_HEADER2_MAGIC */
} *ALLOC;
#define ALLOC_HEADER_SIZE ALLOC_ROUNDUP(sizeof (struct alloc))
#define ALLOC_TAILER_SIZE 6
#define ALLOC_HEADER1_MAGIC_ALLOCATED "FBTAG"
#define ALLOC_HEADER1_MAGIC_FREED "FREED"
#define ALLOC_HEADER2_MAGIC "@START1#"
#define ALLOC_TAILER_MAGIC "END01"
#define ALLOC_FREED_PATTERN 0xCB
#define ALLOC_ALLOC_PATTERN 0xFD
#define ALLOC_FREQUENCY 99 /* how often to validate memory */
static ALLOC gds_alloc_chain = NULL;
static ULONG gds_alloc_call_count = 0;
static ULONG gds_alloc_state = ~0;
static ULONG gds_alloc_nomem = ~0;
static ULONG gds_free_call_count = 0;
static ULONG gds_bug_alloc_count = 0;
static ULONG gds_bug_free_count = 0;
static struct alloc
*gds_alloc_watchpoint = NULL; /* For SUN debugging */
static ULONG gds_alloc_watch_call_count = 0;
#define SAVE_DEBUG_INFO(b,f,l) {\
strncpy (((ALLOC) (b))->alloc_magic1, ALLOC_HEADER1_MAGIC_ALLOCATED, sizeof (((ALLOC) (b))->alloc_magic1)); \
strncpy (((ALLOC) (b))->alloc_magic2, ALLOC_HEADER2_MAGIC, sizeof (((ALLOC) (b))->alloc_magic2)); \
strncpy (((ALLOC) (b))->alloc_filename, f, sizeof (((ALLOC) (b))->alloc_filename)); \
((ALLOC) (b))->alloc_requested_length = size_request; \
((ALLOC) (b))->alloc_flags = 0; \
((ALLOC) (b))->alloc_lineno = l; \
((ALLOC) (b))->alloc_callno = gds_alloc_call_count; \
\
/* Link this block into the list of blocks */ \
((ALLOC) (b))->alloc_next = gds_alloc_chain; \
gds_alloc_chain = ((ALLOC) (b)); \
\
/* Set the "tailer" at the end of the allocated memory */ \
memcpy ( ((BLOB_PTR *) (b)) + ALLOC_HEADER_SIZE + size_request, \
ALLOC_TAILER_MAGIC, ALLOC_TAILER_SIZE); \
\
/* Initialize the allocated memory */ \
memset (((UCHAR *) (b)) + ALLOC_HEADER_SIZE, ALLOC_ALLOC_PATTERN, size_request); \
}
static void gds_alloc_validate(ALLOC);
static BOOLEAN gds_alloc_validate_free_pattern(UCHAR *, ULONG);
static void gds_alloc_validate_freed(ALLOC);
static void validate_memory(void);
#endif
#ifndef SAVE_DEBUG_INFO
#define SAVE_DEBUG_INFO(b,f,l)
#endif
#ifndef ALLOC_HEADER_SIZE
#define ALLOC_HEADER_SIZE ALLOC_ROUNDUP(sizeof (ULONG))
#endif
#ifndef ALLOC_TAILER_SIZE
#define ALLOC_TAILER_SIZE 0
#endif
#define ALLOC_OVERHEAD (ALLOC_HEADER_SIZE + ALLOC_TAILER_SIZE)
#ifndef GDS_ALLOC_EXTEND_SIZE
#ifdef SUPERCLIENT
#define GDS_ALLOC_EXTEND_SIZE 8192
#else
#define GDS_ALLOC_EXTEND_SIZE 102400
#endif
#endif
#ifndef FOPEN_APPEND_TYPE
#define FOPEN_APPEND_TYPE "a"
#endif
#ifndef O_BINARY
#define O_BINARY 0
#endif
#ifndef GENERIC_SQLCODE
#define GENERIC_SQLCODE -999
#endif
static char ib_prefix_val[MAXPATHLEN];
static char ib_prefix_lock_val[MAXPATHLEN];
static char ib_prefix_msg_val[MAXPATHLEN];
#ifndef INCLUDE_FB_TYPES_H
#include "../include/fb_types.h"
#endif
typedef struct gds_msg
{
ULONG msg_top_tree;
int msg_file;
USHORT msg_bucket_size;
USHORT msg_levels;
SCHAR msg_bucket[1];
} *GDS_MSG;
typedef struct ctl
{
UCHAR *ctl_blr; /* Running blr string */
UCHAR *ctl_blr_start; /* Original start of blr string */
void (*ctl_routine) (); /* Call back */
SCHAR *ctl_user_arg; /* User argument */
TEXT *ctl_ptr;
SSHORT ctl_language;
TEXT ctl_buffer[1024];
} *CTL;
#ifdef DEV_BUILD
void GDS_breakpoint(int);
#endif
#ifdef __GNUC__
static void blr_error(CTL, TEXT *, ...) __attribute__ ((format(printf,2,3)));
static void blr_format(CTL, const char *, ...) __attribute__ ((format(printf,2,3)));
#else
static void blr_error(CTL, TEXT *, ...);
static void blr_format(CTL, const char *, ...);
#endif
static void blr_indent(CTL, SSHORT);
static void blr_print_blr(CTL, UCHAR);
static SCHAR blr_print_byte(CTL);
static SCHAR blr_print_char(CTL);
static void blr_print_cond(CTL);
static int blr_print_dtype(CTL);
static void blr_print_join(CTL);
static SLONG blr_print_line(CTL, SSHORT);
static void blr_print_verb(CTL, SSHORT);
static int blr_print_word(CTL);
static void cleanup_malloced_memory(void *);
static ULONG free_memory(void *);
static void init(void);
static int yday(struct tm *);
static void ndate(SLONG, struct tm *);
static GDS_DATE nday(struct tm *);
static void sanitize(TEXT *);
/* Generic cleanup handlers */
typedef struct clean
{
struct clean* clean_next;
void (*clean_routine)(void*);
void* clean_arg;
} *CLEAN;
static CLEAN cleanup_handlers = NULL;
static FREE pool = NULL;
static void** malloced_memory = NULL;
static GDS_MSG default_msg = NULL;
static SLONG initialized = FALSE;
static MUTX_T alloc_mutex;
#ifdef DEV_BUILD
SLONG gds_delta_alloc = 0, gds_max_alloc = 0; /* Used in DBG_memory */
#endif
#ifdef DEBUG_GDS_ALLOC
void* API_ROUTINE gds__alloc_debug(SLONG size_request,
TEXT* filename,
ULONG lineno)
{
return getDefaultMemoryPool()->allocate(size_request, 0, filename, lineno);
// return getDefaultMemoryPool()->calloc(size_request, 0, filename, lineno);
}
#else
void* API_ROUTINE gds__alloc(SLONG size_request)
{
return getDefaultMemoryPool()->allocate(size_request);
// return getDefaultMemoryPool()->calloc(size_request);
}
#endif
ULONG API_ROUTINE gds__free(void* blk) {
getDefaultMemoryPool()->deallocate(blk);
return 0;
}
#ifdef UNIX
static SLONG gds_pid = 0;
#endif
#if (defined SOLARIS && defined SUPERSERVER && !defined(MAP_ANON))
static int anon_fd = -1;
static int page_size = -1;
#endif
/* VMS structure to declare exit handler */
#ifdef VMS
static SLONG exit_status = 0;
static struct
{
SLONG link;
int (*exit_handler) ();
SLONG args;
SLONG arg[1];
} exit_description;
#endif
#ifdef SHLIB_DEFS
#define sprintf (*_libgds_sprintf)
#define vsprintf (*_libgds_vsprintf)
#define strlen (*_libgds_strlen)
#define strcpy (*_libgds_strcpy)
#define _iob (*_libgds__iob)
#define getpid (*_libgds_getpid)
#define ib_fprintf (*_libgds_fprintf)
#define ib_printf (*_libgds_printf)
#define ib_fopen (*_libgds_fopen)
#define ib_fclose (*_libgds_fclose)
#define sys_nerr (*_libgds_sys_nerr)
#define sys_errlist (*_libgds_sys_errlist)
#define malloc (*_libgds_malloc)
#define gettimeofday(*_libgds_gettimeofday)
#define ctime (*_libgds_ctime)
#define getenv (*_libgds_getenv)
#define lseek (*_libgds_lseek)
#define read (*_libgds_read)
#define memcpy (*_libgds_memcpy)
#define open (*_libgds_open)
#define strcat (*_libgds_strcat)
#define ib_fputs (*_libgds_fputs)
#define ib_fputc (*_libgds_fputc)
#define mktemp (*_libgds_mktemp)
#define unlink (*_libgds_unlink)
#define close (*_libgds_close)
#define strncmp (*_libgds_strncmp)
#define time (*_libgds_time)
#define ib_fflush (*_libgds_fflush)
#define errno (*_libgds_errno)
#define umask (*_libgds_umask)
#define atexit (*_libgds_atexit)
#define ib_vfprintf (*_libgds_vfprintf)
extern int sprintf();
extern int vsprintf();
extern int strlen();
extern SCHAR *strcpy();
extern IB_FILE _iob[];
extern pid_t getpid();
extern int ib_fprintf();
extern int ib_printf();
extern IB_FILE *ib_fopen();
extern int ib_fclose();
extern int sys_nerr;
extern SCHAR *sys_errlist[];
extern void *malloc();
extern int gettimeofday();
extern SCHAR *ctime();
extern SCHAR *getenv();
extern off_t lseek();
extern int read();
extern void *memcpy();
extern int open();
extern SCHAR *strcat();
extern int ib_fputs();
extern int ib_fputc();
extern SCHAR *mktemp();
extern int unlink();
extern int close();
extern int strncmp();
extern time_t time();
extern int ib_fflush();
extern int errno;
extern mode_t umask();
extern int atexit();
extern int ib_vfprintf();
#endif /* SHLIB_DEFS */
/* BLR Pretty print stuff */
#define PRINT_VERB blr_print_verb (control, level)
#define PRINT_LINE blr_print_line (control, (SSHORT) offset)
#define PRINT_BYTE blr_print_byte (control)
#define PRINT_CHAR blr_print_char (control)
#define PRINT_WORD blr_print_word (control)
#define PRINT_COND blr_print_cond (control)
#define PRINT_DTYPE blr_print_dtype (control)
#define PRINT_JOIN blr_print_join (control)
#define BLR_BYTE *(control->ctl_blr)++
#define PUT_BYTE(byte) *(control->ctl_ptr)++ = byte
#define op_line 1
#define op_verb 2
#define op_byte 3
#define op_word 4
#define op_pad 5
#define op_dtype 6
#define op_message 7
#define op_literal 8
#define op_begin 9
#define op_map 10
#define op_args 11
#define op_union 12
#define op_indent 13
#define op_join 14
#define op_parameters 15
#define op_error_handler 16
#define op_set_error 17
#define op_literals 18
#define op_relation 20
static const UCHAR
/* generic print formats */
zero[] = { op_line, 0 },
one[] = { op_line, op_verb, 0},
two[] = { op_line, op_verb, op_verb, 0},
three[] = { op_line, op_verb, op_verb, op_verb, 0},
field[] = { op_byte, op_byte, op_literal, op_pad, op_line, 0},
byte[] = { op_byte, op_line, 0},
byte_args[] = { op_byte, op_line, op_args, 0},
byte_byte[] = { op_byte, op_byte, op_line, 0},
byte_verb[] = { op_byte, op_line, op_verb, 0},
byte_verb_verb[] = { op_byte, op_line, op_verb, op_verb, 0},
byte_literal[] = { op_byte, op_literal, op_line, 0},
byte_byte_verb[] = { op_byte, op_byte, op_line, op_verb, 0},
parm[] = { op_byte, op_word, op_line, 0}, /* also field id */
parm2[] = { op_byte, op_word, op_word, op_line, 0},
parm3[] = { op_byte, op_word, op_word, op_word, op_line, 0},
/* formats specific to a verb */
begin[] = { op_line, op_begin, op_verb, 0},
literal[] = { op_dtype, op_literal, op_line, 0},
message[] = { op_byte, op_word, op_line, op_message, 0},
rse[] = { op_byte, op_line, op_begin, op_verb, 0},
relation[] = { op_byte, op_literal, op_pad, op_byte, op_line, 0},
relation2[] = { op_byte, op_literal, op_line, op_indent, op_byte,
op_literal, op_pad, op_byte, op_line, 0},
aggregate[] = { op_byte, op_line, op_verb, op_verb, op_verb, 0},
rid[] = { op_word, op_byte, op_line, 0},
rid2[] = { op_word, op_byte, op_literal, op_pad, op_byte, op_line, 0},
union_ops[] = { op_byte, op_byte, op_line, op_union, 0},
map[] = { op_word, op_line, op_map, 0},
function[] = { op_byte, op_literal, op_byte, op_line, op_args, 0},
gen_id[] = { op_byte, op_literal, op_line, op_verb, 0},
declare[] = { op_word, op_dtype, op_line, 0},
variable[] = { op_word, op_line, 0},
indx[] = { op_line, op_verb, op_indent, op_byte, op_line, op_args, 0},
find[] = { op_byte, op_verb, op_verb, op_indent, op_byte, op_line, op_args, 0},
seek[] = { op_line, op_verb, op_verb, 0},
join[] = { op_join, op_line, 0},
exec_proc[] = { op_byte, op_literal, op_line, op_indent, op_word, op_line,
op_parameters, op_indent, op_word, op_line, op_parameters, 0},
procedure[] = { op_byte, op_literal, op_pad, op_byte, op_line, op_indent,
op_word, op_line, op_parameters, 0},
pid[] = { op_word, op_pad, op_byte, op_line, op_indent, op_word,
op_line, op_parameters, 0},
error_handler[] = { op_word, op_line, op_error_handler, 0},
set_error[] = { op_set_error, op_line, 0},
cast[] = { op_dtype, op_line, op_verb, 0},
indices[] = { op_byte, op_line, op_literals, 0},
lock_relation[] = { op_line, op_indent, op_relation, op_line, op_verb, 0},
range_relation[] = { op_line, op_verb, op_indent, op_relation, op_line, 0},
extract[] = { op_line, op_byte, op_verb, 0};
static const struct
{
const char *blr_string;
const UCHAR *blr_operators;
} FAR_VARIABLE blr_table[] =
{
#pragma FB_COMPILER_MESSAGE("Fix this!")
#include "../jrd/blp.h"
{0, 0}
};
#define FB_ENV "FIREBIRD"
#define FB_LOCK_ENV "FIREBIRD_LOCK"
#define FB_MSG_ENV "FIREBIRD_MSG"
#ifdef WIN_NT
#define EXPAND_PATH(relative, absolute) _fullpath(absolute, relative, MAXPATHLEN)
#define COMPARE_PATH(a,b) _stricmp(a,b)
#else
#define EXPAND_PATH(relative, absolute) realpath(relative, absolute)
#define COMPARE_PATH(a,b) strcmp(a,b)
#endif
#ifdef SUPERSERVER
extern SLONG allr_delta_alloc;
} // extern "C"
extern SLONG alld_delta_alloc;
extern SLONG all_delta_alloc;
extern "C" {
SLONG trace_pools = 0;
SLONG free_map_debug = 0;
static void freemap(int cod);
void gds_print_delta_counters(IB_FILE *);
#endif // SUPERSERVER
#ifdef NOT_USED_OR_REPLACED
#ifdef DEBUG_GDS_ALLOC
void* API_ROUTINE gds__alloc_debug(SLONG size_request,
TEXT* filename,
ULONG lineno)
#else
void* API_ROUTINE gds__alloc(SLONG size_request)
#endif
{
/**************************************
*
* g d s _ $ a l l o c
*
**************************************
*
* Functional description
* Allocate a block of memory.
*
**************************************/
SLONG *block = NULL;
SLONG size;
ULONG factor;
FREE *next, *best, free_blk;
SLONG tail, best_tail = 0;
#ifdef VMS
SLONG status;
#endif
if (size_request <= 0) {
DEV_REPORT("gds__alloc: non-positive size allocation request\n"); /* TXNN */
BREAKPOINT(__LINE__);
return NULL;
}
if (!initialized)
init();
#ifdef DEBUG_GDS_ALLOC
gds_alloc_call_count++;
if (gds_alloc_call_count == gds_bug_alloc_count) {
PRINTF("gds_alloc() call %ld reached\n", gds_bug_alloc_count);
BREAKPOINT(__LINE__);
}
/* If we want to validate all memory on each call, do so now */
if ((gds_alloc_state < gds_alloc_call_count) ||
(gds_bug_alloc_count && (gds_bug_alloc_count < gds_alloc_call_count))
|| ((gds_alloc_call_count % ALLOC_FREQUENCY) == 0)) {
V4_MUTEX_LOCK(&alloc_mutex);
gds_alloc_report(ALLOC_silent, __FILE__, __LINE__);
V4_MUTEX_UNLOCK(&alloc_mutex);
}
/* Simulate out of memory */
if (gds_alloc_nomem <= gds_alloc_call_count)
return NULL;
#endif
/* Add the size of malloc overhead to the block, and then
* round up the block size according to the desired alignment
* for the platform
* Make sure that for a multiple of 1K allocate ALLOC_OVERHEAD
* times the multiple. This ensures that on freeing and subsequent
* re-allocating we don't fragment the free memory list
*/
factor = 1;
if (!(size_request & (1024 - 1)))
{
factor = size_request / 1024;
}
size = size_request + ALLOC_OVERHEAD * factor;
size = ROUNDUP(size, GDS_ALLOC_ALIGNMENT);
#ifdef DEV_BUILD
gds_delta_alloc += size;
#endif
V4_MUTEX_LOCK(&alloc_mutex);
/* Find the free block with the shortest tail */
best = NULL;
for (next = &pool; (free_blk = *next) != NULL; next = &(*next)->free_next)
{
if (free_blk->free_length <= 0 ||
(free_blk->free_next
&& (UCHAR HUGE_PTR *) free_blk + free_blk->free_length >
(UCHAR HUGE_PTR *) free_blk->free_next))
{
DEV_REPORT("gds__alloc: memory pool corrupted\n"); /* TXNN */
BREAKPOINT(__LINE__);
*next = NULL;
break;
}
if ((tail = free_blk->free_length - size) >= 0)
{
if (!best || tail < best_tail)
{
best = next;
if (!(best_tail = tail))
break;
}
}
}
/* If there was a fit, return the block */
if (best)
{
if (best_tail <= sizeof(struct free) + ALLOC_OVERHEAD)
{
/* There's only a little left over, give client the whole chunk */
size = (*best)->free_length;
block = (SLONG *) * best;
*best = (*best)->free_next;
#ifdef DEBUG_GDS_ALLOC
/* We're going to give some memory off the free list - before doing
* so, let's make sure it still looks like free memory.
*/
gds_alloc_validate_freed((ALLOC) block);
#endif /* DEBUG_GDS_ALLOC */
}
else
{
/* Shave off what we need from the end of the free chunk */
block = (SLONG *) ((UCHAR *) * best + best_tail);
(*best)->free_length -= size;
#ifdef DEBUG_GDS_ALLOC
/* We're going to give some memory off the free list - before doing
* so, let's make sure it still looks like free memory.
*/
(void) gds_alloc_validate_free_pattern((UCHAR *) block, (ULONG) size);
#endif /* DEBUG_GDS_ALLOC */
}
block[0] = size;
SAVE_DEBUG_INFO(block, filename, lineno);
V4_MUTEX_UNLOCK(&alloc_mutex);
return (UCHAR *) block + ALLOC_HEADER_SIZE;
}
/* We need to extend the address space. Do so in multiples of the
* extend quantity, release the extra, and return the block.
* If we can't get a multiple of the extend size, try again using
* the minimal amount of memory needed before returning "out of memory".
* The minimal amount must include the malloc/free overhead amounts.
*/
tail = (((size + sizeof(struct free) + ALLOC_ROUNDUP(sizeof(void *)) +
ALLOC_OVERHEAD) / GDS_ALLOC_EXTEND_SIZE) +
1) * GDS_ALLOC_EXTEND_SIZE;
#ifdef VMS
status = lib$get_vm(&tail, &block);
if (!(status & 1))
{
tail = (size + sizeof(struct free) +
ALLOC_ROUNDUP(sizeof(void *)) + ALLOC_OVERHEAD);
status = lib$get_vm(&tail, &block);
if (!(status & 1))
{
V4_MUTEX_UNLOCK(&alloc_mutex);
return NULL;
}
}
#define MEMORY_ALLOCATED
#endif
#ifndef MEMORY_ALLOCATED
if ((block = (SLONG *) malloc(tail)) == NULL)
{
tail = (size + sizeof(struct free) +
ALLOC_ROUNDUP(sizeof(void *)) + ALLOC_OVERHEAD);
if ((block = (SLONG *) malloc(tail)) == NULL)
{
V4_MUTEX_UNLOCK(&alloc_mutex);
return NULL;
}
}
#endif
#undef MEMORY_ALLOCATED
/* Insert the allocated block into a simple linked list */
*((void **) block) = (void *) malloced_memory;
malloced_memory = (void **) block;
block = (SLONG *) ((UCHAR *) block + ALLOC_ROUNDUP(sizeof(void *)));
#ifdef DEV_BUILD
gds_max_alloc += tail;
#endif
tail -= size + ALLOC_ROUNDUP(sizeof(void *));
#ifdef DEV_BUILD
gds_delta_alloc += tail;
#endif
#ifdef DEBUG_GDS_ALLOC
memset((UCHAR *) block, ALLOC_FREED_PATTERN, tail);
#endif
block[0] = tail;
(void) free_memory((SLONG *) ((UCHAR *) block + ALLOC_HEADER_SIZE));
#ifdef SUPERSERVER
if (free_map_debug)
freemap(0);
#endif
block = (SLONG *) ((UCHAR *) block + tail);
block[0] = size;
SAVE_DEBUG_INFO(block, filename, lineno);
V4_MUTEX_UNLOCK(&alloc_mutex);
return (UCHAR *) block + ALLOC_HEADER_SIZE;
}
#endif
STATUS API_ROUTINE gds__decode(STATUS code, USHORT* fac, USHORT* class_)
{
/**************************************
*
* g d s _ $ d e c o d e
*
**************************************
*
* Functional description
* Translate a status codes from system dependent form to
* network form.
*
**************************************/
if (!code)
{
return FB_SUCCESS;
}
else if ((code & ISC_MASK) != ISC_MASK)
{
/* not an ISC error message */
return code;
}
*fac = GET_FACILITY(code);
*class_ = GET_CLASS(code);
return GET_CODE(code);
}
void API_ROUTINE isc_decode_date(GDS_QUAD* date, void* times_arg)
{
/**************************************
*
* i s c _ d e c o d e _ d a t e
*
**************************************
*
* Functional description
* Convert from internal timestamp format to UNIX time structure.
*
* Note: this API is historical - the prefered entrypoint is
* isc_decode_timestamp
*
**************************************/
isc_decode_timestamp((GDS_TIMESTAMP*) date, times_arg);
}
void API_ROUTINE isc_decode_sql_date(GDS_DATE* date, void* times_arg)
{
/**************************************
*
* i s c _ d e c o d e _ s q l _ d a t e
*
**************************************
*
* Functional description
* Convert from internal DATE format to UNIX time structure.
*
**************************************/
struct tm *times;
times = (struct tm *) times_arg;
memset(times, 0, sizeof(*times));
ndate(*date, times);
times->tm_yday = yday(times);
if ((times->tm_wday = (*date + 3) % 7) < 0)
times->tm_wday += 7;
}
void API_ROUTINE isc_decode_sql_time(GDS_TIME * sql_time, void *times_arg)
{
/**************************************
*
* i s c _ d e c o d e _ s q l _ t i m e
*
**************************************
*
* Functional description
* Convert from internal TIME format to UNIX time structure.
*
**************************************/
ULONG minutes;
struct tm *times;
times = (struct tm *) times_arg;
memset(times, 0, sizeof(*times));
minutes = *sql_time / (ISC_TIME_SECONDS_PRECISION * 60);
times->tm_hour = minutes / 60;
times->tm_min = minutes % 60;
times->tm_sec = (*sql_time / ISC_TIME_SECONDS_PRECISION) % 60;
}
void API_ROUTINE isc_decode_timestamp(GDS_TIMESTAMP * date, void *times_arg)
{
/**************************************
*
* i s c _ d e c o d e _ t i m e s t a m p
*
**************************************
*
* Functional description
* Convert from internal timestamp format to UNIX time structure.
*
* Note: the date arguement is really an ISC_TIMESTAMP -- however the
* definition of ISC_TIMESTAMP is not available from all the source
* modules that need to use isc_encode_timestamp
*
**************************************/
SLONG minutes;
struct tm *times;
times = (struct tm *) times_arg;
memset(times, 0, sizeof(*times));
ndate(date->timestamp_date, times);
times->tm_yday = yday(times);
if ((times->tm_wday = (date->timestamp_date + 3) % 7) < 0)
times->tm_wday += 7;
minutes = date->timestamp_time / (ISC_TIME_SECONDS_PRECISION * 60);
times->tm_hour = minutes / 60;
times->tm_min = minutes % 60;
times->tm_sec = (date->timestamp_time / ISC_TIME_SECONDS_PRECISION) % 60;
}
STATUS API_ROUTINE gds__encode(STATUS code, USHORT facility)
{
/**************************************
*
* g d s _ $ e n c o d e
*
**************************************
*
* Functional description
* Translate a status codes from network format to system
* dependent form.
*
**************************************/
if (!code)
return FB_SUCCESS;
return ENCODE_ISC_MSG(code, facility);
}
void API_ROUTINE isc_encode_date(void *times_arg, GDS_QUAD * date)
{
/**************************************
*
* i s c _ e n c o d e _ d a t e
*
**************************************
*
* Functional description
* Convert from UNIX time structure to internal timestamp format.
*
* Note: This API is historical -- the prefered entrypoint is
* isc_encode_timestamp
*
**************************************/
isc_encode_timestamp(times_arg, (GDS_TIMESTAMP *) date);
}
void API_ROUTINE isc_encode_sql_date(void *times_arg, GDS_DATE * date)
{
/**************************************
*
* i s c _ e n c o d e _ s q l _ d a t e
*
**************************************
*
* Functional description
* Convert from UNIX time structure to internal date format.
*
**************************************/
*date = nday((struct tm *) times_arg);
}
void API_ROUTINE isc_encode_sql_time(void *times_arg, GDS_TIME * isc_time)
{
/**************************************
*
* i s c _ e n c o d e _ s q l _ t i m e
*
**************************************
*
* Functional description
* Convert from UNIX time structure to internal TIME format.
*
**************************************/
struct tm *times;
times = (struct tm *) times_arg;
*isc_time = ((times->tm_hour * 60 + times->tm_min) * 60 +
times->tm_sec) * ISC_TIME_SECONDS_PRECISION;
}
void API_ROUTINE isc_encode_timestamp(void *times_arg, GDS_TIMESTAMP * date)
{
/**************************************
*
* i s c _ e n c o d e _ t i m e s t a m p
*
**************************************
*
* Functional description
* Convert from UNIX time structure to internal timestamp format.
*
* Note: the date arguement is really an ISC_TIMESTAMP -- however the
* definition of ISC_TIMESTAMP is not available from all the source
* modules that need to use isc_encode_timestamp
*
**************************************/
struct tm *times;
times = (struct tm *) times_arg;
date->timestamp_date = nday(times);
date->timestamp_time =
((times->tm_hour * 60 + times->tm_min) * 60 +
times->tm_sec) * ISC_TIME_SECONDS_PRECISION;
}
#ifdef NOT_USED_OR_REPLACED
ULONG API_ROUTINE gds__free(void* blk)
{
/**************************************
*
* g d s _ $ f r e e
*
**************************************
*
* Functional description
* Release a block allocated by gds__alloc. Return number
* of bytes released. Called under mutex protection.
*
**************************************/
ULONG released;
V4_MUTEX_LOCK(&alloc_mutex);
#ifdef DEBUG_GDS_ALLOC
gds_free_call_count++;
if (!blk) {
DEV_REPORT("gds__free: tried to release NULL\n");
BREAKPOINT(__LINE__);
}
else
{
ALLOC alloc, *p;
BOOLEAN found = FALSE;
ULONG save_length;
alloc = (ALLOC) (((char *) blk) - ALLOC_HEADER_SIZE);
for (p = &gds_alloc_chain; *p; p = &((*p)->alloc_next))
{
if (*p == alloc)
{
found++;
*p = (*p)->alloc_next;
break;
}
}
if (!found)
{
DEV_REPORT("gds__free: Tried to free unallocated block\n");
BREAKPOINT(__LINE__);
}
else {
/* Always validate the buffer we're about to free */
gds_alloc_validate(alloc);
/* If we want to validate all memory on each call, do so now */
if (gds_alloc_state < gds_alloc_call_count)
gds_alloc_report(ALLOC_silent, __FILE__, __LINE__);
/* Now zap the contents of the freed block - including the debug
* information as it is difficult to distinguish blocks freed
* by the suballocater vs blocks freed from a client
*/
strncpy(alloc->alloc_magic1, ALLOC_HEADER1_MAGIC_FREED,
sizeof(alloc->alloc_magic1));
save_length = alloc->alloc_status.alloc_length;
memset(alloc, ALLOC_FREED_PATTERN, save_length);
alloc->alloc_status.alloc_length = save_length;;
}
};
#endif
released = free_memory(blk);
blk = NULL;
V4_MUTEX_UNLOCK(&alloc_mutex);
return released;
}
#endif
#ifdef DEV_BUILD
void GDS_breakpoint(int parameter)
{
/**************************************
*
* G D S _ b r e a k p o i n t
*
**************************************
*
* Functional description
* Just a handy place to put a debugger breakpoint
* Calls to this can then be embedded for DEV_BUILD
* using the BREAKPOINT(x) macro.
*
**************************************/
/* Put a breakpoint here via the debugger */
}
#endif
SINT64 API_ROUTINE isc_portable_integer(UCHAR* ptr, SSHORT length)
{
/**************************************
*
* i s c _ p o r t a b l e _ i n t e g e r
*
**************************************
*
* Functional description
* Pick up (and convert) a Little Endian (VAX) style integer
* of length 1, 2, 4 or 8 bytes to local system's Endian format.
*
* various parameter blocks (i.e., dpb, tpb, spb) flatten out multibyte
* values into little endian byte ordering before network transmission.
* programs need to use this function in order to get the correct value
* of the integer in the correct Endian format.
*
* **Note**
*
* This function is similar to gds__vax_integer() in functionality.
* The difference is in the return type. Changed from SLONG to SINT64
* Since gds__vax_integer() is a public API routine, it could not be
* changed. This function has been made public so gbak can use it.
*
**************************************/
SINT64 value;
SSHORT shift;
assert(length <= 8);
value = shift = 0;
while (--length >= 0) {
value += ((SINT64) * ptr++) << shift;
shift += 8;
}
return value;
}
#ifdef DEBUG_GDS_ALLOC
static void validate_memory(void)
{
/**************************************
*
* v a l i d a t e _ m e m o r y
*
**************************************
*
* Functional description
* Utility function to call from the debugger, which sometimes
* doesn't like passing parameters.
*
**************************************/
gds_alloc_report(ALLOC_silent, "from debugger", 0);
}
#endif
#ifdef DEBUG_GDS_ALLOC
static void gds_alloc_validate(ALLOC p)
{
/**************************************
*
* g d s _ a l l o c _ v a l i d a t e
*
**************************************
*
* Functional description
* Test a memory buffer to see if it appears intact.
*
**************************************/
// JMB: Need to rework the code for the new pools
#ifdef NOT_USED_OR_REPLACED
USHORT errors = 0;
/* This might be a garbage pointer, so try and validate it first */
if (!p) {
errors++;
DEV_REPORT("gds_alloc_validate: pointer doesn't look right\n");
}
if (!errors &&
strncmp(p->alloc_magic1,
ALLOC_HEADER1_MAGIC_ALLOCATED,
sizeof(p->alloc_magic1)))
{
errors++;
DEV_REPORT("gds_alloc_validate: Header marker1 mismatch\n");
}
if (!errors &&
strncmp(p->alloc_magic2,
ALLOC_HEADER2_MAGIC,
sizeof(p->alloc_magic2)))
{
errors++;
DEV_REPORT("gds_alloc_validate: Header marker2 mismatch\n");
}
if (!errors &&
memcmp(((BLOB_PTR*) p) + ALLOC_HEADER_SIZE + p->alloc_requested_length,
ALLOC_TAILER_MAGIC,
ALLOC_TAILER_SIZE))
{
errors++;
DEV_REPORT("gds_alloc_validate: tailer marker mismatch\n");
}
if (!errors && p->alloc_status.alloc_length < p->alloc_requested_length)
{
errors++;
DEV_REPORT("gds_alloc_validate: Header has been trashed (smaller than req)\n");
}
if (!errors && p->alloc_status.alloc_length < ALLOC_OVERHEAD)
{
errors++;
DEV_REPORT
("gds_alloc_validate: Header has been trashed (smaller than min)\n");
}
if (errors)
{
char buffer[150];
sprintf(buffer,
"gds_alloc_validate: %lx len=%5ld file=%15.30s line=%4ld call=%ld alloc=%ld free=%ld%s",
p,
p->alloc_requested_length,
p->alloc_filename,
p->alloc_lineno,
p->alloc_callno,
gds_alloc_call_count,
gds_free_call_count,
NEWLINE);
DEV_REPORT(buffer);
BREAKPOINT(__LINE__);
}
#endif
}
#endif
#ifdef DEBUG_GDS_ALLOC
static BOOLEAN gds_alloc_validate_free_pattern(UCHAR* ptr,
ULONG len)
{
/**************************************
*
* g d s _ a l l o c _ v a l i d a t e _ f r e e _ p a t t e r n
*
**************************************
*
* Functional description
* Walk a chunk of memory to see if it is all the same
* "color", eg: freed.
*
* Return TRUE if the memory looks OK, FALSE if it has
* been clobbered.
*
**************************************/
// JMB: Need to rework the code for the new pool
#ifdef NOT_USED_OR_REPLACED
while (len--)
{
if (*ptr++ != ALLOC_FREED_PATTERN)
{
char buffer[100];
sprintf(buffer,
"gds_alloc_validate_free_pattern: Write to freed memory at %lx\n",
(ptr - 1));
DEV_REPORT(buffer);
BREAKPOINT(__LINE__);
return FALSE;
}
}
#endif
return TRUE;
}
#endif /* DEBUG_GDS_ALLOC */
#ifdef DEBUG_GDS_ALLOC
static void gds_alloc_validate_freed(ALLOC p)
{
/**************************************
*
* g d s _ a l l o c _ v a l i d a t e _ f r e e d
*
**************************************
*
* Functional description
* Test a memory buffer to see if it appears freed.
*
**************************************/
// JMB: need to rework this code for the new pool
#ifdef NOT_USED_OR_REPLACED
USHORT errors = 0;
/* This might be a garbage pointer, so try and validate it first */
if (!p)
{
errors++;
DEV_REPORT("gds_alloc_validate_freed: pointer doesn't look right\n");
}
/* Now look at the space, is it all the same pattern? */
if (!errors)
{
UCHAR *ptr;
ULONG len;
ptr = ((UCHAR *) p) + sizeof(struct free);
len = p->alloc_status.alloc_freed.free_length - sizeof(struct free);
if (!gds_alloc_validate_free_pattern(ptr, len))
{
errors++;
}
}
if (errors)
{
char buffer[150];
sprintf(buffer,
"gds_alloc_validate_freed: %lx len=%5ld alloc=%ld free=%ld%s",
p,
p->alloc_status.alloc_freed.free_length,
gds_alloc_call_count, gds_free_call_count, NEWLINE);
DEV_REPORT(buffer);
BREAKPOINT(__LINE__);
}
#endif
}
#endif
#ifdef DEBUG_GDS_ALLOC
void gds_alloc_watch(void* p)
{
/**************************************
*
* g d s _ a l l o c _ w a t c h
*
**************************************
*
* Functional description
* Start watching a memory location.
*
**************************************/
// JMB: need to rework this for new pools
#ifdef NOT_USED_OR_REPLACED
gds_alloc_watch_call_count++;
/* Do we have a new place to watch? If so, set our watcher */
if (p)
{
gds_alloc_watchpoint = (ALLOC) ((UCHAR *) p - ALLOC_HEADER_SIZE);
gds_alloc_watch_call_count = 1;
}
/* If we have a watchpoint, check that it's still valid */
if (gds_alloc_watchpoint)
{
gds_alloc_validate(gds_alloc_watchpoint);
}
#endif
}
#endif /* DEBUG_GDS_ALLOC */
#ifdef DEBUG_GDS_ALLOC
void API_ROUTINE gds_alloc_flag_unfreed(void *blk)
{
/**************************************
*
* g d s _ a l l o c _ f l a g _ u n f r e e d
*
**************************************
*
* Functional description
* Flag a buffer as "known to be unfreed" so we
* don't report it in gds_alloc_report
*
**************************************/
// JMB: need to rework this for the new pools
#ifdef NOT_USED_OR_REPLACED
if (!blk)
return;
/* Point to the start of the block */
ALLOC p = (ALLOC) (((UCHAR *) blk) - ALLOC_HEADER_SIZE);
/* Might as well validate it while we're here */
gds_alloc_validate(p);
/* Flag it as "already known" */
p->alloc_flags |= ALLOC_dont_report;
#endif
}
#endif /* DEBUG_GDS_ALLOC */
#ifdef DEBUG_GDS_ALLOC
void API_ROUTINE gds_alloc_report(ULONG flags, char* filename, int lineno)
{
/**************************************
*
* g d s _ a l l o c _ r e p o r t
*
**************************************
*
* Functional description
* Print buffers that might be memory leaks.
* Or that might have been clobbered.
*
**************************************/
// JMB: needs to be reworked for new pools
#ifdef NOT_USED_OR_REPLACED
ALLOC p;
IB_FILE *f = NULL;
char buffer[150];
TEXT name[MAXPATHLEN];
BOOLEAN stderr_reporting = TRUE;
if (flags & ALLOC_check_each_call)
{
gds_alloc_state = gds_alloc_call_count;
}
if (flags & ALLOC_dont_check)
{
gds_alloc_state = ~0;
}
for (p = gds_alloc_chain; p; p = p->alloc_next)
{
gds_alloc_validate(p);
if ((!(p->alloc_flags & ALLOC_dont_report) &&
!(flags & ALLOC_silent)) ||
(flags & ALLOC_verbose))
{
if (f == NULL)
{
#pragma FB_COMPILER_MESSAGE("TMN: Possible change needed")
gds__prefix(name, "iserver.err");
f = ib_fopen(name, "at");
ib_fflush(ib_stdout);
ib_fflush(ib_stderr);
sprintf(buffer,
"%sgds_alloc_report: flags %ld file %15.15s lineno %ld alloc=%ld free=%ld%s%s",
NEWLINE,
flags,
filename ? filename : "?",
lineno,
gds_alloc_call_count,
gds_free_call_count,
NEWLINE,
NEWLINE);
if (getenv("NO_GDS_ALLOC_REPORT"))
{
stderr_reporting = FALSE;
}
if (stderr_reporting)
{
DEV_REPORT(buffer);
}
if (f)
{
ib_fprintf(f, buffer);
}
}
sprintf(buffer,
"Unfreed alloc: %lx len=%5ld file=%15.30s line=%4ld call=%4ld%s",
p,
p->alloc_requested_length,
p->alloc_filename,
p->alloc_lineno, p->alloc_callno, NEWLINE);
if (stderr_reporting)
{
DEV_REPORT(buffer);
}
if (f)
{
ib_fprintf(f, buffer);
}
}
if (flags & ALLOC_mark_current) {
p->alloc_flags |= ALLOC_dont_report;
}
}
/* Walk the list of free blocks and validate they haven't been referenced */
for (p = (ALLOC)pool; p; p = (ALLOC)p->alloc_status.alloc_freed.free_next)
{
gds_alloc_validate_freed(p);
}
if (f)
{
ib_fclose(f);
}
#endif
}
#endif
SLONG API_ROUTINE gds__interprete(char *s, STATUS ** vector)
{
/**************************************
*
* g d s _ $ i n t e r p r e t e
*
**************************************
*
* Functional description
* Translate a status code with arguments to a string. Return the
* length of the string while updating the vector address. If the
* message is null (end of messages) or invalid, return 0;
*
**************************************/
TEXT *p, *q, *temp;
SSHORT l, temp_len;
TEXT **arg, *args[10];
STATUS code, *v;
UCHAR x;
STATUS decoded;
#ifdef VMS
STATUS status;
TEXT flags[4];
struct dsc$descriptor_s desc;
#endif
if (!**vector)
return 0;
temp = NULL;
/* handle a case: "no errors, some warning(s)" */
if ((*vector)[1] == 0 && (*vector)[2] == isc_arg_warning) {
v = *vector + 4;
code = (*vector)[3];
}
else {
v = *vector + 2;
code = (*vector)[1];
}
arg = args;
/* Parse and collect any arguments that may be present */
for (;;) {
x = (UCHAR) * v++;
switch (x) {
case gds_arg_string:
case gds_arg_number:
*arg++ = (TEXT *) * v++;
continue;
case gds_arg_cstring:
if (!temp) {
/* We need a temporary buffer when cstrings are involved.
Give up if we can't get one. */
p = temp = (TEXT*) gds__alloc((SLONG) BUFFER_SMALL);
temp_len = (SSHORT) BUFFER_SMALL;
/* FREE: at procedure exit */
if (!temp) /* NOMEM: */
return 0;
}
l = (SSHORT) * v++;
q = (TEXT *) * v++;
*arg++ = p;
/* ensure that we do not overflow the buffer allocated */
l = (temp_len < l) ? temp_len : l;
if (l)
do
*p++ = *q++;
while (--l);
*p++ = 0;
continue;
default:
break;
}
--v;
break;
}
/* Handle primary code on a system by system basis */
switch ((UCHAR) (*vector)[0]) {
case isc_arg_warning:
case gds_arg_gds:
{
USHORT fac = 0, class_ = 0;
decoded = gds__decode(code, &fac, &class_);
if (gds__msg_format(0, fac, (USHORT) decoded,
128, s, args[0], args[1], args[2], args[3],
args[4]) < 0) {
if ((decoded < FB_NELEM(messages) - 1) && (decoded >= 0))
sprintf(s, messages[decoded], args[0], args[1], args[2],
args[3], args[4]);
else
sprintf(s, "unknown ISC error %ld", code); /* TXNN */
}
}
break;
case gds_arg_interpreted:
p = s;
q = (TEXT *) (*vector)[1];
while ((*p++ = *q++) /*!= NULL*/);
break;
case gds_arg_unix:
if (code > 0 && code < sys_nerr && (p = (TEXT*)sys_errlist[code]))
strcpy(s, p);
else if (code == 60)
strcpy(s, "connection timed out");
else if (code == 61)
strcpy(s, "connection refused");
else
sprintf(s, "unknown unix error %ld", code); /* TXNN */
break;
case gds_arg_dos:
sprintf(s, "unknown dos error %ld", code); /* TXNN */
break;
#ifdef VMS
case gds_arg_vms:
l = 0;
desc.dsc$b_class = DSC$K_CLASS_S;
desc.dsc$b_dtype = DSC$K_DTYPE_T;
desc.dsc$w_length = 128;
desc.dsc$a_pointer = s;
status = sys$getmsg(code, &l, &desc, 15, flags);
if (status & 1)
s[l] = 0;
else
sprintf(s, "uninterpreted VMS code %x", code); /* TXNN */
break;
#endif
#ifdef WIN_NT
case gds_arg_win32:
if (!(l = (SSHORT) FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM,
NULL,
code,
GetUserDefaultLangID(),
s,
128,
NULL))
&& !(l = (SSHORT)FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM,
NULL,
code,
NULL, /* TMN: Fallback to system known language */
s,
128,
NULL))) sprintf(s, "unknown Win32 error %ld", code); /* TXNN */
break;
#endif
default:
if (temp)
gds__free((SLONG *) temp);
return 0;
}
if (temp)
gds__free((SLONG *) temp);
*vector = v;
p = s;
while (*p)
p++;
return p - s;
}
void API_ROUTINE gds__interprete_a(
SCHAR * s,
SSHORT * length,
STATUS * vector, SSHORT * offset)
{
/**************************************
*
* g d s _ $ i n t e r p r e t e _ a
*
**************************************
*
* Functional description
* Translate a status code with arguments to a string. Return the
* length of the string while updating the vector address. If the
* message is null (end of messages) or invalid, return 0;
* Ada being an unsatisfactory excuse for a language, fall back on
* the concept of indexing into the vector.
*
**************************************/
STATUS *v;
v = vector + *offset;
*length = (SSHORT) gds__interprete(s, &v);
*offset = v - vector;
}
void API_ROUTINE gds__log(TEXT * text, ...)
{
/**************************************
*
* g d s _ l o g
*
**************************************
*
* Functional description
* Post an event to a log file.
*
**************************************/
va_list ptr;
IB_FILE *file;
int oldmask;
time_t now;
TEXT name[MAXPATHLEN];
#ifdef HAVE_GETTIMEOFDAY
{
struct timeval tv;
#ifdef GETTIMEOFDAY_RETURNS_TIMEZONE
(void)gettimeofday(&tv, (struct timezone *)0);
#else
(void)gettimeofday(&tv);
#endif
now = tv.tv_sec;
}
#else
now = time((time_t *)0);
#endif
gds__prefix(name, LOGFILE);
oldmask = umask(0111);
if ((file = ib_fopen(name, FOPEN_APPEND_TYPE)) != NULL)
{
ib_fprintf(file, "%s%s\t%.25s\t", ISC_get_host(name, MAXPATHLEN),
gdslogid, ctime(&now));
VA_START(ptr, text);
ib_vfprintf(file, text, ptr);
ib_fprintf(file, "\n\n");
ib_fclose(file);
}
umask(oldmask);
}
void API_ROUTINE gds__log_status(TEXT * database, STATUS * status_vector)
{
/**************************************
*
* g d s _ $ l o g _ s t a t u s
*
**************************************
*
* Functional description
* Log error to error log.
*
**************************************/
TEXT *buffer, *p;
buffer = (TEXT *) gds__alloc((SLONG) BUFFER_XLARGE);
/* FREE: at procedure exit */
if (!buffer) /* NOMEM: */
return;
p = buffer;
sprintf(p, "Database: %s", (database) ? database : "");
do {
while (*p)
p++;
*p++ = '\n';
*p++ = '\t';
} while (gds__interprete(p, &status_vector));
p[-2] = 0;
gds__log(buffer, 0);
gds__free((SLONG *) buffer);
}
int API_ROUTINE gds__msg_close(void *handle)
{
/**************************************
*
* g d s _ $ m s g _ c l o s e
*
**************************************
*
* Functional description
* Close the message file and zero out static variable.
*
**************************************/
GDS_MSG message = reinterpret_cast < GDS_MSG > (handle);
if (!message) {
if (!default_msg) {
return 0;
}
message = default_msg;
}
default_msg = (GDS_MSG) NULL;
int fd = message->msg_file;
FREE_LIB_MEMORY(message);
if (fd <= 0)
return 0;
return close(fd);
}
SSHORT API_ROUTINE gds__msg_format(void* handle,
USHORT facility,
USHORT number,
USHORT length,
TEXT* buffer,
const TEXT* arg1,
const TEXT* arg2,
const TEXT* arg3,
const TEXT* arg4,
const TEXT* arg5)
{
/**************************************
*
* g d s _ $ m s g _ f o r m a t
*
**************************************
*
* Functional description
* Lookup and format message. Return as much of formatted string
* as fits in caller's buffer.
*
**************************************/
TEXT *p, *formatted, *end;
SLONG size;
int n;
size = (SLONG) (((arg1) ? MAX_ERRSTR_LEN : 0) +
((arg2) ? MAX_ERRSTR_LEN : 0) +
((arg3) ? MAX_ERRSTR_LEN : 0) +
((arg4) ? MAX_ERRSTR_LEN : 0) +
((arg5) ? MAX_ERRSTR_LEN : 0) + MAX_ERRMSG_LEN);
size = (size < length) ? length : size;
formatted = (TEXT *) gds__alloc((SLONG) size);
if (!formatted) /* NOMEM: */
return -1;
/* Let's assume that the text to be output will never be shorter
than the raw text of the message to be formatted. Then we can
use the caller's buffer to temporarily hold the raw text. */
n = gds__msg_lookup(handle, facility, number, length, buffer, NULL);
if (n > 0 && n < length)
{
sprintf(formatted, buffer, arg1, arg2, arg3, arg4, arg5);
}
else
{
sprintf(formatted, "can't format message %d:%d -- ", facility,
number);
if (n == -1)
strcat(formatted, "message text not found");
else if (n == -2) {
strcat(formatted, "message file ");
for (p = formatted; *p;)
p++;
gds__prefix_msg(p, MSG_FILE);
strcat(p, " not found");
}
else {
for (p = formatted; *p;)
p++;
sprintf(p, "message system code %d", n);
}
}
USHORT l = strlen(formatted);
end = buffer + length - 1;
for (p = formatted; *p && buffer < end;) {
*buffer++ = *p++;
}
*buffer = 0;
gds__free((SLONG *) formatted);
return ((n > 0) ? l : -l);
}
SSHORT API_ROUTINE gds__msg_lookup(void *handle,
USHORT facility,
USHORT number,
USHORT length,
TEXT * buffer, USHORT * flags)
{
/**************************************
*
* g d s _ $ m s g _ l o o k u p
*
**************************************
*
* Functional description
* Lookup a message. Return as much of the record as fits in the
* callers buffer. Return the FULL size of the message, or negative
* number if we can't find the message.
*
**************************************/
GDS_MSG message;
int status;
USHORT n;
MSGREC leaf;
MSGNOD node, end;
ULONG position, code;
TEXT *p;
/* Handle default message file */
if (!(message = (GDS_MSG) handle) && !(message = default_msg)) {
/* Try environment variable setting first */
if ((p = getenv("ISC_MSGS")) == NULL ||
(status =
gds__msg_open(reinterpret_cast < void **>(&message), p))) {
TEXT translated_msg_file[sizeof(MSG_FILE_LANG) + LOCALE_MAX + 1];
TEXT *msg_file;
/* Try declared language of this attachment */
/* This is not quite the same as the language declared on the
READY statement */
msg_file = (TEXT *) gds__alloc((SLONG) MAXPATHLEN);
/* FREE: at block exit */
if (!msg_file) /* NOMEM: */
return -2;
p = getenv("LC_MESSAGES");
if (p != NULL) {
sanitize(p);
sprintf(translated_msg_file, MSG_FILE_LANG, p);
gds__prefix_msg(msg_file, translated_msg_file);
status =
gds__msg_open(reinterpret_cast < void **>(&message),
msg_file);
}
else
status = 1;
if (status) {
/* Default to standard message file */
gds__prefix_msg(msg_file, MSG_FILE);
status =
gds__msg_open(reinterpret_cast < void **>(&message),
msg_file);
}
gds__free((SLONG *) msg_file);
}
if (status)
return status;
default_msg = message;
}
/* Search down index levels to the leaf. If we get lost, punt */
code = MSG_NUMBER(facility, number);
end = (MSGNOD) ((SCHAR *) message->msg_bucket + message->msg_bucket_size);
position = message->msg_top_tree;
for (n = 1, status = 0; !status; n++) {
if (lseek(message->msg_file, LSEEK_OFFSET_CAST position, 0) < 0)
status = -6;
else if (read(message->msg_file, message->msg_bucket,
message->msg_bucket_size) < 0)
status = -7;
else if (n == message->msg_levels)
break;
else {
for (node = (MSGNOD) message->msg_bucket; !status; node++) {
if (node >= end) {
status = -8;
break;
}
else if (node->msgnod_code >= code) {
position = node->msgnod_seek;
break;
}
}
}
}
if (!status) {
/* Search the leaf */
for (leaf = (MSGREC) message->msg_bucket; !status;
leaf = NEXT_LEAF(leaf)) {
if (leaf >= (MSGREC) end || leaf->msgrec_code > code) {
status = -1;
break;
}
else if (leaf->msgrec_code == code) {
/* We found the correct message, so return it to the user */
n = MIN(length - 1, leaf->msgrec_length);
memcpy(buffer, leaf->msgrec_text, n);
buffer[n] = 0;
if (flags)
*flags = leaf->msgrec_flags;
status = leaf->msgrec_length;
break;
}
}
}
return status;
}
int API_ROUTINE gds__msg_open(void **handle, TEXT * filename)
{
/**************************************
*
* g d s _ $ m s g _ o p e n
*
**************************************
*
* Functional description
* Try to open a message file. If successful, return a status of 0
* and update a message handle.
*
**************************************/
int n;
GDS_MSG message;
ISC_MSGHDR header;
if ((n = open(filename, O_RDONLY | O_BINARY, 0)) < 0)
return -2;
if (read(n, &header, sizeof(header)) < 0) {
(void) close(n);
return -3;
}
if (header.msghdr_major_version != MSG_MAJOR_VERSION ||
(SSHORT) header.msghdr_minor_version < MSG_MINOR_VERSION) {
(void) close(n);
return -4;
}
message =
(GDS_MSG) ALLOC_LIB_MEMORY((SLONG) sizeof(gds_msg) +
header.msghdr_bucket_size - 1);
/* FREE: in gds__msg_close */
if (!message) { /* NOMEM: return non-open error */
(void) close(n);
return -5;
}
#ifdef DEBUG_GDS_ALLOC
/* This will only be freed if the client closes the message file for us */
gds_alloc_flag_unfreed((void *) message);
#endif
message->msg_file = n;
message->msg_bucket_size = header.msghdr_bucket_size;
message->msg_levels = header.msghdr_levels;
message->msg_top_tree = header.msghdr_top_tree;
*handle = message;
return 0;
}
void API_ROUTINE gds__msg_put(
void *handle,
USHORT facility,
USHORT number,
TEXT * arg1,
TEXT * arg2, TEXT * arg3, TEXT * arg4, TEXT * arg5)
{
/**************************************
*
* g d s _ $ m s g _ p u t
*
**************************************
*
* Functional description
* Lookup and format message. Return as much of formatted string
* as fits in callers buffer.
*
**************************************/
TEXT formatted[512];
gds__msg_format(handle, facility, number, sizeof(TEXT) * BUFFER_MEDIUM,
formatted, arg1, arg2, arg3, arg4, arg5);
gds__put_error(formatted);
}
SLONG API_ROUTINE gds__get_prefix(SSHORT arg_type, TEXT * passed_string)
{
/**************************************
*
* g d s _ $ g e t _ p r e f i x
*
**************************************
*
* Functional description
* Find appropriate InterBase command line arguments
* for Interbase file prefix.
*
* arg_type is 0 for $INTERBASE, 1 for $INTERBASE_LOCK
* and 2 for $INTERBASE_MSG
*
* Function returns 0 on success and -1 on failure
**************************************/
char *prefix_ptr;
int count = 0;
#ifdef NOT_USED_OR_REPLACED
char *prefix_array; /* = {ib_prefix, ib_prefix_lock, ib_prefix_msg}; */
#endif
if (arg_type < IB_PREFIX_TYPE || arg_type > IB_PREFIX_MSG_TYPE)
return ((SLONG) - 1);
switch (arg_type) {
case IB_PREFIX_TYPE:
prefix_ptr = ib_prefix = ib_prefix_val;
break;
case IB_PREFIX_LOCK_TYPE:
prefix_ptr = ib_prefix_lock = ib_prefix_lock_val;
break;
case IB_PREFIX_MSG_TYPE:
prefix_ptr = ib_prefix_msg = ib_prefix_msg_val;
break;
default:
return ((SLONG) - 1);
}
/* the command line argument was 'H' for interbase home */
while (*prefix_ptr++ = *passed_string++) {
/* if the next character is space, newline or carriage return OR
number of characters exceeded */
if (*passed_string == ' ' || *passed_string == 10
|| *passed_string == 13 || (count == MAXPATHLEN)) {
*(prefix_ptr++) = '\0';
break;
}
count++;
}
if (!count) {
prefix_ptr = NULL;
return ((SLONG) - 1);
}
return ((SLONG) 0);
}
#ifndef VMS
void API_ROUTINE gds__prefix(TEXT *resultString, TEXT *file)
{
/**************************************
*
* g d s _ $ p r e f i x ( n o n - V M S )
*
**************************************
*
* Functional description
* Find appropriate file prefix.
* Override conditional defines with
* the enviroment variable FIREBIRD if it is set.
*
**************************************/
resultString[0] = 0;
if (ib_prefix == NULL) {
if (!(ib_prefix = getenv(FB_ENV)) || ib_prefix[0] == 0) {
{
// Try and get value from config file
const char *regPrefix = Config::getRootDirectory();
size_t len = strlen(regPrefix);
if (len > 0) {
if (len > sizeof(ib_prefix_val)) {
ib_perror("ib_prefix path size too large - truncated");
}
strncpy(ib_prefix_val, regPrefix, sizeof(ib_prefix_val) -1);
ib_prefix_val[sizeof(ib_prefix_val) -1] = 0;
ib_prefix = ib_prefix_val;
}
else {
ib_prefix = FB_PREFIX;
strcat(ib_prefix_val, ib_prefix);
}
}
ib_prefix = ib_prefix_val;
}
}
strcat(resultString, ib_prefix);
int len = strlen(resultString);
if (resultString[len - 1] != PathUtils::dir_sep) {
resultString[len] = PathUtils::dir_sep;
resultString[len + 1] = 0;
}
strcat(resultString, file);
}
#endif /* !defined(VMS) */
#ifdef VMS
void API_ROUTINE gds__prefix(TEXT * string, TEXT * root)
{
/**************************************
*
* g d s _ $ p r e f i x ( V M S )
*
**************************************
*
* Functional description
* Find appropriate InterBase file prefix.
* Override conditional defines with
* the enviroment variable INTERBASE if it is set.
*
**************************************/
TEXT temp[256], *p, *q;
ISC_VMS_PREFIX prefix;
SSHORT len;
if (*root != '[') {
strcpy(string, root);
return;
}
/* Check for the existence of an InterBase logical name. If there is
one use it, otherwise use the system directories. */
if (ISC_expand_logical_once(ISC_LOGICAL, sizeof(ISC_LOGICAL) - 2, temp)) {
strcpy(string, ISC_LOGICAL);
strcat(string, root);
return;
}
for (p = temp, q = root; *p = UPPER7(*q); q++)
if (*p++ == ']')
break;
len = p - temp;
for (prefix = trans_prefix; prefix->isc_prefix; prefix++)
if (!strncmp(temp, prefix->isc_prefix, len)) {
strcpy(string, prefix->vms_prefix);
strcat(string, &root[len]);
return;
}
strcpy(string, &root[len]);
}
#endif
#ifndef VMS
void API_ROUTINE gds__prefix_lock(TEXT * string, TEXT * root)
{
/********************************************************
*
* g d s _ $ p r e f i x _ l o c k ( n o n - V M S )
*
********************************************************
*
* Functional description
* Find appropriate Firebird lock file prefix.
* Override conditional defines with the enviroment
* variable FIREBIRD_LOCK if it is set.
*
**************************************/
string[0] = 0;
if (ib_prefix_lock == NULL) {
if (!(ib_prefix_lock = getenv(FB_LOCK_ENV))) {
ib_prefix_lock = ib_prefix_lock_val;
gds__prefix(ib_prefix_lock, "");
}
else {
strcat(ib_prefix_lock_val, ib_prefix_lock);
ib_prefix_lock = ib_prefix_lock_val;
}
}
strcat(string, ib_prefix_lock);
int len = strlen(string);
if (string[len - 1] != PathUtils::dir_sep) {
string[len] = PathUtils::dir_sep;
string[len + 1] = 0;
}
strcat(string, root);
}
#endif
#ifdef VMS
void API_ROUTINE gds__prefix_lock(TEXT * string, TEXT * root)
{
/************************************************
*
* g d s _ $ p r e f i x _ l o c k ( V M S )
*
************************************************
*
* Functional description
* Find appropriate Firebird lock file prefix.
* Override conditional defines with the enviroment
* variable FIREBIRD_LOCK if it is set.
*
*************************************************/
TEXT temp[256], *p, *q;
ISC_VMS_PREFIX prefix;
SSHORT len;
if (*root != '[') {
strcpy(string, root);
return;
}
/* Check for the existence of an InterBase logical name. If there is
one use it, otherwise use the system directories. */
if (ISC_expand_logical_once
(ISC_LOGICAL_LOCK, sizeof(ISC_LOGICAL_LOCK) - 2, temp)) {
strcpy(string, ISC_LOGICAL_LOCK);
strcat(string, root);
return;
}
for (p = temp, q = root; *p = UPPER7(*q); q++)
if (*p++ == ']')
break;
len = p - temp;
for (prefix = trans_prefix; prefix->isc_prefix; prefix++)
if (!strncmp(temp, prefix->isc_prefix, len)) {
strcpy(string, prefix->vms_prefix);
strcat(string, &root[len]);
return;
}
strcpy(string, &root[len]);
}
#endif
#ifndef VMS
void API_ROUTINE gds__prefix_msg(TEXT * string, TEXT * root)
{
/********************************************************
*
* g d s _ $ p r e f i x _ m s g ( n o n - V M S )
*
********************************************************
*
* Functional description
* Find appropriate Firebird message file prefix.
* Override conditional defines with the enviroment
* variable FIREBIRD_MSG if it is set.
*
*
**************************************/
string[0] = 0;
if (ib_prefix_msg == NULL) {
if (!(ib_prefix_msg = getenv(FB_MSG_ENV))) {
ib_prefix_msg = ib_prefix_msg_val;
gds__prefix(ib_prefix_msg, "");
}
else {
strcat(ib_prefix_msg_val, ib_prefix_msg);
ib_prefix_msg = ib_prefix_msg_val;
}
}
strcat(string, ib_prefix_msg);
int len = strlen(string);
if (string[len - 1] != PathUtils::dir_sep) {
string[len] = PathUtils::dir_sep;
string[len + 1] = 0;
}
strcat(string, root);
}
#endif
#ifdef VMS
void API_ROUTINE gds__prefix_msg(TEXT * string, TEXT * root)
{
/************************************************
*
* g d s _ $ p r e f i x _ m s g ( V M S )
*
************************************************
*
* Functional description
* Find appropriate Firebird message file prefix.
* Override conditional defines with the enviroment
* variable FIREBIRD_MSG if it is set.
*
*************************************************/
TEXT temp[256], *p, *q;
ISC_VMS_PREFIX prefix;
SSHORT len;
if (*root != '[') {
strcpy(string, root);
return;
}
/* Check for the existence of an InterBase logical name. If there is
one use it, otherwise use the system directories. */
/* ISC_LOGICAL_MSG macro needs to be defined, check non VMS version of routine
for functionality. */
if (ISC_expand_logical_once
(ISC_LOGICAL_MSG, sizeof(ISC_LOGICAL_MSG) - 2, te mp)) {
strcpy(string, ISC_LOGICAL_MSG);
strcat(string, root);
return;
}
for (p = temp, q = root; *p = UPPER7(*q); q++)
if (*p++ == ']')
break;
len = p - temp;
for (prefix = trans_prefix; prefix->isc_prefix; prefix++)
if (!strncmp(temp, prefix->isc_prefix, len)) {
strcpy(string, prefix->vms_prefix);
strcat(string, &root[len]);
return;
}
strcpy(string, &root[len]);
}
#endif
STATUS API_ROUTINE gds__print_status(STATUS * vec)
{
/**************************************
*
* g d s _ p r i n t _ s t a t u s
*
**************************************
*
* Functional description
* Interprete a status vector.
*
**************************************/
STATUS *vector;
TEXT *s;
if (!vec || (!vec[1] && vec[2] == gds_arg_end))
return FB_SUCCESS;
s = (TEXT *) gds__alloc((SLONG) BUFFER_LARGE);
/* FREE: at procedure return */
if (!s) /* NOMEM: */
return vec[1];
vector = vec;
if (!gds__interprete(s, &vector)) {
gds__free((SLONG *) s);
return vec[1];
}
gds__put_error(s);
s[0] = '-';
while (gds__interprete(s + 1, &vector))
gds__put_error(s);
gds__free((SLONG *) s);
return vec[1];
}
USHORT API_ROUTINE gds__parse_bpb(USHORT bpb_length,
UCHAR * bpb,
USHORT * source, USHORT * target)
{
/**************************************
*
* g d s _ p a r s e _ b p b
*
**************************************
*
* Functional description
* Get blob type, source sub_type and target sub-type
* from a blob parameter block.
*
**************************************/
/* SIGN ERROR */
return gds__parse_bpb2(bpb_length, bpb, (SSHORT*)source, (SSHORT*)target, NULL, NULL);
}
USHORT API_ROUTINE gds__parse_bpb2(USHORT bpb_length,
UCHAR * bpb,
SSHORT * source,
SSHORT * target,
USHORT * source_interp,
USHORT * target_interp)
{
/**************************************
*
* g d s _ p a r s e _ b p b 2
*
**************************************
*
* Functional description
* Get blob type, source sub_type and target sub-type
* from a blob parameter block.
* Basically gds__parse_bpb() with additional:
* source_interp and target_interp.
*
**************************************/
UCHAR *p, *end, op;
USHORT type, length;
type = *source = *target = 0;
if (source_interp)
*source_interp = 0;
if (target_interp)
*target_interp = 0;
if (!bpb_length)
return type;
p = bpb;
end = p + bpb_length;
if (*p++ != gds_bpb_version1)
return type;
while (p < end) {
op = *p++;
length = *p++;
switch (op) {
case gds_bpb_source_type:
*source = (USHORT) gds__vax_integer(p, length);
break;
case gds_bpb_target_type:
*target = (USHORT) gds__vax_integer(p, length);
break;
case gds_bpb_type:
type = (USHORT) gds__vax_integer(p, length);
break;
case gds_bpb_source_interp:
if (source_interp)
*source_interp = (USHORT) gds__vax_integer(p, length);
break;
case gds_bpb_target_interp:
if (target_interp)
*target_interp = (USHORT) gds__vax_integer(p, length);
break;
default:
break;
}
p += length;
}
return type;
}
SLONG API_ROUTINE gds__ftof(SCHAR * string,
USHORT GDS_VAL(length1),
SCHAR * field,
USHORT GDS_VAL(length2))
{
/**************************************
*
* g d s _ f t o f
*
**************************************
*
* Functional description
* Move a fixed length string to a fixed length string.
* This is typically generated by the preprocessor to
* move strings around.
*
**************************************/
USHORT l, fill;
fill = GDS_VAL(length2) - GDS_VAL(length1);
if ((l = MIN(GDS_VAL(length1), GDS_VAL(length2))) > 0)
do
*field++ = *string++;
while (--l);
if (fill > 0)
do
*field++ = ' ';
while (--fill);
return 0;
}
int API_ROUTINE gds__print_blr(
UCHAR * blr,
FPTR_VOID routine,
SCHAR * user_arg, SSHORT language)
{
/**************************************
*
* g d s _ $ p r i n t _ b l r
*
**************************************
*
* Functional description
* Pretty print blr thru callback routine.
*
**************************************/
struct ctl ctl, *control;
SCHAR eoc;
SLONG offset;
SSHORT version, level;
try {
control = &ctl;
level = 0;
offset = 0;
if (!routine) {
routine = (void (*)()) PRINTF;
user_arg = "%4d %s\n";
}
control->ctl_routine = routine;
control->ctl_user_arg = user_arg;
control->ctl_blr = control->ctl_blr_start = blr;
control->ctl_ptr = control->ctl_buffer;
control->ctl_language = language;
version = BLR_BYTE;
if ((version != blr_version4) && (version != blr_version5))
blr_error(control, "*** blr version %d is not supported ***",
(int) version);
blr_format(control,
(version == blr_version4) ? "blr_version4," : "blr_version5,");
PRINT_LINE;
PRINT_VERB;
offset = control->ctl_blr - control->ctl_blr_start;
eoc = BLR_BYTE;
if (eoc != blr_eoc)
blr_error(control, "*** expected end of command, encounted %d ***",
(int) eoc);
blr_format(control, "blr_eoc");
PRINT_LINE;
} // try
catch (const std::exception&) {
return -1;
}
return 0;
}
void API_ROUTINE gds__put_error(TEXT * string)
{
/**************************************
*
* g d s _ p u t _ e r r o r
*
**************************************
*
* Functional description
* Put out a line of error text.
*
**************************************/
#ifdef VMS
#define PUT_ERROR
struct dsc$descriptor_s desc;
ISC_make_desc(string, &desc, 0);
lib$put_output(&desc);
#endif
#ifdef PUT_ERROR
#undef PUT_ERROR
#else
ib_fputs(string, ib_stderr);
ib_fputc('\n', ib_stderr);
ib_fflush(ib_stderr);
#endif
}
void API_ROUTINE gds__qtoq(void *quad_in, void *quad_out)
{
/**************************************
*
* g d s _ q t o q
*
**************************************
*
* Functional description
* Move a quad value to another quad value. This
* call is generated by the preprocessor when assigning
* quad values in FORTRAN.
*
**************************************/
*((GDS_QUAD *) quad_out) = *((GDS_QUAD *) quad_in);
}
void API_ROUTINE gds__register_cleanup(FPTR_VOID_PTR routine, void *arg)
{
/**************************************
*
* g d s _ r e g i s t e r _ c l e a n u p
*
**************************************
*
* Functional description
* Register a cleanup handler.
*
**************************************/
/*
* Ifdef out for windows client. We have not implemented any way of
* determining when a task ends, therefore this never gets called.
*/
CLEAN clean;
if (!initialized)
init();
clean = (CLEAN) ALLOC_LIB_MEMORY((SLONG) sizeof(struct clean));
clean->clean_next = cleanup_handlers;
cleanup_handlers = clean;
clean->clean_routine = routine;
clean->clean_arg = arg;
#ifdef DEBUG_GDS_ALLOC
/* Startup function - known to be unfreed */
gds_alloc_flag_unfreed((void *) clean);
#endif
}
SLONG API_ROUTINE gds__sqlcode(STATUS * status_vector)
{
/**************************************
*
* g d s _ s q l c o d e
*
**************************************
*
* Functional description
* Translate GDS error code to SQL error code. This is a little
* imprecise (to say the least) because we don't know the proper
* SQL codes. One must do what what can; stiff upper lip, and all
* that.
*
* Scan a status vector - if we find gds__sqlerr in the "right"
* positions, return the code that follows. Otherwise, for the
* first code for which there is a non-generic SQLCODE, return it.
*
**************************************/
USHORT code;
SLONG sqlcode;
STATUS *s;
USHORT have_sqlcode;
if (!status_vector) {
DEV_REPORT("gds__sqlcode: NULL status vector");
return GENERIC_SQLCODE;
}
have_sqlcode = FALSE;
sqlcode = GENERIC_SQLCODE; /* error of last resort */
/* SQL code -999 (GENERIC_SQLCODE) is generic, meaning "no other sql code
* known". Now scan the status vector, seeing if there is ANY sqlcode
* reported. Make note of the first error in the status vector who's
* SQLCODE is NOT -999, that will be the return code if there is no specific
* sqlerr reported.
*/
s = status_vector;
while (*s != gds_arg_end)
{
if (*s == gds_arg_gds)
{
s++;
if (*s == gds_sqlerr)
{
return *(s + 2);
}
if (!have_sqlcode) {
/* Now check the hard-coded mapping table of gds_codes to
sql_codes */
USHORT fac = 0, class_ = 0;
code = (USHORT) gds__decode(status_vector[1], &fac, &class_);
if ((code < FB_NELEM(gds__sql_code)) &&
(gds__sql_code[code] != GENERIC_SQLCODE))
{
sqlcode = gds__sql_code[code];
have_sqlcode = TRUE;
}
}
s++;
}
else if (*s == gds_arg_cstring)
s += 3; /* skip: gds_arg_cstring <len> <ptr> */
else
s += 2; /* skip: gds_arg_* <item> */
};
return sqlcode;
}
void API_ROUTINE gds__sqlcode_s(STATUS * status_vector, ULONG * sqlcode)
{
/**************************************
*
* g d s _ s q l c o d e _ s
*
**************************************
*
* Functional description
* Translate GDS error code to SQL error code. This is a little
* imprecise (to say the least) because we don't know the proper
* SQL codes. One must do what what can; stiff upper lip, and all
* that. THIS IS THE COBOL VERSION. (Some cobols son't have
* return values for calls...
*
**************************************/
*sqlcode = gds__sqlcode(status_vector);
}
void* API_ROUTINE gds__temp_file( BOOLEAN flag,
TEXT* string,
TEXT* expanded_string)
{
/**************************************
*
* g d s _ $ t e m p _ f i l e
*
**************************************
*
* Functional description
* Create and open a temp file with a given root. Unless the
* address of a buffer for the expanded file name string is
* given, make up the file "pre-deleted". Return -1 on failure.
*
**************************************/
return gds__tmp_file2(flag, string, expanded_string, NULL);
}
void *gds__tmp_file2(
BOOLEAN flag,
TEXT * string, TEXT * expanded_string, TEXT * dir)
{
/**************************************
*
* g d s _ _ t m p _ f i l e 2
*
**************************************
*
* Functional description
* Create and open a temp file with a given location.
* Unless the address of a buffer for the expanded file name string is
* given, make up the file "pre-deleted". Return -1 on failure.
*
**************************************/
IB_FILE *file;
TEXT file_name[256], *p, *q, *end;
SSHORT i;
p = file_name;
end = p + sizeof(file_name) - 8;
#ifdef VMS
q = WORKFILE;
#else
if (!(q = dir) && !(q = getenv("INTERBASE_TMP")) && !(q = getenv("TMP")))
{
#ifdef WIN_NT
TEXT temp_dir[256];
if ((i = (SSHORT) GetTempPath(sizeof(temp_dir), temp_dir)) &&
i < sizeof(temp_dir))
q = temp_dir;
else
#endif
q = WORKFILE;
}
#endif /* VMS */
for (; p < end && *q;)
*p++ = *q++;
#ifndef VMS
#if (defined WIN_NT)
if (p > file_name && p[-1] != '\\' && p[-1] != '/' && p < end)
*p++ = '\\';
#else
if (p > file_name && p[-1] != '/' && p < end)
*p++ = '/';
#endif
#endif
for (q = string; p < end && *q;)
*p++ = *q++;
for (q = TEMP_PATTERN; *q;)
*p++ = *q++;
*p = 0;
#if !(defined FREEBSD || defined NETBSD)
MKTEMP(file_name, string);
if (expanded_string)
strcpy(expanded_string, file_name);
#ifdef VMS
if (flag) {
file = (expanded_string) ?
ib_fopen(file_name, "w") : ib_fopen(file_name, "w", "fop=tmd");
if (!file)
return (void *) -1;
}
else {
file = (expanded_string) ?
open(file_name, O_RDWR | O_CREAT | O_TRUNC, 0600) :
open(file_name, O_RDWR | O_CREAT | O_TRUNC, 0600, "fop=tmd");
if (file == (IB_FILE *) - 1)
return (void *) -1;
}
#else
if (flag) {
for (i = 0; i < IO_RETRY; i++) {
if (file = ib_fopen(file_name, "w+"))
break;
if (!SYSCALL_INTERRUPTED(errno))
return (void *) -1;
}
if (!file)
return (void *) -1;
}
else {
for (i = 0; i < IO_RETRY; i++) {
file =
(IB_FILE *) (int) open(file_name,
O_RDWR | O_CREAT | O_TRUNC | O_BINARY,
0600);
if (file != (IB_FILE *) - 1)
break;
if (!SYSCALL_INTERRUPTED(errno))
return (void *) -1;
}
if (file == (IB_FILE *) - 1)
return (void *) -1;
}
if (!expanded_string)
unlink(file_name);
#endif
#else /* defined FREEBSD/NETBSD */
file = (IB_FILE *) mkstemp(file_name);
if (file == (IB_FILE *) - 1)
return (void *) -1;
if (flag)
if (!(file = fdopen((int) file, "w+")))
return (void *) -1;
if (expanded_string)
strcpy(expanded_string, file_name);
if (!expanded_string) {
unlink(file_name);
}
#endif /* FREEBSD/NETBSD */
return (void *) file;
}
void API_ROUTINE gds__unregister_cleanup(FPTR_VOID_PTR routine, void *arg)
{
/**************************************
*
* g d s _ u n r e g i s t e r _ c l e a n u p
*
**************************************
*
* Functional description
* Una cleanup handler.
*
**************************************/
CLEAN *clean_ptr, clean;
for (clean_ptr = &cleanup_handlers; clean = *clean_ptr;
clean_ptr = &clean->clean_next) {
if (clean->clean_routine == routine
&& clean->clean_arg == arg) {
*clean_ptr = clean->clean_next;
FREE_LIB_MEMORY(clean);
break;
}
}
}
#ifndef VMS
BOOLEAN API_ROUTINE gds__validate_lib_path(TEXT * module,
TEXT * ib_env_var,
TEXT * resolved_module,
SLONG length)
{
/**************************************
*
* g d s _ $ v a l i d a t e _ l i b _ p a t h ( n o n - V M S )
*
**************************************
*
* Functional description
* Find the external library path variable.
* Validate that the path to the library module name
* in the path specified. If the external lib path
* is not defined then accept any path, and return
* TRUE. If the module is in the path then return TRUE
* else, if the module is not in the path return FALSE.
*
**************************************/
TEXT *p, *q;
TEXT *token;
TEXT abs_module[MAXPATHLEN];
TEXT abs_module_path[MAXPATHLEN];
TEXT abs_path[MAXPATHLEN];
TEXT path[MAXPATHLEN];
TEXT *ib_ext_lib_path = 0;
if (!(ib_ext_lib_path = getenv(ib_env_var))) {
strncpy(resolved_module, module, length);
return TRUE; /* The variable is not defined. Retrun TRUE */
}
if (EXPAND_PATH(module, abs_module)) {
/* Extract the path from the absolute module name */
for (p = abs_module, q = NULL; *p; p++)
if ((*p == '\\') || (*p == '/'))
q = p;
memset(abs_module_path, 0, MAXPATHLEN);
strncpy(abs_module_path, abs_module, q - abs_module);
/* Check to see if the module path is in the lib path
if it is return TURE. If it does not find it, then
the module path is not valid so return FALSE */
token = strtok(ib_ext_lib_path, ";");
while (token != NULL) {
strcpy(path, token);
/* make sure that there is no traing slash on the path */
p = path + strlen(path);
if ((p != path) && ((p[-1] == '/') || (p[-1] == '\\')))
p[-1] = 0;
if ((EXPAND_PATH(path, abs_path))
&& (!COMPARE_PATH(abs_path, abs_module_path))) {
strncpy(resolved_module, abs_module, length);
return TRUE;
}
token = strtok(NULL, ";");
}
}
return FALSE;
}
#endif
#ifdef VMS
BOOLEAN API_ROUTINE gds__validate_lib_path(TEXT * module,
TEXT * ib_env_var,
TEXT * resolved_module,
SLONG length)
{
/**************************************
*
* g d s _ $ v a l i d a t e _ l i b _ p a t h ( V M S )
*
**************************************
*
* Functional description
* Find the InterBase external library path variable.
* Validate that the path to the library module name
* in the path specified. If the external lib path
* is not defined then accept any path, and return true.
* If the module is not in the path return FALSE.
*
**************************************/
TEXT *p, *q;
TEXT path[MAXPATHLEN];
TEXT abs_module_path[MAXPATHLEN];
TEXT abs_module[MAXPATHLEN];
/* Check for the existence of an InterBase logical name. If there is
one use it, otherwise use the system directories. */
COMPILER ERROR ! BEFORE DOING A VMS POST PLEASE
MAKE SURE THAT THIS FUNCTION WORKS THE SAME WAY
AS THE NON -
VMS ONE
DOES.if (!
(ISC_expand_logical_once
(ib_env_var, strlen(ib_env_var) - 2, path))) return TRUE;
if (ISC_expand_logical_once(module, strlen(module) - 2, abs_module)) {
/* Extract the path from the module name */
for (p = abs_module, q = NULL; *p; p++)
if (*p == ']')
q = p;
memset(abs_module_path, 0, MAXPATHLEN);
strncpy(abs_module_path, abs_module, q - abs_module + 1);
/* Check to see if the module path is the same as lib path
if it is return TURE. If not then the module path is
not valid so return FALSE */
if (!strcmp(path, abs_module_path))
return TRUE;
}
return FALSE;
}
#endif
SLONG API_ROUTINE gds__vax_integer(const UCHAR * ptr, SSHORT length)
{
/**************************************
*
* g d s _ $ v a x _ i n t e g e r
*
**************************************
*
* Functional description
* Pick up (and convert) a VAX style integer of length 1, 2, or 4
* bytes.
*
**************************************/
SLONG value;
SSHORT shift;
value = shift = 0;
while (--length >= 0) {
value += ((SLONG) * ptr++) << shift;
shift += 8;
}
return value;
}
void API_ROUTINE gds__vtof(
SCHAR * string,
SCHAR * field, USHORT length)
{
/**************************************
*
* g d s _ v t o f
*
**************************************
*
* Functional description
* Move a null terminated string to a fixed length
* field. The call is primarily generated by the
* preprocessor.
*
**************************************/
while (*string) {
*field++ = *string++;
if (--length <= 0)
return;
}
if (length > 0)
do
*field++ = ' ';
while (--length);
}
void API_ROUTINE gds__vtov(const SCHAR* string, char* field, SSHORT length)
{
/**************************************
*
* g d s _ v t o v
*
**************************************
*
* Functional description
* Move a null terminated string to a fixed length
* field. The call is primarily generated by the
* preprocessor. Until gds__vtof, the target string
* is null terminated.
*
**************************************/
--length;
while ((*field++ = *string++) != 0)
if (--length <= 0) {
*field = 0;
return;
}
}
void API_ROUTINE isc_print_sqlerror(SSHORT sqlcode, STATUS * status)
{
/**************************************
*
* i s c _ p r i n t _ s q l e r r o r
*
**************************************
*
* Functional description
* Given an sqlcode, give as much info as possible.
* Decide whether status is worth mentioning.
*
**************************************/
TEXT error_buffer[192], *p;
sprintf(error_buffer, "SQLCODE: %d\nSQL ERROR:\n", sqlcode);
for (p = error_buffer; *p;)
p++;
isc_sql_interprete(sqlcode, p,
(SSHORT) (sizeof(error_buffer) - (p - error_buffer) -
2));
while (*p)
p++;
*p++ = '\n';
*p = 0;
gds__put_error(error_buffer);
if (status && status[1]) {
gds__put_error("ISC STATUS: ");
gds__print_status(status);
}
}
void API_ROUTINE isc_sql_interprete(
SSHORT sqlcode,
TEXT * buffer, SSHORT length)
{
/**************************************
*
* i s c _ s q l _ i n t e r p r e t e
*
**************************************
*
* Functional description
* Given an sqlcode, return a buffer full of message
* This is very simple because all we have is the
* message number, and the sqlmessages have as yet no
* arguments
*
* NOTE: As of 21-APR-1999, sqlmessages HAVE arguments hence use
* an empty string instead of NULLs.
*
**************************************/
TEXT *str = "";
if (sqlcode < 0)
gds__msg_format(0, 13, (USHORT) (1000 + sqlcode), length, buffer,
str, str, str, str, str);
else
gds__msg_format(0, 14, sqlcode, length, buffer,
str, str, str, str, str);
}
#ifdef VMS
int unlink(SCHAR * file)
{
/**************************************
*
* u n l i n k
*
**************************************
*
* Functional description
* Get rid of a file and all of its versions.
*
**************************************/
int status;
struct dsc$descriptor_s desc;
ISC_make_desc(file, &desc, 0);
for (;;) {
status = lib$delete_file(&desc);
if (!(status & 1))
break;
}
if (!status)
return 0;
else if (status != RMS$_FNF)
return -1;
else
return 0;
}
#endif
static void blr_error(CTL control, TEXT * string, ...)
{
/**************************************
*
* b l r _ e r r o r
*
**************************************
*
* Functional description
* Report back an error message and unwind.
*
**************************************/
USHORT offset;
va_list args;
VA_START(args, string);
blr_format(control, string, args);
offset = 0;
PRINT_LINE;
Firebird::status_exception::raise(-1);
}
static void blr_format(CTL control, const char * string, ...)
{
/**************************************
*
* b l r _ f o r m a t
*
**************************************
*
* Functional description
* Format an utterance.
*
**************************************/
va_list ptr;
VA_START(ptr, string);
vsprintf(control->ctl_ptr, string, ptr);
while (*control->ctl_ptr)
control->ctl_ptr++;
}
static void blr_indent(CTL control, SSHORT level)
{
/**************************************
*
* b l r _ i n d e n t
*
**************************************
*
* Functional description
* Indent for pretty printing.
*
**************************************/
level *= 3;
while (--level >= 0)
PUT_BYTE(' ');
}
static void blr_print_blr(CTL control, UCHAR operator_)
{
/**************************************
*
* b l r _ p r i n t _ b l r
*
**************************************
*
* Functional description
* Print a blr item.
*
**************************************/
SCHAR *p;
if (operator_ > FB_NELEM(blr_table) ||
!(p = (SCHAR *) /* const_cast */ blr_table[operator_].blr_string))
blr_error(control, "*** blr operator %d is undefined ***",
(int) operator_);
blr_format(control, "blr_%s, ", p);
}
static SCHAR blr_print_byte(CTL control)
{
/**************************************
*
* b l r _ p r i n t _ b y t e
*
**************************************
*
* Functional description
* Print a byte as a numeric value and return same.
*
**************************************/
UCHAR v;
v = BLR_BYTE;
blr_format(control, (control->ctl_language) ? "chr(%d), " : "%d, ",
(int) v);
return v;
}
static SCHAR blr_print_char(CTL control)
{
/**************************************
*
* b l r _ p r i n t _ c h a r
*
**************************************
*
* Functional description
* Print a byte as a numeric value and return same.
*
**************************************/
SCHAR c;
UCHAR v;
SSHORT printable;
v = c = BLR_BYTE;
printable = (c >= 'a' && c <= 'z') ||
(c >= 'A' && c <= 'Z') ||
(c >= '0' && c <= '9') ||
c == '$' ||
c == '_';
if (printable)
blr_format(control, "'%c',", (char) c);
else if (control->ctl_language)
blr_format(control, "chr(%d),", (int) v);
else
blr_format(control, "%d,", (int) c);
return c;
}
static void blr_print_cond(CTL control)
{
/**************************************
*
* b l r _ p r i n t _ c o n d
*
**************************************
*
* Functional description
* Print an error condition.
*
**************************************/
USHORT ctype;
SSHORT n;
ctype = BLR_BYTE;
switch (ctype) {
case blr_gds_code:
blr_format(control, "blr_gds_code, ");
n = PRINT_BYTE;
while (--n >= 0)
PRINT_CHAR;
break;
case blr_exception:
blr_format(control, "blr_exception, ");
n = PRINT_BYTE;
while (--n >= 0)
PRINT_CHAR;
break;
case blr_sql_code:
blr_format(control, "blr_sql_code, ");
PRINT_WORD;
break;
case blr_default_code:
blr_format(control, "blr_default_code, ");
break;
default:
blr_error(control, "*** invalid condition type ***");
break;
}
return;
}
static int blr_print_dtype(CTL control)
{
/**************************************
*
* b l r _ p r i n t _ d t y p e
*
**************************************
*
* Functional description
* Print a datatype sequence and return the length of the
* data described.
*
**************************************/
USHORT dtype;
SCHAR *string;
SSHORT length;
UCHAR v1, v2;
dtype = BLR_BYTE;
/* Special case blob (261) to keep down the size of the
jump table */
switch (dtype) {
case blr_short:
string = "short";
length = 2;
break;
case blr_long:
string = "long";
length = 4;
break;
case blr_int64:
string = "int64";
length = 8;
break;
case blr_quad:
string = "quad";
length = 8;
break;
case blr_timestamp:
string = "timestamp";
length = 8;
break;
case blr_sql_time:
string = "sql_time";
length = 4;
break;
case blr_sql_date:
string = "sql_date";
length = 4;
break;
case blr_float:
string = "float";
length = 4;
break;
case blr_double:
string = "double";
/* for double literal, return the length of the numeric string */
v1 = *(control->ctl_blr);
v2 = *(control->ctl_blr + 1);
length = ((v2 << 8) | v1) + 2;
break;
case blr_d_float:
string = "d_float";
length = 8;
break;
case blr_text:
string = "text";
break;
case blr_cstring:
string = "cstring";
break;
case blr_varying:
string = "varying";
break;
case blr_text2:
string = "text2";
break;
case blr_cstring2:
string = "cstring2";
break;
case blr_varying2:
string = "varying2";
break;
default:
blr_error(control, "*** invalid data type ***");
break;
}
blr_format(control, "blr_%s, ", string);
switch (dtype) {
case blr_text:
length = PRINT_WORD;
break;
case blr_varying:
length = PRINT_WORD + 2;
break;
case blr_short:
case blr_long:
case blr_quad:
case blr_int64:
PRINT_BYTE;
break;
case blr_text2:
PRINT_WORD;
length = PRINT_WORD;
break;
case blr_varying2:
PRINT_WORD;
length = PRINT_WORD + 2;
break;
case blr_cstring2:
PRINT_WORD;
length = PRINT_WORD;
break;
default:
if (dtype == blr_cstring)
length = PRINT_WORD;
break;
}
return length;
}
static void blr_print_join(CTL control)
{
/**************************************
*
* b l r _ p r i n t _ j o i n
*
**************************************
*
* Functional description
* Print a join type.
*
**************************************/
USHORT join_type;
SCHAR *string;
join_type = BLR_BYTE;
switch (join_type) {
case blr_inner:
string = "inner";
break;
case blr_left:
string = "left";
break;
case blr_right:
string = "right";
break;
case blr_full:
string = "full";
break;
default:
blr_error(control, "*** invalid join type ***");
break;
}
blr_format(control, "blr_%s, ", string);
}
static SLONG blr_print_line(CTL control, SSHORT offset)
{
/**************************************
*
* b l r _ p r i n t _ l i n e
*
**************************************
*
* Functional description
* Invoke callback routine to print (or do something with) a line.
*
**************************************/
*control->ctl_ptr = 0;
#pragma FB_COMPILER_MESSAGE("Fix! Ugly function pointer cast!")
((void (*)(...)) (*control->ctl_routine)) (control->ctl_user_arg, offset, control->ctl_buffer);
control->ctl_ptr = control->ctl_buffer;
return control->ctl_blr - control->ctl_blr_start;
}
static void blr_print_verb(CTL control, SSHORT level)
{
/**************************************
*
* b l r _ p r i n t _ v e r b
*
**************************************
*
* Functional description
* Primary recursive routine to print BLR.
*
**************************************/
SSHORT n, n2;
SLONG offset;
offset = control->ctl_blr - control->ctl_blr_start;
blr_indent(control, level);
UCHAR operator_ = BLR_BYTE;
if ((SCHAR) operator_ == (SCHAR) blr_end) {
blr_format(control, "blr_end, ");
PRINT_LINE;
return;
}
blr_print_blr(control, operator_);
level++;
const UCHAR *ops = blr_table[operator_].blr_operators;
while (*ops)
switch (*ops++) {
case op_verb:
PRINT_VERB;
break;
case op_line:
offset = PRINT_LINE;
break;
case op_byte:
n = PRINT_BYTE;
break;
case op_word:
n = PRINT_WORD;
break;
case op_pad:
PUT_BYTE(' ');
break;
case op_dtype:
n = PRINT_DTYPE;
break;
case op_literal:
while (--n >= 0)
PRINT_CHAR;
break;
case op_join:
PRINT_JOIN;
break;
case op_message:
while (--n >= 0) {
blr_indent(control, level);
PRINT_DTYPE;
offset = PRINT_LINE;
}
break;
case op_parameters:
level++;
while (--n >= 0)
PRINT_VERB;
level--;
break;
case op_error_handler:
while (--n >= 0) {
blr_indent(control, level);
PRINT_COND;
offset = PRINT_LINE;
}
break;
case op_set_error:
PRINT_COND;
break;
case op_indent:
blr_indent(control, level);
break;
case op_begin:
while ((SCHAR) * (control->ctl_blr) != (SCHAR) blr_end)
PRINT_VERB;
break;
case op_map:
while (--n >= 0) {
blr_indent(control, level);
PRINT_WORD;
offset = PRINT_LINE;
PRINT_VERB;
}
break;
case op_args:
while (--n >= 0)
PRINT_VERB;
break;
case op_literals:
while (--n >= 0) {
blr_indent(control, level);
n2 = PRINT_BYTE;
while (--n2 >= 0)
PRINT_CHAR;
offset = PRINT_LINE;
}
break;
case op_union:
while (--n >= 0) {
PRINT_VERB;
PRINT_VERB;
}
break;
case op_relation:
operator_ = BLR_BYTE;
blr_print_blr(control, operator_);
if (operator_ != blr_relation && operator_ != blr_rid)
blr_error(control,
"*** blr_relation or blr_rid must be object of blr_lock_relation, %d found ***",
(int) operator_);
if (operator_ == blr_relation) {
n = PRINT_BYTE;
while (--n >= 0)
PRINT_CHAR;
}
else
PRINT_WORD;
break;
default:
assert(FALSE);
break;
}
}
static int blr_print_word(CTL control)
{
/**************************************
*
* b l r _ p r i n t _ w o r d
*
**************************************
*
* Functional description
* Print a VAX word as a numeric value an return same.
*
**************************************/
UCHAR v1, v2;
v1 = BLR_BYTE;
v2 = BLR_BYTE;
blr_format(control,
(control->ctl_language) ? "chr(%d),chr(%d), " : "%d,%d, ",
(int) v1, (int) v2);
return (v2 << 8) | v1;
}
void gds__cleanup(void)
{
/**************************************
*
* c l e a n u p
*
**************************************
*
* Functional description
* Exit handler for image exit.
*
**************************************/
CLEAN clean;
FPTR_VOID_PTR routine;
void *arg;
#ifdef UNIX
if (gds_pid != getpid())
return;
#endif
gds__msg_close(NULL);
while (clean = cleanup_handlers) {
cleanup_handlers = clean->clean_next;
routine = clean->clean_routine;
arg = clean->clean_arg;
/* We must free the handler before calling it because there
may be a handler (and is) that frees all memory that has
been allocated. */
FREE_LIB_MEMORY(clean);
#pragma FB_COMPILER_MESSAGE("Fix! Ugly function pointer cast!")
((void (*)(void *)) (*routine)) (arg);
}
V4_MUTEX_DESTROY(&alloc_mutex);
/* V4_DESTROY; */
initialized = 0;
}
static void cleanup_malloced_memory(void *arg)
{
/**************************************
*
* c l e a n u p _ m a l l o c e d _ m e m o r y
*
**************************************
*
* Functional description
* Return the real memory that was acquired in gds__alloc
* to the system. Refer to gds__alloc for the cases when
* the malloc pointer is saved and when the malloc pointer
* is not saved.
* This substantive work of this function is only turned on
* for Novell.
*
**************************************/
void *ptr;
#if (defined (WIN_NT) || defined (UNIX))
while (ptr = malloced_memory) {
malloced_memory = reinterpret_cast < void **>(*malloced_memory);
free(ptr);
}
#endif
pool = NULL;
#if (defined SOLARIS && defined SUPERSERVER && !defined(MAP_ANON))
/* Close the file used as the basis for generating
anonymous virtual memory. Since it is created
pre-deleted there's no need to unlink it here. */
close(anon_fd);
anon_fd = -1;
#endif
}
static ULONG free_memory(void *blk)
{
/**************************************
*
* f r e e _ m e m o r y
*
**************************************
*
* Functional description
* Release a block allocated by gds__alloc. Return number
* of bytes released.
*
**************************************/
FREE HUGE_PTR *ptr, prior, free_blk, block;
SLONG length;
#ifdef DEV_BUILD
if (!blk) {
DEV_REPORT("gds__free: Tried to free NULL\n");
BREAKPOINT(__LINE__);
return 0;
}
#endif
block = (FREE) ((UCHAR *) blk - ALLOC_HEADER_SIZE);
blk = block;
#ifdef DEV_BUILD
gds_delta_alloc -= *((ULONG *) blk);
#endif
length = block->free_length = *((ULONG *) blk);
#ifdef DEBUG_GDS_ALLOC
block->free_count = gds_free_call_count;
if (gds_bug_free_count && (gds_free_call_count == gds_bug_free_count)) {
PRINTF("gds__free call %d reached, address=%lx\n", gds_bug_free_count,
blk);
BREAKPOINT(__LINE__);
}
#endif
#ifdef DEV_BUILD
if (length <= (SLONG) ALLOC_OVERHEAD) {
DEV_REPORT("gds__free: attempt to release bad block (too small)\n"); /* TXNN */
BREAKPOINT(__LINE__);
return 0;
}
#endif
/* Find the logical place in the free block list for the returning
block. If it adjoins existing blocks, merge it (them). */
prior = NULL;
for (ptr = &pool; (free_blk = *ptr) != NULL;
prior = free_blk, ptr = &free_blk->free_next) {
if (free_blk->free_next
&& (UCHAR HUGE_PTR *) free_blk >=
(UCHAR HUGE_PTR *) free_blk->free_next) {
DEV_REPORT("gds__free: pool corrupted"); /* TXNN */
BREAKPOINT(__LINE__);
free_blk = *ptr = NULL;
break;
}
if ((UCHAR HUGE_PTR *) block < (UCHAR HUGE_PTR *) free_blk)
break;
}
/* Make sure everything is completely kosher */
if (block->free_length <= 0 ||
(prior
&& (UCHAR HUGE_PTR *) block <
(UCHAR HUGE_PTR *) prior + prior->free_length) || (free_blk
&& (UCHAR HUGE_PTR
*) block +
block->free_length
>
(UCHAR HUGE_PTR *)
free_blk)) {
DEV_REPORT("gds__free: attempt to release bad block\n"); /* TXNN */
BREAKPOINT(__LINE__);
return 0;
}
/* Start by linking block into chain */
block->free_next = free_blk;
*ptr = block;
/* Try to merge the free block with the next block */
if (free_blk
&& (UCHAR HUGE_PTR *) block + block->free_length ==
(UCHAR HUGE_PTR *) free_blk) {
block->free_length += free_blk->free_length;
block->free_next = free_blk->free_next;
#ifdef DEBUG_GDS_ALLOC
/* We're making a larger free chuck - be sure to color the header
* of the "merged" block as freed space
*/
memset(free_blk, ALLOC_FREED_PATTERN, sizeof(struct free));
#endif
}
/* Next, try to merge the free block with the prior block */
if (prior
&& (UCHAR HUGE_PTR *) prior + prior->free_length ==
(UCHAR HUGE_PTR *) block) {
prior->free_length += block->free_length;
prior->free_next = block->free_next;
#ifdef DEBUG_GDS_ALLOC
/* We're making a larger free chuck - be sure to color the header
* of the "merged" block as freed space
*/
memset(block, ALLOC_FREED_PATTERN, sizeof(struct free));
#endif
}
#ifdef DEBUG_GDS_ALLOC
if (gds_bug_free_count && (gds_free_call_count >= gds_bug_free_count))
gds_alloc_report(ALLOC_silent, __FILE__, __LINE__);
#endif
/* The number of bytes freed from the user's point
of view - eg: the block length, less the malloc overhead */
if (length % (1024 + ALLOC_OVERHEAD) == 0) {
return (length - ALLOC_OVERHEAD * (length / (1024 + ALLOC_OVERHEAD)));
}
return length - ALLOC_OVERHEAD;
}
static void init(void)
{
/**************************************
*
* i n i t
*
**************************************
*
* Functional description
* Do anything necessary to initialize module/system.
*
**************************************/
#if defined(VMS)
STATUS status;
#endif
if (initialized)
return;
#ifdef DEBUG_GDS_ALLOC
{
char *t = getenv("DEBUG_GDS_ALLOC");
if (t) {
/* Set which call to gds__alloc starts full memory debugging */
if (*t >= '0' && *t <= '9')
gds_alloc_state = atoi(t);
else
gds_alloc_state = 0;
}
t = getenv("DEBUG_GDS_ALLOC_NOMEM");
if (t) {
/* Set which call to gds__alloc "exhausts" available memory */
if (*t >= '0' && *t <= '9')
gds_alloc_nomem = atoi(t);
else
gds_alloc_nomem = 0;
}
/* Open the memory debugging configuration file, if any. */
IB_FILE *file = ib_fopen(DBG_MEM_FILE, "r");
if (file) {
ib_fscanf(file, "%ld", &gds_bug_alloc_count);
ib_fscanf(file, "%ld", &gds_bug_free_count);
PRINTF("gds_bug_alloc_count=%d, gds_bug_free_count=%d\n",
gds_bug_alloc_count, gds_bug_free_count);
ib_fclose(file);
}
}
BREAKPOINT(__LINE__);
#endif /* DEBUG_GDS_ALLOC */
/* V4_INIT; */
/* V4_GLOBAL_MUTEX_LOCK; */
if (initialized) {
/* V4_GLOBAL_MUTEX_UNLOCK; */
return;
}
V4_MUTEX_INIT(&alloc_mutex);
#ifdef VMS
exit_description.exit_handler = cleanup;
exit_description.args = 1;
exit_description.arg[0] = &exit_status;
status = sys$dclexh(&exit_description);
#endif
#ifdef UNIX
gds_pid = getpid();
#ifdef SUPERSERVER
#if (defined SOLARIS || defined HPUX || defined LINUX)
{
/* Increase max open files to hard limit for Unix
platforms which are known to have low soft limits. */
struct rlimit old, new_;
if (!getrlimit(RLIMIT_NOFILE, &old) && old.rlim_cur < old.rlim_max) {
new_.rlim_cur = new_.rlim_max = old.rlim_max;
if (!setrlimit(RLIMIT_NOFILE, &new_))
{
#if _FILE_OFFSET_BITS == 64 && defined SOLARIS
gds__log("64 bit i/o support is on.");
gds__log("Open file limit increased from %lld to %lld",
old.rlim_cur, new_.rlim_cur);
#else
gds__log("Open file limit increased from %d to %d",
old.rlim_cur, new_.rlim_cur);
#endif
}
}
}
#endif
#endif /* SUPERSERVER */
#endif /* UNIX */
#ifdef HAVE_ATEXIT
atexit(gds__cleanup);
#else
#ifdef HAVE_ON_EXIT
on_exit(gds__cleanup, 0);
#endif
#endif
initialized = 1;
gds__register_cleanup(cleanup_malloced_memory, 0);
#ifndef REQUESTER
ISC_signal_init();
#endif
/* V4_GLOBAL_MUTEX_UNLOCK; */
}
static int yday(struct tm *times)
{
/**************************************
*
* y d a y
*
**************************************
*
* Functional description
* Convert a calendar date to the day-of-year.
*
* The unix time structure considers
* january 1 to be Year day 0, although it
* is day 1 of the month. (Note that QLI,
* when printing Year days takes the other
* view.)
*
**************************************/
SSHORT day, month, year;
const BYTE *days;
static const BYTE month_days[] =
{ 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };
day = times->tm_mday;
month = times->tm_mon;
year = times->tm_year + 1900;
--day;
for (days = month_days; days < month_days + month; days++)
day += *days;
if (month < 2)
return day;
/* Add a day as we're past the leap-day */
if (!(year % 4))
day++;
/* Ooops - year's divisible by 100 aren't leap years */
if (!(year % 100))
day--;
/* Unless they are also divisible by 400! */
if (!(year % 400))
day++;
return day;
}
// Solaris <=2.7 does not support map_anon use old code - nmcc dec2002
#if (defined(SOLARIS) && !defined(MAP_ANON))
UCHAR *mmap_anon(SLONG size)
{
/**************************************
*
* m m a p _ a n o n (S O L A R I S)
*
**************************************
*
* Functional description
* Allocate anonymous virtual memory for Solaris.
* Use an arbitrary file descriptor as a basis
* for file mapping.
*
**************************************/
char *memory, *va, *va_end, *va1;
ULONG chunk, errno1;
int anon_fd=-1, page_size=-1;
/* Choose SYS_ALLOC_CHUNK such that it is always valid for the
underlying mapped file and is a multiple of any conceivable
memory page size that a hardware platform might support. */
#define SYS_ALLOC_CHUNK 0x20000 /* 131,072 bytes */
/* All virtual addresses generated will be remapped or wrapped around
to the start of the mapped file. The mapped file must be at least
SYS_ALLOC_CHUCK large to guarantee valid memory references. */
if (anon_fd == -1) {
anon_fd = (int) gds__temp_file(FALSE, "gds_anon", NULL);
while (ftruncate(anon_fd, SYS_ALLOC_CHUNK) == -1)
if (!SYSCALL_INTERRUPTED(errno)) {
close(anon_fd);
anon_fd = -1;
ERR_post(isc_sys_request, isc_arg_string, "ftruncate",
isc_arg_unix, errno, 0);
}
}
/* Get memory page size for virtual memory checking. */
if ((page_size = sysconf(_SC_PAGESIZE)) == -1)
ERR_post(isc_sys_request, isc_arg_string, "sysconf", isc_arg_unix,
errno, 0);
/* Generate a virtually contiguous address space for size requested.
MAP_PRIVATE is critical here, otherwise the mapped file turns to jelly. */
memory = mmap(NULL, size, (PROT_READ | PROT_WRITE), MAP_PRIVATE, anon_fd, 0);
if (memory == MAP_FAILED) {
/* If no swap space or memory available maybe caller can ask for less.
Punt on unexpected errors. */
if (errno == EAGAIN || errno == ENOMEM)
return NULL;
else
ERR_post(isc_sys_request, isc_arg_string, "mmap", isc_arg_unix,
errno, 0);
}
/* Memory returned from mmap should always be page-aligned. */
assert(((ULONG) memory & (page_size - 1)) == 0);
/* Some of this virtual address space may be invalid because the
mapped file isn't large enough, remap all virtual addresses to
the beginning of the file which will validate the illegal memory
references. The use of MAP_FIXED guarantees that the virtual
address space remains contiguous during this legerdemain. */
va = memory;
va_end = memory + size;
for (va += SYS_ALLOC_CHUNK; va < va_end; va += chunk) {
chunk = MIN(va_end - va, SYS_ALLOC_CHUNK);
va1 = mmap(va, chunk, (PROT_READ | PROT_WRITE),
(MAP_PRIVATE | MAP_FIXED), anon_fd, 0);
if (va1 != va) {
assert(FALSE);
errno1 = errno;
munmap(memory, size);
ERR_post(isc_sys_request, isc_arg_string, "mmap", isc_arg_unix,
errno1, 0);
}
}
/* All modfiy references will transparently revector from the mapped
file to private memory backed by the swap file. However, since the
database buffer pool uses this allocation mechanism, make the
revectoring happen now by touching each memory page. This protects
against DMA (Direct Memory Access) transfers used by "raw" I/O
device interfaces which may bypass the virtual memory manager. */
for (va = memory; va < va_end; va += page_size)
*(ULONG *) va = (ULONG) va;
#ifdef DEV_BUILD
/* Since the mapped object is mapped repeatedly by different
chunks of the virtual address space, check the same physical
memory pages aren't being shared by labeling each virtual page
with its address. Verify every page is still labeled correctly .*/
for (va = memory; va < va_end; va += page_size)
assert(*(ULONG *) va == (ULONG) va);
#endif
return (UCHAR *) memory;
}
#endif
static void ndate(SLONG nday, struct tm *times)
{
/**************************************
*
* n d a t e
*
**************************************
*
* Functional description
* Convert a numeric day to [day, month, year].
*
* Calenders are divided into 4 year cycles.
* 3 Non-Leap years, and 1 leap year.
* Each cycle takes 365*4 + 1 == 1461 days.
* There is a further cycle of 100 4 year cycles.
* Every 100 years, the normally expected leap year
* is not present. Every 400 years it is.
* This cycle takes 100 * 1461 - 3 == 146097 days
* The origin of the constant 2400001 is unknown.
* The origin of the constant 1721119 is unknown.
* The difference between 2400001 and 1721119 is the
* number of days From 0/0/0000 to our base date of
* 11/xx/1858. (678882)
* The origin of the constant 153 is unknown.
*
* This whole routine has problems with ndates
* less than -678882 (Approx 2/1/0000).
*
**************************************/
SLONG year, month, day;
SLONG century;
nday -= 1721119 - 2400001;
century = (4 * nday - 1) / 146097;
nday = 4 * nday - 1 - 146097 * century;
day = nday / 4;
nday = (4 * day + 3) / 1461;
day = 4 * day + 3 - 1461 * nday;
day = (day + 4) / 4;
month = (5 * day - 3) / 153;
day = 5 * day - 3 - 153 * month;
day = (day + 5) / 5;
year = 100 * century + nday;
if (month < 10)
month += 3;
else {
month -= 9;
year += 1;
}
times->tm_mday = (int) day;
times->tm_mon = (int) month - 1;
times->tm_year = (int) year - 1900;
}
static GDS_DATE nday(struct tm *times)
{
/**************************************
*
* n d a y
*
**************************************
*
* Functional description
* Convert a calendar date to a numeric day
* (the number of days since the base date).
*
**************************************/
SSHORT day, month, year;
SLONG c, ya;
day = times->tm_mday;
month = times->tm_mon + 1;
year = times->tm_year + 1900;
if (month > 2)
month -= 3;
else {
month += 9;
year -= 1;
}
c = year / 100;
ya = year - 100 * c;
return (GDS_DATE) (((SINT64) 146097 * c) / 4 +
(1461 * ya) / 4 +
(153 * month + 2) / 5 + day + 1721119 - 2400001);
}
static void sanitize(TEXT * locale)
{
/**************************************
*
* s a n i t i z e
*
**************************************
*
* Functional description
* Clean up a locale to make it acceptable for use in file names
* for Windows NT, PC.
* replace any period with '_' for NT or PC.
*
**************************************/
while (*locale) {
if (*locale == '.')
*locale = '_';
locale++;
}
}
#ifdef DEBUG_GDS_ALLOC
#undef gds__alloc
void* API_ROUTINE gds__alloc(SLONG size)
{
/**************************************
*
* g d s _ $ a l l o c (alternate debug entrypoint)
*
**************************************
*
* Functional description
*
* NOTE: This function should be the last in the file due to
* the undef of gds__alloc above.
*
* For modules not recompiled with DEBUG_GDS_ALLOC, this provides
* an entry point to malloc again.
*
**************************************/
return gds__alloc_debug(size, "-- Unknown --", 0);
}
#endif
#ifdef SUPERSERVER
static void freemap(int cod)
{
FREE *next, free_blk;
ULONG free_list_count = 0;
ULONG total_free = 0;
ib_printf("\nFree Start\t\tFree Length\t\tFree End\n");
for (next = &pool; (free_blk = *next) != NULL; next = &(*next)->free_next) {
++free_list_count;
total_free += free_blk->free_length;
ib_printf("%x\t\t\t%d\t\t\t%x\n", (UCHAR HUGE_PTR *) free_blk,
free_blk->free_length,
(UCHAR HUGE_PTR *) (free_blk + free_blk->free_length));
}
ib_printf("Stats after %s\n", (cod) ? "Merging" : "Allocating");
ib_printf("Free list count = %d\n", free_list_count);
ib_printf("Total Free = %d\n", total_free);
gds_print_delta_counters(ib_stdout);
}
void gds_print_delta_counters(IB_FILE * fptr)
{
#ifdef DEV_BUILD
ib_fprintf(fptr, "gds_delta_alloc = %d\n", gds_delta_alloc);
ib_fprintf(fptr, "gds_max_alloc = %d\n", gds_max_alloc);
#endif
ib_fprintf(fptr, "allr_delta_alloc = %d\n", allr_delta_alloc);
ib_fprintf(fptr, "alld_delta_alloc = %d\n", alld_delta_alloc);
ib_fprintf(fptr, "all_delta_alloc = %d\n", all_delta_alloc);
}
#endif
} // extern "C"