2001-05-23 15:26:42 +02:00
|
|
|
/*
|
|
|
|
* PROGRAM: JRD Remote Interface/Server
|
2003-09-25 13:49:12 +02:00
|
|
|
* MODULE: inet.cpp
|
2001-05-23 15:26:42 +02:00
|
|
|
* DESCRIPTION: TCP/UCP/IP Communications module.
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Interbase Public
|
|
|
|
* License Version 1.0 (the "License"); you may not use this file
|
|
|
|
* except in compliance with the License. You may obtain a copy
|
|
|
|
* of the License at http://www.Inprise.com/IPL.html
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an
|
|
|
|
* "AS IS" basis, WITHOUT WARRANTY OF ANY KIND, either express
|
|
|
|
* or implied. See the License for the specific language governing
|
|
|
|
* rights and limitations under the License.
|
|
|
|
*
|
|
|
|
* The Original Code was created by Inprise Corporation
|
|
|
|
* and its predecessors. Portions created by Inprise Corporation are
|
|
|
|
* Copyright (C) Inprise Corporation.
|
|
|
|
*
|
|
|
|
* All Rights Reserved.
|
|
|
|
* Contributor(s): ______________________________________.
|
2002-02-16 03:21:35 +01:00
|
|
|
*
|
2002-02-16 03:50:01 +01:00
|
|
|
* 2002.02.15 Sean Leyne - Code Cleanup, removed obsolete "EPSON" port
|
|
|
|
* 2002.02.15 Sean Leyne - Code Cleanup, removed obsolete "XENIX" port
|
2002-02-16 04:27:33 +01:00
|
|
|
* 2002.02.15 Sean Leyne - Code Cleanup, removed obsolete "IMP" port
|
2002-02-16 04:33:53 +01:00
|
|
|
* 2002.02.15 Sean Leyne - Code Cleanup, removed obsolete "NCR3000" port
|
2002-02-16 03:21:35 +01:00
|
|
|
*
|
2002-02-23 23:15:24 +01:00
|
|
|
* 2002-02-23 Sean Leyne - Code Cleanup, removed old M88K and NCR3000 port
|
|
|
|
*
|
2002-10-28 05:57:07 +01:00
|
|
|
* 2002.10.27 Sean Leyne - Code Cleanup, removed obsolete "UNIXWARE" port
|
2002-10-28 06:19:52 +01:00
|
|
|
* 2002.10.27 Sean Leyne - Code Cleanup, removed obsolete "Ultrix/MIPS" port
|
2002-10-28 05:57:07 +01:00
|
|
|
*
|
2002-10-29 03:45:09 +01:00
|
|
|
* 2002.10.28 Sean Leyne - Completed removal of obsolete "DGUX" port
|
|
|
|
*
|
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
|
2002-10-31 06:33:35 +01:00
|
|
|
* 2002.10.30 Sean Leyne - Code Cleanup, removed obsolete "SUN3_3" port
|
2005-04-04 14:54:34 +02:00
|
|
|
* 2005.04.01 Konstantin Kuznetsov - allow setting NoNagle option in Classic
|
2002-10-31 06:06:02 +01:00
|
|
|
*
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
2004-12-17 07:22:37 +01:00
|
|
|
|
2001-07-30 01:43:24 +02:00
|
|
|
#include "firebird.h"
|
2004-04-29 00:36:29 +02:00
|
|
|
#include <stdio.h>
|
2003-02-18 13:38:27 +01:00
|
|
|
#ifdef HAVE_SYS_TYPES_H
|
|
|
|
#include <sys/types.h>
|
|
|
|
#endif
|
2001-05-23 15:26:42 +02:00
|
|
|
#include <errno.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include "../jrd/common.h"
|
|
|
|
#include "../jrd/file_params.h"
|
|
|
|
#include <stdarg.h>
|
|
|
|
|
2006-02-03 14:13:27 +01:00
|
|
|
#include "../common/classes/timestamp.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-06-24 15:59:58 +02:00
|
|
|
#ifdef HAVE_PWD_H
|
|
|
|
#include <pwd.h>
|
|
|
|
#endif
|
|
|
|
|
2001-07-12 07:46:06 +02:00
|
|
|
#ifdef HAVE_SYS_PARAM_H
|
|
|
|
#include <sys/param.h>
|
|
|
|
#endif
|
|
|
|
|
2001-12-24 03:51:06 +01:00
|
|
|
#ifdef HAVE_GRP_H
|
2001-07-30 01:43:24 +02:00
|
|
|
#include <grp.h>
|
2001-12-24 03:51:06 +01:00
|
|
|
#endif
|
2001-07-30 01:43:24 +02:00
|
|
|
|
2003-02-18 13:38:27 +01:00
|
|
|
#ifdef HAVE_SYS_SOCKET_H
|
|
|
|
#include <sys/socket.h> /* for socket() */
|
2001-07-12 07:46:06 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
#include <unistd.h>
|
|
|
|
#endif
|
|
|
|
|
2005-04-04 14:54:34 +02:00
|
|
|
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef SUPERSERVER
|
|
|
|
#ifdef WIN_NT
|
|
|
|
#define FD_SETSIZE 1024
|
|
|
|
#endif
|
|
|
|
#endif /* SUPERSERVER */
|
|
|
|
|
|
|
|
#if !(defined VMS || defined WIN_NT)
|
2005-04-06 15:02:13 +02:00
|
|
|
#include <netinet/tcp.h>
|
2001-05-23 15:26:42 +02:00
|
|
|
#include <netdb.h>
|
2003-03-12 12:22:24 +01:00
|
|
|
#include <arpa/inet.h>
|
2003-03-12 14:55:45 +01:00
|
|
|
/* EKU: SINIX-Z does not define INADDR_NONE */
|
|
|
|
#ifndef INADDR_NONE
|
|
|
|
#define INADDR_NONE (unsigned long)-1
|
|
|
|
#endif
|
2004-05-17 12:22:34 +02:00
|
|
|
#endif // !(defined VMS || defined WIN_NT)
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-07-12 07:46:06 +02:00
|
|
|
#ifdef DARWIN
|
2004-03-07 08:58:55 +01:00
|
|
|
extern "C" int innetgr(const char*, const char*, const char*, const char*);
|
2001-07-12 07:46:06 +02:00
|
|
|
#endif
|
|
|
|
|
2004-05-17 12:22:34 +02:00
|
|
|
const int INET_RETRY_CALL = 5;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#include "../remote/remote.h"
|
2003-11-08 17:40:17 +01:00
|
|
|
#include "../jrd/ibase.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/iberr.h"
|
|
|
|
#include "../jrd/thd.h"
|
|
|
|
#include "../remote/inet_proto.h"
|
|
|
|
#include "../remote/proto_proto.h"
|
|
|
|
#include "../remote/remot_proto.h"
|
|
|
|
#include "../jrd/gds_proto.h"
|
|
|
|
#include "../jrd/isc_proto.h"
|
|
|
|
#ifndef REQUESTER
|
2003-09-08 22:23:46 +02:00
|
|
|
#include "../jrd/os/isc_i_proto.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#include "../jrd/sch_proto.h"
|
|
|
|
#endif /* REQUESTER */
|
|
|
|
|
2004-05-18 00:30:09 +02:00
|
|
|
#include "../jrd/thread_proto.h"
|
2002-12-06 13:34:43 +01:00
|
|
|
#include "../common/config/config.h"
|
2006-01-16 09:47:20 +01:00
|
|
|
#include "../common/utils_proto.h"
|
2004-12-26 14:48:01 +01:00
|
|
|
#include "../common/classes/ClumpletWriter.h"
|
2002-12-06 13:34:43 +01:00
|
|
|
|
2002-10-28 06:19:52 +01:00
|
|
|
#if (defined hpux || defined SCO_UNIX)
|
2001-05-23 15:26:42 +02:00
|
|
|
extern int h_errno;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef VMS
|
2004-04-29 00:36:29 +02:00
|
|
|
#include <perror.h>
|
2001-05-23 15:26:42 +02:00
|
|
|
#include <socket.h>
|
|
|
|
#define NO_FORK
|
2004-05-17 12:22:34 +02:00
|
|
|
const USHORT MAX_PTYPE = ptype_batch_send;
|
|
|
|
const char* PROXY_FILE = "[sysmgr]gds_proxy.dat";
|
|
|
|
const char* HOSTS_FILE = "";
|
|
|
|
const char* GDS_HOSTS_FILE = "";
|
2003-02-17 20:33:53 +01:00
|
|
|
#error "vms implementation must be completed"
|
2004-05-17 12:22:34 +02:00
|
|
|
#else // VMS
|
|
|
|
const char* PROXY_FILE = "/etc/gds_proxy";
|
|
|
|
const char* HOSTS_FILE = "/etc/hosts.equiv";
|
|
|
|
#ifdef MULTI_THREAD
|
|
|
|
const USHORT MAX_PTYPE = ptype_batch_send;
|
|
|
|
#else
|
|
|
|
const USHORT MAX_PTYPE = ptype_out_of_band;
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
2004-05-17 12:22:34 +02:00
|
|
|
const char* GDS_HOSTS_FILE = "/etc/gds_hosts.equiv";
|
|
|
|
#endif // VMS
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef WIN_NT
|
2004-05-19 00:01:53 +02:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
#include <fcntl.h>
|
|
|
|
#include <process.h>
|
|
|
|
#include <signal.h>
|
2003-07-15 04:43:36 +02:00
|
|
|
#include "../utilities/install/install_nt.h"
|
2001-05-23 15:26:42 +02:00
|
|
|
#define SOCLOSE closesocket
|
|
|
|
#define INET_RETRY_ERRNO WSAEINPROGRESS
|
|
|
|
#define INET_ADDR_IN_USE WSAEADDRINUSE
|
|
|
|
#define sleep(seconds) Sleep ((seconds) * 1000)
|
2003-02-10 06:42:22 +01:00
|
|
|
|
2004-05-19 00:01:53 +02:00
|
|
|
#else // WIN_NT
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifndef SOCKET
|
|
|
|
#define SOCKET int
|
|
|
|
#endif
|
|
|
|
#ifndef SOCLOSE
|
|
|
|
#define SOCLOSE close
|
|
|
|
#endif
|
|
|
|
#ifndef INET_ADDR_IN_USE
|
|
|
|
#define INET_ADDR_IN_USE EADDRINUSE
|
|
|
|
#endif
|
|
|
|
#ifndef INET_RETRY_ERRNO
|
|
|
|
#define INET_RETRY_ERRNO TRY_AGAIN
|
|
|
|
#endif
|
|
|
|
|
2004-05-19 00:01:53 +02:00
|
|
|
#endif // WIN_NT
|
|
|
|
|
|
|
|
#ifndef INVALID_SOCKET
|
|
|
|
#define INVALID_SOCKET -1
|
|
|
|
#endif
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifndef SIGURG
|
|
|
|
#define SIGURG SIGINT
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef ENOBUFS
|
|
|
|
#define ENOBUFS 0
|
|
|
|
#endif
|
|
|
|
|
2006-10-25 09:05:25 +02:00
|
|
|
#ifndef FB_SEND_FLAGS
|
|
|
|
#define FB_SEND_FLAGS 0
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef FB_SETOPT_FLAGS
|
|
|
|
#define FB_SETOPT_FLAGS 0
|
|
|
|
#endif
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
SLONG INET_remote_buffer;
|
|
|
|
SLONG INET_max_data;
|
2003-09-11 20:59:34 +02:00
|
|
|
static bool first_time = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-19 00:01:53 +02:00
|
|
|
//
|
|
|
|
//#define DEBUG 1
|
|
|
|
//
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifndef REQUESTER
|
|
|
|
#ifdef DEBUG
|
2002-09-18 14:50:13 +02:00
|
|
|
#ifdef HAVE_SYS_TIMEB_H
|
|
|
|
# include <sys/timeb.h>
|
|
|
|
#endif
|
2004-05-18 18:32:07 +02:00
|
|
|
const int TRACE_packets = 1 << 0; /* bit 0 */
|
|
|
|
const int TRACE_operations = 1 << 1; /* bit 1 */
|
|
|
|
const int TRACE_summary = 1 << 2; /* bit 2 */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-10-24 11:13:01 +02:00
|
|
|
static int INET_trace = TRACE_summary | TRACE_packets | TRACE_operations;
|
2001-05-23 15:26:42 +02:00
|
|
|
static time_t INET_start_time = 0;
|
|
|
|
SLONG INET_force_error = -1; /* simulate a network error */
|
|
|
|
static ULONG INET_count_send = 0;
|
|
|
|
static ULONG INET_count_recv = 0;
|
|
|
|
static ULONG INET_bytes_send = 0;
|
|
|
|
static ULONG INET_bytes_recv = 0;
|
|
|
|
|
|
|
|
static ULONG inet_debug_timer(void)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n e t _ d e b u g _ t i m e r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Utility function used in DEBUG mode only to put a timestamp
|
|
|
|
* since start of connection on each debug output message.
|
|
|
|
*
|
|
|
|
* This has been implemented and tested on SOLARIS, and may
|
|
|
|
* need tweeking on any other platform where DEBUG is needed.
|
|
|
|
*
|
|
|
|
**************************************/
|
2002-08-20 13:49:10 +02:00
|
|
|
#ifdef HAVE_GETTIMEOFDAY
|
2002-08-22 12:48:26 +02:00
|
|
|
struct timeval tv;
|
2003-10-31 12:33:45 +01:00
|
|
|
GETTIMEOFDAY(&tv);
|
|
|
|
return (tv.tv_sec * 1000 + tv.tv_usec - INET_start_time);
|
2002-08-20 13:49:10 +02:00
|
|
|
#else
|
2001-05-23 15:26:42 +02:00
|
|
|
struct timeb now;
|
2003-08-28 15:07:29 +02:00
|
|
|
ftime(&now);
|
2001-05-23 15:26:42 +02:00
|
|
|
return (now.time * 1000 + now.millitm - INET_start_time);
|
2002-08-22 12:48:26 +02:00
|
|
|
#endif /* HAVE_GETTIMEOFDAY */
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2002-08-22 12:48:26 +02:00
|
|
|
#endif /* DEBUG */
|
|
|
|
#endif /* !REQUESTER */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-17 12:22:34 +02:00
|
|
|
const SLONG MAX_DATA_LW = 1448; /* Low Water mark */
|
|
|
|
const SLONG MAX_DATA_HW = 32768; /* High Water mark */
|
|
|
|
const SLONG DEF_MAX_DATA = 8192;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-17 12:22:34 +02:00
|
|
|
const int MAX_SEQUENCE = 256;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-17 12:22:34 +02:00
|
|
|
const int MAXHOSTLEN = 64;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-17 12:22:34 +02:00
|
|
|
const int SELECT_TIMEOUT = 60; /* Dispatch thread select timeout (sec) */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-06-09 20:23:27 +02:00
|
|
|
typedef struct slct
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
int slct_width;
|
|
|
|
int slct_count;
|
|
|
|
SLONG slct_time;
|
|
|
|
fd_set slct_fdset;
|
2004-06-09 20:23:27 +02:00
|
|
|
} SLCT;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static int accept_connection(rem_port*, P_CNCT *);
|
2003-03-03 09:37:54 +01:00
|
|
|
#ifdef HAVE_SETITIMER
|
2001-05-23 15:26:42 +02:00
|
|
|
static void alarm_handler(int);
|
2003-03-03 09:37:54 +01:00
|
|
|
#endif
|
2004-01-28 08:50:41 +01:00
|
|
|
static rem_port* alloc_port(rem_port*);
|
2004-02-24 06:34:44 +01:00
|
|
|
static rem_port* aux_connect(rem_port*, PACKET*, t_event_ast);
|
|
|
|
static rem_port* aux_request(rem_port*, PACKET*);
|
2003-03-01 18:51:59 +01:00
|
|
|
#if !defined(WIN_NT)
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifndef VMS
|
2004-03-07 08:58:55 +01:00
|
|
|
static int check_host(rem_port*, TEXT*, const TEXT*, const struct passwd*);
|
2001-05-23 15:26:42 +02:00
|
|
|
#else
|
2004-03-07 08:58:55 +01:00
|
|
|
static int check_host(rem_port*, TEXT*, const TEXT*);
|
2003-11-07 09:06:35 +01:00
|
|
|
#endif // VMS
|
2005-01-29 10:22:36 +01:00
|
|
|
static bool check_proxy(rem_port*, TEXT*, Firebird::string&);
|
2003-11-07 09:06:35 +01:00
|
|
|
#endif // WIN_NT
|
2004-01-28 08:50:41 +01:00
|
|
|
static void cleanup_port(rem_port*);
|
|
|
|
static void disconnect(rem_port*);
|
2001-05-23 15:26:42 +02:00
|
|
|
static void exit_handler(void *);
|
|
|
|
|
|
|
|
#ifdef NO_FORK
|
|
|
|
static int fork(void);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
static int fork(SOCKET, USHORT);
|
|
|
|
#endif
|
|
|
|
|
2003-03-19 14:14:09 +01:00
|
|
|
static in_addr get_bind_address();
|
2006-10-25 09:05:25 +02:00
|
|
|
static in_addr get_host_address(const Firebird::string&);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
static void copy_p_cnct_repeat_array( p_cnct::p_cnct_repeat* pDest,
|
|
|
|
const p_cnct::p_cnct_repeat* pSource,
|
|
|
|
size_t nEntries);
|
|
|
|
|
2006-02-23 07:52:25 +01:00
|
|
|
static void inet_copy(const void*, UCHAR*, int);
|
2001-05-23 15:26:42 +02:00
|
|
|
static int inet_destroy(XDR *);
|
2004-01-28 08:50:41 +01:00
|
|
|
static void inet_gen_error(rem_port*, ISC_STATUS, ...);
|
2001-05-23 15:26:42 +02:00
|
|
|
static bool_t inet_getbytes(XDR *, SCHAR *, u_int);
|
|
|
|
static bool_t inet_getlong(XDR *, SLONG *);
|
|
|
|
static u_int inet_getpostn(XDR *);
|
2003-03-01 18:51:59 +01:00
|
|
|
#if !(defined WIN_NT)
|
2003-09-08 22:23:46 +02:00
|
|
|
static void inet_handler(void* _port);
|
2003-03-01 18:51:59 +01:00
|
|
|
#endif
|
2002-11-18 17:20:21 +01:00
|
|
|
static caddr_t inet_inline(XDR *, u_int);
|
2004-10-20 19:42:36 +02:00
|
|
|
static void inet_error(rem_port*, const TEXT*, ISC_STATUS, int);
|
2004-09-22 10:54:42 +02:00
|
|
|
static bool_t inet_putlong(XDR*, const SLONG*);
|
2003-10-29 11:53:47 +01:00
|
|
|
static bool_t inet_putbytes(XDR*, const SCHAR*, u_int);
|
2001-05-23 15:26:42 +02:00
|
|
|
static bool_t inet_read(XDR *);
|
|
|
|
static bool_t inet_setpostn(XDR *, u_int);
|
2004-01-28 08:50:41 +01:00
|
|
|
static rem_port* inet_try_connect( PACKET*,
|
2001-05-23 15:26:42 +02:00
|
|
|
RDB,
|
2004-11-15 16:52:57 +01:00
|
|
|
Firebird::PathName&,
|
2003-11-07 09:06:35 +01:00
|
|
|
const TEXT*,
|
2003-04-10 12:31:28 +02:00
|
|
|
ISC_STATUS*,
|
2003-10-29 11:53:47 +01:00
|
|
|
const SCHAR*,
|
2001-05-23 15:26:42 +02:00
|
|
|
SSHORT);
|
|
|
|
static bool_t inet_write(XDR *, int);
|
2006-02-23 07:52:25 +01:00
|
|
|
static void inet_zero(sockaddr_in*, int);
|
2003-03-01 18:51:59 +01:00
|
|
|
#if !(defined WIN_NT)
|
2004-03-07 08:58:55 +01:00
|
|
|
static int parse_hosts(const TEXT*, const TEXT*, const TEXT*);
|
|
|
|
static int parse_line(const TEXT*, const TEXT*, const TEXT*, const TEXT*);
|
2003-03-01 18:51:59 +01:00
|
|
|
#endif
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef DEBUG
|
2006-03-07 06:44:13 +01:00
|
|
|
static void packet_print(const TEXT*, const UCHAR*, int, ULONG);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static bool_t packet_receive(rem_port*, UCHAR*, SSHORT, SSHORT*);
|
|
|
|
static bool_t packet_send(rem_port*, const SCHAR*, SSHORT);
|
|
|
|
static rem_port* receive(rem_port*, PACKET *);
|
|
|
|
static rem_port* select_accept(rem_port*);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static rem_port* select_port(rem_port*, SLCT *);
|
2006-03-03 17:20:42 +01:00
|
|
|
static rem_port* select_multi(rem_port*, UCHAR* buffer, SSHORT bufsize, SSHORT* length);
|
2004-01-28 08:50:41 +01:00
|
|
|
static int select_wait(rem_port*, SLCT *);
|
|
|
|
static int send_full(rem_port*, PACKET *);
|
|
|
|
static int send_partial(rem_port*, PACKET *);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef SUPERSERVER
|
2004-01-28 08:50:41 +01:00
|
|
|
static void unhook_disconnected_ports(rem_port*);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static void unhook_port(rem_port*, rem_port*);
|
|
|
|
static int xdrinet_create(XDR *, rem_port*, UCHAR *, USHORT, enum xdr_op);
|
2006-05-24 16:08:06 +02:00
|
|
|
static bool setNoNagleOption(rem_port*);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
|
|
|
|
static XDR::xdr_ops inet_ops =
|
|
|
|
{
|
|
|
|
inet_getlong,
|
|
|
|
inet_putlong,
|
2006-05-02 00:23:31 +02:00
|
|
|
#if defined(SUPERSERVER) && !defined(EMBEDDED)
|
2006-03-03 17:20:42 +01:00
|
|
|
REMOTE_getbytes,
|
|
|
|
#else
|
2002-11-18 17:20:21 +01:00
|
|
|
inet_getbytes,
|
2006-03-03 17:20:42 +01:00
|
|
|
#endif
|
2002-11-18 17:20:21 +01:00
|
|
|
inet_putbytes,
|
2001-05-23 15:26:42 +02:00
|
|
|
inet_getpostn,
|
|
|
|
inet_setpostn,
|
|
|
|
inet_inline,
|
|
|
|
inet_destroy
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
#define MAXCLIENTS NOFILE - 10
|
|
|
|
|
|
|
|
/* Select uses bit masks of file descriptors in longs. */
|
|
|
|
|
|
|
|
#ifndef NBBY
|
|
|
|
#define NBBY 8
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef NFDBITS
|
|
|
|
#if !defined(WIN_NT)
|
|
|
|
#define NFDBITS (sizeof(SLONG) * NBBY)
|
|
|
|
|
2004-12-25 10:44:03 +01:00
|
|
|
#define FD_SET(n, p) ((p)->fds_bits[(n) / NFDBITS] |= (1 << ((n) % NFDBITS)))
|
|
|
|
#define FD_CLR(n, p) ((p)->fds_bits[(n) / NFDBITS] &= ~(1 << ((n) % NFDBITS)))
|
|
|
|
#define FD_ISSET(n, p) ((p)->fds_bits[(n) / NFDBITS] & (1 << ((n) % NFDBITS)))
|
2006-02-23 07:52:25 +01:00
|
|
|
#define FD_ZERO(p) inet_zero(p, sizeof(*(p)))
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
static SLCT INET_select = { 0, 0, 0 };
|
|
|
|
static int INET_max_clients;
|
|
|
|
#ifdef WIN_NT
|
2004-02-24 06:34:44 +01:00
|
|
|
static bool INET_initialized = false;
|
2001-05-23 15:26:42 +02:00
|
|
|
static WSADATA INET_wsadata;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
#define INTERRUPT_ERROR(x) (SYSCALL_INTERRUPTED(x) || (x) == WSAEINTR)
|
|
|
|
#else
|
|
|
|
#define INTERRUPT_ERROR(x) (SYSCALL_INTERRUPTED(x))
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER
|
|
|
|
|
|
|
|
static MUTX_T port_mutex;
|
2003-09-11 20:59:34 +02:00
|
|
|
static bool port_mutex_inited = false;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#define DEFER_PORT_CLEANUP
|
|
|
|
|
2004-05-21 08:16:17 +02:00
|
|
|
inline void START_PORT_CRITICAL() {
|
2004-05-17 12:22:34 +02:00
|
|
|
if (!port_mutex_inited) {
|
|
|
|
port_mutex_inited = true;
|
|
|
|
}
|
|
|
|
THREAD_EXIT();
|
|
|
|
THD_mutex_lock (&port_mutex);
|
|
|
|
THREAD_ENTER();
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-21 08:16:17 +02:00
|
|
|
inline void STOP_PORT_CRITICAL() {
|
2004-05-17 12:22:34 +02:00
|
|
|
THREAD_EXIT();
|
|
|
|
THD_mutex_unlock (&port_mutex);
|
|
|
|
THREAD_ENTER();
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-17 12:22:34 +02:00
|
|
|
#else
|
2004-05-21 08:16:17 +02:00
|
|
|
inline void START_PORT_CRITICAL() {
|
2004-05-17 12:22:34 +02:00
|
|
|
}
|
2004-05-21 08:16:17 +02:00
|
|
|
inline void STOP_PORT_CRITICAL() {
|
2004-05-17 12:22:34 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2004-11-15 16:52:57 +01:00
|
|
|
rem_port* INET_analyze(Firebird::PathName& file_name,
|
2003-04-10 12:31:28 +02:00
|
|
|
ISC_STATUS* status_vector,
|
2003-11-07 09:06:35 +01:00
|
|
|
const TEXT* node_name,
|
|
|
|
const TEXT* user_string,
|
2003-12-03 09:19:24 +01:00
|
|
|
bool uv_flag,
|
2003-10-29 11:53:47 +01:00
|
|
|
const SCHAR* dpb,
|
2001-05-23 15:26:42 +02:00
|
|
|
SSHORT dpb_length)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I N E T _ a n a l y z e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* File_name is on node_name.
|
|
|
|
* Establish an external connection to node_name.
|
|
|
|
*
|
|
|
|
* If a connection is established, return a port block, otherwise
|
|
|
|
* return NULL.
|
|
|
|
*
|
|
|
|
* If the "uv_flag" is non-zero, user verification also takes place.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
/* We need to establish a connection to a remote server. Allocate the necessary
|
|
|
|
blocks and get ready to go. */
|
|
|
|
|
2004-09-22 22:33:07 +02:00
|
|
|
RDB rdb = (RDB) ALLR_block(type_rdb, 0);
|
|
|
|
PACKET* packet = &rdb->rdb_packet;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Pick up some user identification information */
|
2005-11-27 21:53:09 +01:00
|
|
|
Firebird::ClumpletWriter user_id(Firebird::ClumpletReader::UnTagged, MAX_DPB_SIZE);
|
2004-12-26 14:48:01 +01:00
|
|
|
char buffer[BUFFER_SMALL];
|
|
|
|
|
2004-09-22 22:33:07 +02:00
|
|
|
int eff_gid;
|
|
|
|
int eff_uid;
|
2004-12-26 14:48:01 +01:00
|
|
|
ISC_get_user(buffer, &eff_uid, &eff_gid, 0, 0, 0, user_string);
|
|
|
|
user_id.insertString(CNCT_user, buffer, strlen(buffer));
|
|
|
|
|
|
|
|
ISC_get_host(buffer, sizeof(buffer));
|
|
|
|
for (char* p = buffer; *p; p++) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (*p >= 'A' && *p <= 'Z') {
|
|
|
|
*p = *p - 'A' + 'a';
|
|
|
|
}
|
|
|
|
}
|
2004-12-26 14:48:01 +01:00
|
|
|
user_id.insertString(CNCT_host, buffer, strlen(buffer));
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if ((eff_uid == -1) || uv_flag) {
|
2004-12-26 14:48:01 +01:00
|
|
|
user_id.insertTag(CNCT_user_verification);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2002-10-31 06:06:02 +01:00
|
|
|
#if !(defined VMS)
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Communicate group id info to server, as user maybe running under group
|
|
|
|
id other than default specified in /etc/passwd. */
|
|
|
|
|
|
|
|
eff_gid = htonl(eff_gid);
|
2004-12-26 14:48:01 +01:00
|
|
|
user_id.insertBytes(CNCT_group,
|
|
|
|
reinterpret_cast<UCHAR*>(AOF32L(eff_gid)), sizeof(SLONG));
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Establish connection to server */
|
|
|
|
|
|
|
|
/* Note: prior to V3.1E a recievers could not in truth handle more
|
|
|
|
then 5 protocol descriptions, so we try them in chunks of 5 or less */
|
|
|
|
|
|
|
|
/* If we want user verification, we can't speak anything less than version 7 */
|
|
|
|
|
2003-11-07 09:06:35 +01:00
|
|
|
P_CNCT* cnct = &packet->p_cnct;
|
2003-04-09 15:20:56 +02:00
|
|
|
|
2004-12-26 14:48:01 +01:00
|
|
|
cnct->p_cnct_user_id.cstr_length = user_id.getBufferLength();
|
|
|
|
cnct->p_cnct_user_id.cstr_address = const_cast<UCHAR*>(user_id.getBuffer());
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
static const p_cnct::p_cnct_repeat protocols_to_try1[] =
|
|
|
|
{
|
2003-04-09 15:20:56 +02:00
|
|
|
REMOTE_PROTOCOL(PROTOCOL_VERSION8, ptype_rpc, MAX_PTYPE, 1),
|
2006-10-25 09:05:25 +02:00
|
|
|
REMOTE_PROTOCOL(PROTOCOL_VERSION10, ptype_rpc, MAX_PTYPE, 2),
|
2006-12-08 19:38:15 +01:00
|
|
|
REMOTE_PROTOCOL(PROTOCOL_VERSION10, ptype_rpc, ptype_lazy_send, 3),
|
|
|
|
REMOTE_PROTOCOL(PROTOCOL_VERSION11, ptype_rpc, MAX_PTYPE, 4),
|
|
|
|
REMOTE_PROTOCOL(PROTOCOL_VERSION11, ptype_rpc, ptype_lazy_send, 5)
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef SCROLLABLE_CURSORS
|
|
|
|
,
|
2006-12-08 19:38:15 +01:00
|
|
|
REMOTE_PROTOCOL(PROTOCOL_SCROLLABLE_CURSORS, ptype_rpc, MAX_PTYPE, 99)
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
};
|
|
|
|
|
2002-12-02 10:45:52 +01:00
|
|
|
cnct->p_cnct_count = FB_NELEM(protocols_to_try1);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-04-09 15:20:56 +02:00
|
|
|
copy_p_cnct_repeat_array(cnct->p_cnct_versions,
|
|
|
|
protocols_to_try1,
|
|
|
|
cnct->p_cnct_count);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Try connection using first set of protocols. punt if error */
|
|
|
|
|
2004-11-07 15:44:59 +01:00
|
|
|
rem_port* port = inet_try_connect(packet, rdb, file_name,
|
2003-04-09 15:20:56 +02:00
|
|
|
node_name, status_vector, dpb, dpb_length);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!port) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (packet->p_operation == op_reject && !uv_flag)
|
|
|
|
{
|
|
|
|
disconnect(port);
|
|
|
|
|
|
|
|
/* try again with next set of known protocols */
|
|
|
|
|
2004-12-26 14:48:01 +01:00
|
|
|
cnct->p_cnct_user_id.cstr_length = user_id.getBufferLength();
|
|
|
|
cnct->p_cnct_user_id.cstr_address = const_cast<UCHAR*>(user_id.getBuffer());
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
static const p_cnct::p_cnct_repeat protocols_to_try2[] =
|
|
|
|
{
|
2003-04-09 15:20:56 +02:00
|
|
|
REMOTE_PROTOCOL(PROTOCOL_VERSION6, ptype_rpc, ptype_batch_send, 1),
|
|
|
|
REMOTE_PROTOCOL(PROTOCOL_VERSION7, ptype_rpc, MAX_PTYPE, 2)
|
2001-05-23 15:26:42 +02:00
|
|
|
};
|
|
|
|
|
2002-12-02 10:45:52 +01:00
|
|
|
cnct->p_cnct_count = FB_NELEM(protocols_to_try2);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-04-09 15:20:56 +02:00
|
|
|
copy_p_cnct_repeat_array(cnct->p_cnct_versions,
|
|
|
|
protocols_to_try2,
|
|
|
|
cnct->p_cnct_count);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-11-07 15:44:59 +01:00
|
|
|
port = inet_try_connect(packet, rdb, file_name,
|
2001-05-23 15:26:42 +02:00
|
|
|
node_name, status_vector, dpb, dpb_length);
|
2005-12-23 09:52:31 +01:00
|
|
|
if (!port) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return NULL;
|
2005-12-23 09:52:31 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (packet->p_operation == op_reject && !uv_flag)
|
|
|
|
{
|
|
|
|
disconnect(port);
|
|
|
|
|
|
|
|
/* try again with next set of known protocols */
|
|
|
|
|
2004-12-26 14:48:01 +01:00
|
|
|
cnct->p_cnct_user_id.cstr_length = user_id.getBufferLength();
|
|
|
|
cnct->p_cnct_user_id.cstr_address = const_cast<UCHAR*>(user_id.getBuffer());
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
static const p_cnct::p_cnct_repeat protocols_to_try3[] =
|
|
|
|
{
|
2003-04-09 15:20:56 +02:00
|
|
|
REMOTE_PROTOCOL(PROTOCOL_VERSION3, ptype_rpc, ptype_batch_send, 1),
|
|
|
|
REMOTE_PROTOCOL(PROTOCOL_VERSION4, ptype_rpc, ptype_batch_send, 2)
|
2001-05-23 15:26:42 +02:00
|
|
|
};
|
|
|
|
|
2002-12-02 10:45:52 +01:00
|
|
|
cnct->p_cnct_count = FB_NELEM(protocols_to_try3);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-04-09 15:20:56 +02:00
|
|
|
copy_p_cnct_repeat_array(cnct->p_cnct_versions,
|
|
|
|
protocols_to_try3,
|
|
|
|
cnct->p_cnct_count);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-11-07 15:44:59 +01:00
|
|
|
port = inet_try_connect(packet, rdb, file_name,
|
2001-05-23 15:26:42 +02:00
|
|
|
node_name, status_vector, dpb, dpb_length);
|
|
|
|
if (!port) {
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (packet->p_operation != op_accept)
|
|
|
|
{
|
2003-11-08 17:40:17 +01:00
|
|
|
*status_vector++ = isc_arg_gds;
|
|
|
|
*status_vector++ = isc_connect_reject;
|
2001-05-23 15:26:42 +02:00
|
|
|
*status_vector++ = 0;
|
|
|
|
disconnect(port);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
port->port_protocol = packet->p_acpt.p_acpt_version;
|
|
|
|
|
2002-02-16 03:21:35 +01:00
|
|
|
/* once we've decided on a protocol, concatenate the version
|
2001-05-23 15:26:42 +02:00
|
|
|
string to reflect it... */
|
2004-09-22 22:33:07 +02:00
|
|
|
Firebird::string temp;
|
2006-12-08 19:38:15 +01:00
|
|
|
temp.printf("%s/P%d", port->port_version->str_data,
|
|
|
|
port->port_protocol & FB_PROTOCOL_MASK);
|
2001-05-23 15:26:42 +02:00
|
|
|
ALLR_free(port->port_version);
|
2004-09-22 22:33:07 +02:00
|
|
|
port->port_version = REMOTE_make_string(temp.c_str());
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (packet->p_acpt.p_acpt_architecture == ARCHITECTURE) {
|
|
|
|
port->port_flags |= PORT_symmetric;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (packet->p_acpt.p_acpt_type == ptype_rpc) {
|
|
|
|
port->port_flags |= PORT_rpc;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (packet->p_acpt.p_acpt_type != ptype_out_of_band) {
|
|
|
|
port->port_flags |= PORT_no_oob;
|
|
|
|
}
|
|
|
|
|
2006-10-25 09:05:25 +02:00
|
|
|
if (packet->p_acpt.p_acpt_type == ptype_lazy_send) {
|
|
|
|
port->port_flags |= PORT_lazy;
|
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
return port;
|
|
|
|
}
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* INET_connect(const TEXT* name,
|
2003-10-29 11:53:47 +01:00
|
|
|
PACKET* packet,
|
|
|
|
ISC_STATUS* status_vector,
|
|
|
|
USHORT flag, const SCHAR* dpb, SSHORT dpb_length)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I N E T _ c o n n e c t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Establish half of a communication link. If a connect packet is given,
|
|
|
|
* the connection is on behalf of a remote interface. Otherwise the connect
|
|
|
|
* is for a server process.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
#ifdef DEBUG
|
|
|
|
{
|
|
|
|
if (INET_trace & TRACE_operations) {
|
2004-04-29 00:36:29 +02:00
|
|
|
fprintf(stdout, "INET_connect\n");
|
|
|
|
fflush(stdout);
|
2004-11-07 11:38:13 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
INET_start_time = inet_debug_timer();
|
2006-01-14 05:48:59 +01:00
|
|
|
// CVC: I don't see the point in replacing this with fb_utils::readenv().
|
2003-11-07 09:06:35 +01:00
|
|
|
const char* p = getenv("INET_force_error");
|
|
|
|
if (p != NULL) {
|
|
|
|
INET_force_error = atoi(p);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* port = alloc_port(0);
|
2001-05-23 15:26:42 +02:00
|
|
|
port->port_status_vector = status_vector;
|
2003-10-29 11:53:47 +01:00
|
|
|
REMOTE_get_timeout_params(port, reinterpret_cast<const UCHAR*>(dpb),
|
2001-05-23 15:26:42 +02:00
|
|
|
dpb_length);
|
2003-11-08 17:40:17 +01:00
|
|
|
status_vector[0] = isc_arg_gds;
|
2001-05-23 15:26:42 +02:00
|
|
|
status_vector[1] = 0;
|
2003-11-08 17:40:17 +01:00
|
|
|
status_vector[2] = isc_arg_end;
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef VMS
|
|
|
|
ISC_tcp_setup(ISC_wait, gds__completion_ast);
|
|
|
|
#endif
|
|
|
|
|
2006-10-25 09:05:25 +02:00
|
|
|
Firebird::string host;
|
|
|
|
Firebird::string protocol;
|
2003-12-06 16:52:45 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
if (name) {
|
2006-10-25 09:05:25 +02:00
|
|
|
host = name;
|
|
|
|
const size_t pos = host.find("/");
|
|
|
|
if (pos != Firebird::string::npos) {
|
|
|
|
protocol = host.substr(pos + 1);
|
|
|
|
host = host.substr(0, pos);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-10-25 09:05:25 +02:00
|
|
|
if (host.hasData()) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (port->port_connection) {
|
|
|
|
ALLR_free(port->port_connection);
|
|
|
|
}
|
2006-10-25 09:05:25 +02:00
|
|
|
port->port_connection = REMOTE_make_string(host.c_str());
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
2006-10-25 09:05:25 +02:00
|
|
|
host = port->port_host->str_data;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2006-10-25 09:05:25 +02:00
|
|
|
if (protocol.isEmpty()) {
|
2004-05-12 21:39:17 +02:00
|
|
|
const unsigned short port2 = Config::getRemoteServicePort();
|
|
|
|
if (port2) {
|
2006-10-25 09:05:25 +02:00
|
|
|
protocol.printf("%hu", port2);
|
2003-12-11 09:26:46 +01:00
|
|
|
}
|
|
|
|
else {
|
2005-01-16 12:25:18 +01:00
|
|
|
protocol = Config::getRemoteServiceName();
|
2003-12-11 09:26:46 +01:00
|
|
|
}
|
2003-12-06 16:52:45 +01:00
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Set up Inter-Net socket address */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
struct sockaddr_in address;
|
2006-02-23 07:52:25 +01:00
|
|
|
inet_zero(&address, sizeof(address));
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef VMS
|
|
|
|
/* V M S */
|
2006-10-25 09:05:25 +02:00
|
|
|
if (getservport(protocol.c_str(), "tcp", &address.sin_port) == -1) {
|
2001-05-23 15:26:42 +02:00
|
|
|
inet_error(port, "getservbyname", isc_net_connect_err, 0);
|
|
|
|
disconnect(port);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
if (packet) {
|
2006-10-25 09:05:25 +02:00
|
|
|
if (getaddr(host.c_str(), &address) == -1) {
|
2001-05-23 15:26:42 +02:00
|
|
|
inet_error(port, "gethostbyname", isc_net_connect_err, 0);
|
|
|
|
disconnect(port);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
address.sin_addr.s_addr = INADDR_ANY;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
|
2004-09-05 21:37:43 +02:00
|
|
|
TEXT msg[BUFFER_SMALL];
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* U N I X style sockets */
|
2003-03-12 11:13:58 +01:00
|
|
|
|
2003-03-19 14:14:09 +01:00
|
|
|
address.sin_family = AF_INET;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-03-19 14:14:09 +01:00
|
|
|
in_addr host_addr;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-03-19 14:14:09 +01:00
|
|
|
if (packet) {
|
|
|
|
// client connection
|
2006-10-25 09:05:25 +02:00
|
|
|
host_addr = get_host_address(host);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-09-22 22:33:07 +02:00
|
|
|
if (host_addr.s_addr == INADDR_NONE)
|
|
|
|
{
|
|
|
|
SNPRINTF(msg, FB_NELEM(msg),
|
2004-09-05 21:37:43 +02:00
|
|
|
"INET/INET_connect: gethostbyname (%s) failed, error code = %d",
|
2006-10-25 09:05:25 +02:00
|
|
|
host.c_str(), H_ERRNO);
|
2003-03-19 14:14:09 +01:00
|
|
|
gds__log(msg, 0);
|
|
|
|
inet_gen_error(port,
|
|
|
|
isc_network_error,
|
|
|
|
isc_arg_string,
|
|
|
|
port->port_connection->str_data,
|
|
|
|
isc_arg_gds,
|
|
|
|
isc_net_lookup_err, isc_arg_gds, isc_host_unknown, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-03-19 14:14:09 +01:00
|
|
|
disconnect(port);
|
|
|
|
return NULL;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else {
|
2003-03-19 14:14:09 +01:00
|
|
|
// server connection
|
|
|
|
host_addr = get_bind_address();
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2006-02-23 07:52:25 +01:00
|
|
|
inet_copy(&host_addr,
|
|
|
|
(UCHAR*) &address.sin_addr,
|
2003-10-29 11:53:47 +01:00
|
|
|
sizeof(address.sin_addr));
|
2003-03-19 14:14:09 +01:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2006-10-25 09:05:25 +02:00
|
|
|
const struct servent* service = getservbyname(protocol.c_str(), "tcp");
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef WIN_NT
|
|
|
|
/* On Windows NT/9x, getservbyname can only accomodate
|
|
|
|
* 1 call at a time. In this case it returns the error
|
|
|
|
* WSAEINPROGRESS.
|
|
|
|
* If this happens, retry the operation a few times.
|
|
|
|
* NOTE: This still does not guarantee success, but helps.
|
|
|
|
*/
|
|
|
|
if (!service) {
|
|
|
|
if (H_ERRNO == INET_RETRY_ERRNO) {
|
2003-03-12 11:13:58 +01:00
|
|
|
for (int retry = 0; retry < INET_RETRY_CALL; retry++) {
|
2006-10-25 09:05:25 +02:00
|
|
|
if ( (service = getservbyname(protocol.c_str(), "tcp")) )
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif /* WIN_NT */
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Modification by luz (slightly modified by FSG)
|
|
|
|
instead of failing here, try applying hard-wired
|
|
|
|
translation of "gds_db" into "3050"
|
2003-03-12 11:13:58 +01:00
|
|
|
This way, a connection to a remote FB server
|
2001-05-23 15:26:42 +02:00
|
|
|
works even from clients with missing "gds_db"
|
|
|
|
entry in "services" file, which is important
|
|
|
|
for zero-installation clients.
|
|
|
|
*/
|
2003-12-06 16:52:45 +01:00
|
|
|
if (!service) {
|
2006-10-25 09:05:25 +02:00
|
|
|
if (protocol == FB_SERVICE_NAME) {
|
2001-05-23 15:26:42 +02:00
|
|
|
/* apply hardwired translation */
|
2003-06-25 09:39:04 +02:00
|
|
|
address.sin_port = htons(FB_SERVICE_PORT);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
/* modification by FSG 23.MAR.2001 */
|
|
|
|
else {
|
|
|
|
/* modification by FSG 23.MAR.2001 */
|
|
|
|
/* The user has supplied something as protocol
|
|
|
|
* let's see whether this is a port number
|
|
|
|
* instead of a service name
|
|
|
|
*/
|
2006-10-25 09:05:25 +02:00
|
|
|
address.sin_port = htons(atoi(protocol.c_str()));
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-09-22 22:33:07 +02:00
|
|
|
if (address.sin_port == 0)
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
/* end of modification by FSG */
|
|
|
|
/* this is the original code */
|
2004-09-22 22:33:07 +02:00
|
|
|
SNPRINTF(msg, FB_NELEM(msg),
|
2001-05-23 15:26:42 +02:00
|
|
|
"INET/INET_connect: getservbyname failed, error code = %d",
|
|
|
|
H_ERRNO);
|
|
|
|
gds__log(msg, 0);
|
|
|
|
inet_gen_error(port,
|
|
|
|
isc_network_error,
|
|
|
|
isc_arg_string,
|
|
|
|
port->port_connection->str_data,
|
|
|
|
isc_arg_gds,
|
|
|
|
isc_net_lookup_err,
|
|
|
|
isc_arg_gds,
|
|
|
|
isc_service_unknown,
|
|
|
|
isc_arg_string,
|
2006-10-25 09:05:25 +02:00
|
|
|
protocol.c_str(),
|
|
|
|
isc_arg_string, "tcp", 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
return NULL;
|
|
|
|
} /* else / not hardwired gds_db translation */
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* if we have got a service-struct, get port number from there
|
|
|
|
* (in case of hardwired gds_db to 3050 translation, address.sin_port was
|
|
|
|
* already set above */
|
|
|
|
address.sin_port = service->s_port;
|
|
|
|
} /* else (service found) */
|
|
|
|
|
|
|
|
/* end of modifications by luz */
|
|
|
|
#endif /* VMS */
|
|
|
|
|
|
|
|
/* Allocate a port block and initialize a socket for communications */
|
|
|
|
|
|
|
|
port->port_handle = (HANDLE) socket(AF_INET, SOCK_STREAM, 0);
|
|
|
|
|
|
|
|
if ((SOCKET) port->port_handle == INVALID_SOCKET)
|
|
|
|
{
|
2004-05-05 23:55:13 +02:00
|
|
|
inet_error(port, "socket", isc_net_connect_err, INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
disconnect(port);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If we're a host, just make the connection */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
int n;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
if (packet) {
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
n = connect((SOCKET) port->port_handle,
|
|
|
|
(struct sockaddr *) &address, sizeof(address));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
if (n != -1 && send_full(port, packet))
|
|
|
|
return port;
|
|
|
|
else {
|
2004-05-05 23:55:13 +02:00
|
|
|
inet_error(port, "connect", isc_net_connect_err, INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
disconnect(port);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We're a server, so wait for a host to show up */
|
|
|
|
|
|
|
|
if (flag & SRVR_multi_client) {
|
|
|
|
struct linger lingerInfo;
|
|
|
|
|
|
|
|
lingerInfo.l_onoff = 0;
|
|
|
|
lingerInfo.l_linger = 0;
|
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
int optval = TRUE;
|
2001-05-23 15:26:42 +02:00
|
|
|
n = setsockopt((SOCKET) port->port_handle, SOL_SOCKET, SO_REUSEADDR,
|
2003-12-03 09:19:24 +01:00
|
|
|
(SCHAR*) &optval, sizeof(optval));
|
2001-05-23 15:26:42 +02:00
|
|
|
if (n == -1) {
|
|
|
|
inet_error(port, "setsockopt REUSE", isc_net_connect_listen_err,
|
2004-05-05 23:55:13 +02:00
|
|
|
INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
disconnect(port);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get any values for SO_LINGER so that they can be reset during
|
|
|
|
* disconnect. SO_LINGER should be set by default on the socket
|
|
|
|
*/
|
2006-03-07 06:44:13 +01:00
|
|
|
socklen_t optlen = sizeof(port->port_linger);
|
2001-05-23 15:26:42 +02:00
|
|
|
n = getsockopt((SOCKET) port->port_handle, SOL_SOCKET, SO_LINGER,
|
|
|
|
(SCHAR *) & port->port_linger, &optlen);
|
|
|
|
|
|
|
|
if (n != 0) /* getsockopt failed */
|
|
|
|
port->port_linger.l_onoff = 0;
|
|
|
|
|
|
|
|
n = setsockopt((SOCKET) port->port_handle, SOL_SOCKET, SO_LINGER,
|
|
|
|
(SCHAR *) & lingerInfo, sizeof(lingerInfo));
|
|
|
|
if (n == -1) {
|
|
|
|
inet_error(port, "setsockopt LINGER", isc_net_connect_listen_err,
|
2004-05-05 23:55:13 +02:00
|
|
|
INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
disconnect(port);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2006-05-24 16:08:06 +02:00
|
|
|
if (! setNoNagleOption(port)) {
|
|
|
|
inet_error(port, "setsockopt TCP_NODELAY",
|
|
|
|
isc_net_connect_listen_err, INET_ERRNO);
|
|
|
|
disconnect(port);
|
|
|
|
return NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
n = bind((SOCKET) port->port_handle,
|
|
|
|
(struct sockaddr *) &address, sizeof(address));
|
|
|
|
|
|
|
|
if (n == -1) {
|
|
|
|
/* On Linux platform, when the server dies the system holds a port
|
|
|
|
for some time. */
|
|
|
|
|
2004-05-05 23:55:13 +02:00
|
|
|
if (INET_ERRNO == INET_ADDR_IN_USE) {
|
2004-02-24 06:34:44 +01:00
|
|
|
for (int retry = 0; retry < INET_RETRY_CALL; retry++) {
|
2001-05-23 15:26:42 +02:00
|
|
|
sleep(10);
|
|
|
|
n = bind((SOCKET) port->port_handle,
|
|
|
|
(struct sockaddr *) &address, sizeof(address));
|
|
|
|
if (n == 0)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
2004-05-05 23:55:13 +02:00
|
|
|
inet_error(port, "bind", isc_net_connect_listen_err, INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
disconnect(port);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
n = listen((SOCKET) port->port_handle, 5);
|
|
|
|
|
|
|
|
if (n == -1) {
|
2004-05-05 23:55:13 +02:00
|
|
|
inet_error(port, "listen", isc_net_connect_listen_err, INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flag & SRVR_multi_client) {
|
|
|
|
/* Prevent the generation of dummy keepalive packets on the
|
|
|
|
connect port. */
|
|
|
|
|
|
|
|
port->port_dummy_packet_interval = 0;
|
|
|
|
port->port_dummy_timeout = 0;
|
|
|
|
port->port_server_flags |= (SRVR_server | SRVR_multi_client);
|
|
|
|
gds__register_cleanup(exit_handler, (void *) port);
|
|
|
|
return port;
|
|
|
|
}
|
|
|
|
|
2003-03-12 11:13:58 +01:00
|
|
|
while (true) {
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2004-02-24 06:34:44 +01:00
|
|
|
socklen_t l = sizeof(address);
|
|
|
|
SOCKET s = accept((SOCKET) port->port_handle,
|
2001-05-23 15:26:42 +02:00
|
|
|
(struct sockaddr *) &address, &l);
|
|
|
|
if (s == INVALID_SOCKET) {
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2004-05-05 23:55:13 +02:00
|
|
|
inet_error(port, "accept", isc_net_connect_err, INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
disconnect(port);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
#ifdef WIN_NT
|
|
|
|
if ((flag & SRVR_debug) || !fork(s, flag))
|
|
|
|
#else
|
|
|
|
if ((flag & SRVR_debug) || !fork())
|
|
|
|
#endif
|
|
|
|
{
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
SOCLOSE((SOCKET) port->port_handle);
|
|
|
|
port->port_handle = (HANDLE) s;
|
|
|
|
port->port_server_flags |= SRVR_server;
|
|
|
|
return port;
|
|
|
|
}
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
SOCLOSE(s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* INET_reconnect(HANDLE handle, ISC_STATUS* status_vector)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I N E T _ r e c o n n e c t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* A communications link has been established by another
|
|
|
|
* process. We have inheritted the handle. Set up
|
|
|
|
* a port block.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* port = alloc_port(0);
|
2001-05-23 15:26:42 +02:00
|
|
|
port->port_status_vector = status_vector;
|
2003-11-08 17:40:17 +01:00
|
|
|
status_vector[0] = isc_arg_gds;
|
2001-05-23 15:26:42 +02:00
|
|
|
status_vector[1] = 0;
|
2003-11-08 17:40:17 +01:00
|
|
|
status_vector[2] = isc_arg_end;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
port->port_handle = handle;
|
|
|
|
port->port_server_flags |= SRVR_server;
|
|
|
|
|
2006-04-05 07:04:52 +02:00
|
|
|
int n = 0, optval = TRUE;
|
|
|
|
n = setsockopt((SOCKET) port->port_handle, SOL_SOCKET,
|
|
|
|
SO_KEEPALIVE, (SCHAR*) &optval, sizeof(optval));
|
|
|
|
if (n == -1) {
|
|
|
|
gds__log("inet server err: setting KEEPALIVE socket option \n");
|
|
|
|
}
|
|
|
|
|
2006-05-24 16:08:06 +02:00
|
|
|
if (! setNoNagleOption(port)) {
|
|
|
|
gds__log("inet server err: setting NODELAY socket option \n");
|
2006-04-05 07:04:52 +02:00
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
return port;
|
|
|
|
}
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* INET_server(int sock)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I N E T _ s e r v e r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* We have been spawned by a master server with a connection
|
|
|
|
* established. Set up port block with the appropriate socket.
|
|
|
|
*
|
|
|
|
**************************************/
|
2005-04-04 14:54:34 +02:00
|
|
|
int n = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef VMS
|
|
|
|
ISC_tcp_setup(ISC_wait, gds__completion_ast);
|
|
|
|
#endif
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* port = alloc_port(0);
|
2001-05-23 15:26:42 +02:00
|
|
|
port->port_server_flags |= SRVR_server;
|
|
|
|
port->port_handle = (HANDLE) sock;
|
|
|
|
|
2003-11-15 21:16:29 +01:00
|
|
|
int optval = 1;
|
2005-04-04 14:54:34 +02:00
|
|
|
n = setsockopt(sock, SOL_SOCKET, SO_KEEPALIVE,
|
2003-11-15 21:16:29 +01:00
|
|
|
(SCHAR *) & optval, sizeof(optval));
|
2005-04-05 08:47:17 +02:00
|
|
|
if (n == -1) {
|
2005-04-04 14:54:34 +02:00
|
|
|
gds__log("inet server err: setting KEEPALIVE socket option \n");
|
|
|
|
}
|
|
|
|
|
2006-05-24 16:08:06 +02:00
|
|
|
if (! setNoNagleOption(port)) {
|
|
|
|
gds__log("inet server err: setting NODELAY socket option \n");
|
2005-04-04 14:54:34 +02:00
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
return port;
|
|
|
|
}
|
|
|
|
|
|
|
|
void INET_set_clients( int count)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* I N E T _ s e t _ c l i e n t s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Set maxinum number of clients served before
|
|
|
|
* starting new server
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
INET_max_clients = (count && count < MAXCLIENTS) ? count : MAXCLIENTS;
|
|
|
|
}
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static int accept_connection(rem_port* port,
|
2003-09-11 20:59:34 +02:00
|
|
|
P_CNCT* cnct)
|
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
|
|
|
|
* Accept an incoming request for connection. This is purely a lower
|
|
|
|
* level handshaking function, and does not constitute the server
|
|
|
|
* response for protocol selection.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
/* Default account to "guest" (in theory all packets contain a name) */
|
|
|
|
|
2004-12-26 14:48:01 +01:00
|
|
|
Firebird::string name("guest"), password;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Pick up account and password, if given */
|
|
|
|
|
2005-11-27 21:53:09 +01:00
|
|
|
Firebird::ClumpletReader id(Firebird::ClumpletReader::UnTagged, cnct->p_cnct_user_id.cstr_address,
|
2004-12-26 14:48:01 +01:00
|
|
|
cnct->p_cnct_user_id.cstr_length);
|
|
|
|
SLONG eff_gid = -1, eff_uid = -1;
|
2003-09-11 20:59:34 +02:00
|
|
|
bool user_verification = false;
|
2004-12-26 14:48:01 +01:00
|
|
|
for (id.rewind(); !id.isEof(); id.moveNext())
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
2004-12-26 14:48:01 +01:00
|
|
|
switch (id.getClumpTag())
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
case CNCT_user:
|
2003-12-03 09:19:24 +01:00
|
|
|
{
|
2004-12-26 14:48:01 +01:00
|
|
|
id.getString(name);
|
2003-12-03 09:19:24 +01:00
|
|
|
break;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
case CNCT_passwd:
|
2003-12-03 09:19:24 +01:00
|
|
|
{
|
2004-12-26 14:48:01 +01:00
|
|
|
id.getString(password);
|
2003-12-03 09:19:24 +01:00
|
|
|
break;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
case CNCT_group:
|
2003-12-03 09:19:24 +01:00
|
|
|
{
|
2004-12-26 14:48:01 +01:00
|
|
|
int length = id.getClumpLength();
|
2003-12-03 09:19:24 +01:00
|
|
|
if (length != 0) {
|
2004-12-26 14:48:01 +01:00
|
|
|
eff_gid = 0;
|
2006-02-23 07:52:25 +01:00
|
|
|
memcpy(&eff_gid, id.getBytes(), length);
|
2004-12-26 14:48:01 +01:00
|
|
|
eff_gid = ntohl(eff_gid);
|
2003-12-03 09:19:24 +01:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* this case indicates that the client has requested that
|
|
|
|
we force the user name/password to be verified against
|
|
|
|
the security database */
|
|
|
|
|
|
|
|
case CNCT_user_verification:
|
2003-09-11 20:59:34 +02:00
|
|
|
user_verification = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* See if user exists. If not, reject connection */
|
|
|
|
if (user_verification) {
|
|
|
|
eff_gid = eff_uid = -1;
|
2006-02-23 06:08:26 +01:00
|
|
|
port->port_flags |= PORT_not_trusted; // never tested
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2002-10-30 07:40:58 +01:00
|
|
|
#if !defined(WIN_NT)
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef VMS
|
|
|
|
else
|
|
|
|
{
|
|
|
|
TEXT host[MAXHOSTLEN];
|
2004-02-24 06:34:44 +01:00
|
|
|
const int trusted = check_host(port, host, name);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!trusted) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (trusted == -1) {
|
|
|
|
eff_gid = eff_uid = -1;
|
2006-02-23 06:08:26 +01:00
|
|
|
port->port_flags |= PORT_not_trusted; // never tested
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
check_proxy(port, host, name);
|
|
|
|
|
|
|
|
if (!port->port_parent) {
|
|
|
|
if (!chuser(name))
|
|
|
|
return FALSE;
|
|
|
|
ISC_get_user(name, &eff_uid, &eff_gid, 0, 0, 0, 0);
|
|
|
|
}
|
|
|
|
/* ??? How to get uic for multi-client server case ??? */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Security check should be against remote name passed - so do
|
|
|
|
check_host first */
|
|
|
|
|
|
|
|
TEXT host[MAXHOSTLEN];
|
2005-01-27 14:19:16 +01:00
|
|
|
const struct passwd* passwd = getpwnam( name.c_str());
|
|
|
|
const int trusted = check_host(port, host, name.c_str(), passwd);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!trusted) {
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (trusted == -1) {
|
|
|
|
eff_gid = eff_uid = -1;
|
2006-02-23 06:08:26 +01:00
|
|
|
port->port_flags |= PORT_not_trusted; // never tested
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2005-01-29 10:22:36 +01:00
|
|
|
if (check_proxy(port, host, name))
|
2005-01-27 14:19:16 +01:00
|
|
|
passwd = getpwnam(name.c_str());
|
2005-12-23 09:52:31 +01:00
|
|
|
if (!passwd) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return FALSE;
|
2005-12-23 09:52:31 +01:00
|
|
|
}
|
2002-10-02 11:41:05 +02:00
|
|
|
#ifndef HAVE_INITGROUPS
|
2001-05-23 15:26:42 +02:00
|
|
|
eff_gid = passwd->pw_gid;
|
|
|
|
#else
|
|
|
|
|
2004-09-22 22:33:07 +02:00
|
|
|
SLONG gids[BUFFER_TINY];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
initgroups(passwd->pw_name, passwd->pw_gid);
|
|
|
|
if (eff_gid != -1) {
|
2002-12-02 10:45:52 +01:00
|
|
|
const int gid_count = getgroups(FB_NELEM(gids), (gid_t*)gids);
|
2003-12-03 09:19:24 +01:00
|
|
|
int i;
|
2001-07-12 07:46:06 +02:00
|
|
|
for (i = 0; i < gid_count; ++i) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (gids[i] == eff_gid) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if ((i == gid_count) && passwd) {
|
|
|
|
eff_gid = passwd->pw_gid;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
eff_gid = passwd->pw_gid;
|
|
|
|
}
|
2002-10-02 11:41:05 +02:00
|
|
|
#endif /* HAVE_INITGROUPS */
|
2001-05-23 15:26:42 +02:00
|
|
|
eff_uid = passwd->pw_uid;
|
|
|
|
|
|
|
|
/* if not multi-client: set uid, gid and home directory */
|
|
|
|
|
|
|
|
if (!port->port_parent) {
|
|
|
|
if (!eff_gid || setregid(passwd->pw_gid, eff_gid) == -1) {
|
|
|
|
setregid(passwd->pw_gid, passwd->pw_gid);
|
|
|
|
}
|
|
|
|
if (!setreuid(passwd->pw_uid, passwd->pw_uid)) {
|
2003-08-28 15:07:29 +02:00
|
|
|
chdir(passwd->pw_dir);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
2002-02-16 03:21:35 +01:00
|
|
|
/* If the environment varible ISC_INET_SERVER_HOME is set,
|
2001-05-23 15:26:42 +02:00
|
|
|
* change the home directory to the specified directory.
|
|
|
|
* Note that this will overrule the normal setting of
|
|
|
|
* the current directory to the effective user's home directory.
|
|
|
|
* This feature was added primarily for testing via remote
|
|
|
|
* loopback - but does seem to be of good general use, so
|
|
|
|
* is activiated for the production product.
|
2002-02-16 03:21:35 +01:00
|
|
|
* 1995-February-27 David Schnepper
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
2006-01-14 05:48:59 +01:00
|
|
|
Firebird::PathName home;
|
|
|
|
if (fb_utils::readenv("ISC_INET_SERVER_HOME", home))
|
|
|
|
{
|
|
|
|
if (chdir(home.c_str())) {
|
|
|
|
gds__log("inet_server: unable to cd to %s errno %d\n", home.c_str(),
|
2004-05-05 23:55:13 +02:00
|
|
|
INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
/* We continue after the error */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif /* VMS */
|
|
|
|
|
2002-10-30 07:40:58 +01:00
|
|
|
#endif /* !WIN_NT */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* store FULL user identity in port_user_name for security purposes */
|
|
|
|
|
2004-09-22 22:33:07 +02:00
|
|
|
Firebird::string temp;
|
2005-01-13 08:31:17 +01:00
|
|
|
temp.printf("%s.%ld.%ld", name.c_str(), eff_gid, eff_uid);
|
2004-09-22 22:33:07 +02:00
|
|
|
port->port_user_name = REMOTE_make_string(temp.c_str());
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-11-26 02:01:27 +01:00
|
|
|
port->port_protocol_str = REMOTE_make_string("TCPv4");
|
|
|
|
|
|
|
|
struct sockaddr_in address;
|
|
|
|
socklen_t l = sizeof(address);
|
|
|
|
|
2006-02-23 07:52:25 +01:00
|
|
|
inet_zero(&address, sizeof(address));
|
2004-11-26 02:01:27 +01:00
|
|
|
int status = getpeername((SOCKET) port->port_handle, (struct sockaddr *) &address, &l);
|
|
|
|
if (status == 0) {
|
|
|
|
Firebird::string addr_str;
|
|
|
|
UCHAR* ip = (UCHAR*) &address.sin_addr;
|
|
|
|
addr_str.printf(
|
|
|
|
"%d.%d.%d.%d",
|
|
|
|
static_cast<int>(ip[0]),
|
|
|
|
static_cast<int>(ip[1]),
|
|
|
|
static_cast<int>(ip[2]),
|
|
|
|
static_cast<int>(ip[3]) );
|
|
|
|
port->port_address_str = REMOTE_make_string(addr_str.c_str());
|
|
|
|
}
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2003-03-11 06:53:55 +01:00
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static rem_port* alloc_port( rem_port* parent)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* a l l o c _ p o r t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Allocate a port block, link it in to parent (if there is a parent),
|
|
|
|
* and initialize input and output XDR streams.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-03-11 06:53:55 +01:00
|
|
|
|
2004-09-22 22:33:07 +02:00
|
|
|
TEXT buffer[BUFFER_SMALL];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
if (!INET_initialized) {
|
2003-11-07 09:06:35 +01:00
|
|
|
const WORD version = MAKEWORD(2, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (WSAStartup(version, &INET_wsadata)) {
|
|
|
|
if (parent)
|
2004-05-05 23:55:13 +02:00
|
|
|
inet_error(parent, "WSAStartup", isc_net_init_error, INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
else {
|
2004-09-22 22:33:07 +02:00
|
|
|
SNPRINTF(buffer, FB_NELEM(buffer),
|
|
|
|
"INET/alloc_port: WSAStartup failed, error code = %d",
|
|
|
|
INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
gds__log(buffer, 0);
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
gds__register_cleanup(exit_handler, 0);
|
2004-02-24 06:34:44 +01:00
|
|
|
INET_initialized = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2003-09-11 20:59:34 +02:00
|
|
|
if (first_time == true)
|
2003-03-11 06:53:55 +01:00
|
|
|
{
|
2002-12-06 13:34:43 +01:00
|
|
|
INET_remote_buffer = Config::getTcpRemoteBufferSize();
|
2003-03-11 06:53:55 +01:00
|
|
|
if (INET_remote_buffer < MAX_DATA_LW ||
|
|
|
|
INET_remote_buffer > MAX_DATA_HW)
|
|
|
|
{
|
|
|
|
INET_remote_buffer = DEF_MAX_DATA;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
INET_max_data = INET_remote_buffer;
|
|
|
|
#ifdef DEBUG
|
|
|
|
{
|
2004-09-22 22:33:07 +02:00
|
|
|
char msg[BUFFER_SMALL];
|
|
|
|
SNPRINTF(msg, FB_NELEM(msg), " Info: Remote Buffer Size set to %ld",
|
|
|
|
INET_remote_buffer);
|
2004-09-24 08:42:39 +02:00
|
|
|
gds__log(msg, 0);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
2003-09-11 20:59:34 +02:00
|
|
|
first_time = false;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-05-17 12:22:34 +02:00
|
|
|
rem_port* port = (rem_port*) ALLR_block(type_port, INET_remote_buffer * 2);
|
2001-05-23 15:26:42 +02:00
|
|
|
port->port_type = port_inet;
|
|
|
|
port->port_state = state_pending;
|
|
|
|
REMOTE_get_timeout_params(port, 0, 0);
|
|
|
|
|
2002-11-14 07:48:09 +01:00
|
|
|
gethostname(buffer, sizeof(buffer));
|
2001-05-23 15:26:42 +02:00
|
|
|
port->port_host = REMOTE_make_string(buffer);
|
|
|
|
port->port_connection = REMOTE_make_string(buffer);
|
2004-09-22 22:33:07 +02:00
|
|
|
SNPRINTF(buffer, FB_NELEM(buffer), "tcp (%s)", port->port_host->str_data);
|
2001-05-23 15:26:42 +02:00
|
|
|
port->port_version = REMOTE_make_string(buffer);
|
|
|
|
|
2004-05-17 12:22:34 +02:00
|
|
|
START_PORT_CRITICAL();
|
2001-05-23 15:26:42 +02:00
|
|
|
if (parent && !(parent->port_server_flags & SRVR_thread_per_port)) {
|
|
|
|
port->port_parent = parent;
|
|
|
|
port->port_next = parent->port_clients;
|
|
|
|
parent->port_clients = parent->port_next = port;
|
|
|
|
port->port_handle = parent->port_handle;
|
|
|
|
port->port_server = parent->port_server;
|
|
|
|
port->port_server_flags = parent->port_server_flags;
|
|
|
|
}
|
2004-05-17 12:22:34 +02:00
|
|
|
STOP_PORT_CRITICAL();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
port->port_accept = accept_connection;
|
|
|
|
port->port_disconnect = disconnect;
|
|
|
|
port->port_receive_packet = receive;
|
2006-03-03 17:20:42 +01:00
|
|
|
port->port_select_multi = select_multi;
|
2001-05-23 15:26:42 +02:00
|
|
|
port->port_send_packet = send_full;
|
|
|
|
port->port_send_partial = send_partial;
|
2004-02-24 06:34:44 +01:00
|
|
|
port->port_connect = aux_connect;
|
2001-05-23 15:26:42 +02:00
|
|
|
port->port_request = aux_request;
|
|
|
|
port->port_buff_size = (USHORT) INET_remote_buffer;
|
|
|
|
|
|
|
|
xdrinet_create( &port->port_send, port,
|
|
|
|
&port->port_buffer[INET_remote_buffer],
|
|
|
|
(USHORT) INET_remote_buffer,
|
|
|
|
XDR_ENCODE);
|
|
|
|
|
|
|
|
xdrinet_create( &port->port_receive,
|
|
|
|
port, port->port_buffer,
|
|
|
|
0,
|
|
|
|
XDR_DECODE);
|
|
|
|
|
2006-03-03 17:20:42 +01:00
|
|
|
#ifdef SUPERSERVER
|
|
|
|
port->port_queue = FB_NEW(*getDefaultMemoryPool())
|
|
|
|
Firebird::ObjectsArray< Firebird::Array< char > >(*getDefaultMemoryPool());
|
|
|
|
port->port_qoffset = 0;
|
|
|
|
#endif
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
return port;
|
|
|
|
}
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
static rem_port* aux_connect(rem_port* port, PACKET* packet, t_event_ast ast)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* a u x _ c o n n e c t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Try to establish an alternative connection. Somebody has already
|
|
|
|
* done a successfull connect request ("packet" contains the response).
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
struct sockaddr_in address;
|
|
|
|
|
2004-08-01 19:44:24 +02:00
|
|
|
socklen_t l = sizeof(address);
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
/* If this is a server, we're got an auxiliary connection. Accept it */
|
|
|
|
|
|
|
|
if (port->port_server_flags) {
|
2006-03-20 17:29:04 +01:00
|
|
|
|
|
|
|
THREAD_EXIT();
|
2004-02-24 06:34:44 +01:00
|
|
|
SOCKET n = accept(port->port_channel, (struct sockaddr *) &address, &l);
|
2006-03-20 17:29:04 +01:00
|
|
|
THREAD_ENTER();
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
if (n == INVALID_SOCKET) {
|
2004-05-05 23:55:13 +02:00
|
|
|
inet_error(port, "accept", isc_net_event_connect_err, INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
SOCLOSE(port->port_channel);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
SOCLOSE(port->port_channel);
|
|
|
|
port->port_handle = (HANDLE) n;
|
|
|
|
port->port_flags |= PORT_async;
|
|
|
|
return port;
|
|
|
|
}
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* new_port = alloc_port(port->port_parent);
|
2003-11-07 09:06:35 +01:00
|
|
|
port->port_async = new_port;
|
2001-05-23 15:26:42 +02:00
|
|
|
new_port->port_dummy_packet_interval = port->port_dummy_packet_interval;
|
|
|
|
new_port->port_dummy_timeout = new_port->port_dummy_packet_interval;
|
2005-11-30 13:37:00 +01:00
|
|
|
new_port->port_flags = port->port_flags & PORT_no_oob;
|
2001-05-23 15:26:42 +02:00
|
|
|
new_port->port_flags |= PORT_async;
|
2003-11-07 09:06:35 +01:00
|
|
|
P_RESP* response = &packet->p_resp;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Set up new socket */
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
SOCKET n = socket(AF_INET, SOCK_STREAM, 0);
|
|
|
|
if (n == INVALID_SOCKET) {
|
2004-05-05 23:55:13 +02:00
|
|
|
inet_error(port, "socket", isc_net_event_connect_err, INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2004-08-01 19:44:24 +02:00
|
|
|
/*
|
|
|
|
* NJK - Determine address and port to use.
|
|
|
|
*
|
|
|
|
* The address returned by the server may be incorrect if it is behind a NAT box
|
|
|
|
* so we must use the address that was used to connect the main socket, not the
|
|
|
|
* address reported by the server.
|
|
|
|
*
|
|
|
|
* The port number reported by the server is used. For NAT support the port number
|
|
|
|
* should be configured to be a fixed port number in the server configuration.
|
|
|
|
*/
|
|
|
|
|
2006-02-23 07:52:25 +01:00
|
|
|
inet_zero(&address, sizeof(address));
|
2004-08-01 19:44:24 +02:00
|
|
|
int status = getpeername((SOCKET) port->port_handle, (struct sockaddr *) &address, &l);
|
|
|
|
if (status != 0) {
|
|
|
|
inet_error(port, "socket", isc_net_event_connect_err, INET_ERRNO);
|
|
|
|
SOCLOSE(n);
|
|
|
|
return NULL;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
address.sin_family = AF_INET;
|
2004-08-01 19:44:24 +02:00
|
|
|
address.sin_port = ((struct sockaddr_in *)(response->p_resp_data.cstr_address))->sin_port;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2004-08-01 19:44:24 +02:00
|
|
|
status = connect(n, (struct sockaddr *) &address, sizeof(address));
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (status < 0) {
|
2004-05-05 23:55:13 +02:00
|
|
|
inet_error(port, "connect", isc_net_event_connect_err, INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
SOCLOSE(n);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2002-11-12 14:15:13 +01:00
|
|
|
#ifdef SIOCSPGRP
|
2001-05-23 15:26:42 +02:00
|
|
|
if (ast)
|
|
|
|
{
|
2003-11-07 09:06:35 +01:00
|
|
|
int arg;
|
2002-11-12 14:15:13 +01:00
|
|
|
#ifdef HAVE_GETPGRP
|
2001-05-23 15:26:42 +02:00
|
|
|
arg = getpgrp();
|
|
|
|
#else
|
|
|
|
arg = getpid();
|
|
|
|
#endif
|
|
|
|
if (ioctl(n, SIOCSPGRP, &arg) < 0) {
|
|
|
|
inet_error(port, "ioctl/SIOCSPGRP", isc_net_event_connect_err,
|
2004-05-05 23:55:13 +02:00
|
|
|
INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
SOCLOSE(port->port_channel);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
new_port->port_ast = ast;
|
2003-09-08 22:23:46 +02:00
|
|
|
ISC_signal(SIGURG, inet_handler, new_port);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2002-11-12 14:15:13 +01:00
|
|
|
#endif /* SIOCSPGRP */
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
new_port->port_handle = (HANDLE) n;
|
|
|
|
|
|
|
|
return new_port;
|
|
|
|
}
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
static rem_port* aux_request( rem_port* port, PACKET* packet)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* a u x _ r e q u e s t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2002-02-16 03:21:35 +01:00
|
|
|
* A remote interface has requested the server prepare an auxiliary
|
2001-05-23 15:26:42 +02:00
|
|
|
* connection; the server calls aux_request to set up the connection.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-11-07 09:06:35 +01:00
|
|
|
struct sockaddr_in address;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Set up new socket */
|
|
|
|
|
2003-03-12 11:13:58 +01:00
|
|
|
address.sin_family = AF_INET;
|
2003-03-19 14:14:09 +01:00
|
|
|
in_addr bind_addr = get_bind_address();
|
2006-02-23 07:52:25 +01:00
|
|
|
inet_copy(&bind_addr,
|
|
|
|
(UCHAR*) &address.sin_addr,
|
2003-10-29 11:53:47 +01:00
|
|
|
sizeof(address.sin_addr));
|
2003-03-12 11:13:58 +01:00
|
|
|
address.sin_port = htons(Config::getRemoteAuxPort());
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
SOCKET n = socket(AF_INET, SOCK_STREAM, 0);
|
|
|
|
if (n == INVALID_SOCKET) {
|
2004-05-05 23:55:13 +02:00
|
|
|
inet_error(port, "socket", isc_net_event_listen_err, INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2006-10-25 11:54:32 +02:00
|
|
|
int optval = TRUE;
|
2006-10-25 17:09:13 +02:00
|
|
|
int ret = setsockopt(n, SOL_SOCKET, SO_REUSEADDR,
|
|
|
|
(SCHAR *) &optval, sizeof(optval));
|
|
|
|
|
2006-10-25 11:54:32 +02:00
|
|
|
if (ret == -1) {
|
|
|
|
inet_error(port, "setsockopt REUSE", isc_net_event_listen_err, INET_ERRNO);
|
|
|
|
return NULL;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (bind(n, (struct sockaddr *) &address, sizeof(address)) < 0) {
|
2004-05-05 23:55:13 +02:00
|
|
|
inet_error(port, "bind", isc_net_event_listen_err, INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2003-11-07 09:06:35 +01:00
|
|
|
socklen_t length = sizeof(address);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (getsockname(n, (struct sockaddr *) &address, &length) < 0) {
|
2004-05-05 23:55:13 +02:00
|
|
|
inet_error(port, "getsockname", isc_net_event_listen_err, INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (listen(n, 1) < 0) {
|
2004-05-05 23:55:13 +02:00
|
|
|
inet_error(port, "listen", isc_net_event_listen_err, INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* new_port = alloc_port(port->port_parent);
|
2003-11-07 09:06:35 +01:00
|
|
|
port->port_async = new_port;
|
2001-05-23 15:26:42 +02:00
|
|
|
new_port->port_dummy_packet_interval = port->port_dummy_packet_interval;
|
|
|
|
new_port->port_dummy_timeout = new_port->port_dummy_packet_interval;
|
|
|
|
|
|
|
|
new_port->port_server_flags = port->port_server_flags;
|
|
|
|
new_port->port_channel = (int) n;
|
|
|
|
new_port->port_flags = port->port_flags & PORT_no_oob;
|
|
|
|
|
2003-11-07 09:06:35 +01:00
|
|
|
P_RESP* response = &packet->p_resp;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-11-07 09:06:35 +01:00
|
|
|
struct sockaddr_in port_address;
|
2003-03-12 11:13:58 +01:00
|
|
|
if (getsockname((SOCKET) port->port_handle, (struct sockaddr *) &port_address, &length) < 0) {
|
2004-05-05 23:55:13 +02:00
|
|
|
inet_error(port, "getsockname", isc_net_event_listen_err, INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
2006-02-23 07:52:25 +01:00
|
|
|
inet_copy(&port_address.sin_addr,
|
|
|
|
(UCHAR*) &address.sin_addr,
|
2003-10-29 11:53:47 +01:00
|
|
|
sizeof(address.sin_addr));
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-11-15 16:52:57 +01:00
|
|
|
response->p_resp_data.cstr_address =
|
|
|
|
reinterpret_cast<UCHAR*>(& response->p_resp_blob_id);
|
2001-05-23 15:26:42 +02:00
|
|
|
response->p_resp_data.cstr_length = sizeof(response->p_resp_blob_id);
|
2006-02-23 07:52:25 +01:00
|
|
|
inet_copy(&address,
|
|
|
|
response->p_resp_data.cstr_address,
|
2001-05-23 15:26:42 +02:00
|
|
|
response->p_resp_data.cstr_length);
|
|
|
|
|
|
|
|
return new_port;
|
|
|
|
}
|
|
|
|
|
2003-03-01 18:51:59 +01:00
|
|
|
#ifndef WIN_NT
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef VMS
|
2004-01-28 08:50:41 +01:00
|
|
|
static int check_host(rem_port* port,
|
2004-03-07 08:58:55 +01:00
|
|
|
TEXT* host_name,
|
|
|
|
const TEXT* user_name)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c h e c k _ h o s t ( V M S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Check the host on the other end of the socket to see it
|
|
|
|
* it's an equivalent host.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
struct sockaddr_in address;
|
|
|
|
TEXT hosts_file[MAXPATHLEN];
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
int length = sizeof(address);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (getpeername((int) port->port_handle, &address, &length) == -1)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (getnamebyaddr(&address, sizeof(address), host_name) == -1)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
gethosts(hosts_file);
|
2004-02-24 06:34:44 +01:00
|
|
|
const int result = parse_hosts(hosts_file, host_name, user_name);
|
2003-09-13 11:15:06 +02:00
|
|
|
if (result == -1)
|
2001-05-23 15:26:42 +02:00
|
|
|
result = FALSE;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
2003-03-01 18:51:59 +01:00
|
|
|
#else
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
static int check_host(
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* port,
|
2004-03-07 08:58:55 +01:00
|
|
|
TEXT* host_name,
|
|
|
|
const TEXT* user_name, const struct passwd *passwd)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c h e c k _ h o s t ( n o n - V M S )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Check the host on the other end of the socket to see it
|
|
|
|
* it's an equivalent host.
|
|
|
|
* NB.: First check the ~/.rhosts then the HOSTS_FILE - both have
|
|
|
|
* the same line formats (see parse_line)
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
struct sockaddr_in address;
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
socklen_t length = sizeof(address);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2001-07-12 07:46:06 +02:00
|
|
|
if (getpeername((int) port->port_handle, (struct sockaddr*)&address, &length) == -1)
|
2001-05-23 15:26:42 +02:00
|
|
|
return 0;
|
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
const struct hostent* host = gethostbyaddr((SCHAR *) & address.sin_addr,
|
|
|
|
sizeof(address.sin_addr), address.sin_family);
|
|
|
|
if (!host)
|
2003-09-11 20:59:34 +02:00
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
int result = -1;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
strcpy(host_name, host->h_name);
|
|
|
|
|
2004-09-22 22:33:07 +02:00
|
|
|
TEXT user[BUFFER_TINY], rhosts[MAXPATHLEN];
|
2001-05-23 15:26:42 +02:00
|
|
|
if (passwd) {
|
|
|
|
strcpy(user, passwd->pw_name);
|
|
|
|
strcpy(rhosts, passwd->pw_dir);
|
|
|
|
strcat(rhosts, "/.rhosts");
|
|
|
|
result = parse_hosts(rhosts, host_name, user);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
strcpy(user, user_name);
|
|
|
|
|
|
|
|
if (result == -1) {
|
2004-04-29 00:36:29 +02:00
|
|
|
FILE* fp = fopen(GDS_HOSTS_FILE, "r");
|
2004-02-24 06:34:44 +01:00
|
|
|
const TEXT* hosts_file = fp ? (TEXT*)GDS_HOSTS_FILE : (TEXT*)HOSTS_FILE;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (fp)
|
2004-04-29 00:36:29 +02:00
|
|
|
fclose(fp);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-09-13 11:15:06 +02:00
|
|
|
result = parse_hosts(hosts_file, host_name, user);
|
|
|
|
if (result == -1)
|
2005-12-23 09:52:31 +01:00
|
|
|
result = 0;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
2003-12-03 09:19:24 +01:00
|
|
|
#endif // !non VMS
|
|
|
|
#endif // !defined(WIN_NT)
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-10-31 06:06:02 +01:00
|
|
|
#if !(defined WIN_NT)
|
2004-01-28 08:50:41 +01:00
|
|
|
static bool check_proxy(rem_port* port,
|
2005-01-29 10:22:36 +01:00
|
|
|
TEXT* host_name,
|
|
|
|
Firebird::string& user_name)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* c h e c k _ p r o x y
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Lookup <host_name, user_name> in proxy file. If found,
|
|
|
|
* change user_name.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-09-11 20:59:34 +02:00
|
|
|
TEXT proxy_file[MAXPATHLEN];
|
2004-09-22 22:33:07 +02:00
|
|
|
TEXT source_user[BUFFER_TINY];
|
2003-09-11 20:59:34 +02:00
|
|
|
TEXT source_host[MAXHOSTLEN];
|
2004-09-22 22:33:07 +02:00
|
|
|
TEXT target_user[BUFFER_TINY];
|
|
|
|
TEXT line[BUFFER_SMALL];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifndef VMS
|
|
|
|
strcpy(proxy_file, PROXY_FILE);
|
|
|
|
#else
|
|
|
|
gds__prefix(proxy_file, PROXY_FILE);
|
|
|
|
#endif
|
2004-04-29 00:36:29 +02:00
|
|
|
FILE* proxy = fopen(proxy_file, "r");
|
2004-02-24 06:34:44 +01:00
|
|
|
if (!proxy)
|
2003-09-11 20:59:34 +02:00
|
|
|
return false;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Read lines, scan, and compare */
|
|
|
|
|
2003-09-11 20:59:34 +02:00
|
|
|
bool result = false;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
for (;;) {
|
2004-02-24 06:34:44 +01:00
|
|
|
int c;
|
|
|
|
TEXT* p = line;
|
2004-04-29 00:36:29 +02:00
|
|
|
while (((c = getc(proxy)) != 0) && c != EOF && c != '\n')
|
2004-02-24 06:34:44 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
*p++ = c;
|
2004-02-24 06:34:44 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
*p = 0;
|
|
|
|
if (sscanf(line, " %[^:]:%s%s", source_host, source_user, target_user)
|
|
|
|
>= 3)
|
2006-04-09 08:46:28 +02:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if ((!strcmp(source_host, host_name) || !strcmp(source_host, "*"))
|
2005-01-29 10:22:36 +01:00
|
|
|
&& (!strcmp(source_user, user_name.c_str())
|
2004-02-24 06:34:44 +01:00
|
|
|
|| !strcmp(source_user, "*")))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
ALLR_free(port->port_user_name);
|
2004-02-24 06:34:44 +01:00
|
|
|
const SLONG length = strlen(target_user);
|
2004-05-17 12:22:34 +02:00
|
|
|
rem_str* string = (rem_str*) ALLR_block(type_str, (int) length);
|
2004-02-24 06:34:44 +01:00
|
|
|
port->port_user_name = string;
|
2001-05-23 15:26:42 +02:00
|
|
|
string->str_length = length;
|
|
|
|
strncpy(string->str_data, target_user, length);
|
2005-01-29 10:22:36 +01:00
|
|
|
user_name = target_user;
|
2003-09-11 20:59:34 +02:00
|
|
|
result = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
2006-04-09 08:46:28 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
if (c == EOF)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2004-04-29 00:36:29 +02:00
|
|
|
fclose(proxy);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static void disconnect( rem_port* port)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* d i s c o n n e c t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Break a remote connection.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
#ifndef VMS
|
|
|
|
/* SO_LINGER was turned off on the initial bind when the server was started.
|
|
|
|
* This will force a reset to be sent to the client when the socket is closed.
|
|
|
|
* We only want this behavior in the case of the server terminiating
|
|
|
|
* abnormally and not on an orderly shut down. Because of this, turn the
|
|
|
|
* SO_LINGER option back on for the socket. The result of setsockopt isn't
|
|
|
|
* too important at this stage since we are closing the socket anyway. This
|
|
|
|
* is an attempt to return the socket to a state where a graceful shutdown can
|
|
|
|
* occur.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (port->port_linger.l_onoff) {
|
2003-11-07 09:06:35 +01:00
|
|
|
setsockopt((SOCKET) port->port_handle, SOL_SOCKET, SO_LINGER,
|
2001-05-23 15:26:42 +02:00
|
|
|
(SCHAR *) & port->port_linger,
|
|
|
|
sizeof(port->port_linger));
|
|
|
|
}
|
|
|
|
|
|
|
|
#if defined WIN_NT
|
|
|
|
|
|
|
|
if (port->port_handle && (SOCKET) port->port_handle != INVALID_SOCKET) {
|
|
|
|
shutdown((int) port->port_handle, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
#else /* WIN_NT */
|
|
|
|
|
|
|
|
if (port->port_handle) {
|
|
|
|
shutdown((int) port->port_handle, 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif /* WIN_NT */
|
|
|
|
|
|
|
|
#endif /* !VMS */
|
|
|
|
|
2002-10-31 06:06:02 +01:00
|
|
|
#if !(defined VMS || defined WIN_NT)
|
2001-05-23 15:26:42 +02:00
|
|
|
if (port->port_ast) {
|
2003-09-08 22:23:46 +02:00
|
|
|
ISC_signal_cancel(SIGURG, inet_handler, port);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* If this is a sub-port, unlink it from it's parent */
|
|
|
|
|
|
|
|
#ifdef DEFER_PORT_CLEANUP
|
2003-11-07 09:06:35 +01:00
|
|
|
bool defer_cleanup = false;
|
2001-05-23 15:26:42 +02:00
|
|
|
port->port_state = state_disconnected;
|
|
|
|
#endif
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* parent = port->port_parent;
|
2003-11-07 09:06:35 +01:00
|
|
|
if (parent != NULL) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (port->port_async) {
|
|
|
|
disconnect(port->port_async);
|
|
|
|
port->port_async = NULL;
|
|
|
|
}
|
|
|
|
#ifdef DEFER_PORT_CLEANUP
|
2003-11-07 09:06:35 +01:00
|
|
|
defer_cleanup = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
#else
|
|
|
|
unhook_port(port, parent);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else if (port->port_async) {
|
|
|
|
#ifdef MULTI_THREAD
|
|
|
|
port->port_async->port_flags |= PORT_disconnect;
|
|
|
|
#else
|
|
|
|
disconnect(port->port_async);
|
|
|
|
port->port_async = NULL;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
if (port->port_handle) {
|
|
|
|
SOCLOSE((SOCKET) port->port_handle);
|
|
|
|
}
|
|
|
|
|
|
|
|
gds__unregister_cleanup(exit_handler, (void *) port);
|
|
|
|
|
|
|
|
#ifdef DEFER_PORT_CLEANUP
|
|
|
|
if (!defer_cleanup)
|
|
|
|
#endif
|
|
|
|
cleanup_port(port);
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
if (INET_trace & TRACE_summary) {
|
2004-04-29 00:36:29 +02:00
|
|
|
fprintf(stdout, "INET_count_send = %lu packets\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
INET_count_send);
|
2004-04-29 00:36:29 +02:00
|
|
|
fprintf(stdout, "INET_bytes_send = %lu bytes\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
INET_bytes_send);
|
2004-04-29 00:36:29 +02:00
|
|
|
fprintf(stdout, "INET_count_recv = %lu packets\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
INET_count_recv);
|
2004-04-29 00:36:29 +02:00
|
|
|
fprintf(stdout, "INET_bytes_recv = %lu bytes\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
INET_bytes_recv);
|
2004-04-29 00:36:29 +02:00
|
|
|
fflush(stdout);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static void cleanup_port( rem_port* port)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2002-02-16 03:21:35 +01:00
|
|
|
* c l e a n u p _ p o r t
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Walk through the port structure freeing
|
|
|
|
* allocated memory and then free the port.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
if (port->port_version)
|
|
|
|
ALLR_free((UCHAR *) port->port_version);
|
|
|
|
|
|
|
|
if (port->port_connection)
|
|
|
|
ALLR_free((UCHAR *) port->port_connection);
|
|
|
|
|
|
|
|
if (port->port_user_name)
|
|
|
|
ALLR_free((UCHAR *) port->port_user_name);
|
|
|
|
|
|
|
|
if (port->port_host)
|
|
|
|
ALLR_free((UCHAR *) port->port_host);
|
|
|
|
|
|
|
|
if (port->port_object_vector)
|
|
|
|
ALLR_free((UCHAR *) port->port_object_vector);
|
|
|
|
|
2004-11-26 02:01:27 +01:00
|
|
|
if (port->port_protocol_str)
|
|
|
|
ALLR_free((UCHAR *) port->port_protocol_str);
|
|
|
|
|
|
|
|
if (port->port_address_str)
|
|
|
|
ALLR_free((UCHAR *) port->port_address_str);
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef DEBUG_XDR_MEMORY
|
|
|
|
if (port->port_packet_vector)
|
|
|
|
ALLR_free((UCHAR *) port->port_packet_vector);
|
|
|
|
#endif
|
|
|
|
|
2006-03-03 17:20:42 +01:00
|
|
|
#ifdef SUPERSERVER
|
|
|
|
delete port->port_queue;
|
|
|
|
#endif
|
|
|
|
|
2006-12-08 19:38:15 +01:00
|
|
|
#ifdef TRUSTED_AUTH
|
|
|
|
delete port->port_trusted_auth;
|
|
|
|
#endif
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
ALLR_release((UCHAR *) port);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void exit_handler( void *arg)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* e x i t _ h a n d l e r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Shutdown all active connections
|
|
|
|
* to allow restart.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* main_port = (rem_port*) arg;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
if (!main_port) {
|
2005-03-10 11:23:52 +01:00
|
|
|
SleepEx(0, FALSE); // let select in other thread(s) shutdown gracefully
|
2001-05-23 15:26:42 +02:00
|
|
|
WSACleanup();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef VMS
|
2004-01-28 08:50:41 +01:00
|
|
|
for (rem_port* port = main_port; port; port = port->port_next) {
|
2001-05-23 15:26:42 +02:00
|
|
|
shutdown((int) port->port_handle, 2);
|
|
|
|
SOCLOSE((SOCKET) port->port_handle);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef NO_FORK
|
|
|
|
static int fork(void)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* f o r k ( N O _ F O R K )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Hmmm.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
static int fork( SOCKET old_handle, USHORT flag)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* f o r k ( W I N _ N T )
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Create a child process.
|
|
|
|
*
|
|
|
|
**************************************/
|
2006-11-28 11:21:21 +01:00
|
|
|
TEXT name[MAXPATHLEN];
|
|
|
|
GetModuleFileName(NULL, name, sizeof(name));
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
HANDLE new_handle;
|
2001-05-23 15:26:42 +02:00
|
|
|
DuplicateHandle(GetCurrentProcess(), (HANDLE) old_handle,
|
|
|
|
GetCurrentProcess(), &new_handle, 0, TRUE,
|
|
|
|
DUPLICATE_SAME_ACCESS);
|
|
|
|
|
2006-11-28 11:21:21 +01:00
|
|
|
Firebird::string cmdLine;
|
|
|
|
cmdLine.printf("%s -i -h %"SLONGFORMAT, name, (SLONG) new_handle);
|
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
STARTUPINFO start_crud;
|
2001-05-23 15:26:42 +02:00
|
|
|
start_crud.cb = sizeof(STARTUPINFO);
|
|
|
|
start_crud.lpReserved = NULL;
|
|
|
|
start_crud.lpReserved2 = NULL;
|
|
|
|
start_crud.cbReserved2 = 0;
|
|
|
|
start_crud.lpDesktop = NULL;
|
|
|
|
start_crud.lpTitle = NULL;
|
2003-09-24 16:53:25 +02:00
|
|
|
start_crud.dwFlags = STARTF_FORCEOFFFEEDBACK;
|
2004-02-24 06:34:44 +01:00
|
|
|
|
|
|
|
PROCESS_INFORMATION pi;
|
2006-11-28 11:21:21 +01:00
|
|
|
if (CreateProcess(NULL, cmdLine.begin(), NULL, NULL, TRUE,
|
|
|
|
(flag & SRVR_high_priority ?
|
|
|
|
HIGH_PRIORITY_CLASS | DETACHED_PROCESS :
|
|
|
|
NORMAL_PRIORITY_CLASS | DETACHED_PROCESS),
|
|
|
|
NULL, NULL, &start_crud, &pi))
|
2003-12-03 09:19:24 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
CloseHandle(pi.hThread);
|
|
|
|
CloseHandle(pi.hProcess);
|
|
|
|
}
|
|
|
|
CloseHandle(new_handle);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2003-03-19 14:14:09 +01:00
|
|
|
static in_addr get_bind_address()
|
2003-03-12 11:13:58 +01:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* g e t _ b i n d _ a d d r e s s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Return local address to bind sockets to.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-03-19 14:14:09 +01:00
|
|
|
in_addr config_address;
|
|
|
|
|
2003-03-12 11:13:58 +01:00
|
|
|
const char* config_option = Config::getRemoteBindAddress();
|
2003-03-19 14:14:09 +01:00
|
|
|
config_address.s_addr =
|
|
|
|
(config_option) ? inet_addr(config_option) : INADDR_NONE;
|
|
|
|
if (config_address.s_addr == INADDR_NONE) {
|
|
|
|
config_address.s_addr = INADDR_ANY;
|
|
|
|
}
|
|
|
|
return config_address;
|
2003-03-12 11:13:58 +01:00
|
|
|
}
|
|
|
|
|
2006-10-25 09:05:25 +02:00
|
|
|
static in_addr get_host_address(const Firebird::string& name)
|
2003-03-12 11:13:58 +01:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
2003-03-19 14:14:09 +01:00
|
|
|
* g e t _ h o s t _ a d d r e s s
|
2003-03-12 11:13:58 +01:00
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2003-03-19 14:14:09 +01:00
|
|
|
* Return host address.
|
2003-03-12 11:13:58 +01:00
|
|
|
*
|
|
|
|
**************************************/
|
2003-03-19 14:14:09 +01:00
|
|
|
in_addr address;
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-03-19 14:14:09 +01:00
|
|
|
|
2006-10-25 09:05:25 +02:00
|
|
|
address.s_addr = inet_addr(name.c_str());
|
2003-03-19 14:14:09 +01:00
|
|
|
|
|
|
|
if (address.s_addr == INADDR_NONE) {
|
|
|
|
|
2006-10-25 09:05:25 +02:00
|
|
|
const hostent* host = gethostbyname(name.c_str());
|
2003-03-19 14:14:09 +01:00
|
|
|
|
|
|
|
/* On Windows NT/9x, gethostbyname can only accomodate
|
|
|
|
* 1 call at a time. In this case it returns the error
|
|
|
|
* WSAEINPROGRESS. On UNIX systems, this call may not succeed
|
|
|
|
* because of a temporary error. In this case, it returns
|
|
|
|
* h_error set to TRY_AGAIN. When these errors occur,
|
|
|
|
* retry the operation a few times.
|
|
|
|
* NOTE: This still does not guarantee success, but helps.
|
|
|
|
*/
|
|
|
|
if (!host) {
|
|
|
|
if (H_ERRNO == INET_RETRY_ERRNO) {
|
|
|
|
for (int retry = 0; retry < INET_RETRY_CALL; retry++) {
|
2006-10-25 09:05:25 +02:00
|
|
|
if ( (host = gethostbyname(name.c_str())) )
|
2003-03-19 14:14:09 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (host) {
|
2006-02-23 07:52:25 +01:00
|
|
|
inet_copy(host->h_addr, (UCHAR*) &address, sizeof(address));
|
2003-03-19 14:14:09 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2003-03-19 14:14:09 +01:00
|
|
|
|
|
|
|
return address;
|
2003-03-12 11:13:58 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
//____________________________________________________________
|
2002-02-16 03:21:35 +01:00
|
|
|
//
|
2001-05-23 15:26:42 +02:00
|
|
|
// Copy an array of p_cnct::p_cnct_repeat.
|
2002-02-16 03:21:35 +01:00
|
|
|
//
|
2001-05-23 15:26:42 +02:00
|
|
|
static void copy_p_cnct_repeat_array( p_cnct::p_cnct_repeat* pDest,
|
|
|
|
const p_cnct::p_cnct_repeat* pSource,
|
|
|
|
size_t nEntries)
|
|
|
|
{
|
2004-02-24 06:34:44 +01:00
|
|
|
for (size_t i = 0; i < nEntries; ++i) {
|
2001-05-23 15:26:42 +02:00
|
|
|
pDest[i] = pSource[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-02-23 07:52:25 +01:00
|
|
|
static void inet_copy(const void* from, UCHAR* to, int length)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n e t _ c o p y
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Copy a number of bytes;
|
|
|
|
*
|
|
|
|
**************************************/
|
2006-02-23 07:52:25 +01:00
|
|
|
memcpy(to, from, length);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2006-02-23 07:52:25 +01:00
|
|
|
static void inet_zero(sockaddr_in* address, int length)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n e t _ z e r o
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Zero a block of memory.
|
|
|
|
*
|
|
|
|
**************************************/
|
2006-02-23 07:52:25 +01:00
|
|
|
memset(address, 0, length);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2002-10-31 06:06:02 +01:00
|
|
|
#if !(defined WIN_NT)
|
2004-03-07 08:58:55 +01:00
|
|
|
static int parse_hosts( const TEXT* file_name, const TEXT* host_name,
|
|
|
|
const TEXT* user_name)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/*****************************************************************
|
|
|
|
*
|
|
|
|
* p a r s e _ h o s t s
|
|
|
|
*
|
|
|
|
*****************************************************************
|
|
|
|
*
|
|
|
|
* Functional description:
|
|
|
|
* Parse hosts file (.rhosts or hosts.equiv) to determine
|
|
|
|
* if user_name on host_name should be allowed access.
|
|
|
|
*
|
|
|
|
*****************************************************************/
|
2004-09-22 22:33:07 +02:00
|
|
|
TEXT line[BUFFER_SMALL], entry1[BUFFER_SMALL], entry2[BUFFER_SMALL];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-02-24 06:34:44 +01:00
|
|
|
int result = -1;
|
2004-04-29 00:36:29 +02:00
|
|
|
FILE* fp = fopen(file_name, "r");
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (fp) {
|
|
|
|
for (;;) {
|
|
|
|
entry1[0] = entry2[0] = 0;
|
|
|
|
entry1[1] = entry2[1] = 0;
|
2004-02-24 06:34:44 +01:00
|
|
|
int c;
|
|
|
|
TEXT* p = line;
|
2004-04-29 00:36:29 +02:00
|
|
|
while ((c = getc(fp)) != EOF && c != '\n')
|
2001-05-23 15:26:42 +02:00
|
|
|
*p++ = c;
|
|
|
|
*p = 0;
|
|
|
|
sscanf(line, "%s", entry1);
|
|
|
|
sscanf(&line[strlen(entry1)], "%s", entry2);
|
|
|
|
result = parse_line(entry1, entry2, host_name, user_name);
|
|
|
|
if (c == EOF || result > -1)
|
|
|
|
break;
|
|
|
|
}
|
2004-04-29 00:36:29 +02:00
|
|
|
fclose(fp);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2002-10-31 06:06:02 +01:00
|
|
|
#if !(defined WIN_NT)
|
2001-05-23 15:26:42 +02:00
|
|
|
static int parse_line(
|
2004-03-07 08:58:55 +01:00
|
|
|
const TEXT* entry1,
|
|
|
|
const TEXT* entry2,
|
|
|
|
const TEXT* host_name, const TEXT* user_name)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/*****************************************************************
|
|
|
|
*
|
2002-02-16 03:21:35 +01:00
|
|
|
* p a r s e _ l i n e
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
*****************************************************************
|
|
|
|
*
|
|
|
|
* Functional description:
|
|
|
|
* Parse hosts file (.rhosts or hosts.equiv) to determine
|
|
|
|
* if user_name on host_name should be allowed access.
|
|
|
|
*
|
2003-09-11 20:59:34 +02:00
|
|
|
* Returns
|
|
|
|
* 1 if user_name is allowed
|
|
|
|
* 0 if not allowed and
|
|
|
|
* -1 if there is not a host_name or a user_name
|
|
|
|
*
|
2001-05-23 15:26:42 +02:00
|
|
|
* only supporting:
|
|
|
|
* + - anybody on any machine
|
|
|
|
* -@machinegroup [2nd entry optional] - nobody on that machine
|
|
|
|
* +@machinegroup - anybody on that machine
|
|
|
|
* +@machinegroup username - this person on these machines
|
|
|
|
* +@machinegroup +@usergroup - these people on these machines
|
|
|
|
* +@machinegroup -@usergroup - anybody on these machines EXCEPT these people
|
|
|
|
* machinename - anyone on this machine
|
|
|
|
* machinename username - this person on this machine
|
|
|
|
* machinename +@usergroup - these people on this machine
|
|
|
|
* machinename -@usergroup - anybody but these people on this machine
|
|
|
|
* machinename + - anybody on this machine
|
|
|
|
*
|
|
|
|
******************************************************************/
|
|
|
|
|
|
|
|
/* if entry1 is simply a "+" - everyone's in */
|
|
|
|
|
|
|
|
if (!strcmp(entry1, "+"))
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
/* if we don't have a host_name match, don't bother */
|
|
|
|
|
|
|
|
if (strcmp(entry1, host_name))
|
2006-04-09 08:46:28 +02:00
|
|
|
{
|
2004-05-21 01:05:02 +02:00
|
|
|
#if (defined UNIX) && !(defined SINIXZ) && !(defined NETBSD)
|
2006-04-09 08:46:28 +02:00
|
|
|
if (entry1[1] == '@')
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!innetgr(&entry1[2], host_name, 0, 0))
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
#endif
|
|
|
|
return -1;
|
2006-04-09 08:46:28 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* if we have a host_name match and an exclude symbol - they're out */
|
|
|
|
|
|
|
|
if (entry1[0] == '-')
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
/* if we matched the machine and seen a + after the machine
|
|
|
|
name they are in (eg: <machinename> + ) */
|
|
|
|
|
|
|
|
if ((entry2[0] == '+') && (strlen(entry2) == 1))
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
/* if we don't have a second entry OR it matches the user_name - they're in */
|
|
|
|
|
|
|
|
if (!entry2[0] || !strcmp(entry2, user_name))
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
/* if they're in the user group: + they're in, - they're out */
|
|
|
|
|
2004-05-21 01:05:02 +02:00
|
|
|
#if (defined UNIX) && !(defined SINIXZ) && !(defined NETBSD)
|
2001-05-23 15:26:42 +02:00
|
|
|
if (entry2[1] == '@') {
|
|
|
|
if (innetgr(&entry2[2], 0, user_name, 0)) {
|
|
|
|
if (entry2[0] == '+')
|
|
|
|
return TRUE;
|
|
|
|
else
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
else {
|
2002-02-16 03:21:35 +01:00
|
|
|
/* if they're NOT in the user group AND we're excluding it
|
2001-05-23 15:26:42 +02:00
|
|
|
- they're in */
|
|
|
|
|
|
|
|
if (entry2[0] == '-')
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* if we get here - we matched machine but not user - maybe next line ... */
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static rem_port* receive( rem_port* main_port, PACKET * packet)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* r e c e i v e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Receive a message from a port or clients of a port. If the process
|
|
|
|
* is a server and a connection request comes in, generate a new port
|
|
|
|
* block for the client.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2006-03-03 17:20:42 +01:00
|
|
|
#ifndef SUPERSERVER
|
2001-05-23 15:26:42 +02:00
|
|
|
/* If this isn't a multi-client server, just do the operation and get it
|
|
|
|
over with */
|
|
|
|
|
2006-03-07 06:44:13 +01:00
|
|
|
if (!(main_port->port_server_flags & SRVR_multi_client))
|
2006-03-03 17:20:42 +01:00
|
|
|
#endif //SUPERSERVER
|
2006-03-07 06:44:13 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
/* loop as long as we are receiving dummy packets, just
|
2002-02-16 03:21:35 +01:00
|
|
|
throwing them away--note that if we are a server we won't
|
|
|
|
be receiving them, but it is better to check for them at
|
|
|
|
this level rather than try to catch them in all places where
|
2001-05-23 15:26:42 +02:00
|
|
|
this routine is called */
|
|
|
|
|
|
|
|
do {
|
|
|
|
if (!xdr_protocol(&main_port->port_receive, packet))
|
2005-11-24 15:03:26 +01:00
|
|
|
{
|
2006-03-03 17:20:42 +01:00
|
|
|
packet->p_operation = main_port->port_flags & PORT_partial_data ?
|
|
|
|
op_partial : op_exit;
|
|
|
|
main_port->port_flags &= ~PORT_partial_data;
|
2005-11-24 15:03:26 +01:00
|
|
|
break;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef DEBUG
|
|
|
|
{
|
|
|
|
static ULONG op_rec_count = 0;
|
|
|
|
op_rec_count++;
|
|
|
|
if (INET_trace & TRACE_operations) {
|
2004-04-29 00:36:29 +02:00
|
|
|
fprintf(stdout, "%04lu: OP Recd %5lu opcode %d\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
inet_debug_timer(),
|
|
|
|
op_rec_count, packet->p_operation);
|
2004-04-29 00:36:29 +02:00
|
|
|
fflush(stdout);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
while (packet->p_operation == op_dummy);
|
|
|
|
|
|
|
|
return main_port;
|
|
|
|
}
|
|
|
|
|
2006-03-07 06:44:13 +01:00
|
|
|
#ifndef SUPERSERVER
|
2001-05-23 15:26:42 +02:00
|
|
|
/* Multi-client server multiplexes all known ports for incoming packets. */
|
|
|
|
|
|
|
|
for (;;) {
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* port = select_port(main_port, &INET_select);
|
2001-05-23 15:26:42 +02:00
|
|
|
if (port == main_port) {
|
|
|
|
if (port = select_accept(main_port))
|
|
|
|
return port;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (port) {
|
|
|
|
if (port->port_dummy_timeout < 0) {
|
|
|
|
port->port_dummy_timeout = port->port_dummy_packet_interval;
|
|
|
|
if (port->port_flags & PORT_async ||
|
2006-03-07 06:44:13 +01:00
|
|
|
port->port_protocol < PROTOCOL_VERSION8)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
packet->p_operation = op_dummy;
|
|
|
|
return port;
|
|
|
|
}
|
|
|
|
/* We've got data -- lap it up and use it */
|
|
|
|
|
|
|
|
if (!xdr_protocol(&port->port_receive, packet))
|
|
|
|
packet->p_operation = op_exit;
|
|
|
|
#ifdef DEBUG
|
|
|
|
{
|
|
|
|
static ULONG op_rec_count = 0;
|
|
|
|
op_rec_count++;
|
|
|
|
if (INET_trace & TRACE_operations) {
|
2004-04-29 00:36:29 +02:00
|
|
|
fprintf(stdout, "%05lu: OP Recd %5lu opcode %d\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
inet_debug_timer(),
|
|
|
|
op_rec_count, packet->p_operation);
|
2004-04-29 00:36:29 +02:00
|
|
|
fflush(stdout);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
2004-11-07 11:38:13 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Make sure that there are no more messages in this port before blocking
|
|
|
|
ourselves in select_wait. If there are more messages then set the flag
|
|
|
|
corresponding to this port which was cleared in select_port routine.
|
|
|
|
*/
|
|
|
|
if (port->port_receive.x_handy) {
|
|
|
|
FD_SET((SLONG) port->port_handle, &INET_select.slct_fdset);
|
|
|
|
++INET_select.slct_count;
|
|
|
|
}
|
|
|
|
if (packet->p_operation == op_dummy)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
return port;
|
|
|
|
}
|
|
|
|
if (!select_wait(main_port, &INET_select))
|
|
|
|
return NULL;
|
|
|
|
}
|
2006-03-03 17:20:42 +01:00
|
|
|
#endif //SUPERSERVER
|
|
|
|
}
|
|
|
|
|
|
|
|
static rem_port* select_multi(rem_port* main_port, UCHAR* buffer, SSHORT bufsize, SSHORT* length)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e l e c t _ m u l t i
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Receive an IP packet from a port or clients of a port.
|
|
|
|
* Used only by the multiclient server on main server's port.
|
|
|
|
* If a connection request comes in, generate a new port
|
|
|
|
* block for the client.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
fb_assert(main_port->port_server_flags & SRVR_multi_client);
|
|
|
|
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
rem_port* port = select_port(main_port, &INET_select);
|
|
|
|
if (port == main_port)
|
|
|
|
{
|
|
|
|
if (port = select_accept(main_port))
|
|
|
|
{
|
|
|
|
if (!packet_receive(port, buffer, bufsize, length))
|
|
|
|
{
|
|
|
|
*length = 0;
|
|
|
|
}
|
|
|
|
return port;
|
|
|
|
}
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (port)
|
|
|
|
{
|
|
|
|
if (port->port_dummy_timeout < 0)
|
|
|
|
{
|
|
|
|
port->port_dummy_timeout = port->port_dummy_packet_interval;
|
|
|
|
if (port->port_flags & PORT_async ||
|
|
|
|
port->port_protocol < PROTOCOL_VERSION8) continue;
|
|
|
|
*length = 0;
|
|
|
|
return port;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!packet_receive(port, buffer, bufsize, length))
|
|
|
|
{
|
|
|
|
*length = 0;
|
|
|
|
}
|
|
|
|
return port;
|
|
|
|
}
|
|
|
|
if (!select_wait(main_port, &INET_select))
|
|
|
|
{
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static rem_port* select_accept( rem_port* main_port)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e l e c t _ a c c e p t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Accept a new connection request.
|
|
|
|
* If client load is exceeded then
|
|
|
|
* handoff connection responsibility
|
|
|
|
* to a fresh server.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
struct sockaddr_in address;
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* port = alloc_port(main_port);
|
2003-12-03 09:19:24 +01:00
|
|
|
socklen_t l = sizeof(address);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
port->port_handle = (HANDLE) accept((SOCKET) main_port->port_handle,
|
|
|
|
(struct sockaddr *) &address, &l);
|
|
|
|
if ((SOCKET) port->port_handle == INVALID_SOCKET) {
|
2004-05-05 23:55:13 +02:00
|
|
|
inet_error(port, "accept", isc_net_connect_err, INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
disconnect(port);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
int optval = 1;
|
2001-05-23 15:26:42 +02:00
|
|
|
setsockopt((SOCKET) port->port_handle, SOL_SOCKET, SO_KEEPALIVE,
|
2003-12-03 09:19:24 +01:00
|
|
|
(SCHAR*) &optval, sizeof(optval));
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2002-10-31 06:06:02 +01:00
|
|
|
#if !(defined SUPERSERVER || defined VMS || defined WIN_NT)
|
2003-12-03 09:19:24 +01:00
|
|
|
int n;
|
2001-05-23 15:26:42 +02:00
|
|
|
for (n = 0, port = main_port->port_clients; port;
|
|
|
|
n++, port = port->port_next);
|
|
|
|
if (n >= INET_max_clients) {
|
|
|
|
main_port->port_state = state_closed;
|
|
|
|
SOCLOSE((int) main_port->port_handle);
|
2004-09-22 22:33:07 +02:00
|
|
|
TEXT msg[BUFFER_SMALL];
|
|
|
|
SNPRINTF(msg, FB_NELEM(msg),
|
2001-05-23 15:26:42 +02:00
|
|
|
"INET/select_accept: exec new server at client limit: %d", n);
|
|
|
|
gds__log(msg, 0);
|
|
|
|
|
|
|
|
setreuid(0, 0);
|
|
|
|
kill(getppid(), SIGUSR1);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (main_port->port_server_flags & SRVR_thread_per_port) {
|
|
|
|
port->port_server_flags =
|
|
|
|
(SRVR_server | SRVR_inet | SRVR_thread_per_port);
|
|
|
|
return port;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static rem_port* select_port( rem_port* main_port, SLCT * selct)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e l e c t _ p o r t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Select a descriptor that is ready to read
|
|
|
|
* and return the port block. Additionally,
|
|
|
|
* check if a port's keepalive timer has
|
|
|
|
* expired and return the port block so that
|
|
|
|
* a keepalive packet can be queued. Return
|
|
|
|
* NULL if none are active.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
#ifdef WIN_NT
|
|
|
|
|
|
|
|
/* NT's socket handles are addresses */
|
|
|
|
/* TMN: No, they are "black-box" handles. */
|
|
|
|
|
2004-05-17 12:22:34 +02:00
|
|
|
START_PORT_CRITICAL();
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef DEFER_PORT_CLEANUP
|
|
|
|
unhook_disconnected_ports(main_port);
|
|
|
|
#endif
|
2004-01-28 08:50:41 +01:00
|
|
|
for (rem_port* port = main_port; port; port = port->port_next) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (FD_ISSET(port->port_handle, &selct->slct_fdset)) {
|
|
|
|
port->port_dummy_timeout = port->port_dummy_packet_interval;
|
|
|
|
|
|
|
|
#pragma FB_COMPILER_MESSAGE("TODO: Make porthandle a SOCKET on Win32")
|
|
|
|
|
|
|
|
FD_CLR((SOCKET) port->port_handle, &selct->slct_fdset);
|
|
|
|
--selct->slct_count;
|
2004-05-17 12:22:34 +02:00
|
|
|
STOP_PORT_CRITICAL();
|
2001-05-23 15:26:42 +02:00
|
|
|
return port;
|
|
|
|
}
|
|
|
|
else if (port->port_dummy_timeout < 0) {
|
2004-05-17 12:22:34 +02:00
|
|
|
STOP_PORT_CRITICAL();
|
2001-05-23 15:26:42 +02:00
|
|
|
return port;
|
|
|
|
}
|
2003-12-03 09:19:24 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-17 12:22:34 +02:00
|
|
|
STOP_PORT_CRITICAL();
|
2003-12-03 09:19:24 +01:00
|
|
|
#else // !defined(WIN_NT)
|
2004-05-17 12:22:34 +02:00
|
|
|
START_PORT_CRITICAL();
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef DEFER_PORT_CLEANUP
|
|
|
|
unhook_disconnected_ports(main_port);
|
|
|
|
#endif
|
2004-01-28 08:50:41 +01:00
|
|
|
for (rem_port* port = main_port; port; port = port->port_next) {
|
2003-12-03 09:19:24 +01:00
|
|
|
const int n = (int) port->port_handle;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (n < selct->slct_width && FD_ISSET(n, &selct->slct_fdset)) {
|
|
|
|
port->port_dummy_timeout = port->port_dummy_packet_interval;
|
|
|
|
FD_CLR(n, &selct->slct_fdset);
|
|
|
|
--selct->slct_count;
|
2004-05-17 12:22:34 +02:00
|
|
|
STOP_PORT_CRITICAL();
|
2001-05-23 15:26:42 +02:00
|
|
|
return port;
|
|
|
|
}
|
|
|
|
else if (port->port_dummy_timeout < 0) {
|
2004-05-17 12:22:34 +02:00
|
|
|
STOP_PORT_CRITICAL();
|
2001-05-23 15:26:42 +02:00
|
|
|
return port;
|
|
|
|
}
|
|
|
|
}
|
2004-05-17 12:22:34 +02:00
|
|
|
STOP_PORT_CRITICAL();
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static int select_wait( rem_port* main_port, SLCT * selct)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e l e c t _ w a i t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Select interesting descriptors from
|
|
|
|
* port blocks and wait for something
|
|
|
|
* to read from them.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-09-22 22:33:07 +02:00
|
|
|
TEXT msg[BUFFER_SMALL];
|
2001-05-23 15:26:42 +02:00
|
|
|
struct timeval timeout;
|
|
|
|
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
selct->slct_count = selct->slct_width = 0;
|
|
|
|
FD_ZERO(&selct->slct_fdset);
|
2003-12-03 09:19:24 +01:00
|
|
|
bool found = false;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Use the time interval between select() calls to expire
|
|
|
|
keepalive timers on all ports. */
|
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
SLONG delta_time;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (selct->slct_time)
|
|
|
|
{
|
|
|
|
delta_time = (SLONG) time(NULL) - selct->slct_time;
|
|
|
|
selct->slct_time += delta_time;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
delta_time = 0;
|
|
|
|
selct->slct_time = (SLONG) time(NULL);
|
|
|
|
}
|
|
|
|
|
2004-05-17 12:22:34 +02:00
|
|
|
START_PORT_CRITICAL();
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef DEFER_PORT_CLEANUP
|
|
|
|
unhook_disconnected_ports(main_port);
|
|
|
|
#endif
|
2004-01-28 08:50:41 +01:00
|
|
|
for (rem_port* port = main_port; port; port = port->port_next)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
if ((port->port_state == state_active) ||
|
|
|
|
(port->port_state == state_pending))
|
|
|
|
{
|
|
|
|
/* Adjust down the port's keepalive timer. */
|
|
|
|
|
|
|
|
if (port->port_dummy_packet_interval)
|
|
|
|
{
|
|
|
|
port->port_dummy_timeout -= delta_time;
|
|
|
|
}
|
|
|
|
|
|
|
|
FD_SET((SLONG) port->port_handle, &selct->slct_fdset);
|
|
|
|
#ifdef WIN_NT
|
|
|
|
++selct->slct_width;
|
|
|
|
#else
|
|
|
|
selct->slct_width =
|
|
|
|
MAX(selct->slct_width, (int) port->port_handle);
|
|
|
|
#endif
|
2003-09-11 20:59:34 +02:00
|
|
|
found = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
2004-05-17 12:22:34 +02:00
|
|
|
STOP_PORT_CRITICAL();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (!found)
|
|
|
|
{
|
2003-09-11 20:59:34 +02:00
|
|
|
gds__log("INET/select_wait: client rundown complete, server exiting",
|
2001-05-23 15:26:42 +02:00
|
|
|
0);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
++selct->slct_width;
|
|
|
|
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
// Some platforms change the timeout in the select call.
|
|
|
|
// Reset timeout for each iteration to avoid problems.
|
|
|
|
timeout.tv_sec = SELECT_TIMEOUT;
|
|
|
|
timeout.tv_usec = 0;
|
|
|
|
|
|
|
|
#ifdef WIN_NT
|
|
|
|
selct->slct_count = select(FD_SETSIZE, &selct->slct_fdset,
|
2003-08-28 15:07:29 +02:00
|
|
|
NULL, NULL, &timeout);
|
2001-05-23 15:26:42 +02:00
|
|
|
#else
|
|
|
|
selct->slct_count = select(selct->slct_width, &selct->slct_fdset,
|
2003-08-28 15:07:29 +02:00
|
|
|
NULL, NULL, &timeout);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
if (selct->slct_count != -1)
|
|
|
|
{
|
|
|
|
/* if selct->slct_count is zero it means that we timed out of
|
2002-02-16 03:50:01 +01:00
|
|
|
select with nothing to read or accept, so clear the fd_set
|
|
|
|
bit as this value is undefined on some platforms (eg. HP-UX),
|
2001-05-23 15:26:42 +02:00
|
|
|
when the select call times out. Once these bits are cleared
|
|
|
|
they can be used in select_port() */
|
|
|
|
if (selct->slct_count == 0)
|
|
|
|
{
|
2004-01-28 08:50:41 +01:00
|
|
|
for (rem_port* port = main_port; port; port = port->port_next)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
#ifdef WIN_NT
|
|
|
|
FD_CLR((SOCKET)port->port_handle, &selct->slct_fdset);
|
|
|
|
#else
|
|
|
|
FD_CLR(port->port_handle, &selct->slct_fdset);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
2004-05-05 23:55:13 +02:00
|
|
|
else if (INTERRUPT_ERROR(INET_ERRNO))
|
2001-05-23 15:26:42 +02:00
|
|
|
continue;
|
|
|
|
#ifndef WIN_NT
|
2004-05-05 23:55:13 +02:00
|
|
|
else if (INET_ERRNO == EBADF)
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
#else
|
2004-05-05 23:55:13 +02:00
|
|
|
else if (INET_ERRNO == WSAENOTSOCK)
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
#endif
|
|
|
|
else {
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2004-09-22 22:33:07 +02:00
|
|
|
SNPRINTF(msg, FB_NELEM(msg),
|
|
|
|
"INET/select_wait: select failed, errno = %d",
|
|
|
|
INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
gds__log(msg, 0);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
} // for (;;)
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static int send_full( rem_port* port, PACKET * packet)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e n d _ f u l l
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Send a packet across a port to another process.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
if (!xdr_protocol(&port->port_send, packet))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
{
|
|
|
|
static ULONG op_sent_count = 0;
|
|
|
|
op_sent_count++;
|
|
|
|
if (INET_trace & TRACE_operations) {
|
2004-04-29 00:36:29 +02:00
|
|
|
fprintf(stdout, "%05lu: OP Sent %5lu opcode %d\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
inet_debug_timer(),
|
|
|
|
op_sent_count, packet->p_operation);
|
2004-04-29 00:36:29 +02:00
|
|
|
fflush(stdout);
|
2004-11-07 11:38:13 +01:00
|
|
|
}
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
2006-02-04 11:37:15 +01:00
|
|
|
return inet_write(&port->port_send, TRUE);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static int send_partial( rem_port* port, PACKET * packet)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e n d _ p a r t i a l
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Send a packet across a port to another process.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
{
|
|
|
|
static ULONG op_sentp_count = 0;
|
|
|
|
op_sentp_count++;
|
|
|
|
if (INET_trace & TRACE_operations) {
|
2004-04-29 00:36:29 +02:00
|
|
|
fprintf(stdout, "%05lu: OP Sent %5lu opcode %d (partial)\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
inet_debug_timer(),
|
|
|
|
op_sentp_count, packet->p_operation);
|
2004-04-29 00:36:29 +02:00
|
|
|
fflush(stdout);
|
2004-11-07 11:38:13 +01:00
|
|
|
}
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
return xdr_protocol(&port->port_send, packet);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int xdrinet_create(
|
|
|
|
XDR * xdrs,
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* port,
|
2001-05-23 15:26:42 +02:00
|
|
|
UCHAR * buffer, USHORT length, enum xdr_op x_op)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* x d r i n e t _ c r e a t e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Initialize an XDR stream for Apollo mailboxes.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
xdrs->x_public = (caddr_t) port;
|
|
|
|
xdrs->x_base = xdrs->x_private = (SCHAR *) buffer;
|
|
|
|
xdrs->x_handy = length;
|
|
|
|
xdrs->x_ops = (xdr_t::xdr_ops *) & inet_ops;
|
|
|
|
xdrs->x_op = x_op;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2003-03-03 09:37:54 +01:00
|
|
|
#ifdef HAVE_SETITIMER
|
2001-05-23 15:26:42 +02:00
|
|
|
static void alarm_handler( int x)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* a l a r m _ h a n d l e r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Handle an alarm clock interrupt. If we were waiting on
|
|
|
|
* a semaphone, zap it.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
}
|
2003-03-03 09:37:54 +01:00
|
|
|
#endif
|
2003-12-03 09:19:24 +01:00
|
|
|
static XDR_INT inet_destroy( XDR* xdrs)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n e t _ d e s t r o y
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Destroy a stream. A no-op.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2003-02-14 15:26:05 +01:00
|
|
|
return (XDR_INT)0;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static void inet_gen_error( rem_port* port, ISC_STATUS status, ...)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n e t _ g e n _ e r r o r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* An error has occurred. Mark the port as broken.
|
|
|
|
* Format the status vector if there is one and
|
|
|
|
* save the status vector strings in a permanent place.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
port->port_flags |= PORT_broken;
|
|
|
|
port->port_state = state_broken;
|
|
|
|
|
2003-11-07 09:06:35 +01:00
|
|
|
ISC_STATUS* status_vector = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (port->port_context != NULL) {
|
|
|
|
status_vector = port->port_context->rdb_status_vector;
|
|
|
|
}
|
|
|
|
if (status_vector == NULL) {
|
|
|
|
status_vector = port->port_status_vector;
|
|
|
|
}
|
|
|
|
if (status_vector != NULL) {
|
|
|
|
STUFF_STATUS(status_vector, status);
|
|
|
|
REMOTE_save_status_strings(status_vector);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool_t inet_getbytes( XDR * xdrs, SCHAR * buff, u_int count)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n e t _ g e t b y t e s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Get a bunch of bytes from a memory stream if it fits.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SLONG bytecount = count;
|
|
|
|
|
|
|
|
/* Use memcpy to optimize bulk transfers. */
|
|
|
|
|
2003-11-11 13:19:20 +01:00
|
|
|
while (bytecount > (SLONG) sizeof(ISC_QUAD)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (xdrs->x_handy >= bytecount) {
|
|
|
|
memcpy(buff, xdrs->x_private, bytecount);
|
|
|
|
xdrs->x_private += bytecount;
|
|
|
|
xdrs->x_handy -= bytecount;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (xdrs->x_handy > 0) {
|
|
|
|
memcpy(buff, xdrs->x_private, xdrs->x_handy);
|
|
|
|
xdrs->x_private += xdrs->x_handy;
|
|
|
|
buff += xdrs->x_handy;
|
|
|
|
bytecount -= xdrs->x_handy;
|
|
|
|
xdrs->x_handy = 0;
|
|
|
|
}
|
|
|
|
if (!inet_read(xdrs))
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Scalar values and bulk transfer remainder fall thru
|
|
|
|
to be moved byte-by-byte to avoid memcpy setup costs. */
|
|
|
|
|
|
|
|
if (!bytecount)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
if (xdrs->x_handy >= bytecount) {
|
|
|
|
xdrs->x_handy -= bytecount;
|
2006-04-09 08:46:28 +02:00
|
|
|
while (bytecount--)
|
2001-05-23 15:26:42 +02:00
|
|
|
*buff++ = *xdrs->x_private++;
|
2006-04-09 08:46:28 +02:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (--bytecount >= 0) {
|
|
|
|
if (!xdrs->x_handy && !inet_read(xdrs))
|
|
|
|
return FALSE;
|
|
|
|
*buff++ = *xdrs->x_private++;
|
|
|
|
--xdrs->x_handy;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool_t inet_getlong( XDR * xdrs, SLONG * lp)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n e t _ g e t l o n g
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Fetch a longword into a memory stream if it fits.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
SLONG l;
|
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
if (!(*xdrs->x_ops->x_getbytes) (xdrs, reinterpret_cast<char*>(&l), 4))
|
2001-05-23 15:26:42 +02:00
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
*lp = ntohl(l);
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static u_int inet_getpostn( XDR * xdrs)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n e t _ g e t p o s t n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Get the current position (which is also current length) from stream.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
return (u_int) (xdrs->x_private - xdrs->x_base);
|
|
|
|
}
|
|
|
|
|
2002-10-31 06:06:02 +01:00
|
|
|
#if !(defined WIN_NT)
|
2004-01-28 08:50:41 +01:00
|
|
|
static void inet_handler(void* port_void)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n e t _ h a n d l e r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Inet_handler is called the the signal handler on receipt of
|
|
|
|
* a SIGURG signal indicating out-of-band data. Since SIGURG
|
|
|
|
* may be noisy, check to see if any IO is pending on the channel.
|
2002-02-16 03:50:01 +01:00
|
|
|
* If not, ignore the signal. If so, call the port specific
|
2001-05-23 15:26:42 +02:00
|
|
|
* handler to do something appropriate.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* port = static_cast<rem_port*>(port_void);
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If there isn't any out of band data, this signal isn't for us */
|
2003-12-03 09:19:24 +01:00
|
|
|
SCHAR junk;
|
|
|
|
const int n = recv((SOCKET) port->port_handle, &junk, 1, MSG_OOB);
|
2006-04-09 08:46:28 +02:00
|
|
|
if (n < 0) {
|
2001-05-23 15:26:42 +02:00
|
|
|
return;
|
|
|
|
}
|
2003-12-03 09:19:24 +01:00
|
|
|
|
2004-03-07 08:58:55 +01:00
|
|
|
(*port->port_ast) (port);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
static caddr_t inet_inline( XDR* xdrs, u_int bytecount)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n e t _ i n l i n e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Return a pointer to somewhere in the buffer.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
if (bytecount > (u_int) xdrs->x_handy)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
return xdrs->x_base + bytecount;
|
|
|
|
}
|
|
|
|
|
2004-10-20 19:42:36 +02:00
|
|
|
static void inet_error(
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* port,
|
2003-12-03 09:19:24 +01:00
|
|
|
const TEXT* function, ISC_STATUS operation, int status)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n e t _ e r r o r
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* An I/O error has occurred. Call
|
|
|
|
* inet_gen_error with the appropriate args
|
|
|
|
* to format the status vector if any.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-09-22 22:33:07 +02:00
|
|
|
TEXT msg[BUFFER_SMALL];
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (status) {
|
|
|
|
#ifdef VMS
|
|
|
|
if ((status >= sys_nerr || !sys_errlist[status]) &&
|
|
|
|
status < WIN_NERR && win_errlist[status])
|
2004-02-24 06:34:44 +01:00
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
inet_gen_error(port, isc_network_error,
|
2003-11-08 17:40:17 +01:00
|
|
|
isc_arg_string,
|
2003-04-10 12:31:28 +02:00
|
|
|
(ISC_STATUS) port->port_connection->str_data,
|
2003-11-08 17:40:17 +01:00
|
|
|
isc_arg_gds, operation, isc_arg_string,
|
2003-04-10 12:31:28 +02:00
|
|
|
(ISC_STATUS) win_errlist[status], 0);
|
2004-02-24 06:34:44 +01:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
else
|
|
|
|
#endif
|
|
|
|
inet_gen_error(port, isc_network_error,
|
2003-11-08 17:40:17 +01:00
|
|
|
isc_arg_string,
|
2003-04-10 12:31:28 +02:00
|
|
|
(ISC_STATUS) port->port_connection->str_data,
|
2001-05-23 15:26:42 +02:00
|
|
|
isc_arg_gds, operation, SYS_ERR, status, 0);
|
|
|
|
|
2004-09-22 22:33:07 +02:00
|
|
|
SNPRINTF(msg, FB_NELEM(msg),
|
|
|
|
"INET/inet_error: %s errno = %d", function, status);
|
2001-05-23 15:26:42 +02:00
|
|
|
gds__log(msg, 0);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
/* No status value, just format the basic arguments. */
|
|
|
|
|
|
|
|
inet_gen_error(port, isc_network_error,
|
2003-11-08 17:40:17 +01:00
|
|
|
isc_arg_string,
|
2003-04-10 12:31:28 +02:00
|
|
|
(ISC_STATUS) port->port_connection->str_data, isc_arg_gds,
|
2001-05-23 15:26:42 +02:00
|
|
|
operation, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-10-29 11:53:47 +01:00
|
|
|
static bool_t inet_putbytes( XDR* xdrs, const SCHAR* buff, u_int count)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n e t _ p u t b y t e s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Put a bunch of bytes to a memory stream if it fits.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
SLONG bytecount = count;
|
|
|
|
|
|
|
|
/* Use memcpy to optimize bulk transfers. */
|
|
|
|
|
2003-11-11 13:19:20 +01:00
|
|
|
while (bytecount > (SLONG) sizeof(ISC_QUAD)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (xdrs->x_handy >= bytecount) {
|
|
|
|
memcpy(xdrs->x_private, buff, bytecount);
|
|
|
|
xdrs->x_private += bytecount;
|
|
|
|
xdrs->x_handy -= bytecount;
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
if (xdrs->x_handy > 0) {
|
|
|
|
memcpy(xdrs->x_private, buff, xdrs->x_handy);
|
|
|
|
xdrs->x_private += xdrs->x_handy;
|
|
|
|
buff += xdrs->x_handy;
|
|
|
|
bytecount -= xdrs->x_handy;
|
|
|
|
xdrs->x_handy = 0;
|
|
|
|
}
|
|
|
|
if (!inet_write(xdrs, 0))
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Scalar values and bulk transfer remainder fall thru
|
|
|
|
to be moved byte-by-byte to avoid memcpy setup costs. */
|
|
|
|
|
|
|
|
if (!bytecount)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
if (xdrs->x_handy >= bytecount) {
|
|
|
|
xdrs->x_handy -= bytecount;
|
2006-04-09 08:46:28 +02:00
|
|
|
while (bytecount--)
|
2001-05-23 15:26:42 +02:00
|
|
|
*xdrs->x_private++ = *buff++;
|
2006-04-09 08:46:28 +02:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (--bytecount >= 0) {
|
|
|
|
if (xdrs->x_handy <= 0 && !inet_write(xdrs, 0))
|
|
|
|
return FALSE;
|
|
|
|
--xdrs->x_handy;
|
|
|
|
*xdrs->x_private++ = *buff++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
// CVC: It could be const SLONG* lp, but it should fit into xdr_ops' signature.
|
2004-09-22 10:54:42 +02:00
|
|
|
static bool_t inet_putlong( XDR* xdrs, const SLONG* lp)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n e t _ p u t l o n g
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Fetch a longword into a memory stream if it fits.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-12-03 09:19:24 +01:00
|
|
|
const SLONG l = htonl(*lp);
|
2001-05-23 15:26:42 +02:00
|
|
|
return (*xdrs->x_ops->x_putbytes) (xdrs,
|
2003-12-03 09:19:24 +01:00
|
|
|
reinterpret_cast<const char*>(AOF32L(l)),
|
2001-05-23 15:26:42 +02:00
|
|
|
4);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool_t inet_read( XDR * xdrs)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n e t _ r e a d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Read a buffer full of data. If we receive a bad packet,
|
|
|
|
* send the moral equivalent of a NAK and retry. ACK all
|
|
|
|
* partial packets. Don't ACK the last packet -- the next
|
|
|
|
* message sent will handle this.
|
|
|
|
*
|
|
|
|
**************************************/
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* port = (rem_port*) xdrs->x_public;
|
2003-12-03 09:19:24 +01:00
|
|
|
char* p = xdrs->x_base;
|
|
|
|
const char* const end = p + INET_remote_buffer;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If buffer is not completely empty, slide down what's left */
|
|
|
|
|
|
|
|
if (xdrs->x_handy > 0) {
|
|
|
|
memmove(p, xdrs->x_private, xdrs->x_handy);
|
|
|
|
p += xdrs->x_handy;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If an ACK is pending, do an ACK. The alternative is deadlock. */
|
|
|
|
|
|
|
|
/*
|
|
|
|
if (port->port_flags & PORT_pend_ack)
|
|
|
|
if (!packet_send (port, 0, 0))
|
|
|
|
return FALSE;
|
|
|
|
*/
|
|
|
|
|
2003-10-29 11:53:47 +01:00
|
|
|
while (true) {
|
2003-12-03 09:19:24 +01:00
|
|
|
SSHORT length = end - p;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (!packet_receive
|
2006-04-09 08:46:28 +02:00
|
|
|
(port, reinterpret_cast<UCHAR*>(p), length, &length))
|
|
|
|
{
|
2001-05-23 15:26:42 +02:00
|
|
|
return FALSE;
|
2006-04-09 08:46:28 +02:00
|
|
|
/***
|
|
|
|
if (!packet_send (port, 0, 0))
|
|
|
|
return FALSE;
|
|
|
|
continue;
|
|
|
|
***/
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
if (length >= 0) {
|
|
|
|
p += length;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
p -= length;
|
|
|
|
if (!packet_send(port, 0, 0))
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
port->port_flags |= PORT_pend_ack;
|
|
|
|
xdrs->x_handy = (int) ((SCHAR *) p - xdrs->x_base);
|
|
|
|
xdrs->x_private = xdrs->x_base;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool_t inet_setpostn( XDR * xdrs, u_int bytecount)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n e t _ s e t p o s t n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Set the current position (which is also current length) from stream.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
|
|
|
if (bytecount > (u_int) xdrs->x_handy)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
xdrs->x_private = xdrs->x_base + bytecount;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static rem_port* inet_try_connect(
|
2003-10-29 11:53:47 +01:00
|
|
|
PACKET* packet,
|
2001-05-23 15:26:42 +02:00
|
|
|
RDB rdb,
|
2004-11-15 16:52:57 +01:00
|
|
|
Firebird::PathName& file_name,
|
2004-11-07 15:44:59 +01:00
|
|
|
const TEXT* node_name,
|
|
|
|
ISC_STATUS* status_vector,
|
|
|
|
const SCHAR* dpb,
|
|
|
|
SSHORT dpb_length)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n e t _ t r y _ c o n n e c t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Given a packet with formatted protocol infomation,
|
|
|
|
* set header information and try the connection.
|
|
|
|
*
|
|
|
|
* If a connection is established, return a port block, otherwise
|
|
|
|
* return NULL.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-12-03 09:19:24 +01:00
|
|
|
P_CNCT* cnct = &packet->p_cnct;
|
2001-05-23 15:26:42 +02:00
|
|
|
packet->p_operation = op_connect;
|
|
|
|
cnct->p_cnct_operation = op_attach;
|
|
|
|
cnct->p_cnct_cversion = CONNECT_VERSION2;
|
|
|
|
cnct->p_cnct_client = ARCHITECTURE;
|
2004-11-07 15:44:59 +01:00
|
|
|
cnct->p_cnct_file.cstr_length = file_name.length();
|
2004-11-15 16:52:57 +01:00
|
|
|
cnct->p_cnct_file.cstr_address =
|
2004-11-16 07:16:19 +01:00
|
|
|
reinterpret_cast<UCHAR*>(file_name.begin());
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* If we can't talk to a server, punt. Let somebody else generate
|
|
|
|
an error. status_vector will have the network error info. */
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* port =
|
2001-05-23 15:26:42 +02:00
|
|
|
INET_connect(node_name, packet, status_vector, FALSE, dpb,
|
|
|
|
dpb_length);
|
|
|
|
if (!port) {
|
|
|
|
ALLR_release((BLK) rdb);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get response packet from server. */
|
|
|
|
|
|
|
|
rdb->rdb_port = port;
|
|
|
|
port->port_context = rdb;
|
|
|
|
if (!port->receive(packet)) {
|
|
|
|
inet_error(port, "receive in try_connect", isc_net_connect_err,
|
2004-05-05 23:55:13 +02:00
|
|
|
INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
disconnect(port);
|
|
|
|
ALLR_release((BLK) rdb);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return port;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool_t inet_write( XDR * xdrs, bool_t end_flag)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* i n e t _ w r i t e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Write a buffer full of data. If the end_flag isn't set, indicate
|
|
|
|
* that the buffer is a fragment, and reset the XDR for another buffer
|
|
|
|
* load.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
/* Encode the data portion of the packet */
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* port = (rem_port*) xdrs->x_public;
|
2003-12-03 09:19:24 +01:00
|
|
|
const char* p = xdrs->x_base;
|
|
|
|
SSHORT length = xdrs->x_private - p;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Send data in manageable hunks. If a packet is partial, indicate
|
|
|
|
that with a negative length. A positive length marks the end. */
|
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
//p = xdrs->x_base; redundant
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
while (length) {
|
|
|
|
port->port_misc1 = (port->port_misc1 + 1) % MAX_SEQUENCE;
|
2003-12-03 09:19:24 +01:00
|
|
|
const SSHORT l = (SSHORT) MIN(length, INET_max_data);
|
2001-05-23 15:26:42 +02:00
|
|
|
length -= l;
|
|
|
|
if (!packet_send(port, p, (SSHORT) ((length) ? -l : l)))
|
|
|
|
return FALSE;
|
|
|
|
p += l;
|
|
|
|
}
|
|
|
|
|
|
|
|
xdrs->x_private = xdrs->x_base;
|
|
|
|
xdrs->x_handy = INET_remote_buffer;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
#ifdef PIGGYBACK
|
2003-12-03 09:19:24 +01:00
|
|
|
// CVC: Screwed logic here: if I initialize l2 to zero, nothing useful executes.
|
|
|
|
SCHAR aux_buffer[BUFFER_SIZE];
|
|
|
|
SSHORT l2 = 0;
|
|
|
|
#error Assign l2 some meaningful value before running this.
|
2001-05-23 15:26:42 +02:00
|
|
|
/* If the other end has not piggy-backed the next packet, we're done. */
|
|
|
|
|
|
|
|
if (!l2)
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
/* We've got a piggy-backed response. If the packet is partial,
|
|
|
|
send an ACK for part we did receive. */
|
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
char* p2 = aux_buffer;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
while (l2 < 0) {
|
|
|
|
if (!packet_send(port, 0, 0))
|
|
|
|
return FALSE;
|
2003-12-03 09:19:24 +01:00
|
|
|
p2 -= l2;
|
|
|
|
length = aux_buffer + sizeof(aux_buffer) - p2;
|
|
|
|
if (!packet_receive(port, p2, length, &l2)) {
|
|
|
|
p2 += l2;
|
2001-05-23 15:26:42 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
length = p2 - aux_buffer + l2;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
/* Now we're got a encode glump ready to stuff into the read buffer.
|
|
|
|
Unfortunately, if we just add it to the read buffer, we will shortly
|
|
|
|
overflow the buffer. To avoid this, "scrumpf down" the active bits
|
|
|
|
in the read buffer, then add out stuff at the end. */
|
|
|
|
|
|
|
|
xdrs = &port->port_receive;
|
2003-12-03 09:19:24 +01:00
|
|
|
p2 = xdrs->x_base;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
if (xdrs->x_handy && p2 != xdrs->x_private) {
|
|
|
|
memmove(p2, xdrs->x_private, xdrs->x_handy);
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
p2 += xdrs->x_handy;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
xdrs->x_private = xdrs->x_base;
|
|
|
|
/*
|
2003-12-03 09:19:24 +01:00
|
|
|
xdrs->x_handy += JAP_decode (aux_buffer, length, p2);
|
2001-05-23 15:26:42 +02:00
|
|
|
*/
|
|
|
|
port->port_flags |= PORT_pend_ack;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
|
|
|
static void packet_print(
|
2003-11-07 09:06:35 +01:00
|
|
|
const TEXT* string,
|
2006-03-07 06:44:13 +01:00
|
|
|
const UCHAR* packet, int length, ULONG counter)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a c k e t _ p r i n t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Print a summary of packet.
|
|
|
|
*
|
|
|
|
**************************************/
|
2003-11-07 09:06:35 +01:00
|
|
|
int sum = 0;
|
2006-04-09 08:46:28 +02:00
|
|
|
for (int l = length; l > 0; --l)
|
|
|
|
sum += *packet++;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-04-29 00:36:29 +02:00
|
|
|
fprintf(stdout,
|
2006-03-07 06:44:13 +01:00
|
|
|
"%05lu: PKT %s\t(%lu): length = %4d, checksum = %d\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
inet_debug_timer(), string, counter, length, sum);
|
2004-04-29 00:36:29 +02:00
|
|
|
fflush(stdout);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static int packet_receive(
|
2004-01-28 08:50:41 +01:00
|
|
|
rem_port* port,
|
2003-12-03 09:19:24 +01:00
|
|
|
UCHAR* buffer,
|
|
|
|
SSHORT buffer_length, SSHORT* length)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a c k e t _ r e c e i v e
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Receive a packet and pass on it's goodness. If it's good,
|
|
|
|
* return TRUE and the reported length of the packet, and update
|
|
|
|
* the receive sequence number. If it's bad, return FALSE. If it's
|
|
|
|
* a duplicate message, just ignore it.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
#ifndef REQUESTER
|
|
|
|
/* set the time interval for sending dummy packets to the client */
|
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
timeval timeout;
|
2001-05-23 15:26:42 +02:00
|
|
|
timeout.tv_sec = port->port_dummy_packet_interval;
|
|
|
|
timeout.tv_usec = 0;
|
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
timeval* time_ptr = NULL;
|
2001-05-23 15:26:42 +02:00
|
|
|
if (port->port_protocol >= PROTOCOL_VERSION8 &&
|
|
|
|
port->port_dummy_packet_interval > 0)
|
|
|
|
{
|
|
|
|
time_ptr = &timeout;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If the protocol is 0 we are still in the process of establishing
|
|
|
|
a connection. Add a time out to the wait */
|
|
|
|
if (port->port_protocol == 0)
|
|
|
|
{
|
|
|
|
timeout.tv_sec = port->port_connect_timeout;
|
|
|
|
time_ptr = &timeout;
|
|
|
|
}
|
|
|
|
|
|
|
|
// on Linux systems (and possibly others too) select will eventually
|
|
|
|
// change timout values so save it here for later reuse.
|
|
|
|
// Thanks to Brad Pepers who reported this bug FSG 3 MAY 2001
|
2004-06-21 04:48:47 +02:00
|
|
|
const timeval savetime = timeout;
|
2003-12-03 09:19:24 +01:00
|
|
|
|
|
|
|
int ph = (int) port->port_handle;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
// Unsed to send a dummy packet, but too big to be defined in the loop.
|
|
|
|
PACKET packet;
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifdef VMS
|
|
|
|
if (ISC_tcp_gettype() == TCP_TYPE_MULTINET)
|
|
|
|
{
|
|
|
|
ph = ph / 16;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
int n = 0;
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
|
|
|
|
#ifndef REQUESTER
|
|
|
|
|
|
|
|
/* Implement an error-detection protocol to ensure that the client
|
2002-02-16 03:50:01 +01:00
|
|
|
is still there. Use the select() call with a timeout to wait on
|
2001-05-23 15:26:42 +02:00
|
|
|
the connection for an incoming packet. If none comes within a
|
|
|
|
suitable time interval, write a dummy packet on the connection.
|
|
|
|
If the client is not there, an error will be returned on the write.
|
2002-02-16 03:50:01 +01:00
|
|
|
If the client is there, the dummy packet will be ignored by all
|
|
|
|
InterBase clients V4 or greater. This protocol will detect when
|
2001-05-23 15:26:42 +02:00
|
|
|
clients are lost abnormally through reboot or network disconnect. */
|
|
|
|
|
|
|
|
/* Don't send op_dummy packets on aux port; the server won't
|
|
|
|
read them because it only writes to aux ports. */
|
|
|
|
|
|
|
|
if (port->port_flags & PORT_async)
|
|
|
|
{
|
|
|
|
;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
#ifdef VMS
|
|
|
|
if (ISC_tcp_gettype() == TCP_TYPE_MULTINET)
|
|
|
|
#endif
|
|
|
|
{
|
2003-12-03 09:19:24 +01:00
|
|
|
fd_set slct_fdset;
|
2001-05-23 15:26:42 +02:00
|
|
|
FD_ZERO(&slct_fdset);
|
|
|
|
FD_SET(ph, &slct_fdset);
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-12-03 09:19:24 +01:00
|
|
|
int slct_count;
|
2001-05-23 15:26:42 +02:00
|
|
|
for (;;) {
|
2002-10-30 07:40:58 +01:00
|
|
|
#if (defined WIN_NT)
|
2001-05-23 15:26:42 +02:00
|
|
|
slct_count = select(FD_SETSIZE, &slct_fdset,
|
2003-08-28 15:07:29 +02:00
|
|
|
NULL, NULL, time_ptr);
|
2001-05-23 15:26:42 +02:00
|
|
|
#else
|
|
|
|
slct_count =
|
|
|
|
select((SOCKET) port->port_handle + 1, &slct_fdset,
|
2003-08-28 15:07:29 +02:00
|
|
|
NULL, NULL, time_ptr);
|
2001-05-23 15:26:42 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
// restore original timeout value FSG 3 MAY 2001
|
2004-06-14 13:20:37 +02:00
|
|
|
timeout = savetime;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-05-05 23:55:13 +02:00
|
|
|
if (slct_count != -1 || !INTERRUPT_ERROR(INET_ERRNO))
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (slct_count == -1)
|
|
|
|
{
|
2004-10-20 19:42:36 +02:00
|
|
|
inet_error(port, "select in packet_receive",
|
|
|
|
isc_net_read_err, INET_ERRNO);
|
|
|
|
return FALSE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!slct_count && port->port_protocol >= PROTOCOL_VERSION8)
|
|
|
|
{
|
|
|
|
#ifdef DEBUG
|
|
|
|
if (INET_trace & TRACE_operations)
|
|
|
|
{
|
2004-04-29 00:36:29 +02:00
|
|
|
fprintf(stdout, "%05lu: OP Sent: op_dummy\n",
|
2001-05-23 15:26:42 +02:00
|
|
|
inet_debug_timer());
|
2004-04-29 00:36:29 +02:00
|
|
|
fflush(stdout);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
packet.p_operation = op_dummy;
|
|
|
|
if (!send_full(port, &packet))
|
|
|
|
{
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!slct_count && port->port_protocol == 0)
|
|
|
|
{
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif /* REQUESTER */
|
|
|
|
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
n =
|
|
|
|
recv((SOCKET) port->port_handle,
|
2003-12-03 09:19:24 +01:00
|
|
|
reinterpret_cast<char*>(buffer), buffer_length, 0);
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_ENTER();
|
2004-05-05 23:55:13 +02:00
|
|
|
if (n != -1 || !INTERRUPT_ERROR(INET_ERRNO))
|
2001-05-23 15:26:42 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2004-10-20 19:42:36 +02:00
|
|
|
if (n == -1) {
|
|
|
|
inet_error(port, "read", isc_net_read_err, INET_ERRNO);
|
2001-05-23 15:26:42 +02:00
|
|
|
return FALSE;
|
2004-10-20 19:42:36 +02:00
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2004-10-20 19:42:36 +02:00
|
|
|
if (!n) {
|
|
|
|
inet_error(port, "read end_of_file", isc_net_read_err, 0);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
#ifndef REQUESTER
|
|
|
|
#ifdef DEBUG
|
|
|
|
{
|
|
|
|
INET_count_recv++;
|
|
|
|
INET_bytes_recv += n;
|
|
|
|
if (INET_trace & TRACE_packets)
|
|
|
|
packet_print("receive", buffer, n, INET_count_recv);
|
|
|
|
INET_force_error--;
|
|
|
|
if (INET_force_error == 0) {
|
|
|
|
INET_force_error = 1;
|
2004-10-20 19:42:36 +02:00
|
|
|
inet_error(port, "simulated error - read",
|
|
|
|
isc_net_read_err, 0);
|
|
|
|
return FALSE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
*length = n;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2006-10-25 09:05:25 +02:00
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static bool_t packet_send( rem_port* port, const SCHAR* buffer, SSHORT buffer_length)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* p a c k e t _ s e n d
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
2002-02-16 03:50:01 +01:00
|
|
|
* Send some data on it's way.
|
2001-05-23 15:26:42 +02:00
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2003-12-11 11:33:30 +01:00
|
|
|
#ifdef SINIXZ
|
2003-12-05 11:35:47 +01:00
|
|
|
// Please systems with ill-defined send() function, like SINIX-Z.
|
|
|
|
char* data = const_cast<char*>(buffer);
|
2003-12-11 11:33:30 +01:00
|
|
|
#else
|
|
|
|
const char* data = buffer;
|
2003-12-05 11:35:47 +01:00
|
|
|
#endif
|
2006-10-25 09:05:25 +02:00
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
SSHORT length = buffer_length;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
while (length) {
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef DEBUG
|
|
|
|
if (INET_trace & TRACE_operations) {
|
2004-04-29 00:36:29 +02:00
|
|
|
fprintf(stdout, "Before Send\n");
|
|
|
|
fflush(stdout);
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
#endif
|
2003-12-03 09:19:24 +01:00
|
|
|
SSHORT n = -1;
|
2006-10-25 09:05:25 +02:00
|
|
|
n = send((SOCKET) port->port_handle, data, length, FB_SEND_FLAGS);
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef DEBUG
|
|
|
|
if (INET_trace & TRACE_operations) {
|
2004-04-29 00:36:29 +02:00
|
|
|
fprintf(stdout, "After Send n is %d\n", n);
|
|
|
|
fflush(stdout);
|
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 (n == length) {
|
|
|
|
break;
|
|
|
|
}
|
2002-02-16 03:50:01 +01:00
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
if (n == -1)
|
|
|
|
{
|
2004-05-05 23:55:13 +02:00
|
|
|
if (INTERRUPT_ERROR(INET_ERRNO)) {
|
2001-05-23 15:26:42 +02:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2004-10-20 19:42:36 +02:00
|
|
|
inet_error(port, "send", isc_net_write_err, INET_ERRNO);
|
|
|
|
return FALSE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
data += n;
|
|
|
|
length -= n;
|
|
|
|
}
|
|
|
|
|
2004-12-26 05:09:06 +01:00
|
|
|
#ifdef HAVE_SETITIMER
|
|
|
|
struct itimerval internal_timer, client_timer;
|
|
|
|
struct sigaction internal_handler, client_handler;
|
|
|
|
#endif /* HAVE_SETITIMER */
|
|
|
|
|
2001-05-23 15:26:42 +02:00
|
|
|
if ((port->port_flags & PORT_async) && !(port->port_flags & PORT_no_oob))
|
|
|
|
{
|
2004-05-15 02:58:46 +02:00
|
|
|
THREAD_EXIT();
|
2003-12-03 09:19:24 +01:00
|
|
|
int count = 0;
|
|
|
|
SSHORT n;
|
2003-12-11 11:33:30 +01:00
|
|
|
#ifdef SINIXZ
|
2003-12-05 11:35:47 +01:00
|
|
|
char* b = const_cast<char*>(buffer);
|
2003-12-11 11:33:30 +01:00
|
|
|
#else
|
|
|
|
const char* b = buffer;
|
2003-12-05 11:35:47 +01:00
|
|
|
#endif
|
2006-10-25 09:05:25 +02:00
|
|
|
while ((n = send((SOCKET) port->port_handle, b, 1, MSG_OOB | FB_SEND_FLAGS)) == -1 &&
|
2004-05-05 23:55:13 +02:00
|
|
|
(INET_ERRNO == ENOBUFS || INTERRUPT_ERROR(INET_ERRNO)))
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
if (count++ > 20) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2002-11-14 08:35:52 +01:00
|
|
|
#ifndef HAVE_SETITIMER
|
2001-05-23 15:26:42 +02:00
|
|
|
#ifdef WIN_NT
|
|
|
|
SleepEx(50, TRUE);
|
|
|
|
#else
|
|
|
|
sleep(1);
|
|
|
|
#endif
|
2003-12-05 11:35:47 +01:00
|
|
|
} // end of while() loop for systems without setitimer.
|
2002-11-14 08:35:52 +01:00
|
|
|
#else /* HAVE_SETITIMER */
|
2001-05-23 15:26:42 +02:00
|
|
|
if (count == 1)
|
|
|
|
{
|
|
|
|
/* Wait in a loop until the lock becomes available */
|
|
|
|
|
|
|
|
internal_timer.it_interval.tv_sec = 0;
|
|
|
|
internal_timer.it_interval.tv_usec = 0;
|
|
|
|
internal_timer.it_value.tv_sec = 0;
|
|
|
|
internal_timer.it_value.tv_usec = 0;
|
|
|
|
setitimer(ITIMER_REAL, &internal_timer, &client_timer);
|
2003-09-08 22:23:46 +02:00
|
|
|
internal_handler.sa_handler = alarm_handler;
|
|
|
|
sigemptyset(&internal_handler.sa_mask);
|
2001-05-23 15:26:42 +02:00
|
|
|
internal_handler.sa_flags = SA_RESTART;
|
|
|
|
sigaction(SIGALRM, &internal_handler, &client_handler);
|
|
|
|
}
|
|
|
|
|
|
|
|
internal_timer.it_value.tv_sec = 0;
|
|
|
|
internal_timer.it_value.tv_usec = 50000;
|
|
|
|
setitimer(ITIMER_REAL, &internal_timer, NULL);
|
|
|
|
pause();
|
2003-12-05 11:35:47 +01:00
|
|
|
} // end of while() loop for systems with setitimer
|
2001-05-23 15:26:42 +02:00
|
|
|
|
|
|
|
if (count)
|
|
|
|
{
|
|
|
|
/* Restore user's outstanding alarm request and handler */
|
|
|
|
|
|
|
|
internal_timer.it_value.tv_sec = 0;
|
|
|
|
internal_timer.it_value.tv_usec = 0;
|
|
|
|
setitimer(ITIMER_REAL, &internal_timer, NULL);
|
|
|
|
sigaction(SIGALRM, &client_handler, NULL);
|
|
|
|
setitimer(ITIMER_REAL, &client_timer, NULL);
|
|
|
|
}
|
2002-11-14 08:35:52 +01:00
|
|
|
#endif /* HAVE_SETITIMER */
|
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 (n == -1) {
|
2004-10-20 19:42:36 +02:00
|
|
|
inet_error(port, "send/oob", isc_net_write_err, INET_ERRNO);
|
|
|
|
return FALSE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef REQUESTER
|
|
|
|
#ifdef DEBUG
|
|
|
|
{
|
|
|
|
INET_count_send++;
|
|
|
|
INET_bytes_send += buffer_length;
|
|
|
|
if (INET_trace & TRACE_packets)
|
2003-12-05 11:35:47 +01:00
|
|
|
packet_print("send", (const UCHAR*) buffer, buffer_length, INET_count_send);
|
2001-05-23 15:26:42 +02:00
|
|
|
INET_force_error--;
|
|
|
|
if (INET_force_error == 0) {
|
|
|
|
INET_force_error = 1;
|
2004-10-20 19:42:36 +02:00
|
|
|
inet_error(port, "simulated error - send",
|
|
|
|
isc_net_write_err, 0);
|
|
|
|
return FALSE;
|
2001-05-23 15:26:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
port->port_flags &= ~PORT_pend_ack;
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
static void unhook_port( rem_port* port, rem_port* parent)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* u n h o o k _ p a r e n t
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Disconnect a port from its parent
|
|
|
|
* This must be done under
|
|
|
|
* START_PORT_CRITICAL/STOP_PORT_CRITICAL control.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2004-01-28 08:50:41 +01:00
|
|
|
for (rem_port** ptr = &parent->port_clients; *ptr; ptr = &(*ptr)->port_next) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (*ptr == port) {
|
|
|
|
*ptr = port->port_next;
|
|
|
|
if (ptr == &parent->port_clients) {
|
|
|
|
parent->port_next = *ptr;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef SUPERSERVER
|
2004-01-28 08:50:41 +01:00
|
|
|
static void unhook_disconnected_ports(rem_port* main_port)
|
2001-05-23 15:26:42 +02:00
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* u n h o o k _ d i s c o n n e c t e d _ p o r t s
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Go through a list of ports and get rid of any
|
|
|
|
* that are disconnected. Note that this must be
|
|
|
|
* done under START_PORT_CRITICAL/STOP_PORT_CRITICAL
|
|
|
|
* control.
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
bool more = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
|
2003-12-03 09:19:24 +01:00
|
|
|
while (more) {
|
|
|
|
more = false;
|
2004-01-28 08:50:41 +01:00
|
|
|
for (rem_port* port = main_port; port; port = port->port_next) {
|
2001-05-23 15:26:42 +02:00
|
|
|
if (port->port_state == state_disconnected) {
|
2003-12-03 09:19:24 +01:00
|
|
|
more = true;
|
2001-05-23 15:26:42 +02:00
|
|
|
unhook_port(port, port->port_parent);
|
|
|
|
cleanup_port(port);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2006-05-24 16:08:06 +02:00
|
|
|
static bool setNoNagleOption(rem_port* port)
|
|
|
|
{
|
|
|
|
/**************************************
|
|
|
|
*
|
|
|
|
* s e t N o N a g l e O p t i o n
|
|
|
|
*
|
|
|
|
**************************************
|
|
|
|
*
|
|
|
|
* Functional description
|
|
|
|
* Set TCP_NODELAY, return false
|
|
|
|
* in case of unexpected error
|
|
|
|
*
|
|
|
|
**************************************/
|
|
|
|
if (Config::getTcpNoNagle())
|
|
|
|
{
|
|
|
|
int optval = TRUE;
|
2006-06-02 09:16:39 +02:00
|
|
|
int n = setsockopt((SOCKET) port->port_handle, IPPROTO_TCP,
|
2006-05-24 16:08:06 +02:00
|
|
|
TCP_NODELAY, (SCHAR*) &optval, sizeof(optval));
|
2002-03-11 17:34:08 +01:00
|
|
|
|
2006-06-02 09:16:39 +02:00
|
|
|
if (n == -1)
|
2006-05-24 16:08:06 +02:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|