2001-05-23 15:26:42 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: JRD Remote Server
|
2003-09-29 14:43:14 +02:00
|
|
|
* MODULE: server.cpp
|
2001-05-23 15:26:42 +02:00
|
|
|
* DESCRIPTION: Remote server
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Interbase Public
|
|
|
|
* License Version 1.0 (the "License"); you may not use this file
|
|
|
|
* except in compliance with the License. You may obtain a copy
|
|
|
|
* of the License at http://www.Inprise.com/IPL.html
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an
|
|
|
|
* "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either express
|
|
|
|
* or implied. See the License for the specific language governing
|
|
|
|
* rights and limitations under the License.
|
|
|
|
*
|
|
|
|
* The Original Code was created by Inprise Corporation
|
|
|
|
* and its predecessors. Portions created by Inprise Corporation are
|
|
|
|
* Copyright (C) Inprise Corporation.
|
|
|
|
*
|
|
|
|
* All Rights Reserved.
|
|
|
|
* Contributor(s): ______________________________________.
|
2002-06-29 10:49:39 +02:00
|
|
|
*
|
|
|
|
* 2002.02.27 Claudio Valderrama: Fix SF Bug #509995.
|
|
|
|
*
|
2002-10-30 07:40:58 +01:00
|
|
|
* 2002.10.29 Sean Leyne - Removed obsolete "Netware" port
|
|
|
|
*
|
2002-10-31 06:06:02 +01:00
|
|
|
* 2002.10.30 Sean Leyne - Removed support for obsolete "PC_PLATFORM" define
|
|
|
|
*
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
|
|
|
|
2001-07-30 01:43:24 +02:00
|
|
|
#include "firebird.h"
|
2004-03-22 12:38:23 +01:00
|
|
|
#include "../jrd/common.h"
|
2004-04-29 00:36:29 +02:00
|
|
|
#include <stdio.h>
|
2001-05-23 15:26:42 +02:00
|
|
|
#include <string.h>
|
2003-11-08 17:40:17 +01:00
|
|
|
#include "../jrd/ibase.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/gdsassert.h"
|
|
|
|
#include "../remote/remote.h"
|
|
|
|
#include "../jrd/thd.h"
|
|
|
|
#include "../jrd/isc.h"
|
|
|
|
#include "../jrd/license.h"
|
2003-02-14 03:24:44 +01:00
|
|
|
#include "../jrd/jrd_time.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../remote/merge_proto.h"
|
|
|
|
#include "../remote/parse_proto.h"
|
|
|
|
#include "../remote/remot_proto.h"
|
|
|
|
#include "../remote/serve_proto.h"
|
|
|
|
#ifdef WIN_NT
|
2003-07-15 01:29:45 +02:00
|
|
|
#include "../remote/os/win32/cntl_proto.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include <stdlib.h>
|
|
|
|
#endif
|
|
|
|
#include "../jrd/gds_proto.h"
|
|
|
|
#include "../jrd/isc_proto.h"
|
|
|
|
#include "../jrd/isc_s_proto.h"
|
|
|
|
#include "../jrd/sch_proto.h"
|
2004-05-18 00:30:09 +02:00
|
|
|
#include "../jrd/thread_proto.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/why_proto.h"
|
2004-02-13 11:17:50 +01:00
|
|
|
#include "../common/classes/semaphore.h"
|
2004-11-26 02:01:27 +01:00
|
|
|
#include "../common/classes/ClumpletWriter.h"
|
2003-06-30 13:26:38 +02:00
|
|
|
#ifdef DEBUG
|
2003-11-11 13:19:20 +01:00
|
|
|
#include "gen/iberror.h"
|
2003-06-30 13:26:38 +02:00
|
|
|
#endif
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef SUPERSERVER
|
2003-09-08 22:23:46 +02:00
|
|
|
#include "../jrd/os/isc_i_proto.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
#include "../remote/proto_proto.h" // xdr_protocol_overhead()
|
|
|
|
|
|
|
|
/** CHECK_HANDLE checks -
|
|
|
|
if the id passwd is within the vector bounds,
|
|
|
|
that the port_object corresponding to the id is not null,
|
|
|
|
if the port_object's type matches with the expected type
|
|
|
|
**/
|
|
|
|
|
|
|
|
#pragma FB_COMPILER_MESSAGE("What kind of crap is this?! FIX!")
|
|
|
|
|
2004-11-24 10:22:07 +01:00
|
|
|
#define CHECK_HANDLE(blk, cast, type, id, err) \
|
2001-05-23 15:26:42 +02:00
|
|
|
{ \
|
|
|
|
if (id >= port->port_object_vector->vec_count || \
|
|
|
|
!(blk = (cast) port->port_objects [id]) || \
|
|
|
|
((BLK) blk)->blk_type != (UCHAR) type) \
|
|
|
|
{ \
|
|
|
|
status_vector [0] = isc_arg_gds; \
|
|
|
|
status_vector [1] = err; \
|
|
|
|
status_vector [2] = isc_arg_end; \
|
2004-05-12 21:39:17 +02:00
|
|
|
return port->send_response(sendL, 0, 0, status_vector); \
|
2001-05-23 15:26:42 +02:00
|
|
|
} \
|
|
|
|
}
|
|
|
|
|
2004-11-24 10:22:07 +01:00
|
|
|
#define CHECK_HANDLE_MEMBER(blk, cast, type, id, err) \
|
2001-05-23 15:26:42 +02:00
|
|
|
{ \
|
|
|
|
if (id >= this->port_object_vector->vec_count || \
|
|
|
|
!(blk = (cast) this->port_objects [id]) || \
|
|
|
|
((BLK) blk)->blk_type != (UCHAR) type) \
|
|
|
|
{ \
|
|
|
|
status_vector [0] = isc_arg_gds; \
|
|
|
|
status_vector [1] = err; \
|
|
|
|
status_vector [2] = isc_arg_end; \
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector); \
|
2001-05-23 15:26:42 +02:00
|
|
|
} \
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-17 12:22:34 +02:00
|
|
|
const USHORT STMT_BLOB = 1;
|
|
|
|
const USHORT STMT_NO_BATCH = 2;
|
|
|
|
const USHORT STMT_OTHER = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-06-09 20:23:27 +02:00
|
|
|
typedef struct server_req_t
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
server_req_t* req_next;
|
|
|
|
server_req_t* req_chain;
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* req_port;
|
2001-05-23 15:26:42 +02:00
|
|
|
PACKET req_send;
|
|
|
|
PACKET req_receive;
|
2004-06-09 20:23:27 +02:00
|
|
|
} *SERVER_REQ;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-06-09 20:23:27 +02:00
|
|
|
typedef struct srvr
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
struct srvr* srvr_next;
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* srvr_parent_port;
|
2001-12-24 03:51:06 +01:00
|
|
|
enum rem_port_t srvr_port_type;
|
2001-05-23 15:26:42 +02:00
|
|
|
USHORT srvr_flags;
|
2004-06-09 20:23:27 +02:00
|
|
|
} *SRVR;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static bool accept_connection(rem_port*, P_CNCT*, PACKET*);
|
|
|
|
static ISC_STATUS allocate_statement(rem_port*, P_RLSE*, PACKET*);
|
2004-03-14 06:51:54 +01:00
|
|
|
#ifdef MULTI_THREAD
|
2001-05-23 15:26:42 +02:00
|
|
|
static SLONG append_request_chain(SERVER_REQ, SERVER_REQ*);
|
|
|
|
static SLONG append_request_next(SERVER_REQ, SERVER_REQ*);
|
2004-03-14 06:51:54 +01:00
|
|
|
#endif
|
2004-01-28 08:50:41 +01:00
|
|
|
static ISC_STATUS attach_database(rem_port*, P_OP, P_ATCH*, PACKET*);
|
2004-03-14 06:51:54 +01:00
|
|
|
#ifdef NOT_USED_OR_REPLACED
|
2004-01-28 08:50:41 +01:00
|
|
|
static void aux_connect(rem_port*, P_REQ*, PACKET*);
|
2004-03-14 06:51:54 +01:00
|
|
|
#endif
|
2004-01-28 08:50:41 +01:00
|
|
|
static void aux_request(rem_port*, P_REQ*, PACKET*);
|
|
|
|
static ISC_STATUS cancel_events(rem_port*, P_EVENT*, PACKET*);
|
2005-11-27 21:53:09 +01:00
|
|
|
static void addRemoteAddress(Firebird::ClumpletWriter&, UCHAR, const rem_port*);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef CANCEL_OPERATION
|
2004-01-28 08:50:41 +01:00
|
|
|
static void cancel_operation(rem_port*);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static bool check_request(rrq*, USHORT, USHORT);
|
2001-05-23 15:26:42 +02:00
|
|
|
static USHORT check_statement_type(RSR);
|
|
|
|
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
2002-11-19 13:40:54 +01:00
|
|
|
static REM_MSG dump_cache(rrq::rrq_repeat*);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static bool get_next_msg_no(rrq*, USHORT, USHORT*);
|
2004-05-03 01:06:37 +02:00
|
|
|
static RTR make_transaction(RDB, FB_API_HANDLE);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
static void release_blob(RBL);
|
|
|
|
static void release_event(RVNT);
|
2004-01-28 08:50:41 +01:00
|
|
|
static void release_request(rrq*);
|
2001-05-23 15:26:42 +02:00
|
|
|
static void release_statement(RSR*);
|
|
|
|
static void release_sql_request(RSR);
|
|
|
|
static void release_transaction(RTR);
|
|
|
|
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
2002-11-19 13:40:54 +01:00
|
|
|
static REM_MSG scroll_cache(rrq::rrq_repeat*, USHORT *, ULONG *);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
static void server_ast(void*, USHORT, const UCHAR*);
|
2003-04-10 12:31:28 +02:00
|
|
|
static void success(ISC_STATUS *);
|
2003-09-24 00:34:14 +02:00
|
|
|
#ifdef MULTI_THREAD
|
2004-06-08 15:41:08 +02:00
|
|
|
static THREAD_ENTRY_DECLARE loopThread(THREAD_ENTRY_PARAM);
|
2003-09-24 00:34:14 +02:00
|
|
|
#endif
|
2003-09-11 20:59:34 +02:00
|
|
|
static void zap_packet(PACKET*, bool);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
|
|
|
// static data - NOT THREAD SAFE!
|
|
|
|
|
|
|
|
static SLONG threads_waiting = 0;
|
|
|
|
static SLONG extra_threads = 0;
|
|
|
|
static SERVER_REQ request_que = NULL;
|
|
|
|
static SERVER_REQ free_requests = NULL;
|
|
|
|
static SERVER_REQ active_requests = NULL;
|
|
|
|
static SRVR servers;
|
|
|
|
|
2003-09-24 00:34:14 +02:00
|
|
|
#ifdef MULTI_THREAD
|
2003-09-16 22:59:45 +02:00
|
|
|
static Firebird::Semaphore requests_semaphore;
|
2003-09-24 00:34:14 +02:00
|
|
|
#endif
|
2003-09-16 22:59:45 +02:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-02-13 10:33:26 +01:00
|
|
|
static const UCHAR request_info[] =
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
2003-11-08 17:40:17 +01:00
|
|
|
isc_info_state,
|
|
|
|
isc_info_message_number,
|
|
|
|
isc_info_end
|
2001-05-23 15:26:42 +02:00
|
|
|
};
|
|
|
|
|
2003-02-13 10:33:26 +01:00
|
|
|
static const UCHAR sql_info[] =
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
2003-11-08 17:40:17 +01:00
|
|
|
isc_info_sql_stmt_type,
|
2001-05-23 15:26:42 +02:00
|
|
|
isc_info_sql_batch_fetch
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
#define GDS_DSQL_ALLOCATE isc_dsql_allocate_statement
|
|
|
|
#define GDS_DSQL_EXECUTE isc_dsql_execute2_m
|
|
|
|
#define GDS_DSQL_EXECUTE_IMMED isc_dsql_exec_immed3_m
|
|
|
|
#define GDS_DSQL_FETCH isc_dsql_fetch_m
|
|
|
|
#define GDS_DSQL_FREE isc_dsql_free_statement
|
|
|
|
#define GDS_DSQL_INSERT isc_dsql_insert_m
|
|
|
|
#define GDS_DSQL_PREPARE isc_dsql_prepare_m
|
|
|
|
#define GDS_DSQL_SET_CURSOR isc_dsql_set_cursor_name
|
|
|
|
#define GDS_DSQL_SQL_INFO isc_dsql_sql_info
|
|
|
|
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
void SRVR_main(rem_port* main_port, USHORT flags)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* S R V R _ m a i n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Main entrypoint of server.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
ISC_enter(); /* Setup floating point exception handler once and for all. */
|
|
|
|
#endif
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
PACKET send, receive;
|
2003-09-11 20:59:34 +02:00
|
|
|
zap_packet(&receive, true);
|
|
|
|
zap_packet(&send, true);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
set_server(main_port, flags);
|
|
|
|
|
2003-09-11 20:59:34 +02:00
|
|
|
while (true)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
//
|
|
|
|
// Note: The following is cloned in server other SRVR_main instances.
|
|
|
|
//
|
2004-02-24 06:34:44 +01:00
|
|
|
rem_port* port = main_port->receive(&receive);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!port) {
|
|
|
|
break;
|
|
|
|
}
|
2003-03-11 06:53:55 +01:00
|
|
|
if (!process_packet(port, &send, &receive, 0)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2003-03-11 06:53:55 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
void SRVR_multi_thread( rem_port* main_port, USHORT flags)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
2003-09-24 00:34:14 +02:00
|
|
|
#ifdef MULTI_THREAD
|
2001-05-23 15:26:42 +02:00
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* S R V R _ m u l t i _ t h r e a d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Multi-threaded flavor of server.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SERVER_REQ request = NULL, active;
|
2004-03-01 04:35:23 +01:00
|
|
|
rem_port* port = NULL; // Was volatile PORT port = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG pending_requests;
|
|
|
|
P_OP operation;
|
|
|
|
#ifdef DEV_BUILD
|
|
|
|
#ifdef DEBUG
|
|
|
|
SSHORT request_count = 0;
|
|
|
|
#endif /* DEBUG */
|
|
|
|
#endif /* DEV_BUILD */
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2004-08-30 20:11:08 +02:00
|
|
|
trdb thd_context(status_vector);
|
|
|
|
trdb* tdrdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
gds__thread_enable(-1);
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-21 17:22:19 +02:00
|
|
|
REM_set_thread_data(tdrdb, &thd_context);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
try {
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
set_server(main_port, flags);
|
|
|
|
|
|
|
|
/* We need to have this error handler as there is so much underlaying code
|
|
|
|
* that is depending on it being there. The expected failure
|
|
|
|
* that can occur in the call paths from here is out-of-memory
|
|
|
|
* and it is unknown if other failures can get us here
|
|
|
|
* Failures can occur from set_server as well as RECEIVE
|
|
|
|
*
|
|
|
|
* Note that if a failure DOES occur, we reenter the loop and try to continue
|
|
|
|
* operations. This is important as this is the main receive loop for
|
|
|
|
* new incoming requests, if we exit here no new requests can come to the
|
|
|
|
* server.
|
|
|
|
*/
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
try {
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* When this loop exits, the server will no longer receive requests */
|
2003-09-11 20:59:34 +02:00
|
|
|
while (true)
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
port = NULL;
|
|
|
|
|
|
|
|
/* Allocate a memory block to store the request in */
|
|
|
|
|
|
|
|
if (request = free_requests)
|
|
|
|
{
|
|
|
|
free_requests = request->req_next;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* No block on the free list - allocate some new memory */
|
|
|
|
|
|
|
|
request = (SERVER_REQ) gds__alloc((SLONG) sizeof(struct server_req_t));
|
|
|
|
if (request)
|
|
|
|
{
|
2003-09-11 20:59:34 +02:00
|
|
|
zap_packet(&request->req_send, true);
|
|
|
|
zap_packet(&request->req_receive, true);
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("SRVR_multi_thread allocate request %x\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
request);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* System is out of memory, let's delay processing this
|
|
|
|
request and hope another thread will free memory or
|
|
|
|
request blocks that we can then use. */
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
THREAD_SLEEP(1 * 1000);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef DEV_BUILD
|
|
|
|
#ifdef DEBUG
|
2004-03-01 04:35:23 +01:00
|
|
|
if ((request_count++ % 4) == 0)
|
|
|
|
throw std::bad_alloc();
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif /* DEBUG */
|
|
|
|
#endif /* DEV_BUILD */
|
|
|
|
|
|
|
|
if (request) {
|
|
|
|
request->req_next = NULL;
|
|
|
|
request->req_chain = NULL;
|
|
|
|
|
|
|
|
/* We have a request block - now get some information to stick into it */
|
|
|
|
if (!(port = main_port->receive(&request->req_receive)))
|
|
|
|
{
|
|
|
|
gds__log("SRVR_multi_thread/RECEIVE: error on main_port, shutting down");
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2004-05-21 17:22:19 +02:00
|
|
|
REM_restore_thread_data();
|
2001-05-23 15:26:42 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
request->req_port = port;
|
|
|
|
operation = request->req_receive.p_operation;
|
|
|
|
|
|
|
|
#ifdef DEV_BUILD
|
|
|
|
#ifdef DEBUG
|
2004-03-01 04:35:23 +01:00
|
|
|
if ((request_count % 5) == 0)
|
|
|
|
throw std::bad_alloc();
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif /* DEBUG */
|
|
|
|
#endif /* DEV_BUILD */
|
|
|
|
|
|
|
|
/* If port has an active request, link this one in */
|
|
|
|
|
|
|
|
for (active = active_requests; active; active = active->req_next)
|
|
|
|
if (active->req_port == port) {
|
|
|
|
/* Don't queue a dummy keepalive packet if there is
|
|
|
|
an active request running on this port. */
|
|
|
|
|
|
|
|
if (operation == op_dummy) {
|
|
|
|
request->req_next = free_requests;
|
|
|
|
free_requests = request;
|
|
|
|
goto finished;
|
|
|
|
}
|
|
|
|
port->port_requests_queued++;
|
|
|
|
append_request_chain(request, &active->req_chain);
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf
|
2001-05-23 15:26:42 +02:00
|
|
|
("SRVR_multi_thread ACTIVE request_queued %d\n",
|
|
|
|
port->port_requests_queued);
|
2004-04-29 00:36:29 +02:00
|
|
|
fflush(stdout);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
#ifdef CANCEL_OPERATION
|
|
|
|
if (operation == op_exit || operation == op_disconnect)
|
|
|
|
cancel_operation(port);
|
|
|
|
#endif
|
|
|
|
goto finished;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If port has an pending request, link this one in */
|
|
|
|
|
|
|
|
for (active = request_que; active; active = active->req_next)
|
2002-03-31 01:40:08 +01:00
|
|
|
{
|
|
|
|
if (active->req_port == port)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Don't queue a dummy keepalive packet if there is
|
|
|
|
a pending request against this port. */
|
|
|
|
|
|
|
|
if (operation == op_dummy) {
|
|
|
|
request->req_next = free_requests;
|
|
|
|
free_requests = request;
|
|
|
|
goto finished;
|
|
|
|
}
|
|
|
|
port->port_requests_queued++;
|
|
|
|
append_request_chain(request, &active->req_chain);
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf
|
2001-05-23 15:26:42 +02:00
|
|
|
("SRVR_multi_thread PENDING request_queued %d\n",
|
|
|
|
port->port_requests_queued);
|
2004-04-29 00:36:29 +02:00
|
|
|
fflush(stdout);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
#ifdef CANCEL_OPERATION
|
|
|
|
if (operation == op_exit || operation == op_disconnect)
|
|
|
|
cancel_operation(port);
|
|
|
|
#endif
|
|
|
|
goto finished;
|
|
|
|
}
|
2002-03-31 01:40:08 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* No port to assign request to, add it to the waiting queue and wake up a
|
|
|
|
* thread to handle it
|
|
|
|
*/
|
2003-09-16 22:59:45 +02:00
|
|
|
REMOTE_TRACE(("Enqueue request %p", request));
|
2001-05-23 15:26:42 +02:00
|
|
|
pending_requests = append_request_next(request, &request_que);
|
|
|
|
port->port_requests_queued++;
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf
|
2001-05-23 15:26:42 +02:00
|
|
|
("SRVR_multi_thread APPEND_PENDING request_queued %d\n",
|
|
|
|
port->port_requests_queued);
|
2004-04-29 00:36:29 +02:00
|
|
|
fflush(stdout);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* NOTE: we really *should* have something that limits how many
|
|
|
|
* total threads we allow in the system. As each thread will
|
|
|
|
* eat up memory that other threads could use to complete their work
|
|
|
|
*/
|
|
|
|
/* NOTE: The setting up of extra_threads variable is done below to let waiting
|
|
|
|
threads know if their services may be needed for the current set
|
|
|
|
of requests. Otherwise, some idle threads may leave the system
|
|
|
|
freeing up valuable memory.
|
|
|
|
*/
|
|
|
|
extra_threads = threads_waiting - pending_requests;
|
|
|
|
if (extra_threads < 0) {
|
2004-06-08 15:41:08 +02:00
|
|
|
gds__thread_start( loopThread,
|
2003-04-08 03:06:46 +02:00
|
|
|
(void*)(ULONG) flags,
|
2001-05-23 15:26:42 +02:00
|
|
|
THREAD_medium,
|
|
|
|
THREAD_ast,
|
|
|
|
0);
|
|
|
|
}
|
|
|
|
|
2003-09-16 22:59:45 +02:00
|
|
|
REMOTE_TRACE(("Post event"));
|
|
|
|
requests_semaphore.release();
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
finished:
|
|
|
|
;
|
|
|
|
}
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
} // try
|
2003-02-13 14:02:04 +01:00
|
|
|
catch (const std::exception&)
|
2001-12-24 03:51:06 +01:00
|
|
|
{
|
|
|
|
/* If we got as far as having a port allocated before the error, disconnect it
|
|
|
|
* gracefully.
|
|
|
|
*/
|
|
|
|
if (port != NULL)
|
|
|
|
{
|
2003-06-30 13:26:38 +02:00
|
|
|
/*
|
2001-12-24 03:51:06 +01:00
|
|
|
#ifdef DEV_BUILD
|
|
|
|
#ifdef DEBUG
|
|
|
|
ConsolePrintf("%%ISERVER-F-NOPORT, no port in a storm\r\n");
|
2003-06-30 13:26:38 +02:00
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
*/
|
2001-12-24 03:51:06 +01:00
|
|
|
gds__log("SRVR_multi_thread: forcefully disconnecting a port");
|
|
|
|
|
|
|
|
/* To handle recursion within the error handler */
|
|
|
|
try {
|
|
|
|
/* If we have a port, request really should be non-null, but just in case ... */
|
|
|
|
if (request != NULL) {
|
|
|
|
/* Send client a real status indication of why we disconnected them */
|
|
|
|
/* Note that send_response() can post errors that wind up in this same handler */
|
2003-06-30 13:26:38 +02:00
|
|
|
/*
|
2001-12-24 03:51:06 +01:00
|
|
|
#ifdef DEV_BUILD
|
|
|
|
#ifdef DEBUG
|
|
|
|
ConsolePrintf
|
|
|
|
("%%ISERVER-F-NOMEM, virtual memory exhausted\r\n");
|
2003-06-30 13:26:38 +02:00
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
*/
|
2001-12-24 03:51:06 +01:00
|
|
|
port->send_response(&request->req_send, 0, 0,
|
|
|
|
status_vector);
|
|
|
|
port->disconnect(&request->req_send, &request->req_receive);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* Can't tell the client much, just make 'em go away. Their side should detect
|
|
|
|
* a network error
|
|
|
|
*/
|
|
|
|
port->disconnect(NULL, NULL);
|
|
|
|
}
|
|
|
|
port = NULL;
|
|
|
|
|
|
|
|
} // try
|
2003-02-13 14:02:04 +01:00
|
|
|
catch (const std::exception&) {
|
2001-12-24 03:51:06 +01:00
|
|
|
port->disconnect(NULL, NULL);
|
|
|
|
port = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* There was an error in the processing of the request, if we have allocated
|
|
|
|
* a request, free it up and continue.
|
|
|
|
*/
|
|
|
|
if (request != NULL) {
|
|
|
|
request->req_next = free_requests;
|
|
|
|
free_requests = request;
|
|
|
|
request = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // try
|
2003-02-13 14:02:04 +01:00
|
|
|
catch (const std::exception&) {
|
2001-12-24 03:51:06 +01:00
|
|
|
/* Some kind of unhandled error occured during server setup. In lieu
|
|
|
|
* of anything we CAN do, log something (and we might be so hosed
|
|
|
|
* we can't log anything) and give up.
|
|
|
|
* The likely error here is out-of-memory.
|
|
|
|
*/
|
|
|
|
gds__log("SRVR_multi_thread: error during startup, shutting down");
|
|
|
|
}
|
|
|
|
|
2005-11-24 13:15:41 +01:00
|
|
|
THREAD_EXIT();
|
2004-05-21 17:22:19 +02:00
|
|
|
REM_restore_thread_data();
|
2003-09-24 00:34:14 +02:00
|
|
|
#endif
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static bool accept_connection(rem_port* port,
|
2003-09-11 20:59:34 +02:00
|
|
|
P_CNCT * connect,
|
|
|
|
PACKET* send)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* a c c e p t _ c o n n e c t i o n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Process a connect packet.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-08-27 01:27:46 +02:00
|
|
|
P_ARCH architecture = arch_generic;
|
|
|
|
USHORT version = 0;
|
|
|
|
USHORT type = 0;
|
|
|
|
bool accepted = false;
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Accept the physical connection */
|
|
|
|
|
|
|
|
send->p_operation = op_reject;
|
2004-02-24 06:34:44 +01:00
|
|
|
P_ACPT* accept = &send->p_acpt;
|
|
|
|
USHORT weight = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!port->accept(connect)) {
|
|
|
|
port->send(send);
|
2003-09-11 20:59:34 +02:00
|
|
|
return false;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Select the most appropriate protocol (this will get smarter) */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
const p_cnct::p_cnct_repeat* protocol = connect->p_cnct_versions;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
for (const p_cnct::p_cnct_repeat* const end = protocol + connect->p_cnct_count;
|
|
|
|
protocol < end; protocol++)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if ((protocol->p_cnct_version == PROTOCOL_VERSION3 ||
|
|
|
|
protocol->p_cnct_version == PROTOCOL_VERSION4 ||
|
|
|
|
protocol->p_cnct_version == PROTOCOL_VERSION5 ||
|
|
|
|
protocol->p_cnct_version == PROTOCOL_VERSION6 ||
|
|
|
|
protocol->p_cnct_version == PROTOCOL_VERSION7 ||
|
|
|
|
protocol->p_cnct_version == PROTOCOL_VERSION8 ||
|
|
|
|
protocol->p_cnct_version == PROTOCOL_VERSION9 ||
|
|
|
|
protocol->p_cnct_version == PROTOCOL_VERSION10
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
|| protocol->p_cnct_version == PROTOCOL_SCROLLABLE_CURSORS
|
|
|
|
#endif
|
|
|
|
) &&
|
2003-05-02 19:17:09 +02:00
|
|
|
(protocol->p_cnct_architecture == arch_generic ||
|
|
|
|
protocol->p_cnct_architecture == ARCHITECTURE) &&
|
2003-11-03 01:37:11 +01:00
|
|
|
protocol->p_cnct_weight >= weight)
|
|
|
|
{
|
2004-08-27 01:27:46 +02:00
|
|
|
accepted = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
weight = protocol->p_cnct_weight;
|
|
|
|
version = protocol->p_cnct_version;
|
|
|
|
architecture = protocol->p_cnct_architecture;
|
|
|
|
type = MIN(protocol->p_cnct_max_type, ptype_out_of_band);
|
|
|
|
send->p_operation = op_accept;
|
|
|
|
}
|
2004-02-24 06:34:44 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Send off out gracious acceptance or flag rejection */
|
2004-08-27 01:27:46 +02:00
|
|
|
if (!accepted) {
|
2004-08-28 09:30:17 +02:00
|
|
|
port->send(send);
|
2004-08-27 01:27:46 +02:00
|
|
|
return false;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
accept->p_acpt_version = port->port_protocol = version;
|
|
|
|
accept->p_acpt_architecture = architecture;
|
|
|
|
accept->p_acpt_type = type;
|
|
|
|
|
|
|
|
/* and modify the version string to reflect the chosen protocol */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
TEXT buffer[64];
|
2001-05-23 15:26:42 +02:00
|
|
|
sprintf(buffer, "%s/P%d", port->port_version->str_data,
|
|
|
|
port->port_protocol);
|
|
|
|
ALLR_free(port->port_version);
|
|
|
|
port->port_version = REMOTE_make_string(buffer);
|
|
|
|
|
|
|
|
if (architecture == ARCHITECTURE)
|
|
|
|
port->port_flags |= PORT_symmetric;
|
|
|
|
|
|
|
|
if (type == ptype_rpc)
|
|
|
|
port->port_flags |= PORT_rpc;
|
|
|
|
|
|
|
|
if (type != ptype_out_of_band)
|
|
|
|
port->port_flags |= PORT_no_oob;
|
|
|
|
|
|
|
|
port->send(send);
|
|
|
|
|
2003-09-11 20:59:34 +02:00
|
|
|
return true;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static ISC_STATUS allocate_statement( rem_port* port, P_RLSE * allocate, PACKET* send)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* a l l o c a t e _ s t a t e m e n t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Allocate a statement handle.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
RDB rdb = port->port_context;
|
2004-05-03 01:06:37 +02:00
|
|
|
FB_API_HANDLE handle = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-11-03 01:37:11 +01:00
|
|
|
GDS_DSQL_ALLOCATE(status_vector, &rdb->rdb_handle, &handle);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
OBJCT object;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (status_vector[1])
|
|
|
|
object = 0;
|
|
|
|
else {
|
|
|
|
/* Allocate SQL request block */
|
|
|
|
|
2004-05-17 12:22:34 +02:00
|
|
|
RSR statement = (RSR) ALLR_block(type_rsr, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
statement->rsr_rdb = rdb;
|
|
|
|
statement->rsr_handle = handle;
|
|
|
|
if (statement->rsr_id = port->get_id(&statement->rsr_header))
|
|
|
|
{
|
|
|
|
object = statement->rsr_id;
|
|
|
|
statement->rsr_next = rdb->rdb_sql_requests;
|
|
|
|
rdb->rdb_sql_requests = statement;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
object = 0;
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
GDS_DSQL_FREE(status_vector, &statement->rsr_handle, DSQL_drop);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
ALLR_release(statement);
|
|
|
|
status_vector[0] = isc_arg_gds;
|
|
|
|
status_vector[1] = isc_too_many_handles;
|
|
|
|
status_vector[2] = isc_arg_end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return port->send_response(send, object, 0, status_vector);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-14 06:51:54 +01:00
|
|
|
#ifdef MULTI_THREAD
|
2004-05-03 15:35:38 +02:00
|
|
|
static SLONG append_request_chain( SERVER_REQ request, SERVER_REQ * que_inst)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* a p p e n d _ r e q u e s t _ c h a i n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Traverse using req_chain ptr and append
|
|
|
|
* a request at the end of a que.
|
|
|
|
* Return count of pending requests.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SLONG requests;
|
|
|
|
|
2004-05-03 15:35:38 +02:00
|
|
|
for (requests = 1; *que_inst; que_inst = &(*que_inst)->req_chain)
|
2001-05-23 15:26:42 +02:00
|
|
|
++requests;
|
|
|
|
|
2004-05-03 15:35:38 +02:00
|
|
|
*que_inst = request;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return requests;
|
|
|
|
}
|
|
|
|
|
2004-05-03 15:35:38 +02:00
|
|
|
static SLONG append_request_next( SERVER_REQ request, SERVER_REQ * que_inst)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* a p p e n d _ r e q u e s t _ n e x t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Traverse using req_next ptr and append
|
|
|
|
* a request at the end of a que.
|
|
|
|
* Return count of pending requests.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SLONG requests;
|
|
|
|
|
2004-05-03 15:35:38 +02:00
|
|
|
for (requests = 1; *que_inst; que_inst = &(*que_inst)->req_next)
|
2001-05-23 15:26:42 +02:00
|
|
|
++requests;
|
|
|
|
|
2004-05-03 15:35:38 +02:00
|
|
|
*que_inst = request;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return requests;
|
|
|
|
}
|
2004-03-14 06:51:54 +01:00
|
|
|
#endif
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-11-27 21:53:09 +01:00
|
|
|
static void addRemoteAddress(Firebird::ClumpletWriter& dpb_buffer, UCHAR par, const rem_port* port)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* a d d R e m o t e A d d r e s s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Insert remote endpoint data into DPB address stack
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
Firebird::ClumpletWriter address_stack_buffer(Firebird::ClumpletReader::UnTagged, MAX_UCHAR - 2);
|
|
|
|
if (dpb_buffer.find(par)) {
|
|
|
|
address_stack_buffer.reset(dpb_buffer.getBytes(), dpb_buffer.getClumpLength());
|
|
|
|
dpb_buffer.deleteClumplet();
|
|
|
|
}
|
|
|
|
|
|
|
|
Firebird::ClumpletWriter address_record(Firebird::ClumpletReader::UnTagged, MAX_UCHAR - 2);
|
|
|
|
if (port->port_protocol_str)
|
|
|
|
address_record.insertString(isc_dpb_addr_protocol,
|
|
|
|
port->port_protocol_str->str_data, port->port_protocol_str->str_length);
|
|
|
|
if (port->port_address_str)
|
|
|
|
address_record.insertString(isc_dpb_addr_endpoint,
|
|
|
|
port->port_address_str->str_data, port->port_address_str->str_length);
|
|
|
|
|
|
|
|
// We always insert remote address descriptor as a first element
|
|
|
|
// of appropriate clumplet so user cannot fake it and engine may somewhat trust it.
|
|
|
|
fb_assert(address_stack_buffer.getCurOffset() == 0);
|
|
|
|
address_stack_buffer.insertBytes(isc_dpb_address,
|
|
|
|
address_record.getBuffer(), address_record.getBufferLength());
|
|
|
|
|
|
|
|
dpb_buffer.insertBytes(par, address_stack_buffer.getBuffer(),
|
|
|
|
address_stack_buffer.getBufferLength());
|
|
|
|
|
|
|
|
// Remove all remaining isc_*pb_address_path clumplets.
|
|
|
|
// This is the security feature to prevent user from faking remote address
|
|
|
|
// by passing multiple address_path clumplets. Engine assumes that
|
|
|
|
// dpb contains no more than one address_path clumplet and for
|
|
|
|
// clients coming via remote interface it can trust the first address from
|
|
|
|
// address stack. Clients acessing database directly can do whatever they
|
|
|
|
// want with the engine including faking the source address, not much we
|
|
|
|
// can do about it.
|
|
|
|
|
|
|
|
while (!dpb_buffer.isEof()) {
|
|
|
|
if (dpb_buffer.getClumpTag() == par)
|
|
|
|
dpb_buffer.deleteClumplet();
|
|
|
|
else
|
|
|
|
dpb_buffer.moveNext();
|
|
|
|
}
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-04-10 12:31:28 +02:00
|
|
|
static ISC_STATUS attach_database(
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* port,
|
2001-05-23 15:26:42 +02:00
|
|
|
P_OP operation, P_ATCH * attach, PACKET* send)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* a t t a c h _ d a t a b a s e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Process an attach or create packet.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
send->p_operation = op_accept;
|
2004-05-03 01:06:37 +02:00
|
|
|
FB_API_HANDLE handle = 0;
|
2003-10-29 11:53:47 +01:00
|
|
|
const char* file = reinterpret_cast<const char*>(attach->p_atch_file.cstr_address);
|
2003-11-07 09:06:35 +01:00
|
|
|
const USHORT l = attach->p_atch_file.cstr_length;
|
2004-11-26 02:01:27 +01:00
|
|
|
|
|
|
|
const UCHAR* dpb = attach->p_atch_dpb.cstr_address;
|
2003-11-07 09:06:35 +01:00
|
|
|
USHORT dl = attach->p_atch_dpb.cstr_length;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-11-27 21:53:09 +01:00
|
|
|
Firebird::ClumpletWriter dpb_buffer(Firebird::ClumpletReader::Tagged, MAX_SSHORT);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-11-26 02:01:27 +01:00
|
|
|
if (dl)
|
|
|
|
dpb_buffer.reset(dpb, dl);
|
|
|
|
else
|
|
|
|
dpb_buffer.reset(isc_dpb_version1);
|
|
|
|
|
|
|
|
// If we have user identification, append it to database parameter block
|
2004-02-24 06:34:44 +01:00
|
|
|
rem_str* string = port->port_user_name;
|
|
|
|
if (string) {
|
2004-11-26 02:01:27 +01:00
|
|
|
dpb_buffer.setCurOffset(dpb_buffer.getBufferLength());
|
|
|
|
dpb_buffer.insertString(isc_dpb_sys_user_name,
|
|
|
|
string->str_data, string->str_length);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Now, insert remote endpoint data into DPB address stack
|
2005-11-27 21:53:09 +01:00
|
|
|
addRemoteAddress(dpb_buffer, isc_dpb_address_path, port);
|
2004-11-26 02:01:27 +01:00
|
|
|
|
|
|
|
/* Disable remote gsec attachments */
|
|
|
|
for (dpb_buffer.setCurOffset(1); !dpb_buffer.isEof(); ) {
|
|
|
|
if (dpb_buffer.getClumpTag() == isc_dpb_gsec_attach)
|
|
|
|
dpb_buffer.deleteClumplet();
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
2004-11-26 02:01:27 +01:00
|
|
|
dpb_buffer.moveNext();
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-11-26 02:01:27 +01:00
|
|
|
dpb = dpb_buffer.getBuffer();
|
|
|
|
dl = dpb_buffer.getBufferLength();
|
|
|
|
|
2003-11-07 09:06:35 +01:00
|
|
|
/* See if user has specified parameters relevant to the connection,
|
2001-05-23 15:26:42 +02:00
|
|
|
they will be stuffed in the DPB if so. */
|
|
|
|
REMOTE_get_timeout_params(port, dpb, dl);
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
if (operation == op_attach)
|
|
|
|
{
|
2003-10-29 11:53:47 +01:00
|
|
|
isc_attach_database(status_vector, l, file,
|
2003-11-07 09:06:35 +01:00
|
|
|
&handle, dl, reinterpret_cast<const char*>(dpb));
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2003-10-29 11:53:47 +01:00
|
|
|
isc_create_database(status_vector, l, file,
|
2003-11-07 09:06:35 +01:00
|
|
|
&handle, dl, reinterpret_cast<const char*>(dpb), 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!status_vector[1])
|
|
|
|
{
|
2004-05-17 12:22:34 +02:00
|
|
|
RDB rdb = (RDB) ALLR_block(type_rdb, 0);
|
2004-02-24 06:34:44 +01:00
|
|
|
port->port_context = rdb;
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("attach_databases(server) allocate rdb %x\n", rdb);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
rdb->rdb_port = port;
|
|
|
|
rdb->rdb_handle = handle;
|
|
|
|
}
|
|
|
|
|
|
|
|
return port->send_response(send, 0, 0, status_vector);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-03-14 06:51:54 +01:00
|
|
|
#ifdef NOT_USED_OR_REPLACED
|
2004-01-28 08:50:41 +01:00
|
|
|
static void aux_connect( rem_port* port, P_REQ * request, PACKET* send)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* a u x _ c o n n e c t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* We're receive a auxiliary connection on the main communications
|
|
|
|
* channel. Accept connection and reply politely.
|
|
|
|
*
|
2004-03-14 06:51:54 +01:00
|
|
|
* 13-Mar-2004, Nickolay Samofatov
|
|
|
|
* This code is 64-bit unsafe, unused and also has a security hole, thus I disable it for now
|
|
|
|
*
|
2001-05-23 15:26:42 +02:00
|
|
|
**************************************/
|
|
|
|
port->connect(0, 0);
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* partner = (rem_port*) request->p_req_partner;
|
2001-05-23 15:26:42 +02:00
|
|
|
partner->port_async = port;
|
|
|
|
}
|
2004-03-14 06:51:54 +01:00
|
|
|
#endif
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static void aux_request( rem_port* port, P_REQ * request, PACKET* send)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* a u x _ r e q u e s t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Other guy wants to establish a secondary connection.
|
|
|
|
* Humor him.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* save the port status vector */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
ISC_STATUS* const save_status = port->port_status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
port->port_status_vector = status_vector;
|
|
|
|
success(status_vector);
|
|
|
|
|
|
|
|
/* We do this silliness with buffer because the SPX protocol
|
|
|
|
requires a 12 byte buffer to be sent back. Other protocols
|
|
|
|
can do what they want to with cstr_address. */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
CSTRING save_cstring = send->p_resp.p_resp_data;
|
|
|
|
UCHAR buffer[12];
|
2001-05-23 15:26:42 +02:00
|
|
|
send->p_resp.p_resp_data.cstr_address = buffer;
|
2004-02-24 06:34:44 +01:00
|
|
|
rem_port* aux_port = port->request(send);
|
|
|
|
RDB rdb = port->port_context;
|
2001-05-23 15:26:42 +02:00
|
|
|
port->send_response(send, rdb->rdb_id,
|
|
|
|
send->p_resp.p_resp_data.cstr_length, status_vector);
|
|
|
|
|
|
|
|
if (status_vector[1]) {
|
|
|
|
/* restore the port status vector */
|
|
|
|
|
|
|
|
port->port_status_vector = save_status;
|
|
|
|
send->p_resp.p_resp_data = save_cstring;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (aux_port) {
|
|
|
|
aux_port->connect(send, 0);
|
|
|
|
aux_port->port_context = rdb;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* restore the port status vector */
|
|
|
|
|
|
|
|
port->port_status_vector = save_status;
|
|
|
|
send->p_resp.p_resp_data = save_cstring;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static ISC_STATUS cancel_events( rem_port* port, P_EVENT * stuff, PACKET* send)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c a n c e l _ e v e n t s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Cancel events.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
/* Which database ? */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
RDB rdb = port->port_context;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Find the event */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
RVNT event;
|
2001-05-23 15:26:42 +02:00
|
|
|
for (event = rdb->rdb_events; event; event = event->rvnt_next)
|
|
|
|
if (event->rvnt_rid == stuff->p_event_rid)
|
|
|
|
break;
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* If no event found, pretend it was cancelled */
|
|
|
|
|
|
|
|
if (!event)
|
|
|
|
return port->send_response(send, 0, 0, status_vector);
|
|
|
|
|
|
|
|
/* cancel the event */
|
|
|
|
|
|
|
|
if (event->rvnt_id) {
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_cancel_events(status_vector, &rdb->rdb_handle, &event->rvnt_id);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* zero event info */
|
|
|
|
|
|
|
|
event->rvnt_id = 0L;
|
|
|
|
event->rvnt_rid = 0L;
|
2003-12-22 11:00:59 +01:00
|
|
|
event->rvnt_ast = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* return response */
|
|
|
|
|
|
|
|
return port->send_response(send, 0, 0, status_vector);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef CANCEL_OPERATION
|
2004-01-28 08:50:41 +01:00
|
|
|
static void cancel_operation( rem_port* port)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c a n c e l _ o p e r a t i o n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Flag a running operation for cancel.
|
|
|
|
* Service operations are not currently
|
|
|
|
* able to be canceled.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
RDB rdb;
|
|
|
|
if ((port->port_flags & (PORT_async | PORT_disconnect)) ||
|
2003-03-11 06:53:55 +01:00
|
|
|
!(rdb = port->port_context))
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (rdb->rdb_handle)
|
2003-03-11 06:53:55 +01:00
|
|
|
{
|
|
|
|
if (!(rdb->rdb_flags & RDB_service))
|
|
|
|
{
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2004-02-24 06:34:44 +01:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2004-05-03 01:06:37 +02:00
|
|
|
gds__cancel_operation(status_vector, (FB_API_HANDLE*) &rdb->rdb_handle,
|
2003-11-07 14:20:15 +01:00
|
|
|
CANCEL_raise);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2003-03-11 06:53:55 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static bool check_request(rrq* request,
|
2003-09-11 20:59:34 +02:00
|
|
|
USHORT incarnation,
|
|
|
|
USHORT msg_number)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c h e c k _ r e q u e s t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Check to see if a request is ready to send us a particular
|
|
|
|
* message. If so, return TRUE, otherwise FALSE.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
USHORT n;
|
|
|
|
|
|
|
|
if (!get_next_msg_no(request, incarnation, &n))
|
2003-09-11 20:59:34 +02:00
|
|
|
return false;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-11 20:59:34 +02:00
|
|
|
return (msg_number == n);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static USHORT check_statement_type( RSR statement)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c h e c k _ s t a t e m e n t _ t y p e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Return the type of SQL statement.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-02-24 06:34:44 +01:00
|
|
|
UCHAR buffer[16];
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY local_status;
|
2003-09-11 20:59:34 +02:00
|
|
|
USHORT ret = STMT_OTHER;
|
|
|
|
bool done = false;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-09-11 20:59:34 +02:00
|
|
|
if (!GDS_DSQL_SQL_INFO(local_status, &statement->rsr_handle,
|
2003-11-28 07:48:34 +01:00
|
|
|
sizeof(sql_info), (const SCHAR*) sql_info,
|
2003-09-11 20:59:34 +02:00
|
|
|
sizeof(buffer), reinterpret_cast<char*>(buffer)))
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
2004-02-24 06:34:44 +01:00
|
|
|
for (const UCHAR* info = buffer; (*info != isc_info_end) && !done;)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
2004-02-24 06:34:44 +01:00
|
|
|
const USHORT l = (USHORT) gds__vax_integer(info + 1, 2);
|
|
|
|
const USHORT type = (USHORT) gds__vax_integer(info + 3, l);
|
2001-05-23 15:26:42 +02:00
|
|
|
switch (*info)
|
|
|
|
{
|
2003-11-08 17:40:17 +01:00
|
|
|
case isc_info_sql_stmt_type:
|
|
|
|
if (type == isc_info_sql_stmt_get_segment ||
|
|
|
|
type == isc_info_sql_stmt_put_segment)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
ret = STMT_BLOB;
|
2003-09-11 20:59:34 +02:00
|
|
|
done = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case isc_info_sql_batch_fetch:
|
2003-09-11 20:59:34 +02:00
|
|
|
if (type == 0)
|
2001-05-23 15:26:42 +02:00
|
|
|
ret = STMT_NO_BATCH;
|
|
|
|
break;
|
|
|
|
case isc_info_error:
|
|
|
|
case isc_info_truncated:
|
2003-09-11 20:59:34 +02:00
|
|
|
done = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
}
|
|
|
|
info += 3 + l;
|
|
|
|
}
|
|
|
|
}
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::compile(P_CMPL* compileL, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c o m p i l e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Compile and request.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-25 13:49:12 +02:00
|
|
|
RDB rdb = this->port_context;
|
2004-05-03 01:06:37 +02:00
|
|
|
FB_API_HANDLE handle = 0;
|
2004-05-12 21:39:17 +02:00
|
|
|
const UCHAR* blr = compileL->p_cmpl_blr.cstr_address;
|
|
|
|
USHORT blr_length = compileL->p_cmpl_blr.cstr_length;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_compile_request(status_vector, &rdb->rdb_handle, &handle, blr_length,
|
2003-11-07 09:06:35 +01:00
|
|
|
reinterpret_cast<const char*>(blr));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (status_vector[1])
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Parse the request to find the messages */
|
|
|
|
|
2003-09-25 13:49:12 +02:00
|
|
|
REM_MSG message = PARSE_messages(blr, blr_length);
|
|
|
|
USHORT max_msg = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-25 13:49:12 +02:00
|
|
|
REM_MSG next;
|
2001-05-23 15:26:42 +02:00
|
|
|
for (next = message; next; next = next->msg_next)
|
|
|
|
max_msg = MAX(max_msg, next->msg_number);
|
|
|
|
|
|
|
|
/* Allocate block and merge into data structures */
|
|
|
|
|
2004-05-17 12:22:34 +02:00
|
|
|
rrq* requestL = (rrq*) ALLR_block(type_rrq, max_msg + 1);
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("compile(server) allocate request %x\n", request);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2004-05-12 21:39:17 +02:00
|
|
|
requestL->rrq_handle = handle;
|
|
|
|
requestL->rrq_rdb = rdb;
|
|
|
|
requestL->rrq_max_msg = max_msg;
|
2004-02-24 06:34:44 +01:00
|
|
|
OBJCT object = 0;
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
if (requestL->rrq_id = this->get_id(&requestL->rrq_header))
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
2004-05-12 21:39:17 +02:00
|
|
|
object = requestL->rrq_id;
|
|
|
|
requestL->rrq_next = rdb->rdb_requests;
|
|
|
|
rdb->rdb_requests = requestL;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2004-05-12 21:39:17 +02:00
|
|
|
isc_release_request(status_vector, &requestL->rrq_handle);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2004-05-12 21:39:17 +02:00
|
|
|
ALLR_release(requestL);
|
2001-05-23 15:26:42 +02:00
|
|
|
status_vector[0] = isc_arg_gds;
|
|
|
|
status_vector[1] = isc_too_many_handles;
|
|
|
|
status_vector[2] = isc_arg_end;
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
while (message) {
|
|
|
|
next = message->msg_next;
|
|
|
|
message->msg_next = message;
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
message->msg_prior = message;
|
|
|
|
#endif
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
rrq::rrq_repeat* tail = requestL->rrq_rpt + message->msg_number;
|
2001-05-23 15:26:42 +02:00
|
|
|
tail->rrq_message = message;
|
|
|
|
tail->rrq_xdr = message;
|
2004-01-28 08:50:41 +01:00
|
|
|
tail->rrq_format = (rem_fmt*) message->msg_address;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
message->msg_address = NULL;
|
|
|
|
message = next;
|
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, object, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::ddl(P_DDL* ddlL, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* d d l
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Execute isc_ddl call.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
RTR transaction;
|
|
|
|
|
|
|
|
CHECK_HANDLE_MEMBER(transaction,
|
|
|
|
RTR,
|
|
|
|
type_rtr,
|
2004-05-12 21:39:17 +02:00
|
|
|
ddlL->p_ddl_transaction,
|
2001-05-23 15:26:42 +02:00
|
|
|
isc_bad_trans_handle);
|
|
|
|
|
2003-11-05 10:02:33 +01:00
|
|
|
RDB rdb = this->port_context;
|
2004-05-12 21:39:17 +02:00
|
|
|
const UCHAR* blr = ddlL->p_ddl_blr.cstr_address;
|
|
|
|
const USHORT blr_length = ddlL->p_ddl_blr.cstr_length;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_ddl(status_vector, &rdb->rdb_handle, &transaction->rtr_handle,
|
2003-11-05 10:02:33 +01:00
|
|
|
blr_length, reinterpret_cast<const char*>(blr));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
void rem_port::disconnect(PACKET* sendL, PACKET* receiveL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* d i s c o n n e c t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* We've lost the connection to the parent. Stop everything.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-10-20 19:42:36 +02:00
|
|
|
RDB rdb = this->port_context;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
if (this->port_flags & PORT_async)
|
2004-10-20 19:42:36 +02:00
|
|
|
{
|
|
|
|
if (rdb && rdb->rdb_port && !(rdb->rdb_port->port_flags & PORT_disconnect))
|
|
|
|
{
|
|
|
|
PACKET *packet = &rdb->rdb_packet;
|
|
|
|
packet->p_operation = op_dummy;
|
|
|
|
rdb->rdb_port->send(packet);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
return;
|
2004-10-20 19:42:36 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
this->port_flags |= PORT_disconnect;
|
|
|
|
|
|
|
|
if (!rdb) {
|
2004-05-12 21:39:17 +02:00
|
|
|
REMOTE_free_packet(this, sendL);
|
|
|
|
REMOTE_free_packet(this, receiveL);
|
2001-05-23 15:26:42 +02:00
|
|
|
this->disconnect();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2003-05-05 13:44:02 +02:00
|
|
|
/* For WNET and XNET we should send dummy op_disconnect packet
|
|
|
|
to wakeup async port handling events on client side.
|
|
|
|
For INET it's not necessary because INET client's async port
|
|
|
|
wakes up while server performs shutdown(socket) call on its async port.
|
|
|
|
See interface.cpp - event_thread(). */
|
|
|
|
|
|
|
|
PACKET *packet = &rdb->rdb_packet;
|
|
|
|
if ((this->port_async) &&
|
|
|
|
((this->port_type == port_xnet) || (this->port_type == port_pipe)))
|
|
|
|
{
|
|
|
|
packet->p_operation = op_disconnect;
|
|
|
|
this->port_async->send(packet);
|
|
|
|
}
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
if (rdb->rdb_handle)
|
|
|
|
if (!(rdb->rdb_flags & RDB_service)) {
|
|
|
|
#ifdef CANCEL_OPERATION
|
|
|
|
/* Prevent a pending or spurious cancel from aborting
|
|
|
|
a good, clean detach from the database. */
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2004-05-03 01:06:37 +02:00
|
|
|
gds__cancel_operation(status_vector, (FB_API_HANDLE*) &rdb->rdb_handle,
|
2003-11-07 14:20:15 +01:00
|
|
|
CANCEL_disable);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
while (rdb->rdb_requests)
|
|
|
|
release_request(rdb->rdb_requests);
|
|
|
|
while (rdb->rdb_sql_requests)
|
|
|
|
release_sql_request(rdb->rdb_sql_requests);
|
2004-02-24 06:34:44 +01:00
|
|
|
RTR transaction;
|
2001-05-23 15:26:42 +02:00
|
|
|
while (transaction = rdb->rdb_transactions) {
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!(transaction->rtr_flags & RTR_limbo))
|
|
|
|
isc_rollback_transaction(status_vector,
|
2003-08-30 03:43:08 +02:00
|
|
|
&transaction->rtr_handle);
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef SUPERSERVER
|
|
|
|
/* The underlying JRD subsystem will release all
|
|
|
|
memory resources related to a limbo transaction
|
|
|
|
as a side-effect of the database detach call
|
|
|
|
below. However, the y-valve handle must be released
|
|
|
|
as well since an isc_disconnect_transaction()
|
|
|
|
call doesn't exist. */
|
|
|
|
|
|
|
|
else
|
|
|
|
gds__handle_cleanup(status_vector,
|
2004-05-03 01:06:37 +02:00
|
|
|
(FB_API_HANDLE*) &transaction->rtr_handle);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
release_transaction(rdb->rdb_transactions);
|
|
|
|
}
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_detach_database(status_vector, &rdb->rdb_handle);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
while (rdb->rdb_events) {
|
|
|
|
release_event(rdb->rdb_events);
|
|
|
|
}
|
|
|
|
if (this->port_statement) {
|
|
|
|
release_statement(&this->port_statement);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_service_detach(status_vector, &rdb->rdb_handle);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
REMOTE_free_packet(this, sendL);
|
|
|
|
REMOTE_free_packet(this, receiveL);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("disconnect(server) free rdb %x\n", rdb);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
this->port_context = NULL;
|
|
|
|
ALLR_release(rdb);
|
|
|
|
if (this->port_object_vector)
|
|
|
|
{
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("disconnect(server) free vector %x\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
this->port_object_vector);
|
|
|
|
#endif
|
|
|
|
ALLR_release(this->port_object_vector);
|
|
|
|
this->port_object_vector = NULL;
|
|
|
|
}
|
|
|
|
if (this->port_connection)
|
|
|
|
{
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("disconnect(server) free string %x\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
this->port_connection);
|
|
|
|
#endif
|
|
|
|
ALLR_release(this->port_connection);
|
|
|
|
this->port_connection = NULL;
|
|
|
|
}
|
|
|
|
if (this->port_version)
|
|
|
|
{
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("disconnect(server) free string %x\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
this->port_version);
|
|
|
|
#endif
|
|
|
|
ALLR_release(this->port_version);
|
|
|
|
this->port_version = NULL;
|
|
|
|
}
|
|
|
|
if (this->port_passwd)
|
|
|
|
{
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("disconnect(server) free string %x\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
this->port_passwd);
|
|
|
|
#endif
|
|
|
|
ALLR_release(this->port_passwd);
|
|
|
|
this->port_passwd = NULL;
|
|
|
|
}
|
|
|
|
if (this->port_user_name)
|
|
|
|
{
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("disconnect(server) free string %x\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
this->port_user_name);
|
|
|
|
#endif
|
|
|
|
ALLR_release(this->port_user_name);
|
|
|
|
this->port_user_name = NULL;
|
|
|
|
}
|
|
|
|
if (this->port_host)
|
|
|
|
{
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("disconnect(server) free string %x\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
this->port_host);
|
|
|
|
#endif
|
|
|
|
ALLR_release(this->port_host);
|
|
|
|
this->port_host = NULL;
|
|
|
|
}
|
|
|
|
this->disconnect();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
void rem_port::drop_database(P_RLSE* release, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* d r o p _ d a t a b a s e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* End a request.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
RDB rdb = this->port_context;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_drop_database(status_vector, &rdb->rdb_handle);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (status_vector[1]
|
2004-02-24 06:34:44 +01:00
|
|
|
&& (status_vector[1] != isc_drdb_completed_with_errs))
|
|
|
|
{
|
2004-05-12 21:39:17 +02:00
|
|
|
this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
return;
|
2004-11-07 11:38:13 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
while (rdb->rdb_events)
|
|
|
|
release_event(rdb->rdb_events);
|
|
|
|
|
|
|
|
while (rdb->rdb_requests)
|
|
|
|
release_request(rdb->rdb_requests);
|
|
|
|
|
|
|
|
while (rdb->rdb_sql_requests)
|
|
|
|
release_sql_request(rdb->rdb_sql_requests);
|
|
|
|
|
|
|
|
while (rdb->rdb_transactions)
|
|
|
|
release_transaction(rdb->rdb_transactions);
|
|
|
|
|
|
|
|
if (this->port_statement)
|
|
|
|
release_statement(&this->port_statement);
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
2002-11-19 13:40:54 +01:00
|
|
|
static REM_MSG dump_cache( rrq::rrq_repeat* tail)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* d u m p _ c a c h e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* We have encountered a situation where what's in
|
|
|
|
* cache is not useful, so empty the cache in
|
|
|
|
* preparation for refilling it.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-02-24 06:34:44 +01:00
|
|
|
REM_MSG message = tail->rrq_xdr;
|
2003-09-11 20:59:34 +02:00
|
|
|
while (true) {
|
2001-05-23 15:26:42 +02:00
|
|
|
message->msg_address = NULL;
|
|
|
|
message = message->msg_next;
|
|
|
|
if (message == tail->rrq_xdr)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
tail->rrq_message = message;
|
|
|
|
|
|
|
|
return message;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::end_blob(P_OP operation, P_RLSE * release, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* e n d _ b l o b
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* End a blob.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
RBL blob;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
CHECK_HANDLE_MEMBER(blob,
|
|
|
|
RBL,
|
|
|
|
type_rbl,
|
|
|
|
release->p_rlse_object,
|
|
|
|
isc_bad_segstr_handle);
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
if (operation == op_close_blob)
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_close_blob(status_vector, &blob->rbl_handle);
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_cancel_blob(status_vector, &blob->rbl_handle);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!status_vector[1]) {
|
|
|
|
release_blob(blob);
|
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::end_database(P_RLSE * release, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* e n d _ d a t a b a s e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* End a request.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
RDB rdb = this->port_context;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2004-11-24 10:22:07 +01:00
|
|
|
isc_detach_database(status_vector, &rdb->rdb_handle);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (status_vector[1])
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
while (rdb->rdb_events)
|
|
|
|
release_event(rdb->rdb_events);
|
|
|
|
|
|
|
|
while (rdb->rdb_requests)
|
|
|
|
release_request(rdb->rdb_requests);
|
|
|
|
|
|
|
|
while (rdb->rdb_sql_requests)
|
|
|
|
release_sql_request(rdb->rdb_sql_requests);
|
|
|
|
|
|
|
|
while (rdb->rdb_transactions)
|
|
|
|
release_transaction(rdb->rdb_transactions);
|
|
|
|
|
|
|
|
if (this->port_statement)
|
|
|
|
release_statement(&this->port_statement);
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::end_request(P_RLSE * release, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* e n d _ r e q u e s t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* End a request.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-05-12 21:39:17 +02:00
|
|
|
rrq* requestL;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
CHECK_HANDLE_MEMBER(requestL,
|
2004-01-28 08:50:41 +01:00
|
|
|
rrq*,
|
2001-05-23 15:26:42 +02:00
|
|
|
type_rrq,
|
|
|
|
release->p_rlse_object,
|
|
|
|
isc_bad_req_handle);
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2004-05-12 21:39:17 +02:00
|
|
|
isc_release_request(status_vector, &requestL->rrq_handle);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!status_vector[1])
|
2004-05-12 21:39:17 +02:00
|
|
|
release_request(requestL);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::end_statement(P_SQLFREE* free_stmt, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/*****************************************
|
|
|
|
*
|
|
|
|
* e n d _ s t a t e m e n t
|
|
|
|
*
|
|
|
|
*****************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Free a statement.
|
|
|
|
*
|
|
|
|
*****************************************/
|
|
|
|
RSR statement;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
CHECK_HANDLE_MEMBER(statement,
|
|
|
|
RSR,
|
|
|
|
type_rsr,
|
|
|
|
free_stmt->p_sqlfree_statement,
|
|
|
|
isc_bad_req_handle);
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
GDS_DSQL_FREE(status_vector,
|
2002-12-16 17:52:35 +01:00
|
|
|
&statement->rsr_handle,
|
2001-05-23 15:26:42 +02:00
|
|
|
free_stmt->p_sqlfree_option);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (status_vector[1])
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!statement->rsr_handle) {
|
|
|
|
release_sql_request(statement);
|
|
|
|
statement = NULL;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
statement->rsr_flags &= ~RSR_fetched;
|
|
|
|
statement->rsr_rtr = NULL;
|
|
|
|
REMOTE_reset_statement(statement);
|
|
|
|
statement->rsr_message = statement->rsr_buffer;
|
|
|
|
}
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
const USHORT object = (statement) ? statement->rsr_id : (USHORT) - 1;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, object, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::end_transaction(P_OP operation, P_RLSE * release, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* e n d _ t r a n s a c t i o n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* End a transaction.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
RTR transaction;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
CHECK_HANDLE_MEMBER(transaction,
|
|
|
|
RTR,
|
|
|
|
type_rtr,
|
|
|
|
release->p_rlse_object,
|
|
|
|
isc_bad_trans_handle);
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
switch (operation)
|
|
|
|
{
|
|
|
|
case op_commit:
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_commit_transaction(status_vector, &transaction->rtr_handle);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case op_rollback:
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_rollback_transaction(status_vector, &transaction->rtr_handle);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case op_rollback_retaining:
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_rollback_retaining(status_vector, &transaction->rtr_handle);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case op_commit_retaining:
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_commit_retaining(status_vector, &transaction->rtr_handle);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case op_prepare:
|
2003-08-30 03:43:08 +02:00
|
|
|
if (!isc_prepare_transaction(status_vector, &transaction->rtr_handle))
|
2001-05-23 15:26:42 +02:00
|
|
|
transaction->rtr_flags |= RTR_limbo;
|
|
|
|
break;
|
|
|
|
}
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!status_vector[1])
|
|
|
|
if (operation == op_commit || operation == op_rollback) {
|
|
|
|
REMOTE_cleanup_transaction(transaction);
|
|
|
|
release_transaction(transaction);
|
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::execute_immediate(P_OP op, P_SQLST * exnow, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/*****************************************
|
|
|
|
*
|
|
|
|
* e x e c u t e _ i m m e d i a t e
|
|
|
|
*
|
|
|
|
*****************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* process an execute immediate DSQL packet
|
|
|
|
*
|
|
|
|
*****************************************/
|
2003-08-28 15:07:29 +02:00
|
|
|
RTR transaction = NULL;
|
2004-02-24 06:34:44 +01:00
|
|
|
USHORT in_blr_length, in_msg_type, parser_version,
|
|
|
|
out_blr_length, out_msg_type;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
RDB rdb = this->port_context;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/** Do not call CHECK_HANDLE if this is the start of a transaction **/
|
|
|
|
if (this->port_objects && exnow->p_sqlst_transaction) {
|
|
|
|
CHECK_HANDLE_MEMBER(transaction,
|
|
|
|
RTR,
|
|
|
|
type_rtr,
|
|
|
|
exnow->p_sqlst_transaction,
|
|
|
|
isc_bad_trans_handle);
|
|
|
|
}
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
USHORT in_msg_length = 0, out_msg_length = 0;
|
|
|
|
UCHAR* in_msg = NULL;
|
|
|
|
UCHAR* out_msg = NULL;
|
|
|
|
UCHAR* in_blr;
|
|
|
|
UCHAR* out_blr;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (op == op_exec_immediate2)
|
|
|
|
{
|
|
|
|
in_blr_length = exnow->p_sqlst_blr.cstr_length;
|
|
|
|
in_blr = exnow->p_sqlst_blr.cstr_address;
|
|
|
|
in_msg_type = exnow->p_sqlst_message_number;
|
|
|
|
if (this->port_statement->rsr_bind_format)
|
|
|
|
{
|
|
|
|
in_msg_length = this->port_statement->rsr_bind_format->fmt_length;
|
|
|
|
in_msg = this->port_statement->rsr_message->msg_address;
|
|
|
|
}
|
|
|
|
out_blr_length = exnow->p_sqlst_out_blr.cstr_length;
|
|
|
|
out_blr = exnow->p_sqlst_out_blr.cstr_address;
|
|
|
|
out_msg_type = exnow->p_sqlst_out_message_number;
|
|
|
|
if (this->port_statement->rsr_select_format)
|
|
|
|
{
|
|
|
|
out_msg_length =
|
|
|
|
this->port_statement->rsr_select_format->fmt_length;
|
|
|
|
if (!this->port_statement->rsr_message->msg_address)
|
|
|
|
{
|
|
|
|
/* TMN: Obvious bugfix. Please look at your compilers warnings. */
|
|
|
|
/* They are not enemies, they're friends! */
|
|
|
|
/* port->port_statement->rsr_message->msg_address = &port->port_statement->rsr_message->msg_buffer; */
|
|
|
|
this->port_statement->rsr_message->msg_address =
|
|
|
|
this->port_statement->rsr_message->msg_buffer;
|
|
|
|
}
|
|
|
|
out_msg = this->port_statement->rsr_message->msg_address;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
in_blr_length = out_blr_length = 0;
|
|
|
|
in_blr = out_blr = NULL;
|
|
|
|
in_msg_type = out_msg_type = 0;
|
|
|
|
}
|
|
|
|
|
2004-05-03 01:06:37 +02:00
|
|
|
FB_API_HANDLE handle = (transaction) ? transaction->rtr_handle : 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Since the API to GDS_DSQL_EXECUTE_IMMED is public and can not be changed, there needs to
|
|
|
|
* be a way to send the parser version to DSQL so that the parser can compare the keyword
|
|
|
|
* version to the parser version. To accomplish this, the parser version is combined with
|
|
|
|
* the client dialect and sent across that way. In dsql8_execute_immediate, the parser version
|
|
|
|
* and client dialect are separated and passed on to their final desintations. The information
|
|
|
|
* is combined as follows:
|
|
|
|
* Dialect * 10 + parser_version
|
|
|
|
*
|
|
|
|
* and is extracted in dsql8_execute_immediate as follows:
|
2004-09-05 15:53:56 +02:00
|
|
|
* parser_version = ((dialect * 10) + parser_version) % 10
|
|
|
|
* client_dialect = ((dialect * 10) + parser_version) / 10
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
* For example, parser_version = 1 and client dialect = 1
|
|
|
|
*
|
|
|
|
* combined = (1 * 10) + 1 == 11
|
|
|
|
*
|
2004-09-05 15:53:56 +02:00
|
|
|
* parser = (combined) % 10 == 1
|
|
|
|
* dialect = (combined) / 10 == 1
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
parser_version = (this->port_protocol < PROTOCOL_VERSION10) ? 1 : 2;
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
GDS_DSQL_EXECUTE_IMMED(status_vector,
|
2002-12-16 17:52:35 +01:00
|
|
|
&rdb->rdb_handle,
|
|
|
|
&handle,
|
2001-05-23 15:26:42 +02:00
|
|
|
exnow->p_sqlst_SQL_str.cstr_length,
|
|
|
|
reinterpret_cast<char*>(exnow->p_sqlst_SQL_str.cstr_address),
|
|
|
|
(USHORT) ((exnow->p_sqlst_SQL_dialect * 10) +
|
|
|
|
parser_version),
|
|
|
|
in_blr_length,
|
|
|
|
reinterpret_cast<char*>(in_blr),
|
|
|
|
in_msg_type,
|
|
|
|
in_msg_length,
|
|
|
|
reinterpret_cast<char*>(in_msg),
|
|
|
|
out_blr_length,
|
|
|
|
reinterpret_cast<char*>(out_blr),
|
|
|
|
out_msg_type,
|
|
|
|
out_msg_length,
|
|
|
|
reinterpret_cast<char*>(out_msg));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (op == op_exec_immediate2)
|
|
|
|
{
|
|
|
|
this->port_statement->rsr_format =
|
|
|
|
this->port_statement->rsr_select_format;
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
sendL->p_operation = op_sql_response;
|
|
|
|
sendL->p_sqldata.p_sqldata_messages = (status_vector[1]
|
2001-05-23 15:26:42 +02:00
|
|
|
|| !out_msg) ? 0 : 1;
|
2004-05-12 21:39:17 +02:00
|
|
|
this->send_partial(sendL);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!status_vector[1]) {
|
|
|
|
if (transaction && !handle) {
|
|
|
|
REMOTE_cleanup_transaction(transaction);
|
|
|
|
release_transaction(transaction);
|
|
|
|
transaction = NULL;
|
|
|
|
}
|
|
|
|
else if (!transaction && handle) {
|
|
|
|
if (!(transaction = make_transaction(rdb, handle))) {
|
|
|
|
status_vector[0] = isc_arg_gds;
|
|
|
|
status_vector[1] = isc_too_many_handles;
|
|
|
|
status_vector[2] = isc_arg_end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response( sendL,
|
2001-05-23 15:26:42 +02:00
|
|
|
(OBJCT) (transaction ? transaction->rtr_id : 0),
|
|
|
|
0,
|
|
|
|
status_vector);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::execute_statement(P_OP op, P_SQLDATA* sqldata, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/*****************************************
|
|
|
|
*
|
|
|
|
* e x e c u t e _ s t a t e m e n t
|
|
|
|
*
|
|
|
|
*****************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Execute a non-SELECT dynamic SQL statement.
|
|
|
|
*
|
|
|
|
*****************************************/
|
2003-08-28 15:07:29 +02:00
|
|
|
RTR transaction = NULL;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/** Do not call CHECK_HANDLE if this is the start of a transaction **/
|
|
|
|
if (sqldata->p_sqldata_transaction)
|
|
|
|
{
|
|
|
|
CHECK_HANDLE_MEMBER(transaction,
|
|
|
|
RTR,
|
|
|
|
type_rtr,
|
|
|
|
sqldata->p_sqldata_transaction,
|
|
|
|
isc_bad_trans_handle);
|
|
|
|
}
|
2004-02-24 06:34:44 +01:00
|
|
|
|
|
|
|
RSR statement;
|
2001-05-23 15:26:42 +02:00
|
|
|
CHECK_HANDLE_MEMBER(statement,
|
|
|
|
RSR,
|
|
|
|
type_rsr,
|
|
|
|
sqldata->p_sqldata_statement,
|
|
|
|
isc_bad_req_handle);
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
USHORT in_msg_length = 0, out_msg_length = 0;
|
|
|
|
UCHAR* in_msg = NULL;
|
|
|
|
UCHAR* out_msg = NULL;
|
|
|
|
USHORT out_msg_type, out_blr_length;
|
|
|
|
UCHAR* out_blr;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
if (statement->rsr_format)
|
|
|
|
{
|
|
|
|
in_msg_length = statement->rsr_format->fmt_length;
|
|
|
|
in_msg = statement->rsr_message->msg_address;
|
|
|
|
}
|
|
|
|
if (op == op_execute2)
|
|
|
|
{
|
|
|
|
out_blr_length = sqldata->p_sqldata_out_blr.cstr_length;
|
|
|
|
out_blr = sqldata->p_sqldata_out_blr.cstr_address;
|
|
|
|
out_msg_type = sqldata->p_sqldata_out_message_number;
|
|
|
|
if (this->port_statement->rsr_select_format)
|
|
|
|
{
|
|
|
|
out_msg_length =
|
|
|
|
this->port_statement->rsr_select_format->fmt_length;
|
|
|
|
out_msg = this->port_statement->rsr_message->msg_buffer;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
out_blr_length = 0;
|
|
|
|
out_msg_type = 0;
|
|
|
|
out_blr = NULL;
|
|
|
|
}
|
|
|
|
statement->rsr_flags &= ~RSR_fetched;
|
|
|
|
|
2004-05-03 01:06:37 +02:00
|
|
|
FB_API_HANDLE handle = (transaction) ? transaction->rtr_handle : 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
GDS_DSQL_EXECUTE(status_vector,
|
2002-12-16 17:52:35 +01:00
|
|
|
&handle,
|
|
|
|
&statement->rsr_handle,
|
2001-05-23 15:26:42 +02:00
|
|
|
sqldata->p_sqldata_blr.cstr_length,
|
|
|
|
reinterpret_cast<char*>(sqldata->p_sqldata_blr.cstr_address),
|
|
|
|
sqldata->p_sqldata_message_number,
|
|
|
|
in_msg_length,
|
|
|
|
reinterpret_cast<char*>(in_msg),
|
|
|
|
out_blr_length,
|
|
|
|
reinterpret_cast<char*>(out_blr),
|
|
|
|
out_msg_type,
|
|
|
|
out_msg_length,
|
|
|
|
reinterpret_cast<char*>(out_msg));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (op == op_execute2)
|
|
|
|
{
|
|
|
|
this->port_statement->rsr_format =
|
|
|
|
this->port_statement->rsr_select_format;
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
sendL->p_operation = op_sql_response;
|
|
|
|
sendL->p_sqldata.p_sqldata_messages = (status_vector[1]
|
2001-05-23 15:26:42 +02:00
|
|
|
|| !out_msg) ? 0 : 1;
|
2004-05-12 21:39:17 +02:00
|
|
|
this->send_partial(sendL);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!status_vector[1]) {
|
|
|
|
if (transaction && !handle) {
|
|
|
|
REMOTE_cleanup_transaction(transaction);
|
|
|
|
release_transaction(transaction);
|
|
|
|
transaction = NULL;
|
|
|
|
}
|
|
|
|
else if (!transaction && handle) {
|
|
|
|
if (!(transaction = make_transaction(statement->rsr_rdb, handle))) {
|
|
|
|
status_vector[0] = isc_arg_gds;
|
|
|
|
status_vector[1] = isc_too_many_handles;
|
|
|
|
status_vector[2] = isc_arg_end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
statement->rsr_rtr = transaction;
|
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response( sendL,
|
2001-05-23 15:26:42 +02:00
|
|
|
(OBJCT) (transaction ? transaction->rtr_id : 0),
|
|
|
|
0,
|
|
|
|
status_vector);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::fetch(P_SQLDATA * sqldata, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/*****************************************
|
|
|
|
*
|
|
|
|
* f e t c h
|
|
|
|
*
|
|
|
|
*****************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Fetch next record from a dynamic SQL cursor.
|
|
|
|
*
|
|
|
|
*****************************************/
|
|
|
|
RSR statement;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
CHECK_HANDLE_MEMBER(statement,
|
|
|
|
RSR,
|
|
|
|
type_rsr,
|
|
|
|
sqldata->p_sqldata_statement,
|
|
|
|
isc_bad_req_handle);
|
|
|
|
|
|
|
|
if (statement->rsr_flags & RSR_blob) {
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->fetch_blob(sqldata, sendL);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
USHORT msg_length;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (statement->rsr_format) {
|
|
|
|
msg_length = statement->rsr_format->fmt_length;
|
2004-01-28 08:50:41 +01:00
|
|
|
}
|
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
msg_length = 0;
|
|
|
|
}
|
2004-02-24 06:34:44 +01:00
|
|
|
USHORT count = ((this->port_flags & PORT_rpc) ||
|
2001-05-23 15:26:42 +02:00
|
|
|
(statement->rsr_flags & RSR_no_batch)) ? 1 :
|
|
|
|
sqldata->p_sqldata_messages;
|
2004-02-24 06:34:44 +01:00
|
|
|
USHORT count2 = (statement->rsr_flags & RSR_no_batch) ? 0 : count;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* On first fetch, clear the end-of-stream flag & reset the message buffers */
|
|
|
|
|
|
|
|
if (!(statement->rsr_flags & RSR_fetched)) {
|
|
|
|
statement->rsr_flags &= ~(RSR_eof | RSR_stream_err);
|
|
|
|
memset(statement->rsr_status_vector, 0,
|
|
|
|
sizeof(statement->rsr_status_vector));
|
2004-02-24 06:34:44 +01:00
|
|
|
REM_MSG message = statement->rsr_message;
|
|
|
|
if (message != NULL) {
|
2001-05-23 15:26:42 +02:00
|
|
|
statement->rsr_buffer = message;
|
2003-09-11 20:59:34 +02:00
|
|
|
while (true) {
|
2001-05-23 15:26:42 +02:00
|
|
|
message->msg_address = NULL;
|
|
|
|
message = message->msg_next;
|
|
|
|
if (message == statement->rsr_message)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get ready to ship the data out */
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
P_SQLDATA* response = &sendL->p_sqldata;
|
|
|
|
sendL->p_operation = op_fetch_response;
|
2001-05-23 15:26:42 +02:00
|
|
|
response->p_sqldata_statement = sqldata->p_sqldata_statement;
|
|
|
|
response->p_sqldata_status = 0;
|
|
|
|
response->p_sqldata_messages = 1;
|
2004-02-24 06:34:44 +01:00
|
|
|
ISC_STATUS s = 0;
|
|
|
|
REM_MSG message = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Check to see if any messages are already sitting around */
|
|
|
|
|
2003-09-11 20:59:34 +02:00
|
|
|
while (true) {
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Have we exhausted the cache & reached cursor EOF? */
|
|
|
|
if ((statement->rsr_flags & RSR_eof) && !statement->rsr_msgs_waiting) {
|
|
|
|
statement->rsr_flags &= ~RSR_eof;
|
|
|
|
s = 100;
|
|
|
|
count2 = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Have we exhausted the cache & have a pending error? */
|
|
|
|
if ((statement->rsr_flags & RSR_stream_err)
|
2004-02-24 06:34:44 +01:00
|
|
|
&& !statement->rsr_msgs_waiting)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
statement->rsr_flags &= ~RSR_stream_err;
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0,
|
2001-05-23 15:26:42 +02:00
|
|
|
statement->rsr_status_vector);
|
|
|
|
}
|
|
|
|
|
|
|
|
message = statement->rsr_buffer;
|
|
|
|
|
|
|
|
/* Make sure message can be de referenced, if not then return false */
|
|
|
|
if (message == NULL)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
/* If we don't have a message cached, get one from the
|
|
|
|
access method. */
|
|
|
|
|
|
|
|
if (!message->msg_address) {
|
2003-11-04 00:59:24 +01:00
|
|
|
fb_assert(statement->rsr_msgs_waiting == 0);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
s = GDS_DSQL_FETCH(status_vector,
|
2002-12-16 17:52:35 +01:00
|
|
|
&statement->rsr_handle,
|
2001-05-23 15:26:42 +02:00
|
|
|
sqldata->p_sqldata_blr.cstr_length,
|
2003-11-28 07:48:34 +01:00
|
|
|
reinterpret_cast<const char*>(sqldata->p_sqldata_blr.cstr_address),
|
2001-05-23 15:26:42 +02:00
|
|
|
sqldata->p_sqldata_message_number,
|
|
|
|
msg_length,
|
|
|
|
reinterpret_cast<char*>(message->msg_buffer));
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
statement->rsr_flags |= RSR_fetched;
|
|
|
|
if (s) {
|
|
|
|
if (s == 100 || s == 101) {
|
|
|
|
count2 = 0;
|
|
|
|
break;
|
2004-01-28 08:50:41 +01:00
|
|
|
}
|
|
|
|
else {
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
message->msg_address = message->msg_buffer;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* Take a message from the outqoing queue */
|
2003-11-04 00:59:24 +01:00
|
|
|
fb_assert(statement->rsr_msgs_waiting >= 1);
|
2001-05-23 15:26:42 +02:00
|
|
|
statement->rsr_msgs_waiting--;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* For compatibility with Protocol 7, we must break out of the
|
|
|
|
loop before sending the last record. */
|
|
|
|
|
|
|
|
count--;
|
|
|
|
if (this->port_protocol <= PROTOCOL_VERSION7 && count <= 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* There's a buffer waiting -- send it */
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
if (!this->send_partial(sendL)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
message->msg_address = NULL;
|
|
|
|
|
|
|
|
/* If we've sent the requested amount, break out of loop */
|
|
|
|
|
|
|
|
if (count <= 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
response->p_sqldata_status = s;
|
|
|
|
response->p_sqldata_messages = 0;
|
2004-05-12 21:39:17 +02:00
|
|
|
this->send(sendL);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (message) {
|
|
|
|
message->msg_address = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Since we have a little time on our hands while this packet is sent
|
|
|
|
and processed, get the next batch of records. Start by finding the
|
|
|
|
next free buffer. */
|
|
|
|
|
|
|
|
message = statement->rsr_buffer;
|
2004-02-24 06:34:44 +01:00
|
|
|
REM_MSG next = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
while (message->msg_address && message->msg_next != statement->rsr_buffer)
|
|
|
|
message = message->msg_next;
|
|
|
|
|
|
|
|
for (; count2; --count2) {
|
|
|
|
if (message->msg_address) {
|
|
|
|
if (!next)
|
|
|
|
for (next = statement->rsr_buffer; next->msg_next != message;
|
|
|
|
next = next->msg_next);
|
2004-05-17 12:22:34 +02:00
|
|
|
message = (REM_MSG) ALLR_block(type_msg, statement->rsr_fmt_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
message->msg_number = next->msg_number;
|
|
|
|
message->msg_next = next->msg_next;
|
|
|
|
next->msg_next = message;
|
|
|
|
next = message;
|
|
|
|
}
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
s = GDS_DSQL_FETCH(status_vector,
|
2002-12-16 17:52:35 +01:00
|
|
|
&statement->rsr_handle,
|
2001-05-23 15:26:42 +02:00
|
|
|
sqldata->p_sqldata_blr.cstr_length,
|
|
|
|
reinterpret_cast<char*>(sqldata->p_sqldata_blr.cstr_address),
|
|
|
|
sqldata->p_sqldata_message_number,
|
|
|
|
msg_length,
|
|
|
|
reinterpret_cast<char*>(message->msg_buffer));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
if (s) {
|
|
|
|
if (status_vector[1]) {
|
|
|
|
/* If already have an error queued, don't overwrite it */
|
|
|
|
if (!(statement->rsr_flags & RSR_stream_err)) {
|
|
|
|
statement->rsr_flags |= RSR_stream_err;
|
|
|
|
memcpy(statement->rsr_status_vector, status_vector,
|
|
|
|
sizeof(statement->rsr_status_vector));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (s == 100)
|
|
|
|
statement->rsr_flags |= RSR_eof;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
message->msg_address = message->msg_buffer;
|
|
|
|
message = message->msg_next;
|
|
|
|
statement->rsr_msgs_waiting++;
|
|
|
|
}
|
|
|
|
|
2003-09-13 11:15:06 +02:00
|
|
|
return TRUE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::fetch_blob(P_SQLDATA * sqldata, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/*****************************************
|
|
|
|
*
|
|
|
|
* f e t c h _ b l o b
|
|
|
|
*
|
|
|
|
*****************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Fetch next record from a dynamic SQL cursor.
|
|
|
|
*
|
|
|
|
*****************************************/
|
|
|
|
RSR statement;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
CHECK_HANDLE_MEMBER(statement,
|
|
|
|
RSR,
|
|
|
|
type_rsr,
|
|
|
|
sqldata->p_sqldata_statement,
|
|
|
|
isc_bad_req_handle);
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
USHORT msg_length;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (statement->rsr_format)
|
|
|
|
msg_length = statement->rsr_format->fmt_length;
|
|
|
|
else
|
|
|
|
msg_length = 0;
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
REM_MSG message = statement->rsr_message;
|
|
|
|
if (message != NULL)
|
2001-05-23 15:26:42 +02:00
|
|
|
statement->rsr_buffer = message;
|
|
|
|
|
|
|
|
/* Get ready to ship the data out */
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
P_SQLDATA* response = &sendL->p_sqldata;
|
|
|
|
sendL->p_operation = op_fetch_response;
|
2001-05-23 15:26:42 +02:00
|
|
|
response->p_sqldata_statement = sqldata->p_sqldata_statement;
|
|
|
|
response->p_sqldata_status = 0;
|
|
|
|
response->p_sqldata_messages = 1;
|
2004-02-24 06:34:44 +01:00
|
|
|
ISC_STATUS s = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
message = statement->rsr_buffer;
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
s = GDS_DSQL_FETCH(status_vector,
|
2002-12-16 17:52:35 +01:00
|
|
|
&statement->rsr_handle,
|
2001-05-23 15:26:42 +02:00
|
|
|
sqldata->p_sqldata_blr.cstr_length,
|
|
|
|
reinterpret_cast<char*>(sqldata->p_sqldata_blr.cstr_address),
|
|
|
|
sqldata->p_sqldata_message_number,
|
|
|
|
msg_length,
|
|
|
|
reinterpret_cast<char*>(message->msg_buffer));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!status_vector[1] ||
|
2004-02-24 06:34:44 +01:00
|
|
|
status_vector[1] != isc_segment || status_vector[1] != isc_segstr_eof)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
message->msg_address = message->msg_buffer;
|
|
|
|
response->p_sqldata_status = s;
|
|
|
|
response->p_sqldata_messages =
|
2003-11-08 17:40:17 +01:00
|
|
|
(status_vector[1] == isc_segstr_eof) ? 0 : 1;
|
2004-05-12 21:39:17 +02:00
|
|
|
this->send_partial(sendL);
|
2001-05-23 15:26:42 +02:00
|
|
|
message->msg_address = NULL;
|
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
OBJCT rem_port::get_id(BLK block)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* g e t _ i d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Allocate an object slot for an object.
|
|
|
|
* If the object vector cannot be expanded
|
|
|
|
* to accomodate the object slot then
|
|
|
|
* REMOTE_set_object() will return a NULL
|
|
|
|
* object slot.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
/* If there isn't a vector, obvious the object goes in slot 1.
|
|
|
|
Reserve slot 0 so we can distinguish something from nothing.
|
|
|
|
NOTE: prior to server version 4.5.0 id==0 COULD be used - so
|
|
|
|
only the server side can now depend on id==0 meaning "invalid id" */
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
rem_vec* vector = this->port_object_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!vector) {
|
|
|
|
return REMOTE_set_object(this, block, (OBJCT) 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Search vector for an empty slot. If we find one, use it
|
|
|
|
|
2004-02-20 07:43:27 +01:00
|
|
|
blk** p = vector->vec_object + 1;
|
|
|
|
for (const blk* const* const end = vector->vec_object + vector->vec_count;
|
|
|
|
p < end; p++)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
if (!*p) {
|
|
|
|
*p = block;
|
|
|
|
return (OBJCT) (p - vector->vec_object);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Vector is full -- somebody will need to expand it */
|
|
|
|
|
|
|
|
return REMOTE_set_object(this, block, (OBJCT) vector->vec_count);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static bool get_next_msg_no(rrq* request,
|
2003-09-11 20:59:34 +02:00
|
|
|
USHORT incarnation,
|
|
|
|
USHORT * msg_number)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* g e t _ n e x t _ m s g _ n o
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Return the number of the next message
|
|
|
|
* in the request.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2004-02-24 06:34:44 +01:00
|
|
|
UCHAR info_buffer[128];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_request_info(status_vector, &request->rrq_handle, incarnation,
|
2003-10-29 11:53:47 +01:00
|
|
|
sizeof(request_info), reinterpret_cast<const SCHAR*>(request_info),
|
2003-08-30 03:43:08 +02:00
|
|
|
sizeof(info_buffer), reinterpret_cast<char*>(info_buffer));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (status_vector[1])
|
2003-09-11 20:59:34 +02:00
|
|
|
return false;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-11 20:59:34 +02:00
|
|
|
bool result = false;
|
2004-02-24 06:34:44 +01:00
|
|
|
for (const UCHAR* info = info_buffer; *info != isc_info_end;) {
|
|
|
|
const USHORT l = (USHORT) gds__vax_integer(info + 1, 2);
|
|
|
|
const USHORT n = (USHORT) gds__vax_integer(info + 3, l);
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
switch (*info) {
|
2003-11-08 17:40:17 +01:00
|
|
|
case isc_info_state:
|
|
|
|
if (n != isc_info_req_send)
|
2003-09-11 20:59:34 +02:00
|
|
|
return false;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
case isc_info_message_number:
|
2001-05-23 15:26:42 +02:00
|
|
|
*msg_number = n;
|
2003-09-11 20:59:34 +02:00
|
|
|
result = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2003-09-11 20:59:34 +02:00
|
|
|
return false;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
info += 3 + l;
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::get_segment(P_SGMT* segment, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* g e t _ s e g m e n t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Get a single blob segment.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
RBL blob;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
CHECK_HANDLE_MEMBER(blob,
|
|
|
|
RBL,
|
|
|
|
type_rbl,
|
|
|
|
segment->p_sgmt_blob,
|
|
|
|
isc_bad_segstr_handle);
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
UCHAR temp_buffer[BLOB_LENGTH];
|
|
|
|
USHORT buffer_length = segment->p_sgmt_length;
|
|
|
|
UCHAR* buffer;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (buffer_length <= sizeof(temp_buffer))
|
|
|
|
buffer = temp_buffer;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (buffer_length > blob->rbl_buffer_length) {
|
|
|
|
if (blob->rbl_buffer != blob->rbl_data) {
|
|
|
|
ALLR_free(blob->rbl_buffer);
|
|
|
|
blob->rbl_buffer = blob->rbl_data;
|
|
|
|
blob->rbl_buffer_length = 1;
|
|
|
|
}
|
|
|
|
blob->rbl_buffer = ALLR_alloc((SLONG) buffer_length);
|
|
|
|
blob->rbl_buffer_length = buffer_length;
|
|
|
|
}
|
|
|
|
buffer = blob->rbl_buffer;
|
|
|
|
}
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("get_segment(server) allocate buffer %x\n", buffer);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2004-05-12 21:39:17 +02:00
|
|
|
sendL->p_resp.p_resp_data.cstr_address = buffer;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Be backwards compatible */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
USHORT length;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (this->port_flags & PORT_rpc)
|
|
|
|
{
|
|
|
|
length = 0;
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_get_segment(status_vector, &blob->rbl_handle, &length,
|
2001-05-23 15:26:42 +02:00
|
|
|
segment->p_sgmt_length,
|
2003-08-30 03:43:08 +02:00
|
|
|
reinterpret_cast<char*>(buffer));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2004-02-24 06:34:44 +01:00
|
|
|
const ISC_STATUS status =
|
2004-05-12 21:39:17 +02:00
|
|
|
this->send_response(sendL, blob->rbl_id, length, status_vector);
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("get_segment(server) free buffer %x\n", buffer);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2003-11-08 17:40:17 +01:00
|
|
|
if (status_vector[1] == isc_segstr_eof)
|
2001-05-23 15:26:42 +02:00
|
|
|
if (blob->rbl_buffer != blob->rbl_data) {
|
|
|
|
ALLR_free(blob->rbl_buffer);
|
|
|
|
blob->rbl_buffer = blob->rbl_data;
|
|
|
|
blob->rbl_buffer_length = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Gobble up a buffer's worth of segments */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
UCHAR* p = buffer;
|
|
|
|
ISC_STATUS state = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
while (buffer_length > 2) {
|
|
|
|
buffer_length -= 2;
|
|
|
|
p += 2;
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_get_segment(status_vector, &blob->rbl_handle, &length,
|
|
|
|
buffer_length, reinterpret_cast<char*>(p));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2003-11-08 17:40:17 +01:00
|
|
|
if (status_vector[1] == isc_segstr_eof)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
state = 2;
|
|
|
|
success(status_vector);
|
|
|
|
p -= 2;
|
|
|
|
break;
|
|
|
|
}
|
2003-11-08 17:40:17 +01:00
|
|
|
if (status_vector[1] && (status_vector[1] != isc_segment))
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
p -= 2;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
p[-2] = (UCHAR) length;
|
|
|
|
p[-1] = (UCHAR) (length >> 8);
|
|
|
|
p += length;
|
|
|
|
buffer_length -= length;
|
2003-11-08 17:40:17 +01:00
|
|
|
if (status_vector[1] == isc_segment) {
|
2001-05-23 15:26:42 +02:00
|
|
|
state = 1;
|
|
|
|
success(status_vector);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
const ISC_STATUS status = this->send_response(sendL,
|
2001-05-23 15:26:42 +02:00
|
|
|
(OBJCT)state,
|
|
|
|
(USHORT) (p - buffer),
|
|
|
|
status_vector);
|
|
|
|
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("get_segment(server) free buffer %x\n", buffer);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
if (status_vector[1] == isc_segstr_eof)
|
2001-05-23 15:26:42 +02:00
|
|
|
if (blob->rbl_buffer != blob->rbl_data) {
|
|
|
|
ALLR_free(blob->rbl_buffer);
|
|
|
|
blob->rbl_buffer = blob->rbl_data;
|
|
|
|
blob->rbl_buffer_length = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::get_slice(P_SLC * stuff, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* g e t _ s l i c e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Get an array slice.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
RTR transaction;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
RDB rdb = this->port_context;
|
2001-05-23 15:26:42 +02:00
|
|
|
CHECK_HANDLE_MEMBER(transaction,
|
|
|
|
RTR,
|
|
|
|
type_rtr,
|
|
|
|
stuff->p_slc_transaction,
|
|
|
|
isc_bad_trans_handle);
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
UCHAR temp_buffer[4096];
|
|
|
|
UCHAR* slice;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!stuff->p_slc_length)
|
|
|
|
slice = 0;
|
|
|
|
else {
|
|
|
|
if (stuff->p_slc_length <= sizeof(temp_buffer))
|
|
|
|
slice = temp_buffer;
|
|
|
|
else
|
|
|
|
slice = ALLR_alloc((SLONG) stuff->p_slc_length);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (slice) {
|
|
|
|
memset(slice, 0, stuff->p_slc_length);
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("get_slice(server) allocate buffer %x\n", slice);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
}
|
2004-05-12 21:39:17 +02:00
|
|
|
P_SLR* response = &sendL->p_slr;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_get_slice(status_vector, &rdb->rdb_handle, &transaction->rtr_handle,
|
2003-11-11 13:19:20 +01:00
|
|
|
(ISC_QUAD*) &stuff->p_slc_id, stuff->p_slc_sdl.cstr_length,
|
2003-12-03 09:19:24 +01:00
|
|
|
reinterpret_cast<const char*>(stuff->p_slc_sdl.cstr_address),
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_slc_parameters.cstr_length,
|
2003-12-03 09:19:24 +01:00
|
|
|
(const ISC_LONG*) stuff->p_slc_parameters.cstr_address,
|
2003-08-30 03:43:08 +02:00
|
|
|
stuff->p_slc_length, slice,
|
2004-01-21 08:18:30 +01:00
|
|
|
reinterpret_cast<SLONG*>(&response->p_slr_length));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
ISC_STATUS status;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (status_vector[1])
|
2004-05-12 21:39:17 +02:00
|
|
|
status = this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
else {
|
2004-05-12 21:39:17 +02:00
|
|
|
sendL->p_operation = op_slice;
|
2001-05-23 15:26:42 +02:00
|
|
|
response->p_slr_slice.lstr_address = slice;
|
|
|
|
response->p_slr_slice.lstr_length = response->p_slr_length;
|
2003-08-15 12:23:46 +02:00
|
|
|
response->p_slr_sdl = stuff->p_slc_sdl.cstr_address;
|
2001-05-23 15:26:42 +02:00
|
|
|
response->p_slr_sdl_length = stuff->p_slc_sdl.cstr_length;
|
2004-05-12 21:39:17 +02:00
|
|
|
this->send(sendL);
|
2001-05-23 15:26:42 +02:00
|
|
|
response->p_slr_sdl = 0;
|
2002-11-14 09:33:08 +01:00
|
|
|
status = FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (slice) {
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("get_slice(server) free buffer %x\n", slice);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
if (slice != temp_buffer)
|
|
|
|
ALLR_free(slice);
|
|
|
|
}
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::info(P_OP op, P_INFO * stuff, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n f o
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Get info for a blob, database, request, service,
|
|
|
|
* statement, or transaction.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
RBL blob;
|
|
|
|
RTR transaction;
|
|
|
|
RSR statement;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
RDB rdb = this->port_context;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Make sure there is a suitable temporary blob buffer */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
UCHAR* const buffer = ALLR_alloc((SLONG) stuff->p_info_buffer_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
memset(buffer, 0, stuff->p_info_buffer_length);
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("info(server) allocate buffer %x\n", buffer);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
UCHAR temp[1024];
|
|
|
|
UCHAR* temp_buffer;
|
2002-06-29 10:49:39 +02:00
|
|
|
if (op == op_info_database && stuff->p_info_buffer_length > sizeof(temp)) {
|
|
|
|
temp_buffer = ALLR_alloc((SLONG) stuff->p_info_buffer_length);
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("info(server) allocate buffer %x\n", temp_buffer);
|
2002-06-29 10:49:39 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else
|
|
|
|
temp_buffer = temp;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
switch (op) {
|
|
|
|
case op_info_blob:
|
|
|
|
CHECK_HANDLE_MEMBER(blob,
|
|
|
|
RBL,
|
|
|
|
type_rbl,
|
|
|
|
stuff->p_info_object,
|
|
|
|
isc_bad_segstr_handle);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_blob_info(status_vector, &blob->rbl_handle,
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_info_items.cstr_length,
|
2003-08-30 03:43:08 +02:00
|
|
|
reinterpret_cast<char*>(stuff->p_info_items.cstr_address),
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_info_buffer_length,
|
2003-08-30 03:43:08 +02:00
|
|
|
reinterpret_cast<char*>(buffer));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case op_info_database:
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_database_info(status_vector, &rdb->rdb_handle,
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_info_items.cstr_length,
|
2003-10-29 11:53:47 +01:00
|
|
|
reinterpret_cast<const char*>(stuff->p_info_items.cstr_address),
|
2002-06-29 10:49:39 +02:00
|
|
|
stuff->p_info_buffer_length /*sizeof (temp)*/,
|
|
|
|
reinterpret_cast<char*>(temp_buffer) /*temp*/);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!status_vector[1]) {
|
2004-02-24 06:34:44 +01:00
|
|
|
TEXT version[256];
|
2001-05-23 15:26:42 +02:00
|
|
|
sprintf(version, "%s/%s", GDS_VERSION,
|
|
|
|
this->port_version->str_data);
|
2003-11-06 04:03:36 +01:00
|
|
|
MERGE_database_info(temp_buffer /*temp*/, buffer, stuff->p_info_buffer_length,
|
2001-05-23 15:26:42 +02:00
|
|
|
IMPLEMENTATION, 4, 1,
|
2004-02-24 06:34:44 +01:00
|
|
|
reinterpret_cast<const UCHAR*>(version),
|
2001-05-23 15:26:42 +02:00
|
|
|
reinterpret_cast<UCHAR*>(this->port_host->str_data),
|
|
|
|
0);
|
|
|
|
}
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case op_info_request:
|
2004-05-21 08:16:17 +02:00
|
|
|
{
|
2004-05-12 23:47:36 +02:00
|
|
|
rrq* requestL;
|
|
|
|
CHECK_HANDLE_MEMBER(requestL,
|
2004-01-28 08:50:41 +01:00
|
|
|
rrq*,
|
2001-05-23 15:26:42 +02:00
|
|
|
type_rrq,
|
|
|
|
stuff->p_info_object,
|
|
|
|
isc_bad_req_handle);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2004-05-12 23:47:36 +02:00
|
|
|
isc_request_info(status_vector, &requestL->rrq_handle,
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_info_incarnation,
|
|
|
|
stuff->p_info_items.cstr_length,
|
2003-10-29 11:53:47 +01:00
|
|
|
reinterpret_cast<const char*>(stuff->p_info_items.cstr_address),
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_info_buffer_length,
|
2003-10-29 11:53:47 +01:00
|
|
|
reinterpret_cast<char*>(buffer));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2004-05-21 08:16:17 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
case op_info_transaction:
|
|
|
|
CHECK_HANDLE_MEMBER(transaction,
|
|
|
|
RTR,
|
|
|
|
type_rtr,
|
|
|
|
stuff->p_info_object,
|
|
|
|
isc_bad_trans_handle);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_transaction_info(status_vector, &transaction->rtr_handle,
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_info_items.cstr_length,
|
2003-10-29 11:53:47 +01:00
|
|
|
reinterpret_cast<const char*>(stuff->p_info_items.cstr_address),
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_info_buffer_length,
|
2003-08-30 03:43:08 +02:00
|
|
|
reinterpret_cast < char *>(buffer));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case op_service_info:
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
isc_service_query(status_vector,
|
2002-12-16 17:52:35 +01:00
|
|
|
&rdb->rdb_handle,
|
2001-05-23 15:26:42 +02:00
|
|
|
NULL,
|
|
|
|
stuff->p_info_items.cstr_length,
|
2003-10-29 11:53:47 +01:00
|
|
|
reinterpret_cast<
|
|
|
|
const char*>(stuff->p_info_items.cstr_address),
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_info_recv_items.cstr_length,
|
2003-10-29 11:53:47 +01:00
|
|
|
reinterpret_cast<
|
|
|
|
const char*>(stuff->p_info_recv_items.cstr_address),
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_info_buffer_length,
|
2003-11-28 07:48:34 +01:00
|
|
|
reinterpret_cast<char*>(buffer));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case op_info_sql:
|
|
|
|
CHECK_HANDLE_MEMBER(statement,
|
|
|
|
RSR,
|
|
|
|
type_rsr,
|
|
|
|
stuff->p_info_object,
|
|
|
|
isc_bad_req_handle);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
GDS_DSQL_SQL_INFO(status_vector,
|
2002-12-16 17:52:35 +01:00
|
|
|
&statement->rsr_handle,
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_info_items.cstr_length,
|
2003-10-29 11:53:47 +01:00
|
|
|
reinterpret_cast<
|
|
|
|
const char*>(stuff->p_info_items.cstr_address),
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_info_buffer_length,
|
|
|
|
reinterpret_cast < char *>(buffer));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2002-06-29 10:49:39 +02:00
|
|
|
if (temp_buffer != temp) {
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf ("info(server) free buffer %x\n", temp_buffer);
|
2002-06-29 10:49:39 +02:00
|
|
|
#endif
|
|
|
|
ALLR_free(temp_buffer);
|
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Send a response that includes the segment. */
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
sendL->p_resp.p_resp_data.cstr_address = buffer;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
const ISC_STATUS status = this->send_response(sendL, stuff->p_info_object,
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_info_buffer_length, status_vector);
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("info(server) free buffer %x\n", buffer);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
ALLR_free(buffer);
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::insert(P_SQLDATA * sqldata, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/*****************************************
|
|
|
|
*
|
|
|
|
* i n s e r t
|
|
|
|
*
|
|
|
|
*****************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Insert next record into a dynamic SQL cursor.
|
|
|
|
*
|
|
|
|
*****************************************/
|
|
|
|
RSR statement;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
CHECK_HANDLE_MEMBER(statement,
|
|
|
|
RSR,
|
|
|
|
type_rsr,
|
|
|
|
sqldata->p_sqldata_statement,
|
|
|
|
isc_bad_req_handle);
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
USHORT msg_length;
|
|
|
|
const UCHAR* msg;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (statement->rsr_format)
|
|
|
|
{
|
|
|
|
msg_length = statement->rsr_format->fmt_length;
|
|
|
|
msg = statement->rsr_message->msg_address;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
msg_length = 0;
|
|
|
|
msg = NULL;
|
|
|
|
}
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
GDS_DSQL_INSERT(status_vector,
|
2002-12-16 17:52:35 +01:00
|
|
|
&statement->rsr_handle,
|
2001-05-23 15:26:42 +02:00
|
|
|
sqldata->p_sqldata_blr.cstr_length,
|
2003-11-28 07:48:34 +01:00
|
|
|
reinterpret_cast<const char*>(sqldata->p_sqldata_blr.cstr_address),
|
2001-05-23 15:26:42 +02:00
|
|
|
sqldata->p_sqldata_message_number, msg_length,
|
2003-11-28 07:48:34 +01:00
|
|
|
reinterpret_cast<const char*>(msg));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-03 01:06:37 +02:00
|
|
|
static RTR make_transaction (RDB rdb, FB_API_HANDLE handle)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* m a k e _ t r a n s a c t i o n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Create a local transaction handle.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-05-17 12:22:34 +02:00
|
|
|
RTR transaction = (RTR) ALLR_block(type_rtr, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
transaction->rtr_rdb = rdb;
|
|
|
|
transaction->rtr_handle = handle;
|
|
|
|
if (transaction->rtr_id = rdb->rdb_port->get_id(&transaction->rtr_header))
|
|
|
|
{
|
|
|
|
transaction->rtr_next = rdb->rdb_transactions;
|
|
|
|
rdb->rdb_transactions = transaction;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
ALLR_release(transaction);
|
2003-09-01 09:58:04 +02:00
|
|
|
transaction = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return transaction;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::open_blob(P_OP op, P_BLOB* stuff, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* o p e n _ b l o b
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Open or create a new blob.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
RTR transaction;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
CHECK_HANDLE_MEMBER(transaction,
|
|
|
|
RTR,
|
|
|
|
type_rtr,
|
|
|
|
stuff->p_blob_transaction,
|
|
|
|
isc_bad_trans_handle);
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
RDB rdb = this->port_context;
|
2004-05-03 01:06:37 +02:00
|
|
|
FB_API_HANDLE handle = 0;
|
2004-02-24 06:34:44 +01:00
|
|
|
USHORT bpb_length = 0;
|
2003-10-29 11:53:47 +01:00
|
|
|
const UCHAR* bpb = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (op == op_open_blob2 || op == op_create_blob2) {
|
|
|
|
bpb_length = stuff->p_blob_bpb.cstr_length;
|
|
|
|
bpb = stuff->p_blob_bpb.cstr_address;
|
|
|
|
}
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
if (op == op_open_blob || op == op_open_blob2)
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_open_blob2(status_vector, &rdb->rdb_handle,
|
|
|
|
&transaction->rtr_handle, &handle,
|
2003-11-11 13:19:20 +01:00
|
|
|
(ISC_QUAD*) &stuff->p_blob_id, bpb_length, bpb);
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_create_blob2(status_vector, &rdb->rdb_handle, &transaction->rtr_handle,
|
2004-05-12 21:39:17 +02:00
|
|
|
&handle, (ISC_QUAD*) &sendL->p_resp.p_resp_blob_id,
|
2003-10-29 11:53:47 +01:00
|
|
|
bpb_length, reinterpret_cast<const char*>(bpb));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
USHORT object;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (status_vector[1])
|
|
|
|
object = 0;
|
|
|
|
else {
|
2004-05-17 12:22:34 +02:00
|
|
|
RBL blob = (RBL) ALLR_block(type_rbl, 1);
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("open_blob(server) allocate blob %x\n", blob);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
blob->rbl_buffer_length = 1;
|
|
|
|
blob->rbl_buffer = blob->rbl_data;
|
|
|
|
blob->rbl_handle = handle;
|
|
|
|
blob->rbl_rdb = rdb;
|
|
|
|
if (blob->rbl_id = this->get_id(&blob->rbl_header))
|
|
|
|
{
|
|
|
|
object = blob->rbl_id;
|
|
|
|
blob->rbl_rtr = transaction;
|
|
|
|
blob->rbl_next = transaction->rtr_blobs;
|
|
|
|
transaction->rtr_blobs = blob;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
object = 0;
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_cancel_blob(status_vector, &blob->rbl_handle);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
ALLR_release(blob);
|
|
|
|
status_vector[0] = isc_arg_gds;
|
|
|
|
status_vector[1] = isc_too_many_handles;
|
|
|
|
status_vector[2] = isc_arg_end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, object, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::prepare(P_PREP * stuff, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p r e p a r e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* End a transaction.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
RTR transaction;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
CHECK_HANDLE_MEMBER(transaction,
|
|
|
|
RTR,
|
|
|
|
type_rtr,
|
|
|
|
stuff->p_prep_transaction,
|
|
|
|
isc_bad_trans_handle);
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
if (!isc_prepare_transaction2(status_vector, &transaction->rtr_handle,
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_prep_data.cstr_length,
|
2003-08-30 03:43:08 +02:00
|
|
|
stuff->p_prep_data.cstr_address))
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
transaction->rtr_flags |= RTR_limbo;
|
|
|
|
}
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 23:47:36 +02:00
|
|
|
ISC_STATUS rem_port::prepare_statement(P_SQLST * prepareL, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/*****************************************
|
|
|
|
*
|
|
|
|
* p r e p a r e _ s t a t m e n t
|
|
|
|
*
|
|
|
|
*****************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Prepare a dynamic SQL statement for execution.
|
|
|
|
*
|
|
|
|
*****************************************/
|
2003-08-28 15:07:29 +02:00
|
|
|
RTR transaction = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
RSR statement;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/** Do not call CHECK_HANDLE if this is the start of a transaction **/
|
2004-05-12 23:47:36 +02:00
|
|
|
if (prepareL->p_sqlst_transaction)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
CHECK_HANDLE_MEMBER(transaction,
|
|
|
|
RTR,
|
|
|
|
type_rtr,
|
2004-05-12 23:47:36 +02:00
|
|
|
prepareL->p_sqlst_transaction,
|
2001-05-23 15:26:42 +02:00
|
|
|
isc_bad_trans_handle);
|
|
|
|
}
|
|
|
|
CHECK_HANDLE_MEMBER(statement,
|
|
|
|
RSR,
|
|
|
|
type_rsr,
|
2004-05-12 23:47:36 +02:00
|
|
|
prepareL->p_sqlst_statement,
|
2001-05-23 15:26:42 +02:00
|
|
|
isc_bad_req_handle);
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
UCHAR local_buffer[1024];
|
|
|
|
UCHAR* buffer;
|
2004-05-12 23:47:36 +02:00
|
|
|
if (prepareL->p_sqlst_buffer_length > sizeof(local_buffer))
|
|
|
|
buffer = ALLR_alloc((SLONG) prepareL->p_sqlst_buffer_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
|
|
|
buffer = local_buffer;
|
|
|
|
|
2004-05-03 01:06:37 +02:00
|
|
|
FB_API_HANDLE handle = (transaction) ? transaction->rtr_handle : 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
|
|
|
/* Since the API to GDS_DSQL_PREPARE is public and can not be changed, there needs to
|
|
|
|
* be a way to send the parser version to DSQL so that the parser can compare the keyword
|
|
|
|
* version to the parser version. To accomplish this, the parser version is combined with
|
|
|
|
* the client dialect and sent across that way. In dsql8_prepare_statement, the parser version
|
|
|
|
* and client dialect are separated and passed on to their final desintations. The information
|
|
|
|
* is combined as follows:
|
|
|
|
* Dialect * 10 + parser_version
|
|
|
|
*
|
|
|
|
* and is extracted in dsql8_prepare_statement as follows:
|
2004-09-05 15:53:56 +02:00
|
|
|
* parser_version = ((dialect * 10) + parser_version) % 10
|
|
|
|
* client_dialect = ((dialect * 10) + parser_version) / 10
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
* For example, parser_version = 1 and client dialect = 1
|
|
|
|
*
|
|
|
|
* combined = (1 * 10) + 1 == 11
|
|
|
|
*
|
2004-09-05 15:53:56 +02:00
|
|
|
* parser = (combined) % 10 == 1
|
|
|
|
* dialect = (combined) / 10 == 1
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
2004-02-24 06:34:44 +01:00
|
|
|
const USHORT parser_version = (this->port_protocol < PROTOCOL_VERSION10) ? 1 : 2;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
GDS_DSQL_PREPARE(status_vector,
|
2002-12-16 17:52:35 +01:00
|
|
|
&handle,
|
|
|
|
&statement->rsr_handle,
|
2004-05-12 23:47:36 +02:00
|
|
|
prepareL->p_sqlst_SQL_str.cstr_length,
|
|
|
|
reinterpret_cast<const char*>(prepareL->p_sqlst_SQL_str.cstr_address),
|
|
|
|
(USHORT) ((prepareL->p_sqlst_SQL_dialect * 10) +
|
2001-05-23 15:26:42 +02:00
|
|
|
parser_version),
|
2004-05-12 23:47:36 +02:00
|
|
|
prepareL->p_sqlst_items.cstr_length,
|
|
|
|
reinterpret_cast<const char*>(prepareL->p_sqlst_items.cstr_address),
|
|
|
|
prepareL->p_sqlst_buffer_length,
|
2003-11-28 07:48:34 +01:00
|
|
|
reinterpret_cast<char*>(buffer));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (status_vector[1]) {
|
|
|
|
if (buffer != local_buffer)
|
|
|
|
ALLR_free(buffer);
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
REMOTE_reset_statement(statement);
|
|
|
|
|
|
|
|
statement->rsr_flags &= ~(RSR_blob | RSR_no_batch);
|
2004-02-24 06:34:44 +01:00
|
|
|
USHORT state = check_statement_type(statement);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (state == STMT_BLOB)
|
|
|
|
statement->rsr_flags |= RSR_blob;
|
|
|
|
else if (state == STMT_NO_BATCH)
|
|
|
|
statement->rsr_flags |= RSR_no_batch;
|
|
|
|
state = (state == STMT_BLOB) ? 1 : 0;
|
|
|
|
|
|
|
|
/* Send a response that includes the info requested. */
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
sendL->p_resp.p_resp_data.cstr_address = buffer;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
const ISC_STATUS status =
|
2004-05-12 21:39:17 +02:00
|
|
|
this->send_response(sendL,
|
2001-05-23 15:26:42 +02:00
|
|
|
state,
|
2004-05-12 23:47:36 +02:00
|
|
|
prepareL->p_sqlst_buffer_length,
|
2001-05-23 15:26:42 +02:00
|
|
|
status_vector);
|
|
|
|
|
|
|
|
if (buffer != local_buffer) {
|
|
|
|
ALLR_free(buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
bool process_packet(rem_port* port,
|
2004-05-12 21:39:17 +02:00
|
|
|
PACKET* sendL,
|
2003-09-11 20:59:34 +02:00
|
|
|
PACKET* receive,
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port** result)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p r o c e s s _ p a c k e t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Given an packet received from client, process it a packet ready to be
|
|
|
|
* sent.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
TEXT msg[128];
|
2004-08-30 20:11:08 +02:00
|
|
|
trdb thd_context(port->port_status_vector);
|
|
|
|
// BRS: This is the same as REM_set_thread_data
|
2004-05-21 17:22:19 +02:00
|
|
|
trdb* tdrdb = &thd_context;
|
2004-06-21 10:49:46 +02:00
|
|
|
tdrdb->putSpecific();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
try {
|
2004-02-24 06:34:44 +01:00
|
|
|
P_OP op = receive->p_operation;
|
|
|
|
switch (op)
|
2002-03-31 01:40:08 +01:00
|
|
|
{
|
|
|
|
case op_connect:
|
2004-05-12 21:39:17 +02:00
|
|
|
if (!accept_connection(port, &receive->p_cnct, sendL)) {
|
2004-02-24 06:34:44 +01:00
|
|
|
rem_str* string = port->port_user_name;
|
|
|
|
if (string) {
|
2002-03-31 01:40:08 +01:00
|
|
|
sprintf(msg,
|
|
|
|
"SERVER/process_packet: connection rejected for %*.*s",
|
|
|
|
string->str_length, string->str_length,
|
|
|
|
string->str_data);
|
|
|
|
gds__log(msg, 0);
|
|
|
|
}
|
2002-10-31 12:58:41 +01:00
|
|
|
if (port->port_server->srvr_flags & SRVR_multi_client) {
|
2002-03-31 01:40:08 +01:00
|
|
|
port->port_state = state_broken;
|
2002-10-31 12:58:41 +01:00
|
|
|
}
|
2002-03-31 01:40:08 +01:00
|
|
|
else {
|
|
|
|
gds__log
|
|
|
|
("SERVER/process_packet: connect reject, server exiting",
|
|
|
|
0);
|
2004-08-30 20:11:08 +02:00
|
|
|
ThreadData::restoreSpecific();
|
2003-09-11 20:59:34 +02:00
|
|
|
return false;
|
2002-03-31 01:40:08 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_compile:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->compile(&receive->p_cmpl, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_attach:
|
|
|
|
case op_create:
|
2004-05-12 21:39:17 +02:00
|
|
|
attach_database(port, op, &receive->p_atch, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_service_attach:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->service_attach(&receive->p_atch, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_service_start:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->service_start(&receive->p_info, sendL);
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
2002-03-31 01:40:08 +01:00
|
|
|
|
|
|
|
case op_disconnect:
|
|
|
|
case op_exit:
|
2004-02-24 06:34:44 +01:00
|
|
|
{
|
|
|
|
SRVR server = port->port_server;
|
|
|
|
if (!server)
|
|
|
|
break;
|
2002-03-31 01:40:08 +01:00
|
|
|
if ((server->srvr_flags & SRVR_multi_client) &&
|
2004-02-24 06:34:44 +01:00
|
|
|
port != server->srvr_parent_port)
|
|
|
|
{
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
port->disconnect(sendL, receive);
|
2004-02-24 06:34:44 +01:00
|
|
|
port = NULL;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if ((server->srvr_flags & SRVR_multi_client) &&
|
|
|
|
port == server->srvr_parent_port)
|
|
|
|
{
|
2003-05-05 13:44:02 +02:00
|
|
|
gds__log("SERVER/process_packet: Multi-client server shutdown", 0);
|
2004-02-24 06:34:44 +01:00
|
|
|
}
|
2004-05-12 21:39:17 +02:00
|
|
|
port->disconnect(sendL, receive);
|
2004-08-30 20:11:08 +02:00
|
|
|
ThreadData::restoreSpecific();
|
2004-02-24 06:34:44 +01:00
|
|
|
return false;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_receive:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->receive_msg(&receive->p_data, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_send:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->send_msg(&receive->p_data, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_start:
|
|
|
|
case op_start_and_receive:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->start(op, &receive->p_data, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_start_and_send:
|
|
|
|
case op_start_send_and_receive:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->start_and_send(op, &receive->p_data, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_transact:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->transact_request(&receive->p_trrq, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_reconnect:
|
|
|
|
case op_transaction:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->start_transaction(op, &receive->p_sttr, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_prepare:
|
|
|
|
case op_rollback:
|
|
|
|
case op_rollback_retaining:
|
|
|
|
case op_commit:
|
|
|
|
case op_commit_retaining:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->end_transaction(op, &receive->p_rlse, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_detach:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->end_database(&receive->p_rlse, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_service_detach:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->service_end(&receive->p_rlse, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_drop_database:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->drop_database(&receive->p_rlse, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_create_blob:
|
|
|
|
case op_open_blob:
|
|
|
|
case op_create_blob2:
|
|
|
|
case op_open_blob2:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->open_blob(op, &receive->p_blob, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_batch_segments:
|
|
|
|
case op_put_segment:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->put_segment(op, &receive->p_sgmt, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_get_segment:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->get_segment(&receive->p_sgmt, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_seek_blob:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->seek_blob(&receive->p_seek, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_cancel_blob:
|
|
|
|
case op_close_blob:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->end_blob(op, &receive->p_rlse, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_prepare2:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->prepare(&receive->p_prep, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_release:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->end_request(&receive->p_rlse, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_info_blob:
|
|
|
|
case op_info_database:
|
|
|
|
case op_info_request:
|
|
|
|
case op_info_transaction:
|
|
|
|
case op_service_info:
|
|
|
|
case op_info_sql:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->info(op, &receive->p_info, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_que_events:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->que_events(&receive->p_event, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_cancel_events:
|
2004-05-12 21:39:17 +02:00
|
|
|
cancel_events(port, &receive->p_event, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_connect_request:
|
2004-05-12 21:39:17 +02:00
|
|
|
aux_request(port, &receive->p_req, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-03-14 06:51:54 +01:00
|
|
|
#ifdef NOT_USED_OR_REPLACED
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_aux_connect:
|
2004-05-12 21:39:17 +02:00
|
|
|
aux_connect(port, &receive->p_req, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2004-03-14 06:51:54 +01:00
|
|
|
#endif
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_ddl:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->ddl(&receive->p_ddl, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_get_slice:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->get_slice(&receive->p_slc, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_put_slice:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->put_slice(&receive->p_slc, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_allocate_statement:
|
2004-05-12 21:39:17 +02:00
|
|
|
allocate_statement(port, &receive->p_rlse, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_execute:
|
|
|
|
case op_execute2:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->execute_statement(op, &receive->p_sqldata, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_exec_immediate:
|
|
|
|
case op_exec_immediate2:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->execute_immediate(op, &receive->p_sqlst, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_fetch:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->fetch(&receive->p_sqldata, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_free_statement:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->end_statement(&receive->p_sqlfree, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_insert:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->insert(&receive->p_sqldata, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_prepare_statement:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->prepare_statement(&receive->p_sqlst, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_set_cursor:
|
2004-05-12 21:39:17 +02:00
|
|
|
port->set_cursor(&receive->p_sqlcur, sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
case op_dummy:
|
2004-05-12 21:39:17 +02:00
|
|
|
sendL->p_operation = op_dummy;
|
|
|
|
port->send(sendL);
|
2002-03-31 01:40:08 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
default:
|
|
|
|
sprintf(msg, "SERVER/process_packet: don't understand packet type %d",
|
|
|
|
receive->p_operation);
|
|
|
|
gds__log(msg, 0);
|
|
|
|
port->port_state = state_broken;
|
|
|
|
break;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
if (port && port->port_state == state_broken) {
|
|
|
|
if (!port->port_parent) {
|
|
|
|
gds__log("SERVER/process_packet: broken port, server exiting", 0);
|
2005-11-24 13:15:41 +01:00
|
|
|
port->disconnect(sendL, receive);
|
2004-08-30 20:11:08 +02:00
|
|
|
ThreadData::restoreSpecific();
|
2003-09-11 20:59:34 +02:00
|
|
|
return false;
|
2002-03-31 01:40:08 +01:00
|
|
|
}
|
2004-05-12 21:39:17 +02:00
|
|
|
port->disconnect(sendL, receive);
|
2002-03-31 01:40:08 +01:00
|
|
|
port = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
if (result)
|
|
|
|
*result = port;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
} // try
|
2004-11-29 05:39:20 +01:00
|
|
|
catch (const std::exception& ex) {
|
|
|
|
// NS: trdb_status_vector is usually NULL at this point.
|
|
|
|
// This needs to be fixed, probably via eliminating trdb_status_vector
|
|
|
|
// completely, but this work is not for now.
|
|
|
|
ISC_STATUS* status_ptr;
|
|
|
|
|
|
|
|
ISC_STATUS_ARRAY local_status;
|
|
|
|
memset(local_status, 0, sizeof(local_status));
|
|
|
|
|
|
|
|
if (tdrdb->trdb_status_vector) {
|
|
|
|
status_ptr = tdrdb->trdb_status_vector;
|
|
|
|
} else {
|
|
|
|
status_ptr = local_status;
|
|
|
|
}
|
|
|
|
|
|
|
|
Firebird::stuff_exception(status_ptr, ex);
|
2001-12-24 03:51:06 +01:00
|
|
|
|
2004-11-29 05:39:20 +01:00
|
|
|
gds__log_status("unknown, SERVER/process_packet", status_ptr);
|
2001-12-24 03:51:06 +01:00
|
|
|
|
|
|
|
/* It would be nice to log an error to the user, instead of just terminating them! */
|
2004-11-29 05:39:20 +01:00
|
|
|
port->send_response(sendL, 0, 0, status_ptr);
|
2004-05-12 21:39:17 +02:00
|
|
|
port->disconnect(sendL, receive); /* Well, how about this... */
|
2001-12-24 03:51:06 +01:00
|
|
|
|
2004-08-30 20:11:08 +02:00
|
|
|
ThreadData::restoreSpecific();
|
2003-09-11 20:59:34 +02:00
|
|
|
return false;
|
2001-12-24 03:51:06 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-11-24 13:15:41 +01:00
|
|
|
ThreadData::restoreSpecific();
|
2003-09-11 20:59:34 +02:00
|
|
|
return true;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::put_segment(P_OP op, P_SGMT * segment, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p u t _ s e g m e n t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Write a single blob segment.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
RBL blob;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
CHECK_HANDLE_MEMBER(blob,
|
|
|
|
RBL,
|
|
|
|
type_rbl,
|
|
|
|
segment->p_sgmt_blob,
|
|
|
|
isc_bad_segstr_handle);
|
|
|
|
|
2003-10-29 11:53:47 +01:00
|
|
|
const UCHAR* p = segment->p_sgmt_segment.cstr_address;
|
2004-02-24 06:34:44 +01:00
|
|
|
USHORT length = segment->p_sgmt_segment.cstr_length;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Do the signal segment version. If it failed, just pass on the
|
|
|
|
bad news. */
|
|
|
|
|
|
|
|
if (op == op_put_segment) {
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_put_segment(status_vector, &blob->rbl_handle, length,
|
2003-10-29 11:53:47 +01:00
|
|
|
reinterpret_cast<const char*>(p));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* We've got a batch of segments. This is only slightly more complicated */
|
|
|
|
|
2003-10-29 11:53:47 +01:00
|
|
|
const UCHAR* const end = p + length;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
while (p < end) {
|
|
|
|
length = *p++;
|
|
|
|
length += *p++ << 8;
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_put_segment(status_vector, &blob->rbl_handle, length,
|
2003-10-29 11:53:47 +01:00
|
|
|
reinterpret_cast<const char*>(p));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
if (status_vector[1])
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
p += length;
|
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::put_slice(P_SLC * stuff, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p u t _ s l i c e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Put an array slice.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
RTR transaction;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
CHECK_HANDLE_MEMBER(transaction,
|
|
|
|
RTR,
|
|
|
|
type_rtr,
|
|
|
|
stuff->p_slc_transaction,
|
|
|
|
isc_bad_trans_handle);
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
RDB rdb = this->port_context;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2004-05-12 21:39:17 +02:00
|
|
|
sendL->p_resp.p_resp_blob_id = stuff->p_slc_id;
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_put_slice(status_vector, &rdb->rdb_handle, &transaction->rtr_handle,
|
2004-05-12 21:39:17 +02:00
|
|
|
(ISC_QUAD*) &sendL->p_resp.p_resp_blob_id,
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_slc_sdl.cstr_length,
|
2003-08-30 03:43:08 +02:00
|
|
|
reinterpret_cast<char*>(stuff->p_slc_sdl.cstr_address),
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_slc_parameters.cstr_length,
|
2003-08-30 03:43:08 +02:00
|
|
|
(ISC_LONG *) stuff->p_slc_parameters.cstr_address,
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_slc_slice.lstr_length,
|
2003-08-30 03:43:08 +02:00
|
|
|
stuff->p_slc_slice.lstr_address);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::que_events(P_EVENT * stuff, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* q u e _ e v e n t s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-02-24 06:34:44 +01:00
|
|
|
RDB rdb = this->port_context;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Find unused event block or, if necessary, a new one */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
RVNT event;
|
2001-05-23 15:26:42 +02:00
|
|
|
for (event = rdb->rdb_events; event; event = event->rvnt_next) {
|
|
|
|
if (!event->rvnt_id) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!event)
|
|
|
|
{
|
2004-05-17 12:22:34 +02:00
|
|
|
event = (RVNT) ALLR_block(type_rvnt, 0);
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("que_events(server) allocate event %x\n", event);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
event->rvnt_next = rdb->rdb_events;
|
|
|
|
rdb->rdb_events = event;
|
|
|
|
}
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
event->rvnt_ast = stuff->p_event_ast;
|
2004-03-14 06:51:54 +01:00
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
// CVC: Going from SLONG to void*, problems when sizeof(void*) > 4
|
2004-03-14 06:51:54 +01:00
|
|
|
// Nickolay Samofatov. No problem actually. Event argument returned by server
|
|
|
|
// on event firing is not used by the client. Client looks up argument via
|
|
|
|
// seaching client-side event table and remote event ID passed by server on
|
|
|
|
// event registration. As a result we may use this argument value for
|
|
|
|
// server-side debugging only.
|
|
|
|
event->rvnt_arg = (void*)(IPTR) stuff->p_event_arg;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
event->rvnt_rid = stuff->p_event_rid;
|
|
|
|
event->rvnt_rdb = rdb;
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2004-02-24 06:34:44 +01:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_que_events(status_vector, &rdb->rdb_handle, &event->rvnt_id,
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_event_items.cstr_length,
|
2003-11-28 07:48:34 +01:00
|
|
|
reinterpret_cast<const char*>(stuff->p_event_items.cstr_address),
|
2003-12-22 11:00:59 +01:00
|
|
|
server_ast,
|
2003-08-30 03:43:08 +02:00
|
|
|
event);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
const SLONG id = event->rvnt_id;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (status_vector[1]) {
|
|
|
|
event->rvnt_id = 0;
|
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, (OBJCT) id, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
ISC_STATUS rem_port::receive_after_start( P_DATA* data,
|
2004-05-12 21:39:17 +02:00
|
|
|
PACKET* sendL,
|
2003-04-10 12:31:28 +02:00
|
|
|
ISC_STATUS* status_vector)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e c e i v e _ a f t e r _ s t a r t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Receive a message.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-05-12 23:47:36 +02:00
|
|
|
rrq* requestL;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-12 23:47:36 +02:00
|
|
|
CHECK_HANDLE_MEMBER(requestL,
|
2004-01-28 08:50:41 +01:00
|
|
|
rrq*,
|
2001-05-23 15:26:42 +02:00
|
|
|
type_rrq,
|
|
|
|
data->p_data_request,
|
|
|
|
isc_bad_req_handle);
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
const USHORT level = data->p_data_incarnation;
|
2004-05-12 23:47:36 +02:00
|
|
|
requestL = REMOTE_find_request(requestL, level);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Figure out the number of the message that we're stalled on. */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
USHORT msg_number;
|
2004-05-12 23:47:36 +02:00
|
|
|
if (!get_next_msg_no(requestL, level, &msg_number)) {
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
sendL->p_operation = op_response_piggyback;
|
|
|
|
P_RESP* response = &sendL->p_resp;
|
2001-05-23 15:26:42 +02:00
|
|
|
response->p_resp_object = msg_number;
|
|
|
|
response->p_resp_status_vector = status_vector;
|
|
|
|
response->p_resp_data.cstr_length = 0;
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
this->send_partial(sendL);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Fill in packet to fool receive into thinking that it has been
|
|
|
|
called directly by the client. */
|
|
|
|
|
2004-05-12 23:47:36 +02:00
|
|
|
const rrq::rrq_repeat* tail = requestL->rrq_rpt + msg_number;
|
2004-02-24 06:34:44 +01:00
|
|
|
const rem_fmt* format = tail->rrq_format;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
data->p_data_message_number = msg_number;
|
|
|
|
if (this->port_flags & PORT_rpc)
|
|
|
|
{
|
|
|
|
data->p_data_messages = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
data->p_data_messages =
|
|
|
|
(USHORT) REMOTE_compute_batch_size(this,
|
|
|
|
(USHORT)
|
|
|
|
xdr_protocol_overhead
|
|
|
|
(op_response_piggyback),
|
|
|
|
op_send,
|
|
|
|
format);
|
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->receive_msg(data, sendL);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::receive_msg(P_DATA * data, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e c e i v e _ m s g
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Receive a message.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
/* Find the database, request, message number, and the number of
|
2001-05-23 15:26:42 +02:00
|
|
|
messages the client is willing to cope with. Then locate the
|
|
|
|
message control block for the request and message type. */
|
2004-02-24 06:34:44 +01:00
|
|
|
|
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2004-05-12 23:47:36 +02:00
|
|
|
rrq* requestL;
|
|
|
|
CHECK_HANDLE_MEMBER(requestL,
|
2004-01-28 08:50:41 +01:00
|
|
|
rrq*,
|
2001-05-23 15:26:42 +02:00
|
|
|
type_rrq,
|
|
|
|
data->p_data_request,
|
|
|
|
isc_bad_req_handle);
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
const USHORT level = data->p_data_incarnation;
|
2004-05-12 23:47:36 +02:00
|
|
|
requestL = REMOTE_find_request(requestL, level);
|
2004-02-24 06:34:44 +01:00
|
|
|
const USHORT msg_number = data->p_data_message_number;
|
|
|
|
USHORT count, count2;
|
2001-05-23 15:26:42 +02:00
|
|
|
count2 = count =
|
|
|
|
(this->port_flags & PORT_rpc) ? 1 : data->p_data_messages;
|
2004-05-12 23:47:36 +02:00
|
|
|
rrq::rrq_repeat* tail = requestL->rrq_rpt + msg_number;
|
2004-02-24 06:34:44 +01:00
|
|
|
const rem_fmt* format = tail->rrq_format;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Get ready to ship the data out */
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
sendL->p_operation = op_send;
|
|
|
|
sendL->p_data.p_data_request = data->p_data_request;
|
|
|
|
sendL->p_data.p_data_message_number = msg_number;
|
|
|
|
sendL->p_data.p_data_incarnation = level;
|
|
|
|
sendL->p_data.p_data_messages = 1;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
/* check the direction and offset for possible redirection; if the user
|
|
|
|
scrolls in a direction other than we were going or an offset other
|
|
|
|
than 1, then we need to resynchronize:
|
|
|
|
if the direction is the same as the lookahead cache, scroll forward
|
|
|
|
through the cache to see if we find the record; otherwise scroll the
|
|
|
|
next layer down backward by an amount equal to the number of records
|
|
|
|
in the cache, plus the amount asked for by the next level up */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
USHORT direction;
|
|
|
|
ULONG offset;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (this->port_protocol < PROTOCOL_SCROLLABLE_CURSORS)
|
|
|
|
{
|
|
|
|
direction = blr_forward;
|
|
|
|
offset = 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
direction = data->p_data_direction;
|
|
|
|
offset = data->p_data_offset;
|
|
|
|
tail->rrq_xdr = scroll_cache(tail, &direction, &offset);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Check to see if any messages are already sitting around */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
REM_MSG message = 0;
|
|
|
|
|
2003-09-11 20:59:34 +02:00
|
|
|
while (true) {
|
2001-05-23 15:26:42 +02:00
|
|
|
message = tail->rrq_xdr;
|
|
|
|
|
|
|
|
/* If we don't have a message cached, get one from the next layer down. */
|
|
|
|
|
|
|
|
if (!message->msg_address) {
|
|
|
|
/* If we have an error queued for delivery, send it now */
|
|
|
|
|
2004-05-12 23:47:36 +02:00
|
|
|
if (requestL->rrq_status_vector[1]) {
|
2004-02-24 06:34:44 +01:00
|
|
|
const ISC_STATUS res =
|
2004-05-12 21:39:17 +02:00
|
|
|
this->send_response(sendL, 0, 0,
|
2004-05-12 23:47:36 +02:00
|
|
|
requestL->rrq_status_vector);
|
|
|
|
memset(requestL->rrq_status_vector, 0,
|
|
|
|
sizeof(requestL->rrq_status_vector));
|
2001-05-23 15:26:42 +02:00
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef SCROLLABLE_CURSORS
|
2004-05-12 23:47:36 +02:00
|
|
|
isc_receive2(status_vector, &requestL->rrq_handle, msg_number,
|
2003-08-30 03:43:08 +02:00
|
|
|
format->fmt_length, message->msg_buffer, level,
|
|
|
|
direction, offset);
|
2001-05-23 15:26:42 +02:00
|
|
|
#else
|
2004-05-12 23:47:36 +02:00
|
|
|
isc_receive(status_vector, &requestL->rrq_handle, msg_number,
|
2003-08-30 03:43:08 +02:00
|
|
|
format->fmt_length, message->msg_buffer, level);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
if (status_vector[1])
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
/* set the appropriate flags according to the way we just scrolled
|
|
|
|
the next layer down, and calculate the offset from the beginning
|
|
|
|
of the result set */
|
|
|
|
|
|
|
|
switch (direction) {
|
|
|
|
case blr_forward:
|
|
|
|
tail->rrq_flags &= ~RRQ_backward;
|
|
|
|
tail->rrq_absolute +=
|
2004-12-16 04:03:13 +01:00
|
|
|
(tail->rrq_flags & RRQ_absolute_backward) ?
|
|
|
|
-offset : offset;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_backward:
|
|
|
|
tail->rrq_flags |= RRQ_backward;
|
|
|
|
tail->rrq_absolute +=
|
2004-12-16 04:03:13 +01:00
|
|
|
(tail->rrq_flags & RRQ_absolute_backward) ?
|
|
|
|
offset : -offset;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_bof_forward:
|
|
|
|
tail->rrq_flags &= ~RRQ_backward;
|
|
|
|
tail->rrq_flags &= ~RRQ_absolute_backward;
|
|
|
|
tail->rrq_absolute = offset;
|
|
|
|
direction = blr_forward;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_eof_backward:
|
|
|
|
tail->rrq_flags |= RRQ_backward;
|
|
|
|
tail->rrq_flags |= RRQ_absolute_backward;
|
|
|
|
tail->rrq_absolute = offset;
|
|
|
|
direction = blr_backward;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
message->msg_absolute = tail->rrq_absolute;
|
|
|
|
|
|
|
|
/* if we have already scrolled to the location indicated,
|
|
|
|
then we just want to continue one by one in that direction */
|
|
|
|
|
|
|
|
offset = 1;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
message->msg_address = message->msg_buffer;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If there aren't any buffers remaining, break out of loop */
|
|
|
|
|
|
|
|
if (--count <= 0)
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* There's a buffer waiting -- see if the request is ready to send */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
REM_MSG next = message->msg_next;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if ((next == message || !next->msg_address) &&
|
2004-05-12 23:47:36 +02:00
|
|
|
!check_request(requestL, data->p_data_incarnation, msg_number))
|
2004-02-24 06:34:44 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
/* We've reached the end of the RSE - don't prefetch and flush
|
|
|
|
everything we've buffered so far */
|
|
|
|
|
|
|
|
count2 = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
if (!this->send_partial(sendL))
|
2001-05-23 15:26:42 +02:00
|
|
|
return FALSE;
|
|
|
|
message->msg_address = NULL;
|
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
sendL->p_data.p_data_messages = 0;
|
|
|
|
this->send(sendL);
|
2001-05-23 15:26:42 +02:00
|
|
|
message->msg_address = NULL;
|
|
|
|
|
|
|
|
/* Bump up the message pointer to resync with rrq_xdr (rrq_xdr
|
|
|
|
was incremented by xdr_request in the SEND call). */
|
|
|
|
|
|
|
|
tail->rrq_message = message->msg_next;
|
|
|
|
|
|
|
|
/* Since we have a little time on our hands while this packet is sent
|
|
|
|
and processed, get the next batch of records. Start by finding the
|
|
|
|
next free buffer. */
|
|
|
|
|
|
|
|
message = tail->rrq_xdr;
|
2004-02-24 06:34:44 +01:00
|
|
|
REM_MSG prior = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
while (message->msg_address && message->msg_next != tail->rrq_xdr)
|
|
|
|
message = message->msg_next;
|
|
|
|
|
|
|
|
for (;
|
|
|
|
count2
|
2004-05-12 23:47:36 +02:00
|
|
|
&& check_request(requestL, data->p_data_incarnation, msg_number);
|
2004-02-24 06:34:44 +01:00
|
|
|
--count2)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if (message->msg_address) {
|
|
|
|
if (!prior)
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
prior = message->msg_prior;
|
|
|
|
#else
|
|
|
|
for (prior = tail->rrq_xdr; prior->msg_next != message;
|
|
|
|
prior = prior->msg_next);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* allocate a new message block and put it in the cache */
|
|
|
|
|
2004-05-17 12:22:34 +02:00
|
|
|
message = (REM_MSG) ALLR_block(type_msg, format->fmt_length);
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("receive_msg(server) allocate message %x\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
message);
|
|
|
|
#endif
|
|
|
|
message->msg_number = prior->msg_number;
|
|
|
|
message->msg_next = prior->msg_next;
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
message->msg_prior = prior;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
prior->msg_next = message;
|
|
|
|
prior = message;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* fetch the next record into cache; even for scrollable cursors, we are
|
|
|
|
just doing a simple lookahead continuing on in the last direction specified,
|
|
|
|
so there is no reason to do an isc_receive2() */
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2004-05-12 23:47:36 +02:00
|
|
|
isc_receive(status_vector, &requestL->rrq_handle, msg_number,
|
2001-05-23 15:26:42 +02:00
|
|
|
format->fmt_length,
|
|
|
|
message->msg_buffer, data->p_data_incarnation);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Did we have an error? If so, save it for later delivery */
|
|
|
|
|
|
|
|
if (status_vector[1]) {
|
|
|
|
/* If already have an error queued, don't overwrite it */
|
|
|
|
|
2004-05-12 23:47:36 +02:00
|
|
|
if (!requestL->rrq_status_vector[1]) {
|
|
|
|
memcpy(requestL->rrq_status_vector, status_vector,
|
|
|
|
sizeof(requestL->rrq_status_vector));
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
/* if we have already scrolled to the location indicated,
|
|
|
|
then we just want to continue on in that direction */
|
|
|
|
|
|
|
|
switch (direction)
|
|
|
|
{
|
|
|
|
case blr_forward:
|
|
|
|
tail->rrq_absolute +=
|
|
|
|
(tail->rrq_flags & RRQ_absolute_backward) ? -offset : offset;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case blr_backward:
|
|
|
|
tail->rrq_absolute +=
|
|
|
|
(tail->rrq_flags & RRQ_absolute_backward) ? offset : -offset;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
message->msg_absolute = tail->rrq_absolute;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
message->msg_address = message->msg_buffer;
|
|
|
|
message = message->msg_next;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void release_blob(RBL blob)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e l e a s e _ b l o b
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Release a blob block and friends.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
RDB rdb = blob->rbl_rdb;
|
|
|
|
RTR transaction = blob->rbl_rtr;
|
|
|
|
|
|
|
|
rdb->rdb_port->port_objects[blob->rbl_id] = NULL;
|
|
|
|
|
|
|
|
for (RBL* p = &transaction->rtr_blobs; *p; p = &(*p)->rbl_next) {
|
|
|
|
if (*p == blob) {
|
|
|
|
*p = blob->rbl_next;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("release_blob(server) free blob %x\n", blob);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
if (blob->rbl_buffer != blob->rbl_data) {
|
|
|
|
ALLR_free(blob->rbl_buffer);
|
|
|
|
}
|
|
|
|
ALLR_release(blob);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void release_event( RVNT event)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e l e a s e _ e v e n t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Release an event block.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-02-24 06:34:44 +01:00
|
|
|
RDB rdb = event->rvnt_rdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
for (RVNT* p = &rdb->rdb_events; *p; p = &(*p)->rvnt_next)
|
2001-05-23 15:26:42 +02:00
|
|
|
if (*p == event) {
|
|
|
|
*p = event->rvnt_next;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
ALLR_release(event);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static void release_request( rrq* request)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e l e a s e _ r e q u e s t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Release a request block.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-02-24 06:34:44 +01:00
|
|
|
RDB rdb = request->rrq_rdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
rdb->rdb_port->port_objects[request->rrq_id] = NULL;
|
|
|
|
REMOTE_release_request(request);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void release_statement( RSR * statement)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e l e a s e _ s t a t e m e n t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Release a GDML or SQL statement?
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
if ((*statement)->rsr_select_format)
|
|
|
|
ALLR_release((*statement)->rsr_select_format);
|
|
|
|
if ((*statement)->rsr_bind_format)
|
|
|
|
ALLR_release((*statement)->rsr_bind_format);
|
|
|
|
REMOTE_release_messages((*statement)->rsr_message);
|
|
|
|
ALLR_release((*statement));
|
|
|
|
*statement = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void release_sql_request( RSR statement)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e l e a s e _ s q l _ r e q u e s t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Release an SQL request block.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-02-24 06:34:44 +01:00
|
|
|
RDB rdb = statement->rsr_rdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
rdb->rdb_port->port_objects[statement->rsr_id] = NULL;
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
for (RSR* p = &rdb->rdb_sql_requests; *p; p = &(*p)->rsr_next)
|
2001-05-23 15:26:42 +02:00
|
|
|
if (*p == statement) {
|
|
|
|
*p = statement->rsr_next;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
release_statement(&statement);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void release_transaction( RTR transaction)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e l e a s e _ t r a n s a c t i o n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Release a transaction block and friends.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-02-24 06:34:44 +01:00
|
|
|
RDB rdb = transaction->rtr_rdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
rdb->rdb_port->port_objects[transaction->rtr_id] = NULL;
|
|
|
|
|
|
|
|
while (transaction->rtr_blobs)
|
|
|
|
release_blob(transaction->rtr_blobs);
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
for (RTR* p = &rdb->rdb_transactions; *p; p = &(*p)->rtr_next)
|
2001-05-23 15:26:42 +02:00
|
|
|
if (*p == transaction) {
|
|
|
|
*p = transaction->rtr_next;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("release_transact(server) free transaction %x\n", transaction);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
ALLR_release(transaction);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef SCROLLABLE_CURSORS
|
2002-11-19 13:40:54 +01:00
|
|
|
static REM_MSG scroll_cache(
|
2001-05-23 15:26:42 +02:00
|
|
|
rrq::rrq_repeat* tail,
|
|
|
|
USHORT * direction, ULONG * offset)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s c r o l l _ c a c h e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
* Try to fetch the requested record from cache, if possible. This algorithm depends
|
|
|
|
* on all scrollable cursors being INSENSITIVE to database changes, so that absolute
|
|
|
|
* record numbers within the result set will remain constant.
|
|
|
|
*
|
|
|
|
* 1. BOF Forward or EOF Backward: Retain the record number of the offset from the
|
|
|
|
* beginning or end of the result set. If we can figure out the relative offset
|
|
|
|
* from the absolute, then scroll to it. If it's in cache, great, otherwise dump
|
|
|
|
* the cache and have the server scroll the correct number of records.
|
|
|
|
*
|
|
|
|
* 2. Forward or Backward: Try to scroll the desired offset in cache. If we
|
|
|
|
* scroll off the end of the cache, dump the cache and ask the server for a
|
|
|
|
* packetful of records.
|
|
|
|
*
|
|
|
|
* This routine differs from the corresponding routine on the client in that
|
|
|
|
* we are using only a lookahead cache. There is no point in caching records backward,
|
|
|
|
* in that the client already has them and would not request them from us.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
/* if we are to continue in the current direction, set direction to
|
|
|
|
the last direction scrolled; then depending on the direction asked
|
|
|
|
for, save the last direction asked for by the next layer above */
|
|
|
|
|
|
|
|
if (*direction == blr_continue) {
|
|
|
|
if (tail->rrq_flags & RRQ_last_backward)
|
|
|
|
*direction = blr_backward;
|
|
|
|
else
|
|
|
|
*direction = blr_forward;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*direction == blr_forward || *direction == blr_bof_forward)
|
|
|
|
tail->rrq_flags &= ~RRQ_last_backward;
|
|
|
|
else
|
|
|
|
tail->rrq_flags |= RRQ_last_backward;
|
|
|
|
|
|
|
|
/* set to the first message; if it has no record, this means the cache is
|
|
|
|
empty and there is no point in trying to find the record here */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
REM_MSG message = tail->rrq_xdr;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!message->msg_address)
|
|
|
|
return message;
|
|
|
|
|
|
|
|
/* if we are scrolling from BOF and the cache was started from EOF (or vice
|
|
|
|
versa), the cache is unusable. */
|
|
|
|
|
|
|
|
if (
|
|
|
|
(*direction == blr_bof_forward
|
|
|
|
&& (tail->rrq_flags & RRQ_absolute_backward))
|
|
|
|
|| (*direction == blr_eof_backward
|
2004-12-16 04:03:13 +01:00
|
|
|
&& !(tail->rrq_flags & RRQ_absolute_backward)))
|
2004-02-24 06:34:44 +01:00
|
|
|
{
|
|
|
|
return dump_cache(tail);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* if we are going to an absolute position, see if we can find that position
|
|
|
|
in cache, otherwise change to a relative seek from our former position */
|
|
|
|
|
|
|
|
if (*direction == blr_bof_forward || *direction == blr_eof_backward) {
|
|
|
|
/* If offset is before our current position, just dump the cache because
|
|
|
|
the server cache is purely a lookahead cache--we don't bother to cache
|
|
|
|
back records because the client will cache those records, making it
|
|
|
|
unlikely the client would be asking us for a record which is in our cache. */
|
|
|
|
|
|
|
|
if (*offset < message->msg_absolute)
|
|
|
|
return dump_cache(tail);
|
|
|
|
|
|
|
|
/* convert the absolute to relative, and prepare to scroll forward to look for the record */
|
|
|
|
|
|
|
|
*offset -= message->msg_absolute;
|
|
|
|
if (*direction == blr_bof_forward)
|
|
|
|
*direction = blr_forward;
|
|
|
|
else
|
|
|
|
*direction = blr_backward;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((*direction == blr_forward && (tail->rrq_flags & RRQ_backward)) ||
|
2004-02-24 06:34:44 +01:00
|
|
|
(*direction == blr_backward && !(tail->rrq_flags & RRQ_backward)))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
/* lookahead cache was in opposite direction from the scroll direction,
|
|
|
|
so increase the scroll amount by the amount we looked ahead, then
|
|
|
|
dump the cache */
|
|
|
|
|
|
|
|
for (message = tail->rrq_xdr; message->msg_address;) {
|
|
|
|
(*offset)++;
|
|
|
|
message = message->msg_next;
|
|
|
|
if (message == tail->rrq_message)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return dump_cache(tail);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* lookahead cache is in same direction we want to scroll, so scroll
|
|
|
|
forward through the cache, decrementing the offset */
|
|
|
|
|
|
|
|
for (message = tail->rrq_xdr; message->msg_address;) {
|
|
|
|
if (*offset == 1)
|
|
|
|
break;
|
|
|
|
(*offset)--;
|
|
|
|
message = message->msg_next;
|
|
|
|
if (message == tail->rrq_message)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return message;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::seek_blob(P_SEEK * seek, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e e k _ b l o b
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Execute a blob seek operation.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
RBL blob;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
CHECK_HANDLE_MEMBER(blob,
|
|
|
|
RBL,
|
|
|
|
type_rbl,
|
|
|
|
seek->p_seek_blob,
|
|
|
|
isc_bad_segstr_handle);
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
const SSHORT mode = seek->p_seek_mode;
|
|
|
|
const SLONG offset = seek->p_seek_offset;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2004-02-24 06:34:44 +01:00
|
|
|
SLONG result;
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_seek_blob(status_vector, &blob->rbl_handle, mode, offset, &result);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
sendL->p_resp.p_resp_blob_id.bid_number = result;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::send_msg(P_DATA * data, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e n d _ m s g
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Handle a isc_send operation.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-12 23:47:36 +02:00
|
|
|
rrq* requestL;
|
|
|
|
CHECK_HANDLE_MEMBER(requestL,
|
2004-01-28 08:50:41 +01:00
|
|
|
rrq*,
|
2001-05-23 15:26:42 +02:00
|
|
|
type_rrq,
|
|
|
|
data->p_data_request,
|
|
|
|
isc_bad_req_handle);
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
const USHORT number = data->p_data_message_number;
|
2004-05-12 23:47:36 +02:00
|
|
|
requestL = REMOTE_find_request(requestL, data->p_data_incarnation);
|
|
|
|
REM_MSG message = requestL->rrq_rpt[number].rrq_message;
|
|
|
|
const rem_fmt* format = requestL->rrq_rpt[number].rrq_format;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2004-05-12 23:47:36 +02:00
|
|
|
isc_send(status_vector, &requestL->rrq_handle, number, format->fmt_length,
|
2003-08-30 03:43:08 +02:00
|
|
|
message->msg_address, data->p_data_incarnation);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
message->msg_address = NULL;
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::send_response( PACKET* sendL,
|
2001-05-23 15:26:42 +02:00
|
|
|
OBJCT object,
|
|
|
|
USHORT length,
|
2003-09-25 13:49:12 +02:00
|
|
|
const ISC_STATUS* status_vector)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e n d _ r e s p o n s e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Send a response packet.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
/* Start by translating the status vector into "generic" form */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
ISC_STATUS_ARRAY new_vector;
|
2004-01-28 08:50:41 +01:00
|
|
|
ISC_STATUS* v = new_vector;
|
2004-02-24 06:34:44 +01:00
|
|
|
TEXT buffer[1024];
|
2004-09-22 03:59:39 +02:00
|
|
|
const char* const bufferEnd = buffer + sizeof(buffer);
|
2004-01-28 08:50:41 +01:00
|
|
|
TEXT* p = buffer;
|
|
|
|
const ISC_STATUS exit_code = status_vector[1];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
for (bool sw = true; *status_vector && sw;)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
switch ((USHORT) * status_vector)
|
|
|
|
{
|
|
|
|
case isc_arg_warning:
|
2003-11-08 17:40:17 +01:00
|
|
|
case isc_arg_gds:
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/* When talking with older (pre 6.0) clients, do not send
|
|
|
|
* warnings.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (*status_vector == isc_arg_warning &&
|
|
|
|
this->port_protocol < PROTOCOL_VERSION10)
|
|
|
|
{
|
2004-02-24 06:34:44 +01:00
|
|
|
sw = false;
|
2001-05-23 15:26:42 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
*v++ = *status_vector++;
|
|
|
|
|
|
|
|
/* The status codes are converted to their offsets so that they
|
|
|
|
* were compatible with the RDB implementation. This was fine
|
|
|
|
* when status codes were restricted to a single facility. Now
|
|
|
|
* that the facility is part of the status code we need to know
|
|
|
|
* this on the client side, thus when talking with 6.0 and newer
|
|
|
|
* clients, do not decode the status code, just send it to the
|
2004-01-03 11:59:52 +01:00
|
|
|
* client. The same check is made in interface.cpp::check_response
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
|
|
|
|
2004-01-03 11:59:52 +01:00
|
|
|
if (this->port_protocol < PROTOCOL_VERSION10) {
|
|
|
|
USHORT fac = 0, code_class = 0;
|
2004-02-24 06:34:44 +01:00
|
|
|
*v++ = gds__decode(*status_vector++, &fac, &code_class);
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
|
|
|
else {
|
2004-02-24 06:34:44 +01:00
|
|
|
*v++ = *status_vector++;
|
2004-01-03 11:59:52 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
for (;;) {
|
|
|
|
switch (*status_vector) {
|
2003-11-08 17:40:17 +01:00
|
|
|
case isc_arg_string:
|
|
|
|
case isc_arg_number:
|
2001-05-23 15:26:42 +02:00
|
|
|
*v++ = *status_vector++;
|
|
|
|
*v++ = *status_vector++;
|
|
|
|
continue;
|
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
case isc_arg_cstring:
|
2004-02-24 06:34:44 +01:00
|
|
|
{
|
|
|
|
*v++ = isc_arg_string;
|
|
|
|
TEXT** sp = (TEXT**) v;
|
|
|
|
*sp++ = p;
|
|
|
|
v = (ISC_STATUS*) sp;
|
|
|
|
status_vector++;
|
|
|
|
USHORT l = (USHORT) (*status_vector++);
|
|
|
|
const TEXT* q = (TEXT*) * status_vector++;
|
|
|
|
if (l)
|
|
|
|
do {
|
|
|
|
*p++ = *q++;
|
2004-11-18 09:40:13 +01:00
|
|
|
} while (--l && (p < bufferEnd - 1));
|
2004-02-24 06:34:44 +01:00
|
|
|
*p++ = 0;
|
|
|
|
continue;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
case isc_arg_interpreted:
|
2001-05-23 15:26:42 +02:00
|
|
|
*v++ = *status_vector++;
|
|
|
|
*v++ = *status_vector++;
|
|
|
|
continue;
|
|
|
|
}
|
2004-09-22 03:59:39 +02:00
|
|
|
const USHORT l = p < bufferEnd ?
|
2004-11-28 02:03:05 +01:00
|
|
|
(USHORT) fb_interpret(p, bufferEnd - p, &status_vector) : 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (l == 0)
|
|
|
|
break;
|
2004-09-22 03:59:39 +02:00
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
*v++ = isc_arg_interpreted;
|
2004-02-24 06:34:44 +01:00
|
|
|
TEXT** sp = (TEXT**) v;
|
2001-05-23 15:26:42 +02:00
|
|
|
*sp++ = p;
|
2003-04-10 12:31:28 +02:00
|
|
|
v = (ISC_STATUS *) sp;
|
2001-05-23 15:26:42 +02:00
|
|
|
p += l;
|
2004-02-24 06:34:44 +01:00
|
|
|
sw = false;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2003-11-08 17:40:17 +01:00
|
|
|
*v = isc_arg_end;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Format and send response. Note: the blob_id and data address fields
|
|
|
|
of the response packet may contain valid data. Don't trash them. */
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
sendL->p_operation = op_response;
|
2004-02-24 06:34:44 +01:00
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
P_RESP* response = &sendL->p_resp;
|
2001-05-23 15:26:42 +02:00
|
|
|
response->p_resp_object = object;
|
|
|
|
response->p_resp_status_vector = new_vector;
|
|
|
|
response->p_resp_data.cstr_length = length;
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
this->send(sendL);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return exit_code;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-12-22 11:00:59 +01:00
|
|
|
static void server_ast(void* event_void, USHORT length, const UCHAR* items)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e r v e r _ a s t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2003-01-15 13:15:07 +01:00
|
|
|
* Send an asynchrous event packet back to client.
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
2004-01-28 08:50:41 +01:00
|
|
|
RVNT event = static_cast<rvnt*>(event_void);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
event->rvnt_id = 0;
|
2004-01-28 08:50:41 +01:00
|
|
|
RDB rdb = event->rvnt_rdb;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* port = rdb->rdb_port->port_async;
|
|
|
|
if (!port) {
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
PACKET packet;
|
2001-05-23 15:26:42 +02:00
|
|
|
packet.p_operation = op_event;
|
2004-01-28 08:50:41 +01:00
|
|
|
P_EVENT* p_event = &packet.p_event;
|
2001-05-23 15:26:42 +02:00
|
|
|
p_event->p_event_database = rdb->rdb_id;
|
|
|
|
p_event->p_event_items.cstr_length = length;
|
2003-12-22 11:00:59 +01:00
|
|
|
// Probalby should define this item with CSTRING_CONST instead.
|
|
|
|
p_event->p_event_items.cstr_address = const_cast<UCHAR*>(items);
|
2004-01-21 08:18:30 +01:00
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
// Nickolay Samofatov: We keep these values and even pass them to the client
|
2004-01-21 08:18:30 +01:00
|
|
|
// (as 32-bit values) when event is fired, but client ignores them.
|
2003-12-22 11:00:59 +01:00
|
|
|
p_event->p_event_ast = event->rvnt_ast;
|
2004-01-21 08:18:30 +01:00
|
|
|
p_event->p_event_arg = (SLONG)(IPTR) event->rvnt_arg;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
p_event->p_event_rid = event->rvnt_rid;
|
|
|
|
|
|
|
|
port->send(&packet);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::service_attach(P_ATCH* attach, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e r v i c e _ a t t a c h
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Connect to an Interbase service.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-05-12 21:39:17 +02:00
|
|
|
sendL->p_operation = op_accept;
|
2004-05-03 01:06:37 +02:00
|
|
|
FB_API_HANDLE handle = 0;
|
2004-01-28 08:50:41 +01:00
|
|
|
const UCHAR* service_name = attach->p_atch_file.cstr_address;
|
|
|
|
const USHORT service_length = attach->p_atch_file.cstr_length;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-11-27 21:53:09 +01:00
|
|
|
Firebird::ClumpletWriter spb(Firebird::ClumpletReader::SpbAttach, MAX_DPB_SIZE,
|
|
|
|
attach->p_atch_dpb.cstr_address, attach->p_atch_dpb.cstr_length, isc_spb_current_version);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2005-11-27 21:53:09 +01:00
|
|
|
// If we have user identification, append it to database parameter block
|
|
|
|
const rem_str* string = port_user_name;
|
|
|
|
if (string) {
|
|
|
|
spb.setCurOffset(spb.getBufferLength());
|
|
|
|
spb.insertString(isc_spb_sys_user_name,
|
|
|
|
string->str_data, string->str_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2005-11-27 21:53:09 +01:00
|
|
|
// Now, insert remote endpoint data into DPB address stack
|
|
|
|
addRemoteAddress(spb, isc_spb_address_path, this);
|
|
|
|
|
|
|
|
/* See if user has specified parameters relevent to the connection,
|
|
|
|
they will be stuffed in the SPB if so. */
|
|
|
|
REMOTE_get_timeout_params(this, spb.getBuffer(), spb.getBufferLength());
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2004-02-24 06:34:44 +01:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
isc_service_attach(status_vector,
|
|
|
|
service_length,
|
2004-01-28 08:50:41 +01:00
|
|
|
reinterpret_cast<const char*>(service_name),
|
2002-12-16 17:52:35 +01:00
|
|
|
&handle,
|
2005-11-27 21:53:09 +01:00
|
|
|
spb.getBufferLength(),
|
|
|
|
reinterpret_cast<const char*>(spb.getBuffer()));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!status_vector[1]) {
|
2004-05-17 12:22:34 +02:00
|
|
|
RDB rdb = (RDB) ALLR_block(type_rdb, 0);
|
2004-01-28 08:50:41 +01:00
|
|
|
this->port_context = rdb;
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("attach_service(server) allocate rdb %x\n", rdb);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
rdb->rdb_port = this;
|
|
|
|
rdb->rdb_handle = handle;
|
|
|
|
rdb->rdb_flags |= RDB_service;
|
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::service_end(P_RLSE * release, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e r v i c e _ e n d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Close down a service.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
RDB rdb = this->port_context;
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_service_detach(status_vector, &rdb->rdb_handle);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::service_start(P_INFO * stuff, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e r v i c e _ s t a r t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Start a service on the server
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-01-28 08:50:41 +01:00
|
|
|
RDB rdb = this->port_context;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2004-02-24 06:34:44 +01:00
|
|
|
SLONG* reserved = 0; // reserved for future use
|
2001-05-23 15:26:42 +02:00
|
|
|
isc_service_start(status_vector,
|
2002-12-16 17:52:35 +01:00
|
|
|
&rdb->rdb_handle,
|
2004-02-24 06:34:44 +01:00
|
|
|
reserved,
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_info_items.cstr_length,
|
|
|
|
reinterpret_cast<char*>(stuff->p_info_items.cstr_address));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::set_cursor(P_SQLCUR * sqlcur, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/*****************************************
|
|
|
|
*
|
|
|
|
* s e t _ c u r s o r
|
|
|
|
*
|
|
|
|
*****************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Set a cursor name for a dynamic request.
|
|
|
|
*
|
|
|
|
*****************************************/
|
|
|
|
RSR statement;
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
CHECK_HANDLE_MEMBER(statement,
|
|
|
|
RSR,
|
|
|
|
type_rsr,
|
|
|
|
sqlcur->p_sqlcur_statement,
|
|
|
|
isc_bad_req_handle);
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
GDS_DSQL_SET_CURSOR(status_vector,
|
2002-12-16 17:52:35 +01:00
|
|
|
&statement->rsr_handle,
|
2003-11-28 07:48:34 +01:00
|
|
|
reinterpret_cast<const char*>(sqlcur->p_sqlcur_cursor_name.cstr_address),
|
2001-05-23 15:26:42 +02:00
|
|
|
sqlcur->p_sqlcur_type);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
void set_server( rem_port* port, USHORT flags)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e t _ s e r v e r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Look up the server for this type
|
|
|
|
* of port. If one doesn't exist,
|
|
|
|
* create it.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SRVR server;
|
|
|
|
|
|
|
|
for (server = servers; server; server = server->srvr_next) {
|
|
|
|
if (port->port_type == server->srvr_port_type) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!server) {
|
|
|
|
server = (SRVR) ALLR_alloc((SLONG) sizeof(struct srvr));
|
|
|
|
server->srvr_next = servers;
|
|
|
|
servers = server;
|
|
|
|
server->srvr_port_type = port->port_type;
|
|
|
|
server->srvr_parent_port = port;
|
|
|
|
server->srvr_flags = flags;
|
|
|
|
}
|
|
|
|
|
|
|
|
port->port_server = server;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::start(P_OP operation, P_DATA * data, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s t a r t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
RTR transaction;
|
|
|
|
|
|
|
|
CHECK_HANDLE_MEMBER(transaction,
|
|
|
|
RTR,
|
|
|
|
type_rtr,
|
|
|
|
data->p_data_transaction,
|
|
|
|
isc_bad_trans_handle);
|
|
|
|
|
2004-05-12 23:47:36 +02:00
|
|
|
rrq* requestL;
|
|
|
|
CHECK_HANDLE_MEMBER(requestL,
|
2004-01-28 08:50:41 +01:00
|
|
|
rrq*,
|
2001-05-23 15:26:42 +02:00
|
|
|
type_rrq,
|
|
|
|
data->p_data_request,
|
|
|
|
isc_bad_req_handle);
|
|
|
|
|
2004-05-12 23:47:36 +02:00
|
|
|
requestL = REMOTE_find_request(requestL, data->p_data_incarnation);
|
|
|
|
REMOTE_reset_request(requestL, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2004-05-12 23:47:36 +02:00
|
|
|
isc_start_request(status_vector, &requestL->rrq_handle,
|
2003-08-30 03:43:08 +02:00
|
|
|
&transaction->rtr_handle, data->p_data_incarnation);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!status_vector[1]) {
|
2004-05-12 23:47:36 +02:00
|
|
|
requestL->rrq_rtr = transaction;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (operation == op_start_and_receive)
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->receive_after_start(data, sendL, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
ISC_STATUS rem_port::start_and_send(P_OP operation,
|
2001-05-23 15:26:42 +02:00
|
|
|
P_DATA* data,
|
2004-05-12 21:39:17 +02:00
|
|
|
PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s t a r t _ a n d _ s e n d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-02-24 06:34:44 +01:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
RTR transaction;
|
|
|
|
CHECK_HANDLE_MEMBER(transaction,
|
|
|
|
RTR,
|
|
|
|
type_rtr,
|
|
|
|
data->p_data_transaction,
|
|
|
|
isc_bad_trans_handle);
|
|
|
|
|
2004-05-12 23:47:36 +02:00
|
|
|
rrq* requestL;
|
|
|
|
CHECK_HANDLE_MEMBER(requestL,
|
2004-01-28 08:50:41 +01:00
|
|
|
rrq*,
|
2001-05-23 15:26:42 +02:00
|
|
|
type_rrq,
|
|
|
|
data->p_data_request,
|
|
|
|
isc_bad_req_handle);
|
|
|
|
|
2004-05-12 23:47:36 +02:00
|
|
|
requestL = REMOTE_find_request(requestL, data->p_data_incarnation);
|
2004-02-24 06:34:44 +01:00
|
|
|
const USHORT number = data->p_data_message_number;
|
2004-05-12 23:47:36 +02:00
|
|
|
REM_MSG message = requestL->rrq_rpt[number].rrq_message;
|
|
|
|
const rem_fmt* format = requestL->rrq_rpt[number].rrq_format;
|
|
|
|
REMOTE_reset_request(requestL, message);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2004-05-12 23:47:36 +02:00
|
|
|
isc_start_and_send(status_vector, &requestL->rrq_handle,
|
2003-08-30 03:43:08 +02:00
|
|
|
&transaction->rtr_handle, number,
|
|
|
|
format->fmt_length, message->msg_address,
|
2001-05-23 15:26:42 +02:00
|
|
|
data->p_data_incarnation);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!status_vector[1]) {
|
2004-05-12 23:47:36 +02:00
|
|
|
requestL->rrq_rtr = transaction;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (operation == op_start_send_and_receive) {
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->receive_after_start(data, sendL, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::start_transaction(P_OP operation, P_STTR * stuff, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s t a r t _ t r a n s a c t i o n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Start a transaction.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
RDB rdb = this->port_context;
|
2004-05-03 01:06:37 +02:00
|
|
|
FB_API_HANDLE handle = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
if (operation == op_reconnect)
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_reconnect_transaction(status_vector, &rdb->rdb_handle, &handle,
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_sttr_tpb.cstr_length,
|
2003-11-01 11:26:43 +01:00
|
|
|
reinterpret_cast<const char*>(stuff->p_sttr_tpb.cstr_address));
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_start_transaction(status_vector, &handle, (SSHORT) 1, &rdb->rdb_handle,
|
2001-05-23 15:26:42 +02:00
|
|
|
stuff->p_sttr_tpb.cstr_length,
|
2003-08-30 03:43:08 +02:00
|
|
|
stuff->p_sttr_tpb.cstr_address);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
OBJCT object;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (status_vector[1])
|
|
|
|
{
|
|
|
|
object = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2004-02-24 06:34:44 +01:00
|
|
|
RTR transaction = make_transaction(rdb, handle);
|
|
|
|
if (transaction)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
object = transaction->rtr_id;
|
|
|
|
if (operation == op_reconnect)
|
|
|
|
transaction->rtr_flags |= RTR_limbo;
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("start_transaction(server) allocate trans %x\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
transaction);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
object = 0;
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
if (operation != op_reconnect)
|
2003-08-30 03:43:08 +02:00
|
|
|
isc_rollback_transaction(status_vector, &handle);
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef SUPERSERVER
|
|
|
|
/* Note that there is an underlying transaction pool
|
|
|
|
that won't be released until this connection is
|
|
|
|
detached. There is no isc_disconnect_transaction()
|
|
|
|
call that gets rid of a transaction reference --
|
|
|
|
there is only commit and rollback. It's not worth
|
|
|
|
introducing such a call to all subsystems. At least
|
|
|
|
release the y-valve handle. */
|
|
|
|
|
|
|
|
else {
|
2004-05-03 01:06:37 +02:00
|
|
|
gds__handle_cleanup(status_vector, &handle);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
status_vector[0] = isc_arg_gds;
|
|
|
|
status_vector[1] = isc_too_many_handles;
|
|
|
|
status_vector[2] = isc_arg_end;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, object, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-04-10 12:31:28 +02:00
|
|
|
static void success( ISC_STATUS * status_vector)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s u c c e s s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Set status vector to indicate success.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-11-08 17:40:17 +01:00
|
|
|
status_vector[0] = isc_arg_gds;
|
2002-11-14 09:33:08 +01:00
|
|
|
status_vector[1] = FB_SUCCESS;
|
2003-11-08 17:40:17 +01:00
|
|
|
status_vector[2] = isc_arg_end;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2003-09-24 00:34:14 +02:00
|
|
|
#ifdef MULTI_THREAD
|
2004-06-08 15:41:08 +02:00
|
|
|
static THREAD_ENTRY_DECLARE loopThread(THREAD_ENTRY_PARAM flags)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* t h r e a d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Execute requests in a happy loop.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
ISC_enter(); /* Setup floating point exception handler once and for all. */
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
2004-09-24 08:46:04 +02:00
|
|
|
void* thread = NULL; // silence non initialized warning
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!((SLONG) flags & SRVR_non_service))
|
2004-09-24 08:46:04 +02:00
|
|
|
thread = CNTL_insert_thread();
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
USHORT inactive_count = 0;
|
|
|
|
USHORT timedout_count = 0;
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
rem_port* port;
|
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
for (;;)
|
|
|
|
{
|
2004-02-24 06:34:44 +01:00
|
|
|
SERVER_REQ request = request_que;
|
|
|
|
if (request)
|
2002-03-31 01:40:08 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
inactive_count = 0;
|
|
|
|
timedout_count = 0;
|
2003-09-16 22:59:45 +02:00
|
|
|
REMOTE_TRACE(("Dequeue request %p", request_que));
|
2001-05-23 15:26:42 +02:00
|
|
|
request_que = request->req_next;
|
2002-03-31 01:40:08 +01:00
|
|
|
while (request)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Bind a thread to a port. */
|
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
if (request->req_port->port_server_flags & SRVR_thread_per_port)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
port = request->req_port;
|
|
|
|
request->req_next = free_requests;
|
|
|
|
free_requests = request;
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
SRVR_main(port, port->port_server_flags);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
request = 0;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
/* Splice request into list of active requests, execute request,
|
|
|
|
and unsplice */
|
|
|
|
|
|
|
|
request->req_next = active_requests;
|
|
|
|
active_requests = request;
|
|
|
|
|
|
|
|
/* Validate port. If it looks ok, process request */
|
|
|
|
|
2004-03-24 21:12:35 +01:00
|
|
|
if (request->req_port->port_state != state_disconnected)
|
2002-03-31 01:40:08 +01:00
|
|
|
{
|
2004-03-24 21:12:35 +01:00
|
|
|
rem_port* parent_port =
|
|
|
|
request->req_port->port_server->srvr_parent_port;
|
|
|
|
if (parent_port == request->req_port)
|
2002-03-31 01:40:08 +01:00
|
|
|
{
|
2004-03-24 21:12:35 +01:00
|
|
|
process_packet(parent_port, &request->req_send,
|
|
|
|
&request->req_receive, &port);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for (port = parent_port->port_clients; port;
|
|
|
|
port = port->port_next)
|
2004-01-28 08:50:41 +01:00
|
|
|
{
|
2004-03-24 21:12:35 +01:00
|
|
|
if (port == request->req_port)
|
|
|
|
{
|
|
|
|
process_packet(port, &request->req_send,
|
|
|
|
&request->req_receive, &port);
|
|
|
|
break;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2002-03-31 01:40:08 +01:00
|
|
|
}
|
|
|
|
}
|
2004-03-24 21:12:35 +01:00
|
|
|
else
|
|
|
|
{
|
|
|
|
port = NULL;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Take request out of list of active requests */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
for (SERVER_REQ* req_ptr = &active_requests; *req_ptr;
|
2001-05-23 15:26:42 +02:00
|
|
|
req_ptr = &(*req_ptr)->req_next)
|
2002-03-31 01:40:08 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if (*req_ptr == request) {
|
|
|
|
*req_ptr = request->req_next;
|
|
|
|
break;
|
|
|
|
}
|
2002-03-31 01:40:08 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If this is a explicit or implicit disconnect, get rid of
|
|
|
|
any chained requests */
|
|
|
|
|
2002-03-31 01:40:08 +01:00
|
|
|
if (!port)
|
|
|
|
{
|
2004-02-24 06:34:44 +01:00
|
|
|
SERVER_REQ next;
|
2001-05-23 15:26:42 +02:00
|
|
|
while (next = request->req_chain) {
|
|
|
|
request->req_chain = next->req_chain;
|
|
|
|
next->req_next = free_requests;
|
|
|
|
free_requests = next;
|
|
|
|
}
|
|
|
|
if (request->req_send.p_operation == op_void &&
|
2004-01-28 08:50:41 +01:00
|
|
|
request->req_receive.p_operation == op_void)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
gds__free(request);
|
|
|
|
request = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
port->port_requests_queued--;
|
2003-02-11 01:54:58 +01:00
|
|
|
#ifdef DEBUG_REMOTE_MEMORY
|
2004-04-29 00:36:29 +02:00
|
|
|
printf("thread ACTIVE request_queued %d\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
port->port_requests_queued);
|
2004-04-29 00:36:29 +02:00
|
|
|
fflush(stdout);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Pick up any remaining chained request, and free current request */
|
|
|
|
|
|
|
|
if (request) {
|
2004-02-24 06:34:44 +01:00
|
|
|
SERVER_REQ next = request->req_chain;
|
2001-05-23 15:26:42 +02:00
|
|
|
request->req_next = free_requests;
|
|
|
|
free_requests = request;
|
|
|
|
request = next;
|
|
|
|
}
|
2002-03-31 01:40:08 +01:00
|
|
|
} // while (request)
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
inactive_count++;
|
|
|
|
/* If the main server thread is not relying on me to take on a new request
|
|
|
|
and I have been idling for a while, it is time to quit and thereby
|
|
|
|
release some valuable system resources like memory */
|
|
|
|
if ((extra_threads > 1)
|
2004-01-28 08:50:41 +01:00
|
|
|
&& ((inactive_count > 20) || (timedout_count > 2)))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
extra_threads--; /* Count me out */
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
threads_waiting++;
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Wait for 1 minute (60 seconds) on a new request */
|
2003-09-16 22:59:45 +02:00
|
|
|
REMOTE_TRACE(("Wait for event"));
|
|
|
|
if (!requests_semaphore.tryEnter(60)) {
|
|
|
|
REMOTE_TRACE(("timeout!"));
|
2001-05-23 15:26:42 +02:00
|
|
|
timedout_count++;
|
2004-01-28 08:50:41 +01:00
|
|
|
}
|
|
|
|
else {
|
2003-09-16 22:59:45 +02:00
|
|
|
REMOTE_TRACE(("got it"));
|
|
|
|
}
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
--threads_waiting;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
if (!((SLONG) flags & SRVR_non_service))
|
|
|
|
CNTL_remove_thread(thread);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
2003-09-24 00:34:14 +02:00
|
|
|
#endif
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
ISC_STATUS rem_port::transact_request(P_TRRQ* trrq, PACKET* sendL)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* t r a n s a c t _ r e q u e s t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-04-16 12:18:51 +02:00
|
|
|
ISC_STATUS_ARRAY status_vector;
|
2001-05-23 15:26:42 +02:00
|
|
|
RTR transaction;
|
|
|
|
|
|
|
|
CHECK_HANDLE_MEMBER(transaction,
|
|
|
|
RTR,
|
|
|
|
type_rtr,
|
|
|
|
trrq->p_trrq_transaction,
|
|
|
|
isc_bad_trans_handle);
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
RDB rdb = this->port_context;
|
|
|
|
const UCHAR* blr = trrq->p_trrq_blr.cstr_address;
|
|
|
|
const USHORT blr_length = trrq->p_trrq_blr.cstr_length;
|
|
|
|
RPR procedure = this->port_rpr;
|
|
|
|
UCHAR* in_msg =
|
2001-05-23 15:26:42 +02:00
|
|
|
(procedure->rpr_in_msg) ? procedure->rpr_in_msg->msg_address : NULL;
|
2004-02-24 06:34:44 +01:00
|
|
|
const USHORT in_msg_length =
|
2001-05-23 15:26:42 +02:00
|
|
|
(procedure->rpr_in_format) ? procedure->rpr_in_format->fmt_length : 0;
|
2004-02-24 06:34:44 +01:00
|
|
|
UCHAR* out_msg =
|
2001-05-23 15:26:42 +02:00
|
|
|
(procedure->rpr_out_msg) ? procedure->rpr_out_msg->msg_address : NULL;
|
2004-02-24 06:34:44 +01:00
|
|
|
const USHORT out_msg_length =
|
2004-12-16 04:03:13 +01:00
|
|
|
(procedure->rpr_out_format) ? procedure->rpr_out_format->fmt_length : 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
isc_transact_request(status_vector,
|
2002-12-16 17:52:35 +01:00
|
|
|
&rdb->rdb_handle,
|
|
|
|
&transaction->rtr_handle,
|
2001-05-23 15:26:42 +02:00
|
|
|
blr_length,
|
2003-12-03 09:19:24 +01:00
|
|
|
reinterpret_cast<const char*>(blr),
|
2001-05-23 15:26:42 +02:00
|
|
|
in_msg_length,
|
|
|
|
reinterpret_cast<char*>(in_msg),
|
|
|
|
out_msg_length,
|
|
|
|
reinterpret_cast<char*>(out_msg));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (status_vector[1])
|
2004-05-12 21:39:17 +02:00
|
|
|
return this->send_response(sendL, 0, 0, status_vector);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-12 21:39:17 +02:00
|
|
|
P_DATA* data = &sendL->p_data;
|
|
|
|
sendL->p_operation = op_transact_response;
|
2001-05-23 15:26:42 +02:00
|
|
|
data->p_data_messages = 1;
|
2004-05-12 21:39:17 +02:00
|
|
|
this->send(sendL);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-11-14 09:33:08 +01:00
|
|
|
return FB_SUCCESS;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-09-11 20:59:34 +02:00
|
|
|
static void zap_packet(PACKET* packet,
|
2004-02-24 06:34:44 +01:00
|
|
|
bool new_packet)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* z a p _ p a c k e t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Zero out a packet block.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
if (new_packet)
|
2001-05-23 15:26:42 +02:00
|
|
|
memset(packet, 0, sizeof(PACKET));
|
2004-02-24 06:34:44 +01:00
|
|
|
else {
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef DEBUG_XDR_MEMORY
|
|
|
|
/* Don't trash debug xdr memory allocation table of recycled packets. */
|
|
|
|
|
|
|
|
memset(&packet->p_operation, 0,
|
|
|
|
sizeof(PACKET) - OFFSET(PACKET*, p_operation));
|
|
|
|
#else
|
|
|
|
memset(packet, 0, sizeof(PACKET));
|
|
|
|
#endif
|
2004-02-24 06:34:44 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2003-10-29 11:53:47 +01:00
|
|
|
|