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