FSConnServ.c revision ba6a1819
1ba6a1819Smrg/* $Xorg: FSConnServ.c,v 1.4 2001/02/09 02:03:25 xorgcvs Exp $ */
2ba6a1819Smrg
3ba6a1819Smrg/*
4ba6a1819Smrg * Copyright 1990 Network Computing Devices;
5ba6a1819Smrg * Portions Copyright 1987 by Digital Equipment Corporation
6ba6a1819Smrg *
7ba6a1819Smrg * Permission to use, copy, modify, distribute, and sell this software
8ba6a1819Smrg * and its documentation for any purpose is hereby granted without fee,
9ba6a1819Smrg * provided that the above copyright notice appear in all copies and
10ba6a1819Smrg * that both that copyright notice and this permission notice appear
11ba6a1819Smrg * in supporting documentation, and that the names of Network Computing
12ba6a1819Smrg * Devices or Digital not be used in advertising or publicity pertaining
13ba6a1819Smrg * to distribution of the software without specific, written prior
14ba6a1819Smrg * permission. Network Computing Devices or Digital make no representations
15ba6a1819Smrg * about the suitability of this software for any purpose.  It is provided
16ba6a1819Smrg * "as is" without express or implied warranty.
17ba6a1819Smrg *
18ba6a1819Smrg * NETWORK COMPUTING DEVICES AND  DIGITAL DISCLAIM ALL WARRANTIES WITH
19ba6a1819Smrg * REGARD TO THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF
20ba6a1819Smrg * MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL NETWORK COMPUTING DEVICES
21ba6a1819Smrg * OR DIGITAL BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES
22ba6a1819Smrg * OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
23ba6a1819Smrg * WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
24ba6a1819Smrg * ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
25ba6a1819Smrg * SOFTWARE.
26ba6a1819Smrg */
27ba6a1819Smrg
28ba6a1819Smrg/*
29ba6a1819Smrg
30ba6a1819SmrgCopyright 1987, 1994, 1998  The Open Group
31ba6a1819Smrg
32ba6a1819SmrgPermission to use, copy, modify, distribute, and sell this software and its
33ba6a1819Smrgdocumentation for any purpose is hereby granted without fee, provided that
34ba6a1819Smrgthe above copyright notice appear in all copies and that both that
35ba6a1819Smrgcopyright notice and this permission notice appear in supporting
36ba6a1819Smrgdocumentation.
37ba6a1819Smrg
38ba6a1819SmrgThe above copyright notice and this permission notice shall be included in
39ba6a1819Smrgall copies or substantial portions of the Software.
40ba6a1819Smrg
41ba6a1819SmrgTHE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
42ba6a1819SmrgIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
43ba6a1819SmrgFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL THE
44ba6a1819SmrgOPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
45ba6a1819SmrgAN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
46ba6a1819SmrgCONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
47ba6a1819Smrg
48ba6a1819SmrgExcept as contained in this notice, the name of The Open Group shall not be
49ba6a1819Smrgused in advertising or otherwise to promote the sale, use or other dealings
50ba6a1819Smrgin this Software without prior written authorization from The Open Group.
51ba6a1819Smrg
52ba6a1819Smrg*/
53ba6a1819Smrg/* $XFree86: xc/lib/FS/FSConnServ.c,v 3.10 2001/10/28 03:32:27 tsi Exp $ */
54ba6a1819Smrg
55ba6a1819Smrg#ifdef HAVE_CONFIG_H
56ba6a1819Smrg#include <config.h>
57ba6a1819Smrg#endif
58ba6a1819Smrg#include	"FSlibint.h"
59ba6a1819Smrg#include	<stdio.h>
60ba6a1819Smrg#include	"X11/Xpoll.h"
61ba6a1819Smrg#ifdef NCD
62ba6a1819Smrg#include	<fcntl.h>
63ba6a1819Smrg#endif
64ba6a1819Smrg#ifdef WIN32
65ba6a1819Smrg#define ECHECK(err) (WSAGetLastError() == err)
66ba6a1819Smrg#else
67ba6a1819Smrg#ifdef ISC
68ba6a1819Smrg#define ECHECK(err) ((errno == err) || errno == EAGAIN || errno == EWOULDBLOCK)
69ba6a1819Smrg#else
70ba6a1819Smrg#define ECHECK(err) (errno == err)
71ba6a1819Smrg#endif
72ba6a1819Smrg#endif
73ba6a1819Smrg
74ba6a1819Smrg/*
75ba6a1819Smrg * Attempts to connect to server, given server name. Returns transport
76ba6a1819Smrg * connection object or NULL if connection fails.
77ba6a1819Smrg */
78ba6a1819Smrg
79ba6a1819Smrg#define FS_CONNECTION_RETRIES 5
80ba6a1819Smrg
81ba6a1819SmrgXtransConnInfo
82ba6a1819Smrg_FSConnectServer(char *server_name)
83ba6a1819Smrg{
84ba6a1819Smrg    XtransConnInfo trans_conn = NULL;	/* transport connection object */
85ba6a1819Smrg    int retry, connect_stat;
86ba6a1819Smrg    int  madeConnection = 0;
87ba6a1819Smrg
88ba6a1819Smrg    /*
89ba6a1819Smrg     * Open the network connection.
90ba6a1819Smrg     */
91ba6a1819Smrg
92ba6a1819Smrg    for (retry = FS_CONNECTION_RETRIES; retry >= 0; retry--)
93ba6a1819Smrg    {
94ba6a1819Smrg	if ((trans_conn = _FSTransOpenCOTSClient(server_name)) == NULL)
95ba6a1819Smrg	{
96ba6a1819Smrg	    break;
97ba6a1819Smrg	}
98ba6a1819Smrg
99ba6a1819Smrg	if ((connect_stat = _FSTransConnect(trans_conn,server_name)) < 0)
100ba6a1819Smrg	{
101ba6a1819Smrg	    _FSTransClose(trans_conn);
102ba6a1819Smrg
103ba6a1819Smrg	    if (connect_stat == TRANS_TRY_CONNECT_AGAIN)
104ba6a1819Smrg	    {
105ba6a1819Smrg		sleep(1);
106ba6a1819Smrg		continue;
107ba6a1819Smrg	    }
108ba6a1819Smrg	    else
109ba6a1819Smrg		break;
110ba6a1819Smrg	}
111ba6a1819Smrg	else
112ba6a1819Smrg	{
113ba6a1819Smrg	    madeConnection = 1;
114ba6a1819Smrg	    break;
115ba6a1819Smrg	}
116ba6a1819Smrg    }
117ba6a1819Smrg
118ba6a1819Smrg    if (!madeConnection)
119ba6a1819Smrg	return (NULL);
120ba6a1819Smrg
121ba6a1819Smrg
122ba6a1819Smrg    /*
123ba6a1819Smrg     * set it non-blocking.  This is so we can read data when blocked for
124ba6a1819Smrg     * writing in the library.
125ba6a1819Smrg     */
126ba6a1819Smrg
127ba6a1819Smrg    _FSTransSetOption(trans_conn, TRANS_NONBLOCKING, 1);
128ba6a1819Smrg
129ba6a1819Smrg    return (trans_conn);
130ba6a1819Smrg}
131ba6a1819Smrg
132ba6a1819Smrg/*
133ba6a1819Smrg * Disconnect from server.
134ba6a1819Smrg */
135ba6a1819Smrg
136ba6a1819Smrgvoid
137ba6a1819Smrg_FSDisconnectServer(XtransConnInfo trans_conn)
138ba6a1819Smrg
139ba6a1819Smrg{
140ba6a1819Smrg    (void) _FSTransClose(trans_conn);
141ba6a1819Smrg}
142ba6a1819Smrg
143ba6a1819Smrg
144ba6a1819Smrg/*
145ba6a1819Smrg * This is an OS dependent routine which:
146ba6a1819Smrg * 1) returns as soon as the connection can be written on....
147ba6a1819Smrg * 2) if the connection can be read, must enqueue events and handle errors,
148ba6a1819Smrg * until the connection is writable.
149ba6a1819Smrg */
150ba6a1819Smrgvoid _FSWaitForWritable(FSServer *svr)
151ba6a1819Smrg{
152ba6a1819Smrg    fd_set	r_mask;
153ba6a1819Smrg    fd_set	w_mask;
154ba6a1819Smrg    int         nfound;
155ba6a1819Smrg
156ba6a1819Smrg    FD_ZERO(&r_mask);
157ba6a1819Smrg    FD_ZERO(&w_mask);
158ba6a1819Smrg
159ba6a1819Smrg    while (1) {
160ba6a1819Smrg	FD_SET(svr->fd, &r_mask);
161ba6a1819Smrg	FD_SET(svr->fd, &w_mask);
162ba6a1819Smrg
163ba6a1819Smrg	do {
164ba6a1819Smrg	    nfound = Select(svr->fd + 1, &r_mask, &w_mask, NULL, NULL);
165ba6a1819Smrg	    if (nfound < 0 && !ECHECK(EINTR))
166ba6a1819Smrg		(*_FSIOErrorFunction) (svr);
167ba6a1819Smrg	} while (nfound <= 0);
168ba6a1819Smrg
169ba6a1819Smrg	if (XFD_ANYSET(&r_mask)) {
170ba6a1819Smrg	    char        buf[BUFSIZE];
171ba6a1819Smrg	    BytesReadable_t pend_not_register;
172ba6a1819Smrg	    register BytesReadable_t pend;
173ba6a1819Smrg	    register fsEvent *ev;
174ba6a1819Smrg
175ba6a1819Smrg	    /* find out how much data can be read */
176ba6a1819Smrg	    if (_FSTransBytesReadable(svr->trans_conn, &pend_not_register) < 0)
177ba6a1819Smrg		(*_FSIOErrorFunction) (svr);
178ba6a1819Smrg	    pend = pend_not_register;
179ba6a1819Smrg
180ba6a1819Smrg	    /*
181ba6a1819Smrg	     * must read at least one fsEvent; if none is pending, then we'll
182ba6a1819Smrg	     * just block waiting for it
183ba6a1819Smrg	     */
184ba6a1819Smrg	    if (pend < SIZEOF(fsEvent))
185ba6a1819Smrg		pend = SIZEOF(fsEvent);
186ba6a1819Smrg
187ba6a1819Smrg	    /* but we won't read more than the max buffer size */
188ba6a1819Smrg	    if (pend > BUFSIZE)
189ba6a1819Smrg		pend = BUFSIZE;
190ba6a1819Smrg
191ba6a1819Smrg	    /* round down to an integral number of FSReps */
192ba6a1819Smrg	    pend = (pend / SIZEOF(fsEvent)) * SIZEOF(fsEvent);
193ba6a1819Smrg
194ba6a1819Smrg	    _FSRead(svr, buf, pend);
195ba6a1819Smrg
196ba6a1819Smrg	    /* no space between comma and type or else macro will die */
197ba6a1819Smrg	    STARTITERATE(ev, fsEvent, buf, (pend > 0),
198ba6a1819Smrg			 (pend -= SIZEOF(fsEvent))) {
199ba6a1819Smrg		if (ev->type == FS_Error)
200ba6a1819Smrg		    _FSError(svr, (fsError *) ev);
201ba6a1819Smrg		else		/* it's an event packet; enqueue it */
202ba6a1819Smrg		    _FSEnq(svr, ev);
203ba6a1819Smrg	    }
204ba6a1819Smrg	    ENDITERATE
205ba6a1819Smrg	}
206ba6a1819Smrg	if (XFD_ANYSET(&w_mask))
207ba6a1819Smrg	    return;
208ba6a1819Smrg    }
209ba6a1819Smrg}
210ba6a1819Smrg
211ba6a1819Smrg
212ba6a1819Smrgvoid _FSWaitForReadable(FSServer *svr)
213ba6a1819Smrg{
214ba6a1819Smrg    fd_set	r_mask;
215ba6a1819Smrg    int         result;
216ba6a1819Smrg
217ba6a1819Smrg    FD_ZERO(&r_mask);
218ba6a1819Smrg    do {
219ba6a1819Smrg	FD_SET(svr->fd, &r_mask);
220ba6a1819Smrg	result = Select(svr->fd + 1, &r_mask, NULL, NULL, NULL);
221ba6a1819Smrg	if (result == -1 && !ECHECK(EINTR))
222ba6a1819Smrg	    (*_FSIOErrorFunction) (svr);
223ba6a1819Smrg    } while (result <= 0);
224ba6a1819Smrg}
225ba6a1819Smrg
226ba6a1819Smrgvoid _FSSendClientPrefix(
227ba6a1819Smrg    FSServer		*svr,
228ba6a1819Smrg    fsConnClientPrefix	*client)
229ba6a1819Smrg{
230ba6a1819Smrg    struct iovec iovarray[5],
231ba6a1819Smrg               *iov = iovarray;
232ba6a1819Smrg    int         niov = 0;
233ba6a1819Smrg
234ba6a1819Smrg#define add_to_iov(b,l) \
235ba6a1819Smrg	  { iov->iov_base = (b); iov->iov_len = (l); iov++, niov++; }
236ba6a1819Smrg
237ba6a1819Smrg    add_to_iov((caddr_t) client, SIZEOF(fsConnClientPrefix));
238ba6a1819Smrg
239ba6a1819Smrg#undef add_to_iov
240ba6a1819Smrg
241ba6a1819Smrg    (void) _FSTransWritev(svr->trans_conn, iovarray, niov);
242ba6a1819Smrg    return;
243ba6a1819Smrg}
244