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

831 lines
22 KiB
C++
Raw Normal View History

2001-05-23 15:26:42 +02:00
/*
* PROGRAM: JRD Remote Interface
* MODULE: remote.cpp
2001-05-23 15:26:42 +02:00
* DESCRIPTION: Common routines for remote interface/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): ______________________________________.
*/
#include "firebird.h"
2001-05-23 15:26:42 +02:00
#include <string.h>
#include <stdlib.h>
2003-11-08 17:40:17 +01:00
#include "../jrd/ibase.h"
2001-05-23 15:26:42 +02:00
#include "../remote/remote.h"
#include "../jrd/file_params.h"
#include "../jrd/gdsassert.h"
#include "../jrd/isc.h"
#include "../remote/allr_proto.h"
#include "../remote/proto_proto.h"
#include "../remote/remot_proto.h"
#include "../jrd/gds_proto.h"
#include "../jrd/thd.h"
#include "../jrd/thread_proto.h"
2002-12-06 22:35:47 +01:00
#include "../common/config/config.h"
2001-05-23 15:26:42 +02:00
#ifdef REMOTE_DEBUG
2004-06-14 01:47:02 +02:00
IMPLEMENT_TRACE_ROUTINE(remote_trace, "REMOTE")
#endif
int xdrmem_create(XDR *, SCHAR *, u_int, enum xdr_op);
2001-05-23 15:26:42 +02:00
2004-05-17 12:22:34 +02:00
const SLONG DUMMY_INTERVAL = 60; /* seconds */
const int ATTACH_FAILURE_SPACE = 2048; /* bytes */
2001-05-23 15:26:42 +02:00
static TEXT* attach_failures = NULL;
static TEXT* attach_failures_ptr;
2001-05-23 15:26:42 +02:00
static void cleanup_memory(void*);
static SLONG get_parameter(const UCHAR**);
2001-05-23 15:26:42 +02:00
void REMOTE_cleanup_transaction( RTR transaction)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* R E M O T E _ c l e a n u p _ t r a n s a c t i o n
*
**************************************
*
* Functional description
* A transaction is being committed or rolled back.
* Purge any active messages in case the user calls
* receive while we still have something cached.
*
**************************************/
for (rrq* request = transaction->rtr_rdb->rdb_requests; request;
request = request->rrq_next)
{
2001-05-23 15:26:42 +02:00
if (request->rrq_rtr == transaction) {
REMOTE_reset_request(request, 0);
request->rrq_rtr = NULL;
}
for (rrq* level = request->rrq_levels; level; level = level->rrq_next)
2001-05-23 15:26:42 +02:00
if (level->rrq_rtr == transaction) {
REMOTE_reset_request(level, 0);
level->rrq_rtr = NULL;
}
}
for (RSR statement = transaction->rtr_rdb->rdb_sql_requests; statement;
2001-05-23 15:26:42 +02:00
statement = statement->rsr_next)
{
2001-05-23 15:26:42 +02:00
if (statement->rsr_rtr == transaction) {
REMOTE_reset_statement(statement);
statement->rsr_flags &= ~RSR_fetched;
statement->rsr_rtr = NULL;
}
}
2001-05-23 15:26:42 +02:00
}
ULONG REMOTE_compute_batch_size(rem_port* port,
USHORT buffer_used, P_OP op_code,
const rem_fmt* format)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* R E M O T E _ c o m p u t e _ b a t c h _ s i z e
*
**************************************
*
* Functional description
*
* When batches of records are returned, they are returned as
* follows:
* <op_fetch_response> <data_record 1>
* <op_fetch_response> <data_record 2>
* ...
* <op_fetch_response> <data_record n-1>
* <op_fetch_response> <data_record n>
*
* end-of-batch is indicated by setting p_sqldata_messages to
* 0 in the op_fetch_response. End of cursor is indicated
* by setting p_sqldata_status to a non-zero value. Note
* that a fetch CAN be attempted after end of cursor, this
* is sent to the server for the server to return the appropriate
* error code.
*
* Each data block has one overhead packet
* to indicate the data is present.
*
* (See also op_send in receive_msg() - which is a kissing cousin
* to this routine)
*
* Here we make a guess for the optimal number of records to
* send in each batch. This is important as we wait for the
* whole batch to be received before we return the first item
* to the client program. How many are cached on the client also
* impacts client-side memory utilization.
*
* We optimize the number by how many can fit into a packet.
* The client calculates this number (n from the list above)
* and sends it to the server.
*
* I asked why it is that the client doesn't just ask for a packet
* full of records and let the server return however many fits in
* a packet. According to Sudesh, this is because of a bug in
* Superserver which showed up in the WIN_NT 4.2.x kits. So I
* imagine once we up the protocol so that we can be sure we're not
* talking to a 4.2 kit, then we can make this optimization.
* - Deej 2/28/97
*
* Note: A future optimization can look at setting the packet
* size to optimize the transfer.
*
* Note: This calculation must use worst-case to determine the
* packing. Should the data record have VARCHAR data, it is
* often possible to fit more than the packing specification
* into each packet. This is also a candidate for future
* optimization.
*
* The data size is either the XDR data representation, or the
* actual message size (rounded up) if this is a symmetric
* architecture connection.
*
**************************************/
2004-05-17 12:22:34 +02:00
const USHORT MAX_PACKETS_PER_BATCH = 4; /* packets - picked by SWAG */
const USHORT MIN_PACKETS_PER_BATCH = 2; /* packets - picked by SWAG */
const USHORT DESIRED_ROWS_PER_BATCH = 20; /* data rows - picked by SWAG */
const USHORT MIN_ROWS_PER_BATCH = 10; /* data rows - picked by SWAG */
2001-05-23 15:26:42 +02:00
USHORT op_overhead = (USHORT) xdr_protocol_overhead(op_code);
#ifdef DEBUG
2004-04-29 00:36:29 +02:00
fprintf(stderr,
2001-05-23 15:26:42 +02:00
"port_buff_size = %d fmt_net_length = %d fmt_length = %d overhead = %d\n",
port->port_buff_size, format->fmt_net_length,
format->fmt_length, op_overhead);
#endif
ULONG row_size;
2001-05-23 15:26:42 +02:00
if (port->port_flags & PORT_symmetric) {
/* Same architecture connection */
row_size = (ROUNDUP(format->fmt_length, 4)
+ op_overhead);
}
else {
/* Using XDR for data transfer */
row_size = (ROUNDUP(format->fmt_net_length, 4)
+ op_overhead);
}
USHORT num_packets = (USHORT) (((DESIRED_ROWS_PER_BATCH * row_size) /* data set */
+ buffer_used /* used in 1st pkt */
2001-05-23 15:26:42 +02:00
+ (port->port_buff_size - 1)) /* to round up */
/port->port_buff_size);
if (num_packets > MAX_PACKETS_PER_BATCH) {
num_packets = (USHORT) (((MIN_ROWS_PER_BATCH * row_size) /* data set */
+ buffer_used /* used in 1st pkt */
2001-05-23 15:26:42 +02:00
+ (port->port_buff_size - 1)) /* to round up */
/port->port_buff_size);
}
num_packets = MAX(num_packets, MIN_PACKETS_PER_BATCH);
/* Now that we've picked the number of packets in a batch,
pack as many rows as we can into the set of packets */
ULONG result = (num_packets * port->port_buff_size - buffer_used) / row_size;
2001-05-23 15:26:42 +02:00
/* Must always send some messages, even if message size is more
than packet size. */
result = MAX(result, MIN_ROWS_PER_BATCH);
#ifdef DEBUG
{
const char* p = getenv("DEBUG_BATCH_SIZE");
if (p)
2001-05-23 15:26:42 +02:00
result = atoi(p);
2004-04-29 00:36:29 +02:00
fprintf(stderr, "row_size = %lu num_packets = %d\n",
2001-05-23 15:26:42 +02:00
row_size, num_packets);
2004-04-29 00:36:29 +02:00
fprintf(stderr, "result = %lu\n", result);
2001-05-23 15:26:42 +02:00
}
#endif
return result;
}
rrq* REMOTE_find_request(rrq* request, USHORT level)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* R E M O T E _ f i n d _ r e q u e s t
*
**************************************
*
* Functional description
* Find sub-request if level is non-zero.
*
**************************************/
/* See if we already know about the request level */
for (;;) {
if (request->rrq_level == level)
return request;
if (!request->rrq_levels)
break;
request = request->rrq_levels;
}
/* This is a new level -- make up a new request block. */
request->rrq_levels = (rrq*) ALLR_clone(&request->rrq_header);
2001-05-23 15:26:42 +02:00
/* NOMEM: handled by ALLR_clone, FREE: REMOTE_remove_request() */
#ifdef DEBUG_REMOTE_MEMORY
2004-04-29 00:36:29 +02:00
printf("REMOTE_find_request allocate request %x\n",
2001-05-23 15:26:42 +02:00
request->rrq_levels);
#endif
request = request->rrq_levels;
request->rrq_level = level;
request->rrq_levels = NULL;
/* Allocate message block for known messages */
rrq::rrq_repeat* tail = request->rrq_rpt;
const rrq::rrq_repeat* const end = tail + request->rrq_max_msg;
2001-05-23 15:26:42 +02:00
for (; tail <= end; tail++) {
const rem_fmt* format = tail->rrq_format;
if (!format)
2001-05-23 15:26:42 +02:00
continue;
2004-05-17 12:22:34 +02:00
REM_MSG msg = (REM_MSG) ALLR_block(type_msg, format->fmt_length);
tail->rrq_xdr = msg;
#ifdef DEBUG_REMOTE_MEMORY
2004-04-29 00:36:29 +02:00
printf("REMOTE_find_request allocate message %x\n", msg);
2001-05-23 15:26:42 +02:00
#endif
msg->msg_next = msg;
#ifdef SCROLLABLE_CURSORS
msg->msg_prior = msg;
#endif
msg->msg_number = tail->rrq_message->msg_number;
tail->rrq_message = msg;
}
return request;
}
void REMOTE_free_packet( rem_port* port, PACKET * packet)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* R E M O T E _ f r e e _ p a c k e t
*
**************************************
*
* Functional description
* Zero out a packet block.
*
**************************************/
XDR xdr;
USHORT n;
if (packet) {
xdrmem_create(&xdr, reinterpret_cast < char *>(packet),
sizeof(PACKET), XDR_FREE);
xdr.x_public = (caddr_t) port;
for (n = (USHORT) op_connect; n < (USHORT) op_max; n++) {
packet->p_operation = (P_OP) n;
xdr_protocol(&xdr, packet);
}
#ifdef DEBUG_XDR_MEMORY
/* All packet memory allocations should now be voided. */
for (n = 0; n < P_MALLOC_SIZE; n++)
2003-11-04 00:59:24 +01:00
fb_assert(packet->p_malloc[n].p_operation == op_void);
2001-05-23 15:26:42 +02:00
#endif
packet->p_operation = op_void;
}
}
void REMOTE_get_timeout_params(
rem_port* port,
const UCHAR* dpb, USHORT dpb_length)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* R E M O T E _ g e t _ t i m e o u t _ p a r a m s
*
**************************************
*
* Functional description
* Determine the connection timeout parameter values for this newly created
* port. If the client did a specification in the DPB, use those values.
* Otherwise, see if there is anything in the configuration file. The
* configuration file management code will set the default values if there
* is no other specification.
*
**************************************/
2003-09-11 20:59:34 +02:00
bool got_dpb_connect_timeout = false;
bool got_dpb_dummy_packet_interval = false;
2001-05-23 15:26:42 +02:00
2003-11-04 00:59:24 +01:00
fb_assert(isc_dpb_connect_timeout == isc_spb_connect_timeout);
fb_assert(isc_dpb_dummy_packet_interval == isc_spb_dummy_packet_interval);
2001-05-23 15:26:42 +02:00
port->port_flags &= ~PORT_dummy_pckt_set;
if (dpb && dpb_length) {
const UCHAR* p = dpb;
const UCHAR* const end = p + dpb_length;
2001-05-23 15:26:42 +02:00
2003-11-08 17:40:17 +01:00
if (*p++ == isc_dpb_version1) {
2001-05-23 15:26:42 +02:00
while (p < end)
switch (*p++) {
2003-11-08 17:40:17 +01:00
case isc_dpb_connect_timeout:
2001-05-23 15:26:42 +02:00
port->port_connect_timeout = get_parameter(&p);
2003-09-11 20:59:34 +02:00
got_dpb_connect_timeout = true;
2001-05-23 15:26:42 +02:00
break;
2003-08-22 00:30:20 +02:00
// 22 Aug 2003. Do not receive this parameter from the client as dummy packets
// either kill idle client process or cause unexpected disconnections.
// This applies to all IB/FB versions.
2003-11-08 17:40:17 +01:00
// case isc_dpb_dummy_packet_interval:
2003-08-22 00:30:20 +02:00
// port->port_dummy_packet_interval = get_parameter(&p);
2003-09-11 20:59:34 +02:00
// got_dpb_dummy_packet_interval = true;
2003-08-22 00:30:20 +02:00
// port->port_flags |= PORT_dummy_pckt_set;
// break;
2001-05-23 15:26:42 +02:00
case isc_dpb_sys_user_name:
/** Store the user name in thread specific storage.
We need this later while expanding filename to
get the users home directory.
Normally the working directory is stored in
the attachment but in this case the attachment is
not yet created.
Also note that the thread performing this task
has already called THREAD_ENTER
**/
{
char* t_data;
int i = 0;
int l = *(p++);
2001-05-23 15:26:42 +02:00
if (l) {
t_data = (char *) malloc(l + 1);
do {
t_data[i] = *p;
if (t_data[i] == '.')
t_data[i] = 0;
i++;
p++;
} while (--l);
}
else
t_data = (char *) malloc(1);
t_data[i] = 0;
ThreadData::putSpecificData((void *) t_data);
2001-05-23 15:26:42 +02:00
}
break;
default:
{
// Skip over this parameter - not important to us
const USHORT len = *p++;
p += len;
break;
}
2001-05-23 15:26:42 +02:00
}
}
}
if (!got_dpb_connect_timeout || !got_dpb_dummy_packet_interval) {
/* Didn't find all parameters in the dpb, fetch configuration
information from the configuration file and set the missing
values */
if (!got_dpb_connect_timeout)
2002-12-06 22:35:47 +01:00
port->port_connect_timeout = Config::getConnectionTimeout();
2001-05-23 15:26:42 +02:00
if (!got_dpb_dummy_packet_interval) {
2002-12-06 22:35:47 +01:00
port->port_flags |= PORT_dummy_pckt_set;
port->port_dummy_packet_interval = Config::getDummyPacketInterval();
2001-05-23 15:26:42 +02:00
}
}
/* Insure a meaningful keepalive interval has been set. Otherwise, too
many keepalive packets will drain network performance. */
if (port->port_dummy_packet_interval < 0)
port->port_dummy_packet_interval = DUMMY_INTERVAL;
port->port_dummy_timeout = port->port_dummy_packet_interval;
#ifdef DEBUG
2004-04-29 00:36:29 +02:00
printf("REMOTE_get_timeout dummy = %lu conn = %lu\n",
2001-05-23 15:26:42 +02:00
port->port_dummy_packet_interval, port->port_connect_timeout);
fflush(stdout);
2001-05-23 15:26:42 +02:00
#endif
}
rem_str* REMOTE_make_string(const SCHAR* input)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* R E M O T E _ m a k e _ s t r i n g
*
**************************************
*
* Functional description
* Copy a given string to a permanent location, returning
* address of new string.
*
**************************************/
const USHORT length = strlen(input);
2004-05-17 12:22:34 +02:00
rem_str* string = (rem_str*) ALLR_block(type_str, length);
#ifdef DEBUG_REMOTE_MEMORY
2004-04-29 00:36:29 +02:00
printf("REMOTE_make_string allocate string %x\n", string);
2001-05-23 15:26:42 +02:00
#endif
strcpy(string->str_data, input);
string->str_length = length;
2001-05-23 15:26:42 +02:00
return string;
}
void REMOTE_release_messages( REM_MSG messages)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* R E M O T E _ r e l e a s e _ m e s s a g e s
*
**************************************
*
* Functional description
* Release a circular list of messages.
*
**************************************/
REM_MSG message = messages;
if (message)
2003-09-11 20:59:34 +02:00
while (true) {
REM_MSG temp = message;
2001-05-23 15:26:42 +02:00
message = message->msg_next;
#ifdef DEBUG_REMOTE_MEMORY
2004-04-29 00:36:29 +02:00
printf("REMOTE_release_messages free message %x\n",
2001-05-23 15:26:42 +02:00
temp);
#endif
ALLR_release(temp);
if (message == messages)
break;
}
}
void REMOTE_release_request( rrq* request)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* R E M O T E _ r e l e a s e _ r e q u e s t
*
**************************************
*
* Functional description
* Release a request block and friends.
*
**************************************/
RDB rdb = request->rrq_rdb;
2001-05-23 15:26:42 +02:00
for (rrq** p = &rdb->rdb_requests; *p; p = &(*p)->rrq_next)
2001-05-23 15:26:42 +02:00
if (*p == request) {
*p = request->rrq_next;
break;
}
/* Get rid of request and all levels */
for (;;) {
rrq::rrq_repeat* tail = request->rrq_rpt;
rrq::rrq_repeat* const end = tail + request->rrq_max_msg;
2001-05-23 15:26:42 +02:00
for (; tail <= end; tail++)
{
REM_MSG message = tail->rrq_message;
if (message) {
2001-05-23 15:26:42 +02:00
if (!request->rrq_level) {
#ifdef DEBUG_REMOTE_MEMORY
2004-04-29 00:36:29 +02:00
printf
2001-05-23 15:26:42 +02:00
("REMOTE_release_request free format %x\n",
tail->rrq_format);
#endif
ALLR_release(tail->rrq_format);
}
REMOTE_release_messages(message);
}
}
rrq* next = request->rrq_levels;
#ifdef DEBUG_REMOTE_MEMORY
2004-04-29 00:36:29 +02:00
printf("REMOTE_release_request free request %x\n", request);
2001-05-23 15:26:42 +02:00
#endif
ALLR_release(request);
if (!(request = next))
break;
}
}
void REMOTE_reset_request( rrq* request, REM_MSG active_message)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* R E M O T E _ r e s e t _ r e q u e s t
*
**************************************
*
* Functional description
* Clean up a request in preparation to use it again. Since
* there may be an active message (start_and_send), exercise
* some care to avoid zapping that message.
*
**************************************/
rrq::rrq_repeat* tail = request->rrq_rpt;
rrq::rrq_repeat* end = tail + request->rrq_max_msg;
for (; tail <= end; tail++) {
REM_MSG message = tail->rrq_message;
if (message != NULL && message != active_message) {
2001-05-23 15:26:42 +02:00
tail->rrq_xdr = message;
tail->rrq_rows_pending = 0;
tail->rrq_reorder_level = 0;
tail->rrq_batch_count = 0;
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_message)
break;
}
}
}
2001-05-23 15:26:42 +02:00
/* Initialize the request status to FB_SUCCESS */
2001-05-23 15:26:42 +02:00
request->rrq_status_vector[1] = 0;
}
void REMOTE_reset_statement( RSR statement)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* R E M O T E _ r e s e t _ s t a t e m e n t
*
**************************************
*
* Functional description
* Reset a statement by releasing all buffers except 1
*
**************************************/
REM_MSG message;
2001-05-23 15:26:42 +02:00
if ((!statement) || (!(message = statement->rsr_message)))
return;
/* Reset all the pipeline counters */
statement->rsr_rows_pending = 0;
statement->rsr_msgs_waiting = 0;
statement->rsr_reorder_level = 0;
statement->rsr_batch_count = 0;
/* only one entry */
if (message->msg_next == message)
return;
/* find the entry before statement->rsr_message */
REM_MSG temp;
2001-05-23 15:26:42 +02:00
for (temp = message->msg_next; temp->msg_next != message;
temp = temp->msg_next);
temp->msg_next = message->msg_next;
message->msg_next = message;
#ifdef SCROLLABLE_CURSORS
message->msg_prior = message;
#endif
statement->rsr_buffer = statement->rsr_message;
REMOTE_release_messages(temp);
}
void REMOTE_save_status_strings( ISC_STATUS* vector)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* R E M O T E _ s a v e _ s t a t u s _ s t r i n g s
*
**************************************
*
* Functional description
* There has been a failure during attach/create database.
* The included string have been allocated off of the database block,
* which is going to be released before the message gets passed
* back to the user. So, to preserve information, copy any included
* strings to a special buffer.
*
**************************************/
if (!attach_failures)
{
attach_failures =
(TEXT*) ALLOC_LIB_MEMORY((SLONG) ATTACH_FAILURE_SPACE);
2001-05-23 15:26:42 +02:00
/* FREE: freed by exit handler cleanup_memory() */
if (!attach_failures) /* NOMEM: don't bother trying to copy */
return;
#ifdef DEBUG_GDS_ALLOC
/* This buffer is freed by the exit handler - but some of the
* reporting mechanisms will report it anyway, so flag it as
* "known unfreed" to get the reports quiet.
*/
gds_alloc_flag_unfreed((void*) attach_failures);
2001-05-23 15:26:42 +02:00
#endif /* DEBUG_GDS_ALLOC */
attach_failures_ptr = attach_failures;
gds__register_cleanup(cleanup_memory, 0);
}
TEXT* p;
2004-08-27 01:27:46 +02:00
USHORT l = 0; // silence non initialized warning
2001-05-23 15:26:42 +02:00
while (*vector)
{
const ISC_STATUS status = *vector++;
2001-05-23 15:26:42 +02:00
switch (status)
{
2003-11-08 17:40:17 +01:00
case isc_arg_cstring:
l = (USHORT) *vector++;
2001-05-23 15:26:42 +02:00
2003-11-08 17:40:17 +01:00
case isc_arg_interpreted:
case isc_arg_string:
p = (TEXT*) *vector;
2003-11-08 17:40:17 +01:00
if (status != isc_arg_cstring)
2001-05-23 15:26:42 +02:00
l = strlen(p) + 1;
/* If there isn't any more room in the buffer,
start at the beginning again */
if (attach_failures_ptr + l > attach_failures + ATTACH_FAILURE_SPACE)
attach_failures_ptr = attach_failures;
2003-04-10 12:31:28 +02:00
*vector++ = (ISC_STATUS) attach_failures_ptr;
2001-05-23 15:26:42 +02:00
while (l--)
*attach_failures_ptr++ = *p++;
break;
default:
++vector;
break;
}
}
}
OBJCT REMOTE_set_object(rem_port* port, BLK object, OBJCT slot)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* R E M O T E _ s e t _ o b j e c t
*
**************************************
*
* Functional description
* Set an object into the object vector.
*
**************************************/
/* If it fits, do it */
rem_vec* vector = port->port_object_vector;
if ((vector != NULL) && slot < vector->vec_count) {
2001-05-23 15:26:42 +02:00
vector->vec_object[slot] = object;
return slot;
}
/* Prevent the creation of object handles that can't be
transferred by the remote protocol. */
if (slot + 10 > MAX_OBJCT_HANDLES)
return (OBJCT) NULL;
2004-05-17 12:22:34 +02:00
rem_vec* new_vector = (rem_vec*) ALLR_block(type_vec, slot + 10);
port->port_object_vector = new_vector;
#ifdef DEBUG_REMOTE_MEMORY
2004-04-29 00:36:29 +02:00
printf("REMOTE_set_object allocate vector %x\n", new_vector);
2001-05-23 15:26:42 +02:00
#endif
port->port_objects = new_vector->vec_object;
new_vector->vec_count = slot + 10;
if (vector) {
blk** p = new_vector->vec_object;
blk* const* q = vector->vec_object;
const blk* const* const end = q + (int) vector->vec_count;
2001-05-23 15:26:42 +02:00
while (q < end)
*p++ = *q++;
#ifdef DEBUG_REMOTE_MEMORY
2004-04-29 00:36:29 +02:00
printf("REMOTE_release_request free vector %x\n", vector);
2001-05-23 15:26:42 +02:00
#endif
ALLR_release(vector);
}
new_vector->vec_object[slot] = object;
return slot;
}
static void cleanup_memory( void *block)
{
/**************************************
*
* c l e a n u p _ m e m o r y
*
**************************************
*
* Functional description
* Cleanup any allocated memory.
*
**************************************/
if (attach_failures)
FREE_LIB_MEMORY(attach_failures);
gds__unregister_cleanup(cleanup_memory, 0);
attach_failures = NULL;
}
static SLONG get_parameter(const UCHAR** ptr)
2001-05-23 15:26:42 +02:00
{
/**************************************
*
* g e t _ p a r a m e t e r
*
**************************************
*
* Functional description
* Pick up a VAX format parameter from a parameter block, including the
* length byte.
* This is a clone of jrd/jrd.c:get_parameter()
*
**************************************/
const SSHORT l = *(*ptr)++;
const SLONG parameter = gds__vax_integer(*ptr, l);
2001-05-23 15:26:42 +02:00
*ptr += l;
return parameter;
}
2001-07-12 07:46:06 +02:00
2001-05-23 15:26:42 +02:00
// TMN: Beginning of C++ port - ugly but a start
int rem_port::accept(p_cnct* cnct)
2001-05-23 15:26:42 +02:00
{
return (*this->port_accept)(this, cnct);
}
void rem_port::disconnect()
2001-05-23 15:26:42 +02:00
{
(*this->port_disconnect)(this);
}
rem_port* rem_port::receive(PACKET* pckt)
2001-05-23 15:26:42 +02:00
{
return (*this->port_receive_packet)(this, pckt);
}
XDR_INT rem_port::send(PACKET* pckt)
2001-05-23 15:26:42 +02:00
{
return (*this->port_send_packet)(this, pckt);
}
XDR_INT rem_port::send_partial(PACKET* pckt)
2001-05-23 15:26:42 +02:00
{
return (*this->port_send_partial)(this, pckt);
}
rem_port* rem_port::connect(PACKET* pckt, t_event_ast ast)
2001-05-23 15:26:42 +02:00
{
return (*this->port_connect)(this, pckt, ast);
}
rem_port* rem_port::request(PACKET* pckt)
2001-05-23 15:26:42 +02:00
{
return (*this->port_request)(this, pckt);
}