ftp.nice.ch/pub/next/unix/editor/elvis-2.0.N.bs.tar.gz#/elvis-2.0.N.bs/osunix/tcaposix.h

This is tcaposix.h in view mode; [Download] [Up]

/* tcaposix.c */

char id_tcaposix[] = "$Id: tcaposix.h,v 2.8 1996/10/01 19:49:01 steve Exp $";

#include <termios.h>
#include <signal.h>
#include <unistd.h>

/* HPUX does a "#define ttysize winsize".  Elvis doesn't like that. */
#undef ttysize


#if USE_PROTOTYPES
static void catchsig(int signo);
#endif


static struct termios	oldtermio;	/* original tty mode */
static struct termios	newtermio;	/* cbreak/noecho tty mode */


/* this function is used to catch signals */
static void catchsig(signo)
	int	signo;
{
	caught = (1 << signo);
}

/* get the original tty state */
void ttyinit()
{
	/* get the old tty state */
	tcgetattr(0, &oldtermio);
}

/* switch to the tty state that elvis runs in */
void ttyraw(erasekey)
	char	*erasekey;	/* where to store the ERASE char */
{
#ifdef SA_NOMASK
	struct sigaction new;

	/* arrange for signals to be caught */
	new.sa_handler = catchsig;
	new.sa_flags = (SA_INTERRUPT|SA_NOMASK); /* not one-shot */
	new.sa_restorer = NULL;
	sigaction(SIGHUP, &new, NULL);
	sigaction(SIGINT, &new, NULL);
# ifdef SIGWINCH
	sigaction(SIGWINCH, &new, NULL);
# endif
#else
	signal(SIGHUP, catchsig);
	signal(SIGINT, catchsig);
# ifdef SIGWINCH
	signal(SIGWINCH, catchsig);
# endif
#endif
	signal(SIGQUIT, SIG_IGN);

	/* switch to raw mode */
	ospeed = cfgetospeed(&oldtermio);
	*erasekey = oldtermio.c_cc[VERASE];
	newtermio = oldtermio;
	newtermio.c_iflag &= (IXON|IXOFF|ISTRIP|IGNBRK);
	newtermio.c_oflag &= ~OPOST;
	newtermio.c_lflag &= ISIG;
	newtermio.c_cc[VINTR] = ELVCTRL('C'); /* always use ^C for interrupts */
#ifdef NDEBUG
	newtermio.c_cc[VQUIT] = 0;
#endif
	newtermio.c_cc[VMIN] = 1;
	newtermio.c_cc[VTIME] = 0;
#  ifdef VSWTCH
	newtermio.c_cc[VSWTCH] = 0;
#  endif
#  ifdef VSWTC /* is this a bug in Linux's headers? */
	newtermio.c_cc[VSWTC] = 0;
#  endif
#  ifdef VSUSP
	newtermio.c_cc[VSUSP] = 0;
#  endif
#  ifdef VDSUSP
	newtermio.c_cc[VDSUSP] = 0;
#  endif
	tcsetattr(0, TCSADRAIN, &newtermio);
}

/* switch back to the original tty state */
void ttynormal()
{
	tcsetattr(0, TCSADRAIN, &oldtermio);
}

/* Read from keyboard with timeout.  For POSIX, we use VMIN/VTIME to implement
 * the timeout.  For no timeout, VMIN should be 1 and VTIME should be 0; for
 * timeout, VMIN should be 0 and VTIME should be the timeout value.
 */
int ttyread(buf, len, timeout)
	char	*buf;	/* where to place the read characters */
	int	len;	/* maximum number of characters to read */
	int	timeout;/* timeout (0 for none) */
{
	struct termios t, oldt;
	int	bytes;	/* number of bytes actually read */

	/* clear the "caught" variable */
	caught = 0;

#ifndef SA_NOMASK
	/* make sure the signal catcher hasn't been reset */
	signal(SIGHUP, catchsig);
	signal(SIGINT, catchsig);
# ifdef SIGWINCH
	signal(SIGWINCH, catchsig);
# endif
#endif

	/* arrange for timeout, and disable control chars */
	tcgetattr(0, &t);
	oldt = t;
	if (timeout)
	{
		t.c_cc[VMIN] = 0;
		t.c_cc[VTIME] = timeout;
	}
	else
	{
		t.c_cc[VMIN] = 1;
		t.c_cc[VTIME] = 0;
	}
	t.c_cc[VINTR] = t.c_cc[VQUIT] = t.c_cc[VSTART] = t.c_cc[VSTOP] = 0;
	tcsetattr(0, TCSANOW, &t);

	/* Perform the read. */
	bytes = read(0, buf, (unsigned)len);

	/* revert to previous mode */
	tcsetattr(0, TCSANOW, &oldt);

	/* return the number of bytes read */
	return bytes;
}

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