8
0
mirror of https://github.com/FirebirdSQL/firebird.git synced 2025-01-27 18:03:04 +01:00
firebird-mirror/src/jrd/why.cpp

6058 lines
146 KiB
C++
Raw Normal View History

/*
* PROGRAM: JRD Access Method
* MODULE: why.cpp
* DESCRIPTION: Universal Y-valve
*
* 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): ______________________________________.
*
* 23-Feb-2002 Dmitry Yemanov - Events wildcarding
*
*
* 2002-02-23 Sean Leyne - Code Cleanup, removed old Win3.1 port (Windows_Only)
* 2002-02-23 Sean Leyne - Code Cleanup, removed old M88K and NCR3000 port
*
* 2002.10.27 Sean Leyne - Completed removal of obsolete "IMP" port
* 2002.10.27 Sean Leyne - Completed removal of obsolete "DG_X86" port
*
* 2002.10.28 Sean Leyne - Code cleanup, removed obsolete "MPEXL" port
* 2002.10.28 Sean Leyne - Code cleanup, removed obsolete "DecOSF" port
* 2002.10.28 Sean Leyne - Code cleanup, removed obsolete "SGI" port
*
* 2002.10.29 Sean Leyne - Removed obsolete "Netware" port
*
* 2002.10.30 Sean Leyne - Removed support for obsolete "PC_PLATFORM" define
*
* 2002.12.10 Alex Peshkoff: 1. Moved struct hndl declaration to h-file
* 2. renamed all specific objects to WHY_*
*
*/
#include "firebird.h"
#include <stdlib.h>
#include <string.h>
#include "../jrd/common.h"
#include <stdarg.h>
2004-04-29 00:43:34 +02:00
#include <stdio.h>
2003-11-04 00:59:24 +01:00
#include "../jrd/gdsassert.h"
#include "../jrd/y_handle.h"
#include "gen/iberror.h"
#include "../jrd/msg_encode.h"
#include "gen/msg_facs.h"
#include "../jrd/acl.h"
#include "../jrd/inf_pub.h"
#include "../jrd/thd.h"
#include "../jrd/isc.h"
#include "../jrd/fil.h"
2003-11-26 12:15:32 +01:00
#include "../jrd/flu.h"
#include "../jrd/db_alias.h"
#include "../common/classes/ClumpletWriter.h"
/* includes specific for DSQL */
#include "../dsql/sqlda.h"
/* end DSQL-specific includes */
#include "../jrd/flu_proto.h"
#include "../jrd/gds_proto.h"
#include "../jrd/isc_proto.h"
#include "../jrd/isc_f_proto.h"
#ifndef REQUESTER
#include "../jrd/os/isc_i_proto.h"
#include "../jrd/isc_s_proto.h"
#include "../jrd/sch_proto.h"
#endif
#include "../jrd/thread_proto.h"
#include "../jrd/utl_proto.h"
#include "../dsql/dsql_proto.h"
#include "../dsql/prepa_proto.h"
#include "../dsql/utld_proto.h"
#include "../jrd/why_proto.h"
2004-05-03 01:06:37 +02:00
#include "../common/classes/rwlock.h"
2004-05-03 03:53:24 +02:00
#include "../common/classes/auto.h"
// In 2.0 it's hard to include ibase.h in why.cpp due to API declaration conflicts.
// Taking into account that given y-valve lives it's last version,
// in which dpb version is not likely to change, define it here.
// #include "../jrd/ibase.h"
const UCHAR isc_dpb_version1 = 1;
#ifdef HAVE_ERRNO_H
#include <errno.h>
#endif
#ifdef HAVE_FCNTL_H
#include <fcntl.h>
#endif
#ifdef HAVE_UNISTD_H
#include <unistd.h>
#endif
#ifdef HAVE_FLOCK
#include <sys/file.h> /* for flock() */
#endif
#ifdef WIN_NT
#include <windows.h>
#endif
#ifndef F_OK
#define F_OK 0
#endif
#ifndef F_TLOCK
#define F_TLOCK 2
#endif
2004-05-18 18:48:26 +02:00
const int IO_RETRY = 20;
#ifdef DEV_BUILD
#define CHECK_STATUS(v) check_status_vector(v, !FB_SUCCESS)
#define CHECK_STATUS_SUCCESS(v) check_status_vector(v, FB_SUCCESS)
2004-05-18 18:48:26 +02:00
#else
#define CHECK_STATUS(v) /* nothing */
#define CHECK_STATUS_SUCCESS(v) /* nothing */
#endif
inline void init_status(ISC_STATUS* vector)
{
vector[0] = isc_arg_gds;
vector[1] = FB_SUCCESS;
vector[2] = isc_arg_end;
}
inline bool is_network_error(const ISC_STATUS* vector)
{
return vector[1] == isc_network_error || vector[1] == isc_net_write_err ||
vector[1] == isc_net_read_err;
}
2004-05-03 01:06:37 +02:00
#define TRANSLATE_HANDLE(public_handle, handle, handle_type, error_code) \
if (!((handle) = WHY_translate_handle(public_handle)) || (handle)->type != handle_type) \
return bad_handle (user_status, error_code);
#define FIND_TRANSACTION(public_handle, handle, database, error_code) \
if (!((handle) = find_transaction(database, WHY_translate_handle(public_handle))) || (handle)->type != HANDLE_transaction) \
return bad_handle (user_status, error_code);
#define NULL_CHECK(ptr, code, type) if (*ptr) return bad_handle (user_status, code)
#define GET_STATUS { if (!(status = user_status)) status = local; init_status(status); }
#define RETURN_SUCCESS { subsystem_exit(); CHECK_STATUS_SUCCESS (status); return FB_SUCCESS; }
#if defined(REQUESTER) || defined (SUPERCLIENT)
#define NO_LOCAL_DSQL
#endif
#if defined (SERVER_SHUTDOWN) && !defined (SUPERCLIENT) && !defined (REQUESTER)
static BOOLEAN shutdown_flag = FALSE;
#endif /* SERVER_SHUTDOWN && !SUPERCLIENT && !REQUESTER */
2004-05-09 07:48:33 +02:00
typedef ISC_STATUS(*PTR) (ISC_STATUS* user_status, ...);
/* Database cleanup handlers */
2004-05-24 19:31:47 +02:00
struct clean
{
struct clean* clean_next;
union {
DatabaseCleanupRoutine* DatabaseRoutine;
TransactionCleanupRoutine *TransactionRoutine;
};
void* clean_arg;
2004-05-24 19:31:47 +02:00
};
typedef clean *CLEAN;
/* Transaction element block */
2004-05-24 19:31:47 +02:00
struct teb
{
2004-05-03 01:06:37 +02:00
FB_API_HANDLE *teb_database;
int teb_tpb_length;
UCHAR *teb_tpb;
2004-05-24 19:31:47 +02:00
};
typedef teb TEB;
2004-05-03 01:06:37 +02:00
#ifdef DEBUG_GDS_ALLOC
#define alloc(x) alloc_debug(x, __FILE__, __LINE__)
static SCHAR *alloc_debug(SLONG, const char*, int);
#else
static SCHAR *alloc(SLONG);
#endif
static void free_block(void*);
class HandlePublicKey {
public:
static const FB_API_HANDLE& generate(const void* sender, why_hndl* value) {
return value->public_handle;
}
};
typedef Firebird::BePlusTree<why_hndl*, FB_API_HANDLE, MemoryPool, HandlePublicKey> HandleMapping;
2004-05-03 03:53:24 +02:00
static Firebird::AutoPtr<HandleMapping> handleMapping;
2004-05-03 01:06:37 +02:00
static ULONG handle_sequence_number = 0;
static Firebird::RWLock handleMappingLock;
why_hndl* WHY_alloc_handle(int implementation, int handle_type)
{
/**************************************
*
* W H Y _ a l l o c _ h a n d l e
*
**************************************
*
* Functional description
* Allocate an indirect handle.
*
**************************************/
WHY_HNDL handle = (WHY_HNDL) alloc((SLONG) sizeof(why_hndl));
if (handle)
{
handle->implementation = implementation;
handle->type = handle_type;
#ifdef DEBUG_GDS_ALLOC
/* As the memory for the handle is handed back to the client, InterBase
* cannot free the memory unless the client returns to us. As a result,
* the memory allocator might try to report this as unfreed, but it
* ain't our fault. So flag it to make the allocator be happy.
*/
gds_alloc_flag_unfreed((void *) handle);
#endif
handleMappingLock.beginWrite();
try {
2004-05-03 03:53:24 +02:00
if (!handleMapping)
handleMapping = FB_NEW(*getDefaultMemoryPool())
HandleMapping(getDefaultMemoryPool());
2004-05-03 01:06:37 +02:00
// Loop until we find an empty handle slot.
// This is to care of case when counter rolls over
do {
// Generate handle number using rolling counter.
// This way we tend to give out unique handle numbers and closed
// handles do not appear as valid to our clients.
ULONG temp = ++handle_sequence_number;
// Avoid generating NULL handle when sequence number wraps
if (!temp)
temp = ++handle_sequence_number;
handle->public_handle = reinterpret_cast<FB_API_HANDLE>(temp);
} while (!handleMapping->add(handle));
2004-05-03 01:06:37 +02:00
handleMappingLock.endWrite();
2004-05-03 03:53:24 +02:00
} catch(const std::exception&) {
2004-05-03 01:06:37 +02:00
// Handle out-of-memory conditions
handleMappingLock.endWrite();
free_block(handle);
handle = NULL;
}
}
return handle;
}
why_hndl* WHY_translate_handle(FB_API_HANDLE handle) {
Firebird::ReadLockGuard sync(handleMappingLock);
2004-05-03 03:53:24 +02:00
if (!handleMapping) return NULL;
HandleMapping::Accessor accessor(handleMapping);
2004-05-03 01:06:37 +02:00
if (accessor.locate(handle))
return accessor.current();
return NULL;
}
void WHY_free_handle(FB_API_HANDLE handle) {
Firebird::WriteLockGuard sync(handleMappingLock);
2004-05-03 03:53:24 +02:00
// Silently ignore bad handles for now
if (handleMapping && handleMapping->locate(handle)) {
why_hndl* temp = handleMapping->current();
handleMapping->fastRemove();
2004-05-03 01:06:37 +02:00
free_block(temp);
}
}
inline static WHY_HNDL allocate_handle(int implementation, why_hndl* h, int handle_type)
{
2004-05-03 01:06:37 +02:00
WHY_HNDL handle = WHY_alloc_handle(implementation, handle_type);
handle->handle.h_why = h;
return handle;
}
2004-05-03 01:06:37 +02:00
inline static WHY_HNDL allocate_handle(int implementation, dsql_req* h, int handle_type)
{
2004-05-03 01:06:37 +02:00
WHY_HNDL handle = WHY_alloc_handle(implementation, handle_type);
handle->handle.h_dsql = h;
return handle;
}
2004-05-03 01:06:37 +02:00
inline static WHY_HNDL allocate_handle(int implementation, Jrd::jrd_tra* h, int handle_type)
{
2004-05-03 01:06:37 +02:00
WHY_HNDL handle = WHY_alloc_handle(implementation, handle_type);
handle->handle.h_tra = h;
return handle;
}
// CVC: Just don't get the idea of private functions with C linkage.
// Probably a finer grain would be better here.
// There's an exported variable several lines below.
2003-04-10 08:49:16 +02:00
static ISC_STATUS bad_handle(ISC_STATUS *, ISC_STATUS);
#ifdef DEV_BUILD
2004-05-09 07:48:33 +02:00
static void check_status_vector(const ISC_STATUS*, ISC_STATUS);
#endif
2004-05-09 07:48:33 +02:00
static ISC_STATUS error(const ISC_STATUS*, const ISC_STATUS*);
static ISC_STATUS error2(const ISC_STATUS*, const ISC_STATUS*);
2003-12-22 11:00:59 +01:00
static void event_ast(void*, USHORT, const UCHAR*);
2004-02-02 12:02:12 +01:00
static void exit_handler(event_t*);
static WHY_TRA find_transaction(WHY_DBB, WHY_TRA);
2003-04-10 08:49:16 +02:00
static int get_database_info(ISC_STATUS *, WHY_TRA, TEXT **);
2003-02-13 11:11:35 +01:00
static const PTR get_entrypoint(int, int);
static SCHAR *get_sqlda_buffer(SCHAR *, USHORT, XSQLDA *, USHORT, USHORT *);
2003-04-10 08:49:16 +02:00
static ISC_STATUS get_transaction_info(ISC_STATUS *, WHY_TRA, TEXT **);
2004-05-03 01:06:37 +02:00
static void iterative_sql_info(ISC_STATUS *, FB_API_HANDLE*, SSHORT, const SCHAR *, SSHORT,
SCHAR *, USHORT, XSQLDA *);
2004-05-03 01:06:37 +02:00
static ISC_STATUS open_blob(ISC_STATUS*, FB_API_HANDLE*, FB_API_HANDLE*, FB_API_HANDLE*, SLONG*,
USHORT, const UCHAR*, SSHORT, SSHORT);
2003-04-10 08:49:16 +02:00
static ISC_STATUS prepare(ISC_STATUS *, WHY_TRA);
2004-02-02 12:02:12 +01:00
static void release_dsql_support(sqlda_sup*);
static void release_handle(WHY_HNDL);
2003-04-10 08:49:16 +02:00
static void save_error_string(ISC_STATUS *);
static void subsystem_enter(void);
static void subsystem_exit(void);
#ifndef REQUESTER
static event_t why_event[1];
static SSHORT why_initialized = 0;
#endif
static SLONG why_enabled = 0;
/* subsystem_usage is used to count how many active ATTACHMENTs are
* running though the why valve. For the first active attachment
* request we reset the InterBase FPE handler.
* This counter is incremented for each ATTACH DATABASE, ATTACH SERVER,
* or CREATE DATABASE. This counter is decremented for each
* DETACH DATABASE, DETACH SERVER, or DROP DATABASE.
*
* A client-only API call, isc_reset_fpe() also controls the re-setting of
* the FPE handler.
* isc_reset_fpe (0); (default)
* Initialize the FPE handler the first time the gds
* library is made active.
* isc_reset_fpe (1);
* Initialize the FPE handler the NEXT time an API call is
* invoked.
* isc_reset_fpe (2);
* Revert to InterBase pre-V4.1.0 behavior, reset the FPE
* handler on every API call.
*/
static const USHORT FPE_RESET_INIT_ONLY = 0x0; /* Don't reset FPE after init */
static const USHORT FPE_RESET_NEXT_API_CALL = 0x1; /* Reset FPE on next gds call */
static const USHORT FPE_RESET_ALL_API_CALL = 0x2; /* Reset FPE on all gds call */
#if !(defined REQUESTER || defined SUPERCLIENT || defined SUPERSERVER)
extern ULONG isc_enter_count;
static ULONG subsystem_usage = 0;
static USHORT subsystem_FPE_reset = FPE_RESET_INIT_ONLY;
#define SUBSYSTEM_USAGE_INCR subsystem_usage++
#define SUBSYSTEM_USAGE_DECR subsystem_usage--
#else
#define SUBSYSTEM_USAGE_INCR /* nothing */
#define SUBSYSTEM_USAGE_DECR /* nothing */
#endif
#ifdef UNIX
static TEXT marker_failures[1024];
static TEXT *marker_failures_ptr = marker_failures;
#endif
/*
* Global array to store string from the status vector in
* save_error_string.
*/
const int MAXERRORSTRINGLENGTH = 250;
static TEXT glbstr1[MAXERRORSTRINGLENGTH];
2003-02-13 11:11:35 +01:00
static const TEXT glbunknown[10] = "<unknown>";
#ifdef VMS
#define CALL(proc, handle) (*get_entrypoint(proc, handle))
#else
#define CALL(proc, handle) (get_entrypoint(proc, handle))
#endif
#define GDS_ATTACH_DATABASE isc_attach_database
#define GDS_BLOB_INFO isc_blob_info
#define GDS_CANCEL_BLOB isc_cancel_blob
#define GDS_CANCEL_EVENTS isc_cancel_events
#define GDS_CANCEL_OPERATION gds__cancel_operation
#define GDS_CLOSE_BLOB isc_close_blob
#define GDS_COMMIT isc_commit_transaction
#define GDS_COMMIT_RETAINING isc_commit_retaining
#define GDS_COMPILE isc_compile_request
#define GDS_COMPILE2 isc_compile_request2
#define GDS_CREATE_BLOB isc_create_blob
#define GDS_CREATE_BLOB2 isc_create_blob2
#define GDS_CREATE_DATABASE isc_create_database
#define GDS_DATABASE_INFO isc_database_info
#define GDS_DDL isc_ddl
#define GDS_DETACH isc_detach_database
#define GDS_DROP_DATABASE isc_drop_database
//#define GDS_EVENT_WAIT gds__event_wait
#define GDS_INTL_FUNCTION gds__intl_function
#define GDS_GET_SEGMENT isc_get_segment
#define GDS_GET_SLICE isc_get_slice
#define GDS_OPEN_BLOB isc_open_blob
#define GDS_OPEN_BLOB2 isc_open_blob2
#define GDS_PREPARE isc_prepare_transaction
#define GDS_PREPARE2 isc_prepare_transaction2
#define GDS_PUT_SEGMENT isc_put_segment
#define GDS_PUT_SLICE isc_put_slice
#define GDS_QUE_EVENTS isc_que_events
#define GDS_RECONNECT isc_reconnect_transaction
#define GDS_RECEIVE isc_receive
#ifdef SCROLLABLE_CURSORS
#define GDS_RECEIVE2 isc_receive2
#endif
#define GDS_RELEASE_REQUEST isc_release_request
#define GDS_REQUEST_INFO isc_request_info
#define GDS_ROLLBACK isc_rollback_transaction
#define GDS_ROLLBACK_RETAINING isc_rollback_retaining
#define GDS_SEEK_BLOB isc_seek_blob
#define GDS_SEND isc_send
#define GDS_START_AND_SEND isc_start_and_send
#define GDS_START isc_start_request
#define GDS_START_MULTIPLE isc_start_multiple
#define GDS_START_TRANSACTION isc_start_transaction
#define GDS_TRANSACT_REQUEST isc_transact_request
#define GDS_TRANSACTION_INFO isc_transaction_info
#define GDS_UNWIND isc_unwind_request
#define GDS_DSQL_ALLOCATE isc_dsql_allocate_statement
#define GDS_DSQL_ALLOC isc_dsql_alloc_statement
#define GDS_DSQL_ALLOC2 isc_dsql_alloc_statement2
#define GDS_DSQL_EXECUTE isc_dsql_execute
#define GDS_DSQL_EXECUTE2 isc_dsql_execute2
#define GDS_DSQL_EXECUTE_M isc_dsql_execute_m
#define GDS_DSQL_EXECUTE2_M isc_dsql_execute2_m
#define GDS_DSQL_EXECUTE_IMMED isc_dsql_execute_immediate
#define GDS_DSQL_EXECUTE_IMM_M isc_dsql_execute_immediate_m
#define GDS_DSQL_EXEC_IMMED isc_dsql_exec_immediate
#define GDS_DSQL_EXEC_IMMED2 isc_dsql_exec_immed2
#define GDS_DSQL_EXEC_IMM_M isc_dsql_exec_immediate_m
#define GDS_DSQL_EXEC_IMM2_M isc_dsql_exec_immed2_m
#define GDS_DSQL_EXEC_IMM3_M isc_dsql_exec_immed3_m
#define GDS_DSQL_FETCH isc_dsql_fetch
#define GDS_DSQL_FETCH2 isc_dsql_fetch2
#define GDS_DSQL_FETCH_M isc_dsql_fetch_m
#define GDS_DSQL_FETCH2_M isc_dsql_fetch2_m
#define GDS_DSQL_FREE isc_dsql_free_statement
#define GDS_DSQL_INSERT isc_dsql_insert
#define GDS_DSQL_INSERT_M isc_dsql_insert_m
#define GDS_DSQL_PREPARE isc_dsql_prepare
#define GDS_DSQL_PREPARE_M isc_dsql_prepare_m
#define GDS_DSQL_SET_CURSOR isc_dsql_set_cursor_name
#define GDS_DSQL_SQL_INFO isc_dsql_sql_info
#define GDS_SERVICE_ATTACH isc_service_attach
#define GDS_SERVICE_DETACH isc_service_detach
#define GDS_SERVICE_QUERY isc_service_query
#define GDS_SERVICE_START isc_service_start
/*****************************************************
* IMPORTANT IMPORTANT IMPORTANT IMPORTANT IMPORTANT
*
* The order in which these defines appear MUST match
* the order in which the entrypoint appears in
* source/jrd/entry.h. Failure to do so will result in
* much frustration
******************************************************/
2004-05-18 18:48:26 +02:00
const int PROC_ATTACH_DATABASE = 0;
const int PROC_BLOB_INFO = 1;
const int PROC_CANCEL_BLOB = 2;
const int PROC_CLOSE_BLOB = 3;
const int PROC_COMMIT = 4;
const int PROC_COMPILE = 5;
const int PROC_CREATE_BLOB = 6;
const int PROC_CREATE_DATABASE = 7;
const int PROC_DATABASE_INFO = 8;
const int PROC_DETACH = 9;
const int PROC_GET_SEGMENT = 10;
const int PROC_OPEN_BLOB = 11;
const int PROC_PREPARE = 12;
const int PROC_PUT_SEGMENT = 13;
const int PROC_RECONNECT = 14;
const int PROC_RECEIVE = 15;
const int PROC_RELEASE_REQUEST = 16;
const int PROC_REQUEST_INFO = 17;
const int PROC_ROLLBACK = 18;
const int PROC_SEND = 19;
const int PROC_START_AND_SEND = 20;
const int PROC_START = 21;
const int PROC_START_MULTIPLE = 22;
const int PROC_START_TRANSACTION= 23;
const int PROC_TRANSACTION_INFO = 24;
const int PROC_UNWIND = 25;
const int PROC_COMMIT_RETAINING = 26;
const int PROC_QUE_EVENTS = 27;
const int PROC_CANCEL_EVENTS = 28;
const int PROC_DDL = 29;
const int PROC_OPEN_BLOB2 = 30;
const int PROC_CREATE_BLOB2 = 31;
const int PROC_GET_SLICE = 32;
const int PROC_PUT_SLICE = 33;
const int PROC_SEEK_BLOB = 34;
const int PROC_TRANSACT_REQUEST = 35;
const int PROC_DROP_DATABASE = 36;
const int PROC_DSQL_ALLOCATE = 37;
const int PROC_DSQL_EXECUTE = 38;
const int PROC_DSQL_EXECUTE2 = 39;
const int PROC_DSQL_EXEC_IMMED = 40;
const int PROC_DSQL_EXEC_IMMED2 = 41;
const int PROC_DSQL_FETCH = 42;
const int PROC_DSQL_FREE = 43;
const int PROC_DSQL_INSERT = 44;
const int PROC_DSQL_PREPARE = 45;
const int PROC_DSQL_SET_CURSOR = 46;
const int PROC_DSQL_SQL_INFO = 47;
const int PROC_SERVICE_ATTACH = 48;
const int PROC_SERVICE_DETACH = 49;
const int PROC_SERVICE_QUERY = 50;
const int PROC_SERVICE_START = 51;
const int PROC_ROLLBACK_RETAINING = 52;
const int PROC_CANCEL_OPERATION = 53;
const int PROC_INTL_FUNCTION = 54;
2004-05-18 18:48:26 +02:00
const int PROC_count = 55;
2004-05-24 19:31:47 +02:00
struct ENTRY
{
const TEXT* name;
PTR address;
2004-05-24 19:31:47 +02:00
};
2004-05-24 19:31:47 +02:00
struct IMAGE
{
const TEXT* name;
TEXT path[MAXPATHLEN]; // avoid problems with code changing literals.
2004-05-24 19:31:47 +02:00
};
/* Define complicated table for multi-subsystem world */
2003-11-06 04:03:36 +01:00
extern "C" {
static ISC_STATUS no_entrypoint(ISC_STATUS * user_status, ...);
#ifdef VMS
#define RDB
#endif
#ifdef SUPERCLIENT
2003-11-07 14:25:53 +01:00
#define ENTRYPOINT(gen,cur,bridge,rem,os2_rem,csi,rdb,pipe,bridge_pipe,win,winipi) ISC_STATUS rem(ISC_STATUS * user_status, ...);
#else
2003-11-07 14:25:53 +01:00
#define ENTRYPOINT(gen,cur,bridge,rem,os2_rem,csi,rdb,pipe,bridge_pipe,win,winipi) ISC_STATUS rem(ISC_STATUS * user_status, ...), cur(ISC_STATUS * user_status, ...);
#endif
#include "../jrd/entry.h"
#ifdef RDB
2003-11-07 14:25:53 +01:00
#define ENTRYPOINT(gen,cur,bridge,rem,os2_rem,csi,rdb,pipe,bridge_pipe,win,winipi) ISC_STATUS rdb(ISC_STATUS * user_status, ...);
#include "../jrd/entry.h"
#endif
2004-11-30 07:08:26 +01:00
static const IMAGE images[] =
{
{"REMINT", "REMINT"}, /* Remote */
# if !defined(REQUESTER) && !defined(SUPERCLIENT)
{"GDSSHR", "GDSSHR"}, /* Primary access method */
#endif
#ifdef RDB
{"GDSRDB", "GDSRDB"}, /* Rdb Interface */
#endif
};
#define SUBSYSTEMS sizeof (images) / (sizeof (IMAGE))
static ENTRY entrypoints[PROC_count * SUBSYSTEMS] =
{
#define ENTRYPOINT(gen,cur,bridge,rem,os2_rem,csi,rdb,pipe,bridge_pipe,win,winipi) {NULL, rem},
#include "../jrd/entry.h"
# if !defined(REQUESTER) && !defined(SUPERCLIENT)
#define ENTRYPOINT(gen,cur,bridge,rem,os2_rem,csi,rdb,pipe,bridge_pipe,win,winipi) {NULL, cur},
#include "../jrd/entry.h"
#endif
#ifdef RDB
#define ENTRYPOINT(gen,cur,bridge,rem,os2_rem,csi,rdb,pipe,bridge_pipe,win,winipi) {NULL, rdb},
#include "../jrd/entry.h"
#endif
};
#ifndef SUPERCLIENT
2003-02-13 11:11:35 +01:00
static const TEXT *generic[] = {
#define ENTRYPOINT(gen,cur,bridge,rem,os2_rem,csi,rdb,pipe,bridge_pipe,win,winipi) gen,
#include "../jrd/entry.h"
};
#endif
2003-11-06 04:03:36 +01:00
} // extern "C"
/* Information items for two phase commit */
2003-02-13 11:11:35 +01:00
static const UCHAR prepare_tr_info[] =
{
2003-11-08 17:40:17 +01:00
isc_info_tra_id,
isc_info_end
};
/* Information items for DSQL prepare */
2003-02-13 11:11:35 +01:00
static const SCHAR sql_prepare_info[] =
{
2003-11-08 17:40:17 +01:00
isc_info_sql_select,
isc_info_sql_describe_vars,
isc_info_sql_sqlda_seq,
isc_info_sql_type,
isc_info_sql_sub_type,
isc_info_sql_scale,
isc_info_sql_length,
isc_info_sql_field,
isc_info_sql_relation,
isc_info_sql_owner,
isc_info_sql_alias,
isc_info_sql_describe_end
};
/* Information items for SQL info */
2003-02-13 11:11:35 +01:00
static const SCHAR describe_select_info[] =
{
2003-11-08 17:40:17 +01:00
isc_info_sql_select,
isc_info_sql_describe_vars,
isc_info_sql_sqlda_seq,
isc_info_sql_type,
isc_info_sql_sub_type,
isc_info_sql_scale,
isc_info_sql_length,
isc_info_sql_field,
isc_info_sql_relation,
isc_info_sql_owner,
isc_info_sql_alias,
isc_info_sql_describe_end
};
2003-02-13 11:11:35 +01:00
static const SCHAR describe_bind_info[] =
{
2003-11-08 17:40:17 +01:00
isc_info_sql_bind,
isc_info_sql_describe_vars,
isc_info_sql_sqlda_seq,
isc_info_sql_type,
isc_info_sql_sub_type,
isc_info_sql_scale,
isc_info_sql_length,
isc_info_sql_field,
isc_info_sql_relation,
isc_info_sql_owner,
isc_info_sql_alias,
isc_info_sql_describe_end
};
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE GDS_ATTACH_DATABASE(ISC_STATUS* user_status,
SSHORT file_length,
const TEXT* file_name,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* public_handle,
SSHORT dpb_length,
const SCHAR* dpb)
{
/**************************************
*
* g d s _ $ a t t a c h _ d a t a b a s e
*
**************************************
*
* Functional description
* Attach a database through the first subsystem
* that recognizes it.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status, *ptr;
ISC_STATUS_ARRAY local, temp;
USHORT n, length, org_length, temp_length;
WHY_DBB database;
TEXT expanded_filename[MAXPATHLEN], temp_filebuf[MAXPATHLEN];
#if defined(UNIX)
TEXT single_user[5];
#endif
GET_STATUS;
2004-05-03 01:06:37 +02:00
NULL_CHECK(public_handle, isc_bad_db_handle, HANDLE_database);
if (!file_name)
{
status[0] = isc_arg_gds;
status[1] = isc_bad_db_format;
status[2] = isc_arg_string;
2003-04-10 08:49:16 +02:00
status[3] = (ISC_STATUS) "";
status[4] = isc_arg_end;
return error2(status, local);
}
if (dpb_length > 0 && !dpb)
{
status[0] = isc_arg_gds;
status[1] = isc_bad_dpb_form;
status[2] = isc_arg_end;
return error2(status, local);
}
#if defined (SERVER_SHUTDOWN) && !defined (SUPERCLIENT) && !defined (REQUESTER)
if (shutdown_flag)
{
status[0] = isc_arg_gds;
status[1] = isc_shutwarn;
status[2] = isc_arg_end;
return error2(status, local);
}
#endif /* SERVER_SHUTDOWN && !SUPERCLIENT && !REQUESTER */
subsystem_enter();
SUBSYSTEM_USAGE_INCR;
TEXT* temp_filename = temp_filebuf;
ptr = status;
org_length = file_length;
if (org_length)
{
const TEXT* p = file_name + org_length - 1;
while (*p == ' ')
{
--p;
}
org_length = p - file_name + 1;
}
/* copy the file name to a temp buffer, since some of the following
utilities can modify it */
temp_length = org_length ? org_length : strlen(file_name);
memcpy(temp_filename, file_name, temp_length);
temp_filename[temp_length] = '\0';
if (!ISC_check_if_remote(temp_filename, true))
{
Firebird::PathName database;
if (ResolveDatabaseAlias(temp_filename, database))
{
ISC_expand_filename(database, false);
strcpy(expanded_filename, database.c_str());
}
else if (isc_set_path(temp_filename, org_length, expanded_filename))
{
temp_filename = expanded_filename;
org_length = strlen(temp_filename);
}
else
{
ISC_expand_filename(temp_filename, org_length,
expanded_filename, sizeof(expanded_filename),
true);
}
}
else
{
ISC_expand_filename(temp_filename, org_length,
expanded_filename, sizeof(expanded_filename),
true);
}
Firebird::ClumpletWriter newDpb(Firebird::ClumpletReader::Tagged, MAX_DPB_SIZE,
reinterpret_cast<const UCHAR*>(dpb), dpb_length, isc_dpb_version1);
setLogin(newDpb);
for (n = 0; n < SUBSYSTEMS; n++)
{
if (why_enabled && !(why_enabled & (1 << n)))
{
continue;
}
2004-05-03 01:06:37 +02:00
WHY_ATT handle = NULL;
if (!CALL(PROC_ATTACH_DATABASE, n) (ptr, org_length, temp_filename,
&handle, newDpb.getBufferLength(),
reinterpret_cast<const char*>(newDpb.getBuffer()),
expanded_filename))
{
length = strlen(expanded_filename);
2004-05-03 01:06:37 +02:00
database = allocate_handle(n, handle, HANDLE_database);
if (database)
{
database->db_path = (TEXT*) alloc((SLONG) (length + 1));
}
if (!database || !database->db_path)
{
/* No memory. Make a half-hearted to detach and get out. */
if (database)
release_handle(database);
CALL(PROC_DETACH, n) (ptr, handle);
status[0] = isc_arg_gds;
status[1] = isc_virmemexh;
status[2] = isc_arg_end;
break;
}
2004-05-03 01:06:37 +02:00
*public_handle = database->public_handle;
TEXT* p = database->db_path;
memcpy(p, expanded_filename, length);
p[length] = 0;
database->cleanup = NULL;
status[0] = isc_arg_gds;
status[1] = 0;
/* Check to make sure that status[2] is not a warning. If it is, then
* the rest of the vector should be untouched. If it is not, then make
* this element isc_arg_end
*
* This cleanup is done to remove any erroneous errors from trying multiple
* protocols for a database attach
*/
if (status[2] != isc_arg_warning) {
status[2] = isc_arg_end;
}
subsystem_exit();
CHECK_STATUS_SUCCESS(status);
return status[1];
}
if (ptr[1] != isc_unavailable) {
ptr = temp;
}
}
SUBSYSTEM_USAGE_DECR;
return error(status, local);
}
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE GDS_BLOB_INFO(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* blob_handle,
SSHORT item_length,
const SCHAR* items,
SSHORT buffer_length,
SCHAR* buffer)
{
/**************************************
*
* g d s _ $ b l o b _ i n f o
*
**************************************
*
* Functional description
* Provide information on blob object.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS_ARRAY local;
ISC_STATUS *status;
GET_STATUS;
2004-05-03 01:06:37 +02:00
const why_hndl* blob;
TRANSLATE_HANDLE(*blob_handle, blob, HANDLE_blob, isc_bad_segstr_handle);
subsystem_enter();
CALL(PROC_BLOB_INFO, blob->implementation) (status, &blob->handle,
item_length, items,
buffer_length, buffer);
if (status[1])
return error(status, local);
RETURN_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_CANCEL_BLOB(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * blob_handle)
{
/**************************************
*
* g d s _ $ c a n c e l _ b l o b
*
**************************************
*
* Functional description
* Abort a partially completed blob.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_BLB blob, *ptr;
WHY_ATT dbb;
if (!*blob_handle) {
if (user_status) {
user_status[0] = isc_arg_gds;
user_status[1] = 0;
user_status[2] = isc_arg_end;
CHECK_STATUS_SUCCESS(user_status);
}
return FB_SUCCESS;
}
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*blob_handle, blob, HANDLE_blob, isc_bad_segstr_handle);
subsystem_enter();
if (CALL(PROC_CANCEL_BLOB, blob->implementation) (status, &blob->handle))
return error(status, local);
/* Get rid of connections to database */
dbb = blob->parent;
for (ptr = &dbb->blobs; *ptr; ptr = &(*ptr)->next)
if (*ptr == blob) {
*ptr = blob->next;
break;
}
release_handle(blob);
2004-05-03 01:06:37 +02:00
*blob_handle = 0;
RETURN_SUCCESS;
}
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE GDS_CANCEL_EVENTS(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * handle,
SLONG * id)
{
/**************************************
*
* g d s _ $ c a n c e l _ e v e n t s
*
**************************************
*
* Functional description
* Try to cancel an event.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_ATT database;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*handle, database, HANDLE_database, isc_bad_db_handle);
subsystem_enter();
if (CALL(PROC_CANCEL_EVENTS, database->implementation) (status,
&database->handle,
id))
return error(status, local);
RETURN_SUCCESS;
}
#ifdef CANCEL_OPERATION
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE GDS_CANCEL_OPERATION(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * handle,
USHORT option)
{
/**************************************
*
* g d s _ $ c a n c e l _ o p e r a t i o n
*
**************************************
*
* Functional description
* Try to cancel an operation.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_ATT database;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*handle, database, HANDLE_database, isc_bad_db_handle);
subsystem_enter();
if (CALL(PROC_CANCEL_OPERATION, database->implementation) (status,
&database->
handle,
option))
{
return error(status, local);
}
RETURN_SUCCESS;
}
#endif
ISC_STATUS API_ROUTINE GDS_CLOSE_BLOB(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * blob_handle)
{
/**************************************
*
* g d s _ $ c l o s e _ b l o b
*
**************************************
*
* Functional description
* Abort a partially completed blob.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_BLB blob, *ptr;
WHY_ATT dbb;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*blob_handle, blob, HANDLE_blob, isc_bad_segstr_handle);
subsystem_enter();
CALL(PROC_CLOSE_BLOB, blob->implementation) (status, &blob->handle);
if (status[1])
return error(status, local);
/* Get rid of connections to database */
dbb = blob->parent;
for (ptr = &dbb->blobs; *ptr; ptr = &(*ptr)->next)
if (*ptr == blob) {
*ptr = blob->next;
break;
}
release_handle(blob);
2004-05-03 01:06:37 +02:00
*blob_handle = 0;
RETURN_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_COMMIT(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * tra_handle)
{
/**************************************
*
* g d s _ $ c o m m i t
*
**************************************
*
* Functional description
* Commit a transaction.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_TRA transaction, sub;
//CLEAN clean;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*tra_handle, transaction, HANDLE_transaction, isc_bad_trans_handle);
subsystem_enter();
/* Handle single transaction case */
if (transaction->implementation != SUBSYSTEMS) {
if (CALL(PROC_COMMIT, transaction->implementation) (status,
&transaction->
handle))
{
return error(status, local);
}
}
else {
/* Handle two phase commit. Start by putting everybody into
limbo. If anybody fails, punt */
if (!(transaction->flags & HANDLE_TRANSACTION_limbo))
if (prepare(status, transaction))
return error(status, local);
/* Everybody is in limbo, now commit everybody.
In theory, this can't fail */
for (sub = transaction->next; sub; sub = sub->next)
if (CALL(PROC_COMMIT, sub->implementation) (status, &sub->handle))
return error(status, local);
}
subsystem_exit();
/* Call the associated cleanup handlers */
2004-02-28 20:32:46 +01:00
WHY_cleanup_transaction(transaction);
//while (clean = transaction->cleanup) {
// transaction->cleanup = clean->clean_next;
// clean->TransactionRoutine(transaction, clean->clean_arg);
// free_block(clean);
//}
while (sub = transaction) {
transaction = sub->next;
release_handle(sub);
}
2004-05-03 01:06:37 +02:00
*tra_handle = 0;
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE GDS_COMMIT_RETAINING(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * tra_handle)
{
/**************************************
*
* g d s _ $ c o m m i t _ r e t a i n i n g
*
**************************************
*
* Functional description
* Do a commit retaining.
*
* N.B., the transaction cleanup handlers are NOT
* called here since, conceptually, the transaction
* is still running.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_TRA transaction, sub;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*tra_handle, transaction, HANDLE_transaction, isc_bad_trans_handle);
subsystem_enter();
for (sub = transaction; sub; sub = sub->next)
if (sub->implementation != SUBSYSTEMS &&
CALL(PROC_COMMIT_RETAINING, sub->implementation) (status,
&sub->handle))
return error(status, local);
transaction->flags |= HANDLE_TRANSACTION_limbo;
RETURN_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_COMPILE(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* req_handle,
USHORT blr_length,
const SCHAR* blr)
{
/**************************************
*
* g d s _ $ c o m p i l e
*
**************************************
*
* Functional description
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_ATT dbb;
2004-05-03 01:06:37 +02:00
WHY_REQ request = NULL;
GET_STATUS;
NULL_CHECK(req_handle, isc_bad_req_handle, HANDLE_request);
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*db_handle, dbb, HANDLE_database, isc_bad_db_handle);
subsystem_enter();
if (CALL(PROC_COMPILE, dbb->implementation) (status, &dbb->handle,
2004-05-03 01:06:37 +02:00
&request, blr_length,
blr))
{
return error(status, local);
}
2004-05-03 01:06:37 +02:00
request = allocate_handle(dbb->implementation, request, HANDLE_request);
if (!request) {
/* No memory. Make a half-hearted attempt to release request. */
2004-05-03 01:06:37 +02:00
CALL(PROC_RELEASE_REQUEST, dbb->implementation) (status, request->handle.h_why);
status[0] = isc_arg_gds;
status[1] = isc_virmemexh;
status[2] = isc_arg_end;
return error(status, local);
}
2004-05-03 01:06:37 +02:00
*req_handle = request->public_handle;
request->parent = dbb;
request->next = dbb->requests;
dbb->requests = request;
RETURN_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_COMPILE2(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* req_handle,
USHORT blr_length,
const SCHAR* blr)
{
/**************************************
*
* g d s _ $ c o m p i l e 2
*
**************************************
*
* Functional description
*
**************************************/
if (GDS_COMPILE(user_status, db_handle, req_handle, blr_length, blr))
/* Note: if user_status == NULL then GDS_COMPILE handled it */
return user_status[1];
2004-05-03 01:06:37 +02:00
WHY_REQ request = WHY_translate_handle(*req_handle);
request->user_handle = req_handle;
return FB_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_CREATE_BLOB(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* tra_handle,
FB_API_HANDLE* blob_handle,
SLONG* blob_id)
{
/**************************************
*
* g d s _ $ c r e a t e _ b l o b
*
**************************************
*
* Functional description
* Open an existing blob.
*
**************************************/
return open_blob(user_status, db_handle, tra_handle, blob_handle, blob_id,
0, 0, PROC_CREATE_BLOB, PROC_CREATE_BLOB2);
}
ISC_STATUS API_ROUTINE GDS_CREATE_BLOB2(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* tra_handle,
FB_API_HANDLE* blob_handle,
SLONG* blob_id,
SSHORT bpb_length,
const UCHAR* bpb)
{
/**************************************
*
* g d s _ $ c r e a t e _ b l o b 2
*
**************************************
*
* Functional description
* Open an existing blob.
*
**************************************/
return open_blob(user_status, db_handle, tra_handle, blob_handle, blob_id,
bpb_length, bpb, PROC_CREATE_BLOB,
PROC_CREATE_BLOB2);
}
ISC_STATUS API_ROUTINE GDS_CREATE_DATABASE(ISC_STATUS* user_status,
USHORT file_length,
const TEXT* file_name,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* public_handle,
SSHORT dpb_length,
const UCHAR* dpb,
USHORT db_type)
{
/**************************************
*
* g d s _ $ c r e a t e _ d a t a b a s e
*
**************************************
*
* Functional description
* Create a nice, squeaky clean database, uncorrupted by user data.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status, *ptr;
ISC_STATUS_ARRAY local, temp;
USHORT n, length, org_length, temp_length;
WHY_DBB database;
TEXT expanded_filename[MAXPATHLEN], temp_filebuf[MAXPATHLEN];
#ifdef UNIX
TEXT single_user[5];
#endif
GET_STATUS;
2004-05-03 01:06:37 +02:00
NULL_CHECK(public_handle, isc_bad_db_handle, HANDLE_database);
if (!file_name)
{
status[0] = isc_arg_gds;
status[1] = isc_bad_db_format;
status[2] = isc_arg_string;
2003-04-10 08:49:16 +02:00
status[3] = (ISC_STATUS) "";
status[4] = isc_arg_end;
return error2(status, local);
}
if (dpb_length > 0 && !dpb)
{
status[0] = isc_arg_gds;
status[1] = isc_bad_dpb_form;
status[2] = isc_arg_end;
return error2(status, local);
}
#if defined (SERVER_SHUTDOWN) && !defined (SUPERCLIENT) && !defined (REQUESTER)
if (shutdown_flag)
{
status[0] = isc_arg_gds;
status[1] = isc_shutwarn;
status[2] = isc_arg_end;
return error2(status, local);
}
#endif /* SERVER_SHUTDOWN && !SUPERCLIENT && !REQUESTER */
subsystem_enter();
SUBSYSTEM_USAGE_INCR;
TEXT* temp_filename = temp_filebuf;
ptr = status;
org_length = file_length;
if (org_length)
{
const TEXT* p = file_name + org_length - 1;
while (*p == ' ')
--p;
org_length = p - file_name + 1;
}
/* copy the file name to a temp buffer, since some of the following
utilities can modify it */
if (org_length)
temp_length = org_length;
else
temp_length = strlen(file_name);
memcpy(temp_filename, file_name, temp_length);
temp_filename[temp_length] = '\0';
if (!ISC_check_if_remote(temp_filename, true))
{
Firebird::PathName database;
if (ResolveDatabaseAlias(temp_filename, database))
{
ISC_expand_filename(database, false);
strcpy(expanded_filename, database.c_str());
}
else if (isc_set_path(temp_filename, org_length, expanded_filename))
{
temp_filename = expanded_filename;
org_length = strlen(temp_filename);
}
else
{
ISC_expand_filename(temp_filename, org_length,
expanded_filename, sizeof(expanded_filename),
true);
}
}
else
{
ISC_expand_filename(temp_filename, org_length,
expanded_filename, sizeof(expanded_filename),
true);
}
Firebird::ClumpletWriter newDpb(Firebird::ClumpletReader::Tagged, MAX_DPB_SIZE,
reinterpret_cast<const UCHAR*>(dpb), dpb_length, isc_dpb_version1);
setLogin(newDpb);
for (n = 0; n < SUBSYSTEMS; n++)
{
if (why_enabled && !(why_enabled & (1 << n)))
continue;
2004-05-03 01:06:37 +02:00
WHY_ATT handle = NULL;
if (!CALL(PROC_CREATE_DATABASE, n) (ptr, org_length, temp_filename,
&handle, newDpb.getBufferLength(),
reinterpret_cast<const char*>(newDpb.getBuffer()),
0, expanded_filename))
{
#ifdef WIN_NT
/* Now we can expand, the file exists. */
length = ISC_expand_filename (temp_filename, org_length,
expanded_filename, sizeof(expanded_filename), true);
#else
length = org_length;
if (!length)
{
length = strlen(temp_filename);
}
#endif
2004-05-03 01:06:37 +02:00
database = allocate_handle(n, handle, HANDLE_database);
if (database)
{
database->db_path = (TEXT *) alloc((SLONG) (length + 1));
}
if (!database || !database->db_path)
{
/* No memory. Make a half-hearted to drop database. The
database was successfully created but the user wouldn't
be able to tell. */
if (database)
release_handle(database);
CALL(PROC_DROP_DATABASE, n) (ptr, handle);
status[0] = isc_arg_gds;
status[1] = isc_virmemexh;
status[2] = isc_arg_end;
break;
}
2003-11-04 00:59:24 +01:00
fb_assert(database);
fb_assert(database->db_path);
2004-05-03 01:06:37 +02:00
*public_handle = database->public_handle;
TEXT* p = database->db_path;
#ifdef WIN_NT
/* for (q = (*handle)->dbb_filename->str_data; length; length--) */
memcpy(p, expanded_filename, length);
#else
memcpy(p, temp_filename, length);
#endif
p[length] = 0;
database->cleanup = NULL;
status[0] = isc_arg_gds;
status[1] = 0;
if (status[2] != isc_arg_warning)
status[2] = isc_arg_end;
subsystem_exit();
CHECK_STATUS_SUCCESS(status);
return status[1];
}
if (ptr[1] != isc_unavailable)
ptr = temp;
}
SUBSYSTEM_USAGE_DECR;
return error(status, local);
}
ISC_STATUS API_ROUTINE isc_database_cleanup(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * handle,
DatabaseCleanupRoutine * routine,
void* arg)
{
/**************************************
*
* g d s _ $ d a t a b a s e _ c l e a n u p
*
**************************************
*
* Functional description
* Register a database specific cleanup handler.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_ATT database;
CLEAN clean;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*handle, database, HANDLE_database, isc_bad_db_handle);
if (!(clean = (CLEAN) alloc((SLONG) sizeof(struct clean)))) {
status[0] = isc_arg_gds;
status[1] = isc_virmemexh;
status[2] = isc_arg_end;
return error2(status, local);
}
clean->clean_next = database->cleanup;
database->cleanup = clean;
clean->DatabaseRoutine = routine;
clean->clean_arg = arg;
status[0] = isc_arg_gds;
status[1] = 0;
status[2] = isc_arg_end;
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_DATABASE_INFO(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* handle,
SSHORT item_length,
const SCHAR* items,
SSHORT buffer_length,
SCHAR* buffer)
{
/**************************************
*
* g d s _ $ d a t a b a s e _ i n f o
*
**************************************
*
* Functional description
* Provide information on database object.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_ATT database;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*handle, database, HANDLE_database, isc_bad_db_handle);
subsystem_enter();
if (CALL(PROC_DATABASE_INFO, database->implementation) (status,
&database->handle,
item_length,
items,
buffer_length,
buffer))
{
return error(status, local);
}
RETURN_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_DDL(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* tra_handle,
2003-11-06 04:03:36 +01:00
SSHORT length,
const UCHAR* ddl)
{
/**************************************
*
* g d s _ $ d d l
*
**************************************
*
* Functional description
* Do meta-data update.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_ATT database;
WHY_TRA transaction;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*db_handle, database, HANDLE_database, isc_bad_db_handle);
FIND_TRANSACTION(*tra_handle, transaction, database, isc_bad_trans_handle);
subsystem_enter();
if (get_entrypoint(PROC_DDL, database->implementation) != no_entrypoint)
{
if (!CALL(PROC_DDL, database->implementation) (status,
&database->handle,
&transaction->handle,
length, ddl))
{
RETURN_SUCCESS;
}
if (status[1] != isc_unavailable)
{
return error(status, local);
}
}
subsystem_exit();
/* Assume that we won't find an entrypoint to call. */
no_entrypoint(status);
#ifndef SUPERCLIENT
PTR entrypoint;
2004-11-30 07:08:26 +01:00
const TEXT* image = images[database->implementation].path;
if (image != NULL &&
2004-11-30 07:08:26 +01:00
((entrypoint = (PTR) Jrd::Module::lookup(image, "DYN_ddl")) !=
NULL ||
FALSE) &&
2003-12-22 11:00:59 +01:00
!((*entrypoint) (status, db_handle, tra_handle, length, ddl)))
{
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
#endif
return error2(status, local);
}
ISC_STATUS API_ROUTINE GDS_DETACH(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * handle)
{
/**************************************
*
* g d s _ $ d e t a c h
*
**************************************
*
* Functional description
* Close down a database.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_REQ request;
WHY_STMT statement;
WHY_BLB blob;
//CLEAN clean;
GET_STATUS;
2004-05-03 01:06:37 +02:00
WHY_ATT dbb;
#ifdef WIN_NT
/* This code was added to fix an IDAPI problem where our DLL exit handler
* was getting called before theirs. Our exit handler frees the attachment
* but does not NULL the handle. Their exit handler trys to detach, causing
* a GPF.
* We should check with IDAPI periodically to see if we still need this.
*/
2004-05-03 03:53:24 +02:00
// 02-May-2004, Nickolay Samofatov. We really need to check if BDE still
// needs this junk and how exactly problem looks like. I disable the code
// for now because handleMapping provides means for reliable detection of
// bad handles, no tricks are needed.
//
// if (IsBadReadPtr(handle, sizeof(FB_API_HANDLE)))
// return bad_handle(user_status, isc_bad_db_handle);
#endif /* WIN_NT */
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*handle, dbb, HANDLE_database, isc_bad_db_handle);
2006-03-07 10:23:44 +01:00
#if defined(SUPERSERVER) && !defined(EMBEDDED)
/* Drop all DSQL statements to reclaim DSQL memory pools. */
while (statement = dbb->statements) {
2004-05-03 01:06:37 +02:00
FB_API_HANDLE temp_handle;
if (!statement->user_handle) {
temp_handle = statement->public_handle;
statement->user_handle = &temp_handle;
}
if (GDS_DSQL_FREE(status, statement->user_handle, DSQL_drop))
return error2(status, local);
}
#endif
subsystem_enter();
if (CALL(PROC_DETACH, dbb->implementation) (status, &dbb->handle))
return error(status, local);
/* Release associated request handles */
if (dbb->db_path)
free_block(dbb->db_path);
while (request = dbb->requests) {
dbb->requests = request->next;
if (request->user_handle)
2004-05-03 01:06:37 +02:00
*request->user_handle = 0;
release_handle(request);
}
#ifndef SUPERSERVER
while (statement = dbb->statements) {
dbb->statements = statement->next;
if (statement->user_handle) {
2004-05-03 01:06:37 +02:00
*statement->user_handle = 0;
}
release_dsql_support(statement->das);
release_handle(statement);
}
#endif
while (blob = dbb->blobs) {
dbb->blobs = blob->next;
if (blob->user_handle)
2004-05-03 01:06:37 +02:00
*blob->user_handle = 0;
release_handle(blob);
}
SUBSYSTEM_USAGE_DECR;
subsystem_exit();
/* Call the associated cleanup handlers */
2004-02-28 20:32:46 +01:00
// Obviously, this code can't be replaced by WHY_cleanup_transaction!
2004-03-07 08:58:55 +01:00
// But this requires changes in database and transaction cleanup
2004-02-28 20:32:46 +01:00
// routines first parameter.
for (clean* cln = dbb->cleanup; cln; cln = dbb->cleanup)
{
dbb->cleanup = cln->clean_next;
cln->DatabaseRoutine(handle, cln->clean_arg);
free_block(cln);
}
release_handle(dbb);
2004-05-03 01:06:37 +02:00
*handle = 0;
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
int API_ROUTINE gds__disable_subsystem(TEXT * subsystem)
{
/**************************************
*
* g d s _ $ d i s a b l e _ s u b s y s t e m
*
**************************************
*
* Functional description
* Disable access to a specific subsystem. If no subsystem
* has been explicitly disabled, all are available.
*
**************************************/
2003-12-22 11:00:59 +01:00
const IMAGE* sys = images;
for (const IMAGE* const end = sys + SUBSYSTEMS; sys < end; sys++)
{
if (!strcmp(sys->name, subsystem)) {
if (!why_enabled)
why_enabled = ~why_enabled;
why_enabled &= ~(1 << (sys - images));
return TRUE;
}
}
return FALSE;
}
ISC_STATUS API_ROUTINE GDS_DROP_DATABASE(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * handle)
{
/**************************************
*
* i s c _ d r o p _ d a t a b a s e
*
**************************************
*
* Functional description
* Close down a database and then purge it.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_REQ request;
WHY_STMT statement;
WHY_BLB blob;
CLEAN clean;
GET_STATUS;
2004-05-03 01:06:37 +02:00
WHY_ATT dbb;
TRANSLATE_HANDLE(*handle, dbb, HANDLE_database, isc_bad_db_handle);
2006-03-07 10:23:44 +01:00
#if defined(SUPERSERVER) && !defined(EMBEDDED)
/* Drop all DSQL statements to reclaim DSQL memory pools. */
2004-05-03 01:06:37 +02:00
while (statement = dbb->statements) {
2004-05-03 01:06:37 +02:00
FB_API_HANDLE temp_handle;
if (!statement->user_handle) {
temp_handle = statement->public_handle;
statement->user_handle = &temp_handle;
}
if (GDS_DSQL_FREE(status, statement->user_handle, DSQL_drop))
return error2(status, local);
}
#endif
subsystem_enter();
(CALL(PROC_DROP_DATABASE, dbb->implementation) (status, &dbb->handle));
if (status[1] && (status[1] != isc_drdb_completed_with_errs))
return error(status, local);
/* Release associated request handles */
if (dbb->db_path)
free_block(dbb->db_path);
while (request = dbb->requests) {
dbb->requests = request->next;
if (request->user_handle)
2004-05-03 01:06:37 +02:00
*request->user_handle = 0;
release_handle(request);
}
#ifndef SUPERSERVER
while (statement = dbb->statements) {
dbb->statements = statement->next;
if (statement->user_handle)
2004-05-03 01:06:37 +02:00
*statement->user_handle = 0;
release_dsql_support(statement->das);
release_handle(statement);
}
#endif
while (blob = dbb->blobs) {
dbb->blobs = blob->next;
if (blob->user_handle)
2004-05-03 01:06:37 +02:00
*blob->user_handle = 0;
release_handle(blob);
}
SUBSYSTEM_USAGE_DECR;
subsystem_exit();
/* Call the associated cleanup handlers */
while ((clean = dbb->cleanup) != NULL) {
dbb->cleanup = clean->clean_next;
clean->DatabaseRoutine(handle, clean->clean_arg);
free_block(clean);
}
release_handle(dbb);
2004-05-03 01:06:37 +02:00
*handle = 0;
if (status[1])
return error2(status, local);
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE GDS_DSQL_ALLOC(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * db_handle,
FB_API_HANDLE * stmt_handle)
{
/**************************************
*
* i s c _ d s q l _ a l l o c _ s t a t e m e n t
*
**************************************/
return GDS_DSQL_ALLOCATE(user_status, db_handle, stmt_handle);
}
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE GDS_DSQL_ALLOC2(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * db_handle,
FB_API_HANDLE * stmt_handle)
{
/**************************************
*
* i s c _ d s q l _ a l l o c _ s t a t e m e n t 2
*
**************************************
*
* Functional description
* Allocate a statement handle.
*
**************************************/
if (GDS_DSQL_ALLOCATE(user_status, db_handle, stmt_handle))
return user_status[1];
2004-05-03 01:06:37 +02:00
WHY_STMT statement = WHY_translate_handle(*stmt_handle);
statement->user_handle = stmt_handle;
return FB_SUCCESS;
}
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE GDS_DSQL_ALLOCATE(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * db_handle,
FB_API_HANDLE * public_stmt_handle)
{
/**************************************
*
* i s c _ d s q l _ a l l o c a t e _ s t a t e m e n t
*
**************************************
*
* Functional description
* Allocate a statement handle.
*
**************************************/
2003-12-22 11:00:59 +01:00
ISC_STATUS* status;
2003-04-16 12:18:51 +02:00
ISC_STATUS_ARRAY local;
WHY_STMT statement;
GET_STATUS;
/* check the statement handle to make sure it's NULL and then initialize it. */
2004-05-03 01:06:37 +02:00
NULL_CHECK(public_stmt_handle, isc_bad_stmt_handle, HANDLE_statement);
WHY_ATT dbb;
TRANSLATE_HANDLE(*db_handle, dbb, HANDLE_database, isc_bad_db_handle);
/* Attempt to have the implementation which processed the database attach
process the allocate statement. This may not be feasible (e.g., the
server doesn't support remote DSQL because it's the wrong version or
something) in which case, execute the functionality locally (and hence
remotely through the original Y-valve). */
2003-12-22 11:00:59 +01:00
ISC_STATUS s = isc_unavailable;
2004-05-03 01:06:37 +02:00
WHY_STMT stmt_handle = NULL;
2003-12-22 11:00:59 +01:00
PTR entry = get_entrypoint(PROC_DSQL_ALLOCATE, dbb->implementation);
if (entry != no_entrypoint) {
subsystem_enter();
2004-05-03 01:06:37 +02:00
s = (*entry) (status, &dbb->handle, &stmt_handle);
subsystem_exit();
}
2003-12-22 11:00:59 +01:00
UCHAR flag = 0;
#ifndef NO_LOCAL_DSQL
2003-12-22 11:00:59 +01:00
dsql_req* dstatement = 0;
if (s == isc_unavailable) {
/* if the entry point didn't exist or if the routine said the server
didn't support the protocol... do it locally */
flag = HANDLE_STATEMENT_local;
subsystem_enter();
s = dsql8_allocate_statement(status, db_handle, &dstatement);
subsystem_exit();
}
#endif
if (status[1])
return error2(status, local);
statement =
#ifndef NO_LOCAL_DSQL
(flag & HANDLE_STATEMENT_local) ?
allocate_handle(dbb->implementation, dstatement, HANDLE_statement) :
#endif
2004-05-03 01:06:37 +02:00
allocate_handle(dbb->implementation, stmt_handle, HANDLE_statement);
if (!statement) {
/* No memory. Make a half-hearted attempt to drop statement. */
subsystem_enter();
#ifndef NO_LOCAL_DSQL
if (flag & HANDLE_STATEMENT_local)
dsql8_free_statement(status, &dstatement, DSQL_drop);
else
#endif
CALL(PROC_DSQL_FREE, dbb->implementation) (status, stmt_handle,
DSQL_drop);
subsystem_exit();
status[0] = isc_arg_gds;
status[1] = isc_virmemexh;
status[2] = isc_arg_end;
return error2(status, local);
}
2004-05-03 01:06:37 +02:00
*public_stmt_handle = statement->public_handle;
statement->parent = dbb;
statement->next = dbb->statements;
dbb->statements = statement;
statement->flags = flag;
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE isc_dsql_describe(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * stmt_handle,
USHORT dialect,
XSQLDA * sqlda)
{
/**************************************
*
* i s c _ d s q l _ d e s c r i b e
*
**************************************
*
* Functional description
* Describe output parameters for a prepared statement.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
USHORT buffer_len;
SCHAR *buffer, local_buffer[512];
GET_STATUS;
if (!(buffer = get_sqlda_buffer(local_buffer, sizeof(local_buffer), sqlda,
dialect, &buffer_len)))
{
status[0] = isc_arg_gds;
status[1] = isc_virmemexh;
status[2] = isc_arg_end;
return error2(status, local);
}
if (!GDS_DSQL_SQL_INFO( status,
stmt_handle,
sizeof(describe_select_info),
describe_select_info,
buffer_len,
buffer))
{
iterative_sql_info( status,
stmt_handle,
sizeof(describe_select_info),
describe_select_info,
buffer_len,
buffer,
dialect,
sqlda);
}
if (buffer != local_buffer) {
free_block(buffer);
}
if (status[1]) {
return error2(status, local);
}
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE isc_dsql_describe_bind(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * stmt_handle,
USHORT dialect,
XSQLDA * sqlda)
{
/**************************************
*
* i s c _ d s q l _ d e s c r i b e _ b i n d
*
**************************************
*
* Functional description
* Describe input parameters for a prepared statement.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
USHORT buffer_len;
SCHAR *buffer, local_buffer[512];
GET_STATUS;
if (!(buffer = get_sqlda_buffer(local_buffer, sizeof(local_buffer), sqlda,
2004-12-24 09:52:39 +01:00
dialect, &buffer_len)))
{
status[0] = isc_arg_gds;
status[1] = isc_virmemexh;
status[2] = isc_arg_end;
return error2(status, local);
}
if (!GDS_DSQL_SQL_INFO( status,
stmt_handle,
sizeof(describe_bind_info),
describe_bind_info,
buffer_len,
buffer))
{
iterative_sql_info( status,
stmt_handle,
sizeof(describe_bind_info),
describe_bind_info,
buffer_len,
buffer,
dialect,
sqlda);
}
if (buffer != local_buffer) {
free_block(buffer);
}
if (status[1]) {
return error2(status, local);
}
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_DSQL_EXECUTE(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* tra_handle,
FB_API_HANDLE* stmt_handle,
USHORT dialect,
XSQLDA* sqlda)
{
/**************************************
*
* i s c _ d s q l _ e x e c u t e
*
**************************************
*
* Functional description
* Execute a non-SELECT dynamic SQL statement.
*
**************************************/
return GDS_DSQL_EXECUTE2(user_status, tra_handle, stmt_handle, dialect,
sqlda, NULL);
}
ISC_STATUS API_ROUTINE GDS_DSQL_EXECUTE2(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* tra_handle,
FB_API_HANDLE* stmt_handle,
USHORT dialect,
XSQLDA* in_sqlda,
XSQLDA* out_sqlda)
{
/**************************************
*
* i s c _ d s q l _ e x e c u t e 2
*
**************************************
*
* Functional description
* Execute a non-SELECT dynamic SQL statement.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_STMT statement;
USHORT in_blr_length, in_msg_type, in_msg_length,
out_blr_length, out_msg_type, out_msg_length;
2004-02-02 12:02:12 +01:00
sqlda_sup* dasup;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*stmt_handle, statement, HANDLE_statement, isc_bad_stmt_handle);
if (! (dasup = statement->das))
return bad_handle(user_status, isc_unprepared_stmt);
if (UTLD_parse_sqlda(status, dasup, &in_blr_length, &in_msg_type,
&in_msg_length, dialect, in_sqlda,
DASUP_CLAUSE_bind))
{
return error2(status, local);
}
if (UTLD_parse_sqlda
(status, dasup, &out_blr_length, &out_msg_type, &out_msg_length,
dialect, out_sqlda, DASUP_CLAUSE_select))
{
return error2(status, local);
}
if (GDS_DSQL_EXECUTE2_M(status, tra_handle, stmt_handle,
in_blr_length,
dasup->dasup_clauses[DASUP_CLAUSE_bind].dasup_blr,
in_msg_type, in_msg_length,
dasup->dasup_clauses[DASUP_CLAUSE_bind].dasup_msg,
out_blr_length,
dasup->dasup_clauses[DASUP_CLAUSE_select].
dasup_blr, out_msg_type, out_msg_length,
dasup->dasup_clauses[DASUP_CLAUSE_select].
dasup_msg))
{
return error2(status, local);
}
if (UTLD_parse_sqlda(status, dasup, NULL, NULL, NULL,
dialect, out_sqlda, DASUP_CLAUSE_select))
{
return error2(status, local);
}
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_DSQL_EXECUTE_M(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* tra_handle,
FB_API_HANDLE* stmt_handle,
USHORT blr_length,
const SCHAR* blr,
USHORT msg_type,
USHORT msg_length,
SCHAR* msg)
{
/**************************************
*
* i s c _ d s q l _ e x e c u t e _ m
*
**************************************
*
* Functional description
* Execute a non-SELECT dynamic SQL statement.
*
**************************************/
return GDS_DSQL_EXECUTE2_M(user_status, tra_handle, stmt_handle,
blr_length, blr, msg_type, msg_length, msg,
0, NULL, 0, 0, NULL);
}
ISC_STATUS API_ROUTINE GDS_DSQL_EXECUTE2_M(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* tra_handle,
FB_API_HANDLE* stmt_handle,
USHORT in_blr_length,
const SCHAR* in_blr,
USHORT in_msg_type,
USHORT in_msg_length,
SCHAR* in_msg,
USHORT out_blr_length,
SCHAR* out_blr,
USHORT out_msg_type,
USHORT out_msg_length,
SCHAR* out_msg)
{
/**************************************
*
* i s c _ d s q l _ e x e c u t e 2 _ m
*
**************************************
*
* Functional description
* Execute a non-SELECT dynamic SQL statement.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_STMT statement;
2004-05-03 01:06:37 +02:00
WHY_TRA transaction = NULL, handle = NULL;
PTR entry;
//CLEAN clean;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*stmt_handle, statement, HANDLE_statement, isc_bad_stmt_handle);
if (*tra_handle)
TRANSLATE_HANDLE(*tra_handle, transaction, HANDLE_transaction, isc_bad_trans_handle);
#ifndef NO_LOCAL_DSQL
if (statement->flags & HANDLE_STATEMENT_local) {
subsystem_enter();
dsql8_execute(status, tra_handle, &statement->handle.h_dsql,
in_blr_length, in_blr, in_msg_type, in_msg_length,
in_msg, out_blr_length, out_blr, out_msg_type,
out_msg_length, out_msg);
subsystem_exit();
}
else
#endif
{
if (transaction) {
handle = find_transaction(statement->parent, transaction);
2004-05-03 01:06:37 +02:00
if (!handle)
return bad_handle (user_status, isc_bad_trans_handle);
handle = handle->handle.h_why;
}
subsystem_enter();
entry = get_entrypoint(PROC_DSQL_EXECUTE2, statement->implementation);
if (entry != no_entrypoint &&
(*entry) (status,
&handle,
&statement->handle,
in_blr_length,
in_blr,
in_msg_type,
in_msg_length,
in_msg,
out_blr_length,
out_blr,
out_msg_type,
out_msg_length, out_msg) != isc_unavailable);
else if (!out_blr_length && !out_msg_type && !out_msg_length)
CALL(PROC_DSQL_EXECUTE, statement->implementation) (status,
&handle,
&statement->
handle,
in_blr_length,
in_blr,
in_msg_type,
in_msg_length,
in_msg);
else
no_entrypoint(status);
subsystem_exit();
if (!status[1])
{
if (transaction && !handle) {
/* Call the associated cleanup handlers */
2004-02-28 20:32:46 +01:00
WHY_cleanup_transaction(transaction);
//while (clean = transaction->cleanup) {
// transaction->cleanup = clean->clean_next;
// clean->TransactionRoutine(transaction, clean->clean_arg);
// free_block(clean);
//}
release_handle(transaction);
2004-05-03 01:06:37 +02:00
*tra_handle = 0;
}
else if (!transaction && handle)
{
2004-05-03 01:06:37 +02:00
transaction = allocate_handle( statement->implementation,
handle,
HANDLE_transaction);
2004-05-03 01:06:37 +02:00
if (transaction) {
transaction->parent = statement->parent;
*tra_handle = transaction->public_handle;
}
else {
status[0] = isc_arg_gds;
status[1] = isc_virmemexh;
status[2] = isc_arg_end;
}
}
}
}
if (status[1]) {
return error2(status, local);
}
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_DSQL_EXEC_IMMED(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* tra_handle,
USHORT length,
const SCHAR* string,
USHORT dialect,
XSQLDA* sqlda)
{
/**************************************
*
* i s c _ d s q l _ e x e c _ i m m e d
*
**************************************
*
* Functional description
*
**************************************/
return GDS_DSQL_EXECUTE_IMMED(user_status,
db_handle, tra_handle, length, string,
dialect, sqlda);
}
ISC_STATUS API_ROUTINE GDS_DSQL_EXECUTE_IMMED(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* tra_handle,
USHORT length,
const SCHAR* string,
USHORT dialect, XSQLDA* sqlda)
{
/**************************************
*
* i s c _ d s q l _ e x e c u t e _ i m m e d i a t e
*
**************************************
*
* Functional description
* Prepare a statement for execution.
*
**************************************/
return GDS_DSQL_EXEC_IMMED2(user_status,
db_handle, tra_handle, length, string,
dialect, sqlda, NULL);
}
ISC_STATUS API_ROUTINE GDS_DSQL_EXEC_IMMED2(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* tra_handle,
USHORT length,
const SCHAR* string,
USHORT dialect,
XSQLDA* in_sqlda,
XSQLDA* out_sqlda)
{
/**************************************
*
* i s c _ d s q l _ e x e c _ i m m e d 2
*
**************************************
*
* Functional description
* Prepare a statement for execution.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS s, *status;
ISC_STATUS_ARRAY local;
USHORT in_blr_length, in_msg_type, in_msg_length,
out_blr_length, out_msg_type, out_msg_length;
2004-02-02 12:02:12 +01:00
sqlda_sup dasup;
GET_STATUS;
2004-05-03 01:06:37 +02:00
2004-02-02 12:02:12 +01:00
memset(&dasup, 0, sizeof(sqlda_sup));
if (UTLD_parse_sqlda(status, &dasup, &in_blr_length, &in_msg_type,
&in_msg_length, dialect, in_sqlda,
DASUP_CLAUSE_bind))
{
return error2(status, local);
}
if (UTLD_parse_sqlda
(status, &dasup, &out_blr_length, &out_msg_type, &out_msg_length,
dialect, out_sqlda, DASUP_CLAUSE_select))
{
return error2(status, local);
}
s = GDS_DSQL_EXEC_IMM2_M(status, db_handle, tra_handle,
length, string, dialect,
in_blr_length,
dasup.dasup_clauses[DASUP_CLAUSE_bind].dasup_blr,
in_msg_type, in_msg_length,
dasup.dasup_clauses[DASUP_CLAUSE_bind].dasup_msg,
out_blr_length,
dasup.dasup_clauses[DASUP_CLAUSE_select].dasup_blr,
out_msg_type, out_msg_length,
dasup.dasup_clauses[DASUP_CLAUSE_select].dasup_msg);
if (!s)
{
s = UTLD_parse_sqlda(status, &dasup, NULL, NULL, NULL, dialect,
out_sqlda, DASUP_CLAUSE_select);
}
if (dasup.dasup_clauses[DASUP_CLAUSE_bind].dasup_blr)
gds__free((SLONG *)
(dasup.dasup_clauses[DASUP_CLAUSE_bind].dasup_blr));
if (dasup.dasup_clauses[DASUP_CLAUSE_bind].dasup_msg)
gds__free((SLONG *)
(dasup.dasup_clauses[DASUP_CLAUSE_bind].dasup_msg));
if (dasup.dasup_clauses[DASUP_CLAUSE_select].dasup_blr)
gds__free((SLONG *)
(dasup.dasup_clauses[DASUP_CLAUSE_select].dasup_blr));
if (dasup.dasup_clauses[DASUP_CLAUSE_select].dasup_msg)
gds__free((SLONG *)
(dasup.dasup_clauses[DASUP_CLAUSE_select].dasup_msg));
CHECK_STATUS(status);
return s;
}
ISC_STATUS API_ROUTINE GDS_DSQL_EXEC_IMM_M(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* tra_handle,
USHORT length,
const SCHAR* string,
USHORT dialect,
USHORT blr_length,
USHORT msg_type,
USHORT msg_length,
const SCHAR* blr,
SCHAR* msg)
{
/**************************************
*
* i s c _ d s q l _ e x e c _ i m m e d _ m
*
**************************************
*
* Functional description
*
**************************************/
return GDS_DSQL_EXECUTE_IMM_M(user_status, db_handle, tra_handle,
length, string, dialect, blr_length, blr,
msg_type, msg_length, msg);
}
ISC_STATUS API_ROUTINE GDS_DSQL_EXECUTE_IMM_M(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* tra_handle,
USHORT length,
const SCHAR* string,
USHORT dialect,
USHORT blr_length,
const SCHAR* blr,
USHORT msg_type,
USHORT msg_length,
SCHAR* msg)
{
/**************************************
*
* i s c _ d s q l _ e x e c u t e _ i m m e d i a t e _ m
*
**************************************
*
* Functional description
* Prepare a statement for execution.
*
**************************************/
return GDS_DSQL_EXEC_IMM2_M(user_status, db_handle, tra_handle,
length, string, dialect, blr_length, blr,
msg_type, msg_length, msg, 0, NULL, 0, 0,
NULL);
}
ISC_STATUS API_ROUTINE GDS_DSQL_EXEC_IMM2_M(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* tra_handle,
USHORT length,
const SCHAR* string,
USHORT dialect,
USHORT in_blr_length,
const SCHAR* in_blr,
USHORT in_msg_type,
USHORT in_msg_length,
SCHAR* in_msg,
USHORT out_blr_length,
SCHAR* out_blr,
USHORT out_msg_type,
USHORT out_msg_length,
SCHAR* out_msg)
{
/**************************************
*
* i s c _ d s q l _ e x e c _ i m m 2 _ m
*
**************************************
*
* Functional description
* Prepare a statement for execution.
*
**************************************/
2004-05-03 01:06:37 +02:00
FB_API_HANDLE crdb_trans_handle;
2003-04-16 12:18:51 +02:00
ISC_STATUS_ARRAY local, temp_status;
SCHAR buffer[16];
BOOLEAN ret_v3_error;
ISC_STATUS* status;
GET_STATUS;
bool stmt_eaten;
if (PREPARSE_execute( status,
db_handle,
tra_handle,
length,
string,
&stmt_eaten,
dialect))
{
if (status[1])
return error2(status, local);
2004-05-03 01:06:37 +02:00
crdb_trans_handle = 0;
if (GDS_START_TRANSACTION(status, &crdb_trans_handle, 1,
2004-12-24 09:52:39 +01:00
db_handle, 0, 0))
{
save_error_string(status);
GDS_DROP_DATABASE(temp_status, db_handle);
2004-05-03 01:06:37 +02:00
*db_handle = 0;
return error2(status, local);
}
ret_v3_error = FALSE;
if (!stmt_eaten) {
/* Check if against < 4.0 database */
2003-11-08 17:40:17 +01:00
const SCHAR ch = isc_info_base_level;
if (!GDS_DATABASE_INFO(status, db_handle, 1, &ch, sizeof(buffer),
buffer))
{
2003-11-08 17:40:17 +01:00
if ((buffer[0] != isc_info_base_level) || (buffer[4] > 3))
GDS_DSQL_EXEC_IMM3_M(status, db_handle,
&crdb_trans_handle, length, string,
dialect, in_blr_length, in_blr,
in_msg_type, in_msg_length, in_msg,
out_blr_length, out_blr,
out_msg_type, out_msg_length,
out_msg);
else
ret_v3_error = TRUE;
}
}
if (status[1]) {
GDS_ROLLBACK(temp_status, &crdb_trans_handle);
save_error_string(status);
GDS_DROP_DATABASE(temp_status, db_handle);
2004-05-03 01:06:37 +02:00
*db_handle = 0;
return error2(status, local);
}
else {
if (GDS_COMMIT(status, &crdb_trans_handle)) {
GDS_ROLLBACK(temp_status, &crdb_trans_handle);
save_error_string(status);
GDS_DROP_DATABASE(temp_status, db_handle);
2004-05-03 01:06:37 +02:00
*db_handle = 0;
return error2(status, local);
}
}
if (ret_v3_error) {
ISC_STATUS* s = status;
*s++ = isc_arg_gds;
*s++ = isc_srvr_version_too_old;
*s = isc_arg_end;
return error2(status, local);
}
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
else
return GDS_DSQL_EXEC_IMM3_M(user_status, db_handle, tra_handle,
length, string, dialect,
in_blr_length, in_blr, in_msg_type,
in_msg_length, in_msg, out_blr_length,
out_blr, out_msg_type, out_msg_length,
out_msg);
}
ISC_STATUS API_ROUTINE GDS_DSQL_EXEC_IMM3_M(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* tra_handle,
USHORT length,
const SCHAR* string,
USHORT dialect,
USHORT in_blr_length,
const SCHAR* in_blr,
USHORT in_msg_type,
USHORT in_msg_length,
SCHAR* in_msg,
USHORT out_blr_length,
SCHAR* out_blr,
USHORT out_msg_type,
USHORT out_msg_length,
SCHAR* out_msg)
{
/**************************************
*
* i s c _ d s q l _ e x e c _ i m m 3 _ m
*
**************************************
*
* Functional description
* Prepare a statement for execution.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS s, *status;
ISC_STATUS_ARRAY local;
WHY_ATT dbb;
2004-05-03 01:06:37 +02:00
WHY_TRA transaction = NULL, handle = NULL;
PTR entry;
//CLEAN clean;
/* If we haven't been initialized yet, do it now */
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*db_handle, dbb, HANDLE_database, isc_bad_db_handle);
if (*tra_handle) {
TRANSLATE_HANDLE(*tra_handle, transaction, HANDLE_transaction, isc_bad_trans_handle);
handle = find_transaction(dbb, transaction);
2004-05-03 01:06:37 +02:00
if (!handle)
return bad_handle (user_status, isc_bad_trans_handle);
handle = handle->handle.h_why;
}
/* Attempt to have the implementation which processed the database attach
process the prepare statement. This may not be feasible (e.g., the
server doesn't support remote DSQL because it's the wrong version or
something) in which case, execute the functionality locally (and hence
remotely through the original Y-valve). */
s = isc_unavailable;
entry = get_entrypoint(PROC_DSQL_EXEC_IMMED2, dbb->implementation);
if (entry != no_entrypoint) {
subsystem_enter();
s = (*entry) (status,
&dbb->handle,
&handle,
length,
string,
dialect,
in_blr_length,
in_blr,
in_msg_type,
in_msg_length,
in_msg,
out_blr_length,
out_blr, out_msg_type, out_msg_length, out_msg);
subsystem_exit();
}
if (s == isc_unavailable && !out_msg_length) {
entry = get_entrypoint(PROC_DSQL_EXEC_IMMED, dbb->implementation);
if (entry != no_entrypoint)
{
subsystem_enter();
s = (*entry) (status,
&dbb->handle,
&handle,
length,
string,
dialect,
in_blr_length,
in_blr, in_msg_type, in_msg_length, in_msg);
subsystem_exit();
}
}
if (s != isc_unavailable && !status[1])
if (transaction && !handle) {
/* Call the associated cleanup handlers */
2004-02-28 20:32:46 +01:00
WHY_cleanup_transaction(transaction);
//while (clean = transaction->cleanup) {
// transaction->cleanup = clean->clean_next;
// clean->TransactionRoutine(transaction, clean->clean_arg);
// free_block(clean);
//}
release_handle(transaction);
2004-05-03 01:06:37 +02:00
*tra_handle = 0;
}
else if (!transaction && handle) {
2004-05-03 01:06:37 +02:00
transaction = allocate_handle(dbb->implementation, handle, HANDLE_transaction);
if (transaction) {
transaction->parent = dbb;
*tra_handle = transaction->public_handle;
}
else {
status[0] = isc_arg_gds;
status[1] = isc_virmemexh;
status[2] = isc_arg_end;
return error2(status, local);
}
}
#ifndef NO_LOCAL_DSQL
if (s == isc_unavailable) {
/* if the entry point didn't exist or if the routine said the server
didn't support the protocol... do it locally */
subsystem_enter();
dsql8_execute_immediate(status, db_handle, tra_handle,
length, string, dialect,
in_blr_length, in_blr, in_msg_type,
in_msg_length, in_msg, out_blr_length,
out_blr, out_msg_type, out_msg_length,
out_msg);
subsystem_exit();
}
#endif
if (status[1])
return error2(status, local);
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_DSQL_FETCH(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* stmt_handle,
USHORT dialect,
XSQLDA* sqlda)
{
/**************************************
*
* i s c _ d s q l _ f e t c h
*
**************************************
*
* Functional description
* Fetch next record from a dynamic SQL cursor
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS s, *status;
ISC_STATUS_ARRAY local;
WHY_STMT statement;
USHORT blr_length, msg_type, msg_length;
2004-02-02 12:02:12 +01:00
sqlda_sup* dasup;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*stmt_handle, statement, HANDLE_statement, isc_bad_stmt_handle);
if (!sqlda) {
status[0] = isc_arg_gds;
status[1] = isc_dsql_sqlda_err;
status[2] = isc_arg_end;
return error2(status, local);
}
if (!(dasup = statement->das))
return bad_handle(user_status, isc_unprepared_stmt);
if (UTLD_parse_sqlda(status, dasup, &blr_length, &msg_type, &msg_length,
dialect, sqlda, DASUP_CLAUSE_select))
return error2(status, local);
if ((s = GDS_DSQL_FETCH_M(status, stmt_handle, blr_length,
dasup->dasup_clauses[DASUP_CLAUSE_select].dasup_blr,
0, msg_length,
dasup->dasup_clauses[DASUP_CLAUSE_select].dasup_msg))
&& s != 101)
{
CHECK_STATUS(status);
return s;
}
if (UTLD_parse_sqlda(status, dasup, NULL, NULL, NULL,
dialect, sqlda, DASUP_CLAUSE_select))
return error2(status, local);
CHECK_STATUS(status);
return s;
}
#ifdef SCROLLABLE_CURSORS
ISC_STATUS API_ROUTINE GDS_DSQL_FETCH2(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* stmt_handle,
USHORT dialect,
XSQLDA* sqlda,
USHORT direction,
SLONG offset)
{
/**************************************
*
* i s c _ d s q l _ f e t c h 2
*
**************************************
*
* Functional description
* Fetch next record from a dynamic SQL cursor
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS s, *status;
ISC_STATUS_ARRAY local;
WHY_STMT statement;
USHORT blr_length, msg_type, msg_length;
2004-02-02 12:02:12 +01:00
sqlda_sup* dasup;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*stmt_handle, statement, HANDLE_statement, isc_bad_stmt_handle);
if (!(dasup = statement->das))
return bad_handle(user_status, isc_unprepared_stmt);
if (UTLD_parse_sqlda(status, dasup, &blr_length, &msg_type, &msg_length,
dialect, sqlda, DASUP_CLAUSE_select))
return error2(status, local);
if ((s = GDS_DSQL_FETCH2_M(status, stmt_handle, blr_length,
dasup->dasup_clauses[DASUP_CLAUSE_select].dasup_blr,
0, msg_length,
dasup->dasup_clauses[DASUP_CLAUSE_select].dasup_msg,
direction, offset))
&& s != 101)
{
CHECK_STATUS(status);
return s;
}
if (UTLD_parse_sqlda(status, dasup, NULL, NULL, NULL,
dialect, sqlda, DASUP_CLAUSE_select))
return error2(status, local);
CHECK_STATUS(status);
return s;
}
#endif
ISC_STATUS API_ROUTINE GDS_DSQL_FETCH_M(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* stmt_handle,
USHORT blr_length,
const SCHAR* blr,
USHORT msg_type,
USHORT msg_length,
SCHAR* msg)
{
/**************************************
*
* i s c _ d s q l _ f e t c h _ m
*
**************************************
*
* Functional description
* Fetch next record from a dynamic SQL cursor
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS s, *status;
ISC_STATUS_ARRAY local;
WHY_STMT statement;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*stmt_handle, statement, HANDLE_statement, isc_bad_stmt_handle);
subsystem_enter();
#ifndef NO_LOCAL_DSQL
if (statement->flags & HANDLE_STATEMENT_local)
s = dsql8_fetch(status,
&statement->handle.h_dsql, blr_length, blr, msg_type,
msg_length, msg
#ifdef SCROLLABLE_CURSORS
, (USHORT) 0, (ULONG) 1);
#else
);
#endif
else
#endif
s = CALL(PROC_DSQL_FETCH, statement->implementation) (status,
&statement->
handle,
blr_length, blr,
msg_type,
msg_length, msg
#ifdef SCROLLABLE_CURSORS
,
(USHORT) 0,
(ULONG) 1);
#else
);
#endif
subsystem_exit();
CHECK_STATUS(status);
if (s == 100 || s == 101)
return s;
else if (s)
return error2(status, local);
return FB_SUCCESS;
}
#ifdef SCROLLABLE_CURSORS
ISC_STATUS API_ROUTINE GDS_DSQL_FETCH2_M(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* stmt_handle,
USHORT blr_length,
const SCHAR* blr,
USHORT msg_type,
USHORT msg_length,
SCHAR* msg,
USHORT direction,
SLONG offset)
{
/**************************************
*
* i s c _ d s q l _ f e t c h 2 _ m
*
**************************************
*
* Functional description
* Fetch next record from a dynamic SQL cursor
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS s, *status;
ISC_STATUS_ARRAY local;
WHY_STMT statement;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*stmt_handle, statement, HANDLE_statement, isc_bad_stmt_handle);
subsystem_enter();
#ifndef NO_LOCAL_DSQL
if (statement->flags & HANDLE_STATEMENT_local)
s = dsql8_fetch(status,
&statement->handle, blr_length, blr, msg_type,
msg_length, msg, direction, offset);
else
#endif
s = CALL(PROC_DSQL_FETCH, statement->implementation) (status,
&statement->
handle,
blr_length, blr,
msg_type,
msg_length, msg,
direction,
offset);
subsystem_exit();
CHECK_STATUS(status);
if (s == 100 || s == 101)
return s;
else if (s)
return error2(status, local);
return FB_SUCCESS;
}
#endif
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE GDS_DSQL_FREE(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * stmt_handle,
USHORT option)
{
/*****************************************
*
* i s c _ d s q l _ f r e e _ s t a t e m e n t
*
*****************************************
*
* Functional Description
* release request for an sql statement
*
*****************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_STMT statement;
WHY_DBB dbb, *ptr;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*stmt_handle, statement, HANDLE_statement, isc_bad_stmt_handle);
subsystem_enter();
#ifndef NO_LOCAL_DSQL
if (statement->flags & HANDLE_STATEMENT_local)
dsql8_free_statement(status, &statement->handle.h_dsql, option);
else
#endif
CALL(PROC_DSQL_FREE, statement->implementation) (status,
&statement->handle,
option);
subsystem_exit();
if (status[1])
return error2(status, local);
/* Release the handle and any request hanging off of it. */
if (option & DSQL_drop) {
/* Get rid of connections to database */
dbb = statement->parent;
for (ptr = &dbb->statements; *ptr; ptr = &(*ptr)->next)
if (*ptr == statement) {
*ptr = statement->next;
break;
}
release_dsql_support(statement->das);
release_handle(statement);
2004-05-03 01:06:37 +02:00
*stmt_handle = 0;
}
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_DSQL_INSERT(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* stmt_handle,
USHORT dialect,
XSQLDA* sqlda)
{
/**************************************
*
* i s c _ d s q l _ i n s e r t
*
**************************************
*
* Functional description
* Insert next record into a dynamic SQL cursor
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_STMT statement;
USHORT blr_length, msg_type, msg_length;
2004-02-02 12:02:12 +01:00
sqlda_sup* dasup;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*stmt_handle, statement, HANDLE_statement, isc_bad_stmt_handle);
if (!(dasup = statement->das))
return bad_handle(user_status, isc_unprepared_stmt);
if (UTLD_parse_sqlda(status, dasup, &blr_length, &msg_type, &msg_length,
dialect, sqlda, DASUP_CLAUSE_bind))
return error2(status, local);
return GDS_DSQL_INSERT_M(status, stmt_handle, blr_length,
dasup->dasup_clauses[DASUP_CLAUSE_bind].
dasup_blr, 0, msg_length,
dasup->dasup_clauses[DASUP_CLAUSE_bind].
dasup_msg);
}
ISC_STATUS API_ROUTINE GDS_DSQL_INSERT_M(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* stmt_handle,
USHORT blr_length,
const SCHAR* blr,
USHORT msg_type,
USHORT msg_length,
const SCHAR* msg)
{
/**************************************
*
* i s c _ d s q l _ i n s e r t _ m
*
**************************************
*
* Functional description
* Insert next record into a dynamic SQL cursor
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS s, *status;
ISC_STATUS_ARRAY local;
WHY_STMT statement;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*stmt_handle, statement, HANDLE_statement, isc_bad_stmt_handle);
subsystem_enter();
#ifndef NO_LOCAL_DSQL
if (statement->flags & HANDLE_STATEMENT_local)
s = dsql8_insert(status,
&statement->handle.h_dsql, blr_length, blr, msg_type,
msg_length, msg);
else
#endif
s = CALL(PROC_DSQL_INSERT, statement->implementation) (status,
&statement->
handle,
blr_length,
blr, msg_type,
msg_length,
msg);
subsystem_exit();
if (s)
return error2(status, local);
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_DSQL_PREPARE(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* tra_handle,
FB_API_HANDLE* stmt_handle,
USHORT length,
const SCHAR* string,
USHORT dialect,
XSQLDA* sqlda)
{
/**************************************
*
* i s c _ d s q l _ p r e p a r e
*
**************************************
*
* Functional description
* Prepare a statement for execution.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
USHORT buffer_len;
SCHAR *buffer, local_buffer[BUFFER_MEDIUM];
2004-02-02 12:02:12 +01:00
sqlda_sup* dasup;
GET_STATUS;
if (!(buffer = get_sqlda_buffer(local_buffer, sizeof(local_buffer), sqlda,
dialect, &buffer_len)))
{
status[0] = isc_arg_gds;
status[1] = isc_virmemexh;
status[2] = isc_arg_end;
return error2(status, local);
}
if (!GDS_DSQL_PREPARE_M(status,
tra_handle,
stmt_handle,
length,
string,
dialect,
sizeof(sql_prepare_info),
sql_prepare_info,
buffer_len,
buffer))
{
2004-05-03 01:06:37 +02:00
WHY_STMT statement = WHY_translate_handle(*stmt_handle);
release_dsql_support(statement->das);
2004-02-02 12:02:12 +01:00
if (!(dasup = (sqlda_sup*) alloc((SLONG) sizeof(sqlda_sup)))) {
2004-05-03 01:06:37 +02:00
statement->requests = 0;
status[0] = isc_arg_gds;
status[1] = isc_virmemexh;
status[2] = isc_arg_end;
}
else {
2004-05-03 01:06:37 +02:00
statement->das = dasup;
dasup->dasup_dialect = dialect;
iterative_sql_info(status, stmt_handle, sizeof(sql_prepare_info),
sql_prepare_info, buffer_len, buffer, dialect,
sqlda);
}
}
if (buffer != local_buffer)
free_block(buffer);
if (status[1])
return error2(status, local);
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_DSQL_PREPARE_M(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* tra_handle,
FB_API_HANDLE* stmt_handle,
USHORT length,
const SCHAR* string,
USHORT dialect,
USHORT item_length,
const SCHAR* items,
USHORT buffer_length,
SCHAR* buffer)
{
/**************************************
*
* i s c _ d s q l _ p r e p a r e _ m
*
**************************************
*
* Functional description
* Prepare a statement for execution.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_STMT statement;
2004-05-03 01:06:37 +02:00
WHY_TRA handle = NULL, transaction = NULL;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*stmt_handle, statement, HANDLE_statement, isc_bad_stmt_handle);
if (*tra_handle) {
TRANSLATE_HANDLE(*tra_handle, transaction, HANDLE_transaction, isc_bad_trans_handle);
handle = find_transaction(statement->parent, transaction);
2004-05-03 01:06:37 +02:00
if (!handle)
return bad_handle (user_status, isc_bad_trans_handle);
handle = handle->handle.h_why;
}
subsystem_enter();
#ifndef NO_LOCAL_DSQL
if (statement->flags & HANDLE_STATEMENT_local)
dsql8_prepare(status, tra_handle, &statement->handle.h_dsql,
length, string, dialect, item_length, items,
buffer_length, buffer);
else
#endif
{
CALL(PROC_DSQL_PREPARE, statement->implementation) (status,
&handle,
&statement->
handle, length,
string, dialect,
item_length,
items,
buffer_length,
buffer);
}
subsystem_exit();
if (status[1])
return error2(status, local);
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_DSQL_SET_CURSOR(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* stmt_handle,
const SCHAR* cursor,
USHORT type)
{
/**************************************
*
* i s c _ d s q l _ s e t _ c u r s o r
*
**************************************
*
* Functional description
* Set a cursor name for a dynamic request.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_STMT statement;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*stmt_handle, statement, HANDLE_statement, isc_bad_stmt_handle);
subsystem_enter();
#ifndef NO_LOCAL_DSQL
if (statement->flags & HANDLE_STATEMENT_local)
dsql8_set_cursor(status, &statement->handle.h_dsql, cursor, type);
else
#endif
CALL(PROC_DSQL_SET_CURSOR, statement->implementation) (status,
&statement->
handle, cursor,
type);
subsystem_exit();
if (status[1])
return error2(status, local);
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_DSQL_SQL_INFO(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* stmt_handle,
SSHORT item_length,
const SCHAR* items,
SSHORT buffer_length,
SCHAR* buffer)
{
/**************************************
*
* i s c _ d s q l _ s q l _ i n f o
*
**************************************
*
* Functional description
* Provide information on sql statement.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_STMT statement;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*stmt_handle, statement, HANDLE_statement, isc_bad_stmt_handle);
subsystem_enter();
#ifndef NO_LOCAL_DSQL
if (statement->flags & HANDLE_STATEMENT_local)
dsql8_sql_info(status, &statement->handle.h_dsql, item_length, items,
buffer_length, buffer);
else
#endif
CALL(PROC_DSQL_SQL_INFO, statement->implementation) (status,
&statement->
handle,
item_length,
items,
buffer_length,
buffer);
subsystem_exit();
if (status[1])
return error2(status, local);
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
int API_ROUTINE gds__enable_subsystem(TEXT * subsystem)
{
/**************************************
*
* g d s _ $ e n a b l e _ s u b s y s t e m
*
**************************************
*
* Functional description
* Enable access to a specific subsystem. If no subsystem
* has been explicitly enabled, all are available.
*
**************************************/
2003-02-13 11:11:35 +01:00
const IMAGE *sys, *end;
for (sys = images, end = sys + SUBSYSTEMS; sys < end; sys++)
if (!strcmp(sys->name, subsystem)) {
if (!~why_enabled)
why_enabled = 0;
why_enabled |= (1 << (sys - images));
return TRUE;
}
return FALSE;
}
#ifndef REQUESTER
ISC_STATUS API_ROUTINE isc_wait_for_event(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * handle,
USHORT length,
const SCHAR* events,
SCHAR* buffer)
{
/**************************************
*
* g d s _ $ e v e n t _ w a i t
*
**************************************
*
* Functional description
* Que request for event notification.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
SLONG value, id;
2004-02-02 12:02:12 +01:00
event_t* event_ptr;
GET_STATUS;
if (!why_initialized) {
gds__register_cleanup((FPTR_VOID_PTR) exit_handler, why_event);
why_initialized = TRUE;
ISC_event_init(why_event, 0, 0);
}
value = ISC_event_clear(why_event);
if (GDS_QUE_EVENTS
(status, handle, &id, length, (const UCHAR*) events, event_ast, (UCHAR*) buffer))
{
return error2(status, local);
}
event_ptr = why_event;
ISC_event_wait(1, &event_ptr, &value, -1, 0, NULL);
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
#endif
ISC_STATUS API_ROUTINE GDS_INTL_FUNCTION(ISC_STATUS * user_status,
FB_API_HANDLE * handle,
USHORT function,
UCHAR charSetNumber,
USHORT strLen,
const UCHAR* str,
USHORT* result)
{
/**************************************
*
* g d s _ i n t l _ f u n c t i o n
*
**************************************
*
* Functional description
* Return INTL informations.
* (candidate for removal when engine functions can be called by DSQL)
*
**************************************/
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_ATT database;
GET_STATUS;
TRANSLATE_HANDLE(*handle, database, HANDLE_database, isc_bad_db_handle);
subsystem_enter();
if (CALL(PROC_INTL_FUNCTION, database->implementation) (status,
&database->handle,
function,
charSetNumber,
strLen,
str,
result))
{
return error(status, local);
}
RETURN_SUCCESS;
}
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE GDS_GET_SEGMENT(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * blob_handle,
USHORT * length,
USHORT buffer_length,
UCHAR * buffer)
{
/**************************************
*
* g d s _ $ g e t _ s e g m e n t
*
**************************************
*
* Functional description
* Abort a partially completed blob.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status, code;
ISC_STATUS_ARRAY local;
WHY_BLB blob;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*blob_handle, blob, HANDLE_blob, isc_bad_segstr_handle);
subsystem_enter();
code = CALL(PROC_GET_SEGMENT, blob->implementation) (status,
&blob->handle,
length,
buffer_length,
buffer);
if (code) {
if (code == isc_segstr_eof || code == isc_segment) {
subsystem_exit();
CHECK_STATUS(status);
return code;
}
return error(status, local);
}
RETURN_SUCCESS;
}
2003-12-03 09:19:24 +01:00
ISC_STATUS API_ROUTINE GDS_GET_SLICE(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* tra_handle,
ISC_QUAD* array_id,
USHORT sdl_length,
2003-12-03 09:19:24 +01:00
const UCHAR* sdl,
USHORT param_length,
2003-12-03 09:19:24 +01:00
const UCHAR* param,
SLONG slice_length,
2003-12-03 09:19:24 +01:00
UCHAR* slice,
SLONG* return_length)
{
/**************************************
*
* g d s _ $ g e t _ s l i c e
*
**************************************
*
* Functional description
* Snatch a slice of an array.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_ATT dbb;
WHY_TRA transaction;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*db_handle, dbb, HANDLE_database, isc_bad_db_handle);
FIND_TRANSACTION(*tra_handle, transaction, dbb, isc_bad_trans_handle);
subsystem_enter();
if (CALL(PROC_GET_SLICE, dbb->implementation) (status,
&dbb->handle,
&transaction->handle,
array_id,
sdl_length,
sdl,
param_length,
param,
slice_length,
slice,
return_length))
return error(status, local);
RETURN_SUCCESS;
}
ISC_STATUS gds__handle_cleanup(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * user_handle)
{
/**************************************
*
* g d s _ $ h a n d l e _ c l e a n u p
*
**************************************
*
* Functional description
* Clean up a dangling y-valve handle.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_HNDL handle;
WHY_TRA transaction, sub;
//CLEAN clean;
GET_STATUS;
2004-05-03 01:06:37 +02:00
handle = WHY_translate_handle(*user_handle);
if (!handle) {
status[0] = isc_arg_gds;
status[1] = isc_bad_db_handle;
status[2] = isc_arg_end;
return error2(status, local);
}
switch (handle->type) {
case HANDLE_transaction:
/* Call the associated cleanup handlers */
transaction = (WHY_TRA) handle;
2004-02-28 20:32:46 +01:00
WHY_cleanup_transaction(transaction);
//while (clean = transaction->cleanup) {
// transaction->cleanup = clean->clean_next;
// clean->TransactionRoutine(transaction, clean->clean_arg);
// free_block(clean);
//}
while (sub = transaction) {
transaction = sub->next;
release_handle(sub);
}
break;
default:
status[0] = isc_arg_gds;
status[1] = isc_bad_db_handle;
status[2] = isc_arg_end;
return error2(status, local);
}
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_OPEN_BLOB(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* tra_handle,
FB_API_HANDLE* blob_handle,
SLONG* blob_id)
{
/**************************************
*
* g d s _ $ o p e n _ b l o b
*
**************************************
*
* Functional description
* Open an existing blob.
*
**************************************/
return open_blob(user_status, db_handle, tra_handle, blob_handle, blob_id,
0, 0, PROC_OPEN_BLOB, PROC_OPEN_BLOB2);
}
ISC_STATUS API_ROUTINE GDS_OPEN_BLOB2(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* tra_handle,
FB_API_HANDLE* blob_handle,
SLONG* blob_id,
SSHORT bpb_length,
const UCHAR* bpb)
{
/**************************************
*
* g d s _ $ o p e n _ b l o b 2
*
**************************************
*
* Functional description
* Open an existing blob (extended edition).
*
**************************************/
return open_blob(user_status, db_handle, tra_handle, blob_handle, blob_id,
bpb_length, bpb, PROC_OPEN_BLOB,
PROC_OPEN_BLOB2);
}
ISC_STATUS API_ROUTINE GDS_PREPARE(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* tra_handle)
{
/**************************************
*
* g d s _ $ p r e p a r e
*
**************************************
*
* Functional description
* Prepare a transaction for commit. First phase of a two
* phase commit.
*
**************************************/
return GDS_PREPARE2(user_status, tra_handle, 0, 0);
}
ISC_STATUS API_ROUTINE GDS_PREPARE2(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* tra_handle,
USHORT msg_length,
const UCHAR* msg)
{
/**************************************
*
* g d s _ $ p r e p a r e 2
*
**************************************
*
* Functional description
* Prepare a transaction for commit. First phase of a two
* phase commit.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_TRA transaction, sub;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*tra_handle, transaction, HANDLE_transaction, isc_bad_trans_handle);
subsystem_enter();
for (sub = transaction; sub; sub = sub->next)
if (sub->implementation != SUBSYSTEMS &&
CALL(PROC_PREPARE, sub->implementation) (status, &sub->handle,
msg_length, msg))
return error(status, local);
transaction->flags |= HANDLE_TRANSACTION_limbo;
RETURN_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_PUT_SEGMENT(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* blob_handle,
USHORT buffer_length,
const UCHAR* buffer)
{
/**************************************
*
* g d s _ $ p u t _ s e g m e n t
*
**************************************
*
* Functional description
* Abort a partially completed blob.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_BLB blob;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*blob_handle, blob, HANDLE_blob, isc_bad_segstr_handle);
subsystem_enter();
if (CALL(PROC_PUT_SEGMENT, blob->implementation) (status,
&blob->handle,
buffer_length,
buffer))
return error(status, local);
RETURN_SUCCESS;
}
2003-12-03 09:19:24 +01:00
ISC_STATUS API_ROUTINE GDS_PUT_SLICE(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* tra_handle,
ISC_QUAD* array_id,
USHORT sdl_length,
2003-12-03 09:19:24 +01:00
const UCHAR* sdl,
USHORT param_length,
2004-02-03 12:25:29 +01:00
const SLONG* param,
SLONG slice_length,
2003-12-03 09:19:24 +01:00
UCHAR* slice)
{
/**************************************
*
* g d s _ $ p u t _ s l i c e
*
**************************************
*
* Functional description
* Snatch a slice of an array.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_ATT dbb;
WHY_TRA transaction;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*db_handle, dbb, HANDLE_database, isc_bad_db_handle);
FIND_TRANSACTION(*tra_handle, transaction, dbb, isc_bad_trans_handle);
subsystem_enter();
if (CALL(PROC_PUT_SLICE, dbb->implementation) (status,
&dbb->handle,
&transaction->handle,
array_id,
sdl_length,
sdl,
param_length,
param,
slice_length,
slice))
return error(status, local);
RETURN_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_QUE_EVENTS(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* handle,
SLONG* id,
USHORT length,
const UCHAR* events,
2003-12-22 11:00:59 +01:00
FPTR_EVENT_CALLBACK ast,
void* arg)
{
/**************************************
*
* g d s _ $ q u e _ e v e n t s
*
**************************************
*
* Functional description
* Que request for event notification.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_ATT database;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*handle, database, HANDLE_database, isc_bad_db_handle);
subsystem_enter();
if (CALL(PROC_QUE_EVENTS, database->implementation) (status,
&database->handle,
id, length, events,
ast, arg))
return error(status, local);
RETURN_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_RECEIVE(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* req_handle,
USHORT msg_type,
USHORT msg_length,
SCHAR* msg,
SSHORT level)
{
/**************************************
*
* g d s _ $ r e c e i v e
*
**************************************
*
* Functional description
* Get a record from the host program.
*
**************************************/
#ifdef SCROLLABLE_CURSORS
return GDS_RECEIVE2(user_status, req_handle, msg_type, msg_length,
msg, level, (USHORT) blr_continue, /* means continue in same direction as before */
(ULONG) 1);
#else
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_REQ request;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*req_handle, request, HANDLE_request, isc_bad_req_handle);
subsystem_enter();
if (CALL(PROC_RECEIVE, request->implementation) (status,
&request->handle,
msg_type,
msg_length,
msg,
level))
return error(status, local);
RETURN_SUCCESS;
#endif
}
#ifdef SCROLLABLE_CURSORS
ISC_STATUS API_ROUTINE GDS_RECEIVE2(ISC_STATUS* user_status,
WHY_REQ* req_handle,
USHORT msg_type,
USHORT msg_length,
SCHAR* msg,
SSHORT level,
USHORT direction,
ULONG offset)
{
/**************************************
*
* i s c _ r e c e i v e 2
*
**************************************
*
* Functional description
* Scroll through the request output stream,
* then get a record from the host program.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_REQ request;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*req_handle, request, HANDLE_request, isc_bad_req_handle);
subsystem_enter();
if (CALL(PROC_RECEIVE, request->implementation) (status,
&request->handle,
msg_type,
msg_length,
msg,
level,
direction,
offset))
return error(status, local);
RETURN_SUCCESS;
}
#endif
ISC_STATUS API_ROUTINE GDS_RECONNECT(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* tra_handle,
SSHORT length,
const UCHAR* id)
{
/**************************************
*
* g d s _ $ r e c o n n e c t
*
**************************************
*
* Functional description
* Connect to a transaction in limbo.
*
**************************************/
ISC_STATUS* status;
2003-04-16 12:18:51 +02:00
ISC_STATUS_ARRAY local;
GET_STATUS;
NULL_CHECK(tra_handle, isc_bad_trans_handle, HANDLE_transaction);
2004-05-03 01:06:37 +02:00
WHY_ATT database;
TRANSLATE_HANDLE(*db_handle, database, HANDLE_database, isc_bad_db_handle);
subsystem_enter();
2004-05-03 01:06:37 +02:00
WHY_TRA transaction = NULL;
if (CALL(PROC_RECONNECT, database->implementation) (status,
&database->handle,
2004-05-03 01:06:37 +02:00
&transaction,
length, id))
{
return error(status, local);
}
2004-05-03 01:06:37 +02:00
transaction = allocate_handle( database->implementation,
transaction,
HANDLE_transaction);
2004-05-03 01:06:37 +02:00
if (transaction) {
transaction->parent = database;
*tra_handle = transaction->public_handle;
}
else {
status[0] = isc_arg_gds;
status[1] = isc_virmemexh;
status[2] = isc_arg_end;
return error(status, local);
}
RETURN_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_RELEASE_REQUEST(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * req_handle)
{
/**************************************
*
* g d s _ $ r e l e a s e _ r e q u e s t
*
**************************************
*
* Functional description
* Release a request.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_REQ request, *ptr;
WHY_DBB dbb;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*req_handle, request, HANDLE_request, isc_bad_req_handle);
subsystem_enter();
if (CALL(PROC_RELEASE_REQUEST, request->implementation) (status,
&request->handle))
{
return error(status, local);
}
/* Get rid of connections to database */
dbb = request->parent;
for (ptr = &dbb->requests; *ptr; ptr = &(*ptr)->next)
if (*ptr == request) {
*ptr = request->next;
break;
}
release_handle(request);
2004-05-03 01:06:37 +02:00
*req_handle = 0;
RETURN_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_REQUEST_INFO(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* req_handle,
SSHORT level,
SSHORT item_length,
const SCHAR* items,
SSHORT buffer_length,
SCHAR* buffer)
{
/**************************************
*
* g d s _ $ r e q u e s t _ i n f o
*
**************************************
*
* Functional description
* Provide information on blob object.
*
**************************************/
ISC_STATUS* status;
2003-04-16 12:18:51 +02:00
ISC_STATUS_ARRAY local;
WHY_REQ request;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*req_handle, request, HANDLE_request, isc_bad_req_handle);
subsystem_enter();
if (CALL(PROC_REQUEST_INFO, request->implementation) (status,
&request->handle,
level,
item_length,
items,
buffer_length,
buffer))
{
return error(status, local);
}
RETURN_SUCCESS;
}
2005-06-16 14:08:17 +02:00
#if defined (SOLARIS) || defined (WIN_NT)
extern "C"
#endif
SLONG API_ROUTINE isc_reset_fpe(USHORT fpe_status)
{
/**************************************
*
* i s c _ r e s e t _ f p e
*
**************************************
*
* Functional description
* API to be used to tell InterBase to reset it's
* FPE handler - eg: client has an FPE of it's own
* and just changed it.
*
* Returns
* Prior setting of the FPE reset flag
*
**************************************/
#if !(defined REQUESTER || defined SUPERCLIENT || defined SUPERSERVER)
SLONG prior;
prior = (SLONG) subsystem_FPE_reset;
switch (fpe_status) {
case FPE_RESET_INIT_ONLY:
subsystem_FPE_reset = fpe_status;
break;
case FPE_RESET_NEXT_API_CALL:
subsystem_FPE_reset = fpe_status;
break;
case FPE_RESET_ALL_API_CALL:
subsystem_FPE_reset = fpe_status;
break;
default:
break;
}
return prior;
#else
return FPE_RESET_INIT_ONLY;
#endif
}
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE GDS_ROLLBACK_RETAINING(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * tra_handle)
{
/**************************************
*
* i s c _ r o l l b a c k _ r e t a i n i n g
*
**************************************
*
* Functional description
* Abort a transaction, but keep all cursors open.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_TRA transaction, sub;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*tra_handle, transaction, HANDLE_transaction, isc_bad_trans_handle);
subsystem_enter();
for (sub = transaction; sub; sub = sub->next)
if (sub->implementation != SUBSYSTEMS &&
CALL(PROC_ROLLBACK_RETAINING, sub->implementation) (status,
&sub->handle))
return error(status, local);
transaction->flags |= HANDLE_TRANSACTION_limbo;
RETURN_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_ROLLBACK(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * tra_handle)
{
/**************************************
*
* g d s _ $ r o l l b a c k
*
**************************************
*
* Functional description
* Abort a transaction.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_TRA transaction, sub;
//CLEAN clean;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*tra_handle, transaction, HANDLE_transaction, isc_bad_trans_handle);
subsystem_enter();
for (sub = transaction; sub; sub = sub->next)
if (sub->implementation != SUBSYSTEMS &&
CALL(PROC_ROLLBACK, sub->implementation) (status, &sub->handle))
{
if (!is_network_error(status) ||
(transaction->flags & HANDLE_TRANSACTION_limbo) )
{
return error(status, local);
}
}
if (is_network_error(status))
init_status(status);
subsystem_exit();
/* Call the associated cleanup handlers */
2004-02-28 20:32:46 +01:00
WHY_cleanup_transaction(transaction);
//while (clean = transaction->cleanup) {
// transaction->cleanup = clean->clean_next;
// clean->TransactionRoutine(transaction, clean->clean_arg);
// free_block(clean);
//}
while (sub = transaction) {
transaction = sub->next;
release_handle(sub);
}
2004-05-03 01:06:37 +02:00
*tra_handle = 0;
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE GDS_SEEK_BLOB(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * blob_handle,
SSHORT mode,
SLONG offset,
SLONG * result)
{
/**************************************
*
* g d s _ $ s e e k _ b l o b
*
**************************************
*
* Functional description
* Seek a blob.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_BLB blob;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*blob_handle, blob, HANDLE_blob, isc_bad_segstr_handle);
subsystem_enter();
/***
if (blob->flags & HANDLE_BLOB_filter)
{
subsystem_exit();
BLF_close_blob (status, &blob->handle);
subsystem_enter();
}
else
***/
CALL(PROC_SEEK_BLOB, blob->implementation) (status,
&blob->handle,
mode,
offset, result);
if (status[1])
return error(status, local);
RETURN_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_SEND(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* req_handle,
USHORT msg_type,
USHORT msg_length,
const SCHAR* msg,
SSHORT level)
{
/**************************************
*
* g d s _ $ s e n d
*
**************************************
*
* Functional description
* Get a record from the host program.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_REQ request;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*req_handle, request, HANDLE_request, isc_bad_req_handle);
subsystem_enter();
if (CALL(PROC_SEND, request->implementation) (status, &request->handle,
msg_type, msg_length, msg,
level))
return error(status, local);
RETURN_SUCCESS;
}
2003-12-03 09:19:24 +01:00
ISC_STATUS API_ROUTINE GDS_SERVICE_ATTACH(ISC_STATUS* user_status,
USHORT service_length,
2003-12-03 09:19:24 +01:00
const TEXT* service_name,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* public_handle,
USHORT spb_length,
2003-12-03 09:19:24 +01:00
const SCHAR* spb)
{
/**************************************
*
* i s c _ s e r v i c e _ a t t a c h
*
**************************************
*
* Functional description
* Attach a service through the first subsystem
* that recognizes it.
*
**************************************/
2003-12-03 09:19:24 +01:00
ISC_STATUS *status;
2003-04-16 12:18:51 +02:00
ISC_STATUS_ARRAY local, temp;
GET_STATUS;
2004-05-03 01:06:37 +02:00
NULL_CHECK(public_handle, isc_bad_svc_handle, HANDLE_service);
if (!service_name) {
status[0] = isc_arg_gds;
status[1] = isc_service_att_err;
status[2] = isc_arg_gds;
status[3] = isc_svc_name_missing;
status[4] = isc_arg_end;
return error2(status, local);
}
if (spb_length > 0 && !spb) {
status[0] = isc_arg_gds;
status[1] = isc_bad_spb_form;
status[2] = isc_arg_end;
return error2(status, local);
}
#if defined (SERVER_SHUTDOWN) && !defined (SUPERCLIENT) && !defined (REQUESTER)
if (shutdown_flag) {
status[0] = isc_arg_gds;
status[1] = isc_shutwarn;
status[2] = isc_arg_end;
return error2(status, local);
}
#endif /* SERVER_SHUTDOWN && !SUPERCLIENT && !REQUESTER */
subsystem_enter();
SUBSYSTEM_USAGE_INCR;
2003-12-03 09:19:24 +01:00
ISC_STATUS* ptr = status;
2003-12-03 09:19:24 +01:00
USHORT org_length = service_length;
if (org_length) {
2003-12-03 09:19:24 +01:00
const TEXT* p = service_name + org_length - 1;
while (*p == ' ')
p--;
org_length = p - service_name + 1;
}
2003-12-03 09:19:24 +01:00
for (USHORT n = 0; n < SUBSYSTEMS; n++) {
if (why_enabled && !(why_enabled & (1 << n)))
continue;
2004-05-03 01:06:37 +02:00
WHY_SVC handle = 0;
if (!CALL(PROC_SERVICE_ATTACH, n) (ptr,
org_length,
service_name,
2004-05-03 01:06:37 +02:00
&handle, spb_length, spb))
{
2004-05-03 01:06:37 +02:00
WHY_SVC service = allocate_handle(n, handle, HANDLE_service);
if (!service)
{
/* No memory. Make a half-hearted attempt to detach service. */
CALL(PROC_SERVICE_DETACH, n) (ptr, handle);
2004-05-03 01:06:37 +02:00
*public_handle = 0;
status[0] = isc_arg_gds;
status[1] = isc_virmemexh;
status[2] = isc_arg_end;
break;
}
2004-05-03 01:06:37 +02:00
*public_handle = service->public_handle;
service->cleanup = NULL;
status[0] = isc_arg_gds;
status[1] = 0;
if (status[2] != isc_arg_warning)
status[2] = isc_arg_end;
subsystem_exit();
CHECK_STATUS_SUCCESS(status);
return status[1];
}
if (ptr[1] != isc_unavailable)
ptr = temp;
}
SUBSYSTEM_USAGE_DECR;
if (status[1] == isc_unavailable)
status[1] = isc_service_att_err;
return error(status, local);
}
ISC_STATUS API_ROUTINE GDS_SERVICE_DETACH(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * handle)
{
/**************************************
*
* i s c _ s e r v i c e _ d e t a c h
*
**************************************
*
* Functional description
* Close down a service.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_SVC service;
CLEAN clean;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*handle, service, HANDLE_service, isc_bad_svc_handle);
subsystem_enter();
if (CALL(PROC_SERVICE_DETACH, service->implementation) (status,
&service->handle))
return error(status, local);
SUBSYSTEM_USAGE_DECR;
subsystem_exit();
/* Call the associated cleanup handlers */
while ((clean = service->cleanup) != NULL) {
service->cleanup = clean->clean_next;
clean->DatabaseRoutine(handle, clean->clean_arg);
free_block(clean);
}
release_handle(service);
2004-05-03 01:06:37 +02:00
*handle = 0;
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_SERVICE_QUERY(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* handle,
ULONG* reserved,
USHORT send_item_length,
const SCHAR* send_items,
USHORT recv_item_length,
const SCHAR* recv_items,
USHORT buffer_length,
SCHAR* buffer)
{
/**************************************
*
* i s c _ s e r v i c e _ q u e r y
*
**************************************
*
* Functional description
* Provide information on service object.
*
* NOTE: The parameter RESERVED must not be used
* for any purpose as there are networking issues
* involved (as with any handle that goes over the
* network). This parameter will be implemented at
* a later date.
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_SVC service;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*handle, service, HANDLE_service, isc_bad_svc_handle);
subsystem_enter();
if (CALL(PROC_SERVICE_QUERY, service->implementation) (status, &service->handle, 0, /* reserved */
send_item_length,
send_items,
recv_item_length,
recv_items,
buffer_length,
buffer))
return error(status, local);
RETURN_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_SERVICE_START(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* handle,
ULONG* reserved,
USHORT spb_length,
const SCHAR* spb)
{
/**************************************
*
* i s c _ s e r v i c e _ s t a r t
*
**************************************
*
* Functional description
* Starts a service thread
*
* NOTE: The parameter RESERVED must not be used
* for any purpose as there are networking issues
* involved (as with any handle that goes over the
* network). This parameter will be implemented at
* a later date.
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_SVC service;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*handle, service, HANDLE_service, isc_bad_svc_handle);
subsystem_enter();
if (CALL(PROC_SERVICE_START, service->implementation) (status,
&service->handle,
NULL,
2004-12-24 09:52:39 +01:00
spb_length, spb))
{
return error(status, local);
}
RETURN_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_START_AND_SEND(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* req_handle,
FB_API_HANDLE* tra_handle,
USHORT msg_type,
USHORT msg_length,
const SCHAR* msg,
SSHORT level)
{
/**************************************
*
* g d s _ $ s t a r t _ a n d _ s e n d
*
**************************************
*
* Functional description
* Get a record from the host program.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_REQ request;
WHY_TRA transaction;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*req_handle, request, HANDLE_request, isc_bad_req_handle);
FIND_TRANSACTION(*tra_handle, transaction, request->parent, isc_bad_trans_handle);
subsystem_enter();
if (CALL(PROC_START_AND_SEND, request->implementation) (status,
&request->handle,
&transaction->
handle, msg_type,
msg_length, msg,
level))
return error(status, local);
RETURN_SUCCESS;
}
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE GDS_START(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * req_handle,
FB_API_HANDLE * tra_handle,
SSHORT level)
{
/**************************************
*
* g d s _ $ s t a r t
*
**************************************
*
* Functional description
* Get a record from the host program.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_REQ request;
WHY_TRA transaction;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*req_handle, request, HANDLE_request, isc_bad_req_handle);
FIND_TRANSACTION(*tra_handle, transaction, request->parent, isc_bad_trans_handle);
subsystem_enter();
if (CALL(PROC_START, request->implementation) (status,
&request->handle,
&transaction->handle,
level))
return error(status, local);
RETURN_SUCCESS;
}
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE GDS_START_MULTIPLE(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * tra_handle,
2003-11-06 04:03:36 +01:00
SSHORT count,
// TEB * vector)
void* vec)
{
/**************************************
*
* g d s _ $ s t a r t _ m u l t i p l e
*
**************************************
*
* Functional description
* Start a transaction.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status, *s;
ISC_STATUS_ARRAY local, temp;
WHY_TRA transaction, sub, *ptr;
WHY_DBB database;
USHORT n;
2003-11-06 04:03:36 +01:00
TEB* vector = (TEB*) vec;
GET_STATUS;
NULL_CHECK(tra_handle, isc_bad_trans_handle, HANDLE_transaction);
transaction = NULL;
subsystem_enter();
for (n = 0, ptr = &transaction; n < count;
n++, ptr = &(*ptr)->next, vector++) {
2004-05-03 01:06:37 +02:00
database = WHY_translate_handle(*vector->teb_database);
if (!database || database->type != HANDLE_database) {
s = status;
*s++ = isc_arg_gds;
*s++ = isc_bad_db_handle;
*s = isc_arg_end;
return error(status, local);
}
if (CALL(PROC_START_TRANSACTION, database->implementation) (status,
ptr,
1,
2004-05-03 01:06:37 +02:00
&database->handle,
vector->teb_tpb_length,
vector->teb_tpb))
{
while (sub = transaction) {
transaction = sub->next;
CALL(PROC_ROLLBACK, sub->implementation) (temp, &sub->handle);
release_handle(sub);
}
return error(status, local);
}
sub = allocate_handle( database->implementation,
*ptr,
HANDLE_transaction);
if (!sub)
{
/* No memory. Make a half-hearted attempt to rollback all sub-transactions. */
CALL(PROC_ROLLBACK, database->implementation) (temp, ptr);
*ptr = 0;
while (sub = transaction) {
transaction = sub->next;
CALL(PROC_ROLLBACK, sub->implementation) (temp, &sub->handle);
release_handle(sub);
}
status[0] = isc_arg_gds;
status[1] = isc_virmemexh;
status[2] = isc_arg_end;
return error(status, local);
}
sub->parent = database;
*ptr = sub;
}
if (transaction->next)
{
sub = allocate_handle(SUBSYSTEMS, (class Jrd::jrd_tra *)0, HANDLE_transaction);
if (!sub)
{
/* No memory. Make a half-hearted attempt to rollback all sub-transactions. */
while (sub = transaction) {
transaction = sub->next;
CALL(PROC_ROLLBACK, sub->implementation) (temp, &sub->handle);
release_handle(sub);
}
status[0] = isc_arg_gds;
status[1] = isc_virmemexh;
status[2] = isc_arg_end;
return error(status, local);
}
sub->next = transaction;
2004-05-03 01:06:37 +02:00
*tra_handle = sub->public_handle;
}
else
2004-05-03 01:06:37 +02:00
*tra_handle = transaction->public_handle;
RETURN_SUCCESS;
}
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE_VARARG GDS_START_TRANSACTION(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * tra_handle,
SSHORT count, ...)
{
/**************************************
*
* g d s _ $ s t a r t _ t r a n s a c t i o n
*
**************************************
*
* Functional description
* Start a transaction.
*
**************************************/
TEB tebs[16], *teb, *end;
2003-04-10 08:49:16 +02:00
ISC_STATUS status;
va_list ptr;
if (count <= FB_NELEM(tebs))
teb = tebs;
else
teb = (TEB *) alloc((SLONG) (sizeof(struct teb) * count));
if (!teb) {
user_status[0] = isc_arg_gds;
user_status[1] = status = isc_virmemexh;
user_status[2] = isc_arg_end;
return status;
}
end = teb + count;
va_start(ptr, count);
for (; teb < end; teb++) {
2004-05-03 01:06:37 +02:00
teb->teb_database = va_arg(ptr, FB_API_HANDLE*);
teb->teb_tpb_length = va_arg(ptr, int);
teb->teb_tpb = va_arg(ptr, UCHAR *);
}
va_end(ptr);
teb = end - count;
status = GDS_START_MULTIPLE(user_status, tra_handle, count, teb);
if (teb != tebs)
free_block(teb);
return status;
}
2003-12-03 09:19:24 +01:00
ISC_STATUS API_ROUTINE GDS_TRANSACT_REQUEST(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* tra_handle,
USHORT blr_length,
2003-12-03 09:19:24 +01:00
const SCHAR* blr,
USHORT in_msg_length,
2003-12-03 09:19:24 +01:00
SCHAR* in_msg,
USHORT out_msg_length,
2003-12-03 09:19:24 +01:00
SCHAR* out_msg)
{
/**************************************
*
* i s c _ t r a n s a c t _ r e q u e s t
*
**************************************
*
* Functional description
* Execute a procedure.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_ATT dbb;
WHY_TRA transaction;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*db_handle, dbb, HANDLE_database, isc_bad_db_handle);
TRANSLATE_HANDLE(*tra_handle, transaction, HANDLE_transaction, isc_bad_trans_handle);
subsystem_enter();
if (CALL(PROC_TRANSACT_REQUEST, dbb->implementation) (status,
&dbb->handle,
&transaction->
handle, blr_length,
blr, in_msg_length,
in_msg,
out_msg_length,
out_msg))
{
return error(status, local);
}
RETURN_SUCCESS;
}
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE gds__transaction_cleanup(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * tra_handle,
TransactionCleanupRoutine *routine,
void* arg)
{
/**************************************
*
* g d s _ $ t r a n s a c t i o n _ c l e a n u p
*
**************************************
*
* Functional description
* Register a transaction specific cleanup handler.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status, *s;
ISC_STATUS_ARRAY local;
WHY_TRA transaction;
CLEAN clean;
GET_STATUS;
2004-05-03 01:06:37 +02:00
transaction = WHY_translate_handle(*tra_handle);
if (!transaction || transaction->type != HANDLE_transaction) {
s = status;
*s++ = isc_arg_gds;
*s++ = isc_bad_db_handle;
*s = isc_arg_end;
return error2(status, local);
}
/* Only add the cleanup handler if the transaction doesn't already know
about it. */
for (clean = transaction->cleanup; clean; clean = clean->clean_next)
{
if (clean->TransactionRoutine == routine && clean->clean_arg == arg)
{
break;
}
}
if (!clean)
{
if (clean = (CLEAN) alloc((SLONG) sizeof(struct clean)))
{
#ifdef DEBUG_GDS_ALLOC
/* If client doesn't commit/rollback/detach
or drop, this could be left unfreed. */
gds_alloc_flag_unfreed((void *) clean);
#endif
clean->clean_next = transaction->cleanup;
transaction->cleanup = clean;
clean->TransactionRoutine = routine;
clean->clean_arg = arg;
}
else {
status[0] = isc_arg_gds;
status[1] = isc_virmemexh;
status[2] = isc_arg_end;
return error2(status, local);
}
}
status[0] = isc_arg_gds;
status[1] = FB_SUCCESS;
status[2] = isc_arg_end;
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
ISC_STATUS API_ROUTINE GDS_TRANSACTION_INFO(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* tra_handle,
SSHORT item_length,
const SCHAR* items,
SSHORT buffer_length,
UCHAR* buffer)
{
/**************************************
*
* g d s _ $ t r a n s a c t i o n _ i n f o
*
**************************************
*
* Functional description
* Provide information on transaction object.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_TRA transaction, sub;
SSHORT buffer_len, item_len;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*tra_handle, transaction, HANDLE_transaction, isc_bad_trans_handle);
subsystem_enter();
if (transaction->implementation != SUBSYSTEMS) {
if (CALL(PROC_TRANSACTION_INFO, transaction->implementation) (status,
&transaction->
handle,
item_length,
items,
buffer_length,
buffer))
return error(status, local);
}
else {
item_len = item_length;
buffer_len = buffer_length;
for (sub = transaction->next; sub; sub = sub->next) {
if (CALL(PROC_TRANSACTION_INFO, sub->implementation) (status,
&sub->
handle,
item_len,
items,
buffer_len,
buffer))
return error(status, local);
UCHAR* ptr = buffer;
const UCHAR* const end = buffer + buffer_len;
2003-11-08 17:40:17 +01:00
while (ptr < end && *ptr == isc_info_tra_id)
ptr += 3 + gds__vax_integer(ptr + 1, 2);
2003-11-08 17:40:17 +01:00
if (ptr >= end || *ptr != isc_info_end) {
RETURN_SUCCESS;
}
buffer_len = end - ptr;
buffer = ptr;
}
}
RETURN_SUCCESS;
}
2003-04-10 08:49:16 +02:00
ISC_STATUS API_ROUTINE GDS_UNWIND(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE * req_handle,
SSHORT level)
{
/**************************************
*
* g d s _ $ u n w i n d
*
**************************************
*
* Functional description
* Unwind a running request. This is potentially nasty since it can be called
* asynchronously.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_REQ request;
GET_STATUS;
2004-05-03 01:06:37 +02:00
TRANSLATE_HANDLE(*req_handle, request, HANDLE_request, isc_bad_req_handle);
subsystem_enter();
if (CALL(PROC_UNWIND, request->implementation) (status,
&request->handle,
level))
return error(status, local);
RETURN_SUCCESS;
}
2004-02-28 20:32:46 +01:00
#ifdef DEBUG_GDS_ALLOC
static SCHAR *alloc_debug(SLONG length, const char* file, int line)
#else
static SCHAR *alloc(SLONG length)
2004-02-28 20:32:46 +01:00
#endif
{
/**************************************
*
* a l l o c
*
**************************************
*
* Functional description
* Allocate some memory.
*
**************************************/
SCHAR *block;
2004-02-28 20:32:46 +01:00
#ifdef DEBUG_GDS_ALLOC
if (block = reinterpret_cast<SCHAR *>(gds__alloc_debug((SLONG) (sizeof(SCHAR) * length), file, line)))
#else
2002-12-22 14:32:04 +01:00
if (block = reinterpret_cast<SCHAR *>(gds__alloc((SLONG) (sizeof(SCHAR) * length))))
2004-02-28 20:32:46 +01:00
#endif
memset(block, 0, length);
return block;
}
static ISC_STATUS bad_handle(ISC_STATUS * user_status,
ISC_STATUS code)
{
/**************************************
*
* b a d _ h a n d l e
*
**************************************
*
* Functional description
* Generate an error for a bad handle.
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *s, *status;
ISC_STATUS_ARRAY local;
GET_STATUS;
s = status;
*s++ = isc_arg_gds;
*s++ = code;
*s = isc_arg_end;
return error2(status, local);
}
#ifdef DEV_BUILD
2004-05-09 07:48:33 +02:00
static void check_status_vector(const ISC_STATUS* status,
ISC_STATUS expected)
{
/**************************************
*
* c h e c k _ s t a t u s _ v e c t o r
*
**************************************
*
* Functional description
* Validate that a status vector looks valid.
*
**************************************/
#define SV_MSG(x) { gds__log ("%s %d check_status_vector: %s", __FILE__, __LINE__, (x)); BREAKPOINT (__LINE__); }
2004-05-09 07:48:33 +02:00
const ISC_STATUS* s = status;
if (!s) {
SV_MSG("Invalid status vector");
return;
}
if (*s != isc_arg_gds) {
SV_MSG("Must start with isc_arg_gds");
return;
}
/* Vector [2] could either end the vector, or start a warning
in either case the status vector is a success */
if ((expected == FB_SUCCESS)
&& (s[1] != FB_SUCCESS
|| (s[2] != isc_arg_end && s[2] != isc_arg_gds
&& s[2] !=
2004-05-09 07:48:33 +02:00
isc_arg_warning)))
{
SV_MSG("Success vector expected");
}
ULONG length;
while (*s != isc_arg_end) {
2004-05-09 07:48:33 +02:00
const ISC_STATUS code = *s++;
switch (code) {
case isc_arg_warning:
case isc_arg_gds:
/* The next element must either be 0 (indicating no error) or a
* valid isc error message, let's check */
if (*s && (*s & ISC_MASK) != ISC_MASK) {
if (code == isc_arg_warning) {
SV_MSG("warning code not a valid ISC message");
}
else {
SV_MSG("error code not a valid ISC message");
}
}
/* If the error code is valid, then I better be able to retrieve a
* proper facility code from it ... let's find out */
if (*s && (*s & ISC_MASK) == ISC_MASK) {
2004-05-09 07:48:33 +02:00
bool found = false;
2004-05-09 07:48:33 +02:00
const struct _facilities* facs = facilities;
const int fac_code = GET_FACILITY(*s);
while (facs->facility) {
if (facs->fac_code == fac_code) {
2004-05-09 07:48:33 +02:00
found = true;
break;
}
facs++;
}
if (!found)
if (code == isc_arg_warning) {
SV_MSG
("warning code does not contain a valid facility");
}
else {
SV_MSG
("error code does not contain a valid facility");
}
}
s++;
break;
case isc_arg_interpreted:
case isc_arg_string:
2004-05-09 07:48:33 +02:00
length = strlen((const char*) *s);
/* This check is heuristic, not deterministic */
if (length > 1024 - 1)
SV_MSG("suspect length value");
2004-05-09 07:48:33 +02:00
if (*((const UCHAR *) * s) == 0xCB)
SV_MSG("string in freed memory");
s++;
break;
case isc_arg_cstring:
length = (ULONG) * s;
s++;
/* This check is heuristic, not deterministic */
/* Note: This can never happen anyway, as the length is coming
from a byte value */
if (length > 1024 - 1)
SV_MSG("suspect length value");
2004-05-09 07:48:33 +02:00
if (*((const UCHAR *) * s) == 0xCB)
SV_MSG("string in freed memory");
s++;
break;
case isc_arg_number:
case isc_arg_vms:
case isc_arg_unix:
case isc_arg_win32:
s++;
break;
default:
SV_MSG("invalid status code");
return;
}
2004-05-09 07:48:33 +02:00
if ((s - status) >= ISC_STATUS_LENGTH)
SV_MSG("vector too long");
}
#undef SV_MSG
}
#endif
// Make this repetitive block a function.
// Call all cleanup routines registered with the transaction.
2004-02-28 20:32:46 +01:00
void WHY_cleanup_transaction(WHY_TRA transaction)
{
for (clean* cln = transaction->cleanup; cln; cln = transaction->cleanup)
{
transaction->cleanup = cln->clean_next;
2004-05-03 01:06:37 +02:00
cln->TransactionRoutine(transaction->public_handle, cln->clean_arg);
free_block(cln);
}
}
2004-05-09 07:48:33 +02:00
static ISC_STATUS error(const ISC_STATUS* user_status,
const ISC_STATUS* local)
{
/**************************************
*
* e r r o r
*
**************************************
*
* Functional description
* An error returned has been trapped. If the user specified
* a status vector, return a status code. Otherwise print the
* error code(s) and abort.
*
**************************************/
subsystem_exit();
return error2(user_status, local);
}
2004-05-09 07:48:33 +02:00
static ISC_STATUS error2(const ISC_STATUS* user_status,
const ISC_STATUS* local)
{
/**************************************
*
* e r r o r 2
*
**************************************
*
* Functional description
* An error returned has been trapped. If the user specified
* a status vector, return a status code. Otherwise print the
* error code(s) and abort.
*
**************************************/
CHECK_STATUS(user_status);
#ifdef SUPERSERVER
return user_status[1];
#else
if (user_status != local)
return user_status[1];
gds__print_status(user_status);
exit((int) user_status[1]);
return FB_SUCCESS;
#endif
}
#ifndef REQUESTER
2003-12-22 11:00:59 +01:00
static void event_ast(void* buffer_void,
USHORT length,
2003-12-22 11:00:59 +01:00
const UCHAR* items)
{
/**************************************
*
* e v e n t _ a s t
*
**************************************
*
* Functional description
* We're had an event complete.
*
**************************************/
memcpy(buffer_void, items, length);
ISC_event_post(why_event);
}
#endif
#ifndef REQUESTER
2004-05-12 21:39:17 +02:00
static void exit_handler(event_t* why_eventL)
{
/**************************************
*
* e x i t _ h a n d l e r
*
**************************************
*
* Functional description
* Cleanup shared image.
*
**************************************/
#ifdef WIN_NT
2004-05-12 21:39:17 +02:00
CloseHandle((void *) why_eventL->event_handle);
#endif
why_initialized = FALSE;
why_enabled = 0;
#if !(defined REQUESTER || defined SUPERCLIENT || defined SUPERSERVER)
isc_enter_count = 0;
subsystem_usage = 0;
subsystem_FPE_reset = FPE_RESET_INIT_ONLY;
#endif
}
#endif
static WHY_TRA find_transaction(WHY_DBB dbb,
WHY_TRA transaction)
{
/**************************************
*
* f i n d _ t r a n s a c t i o n
*
**************************************
*
* Functional description
* Find the element of a possible multiple database transaction
* that corresponds to the current database.
*
**************************************/
for (; transaction; transaction = transaction->next)
if (transaction->parent == dbb)
return transaction;
return NULL;
}
static void free_block(void* block)
{
/**************************************
*
* f r e e _ b l o c k
*
**************************************
*
* Functional description
* Release some memory.
*
**************************************/
gds__free((SLONG *) block);
}
static int get_database_info(ISC_STATUS * status,
WHY_TRA transaction,
TEXT ** ptr)
{
/**************************************
*
* g e t _ d a t a b a s e _ i n f o
*
**************************************
*
* Functional description
* Get the full database pathname
* and put it in the transaction
* description record.
*
**************************************/
// Look at the changed code: we don't support here more than 254 bytes in the path
// so it's better to truncate or we'll have corrupt data in the trans desc record:
// the length in one byte would wrap and we would copy more bytes that expected.
// Our caller (prepare) assumed each call consumes at most 256 bytes (item, len, data)
// hence if we don't check here, we have a B.O.
TEXT* p = *ptr;
2004-09-26 09:45:19 +02:00
WHY_DBB database = transaction->parent;
*p++ = TDR_DATABASE_PATH;
2006-02-23 07:52:25 +01:00
const TEXT* q = database->db_path;
size_t len = strlen(q);
if (len > 254)
len = 254;
*p++ = (TEXT) len;
memcpy(p, q, len);
*ptr = p + len;
return FB_SUCCESS;
}
static const PTR get_entrypoint(int proc,
int implementation)
{
/**************************************
*
* g e t _ e n t r y p o i n t
*
**************************************
*
* Functional description
* Lookup entrypoint for procedure.
*
**************************************/
ENTRY *ent = entrypoints + implementation * PROC_count + proc;
2003-02-13 11:11:35 +01:00
const PTR entrypoint = ent->address;
if (entrypoint)
{
return entrypoint;
}
#ifndef SUPERCLIENT
2004-11-30 07:08:26 +01:00
const TEXT* image = images[implementation].path;
const TEXT* name = ent->name;
if (!name)
{
name = generic[proc];
}
if (image && name)
{
2004-11-30 07:08:26 +01:00
PTR entry = (PTR) Jrd::Module::lookup(image, name);
if (entry)
{
ent->address = entry;
return entry;
}
}
#endif
return &no_entrypoint;
}
static SCHAR *get_sqlda_buffer(SCHAR * buffer,
USHORT local_buffer_length,
XSQLDA * sqlda,
USHORT dialect,
USHORT * buffer_length)
{
/**************************************
*
* g e t _ s q l d a _ b u f f e r
*
**************************************
*
* Functional description
* Get a buffer that is large enough to store
* the info items relating to an SQLDA.
*
**************************************/
USHORT n_variables;
SLONG length;
USHORT sql_dialect;
/* If dialect / 10 == 0, then it has not been combined with the
parser version for a prepare statement. If it has been combined, then
the dialect needs to be pulled out to compare to DIALECT_xsqlda
*/
if ((sql_dialect = dialect / 10) == 0)
sql_dialect = dialect;
if (!sqlda)
n_variables = 0;
else if (sql_dialect >= DIALECT_xsqlda)
n_variables = sqlda->sqln;
else
n_variables = ((SQLDA *) sqlda)->sqln;
length = 32 + n_variables * 172;
*buffer_length = (USHORT)((length > 65500L) ? 65500L : length);
if (*buffer_length > local_buffer_length)
buffer = alloc((SLONG) * buffer_length);
return buffer;
}
2003-04-10 08:49:16 +02:00
static ISC_STATUS get_transaction_info(ISC_STATUS * status,
WHY_TRA transaction,
TEXT ** ptr)
{
/**************************************
*
* g e t _ t r a n s a c t i o n _ i n f o
*
**************************************
*
* Functional description
* Put a transaction's id into the transaction
* description record.
*
**************************************/
2004-09-26 09:45:19 +02:00
TEXT buffer[16];
2004-09-26 09:45:19 +02:00
TEXT* p = *ptr;
if (CALL(PROC_TRANSACTION_INFO, transaction->implementation) (status,
&transaction->
handle,
sizeof
(prepare_tr_info),
prepare_tr_info,
sizeof
(buffer),
2004-09-26 09:45:19 +02:00
buffer))
{
CHECK_STATUS(status);
return status[1];
}
const TEXT* q = buffer + 3;
*p++ = TDR_TRANSACTION_ID;
const USHORT length = (USHORT)gds__vax_integer(reinterpret_cast<UCHAR*>(buffer + 1), 2);
*p++ = length; // Warning: USHORT coerced to char
2006-02-23 07:52:25 +01:00
memcpy(p, q, length);
*ptr = p + length;
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
2003-04-10 08:49:16 +02:00
static void iterative_sql_info(ISC_STATUS * user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* stmt_handle,
SSHORT item_length,
2003-02-13 11:11:35 +01:00
const SCHAR * items,
SSHORT buffer_length,
SCHAR* buffer,
USHORT dialect,
XSQLDA * sqlda)
{
/**************************************
*
* i t e r a t i v e _ s q l _ i n f o
*
**************************************
*
* Functional description
* Turn an sql info buffer into an SQLDA. If the info
* buffer was incomplete, make another request, beginning
* where the previous info call left off.
*
**************************************/
USHORT last_index;
SCHAR new_items[32];
while (UTLD_parse_sql_info( user_status,
dialect,
buffer,
sqlda,
&last_index) && last_index)
{
char* p = new_items;
2003-11-08 17:40:17 +01:00
*p++ = isc_info_sql_sqlda_start;
*p++ = 2;
*p++ = last_index;
*p++ = last_index >> 8;
memcpy(p, items, (int) item_length);
p += item_length;
if (GDS_DSQL_SQL_INFO( user_status,
stmt_handle,
(SSHORT) (p - new_items),
new_items,
buffer_length,
buffer))
{
break;
}
}
}
static ISC_STATUS open_blob(ISC_STATUS* user_status,
2004-05-03 01:06:37 +02:00
FB_API_HANDLE* db_handle,
FB_API_HANDLE* tra_handle,
FB_API_HANDLE* public_blob_handle,
SLONG* blob_id,
USHORT bpb_length,
const UCHAR* bpb,
SSHORT proc,
SSHORT proc2)
{
/**************************************
*
* o p e n _ b l o b
*
**************************************
*
* Functional description
* Open an existing blob (extended edition).
*
**************************************/
2003-04-16 12:18:51 +02:00
ISC_STATUS *status;
ISC_STATUS_ARRAY local;
WHY_TRA transaction;
GET_STATUS;
2004-05-03 01:06:37 +02:00
NULL_CHECK(public_blob_handle, isc_bad_segstr_handle, HANDLE_blob);
2004-05-03 01:06:37 +02:00
WHY_ATT dbb;
TRANSLATE_HANDLE(*db_handle, dbb, HANDLE_database, isc_bad_db_handle);
FIND_TRANSACTION(*tra_handle, transaction, dbb, isc_bad_trans_handle);
subsystem_enter();
2003-12-22 11:00:59 +01:00
USHORT flags = 0;
USHORT from, to;
gds__parse_bpb(bpb_length, bpb, &from, &to);
2004-05-03 01:06:37 +02:00
WHY_BLB blob_handle = NULL;
if (get_entrypoint(proc2, dbb->implementation) != no_entrypoint &&
CALL(proc2, dbb->implementation) (status,
&dbb->handle,
&transaction->handle,
2004-05-03 01:06:37 +02:00
&blob_handle,
blob_id,
bpb_length,
bpb) != isc_unavailable)
{
flags = 0;
}
else if (!to || from == to)
CALL(proc, dbb->implementation) (status,
&dbb->handle,
&transaction->handle,
2004-05-03 01:06:37 +02:00
&blob_handle, blob_id);
if (status[1]) {
return error(status, local);
}
2004-05-03 01:06:37 +02:00
WHY_BLB blob = allocate_handle(dbb->implementation, blob_handle, HANDLE_blob);
if (!blob)
{
/* No memory. Make a half-hearted attempt to cancel the blob. */
CALL(PROC_CANCEL_BLOB, dbb->implementation) (status, blob_handle);
status[0] = isc_arg_gds;
status[1] = isc_virmemexh;
status[2] = isc_arg_end;
return error(status, local);
}
2004-05-03 01:06:37 +02:00
*public_blob_handle = blob->public_handle;
blob->flags |= flags;
blob->parent = dbb;
blob->next = dbb->blobs;
dbb->blobs = blob;
RETURN_SUCCESS;
}
2003-11-06 04:03:36 +01:00
extern "C" {
2003-04-10 08:49:16 +02:00
static ISC_STATUS no_entrypoint(ISC_STATUS * user_status, ...)
{
/**************************************
*
* n o _ e n t r y p o i n t
*
**************************************
*
* Functional description
* No_entrypoint is called if there is not entrypoint for a given routine.
*
**************************************/
*user_status++ = isc_arg_gds;
*user_status++ = isc_unavailable;
*user_status = isc_arg_end;
return isc_unavailable;
}
2003-11-06 04:03:36 +01:00
} // extern "C"
static ISC_STATUS prepare(ISC_STATUS * status,
WHY_TRA transaction)
{
/**************************************
*
* p r e p a r e
*
**************************************
*
* Functional description
* Perform the first phase of a two-phase commit
* for a multi-database transaction.
*
**************************************/
WHY_TRA sub;
TEXT tdr_buffer[1024];
int length = 0, transcount = 0;
for (sub = transaction->next; sub; sub = sub->next)
{
length += 256;
++transcount;
}
// To do: use transcount to check the maximum allowed dbs in a two phase commit.
TEXT host[64];
ISC_get_host(host, sizeof(host));
const size_t hostlen = strlen(host);
length += hostlen + 3; // TDR_version + TDR_host_site + UCHAR(strlen(host))
2003-12-22 11:00:59 +01:00
TEXT* const description = (length > sizeof(tdr_buffer)) ?
(TEXT *) gds__alloc(length) : tdr_buffer;
/* build a transaction description record containing
the host site and database/transaction
information for the target databases. */
2003-12-22 11:00:59 +01:00
TEXT* p = description;
if (!p) {
status[0] = isc_arg_gds;
status[1] = isc_virmemexh;
status[2] = isc_arg_end;
CHECK_STATUS(status);
return status[1];
}
*p++ = TDR_VERSION;
*p++ = TDR_HOST_SITE;
*p++ = UCHAR(hostlen);
memcpy(p, host, hostlen);
p += hostlen;
/* Get database and transaction stuff for each sub-transaction */
for (sub = transaction->next; sub; sub = sub->next) {
get_database_info(status, sub, &p);
get_transaction_info(status, sub, &p);
}
/* So far so good -- prepare each sub-transaction */
length = p - description;
for (sub = transaction->next; sub; sub = sub->next)
if (CALL(PROC_PREPARE, sub->implementation) (status,
&sub->handle,
length, description))
{
if (description != tdr_buffer) {
free_block(description);
}
CHECK_STATUS(status);
return status[1];
}
if (description != tdr_buffer)
free_block(description);
CHECK_STATUS_SUCCESS(status);
return FB_SUCCESS;
}
static void why_priv_gds__free_if_set(void* pMem)
{
if (pMem) {
gds__free(pMem);
}
}
2004-02-02 12:02:12 +01:00
static void release_dsql_support(sqlda_sup* dasup)
{
/**************************************
*
* r e l e a s e _ d s q l _ s u p p o r t
*
**************************************
*
* Functional description
* Release some memory.
*
**************************************/
if (!dasup) {
return;
}
/* for C++, add "dasup::" before "dasup_clause" */
2004-02-02 12:02:12 +01:00
sqlda_sup::dasup_clause* pClauses = dasup->dasup_clauses;
why_priv_gds__free_if_set(pClauses[DASUP_CLAUSE_bind].dasup_blr);
why_priv_gds__free_if_set(pClauses[DASUP_CLAUSE_select].dasup_blr);
why_priv_gds__free_if_set(pClauses[DASUP_CLAUSE_bind].dasup_msg);
why_priv_gds__free_if_set(pClauses[DASUP_CLAUSE_select].dasup_msg);
free_block(dasup);
}
static void release_handle(WHY_HNDL handle)
{
/**************************************
*
* r e l e a s e _ h a n d l e
*
**************************************
*
* Functional description
* Release unused and unloved handle.
*
**************************************/
2004-05-03 01:06:37 +02:00
WHY_free_handle(handle->public_handle);
}
2003-04-10 08:49:16 +02:00
static void save_error_string(ISC_STATUS * status)
{
/**************************************
*
* s a v e _ e r r o r _ s t r i n g
*
**************************************
*
* Functional description
* This is need because there are cases
* where the memory allocated for strings
* in the status vector is freed prior to
* surfacing them to the user. This is an
* attempt to save off 1 string to surface to
* the user. Any other strings will be set to
* a standard <Unknown> string.
*
**************************************/
2003-11-04 00:59:24 +01:00
fb_assert(status != NULL);
2003-12-22 11:00:59 +01:00
TEXT* p = glbstr1;
ULONG len = sizeof(glbstr1) - 1;
while (*status != isc_arg_end)
{
2003-12-22 11:00:59 +01:00
ULONG l;
switch (*status++)
{
case isc_arg_cstring:
l = (ULONG) * status;
if (l < len)
{
status++; /* Length is unchanged */
/*
* This strncpy should really be a memcpy
*/
strncpy(p, reinterpret_cast<char*>(*status), l);
2003-04-10 08:49:16 +02:00
*status++ = (ISC_STATUS) p; /* string in static memory */
p += l;
len -= l;
}
else {
2003-04-10 08:49:16 +02:00
*status++ = (ISC_STATUS) strlen(glbunknown);
*status++ = (ISC_STATUS) glbunknown;
}
break;
case isc_arg_interpreted:
case isc_arg_string:
l = (ULONG) strlen(reinterpret_cast<char*>(*status)) + 1;
if (l < len)
{
strncpy(p, reinterpret_cast<char*>(*status), l);
2003-04-10 08:49:16 +02:00
*status++ = (ISC_STATUS) p; /* string in static memory */
p += l;
len -= l;
}
else
{
2003-04-10 08:49:16 +02:00
*status++ = (ISC_STATUS) glbunknown;
}
break;
default:
2003-11-04 00:59:24 +01:00
fb_assert(FALSE);
case isc_arg_gds:
case isc_arg_number:
case isc_arg_vms:
case isc_arg_unix:
case isc_arg_win32:
status++; /* Skip parameter */
break;
}
}
}
static void subsystem_enter(void)
{
/**************************************
*
* s u b s y s t e m _ e n t e r
*
**************************************
*
* Functional description
* Enter subsystem.
*
**************************************/
#ifdef EMBEDDED
THD_INIT;
#endif
2004-05-15 02:58:46 +02:00
THREAD_ENTER();
#if !(defined REQUESTER || defined SUPERCLIENT || defined SUPERSERVER)
isc_enter_count++;
if (subsystem_usage == 0 ||
(subsystem_FPE_reset &
(FPE_RESET_NEXT_API_CALL | FPE_RESET_ALL_API_CALL)))
{
ISC_enter();
subsystem_FPE_reset &= ~FPE_RESET_NEXT_API_CALL;
}
#endif
#ifdef DEBUG_FPE_HANDLING
{
/* It's difficult to make a FPE to occur inside the engine - for debugging
* just force one to occur every-so-often. */
static ULONG counter = 0;
if (((counter++) % 10) == 0)
{
2004-04-29 00:43:34 +02:00
fprintf(stderr, "Forcing FPE to occur within engine\n");
kill(getpid(), SIGFPE);
}
}
#endif /* DEBUG_FPE_HANDLING */
}
static void subsystem_exit(void)
{
/**************************************
*
* s u b s y s t e m _ e x i t
*
**************************************
*
* Functional description
* Exit subsystem.
*
**************************************/
#if !(defined REQUESTER || defined SUPERCLIENT || defined SUPERSERVER)
if (subsystem_usage == 0 ||
(subsystem_FPE_reset &
(FPE_RESET_NEXT_API_CALL | FPE_RESET_ALL_API_CALL)))
{
ISC_exit();
}
isc_enter_count--;
#endif
2004-05-15 02:58:46 +02:00
THREAD_EXIT();
}
#if defined (SERVER_SHUTDOWN) && !defined (SUPERCLIENT) && !defined (REQUESTER)
BOOLEAN WHY_set_shutdown(BOOLEAN flag)
{
/**************************************
*
* W H Y _ s e t _ s h u t d o w n
*
**************************************
*
* Functional description
* Set shutdown_flag to either TRUE or FALSE.
* TRUE = accept new connections
* FALSE= refuse new connections
* Returns the prior state of the flag (server).
*
**************************************/
const BOOLEAN old_flag = shutdown_flag;
shutdown_flag = flag;
return old_flag;
}
BOOLEAN WHY_get_shutdown()
{
/**************************************
*
* W H Y _ g e t _ s h u t d o w n
*
**************************************
*
* Functional description
* Returns the current value of shutdown_flag.
*
**************************************/
return shutdown_flag;
}
#endif /* SERVER_SHUTDOWN && !SUPERCLIENT && !REQUESTER */