home *** CD-ROM | disk | FTP | other *** search
- /* $XConsortium: io.c,v 1.9 92/05/18 13:50:44 gildea Exp $ */
- /*
- * i/o functions
- */
- /*
- * Copyright 1990, 1991 Network Computing Devices;
- * Portions Copyright 1987 by Digital Equipment Corporation and the
- * Massachusetts Institute of Technology
- *
- * Permission to use, copy, modify, and distribute this protoype software
- * and its documentation to Members and Affiliates of the MIT X Consortium
- * any purpose and without fee is hereby granted, provided
- * that the above copyright notice appear in all copies and that both that
- * copyright notice and this permission notice appear in supporting
- * documentation, and that the names of Network Computing Devices, Digital or
- * MIT not be used in advertising or publicity pertaining to distribution of
- * the software without specific, written prior permission.
- *
- * NETWORK COMPUTING DEVICES, DIGITAL AND MIT DISCLAIM ALL WARRANTIES WITH
- * REGARD TO THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY
- * AND FITNESS, IN NO EVENT SHALL NETWORK COMPUTING DEVICES, DIGITAL OR MIT BE
- * LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
- * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
- * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
- * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
- */
-
- #include <stdio.h>
- #include <errno.h>
- #include <sys/types.h>
- #include <sys/param.h>
- #include <sys/uio.h>
-
- #include "FSproto.h"
- #include "clientstr.h"
- #include "osdep.h"
- #include "globals.h"
-
- extern int errno;
-
- /* check for both EAGAIN and EWOULDBLOCK, because some supposedly POSIX
- * systems are broken and return EWOULDBLOCK when they should return EAGAIN
- */
-
- #if defined(EAGAIN) && defined(EWOULDBLOCK)
- #define ETEST(err) (err == EAGAIN || err == EWOULDBLOCK)
- #else
-
- #ifdef EAGAIN
- #define ETEST(err) (err == EAGAIN)
- #else
- #define ETEST(err) (err == EWOULDBLOCK)
- #endif
-
- #endif
-
-
- extern long ClientsWithInput[];
- extern long ClientsWriteBlocked[];
- extern long OutputPending[];
-
- extern long OutputBufferSize;
-
- extern int ConnectionTranslation[];
-
- extern Bool AnyClientsWriteBlocked;
- extern Bool NewOutputPending;
-
- static int timesThisConnection = 0;
- static ConnectionInputPtr FreeInputs = (ConnectionInputPtr) NULL;
- static ConnectionOutputPtr FreeOutputs = (ConnectionOutputPtr) NULL;
- static OsCommPtr AvailableInput = (OsCommPtr) NULL;
-
- static ConnectionInputPtr AllocateInputBuffer();
- static ConnectionOutputPtr AllocateOutputBuffer();
-
-
- #define MAX_TIMES_PER 10
-
- #define yield_control() \
- { isItTimeToYield = TRUE; \
- timesThisConnection = 0; }
-
- #define yield_control_no_input() \
- { yield_control(); \
- BITCLEAR(ClientsWithInput, fd); }
-
- #define yield_control_death() \
- { timesThisConnection = 0; }
-
- #define request_length(req, client) \
- (((client)->swapped ? lswaps((req)->length) : (req)->length) << 2)
-
- int
- ReadRequest(client)
- ClientPtr client;
- {
- OsCommPtr oc = (OsCommPtr) client->osPrivate;
- ConnectionInputPtr oci = oc->input;
- fsReq *request;
- int fd = oc->fd;
- int result,
- gotnow,
- needed;
-
- if (AvailableInput) {
- if (AvailableInput != oc) {
- ConnectionInputPtr aci = AvailableInput->input;
-
- if (aci->size > BUFWATERMARK) {
- fsfree(aci->buffer);
- fsfree(aci);
- } else {
- aci->next = FreeInputs;
- FreeInputs = aci;
- }
- AvailableInput->input = (ConnectionInputPtr) NULL;
- }
- AvailableInput = (OsCommPtr) NULL;
- }
- if (!oci) {
- if ((oci = FreeInputs ) != (ConnectionInputPtr) 0) {
- FreeInputs = oci->next;
- } else if (!(oci = AllocateInputBuffer())) {
- yield_control_death();
- return -1;
- }
- oc->input = oci;
- }
- oci->bufptr += oci->lenLastReq;
-
- request = (fsReq *) oci->bufptr;
-
- gotnow = oci->bufcnt + oci->buffer - oci->bufptr;
-
- /* not enough for a request */
- if ((gotnow < sizeof(fsReq)) ||
- (gotnow < (needed = request_length(request, client)))) {
- oci->lenLastReq = 0;
- if ((gotnow < sizeof(fsReq)) || needed == 0)
- needed = sizeof(fsReq);
- else if (needed > MAXBUFSIZE) {
- yield_control_death();
- return -1;
- }
- /* see if we need to shift up a partial request so the rest can fit */
- if ((gotnow == 0) ||
- ((oci->bufptr - oci->buffer + needed) > oci->size)) {
- if ((gotnow > 0) && (oci->bufptr != oci->buffer))
- bcopy(oci->bufptr, oci->buffer, gotnow);
- /* grow buffer if necessary */
- if (needed > oci->size) {
- char *ibuf;
-
- ibuf = (char *) fsrealloc(oci->buffer, needed);
- if (!ibuf) {
- yield_control_death();
- return -1;
- }
- oci->size = needed;
- oci->buffer = ibuf;
- }
- oci->bufptr = oci->buffer;
- oci->bufcnt = gotnow;
- }
- /* fill 'er up */
- result = read(fd, oci->buffer + oci->bufcnt,
- oci->size - oci->bufcnt);
- if (result <= 0) {
- if ((result < 0) && ETEST(errno)) {
- yield_control_no_input();
- return 0;
- } else {
- yield_control_death();
- return -1;
- }
- }
- oci->bufcnt += result;
- gotnow += result;
-
- /* free up space after huge requests */
- if ((oci->size > BUFWATERMARK) &&
- (oci->bufcnt < BUFSIZE) && (needed < BUFSIZE)) {
- char *ibuf;
-
- ibuf = (char *) fsrealloc(oci, BUFSIZE);
- if (ibuf) {
- oci->size = BUFSIZE;
- oci->buffer = ibuf;
- oci->bufptr = ibuf + oci->bufcnt - gotnow;
- }
- }
- request = (fsReq *) oci->bufptr;
- if ((gotnow < sizeof(fsReq)) ||
- (gotnow < (needed = request_length(request, client)))) {
- yield_control_no_input();
- return 0;
- }
- }
- if (needed == 0)
- needed = sizeof(fsReq);
- oci->lenLastReq = needed;
- /*
- * Check to see if client has at least one whole request in the buffer. If
- * there is only a partial request, treat like buffer is empty so that
- * select() will be called again and other clients can get into the queue.
- */
-
- if (gotnow >= needed + sizeof(fsReq)) {
- request = (fsReq *) (oci->bufptr + needed);
- if (gotnow >= needed + request_length(request, client))
- BITSET(ClientsWithInput, fd);
- else
- yield_control_no_input();
- } else {
- if (gotnow == needed)
- AvailableInput = oc;
- yield_control_no_input();
- }
-
- if (++timesThisConnection >= MAX_TIMES_PER)
- yield_control();
-
- client->requestBuffer = (pointer) oci->bufptr;
- return needed;
- }
-
- Bool
- InsertFakeRequest(client, data, count)
- ClientPtr client;
- char *data;
- int count;
- {
- OsCommPtr oc = (OsCommPtr) client->osPrivate;
- ConnectionInputPtr oci = oc->input;
- int fd = oc->fd;
- fsReq *request;
- int gotnow,
- moveup;
-
- if (AvailableInput) {
- if (AvailableInput != oc) {
- register ConnectionInputPtr aci = AvailableInput->input;
-
- if (aci->size > BUFWATERMARK) {
- fsfree(aci->buffer);
- fsfree(aci);
- } else {
- aci->next = FreeInputs;
- FreeInputs = aci;
- }
- AvailableInput->input = (ConnectionInputPtr) NULL;
- }
- AvailableInput = (OsCommPtr) NULL;
- }
- if (!oci) {
- if ((oci = FreeInputs) != (ConnectionInputPtr) 0)
- FreeInputs = oci->next;
- else if (!(oci = AllocateInputBuffer()))
- return FALSE;
- oc->input = oci;
-
- }
- oci->bufptr += oci->lenLastReq;
- oci->lenLastReq = 0;
- gotnow = oci->bufcnt + oci->buffer - oci->bufptr;
- if ((gotnow + count) > oci->size) {
- char *ibuf;
-
- ibuf = (char *) fsrealloc(oci->buffer, gotnow + count);
- if (!ibuf)
- return FALSE;
- oci->size = gotnow + count;
- oci->buffer = ibuf;
- oci->bufptr = ibuf + oci->bufcnt - gotnow;
- }
- moveup = count - (oci->bufptr - oci->buffer);
- if (moveup > 0) {
- if (gotnow > 0)
- bcopy(oci->bufptr, oci->bufptr + moveup, gotnow);
- oci->bufptr += moveup;
- oci->bufcnt += moveup;
- }
- bcopy(data, oci->bufptr - count, count);
- oci->bufptr -= count;
- request = (fsReq *) oci->bufptr;
- gotnow += count;
- if ((gotnow >= sizeof(fsReq)) &&
- (gotnow >= request_length(request, client)))
- BITSET(ClientsWithInput, fd);
- else
- yield_control_no_input();
- return TRUE;
- }
-
- ResetCurrentRequest(client)
- ClientPtr client;
- {
- OsCommPtr oc = (OsCommPtr) client->osPrivate;
- ConnectionInputPtr oci = oc->input;
- int fd = oc->fd;
- fsReq *request;
- int gotnow;
-
- if (AvailableInput == oc)
- AvailableInput = (OsCommPtr) NULL;
- oci->lenLastReq = 0;
- request = (fsReq *) oci->bufptr;
- gotnow = oci->bufcnt + oci->buffer - oci->bufptr;
- if ((gotnow >= sizeof(fsReq)) &&
- (gotnow >= request_length(request, client))) {
- BITSET(ClientsWithInput, fd);
- yield_control();
- } else {
- yield_control_no_input();
- }
- }
-
- int
- FlushClient(client, oc, extraBuf, extraCount, padsize)
- ClientPtr client;
- OsCommPtr oc;
- char *extraBuf;
- int extraCount;
- int padsize;
- {
- ConnectionOutputPtr oco = oc->output;
- int fd = oc->fd;
- struct iovec iov[3];
- char padBuffer[3];
- long written;
- long notWritten;
- long todo;
-
- if (!oco)
- return 0;
- written = 0;
- notWritten = oco->count + extraCount + padsize;
- todo = notWritten;
- while (notWritten) {
- long before = written;
- long remain = todo;
- int i = 0;
- long len;
-
- /*-
- * You could be very general here and have "in" and "out" iovecs and
- * write a loop without using a macro, but what the heck. This
- * translates to:
- *
- * how much of this piece is new?
- * if more new then we are trying this time, clamp
- * if nothing new
- * then bump down amount already written, for next piece
- * else put new stuff in iovec, will need all of next piece
- *
- * Note that todo had better be at least 1 or else we'll end up
- * writing 0 iovecs.
- */
-
- #define InsertIOV(pointer, length) \
- len = (length) - before; \
- if (len > remain) \
- len = remain; \
- if (len <= 0) { \
- before = (-len); \
- } else { \
- iov[i].iov_len = len; \
- iov[i].iov_base = (pointer) + before; \
- i++; \
- remain -= len; \
- before = 0; \
- }
-
- InsertIOV((char *) oco->buf, oco->count);
- InsertIOV(extraBuf, extraCount);
- InsertIOV(padBuffer, padsize);
-
- errno = 0;
- if ((len = writev(fd, iov, i)) >= 0) {
- written += len;
- notWritten -= len;
- todo = notWritten;
- } else if (ETEST(errno)) {
- BITSET(ClientsWriteBlocked, fd);
- AnyClientsWriteBlocked = TRUE;
-
- if (written < oco->count) {
- if (written > 0) {
- oco->count -= written;
- bcopy((char *) oco->buf + written,
- (char *) oco->buf, oco->count);
- written = 0;
- }
- } else {
- written -= oco->count;
- oco->count = 0;
- }
-
- /* grow buffer if necessary */
- if (notWritten > oco->size) {
- unsigned char *obuf;
-
- obuf = (unsigned char *) fsrealloc(oco->buf,
- notWritten + OutputBufferSize);
- if (!obuf) {
- close(fd);
- MarkClientException(client);
- oco->count = 0;
- return -1;
- }
- oco->size = notWritten + OutputBufferSize;
- oco->buf = obuf;
- }
- if ((len = extraCount - written) > 0) {
- bcopy(extraBuf + written,
- (char *) oco->buf + oco->count, len);
- }
- oco->count = notWritten;
- return extraCount;
- } else {
- close(fd);
- MarkClientException(client);
- oco->count = 0;
- return -1;
- }
- }
-
- /* everything was flushed */
- oco->count = 0;
-
- /* clear the write block if it was set */
- if (AnyClientsWriteBlocked) {
- BITCLEAR(ClientsWriteBlocked, fd);
- if (!ANYSET(ClientsWriteBlocked))
- AnyClientsWriteBlocked = FALSE;
- }
- if (oco->size > BUFWATERMARK) {
- fsfree(oco->buf);
- fsfree(oco);
- } else {
- oco->next = FreeOutputs;
- FreeOutputs = oco;
- }
- oc->output = (ConnectionOutputPtr) NULL;
-
- return extraCount;
- }
-
- void
- FlushAllOutput()
- {
- int index,
- base,
- mask;
- OsCommPtr oc;
- ClientPtr client;
-
- if (!NewOutputPending)
- return;
-
- NewOutputPending = FALSE;
-
- for (base = 0; base < mskcnt; base++) {
- mask = OutputPending[base];
- OutputPending[base] = 0;
- while (mask) {
- index = ffs(mask) - 1;
- mask &= ~lowbit(mask);
- if ((index = ConnectionTranslation[(base << 5) + index]) == 0)
- continue;
- client = clients[index];
- if (client->clientGone == CLIENT_GONE)
- continue;
- oc = (OsCommPtr) client->osPrivate;
- if (GETBIT(ClientsWithInput, oc->fd)) {
- BITSET(OutputPending, oc->fd);
- NewOutputPending = TRUE;
- } else {
- (void) FlushClient(client, oc, (char *) NULL, 0, 0);
- }
- }
- }
- }
-
- /*
- * returns number of bytes written
- */
- static int
- write_to_client_internal(client, count, buf, padBytes)
- ClientPtr client;
- int count;
- char *buf;
- int padBytes;
- {
- OsCommPtr oc = (OsCommPtr) client->osPrivate;
- ConnectionOutputPtr oco = oc->output;
-
- if (!count)
- return 0;
-
- if (!oco) {
- if ((oco = FreeOutputs) != (ConnectionOutputPtr) 0) {
- FreeOutputs = oco->next;
- } else if (!(oco = AllocateOutputBuffer())) {
- close(oc->fd);
- MarkClientException(client);
- return -1;
- }
- oc->output = oco;
- }
- if (oco->count + count + padBytes > oco->size) {
- BITCLEAR(OutputPending, oc->fd);
- NewOutputPending = FALSE;
- return FlushClient(client, oc, buf, count, padBytes);
- }
- NewOutputPending = TRUE;
- BITSET(OutputPending, oc->fd);
- bcopy(buf, (char *) oco->buf + oco->count, count);
- oco->count += count + padBytes;
-
- return count;
- }
-
- WriteToClientUnpadded(client, count, buf)
- ClientPtr client;
- int count;
- char *buf;
- {
- write_to_client_internal(client, count, buf, 0);
- }
-
- static int padlength[4] = {0, 3, 2, 1};
-
- WriteToClient(client, count, buf)
- ClientPtr client;
- int count;
- char *buf;
- {
- write_to_client_internal(client, count, buf, padlength[count & 3]);
- }
-
- static ConnectionInputPtr
- AllocateInputBuffer()
- {
- register ConnectionInputPtr oci;
-
- oci = (ConnectionInputPtr) fsalloc(sizeof(ConnectionInput));
- if (!oci)
- return (ConnectionInputPtr) NULL;
- oci->buffer = (char *) fsalloc(BUFSIZE);
- if (!oci->buffer) {
- fsfree(oci);
- return (ConnectionInputPtr) NULL;
- }
- oci->next = 0;
- oci->size = BUFSIZE;
- oci->bufptr = oci->buffer;
- oci->bufcnt = 0;
- oci->lenLastReq = 0;
- return oci;
- }
- static ConnectionOutputPtr
- AllocateOutputBuffer()
- {
- register ConnectionOutputPtr oco;
-
- oco = (ConnectionOutputPtr) fsalloc(sizeof(ConnectionOutput));
- if (!oco)
- return (ConnectionOutputPtr) NULL;
- oco->buf = (unsigned char *) fsalloc(BUFSIZE);
- if (!oco->buf) {
- fsfree(oco);
- return (ConnectionOutputPtr) NULL;
- }
- oco->size = BUFSIZE;
- oco->count = 0;
- return oco;
- }
-
-
- void
- FreeOsBuffers(oc)
- OsCommPtr oc;
- {
- register ConnectionInputPtr oci;
- register ConnectionOutputPtr oco;
-
- if (AvailableInput == oc)
- AvailableInput = (OsCommPtr) NULL;
- if ((oci = oc->input) != (ConnectionInputPtr) 0) {
- if (FreeInputs) {
- fsfree(oci->buffer);
- fsfree(oci);
- } else {
- FreeInputs = oci;
- oci->next = (ConnectionInputPtr) NULL;
- oci->bufptr = oci->buffer;
- oci->bufcnt = 0;
- oci->lenLastReq = 0;
- }
- }
- if ((oco = oc->output) != (ConnectionOutputPtr) 0) {
- if (FreeOutputs) {
- fsfree(oco->buf);
- fsfree(oco);
- } else {
- FreeOutputs = oco;
- oco->next = (ConnectionOutputPtr) NULL;
- oco->count = 0;
- }
- }
- }
-
- void
- ResetOsBuffers()
- {
- register ConnectionInputPtr oci;
- register ConnectionOutputPtr oco;
-
- while ((oci = FreeInputs) != (ConnectionInputPtr) 0) {
- FreeInputs = oci->next;
- fsfree(oci->buffer);
- fsfree(oci);
- }
- while ((oco = FreeOutputs) != (ConnectionOutputPtr) 0) {
- FreeOutputs = oco->next;
- fsfree(oco->buf);
- fsfree(oco);
- }
- }
-