diff options
Diffstat (limited to 'src/tet3/inetlib/connect.c')
-rw-r--r-- | src/tet3/inetlib/connect.c | 183 |
1 files changed, 183 insertions, 0 deletions
diff --git a/src/tet3/inetlib/connect.c b/src/tet3/inetlib/connect.c new file mode 100644 index 00000000..42ecb6c2 --- /dev/null +++ b/src/tet3/inetlib/connect.c @@ -0,0 +1,183 @@ +/* + * SCCS: @(#)connect.c 1.12 (98/09/01) + * + * UniSoft Ltd., London, England + * + * (C) Copyright 1992 X/Open Company Limited + * + * All rights reserved. No part of this source code may be reproduced, + * stored in a retrieval system, or transmitted, in any form or by any + * means, electronic, mechanical, photocopying, recording or otherwise, + * except as stated in the end-user licence agreement, without the prior + * permission of the copyright owners. + * + * X/Open and the 'X' symbol are trademarks of X/Open Company Limited in + * the UK and other countries. + */ + +#ifndef lint +static char sccsid[] = "@(#)connect.c 1.12 (98/09/01) TET3 release 3.3"; +#endif + +/************************************************************************ + +SCCS: @(#)connect.c 1.12 98/09/01 TETware release 3.3 +NAME: connect.c +PRODUCT: TETware +AUTHOR: Andrew Dingwall, UniSoft Ltd. +DATE CREATED: April 1992 + +DESCRIPTION: + required transport-specific library interface + + function to initiate connection to remote process + +MODIFICATIONS: + Denis McConalogue, UniSoft Limited, September 1993 + added tet_ts_disconnect() function + + Denis McConalogue, UniSoft Limited, September 1993 + make sure socket descriptor allocated is not 0, 1, or 2. + + Andrew Dingwall, UniSoft Ltd., December 1993 + moved disconnect stuff to a separate file + + Andrew Dingwall, UniSoft Ltd., February 1995 + clear sockaddr_in before using it + + Andrew Dingwall, UniSoft Ltd., March 1997 + remove #ifndef __hpux from #include <arpa/inet.h> + since current HP-UX implementations now have this file + + Andrew Dingwall, UniSoft Ltd., July 1998 + Report destination address/port if connect fails. + Added support for shared API libraries. + +************************************************************************/ + +#include <stdio.h> +#include <errno.h> +#include <fcntl.h> +#include <sys/types.h> +# include <unistd.h> +# include <sys/uio.h> +# include <sys/socket.h> +# include <netinet/in.h> +# include <arpa/inet.h> +#include "dtmac.h" +#include "dtmsg.h" +#include "ptab.h" +#include "tptab_in.h" +#include "error.h" +#include "ltoa.h" +#include "bstring.h" +#include "tslib.h" +#include "server_in.h" + +#ifdef NEEDsrcFile +static char srcFile[] = __FILE__; /* file name for error reporting */ +#endif + +/* +** tet_ts_connect() - make a connection to a remote process +*/ + +TET_IMPORT void tet_ts_connect(pp) +register struct ptab *pp; +{ + register struct tptab *tp = (struct tptab *) pp->pt_tdata; + register int rc; + register SOCKET sd; + register int nsd; + struct sockaddr_in sin; + int err; + static char fmt[] = "connect to %.16s port %d failed"; + char msg[sizeof fmt + 16 + LNUMSZ]; + + /* get a socket for the connection */ + if ((sd = socket(AF_INET, SOCK_STREAM, 0)) == INVALID_SOCKET) { + error(SOCKET_ERRNO, "can't get socket", (char *) 0); + pp->pt_state = PS_DEAD; + pp->pt_flags |= PF_ATTENTION; + return; + } + + + + /* ensure that socket is not on stdin, stdout or stderr */ + if (sd < 3) { + bzero((char *) &sin, sizeof sin); + sin.sin_family = AF_INET; + sin.sin_addr.s_addr = INADDR_ANY; + sin.sin_port = 0; + + errno = 0; + if (bind(sd, (struct sockaddr *) &sin, sizeof sin) < 0) { + error(errno, "can't bind client socket", (char *) 0); + pp->pt_state = PS_DEAD; + pp->pt_flags |= PF_ATTENTION; + return; + } + + if ((nsd = fcntl(sd, F_DUPFD, 3)) < 3) { + error(errno, "can't dup socket", (char *) 0); + pp->pt_state = PS_DEAD; + pp->pt_flags |= PF_ATTENTION; + return; + } + (void) close(sd); + sd = nsd; + } + + + + tp->tp_sd = sd; + + /* call the server-specific connect routine to massage the socket and + fill in network address */ + if (tet_ss_tsconnect(pp) < 0) { + pp->pt_state = PS_DEAD; + pp->pt_flags |= PF_ATTENTION; + return; + } + + TRACE5(tet_Tio, 4, "connect to %s on port %s using sd %s %s", + inet_ntoa(tp->tp_sin.sin_addr), + tet_i2a(ntohs(tp->tp_sin.sin_port)), + tet_i2a(tp->tp_sd), tet_r2a(&pp->pt_rid)); + + /* attempt the connection */ + do { + err = 0; + if ((rc = connect(sd, (struct sockaddr *) &tp->tp_sin, sizeof tp->tp_sin)) == SOCKET_ERROR) + err = SOCKET_ERRNO; + } while (rc == SOCKET_ERROR && err == SOCKET_EINTR); + + /* handle errors */ + if (rc == SOCKET_ERROR) { + switch (err) { + case SOCKET_EWOULDBLOCK: + case SOCKET_EINPROGRESS: + if (pp->pt_flags & PF_NBIO) { + TRACE1(tet_Tio, 6, "connect in progress"); + pp->pt_flags |= PF_INPROGRESS; + return; + } + /* else fall through */ + default: + (void) sprintf(msg, fmt, + inet_ntoa(tp->tp_sin.sin_addr), + (int) ntohs(tp->tp_sin.sin_port)); + error(err, msg, tet_r2a(&pp->pt_rid)); + pp->pt_state = PS_DEAD; + } + } + else { + TRACE1(tet_Tio, 6, "connect succeeded"); + pp->pt_flags |= PF_CONNECTED; + } + + /* here if the connect call completed one way or the other */ + pp->pt_flags |= PF_ATTENTION; +} + |