DataMuseum.dk

Presents historical artifacts from the history of:

DKUUG/EUUG Conference tapes

This is an automatic "excavation" of a thematic subset of
artifacts from Datamuseum.dk's BitArchive.

See our Wiki for more about DKUUG/EUUG Conference tapes

Excavated with: AutoArchaeologist - Free & Open Source Software.


top - metrics - download
Index: T i

⟦490431570⟧ TextFile

    Length: 20742 (0x5106)
    Types: TextFile
    Names: »io.c«

Derivation

└─⟦b20c6495f⟧ Bits:30007238 EUUGD18: Wien-båndet, efterår 1987
    └─⟦this⟧ »EUUGD18/General/Ularn/io.c« 

TextFile

/* io.c	
 *
 *	Below are the functions in this file:
 *
 *	setupvt100() 	Subroutine to set up terminal in correct mode for game
 *	clearvt100()  	Subroutine to clean up terminal when the game is over
 *	getcharacter() 	Routine to read in one character from the terminal
 *	scbr()			Function to set cbreak -echo for the terminal
 *	sncbr()			Function to set -cbreak echo for the terminal
 *	newgame() 	Subroutine to save the initial time and seed rnd()
 *
 *	FILE OUTPUT ROUTINES
 *
 *	lprintf(format,args . . .)	printf to the output buffer
 *	lprint(integer)			send binary integer to output buffer
 *	lwrite(buf,len)			write a buffer to the output buffer
 *	lprcat(str)			sent string to output buffer
 *
 *	FILE OUTPUT MACROS (in header.h)
 *
 *	lprc(character)			put the character into the output buffer
 *
 *	FILE INPUT ROUTINES
 *
 *	long lgetc()			read one character from input buffer
 *	long lrint()			read one integer from input buffer
 *	lrfill(address,number)		put input bytes into a buffer
 *	char *lgetw()			get a whitespace ended word from input
 *	char *lgetl()			get a \n or EOF ended line from input
 *
 *	FILE OPEN / CLOSE ROUTINES
 *
 *	lcreat(filename)		create a new file for write
 *	lopen(filename)			open a file for read
 *	lappend(filename)		open for append to an existing file
 *	lrclose()			close the input file
 *	lwclose()			close output file
 *	lflush()			flush the output buffer
 *
 *	Other Routines
 *
 *	cursor(x,y)		position cursor at [x,y]
 *	cursors()		position cursor at [1,24] (saves memory)
 *  cl_line(x,y)         	Clear line at [1,y] and leave cursor at [x,y]
 *  cl_up(x,y)    		Clear screen from [x,1] to current line.
 *  cl_dn(x,y) 			Clear screen from [1,y] to end of display. 
 *  standout(str)	 	Print the string in standout mode.
 *  set_score_output() 		Called when output should be literally printed.
 ** putcharacter(ch)		Print one character in decoded output buffer.
 ** flush_buf()			Flush buffer with decoded output.
 ** init_term()			Terminal initialization -- setup termcap info
 **char *tmcapcnv(sd,ss)  	Routine to convert vt100 \33's to termcap format
 *	beep() 	  Routine to emit a beep if enabled (see no-beep in .Ularnopts)
 *
 * Note: ** entries are available only in termcap mode.
 */
#include "header.h"

#ifdef BSD
#    define	GTTY(arg)	(ioctl(0, TIOCGETP, arg))
#    define	STTY(arg)	(ioctl(0, TIOCSETP, arg))
	static struct sgttyb inittyb, curttyb;
#else /* SYSV */
#    define GTTY(arg)		(ioctl(0, TCGETA, arg))
#    define STTY(arg)		(ioctl(0, TCSETAW, arg))
#    define SPEED(x)	((x).c_cflag & CBAUD)
	static struct termio inittyb, curttyb;
#endif /* BSD */

#define	ON	1
#define OFF	0

extern short ospeed;

#define LINBUFSIZE 128		/* size of the lgetw() and lgetl() buffer */
int lfd;			/*  output file numbers	*/
int fd;				/*  input file numbers	*/
static int ipoint=MAXIBUF,iepoint=MAXIBUF;	/*  input buffering pointers  */
static char lgetwbuf[LINBUFSIZE];	/* get line (word) buffer*/

/*
 *	setupvt100() 		
 *		Subroutine to set up terminal in correct mode for game
 *	Attributes off, clear screen, set scrolling region, set tty mode 
 */
setupvt100()
{
	clear();  
	setscroll();  
	gettty();
}

/*
 *	clearvt100() 	 	
 *		Subroutine to clean up terminal when the game is over
 *	Attributes off, clear screen, unset scrolling region, restore tty mode 
 */
clearvt100()
{
	resetscroll();  
	clear();  
	settty();
}

/*
**	gettty
**
** Get initial state of terminal, set ospeed (for termcap routines)
** and switch off tab expansion
*/
gettty()
{
	if(GTTY(&inittyb) < 0) {
		fprintf(stderr, "OOPS: gettty failed\n");
		fflush(stderr);
		return;
	}
	curttyb = inittyb;
#ifndef BSD
	ospeed = SPEED(inittyb);
	/* do not expand tabs - they might be needed inside a cm sequence */
	if(curttyb.c_oflag & TAB3) {
		curttyb.c_oflag &= ~TAB3;
		setctty();
	}
#endif /* BSD */
	setuptty();
}

/*
**	settty
**
** 		reset terminal to original state 
*/
settty() 
{
	if(STTY(&inittyb) < 0) {
		fprintf(stderr, "OOPS: settty failed\n");
		fflush(stderr);
	}
}

/*
**	setctty
**
**		set current tty
*/
setctty(){
	if(STTY(&curttyb) < 0) {
		fprintf(stderr, "OOPS: setctty failed\n");
		fflush(stderr);
	}
}


/*
**
**	function to setup all required terminal modes for game
*/
setuptty(){
	int change = 0;
#ifdef BSD
	scbr();
#else /* SYSV */
	if((curttyb.c_lflag & ECHO) != OFF){
		curttyb.c_lflag &= ~ECHO;
		change++;
	}
	if((curttyb.c_lflag & ICANON) != !(ICANON)){
		curttyb.c_lflag &= ~ICANON;
		curttyb.c_lflag |= !(ICANON);
		/* be satisfied with one character; no timeout */
		curttyb.c_cc[VMIN] = 1;		/* was VEOF */
		curttyb.c_cc[VTIME] = 0;	/* was VEOL */
		change++;
	}
	if(change)
		setctty();
#endif /* BSD */
}

/*
 *	scbr()		Function to set cbreak -echo for the terminal
 *
 *	like: system("stty cbreak -echo")
 */
scbr()
{
#ifdef BSD
	curttyb.sg_flags |= CBREAK;
	curttyb.sg_flags &= ~ECHO;
#else /* SYSV */
	curttyb.c_lflag &= ~ECHO;
	curttyb.c_lflag &= ~ICANON;
#endif /* BSD */
	setctty();
}

/*
 *	sncbr()		Function to set -cbreak echo for the terminal
 *
 *	like: system("stty -cbreak echo")
 */
sncbr()
{
#ifdef BSD
 	curttyb.sg_flags &= ~CBREAK;
	curttyb.sg_flags |= ECHO;
#else /* SYSV */
	curttyb.c_lflag |= ECHO;
	curttyb.c_lflag |= ICANON;
#endif /* BSD */
	setctty();
}

/*
 *	getcharacter() 	Routine to read in one character from the terminal
 */
getcharacter()
{
	char byt;

	lflush();		/* be sure output buffer is flushed */
	read(0,&byt,1); 	/* get byte from terminal */
	return(byt);
}

/*
 *	newgame() 		
 *		Subroutine to save the initial time and seed rnd()
 */
newgame()
{
	register long *p,*pe;

	for (p=c,pe=c+100; p<pe; *p++ =0)
		;
	time(&initialtime);		
	srand(initialtime);
	lcreat((char*)0);	/* open buffering for output to terminal */
}

/*
 *	lprintf(format,args . . .)		printf to the output buffer
 *		char *format;
 *		??? args . . .
 *
 *	Enter with the format string in "format", as per printf() usage
 *		and any needed arguments following it
 *Note: lprintf() only supports %s, %c and %d, with width modifier and left
 *	or right justification.
 *	No correct checking for output buffer overflow is done, but flushes 
 *		are done beforehand if needed.
 *	Returns nothing of value.
 */
/*VARARGS*/
lprintf(va_alist)
va_dcl
{
	va_list ap;	/* pointer for variable argument list */
	char *fmt;
	char *outb,*tmpb;
	long wide,left,cont,n;		/* data for lprintf	*/
	char db[12];			/* %d buffer in lprintf	*/

	va_start(ap);	/* initialize the var args pointer */
	fmt = va_arg(ap, char *);	/* pointer to format string */
	if (lpnt >= lpend) lflush(); 
	outb = lpnt;
	for ( ; ; ) {
		while (*fmt != '%')
			if (*fmt) *outb++ = *fmt++;  
			else { 
				lpnt=outb;  
				return; 
			}
		wide = 0;	
		left = 1;	
		cont=1;
		while (cont)
			switch(*(++fmt)) {
			case 'd':	
				n = va_arg(ap, long);
				if (n<0) { 
					n = -n;  
					*outb++ = '-';  
					if (wide) --wide; 
				}
				tmpb = db+11;	
				*tmpb = (char)(n % 10 + '0');
				while (n>9)  
					*(--tmpb) = (char)((n /= 10) % 10+'0');
				if (wide==0)  
					while (tmpb < db+12) 
						*outb++ = *tmpb++;
				else {
					wide -= db-tmpb+12;
					if (left)  
						while (wide-- > 0) 
							*outb++ = ' ';
					while (tmpb < db+12) 
						*outb++ = *tmpb++;
					if (left==0)  
						while (wide-- > 0) 
							*outb++ = ' ';
				}
				cont=0;	
				break;

			case 's':	
				tmpb = va_arg(ap, char *);
				if (wide==0)  { 
					while (*outb++ = *tmpb++);  
					--outb; 
				} 
				else {
					n = wide - strlen(tmpb);
					if (left)  
						while (n-- > 0) 
							*outb++ = ' ';
					while (*outb++ = *tmpb++)
						;  
					--outb;
					if (left==0)  
						while (n-- > 0) 
							*outb++ = ' ';
				}
				cont=0;	
				break;

			case 'c':	
				*outb++ = va_arg(ap, int);	
				cont=0;  
				break;

			case '0':
			case '1':
			case '2':
			case '3':
			case '4':
			case '5':
			case '6':
			case '7':
			case '8':
			case '9':	
				wide = 10*wide + *fmt - '0';	
				break;
			case '-':	
				left = 0;	
				break;
			default:	
				*outb++ = *fmt;  
				cont=0;	
				break;
			};
		fmt++;
	}
	va_end(ap);
}

/*
 *	lprint(long-integer)				
 *		long integer;
 *
 *		send binary integer to output buffer
 *
 *	+---------+---------+---------+---------+
 *	|   high  |	    |	      |	  low	|
 *	|  order  |	    |	      |  order	|
 *	|   byte  |	    |	      |	  byte	|
 *	+---------+---------+---------+---------+
 *	 31 --- 24 23 --- 16 15 ---  8 7  ---  0
 *
 *	The save order is low order first, to high order (4 bytes total)
 *		and is written to be system independent.
 *	No checking for output buffer overflow is done, but flushes if needed!
 *	Returns nothing of value.
 */
lprint(x)
long x;
{
	if (lpnt >= lpend) 
		lflush();
	*lpnt++ =  255 & x;			
	*lpnt++ =  255 & (x>>8);
	*lpnt++ =  255 & (x>>16);	
	*lpnt++ =  255 & (x>>24);
}

/*
 *	lwrite(buf,len)					
 *		char *buf;
 *		int len;
 *	
 *		write a buffer to the output buffer
 *
 *	Enter with the address and number of bytes to write out
 *	Returns nothing of value
 */
lwrite(buf,len)
char *buf;
int len;
{
	register char *str;
	register int num2;

	if (len > 399)  /* don't copy data if can just write it */
	{
		for (str=buf;  len>0; --len)
			lprc(*str++);
		lflush();
		write(lfd,buf,len);
	} 
	else while (len) {
		if (lpnt >= lpend) 
			lflush();	/* if buffer is full flush it	*/

		/*	# bytes left in output buffer	*/
		num2 = lpbuf+BUFBIG-lpnt;	

		if (num2 > len) num2=len;
		str = lpnt;  
		len -= num2;
		while (num2--)  *str++ = *buf++;	/* copy in the bytes */
		lpnt = str;
	}
}

/*
 *	long lgetc()			Read one character from input buffer
 *
 *  Returns 0 if EOF, otherwise the character
 */
long lgetc()
{
	int i;

	if (ipoint != iepoint)  
		return(inbuffer[ipoint++]);
	if (iepoint!=MAXIBUF)   
		return(0);
	if ((i=read(fd,inbuffer,MAXIBUF))<=0) {
		if (i!=0)  
			fprintf(stderr,"error reading from input file\n");
		iepoint = ipoint = 0;		
		return(0);
	}
	ipoint=1;  
	iepoint=i;  
	return(*inbuffer);
}

/*
 *	long lrint()			Read one integer from input buffer
 *
 *		+---------+---------+---------+---------+
 *		|  high   |	    |	      |   low   |
 *		|  order  |	    |	      |  order	|
 *		|   byte  |	    |	      |	  byte	|
 *		+---------+---------+---------+---------+
 *	        31  ---  24 23 --- 16 15 ---  8 7  ---   0
 *
 *	The save order is low order first, to high order (4 bytes total)
 *	Returns the int read
 */
long lrint()
{
	unsigned long i;

	i  = 255 & lgetc();				
	i |= (255 & lgetc()) << 8;
	i |= (255 & lgetc()) << 16;		
	i |= (255 & lgetc()) << 24;
	return(i);
}

/*
 *	lrfill(address,number)			put input bytes into a buffer
 *		char *address;
 *		int number;
 *
 *	Reads "number" bytes into the buffer pointed to by "address".
 *	Returns nothing of value
 */
lrfill(adr,num)
char *adr;
int num;
{
	char *pnt;
	int num2;

	while (num) {
		if (iepoint == ipoint) {
			if (num>5) {
				if (read(fd,adr,num) != num)
					fprintf(stderr,"error reading from input file\n");
				num=0;
			}
			else { 
				*adr++ = lgetc();  
				--num; 
			}
		}
		else {
		/*	# of bytes left in the buffer	*/
			num2 = iepoint-ipoint;	

			if (num2 > num) 
				num2=num;
			pnt = inbuffer+ipoint;	
			num -= num2;  
			ipoint += num2;
			while (num2--)  
				*adr++ = *pnt++;
		}
	}
}

/*
 *	char *lgetw()			Get a whitespace ended word from input
 *
 *	Returns pointer to a buffer that contains word.  If EOF, returns a 0
 */
char *lgetw()
{
	char *lgp,cc;
	int n=LINBUFSIZE,quote=0;

	lgp = lgetwbuf;

	do 
		cc=lgetc();
	while ((cc <= ' ') && (cc > 0));  /* eat whitespace */

	for ( ; ; --n,cc=lgetc()) {
		if ((cc==0) && (lgp==lgetwbuf))  return(0);	/* EOF */
		if ((n<=1) || ((cc<=32) && (quote==0))) { 
			*lgp=0; 
			return(lgetwbuf); 
		}
		if (cc != '"') *lgp++ = cc;   
		else quote ^= 1;
	}
}

/*
 *	char *lgetl()	
 * 		Function to read in a line ended by newline or EOF
 *
 *Returns pointer to a buffer that contains the line.  If EOF, returns 0
 */
char *lgetl()
{
	int i=LINBUFSIZE,ch;
	char *str=lgetwbuf;

	for ( ; ; --i) {
		if ((*str++ = ch = lgetc()) == 0) {
			if (str == lgetwbuf+1)  return(0); /* EOF */
ot:	
			*str = 0;	
			return(lgetwbuf);	/* line ended by EOF */
		}
		if ((ch=='\n') || (i<=1))  
			goto ot; /* line ended by \n */
	}
}

/*
 *	lcreat(filename)			Create a new file for write
 *		char *filename;
 *
 *	lcreat((char*)0); means to the terminal
 *	Returns -1 if error, otherwise the file descriptor opened.
 */
lcreat(str)
char *str;
{
	lpnt = lpbuf;	
	lpend = lpbuf+BUFBIG;
	if (str==0) return(lfd=1);
	if ((lfd=creat(str,0644)) < 0) {
		lfd=1; 
		lprintf("error creating file <%s>\n",str); 
		lflush(); 
		return(-1);
	}
	return(lfd);
}

/*
 *	lopen(filename)			Open a file for read
 *		char *filename;
 *
 *	lopen(0) means from the terminal
 *	Returns -1 if error, otherwise the file descriptor opened.
 */
lopen(str)
char *str;
{
	ipoint = iepoint = MAXIBUF;
	if (str==0) return(fd=0);
	if ((fd=open(str,0)) < 0) {
		lwclose(); 
		lfd=1; 
		lpnt=lpbuf; 
		return(-1);
	}
	return(fd);
}

/*
 *	lappend(filename)		Open for append to an existing file
 *		char *filename;
 *
 *	lappend(0) means to the terminal
 *	Returns -1 if error, otherwise the file descriptor opened.
 */
lappend(str)
char *str;
{
	lpnt = lpbuf;	
	lpend = lpbuf+BUFBIG;
	if (str==0) return(lfd=1);
	if ((lfd=open(str,2)) < 0) {
		lfd=1; 
		return(-1);
	}
	lseek(lfd,0,2);	/* seek to end of file */
	return(lfd);
}

/*
 *	lrclose()						
 *			close the input file
 *
 *	Returns nothing of value.
 */
lrclose()
{
	if (fd > 0) close(fd);
}

/*
 *	lwclose()			close output file flushing if needed
 *
 *	Returns nothing of value.
 */
lwclose()
{
	lflush();	
	if (lfd > 2) close(lfd);
}

/*
 *	lprcat(string)			append a string to the output buffer
 *					avoids calls to lprintf (time consuming)
 */
lprcat(str)
char *str;
{
	char *str2;

	if (lpnt >= lpend) 
		lflush(); 
	str2 = lpnt;
	while (*str2++ = *str++)
		;
	lpnt = str2 - 1;
}

/*
 * cursor(x,y)	  Put cursor at specified coordinates staring at [1,1] (termcap)
 */
cursor (x,y)
int x,y;
{
	if (lpnt >= lpend) lflush ();

	*lpnt++ = CURSOR;		
	*lpnt++ = x;		
	*lpnt++ = y;
}

/*
 *	Routine to position cursor at beginning of 24th line
 */
cursors()
{
	cursor(1,24);
}

/*
 * Warning: ringing the bell is control code 7. Don't use in defines.
 * Don't change the order of these defines.
 * Also used in helpfiles. Codes used in helpfiles should be \E[1 to \E[7 with
 * obvious meanings.
 */

static char cap[256];
char *CM, *CE, *CD, *CL, *SO, *SE, *AL, *DL;/* Termcap capabilities */
static char *outbuf=0;	/* translated output buffer */

int putcharacter ();

/*
 * init_term()		Terminal initialization -- setup termcap info
 */
init_term()
{
	char termbuf[1024];
	char *capptr = cap+10;
	char *term;

	switch (tgetent(termbuf, term = getenv("TERM"))) {
	case -1: 
		fprintf(stderr, "Cannot open termcap file.\n"); 
		fflush(stderr);
		exit(1);
	case 0: 
		fprintf(stderr, "Cannot find entry of ");
		fprintf(stderr, term);
		fprintf(stderr, " in termcap\n");
		fflush(stderr);
		exit(1);
	};

	CM = tgetstr("cm", &capptr);  /* Cursor motion */
	CE = tgetstr("ce", &capptr);  /* Clear to eoln */
	CL = tgetstr("cl", &capptr);  /* Clear screen */

	/* OPTIONAL */
	AL = tgetstr("al", &capptr);  /* Insert line */
	DL = tgetstr("dl", &capptr);  /* Delete line */
	SO = tgetstr("so", &capptr);  /* Begin standout mode */
	SE = tgetstr("se", &capptr);  /* End standout mode */
	CD = tgetstr("cd", &capptr);  /* Clear to end of display */

	if (!CM)	/* can't find cursor motion entry */
	{
		fprintf(stderr, "Sorry, for a ");		
		fprintf(stderr, term);
		fprintf(stderr, ", I can't find the cursor motion entry in termcap\n");
		fflush(stderr);
		exit(1);
	}
	if (!CE)	/* can't find clear to end of line entry */
	{
		fprintf(stderr, "Sorry, for a ");		
		fprintf(stderr, term);
		fprintf(stderr,", I can't find the clear to end of line entry in termcap\n");
		fflush(stderr);
		exit(1);
	}
	if (!CL)	/* can't find clear entire screen entry */
	{
		fprintf(stderr, "Sorry, for a ");		
		fprintf(stderr, term);
		fprintf(stderr, ", I can't find the clear entire screen entry in termcap\n");
		fflush(stderr);
		exit(1);
	}
	/* get memory for decoded output buffer*/
	if ((outbuf=malloc(BUFBIG+16))==0) {
	  fprintf(stderr,"Error malloc'ing memory for decoded output buffer\n");
		fflush(stderr);
		died(-285);	/* malloc() failure */
	}
}

/*
 * cl_line(x,y)  Clear the whole line indicated by 'y' and leave cursor at [x,y]
 */
cl_line(x,y)
int x,y;
{
	cursor(1,y);		
	*lpnt++ = CL_LINE;		
	cursor(x,y);
}

/*
 * cl_up(x,y) Clear screen from [x,1] to current position. Leave cursor at [x,y]
 */
cl_up(x,y)
register int x,y;
{
	register int i;
	cursor(1,1);
	for (i=1; i<=y; i++)   { 
		*lpnt++ = CL_LINE;  
		*lpnt++ = '\n'; 
	}
	cursor(x,y);
}

/*
 * cl_dn(x,y) 	Clear screen from [1,y] to end of display. Leave cursor at [x,y]
 */
cl_dn(x,y)
int x,y;
{
	int i;

	cursor(1,y);
	if (!CD) {
		*lpnt++ = CL_LINE;
		for (i=y; i<=24; i++) { 
			*lpnt++ = CL_LINE;  
			if (i!=24) *lpnt++ = '\n'; 
		}
		cursor(x,y);
	}
	else
		*lpnt++ = CL_DOWN;
	cursor(x,y);
}

/*
 * standout(str)  Print the argument string in inverse video (standout mode).
 */
standout(str)
register char *str;
{
	if (boldon == 0) {
		lprcat(str);
		return;
	}
	*lpnt++ = ST_START;
	while (*str)
		*lpnt++ = *str++;
	*lpnt++ = ST_END;
}

/*
 * set_score_output() 	Called when output should be literally printed.
 */
set_score_output()
{
	enable_scroll = -1;
}

/*
 *	lflush()					Flush the output buffer
 *
 *	Returns nothing of value.
 *	for termcap version: Flush output in output buffer according to output
 *					status as indicated by `enable_scroll'
 */
static int scrline=18; /* line # for wraparound instead of scrolling if no DL */
lflush ()
{
	int lpoint;
	char *str;
	static int curx = 0;
	static int cury = 0;

	if ((lpoint = lpnt - lpbuf) > 0) {
		if (enable_scroll <= -1) {
			flush_buf();
			if (write(lfd,lpbuf,lpoint) != lpoint) {
			       fprintf(stderr,"error writing to output file\n");
				fflush(stderr);
			}
			lpnt = lpbuf;	/* point back to beginning of buffer */
			return;
		}
		for (str = lpbuf; str < lpnt; str++) {
			if (*str>=32)	{ 
				putcharacter (*str); 
				curx++; 
			}
			else switch (*str) {
			case CLEAR:		
				tputs (CL, 0, putcharacter);		
				curx = cury = 0;
				break;

			case CL_LINE:	
				tputs (CE, 0, putcharacter);
				break;

			case CL_DOWN:	
				tputs (CD, 0, putcharacter);
				break;

			case ST_START:	
				tputs (SO, 0, putcharacter);
				break;

			case ST_END:	
				tputs (SE, 0, putcharacter);
				break;

			case CURSOR:	
				curx = *++str - 1;		
				cury = *++str - 1;
				tputs (tgoto (CM, curx, cury), 0, putcharacter);
				break;
			case '\n':		
				if ((cury == 23) && enable_scroll) {
					if (!DL || !AL) {
						if (++scrline > 23) scrline=19;
						if (++scrline > 23) scrline=19;
						tputs (tgoto (CM, 0, scrline), 0, putcharacter);
						tputs (CE, 0, putcharacter);

						if (--scrline < 19) scrline=23;
						tputs (tgoto (CM, 0, scrline), 0, putcharacter);
						tputs (CE, 0, putcharacter);
					}
					else {
						tputs (tgoto (CM, 0, 19), 0, putcharacter);
						tputs (DL, 0, putcharacter);
						tputs (tgoto (CM, 0, 23), 0, putcharacter);
					}
				}
				else {
					putcharacter ('\n');		
					cury++;
				}
				curx = 0;
				break;

			default:		
				putcharacter (*str); 
				curx++;
			};
		}
	}
	lpnt = lpbuf;
	flush_buf();	/* flush real output buffer now */
}

static int index=0;

/*
 * putcharacter(ch)		Print one character in decoded output buffer.
 */
int putcharacter(c)
int c;
{
	outbuf[index++] = c;
	if (index >= BUFBIG)  
		flush_buf();
}

/*
 * flush_buf()			Flush buffer with decoded output.
 */
flush_buf()
{
	if (index) 
		write(lfd, outbuf, index);
	index = 0;
}

/*
 *	char *tmcapcnv(sd,ss)  
 *		Routine to convert vt100 escapes to termcap format
 *
 *	Processes only the \33[#m sequence (converts . files for termcap use 
 */
char *tmcapcnv(sd,ss)
char *sd,*ss;
{
	int tmstate=0;	/* 0=normal, 1=\33 2=[ 3=# */
	char tmdigit=0;	/* the # in \33[#m */

	while (*ss) {
		switch(tmstate) {
		case 0:	
			if (*ss=='\33')  { 
				tmstate++; 
				break; 
			}
ign:  
			*sd++ = *ss;
ign2: 
			tmstate = 0;
			break;
		case 1: 
			if (*ss!='[') goto ign;
			tmstate++;
			break;
		case 2: 
			if (isdigit(*ss)) { 
				tmdigit= *ss-'0'; 
				tmstate++; 
				break; 
			}
			if (*ss == 'm') { 
				*sd++ = ST_END; 
				goto ign2; 
			}
			goto ign;
		case 3: 
			if (*ss == 'm') {
				if (tmdigit) *sd++ = ST_START;
				else *sd++ = ST_END;
				goto ign2;
			}
		default: 
			goto ign;
		};
		ss++;
	}
	*sd=0; /* 0 terminator */
	return(sd);
}

/*
 *	beep()	Routine to emit a beep if enabled (see no-beep in .Ularnopts)
 */
beep()
{
	if (!nobeep) *lpnt++ = '\7';
}