ftp.nice.ch/pub/next/text/tex/teTeX/distrib/sources/teTeX-src-0.4.tar.gz#/teTeX-src-0.4/kpse-2.6/xdvik/psnews.c

This is psnews.c in view mode; [Download] [Up]

/*
 * Copyright (c) 1994 Paul Vojta.  All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in the
 *    documentation and/or other materials provided with the distribution.
 *
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 * SUCH DAMAGE.
 *
 * NOTES:
 *	This code was originally written by Ricardo Telichevesky
 *	(ricardo@rle-vlsi-mit.edu) and Luis Miguel Silveira
 *	(lms@rle-vlsi-mit.edu).
 *	It was largely influenced by similar code in the SeeTeX/XTeX
 *	package by Dirk Grunwald (grunwald@colorado.edu).
 */

/* ||| To do:
 *	ALWAYS_CLOSE_SERVER_CONNECTION?
 *	Is there some way of interrupting a process?
 *	fork
 *	extra bytes on input
 */

#ifdef PS_NEWS /* whole file */

#include "config.h"
#include <signal.h>
#include <X11/X.h>
#include <X11/Xlib.h>
#undef SYSV /* To avoid defined SYSV_{WAIT,UCONTEXT} in xview/notify.h.  */
#include <NeWS/psio.h>
#include <xvps/pscanvas.h>

/* if POSIX O_NONBLOCK is not available, use O_NDELAY */
#if !defined(O_NONBLOCK) && defined(O_NDELAY)
#define	O_NONBLOCK O_NDELAY
#endif

/* Condition for retrying a write */
#include <errno.h>
#ifdef	EWOULDBLOCK
#ifdef	EAGAIN
#define	AGAIN_CONDITION	(errno == EWOULDBLOCK || errno == EAGAIN)
#else	/* EAGAIN */
#define	AGAIN_CONDITION	(errno == EWOULDBLOCK)
#endif	/* EAGAIN */
#else	/* EWOULDBLOCK */
#ifdef	EAGAIN
#define	AGAIN_CONDITION	(errno == EAGAIN)
#endif	/* EAGAIN */
#endif	/* EWOULDBLOCK */

#ifdef	STREAMSCONN
#include <poll.h>
#endif

#if	HAS_SIGIO
#include <fcntl.h>
#include <signal.h>
#ifndef	FASYNC
#undef	HAS_SIGIO
#define	HAS_SIGIO 0
#endif
#endif


#define	Fprintf	(void) fprintf


/* define ALWAYS_CLOSE_SERVER_CONNECTION if you want to close the server
   connection all the time */
#undef	ALWAYS_CLOSE_SERVER_CONNECTION


		/*
		 * Some setup code.
		 */
static	_Xconst	char	str0[]	= "\
/OW2? version cvi 2 eq def \
OW2? \
{ /setlinewidth { pop } def} \
{ /NeWS 3 0 findpackage beginpackage \
  /X11 3 0 findpackage beginpackage} \
ifelse \
currentcanvas /Color get \
currentcanvas /Colormap get getcubedescription null eq and \
   {8 {{currentcanvas /Colormap get 1 index dup dup dup newcube} stopped \
    {pop pop pop pop pop} {exit} ifelse \
    2 div cvi dup 1 eq {exit} if} loop pop} \
if\n";
		/*
		 * This string reads chunks (delimited by %%xdvimark).
		 * The first character of a chunk tells whether a given chunk
		 * is to be done within save/restore or not.
		 * The `H' at the end tells it that the first group is a
		 * header; i.e., no save/restore.
		 */
static	_Xconst	char	preamble[]	= "\
/xdvi$line 81 string def \
/xdvi$run {$error null ne {$error /newerror false put} if \
 {currentfile cvx stopped \
 $error null eq {false} {$error /newerror get} ifelse and \
 {handleerror} if} stopped pop} def \
/xdvi$dslen countdictstack def \
{currentfile read not {exit} if 72 eq \
    {xdvi$run} \
    {/xdvi$sav save def xdvi$run \
      clear countdictstack xdvi$dslen sub {end} repeat xdvi$sav restore} \
  ifelse \
  {(%%xdvimark) currentfile xdvi$line {readline} stopped \
    {clear} {{eq {false exit} if} {true exit} ifelse} ifelse }loop {exit} if \
  58 tagprint flush \
}loop\nH";

extern	_Xconst	char	psheader[];
extern	int	psheaderlen;

static	_Xconst	char	preamble2[]	= " stop\n%%xdvimark\n";
#define	stopstring	preamble2

#define	postscript	resource._postscript


/* global procedures (besides initNeWS) */

static	void	toggleNeWS ARGS((void));
static	void	destroyNeWS ARGS((void));
static	void	interruptNeWS ARGS((void));
static	void	endpageNeWS ARGS((void));
static	void	drawbeginNeWS ARGS((Drawable, int, int, int, int, double,
				char *));
static	void	drawrawNeWS ARGS((char *));
static	void	drawfileNeWS ARGS((char *));
static	void	drawendNeWS ARGS((char *));

static	struct psprocs	news_procs = {
	/* toggle */		toggleNeWS,
	/* destroy */		destroyNeWS,
	/* interrupt */		interruptNeWS,
	/* endpage */		endpageNeWS,
	/* drawbegin */		drawbeginNeWS,
	/* drawraw */		drawrawNeWS,
	/* drawfile */		drawfileNeWS,
	/* drawend */		drawendNeWS};

/* signal handler to hairy PostScript code */
static void psio_sigpipe_handler();

/* define local static variables */
static	int	NeWS_mag;		/* magnification currently in use */
static	int	NeWS_shrink;		/* shrink factor currently in use */
static	Boolean	NeWS_active;		/* if we've started a page yet */
static	int	NeWS_pending;		/* number of ack's we're expecting */
static	int	NeWS_sending;		/* level of nesting in send() */
static	Boolean	NeWS_pending_int;	/* if interrupt rec'd while in send() */
static	Boolean	NeWS_destroyed	= False;


/*
 *	NeWS I/O code.  This should send PS code to NeWS,
 *	receive acknowledgements, and receive X events in the meantime.
 *	It also checks for SIGPIPE errors.
 */

#ifndef	STREAMSCONN
static	int		numfds;
static	fd_set		readfds;
static	fd_set		writefds;
#define	XDVI_ISSET(a, b, c)	FD_ISSET(a, b)
#else	/* STREAMSCONN */
struct pollfd		fds[3] = {{0, POLLOUT, 0},
				  {0, POLLIN, 0},
				  {0, POLLIN, 0}};
#define	XDVI_ISSET(a, b, c)	(fds[c].revents)
#endif	/* STREAMSCONN */


/*---------------------------------------------------------------------------*
  psio_sigpipe_handler  ()

  Arguments: sig, code, scp, addr (see man page for signal)
  Returns: (void)
  Side-Effects: SIGPIPE signal is flagged as sigpipe_error variable is set.

  Description:
  Handler for SIGPIPE error generated by a broken pipe in the connection
  to the NeWS server; this may be duer to some abnormal condition, or some
  hairy PostScript code containing commands not implemented by the server.

+----------------------------------------------------------------------------*/

static	Boolean	sigpipe_error = False;

static	struct sigaction psio_sigpipe_handler_struct;
	/* initialized to {psio_sigpipe_handler, (sigset_t) 0, 0} in initNeWS */

/* ARGSUSED */
static	void
psio_sigpipe_handler(sig, code, scp, addr)
	int		sig;
	int		code;
	struct sigcontext *scp;
	char		*addr;
{
	sigpipe_error = True;
}


/*
 *	read_from_NeWS - This does the actual retrieving of acknowledgements.
 *	If other bytes appear on the file - tough.
 */

static	void
read_from_NeWS()
{
	for (;;) {
	    int retval;

	    retval = ps_checkfor(PostScriptInput, PSIO_FIND_TAG, 58);
	    if (retval == 0) break;
	    if (retval < 0) {
		Fprintf(stderr, "ps_checkfor: %d\n", retval);
		return;
	    }
	    (void) ps_checkfor(PostScriptInput, PSIO_WAIT_TAG, 58);
	    --NeWS_pending;
	    if (debug & DBG_PS)
		Printf("Got NeWS ack; %d pending.\n", NeWS_pending);
	}
}


/*
 *	This actually sends the bytes to NeWS.
 */

static	void
send(cp, len)
	_Xconst	char	*cp;
	int		len;
{
	struct sigaction orig;
#ifdef	STREAMSCONN
	int	retval;
#endif

	if (PostScript == (PSFILE *) NULL) return;

	if (!NeWS_sending) {
	    (void) sigaction(SIGPIPE, &psio_sigpipe_handler_struct, &orig);
	    sigpipe_error = False;
	}
	++NeWS_sending;

#if	HAS_SIGIO
	(void) fcntl(ConnectionNumber(DISP), F_SETFL,
	    fcntl(ConnectionNumber(DISP), F_GETFL, 0) & ~FASYNC);
#endif

#ifndef	STREAMSCONN
	FD_ZERO(&readfds);
	FD_ZERO(&writefds);
#endif

	for (;;) {

#ifndef	STREAMSCONN
	    FD_SET(ConnectionNumber(DISP), &readfds);
	    FD_SET(PostScript->file, &writefds);
	    FD_SET(PostScriptInput->file, &readfds);

	    if (select(numfds, &readfds, &writefds, (fd_set *) NULL,
		    (struct timeval *) NULL) < 0 && errno != EINTR) {
		perror("select (NeWS_send)");
		break;
	    }
#else	/* STREAMSCONN */
	    for (;;) {
		retval = poll(fds, XtNumber(fds), -1);
		if (retval >= 0 || errno != EAGAIN) break;
	    }
	    if (retval < 0) {
		perror("poll (NeWS_send)");
		break;
	    }
#endif	/* STREAMSCONN */

	    if (XDVI_ISSET(PostScriptInput->file, &readfds, 1))
		read_from_NeWS();
	    if (XDVI_ISSET(PostScript->file, &writefds, 0)) {
		int	old_flags;
		int	bytes;

		old_flags = fcntl(PostScript->file, F_GETFL, 0);
		if (old_flags < 0) break;
		/* set to be non-blocking */
		if (fcntl(PostScript->file, F_SETFL, old_flags | O_NONBLOCK)
			< 0)
		    break;
		bytes = write(PostScript->file, cp, len);
		if (bytes == -1) {
		    if (!AGAIN_CONDITION) perror("psnews_send");
		}
		else {
		    cp += bytes;
		    len -= bytes;
		}
		if (fcntl(PostScript->file, F_SETFL, old_flags) < 0) break;
		if (len == 0 || sigpipe_error) break;
	    }
	    if (XDVI_ISSET(ConnectionNumber(DISP), &readfds, 2)) {
		allow_can = False;
		read_events(False);
		allow_can = True;
		if (PostScript == (PSFILE *) NULL) break;	/* if timeout occurred */
	    }
	}

#if	HAS_SIGIO
	(void) fcntl(ConnectionNumber(DISP), F_SETFL,
	    fcntl(ConnectionNumber(DISP), F_GETFL, 0) | FASYNC);
#endif

	if (--NeWS_sending == 0) {
	    /* put back generic handler for SIGPIPE */
	    (void) sigaction(SIGPIPE, &orig, (struct sigaction *) NULL);

	    if (sigpipe_error) {
		Fputs("NeWS died unexpectedly.\n", stderr);
		destroyNeWS();
		draw_bbox();
	    }

	    if (NeWS_pending_int) {
		NeWS_pending_int = False;
		interruptNeWS();
	    }
	}
}

/*
 *	Wait for acknowledgement from NeWS.  With NeWS we have no choice but
 *	to wait (||| I think).
 */

static	void
waitack()
{
#ifdef	STREAMSCONN
	int	retval;
#endif
#if	HAS_SIGIO
	int	oldflags;
#endif

	if (PostScript == (PSFILE *) NULL) return;

#if	HAS_SIGIO
	oldflags = fcntl(ConnectionNumber(DISP), F_GETFL, 0);
	(void) fcntl(ConnectionNumber(DISP), F_SETFL, oldflags & ~FASYNC);
#endif
#ifndef	STREAMSCONN
	FD_ZERO(&readfds);
#endif
	while (NeWS_pending > 0) {
#ifndef	STREAMSCONN
	    FD_SET(ConnectionNumber(DISP), &readfds);
	    FD_SET(PostScriptInput->file, &readfds);
	    if (select(numfds, &readfds, (fd_set *) NULL, (fd_set *) NULL,
		    (struct timeval *) NULL) < 0 && errno != EINTR) {
		perror("select (gs_waitack)");
		break;
	    }
#else	/* STREAMSCONN */
	    for (;;) {
		retval = poll(fds + 1, XtNumber(fds) - 1, -1);
		if (retval >= 0 || errno != EAGAIN) break;
	    }
	    if (retval < 0) {
		perror("poll (gs_waitack)");
		break;
	    }
#endif	/* STREAMSCONN */


	    if (XDVI_ISSET(PostScriptInput->file, &readfds, 1))
		read_from_NeWS();
	    if (XDVI_ISSET(ConnectionNumber(DISP), &readfds, 2)) {
		allow_can = False;
		read_events(False);
		allow_can = True;
		if (PostScript == (PSFILE *) NULL) break;	/* if timeout occurred */
	    }
	}
#if	HAS_SIGIO
	(void) fcntl(ConnectionNumber(DISP), F_SETFL, oldflags);
#endif
}


/*---------------------------------------------------------------------------*
  initNeWS()

  Arguments: None.
  Returns: True if and only if initialization succeeded
  Side-Effects: Static variables may be set.

  Description:
  Initializes variables for the application main loop.

+----------------------------------------------------------------------------*/

Boolean
initNeWS()
{
	static	NeWStoken newstoken;

	/* now try to open the connection to the NeWS server */
	if (ps_open_PostScript() == (PSFILE *) NULL)
	    return False;

#ifndef	STREAMSCONN
	numfds = ConnectionNumber(DISP);
	if (numfds < PostScript->file) numfds = PostScript->file;
	if (numfds < PostScriptInput->file) numfds = PostScriptInput->file;
	++numfds;
#else	/* STREAMSCONN */
	fds[0].fd = PostScript->file;
	fds[1].fd = PostScriptInput->file;
	fds[2].fd = ConnectionNumber(DISP);
#endif	/* STREAMSCONN */

	psio_sigpipe_handler_struct.sa_handler = psio_sigpipe_handler;
	sigemptyset(&psio_sigpipe_handler_struct.sa_mask);

	NeWS_active = NeWS_pending_int = False;
	NeWS_sending = 0;
	NeWS_pending = 1;

	ps_flush_PostScript();
	send(str0, sizeof(str0) - 1);
	/* get xid of window, then make this window the NeWS canvas */
	(void) ps_token_from_xid(mane.win, &newstoken);
	if (newstoken != -1) {
	    ps_setcanvas(newstoken);
	    ps_flush_PostScript();
	    send(preamble, sizeof(preamble) - 1);
	    send(psheader, psheaderlen);
	    send(preamble2, sizeof(preamble2) - 1);
	}

	if (NeWS_destroyed) return False;

	/* success */

	NeWS_mag = NeWS_shrink = -1;

	psp = news_procs;
	if (!postscript) toggleNeWS();	/* if we got a 'v' already */

	return True;
}


/*---------------------------------------------------------------------------*
  toggleNeWS()

  Arguments: none
  Returns: (void)
  Side-Effects: psp.drawbegin is changed

  Description:
  Used to toggle the rendering of PostScript by the NeWS server

+----------------------------------------------------------------------------*/

static	void
toggleNeWS()
{
	if (postscript) psp.drawbegin = drawbeginNeWS;
	else {
	    interruptNeWS();
	    psp.drawbegin = drawbegin_none;
	}
}


/*---------------------------------------------------------------------------*
  destroyNeWS()

  Arguments: none
  Returns: (void)
  Side-Effects: the pointer to the NeWS file is nulled

  Description:
  Close the connection to the NeWS server; used when rendering is terminated
  in any way.

+----------------------------------------------------------------------------*/

static	void
destroyNeWS()
{
	psp = no_ps_procs;
	NeWS_destroyed = True;
}


/*---------------------------------------------------------------------------*
  interruptNeWS()

  Arguments: none
  Returns: void

  Description:
  Close the connection to the NeWS server; used when rendering is terminated
  because of an interruption in the viewing of the current page.
  ||| It would be nice if we could asynchronously ``wake up'' a NeWS process
  (preferably by sending something along the X socket); then we could do
  better than just to wait.

+----------------------------------------------------------------------------*/

static	void
interruptNeWS()
{
	if (debug & DBG_PS) Puts("Running interruptNeWS()");
	if (NeWS_sending) NeWS_pending_int = True;
	else {
	    if (NeWS_active) {
		send(stopstring, sizeof(stopstring) - 1);
		NeWS_active = False;
	    }
	    psp.interrupt = NullProc;	/* prevent deep recursion in waitack */
	    waitack();
	    psp.interrupt = interruptNeWS;
	}
}


/*---------------------------------------------------------------------------*
  endpageNeWS()

  Arguments: none
  Returns: (void)
  Side-Effects: the NeWS_active variable is cleared.

  Description:
  Should be called at the end of a page to end this chunk for the NeWS server.

+----------------------------------------------------------------------------*/

static	void
endpageNeWS()
{
	if (debug & DBG_PS)
	    Puts("endpage sent to NeWS Server");
	if (NeWS_active) {
	    send(stopstring, sizeof(stopstring) - 1);
	    NeWS_active = False;
	    waitack();
	}
}


/*---------------------------------------------------------------------------*
  drawbeginNeWS  ()

  Arguments: xul, yul - coordinates of the upper left corner of the figure
	     cp - string with the bounding box line data
  Returns: (void)

  Description:
  Opens a connection to the NeWS server and send in the preamble and the
  bounding box information after correctly computing resolution factors.
  In case no rendering is to be done, outlines the figure.  An outline is
  also generated whenever the PostScript code is too hairy and generates
  a SIGPIPE signal.

+----------------------------------------------------------------------------*/

static	void
drawbeginNeWS(xul, yul, cp)
	int		xul, yul;
	char		*cp;
{
	char	buf[100];
	static	_Xconst	char	str[]	= " TeXDict begin\n";

	if (debug & DBG_PS) {
	    Printf("xul= %d yul= %d\n", xul, yul);
	    Printf("String = < %s >\n", cp);
	}

	/* catch up on the X side */
	XSync(DISP, 0);

	if (!NeWS_active) {
	    /* send initialization to NeWS server */
	    if (magnification != NeWS_mag) {
		Sprintf(buf, "H TeXDict begin /DVImag %d 1000 div def \
end stop\n%%%%xdvimark\n",
		    NeWS_mag = magnification);
		send(buf, strlen(buf));
		++NeWS_pending;
	    }
	    if (mane.shrinkfactor != NeWS_shrink) {
		Sprintf(buf, "H TeXDict begin %d %d div dup \
/Resolution X /VResolution X \
end stop\n%%%%xdvimark\n",
		    pixels_per_inch, NeWS_shrink = mane.shrinkfactor);
		send(buf, strlen(buf));
		++NeWS_pending;
	    }
	    send(str, sizeof(str) - 1);
	    NeWS_active = True;
	    ++NeWS_pending;
	}

	Sprintf(buf, "%d %d moveto\n", xul, yul);
	send(buf, strlen(buf));
	send(cp, strlen(cp));
}


/*---------------------------------------------------------------------------*
  drawrawNeWS()

  Arguments: origcp - the raw string to be sent to the postscript interpreter
  Returns: (void)
  Side-Effects: (none)

  Description:
  If there is a valid connection to the NeWS server, just send the string to
  the interpreter, else leave.

+----------------------------------------------------------------------------*/

static	void
drawrawNeWS(origcp)
	char	*origcp;
{
	char		*pt, *ptm1, *cp1, *ocp1;
	static	char	*cp;
	static	unsigned int cplen = 0;
	unsigned int	len;
	double		angle;
	Boolean		found	= False;

	if (!NeWS_active)
	    return;

	if (debug & DBG_PS)
	    Printf("Raw PS sent to context: <%s>\n", origcp);

	/* take a look at the string:  NeWS bums on certain rotations */
	len = strlen(origcp) + 4;
	if (cplen < len) {
	    if (cplen != 0) free(cp);
	    cplen = len;
	    cp = xmalloc(cplen, "string in drawrawNeWS");
	}
	ocp1 = origcp;
	pt = origcp;
	while (*pt == ' ' || *pt == '\t') ++pt;
	cp1 = cp;
	for (;;) {
	    ptm1 = pt;
	    while (*pt != '\0' && *pt != ' ' && *pt != '\t') ++pt;
	    if (*pt == '\0') break;
	    while (*pt == ' ' || *pt == '\t') ++pt;
	    if (strncmp(pt, "rotate", 6) == 0
		    && (pt[6] == '\0' || pt[6] == ' ' || pt[6] == '\t')) {
		/* found rotate; check angle */
		if (sscanf(ptm1, "%lf", &angle) >= 1) {
		    found = True;
		    while (angle > 360.0)
			angle -= 360;
		    while (angle < -360.0)
			angle += 360;
		    if (angle == 90.0) {
			angle = 89.999;
			(void) memcpy(cp1, ocp1, ptm1 - ocp1);
			cp1 += ptm1 - ocp1;
			Strcpy(cp1, "89.999 rotate ");
			cp1 += strlen(cp1);
			while (*pt != '\0' && *pt != ' ' && *pt != '\t') ++pt;
			while (*pt == ' ' || *pt == '\t') ++pt;
			ocp1 = pt;
		    } else if (angle == -90.0) {
			angle = -89.999;
			(void) memcpy(cp1, ocp1, ptm1 - ocp1);
			cp1 += ptm1 - ocp1;
			Strcpy(cp1, "-89.999 rotate ");
			cp1 += strlen(cp1);
			while (*pt != '\0' && *pt != ' ' && *pt != '\t') ++pt;
			while (*pt == ' ' || *pt == '\t') ++pt;
			ocp1 = pt;
		    } else if (angle == 0.0) {
			(void) memcpy(cp1, ocp1, ptm1 - ocp1);
			cp1 += ptm1 - ocp1;
			while (*pt != '\0' && *pt != ' ' && *pt != '\t') ++pt;
			while (*pt == ' ' || *pt == '\t') ++pt;
			ocp1 = pt;
		    }
		}
	    }
	}
	Strcpy(cp1, ocp1);
	if ((debug & DBG_PS) && found) {
	    Printf("String is now <%s>\n", cp);
	    Printf("Found rotate string.  Angle is %g degrees.\n", angle);
	}

	len = strlen(cp);
	cp[len] = '\n';
	send(cp, len + 1);
}


/*---------------------------------------------------------------------------*
  drawfileNeWS()

  Arguments: cp - string with the postscript file pathname
  Returns: (void)
  Side-Effects: none

  Description:
  Postscript file containing the figure is opened and sent to the NeWS server.
  Figure is outlined in case hairy code produces a SIGPIPE signal.

+----------------------------------------------------------------------------*/

static	void
drawfileNeWS(cp)
	char	*cp;
{
	char		buffer[1025];
	int		blen;
	int		bytes;
	FILE		*psfile;
	struct sigaction orig;

	if (!NeWS_active)
	    return;

#ifndef	VMS
	if ((psfile = xfopen(cp, "r")) == NULL)
#else
	if ((psfile = xfopen(cp, "r", "?")) == NULL)
#endif
	{
	    Fprintf(stderr,"[%%NeWS Server: cannot access file %s%%]\n", cp);
	    draw_bbox();
	} else {
	    if (debug & DBG_PS)
		Printf("printing file %s\n", cp);
	    /* some hairy PS code generates SIGPIPE signals; handle them */
	    (void) sigaction(SIGPIPE, &psio_sigpipe_handler_struct, &orig);
	    sigpipe_error = False;
	    NeWS_sending = 1;
	    if (!sigpipe_error)
		for (;;) {
		    blen = fread(buffer, sizeof(char), 1024, psfile);
		    if (blen == 0) break;
		    send(buffer, blen);
		    if (sigpipe_error) break;
		}
	    (void) fclose(psfile);

	    --NeWS_sending;
	    /* put back generic handler for SIGPIPE */
	    (void) sigaction(SIGPIPE, &orig, (struct sigaction *) NULL);

	    if (sigpipe_error) {
		Fputs("NeWS died unexpectedly.\n", stderr);
		destroyNeWS();
		draw_bbox();
	    }

	    if (NeWS_pending_int) {
		NeWS_pending_int = False;
		interruptNeWS();
	    }
	}
}


/*---------------------------------------------------------------------------*
  drawendNeWS()

  Arguments: cp - string with indication of the end of the special
  Returns: (void)

  Description:
  Sends the indication of end of the figure PostScript code.

+----------------------------------------------------------------------------*/

static	void
drawendNeWS(cp)
	char	*cp;
{
	if (!NeWS_active)
	    return;

	if (debug & DBG_PS)
	    Puts("drawend sent to NeWS Server");
	send(cp, strlen(cp));
	send("\n", 1);
}

#endif /* PS_NEWS */

These are the contents of the former NiCE NeXT User Group NeXTSTEP/OpenStep software archive, currently hosted by Netfuture.ch.