xref: /titanic_51/usr/src/cmd/lp/model/lp.cat.c (revision de81e71e031139a0a7f13b7bf64152c3faa76698)
17c478bd9Sstevel@tonic-gate /*
27c478bd9Sstevel@tonic-gate  * CDDL HEADER START
37c478bd9Sstevel@tonic-gate  *
47c478bd9Sstevel@tonic-gate  * The contents of this file are subject to the terms of the
5*de81e71eSTim Marsland  * Common Development and Distribution License (the "License").
6*de81e71eSTim Marsland  * You may not use this file except in compliance with the License.
77c478bd9Sstevel@tonic-gate  *
87c478bd9Sstevel@tonic-gate  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
97c478bd9Sstevel@tonic-gate  * or http://www.opensolaris.org/os/licensing.
107c478bd9Sstevel@tonic-gate  * See the License for the specific language governing permissions
117c478bd9Sstevel@tonic-gate  * and limitations under the License.
127c478bd9Sstevel@tonic-gate  *
137c478bd9Sstevel@tonic-gate  * When distributing Covered Code, include this CDDL HEADER in each
147c478bd9Sstevel@tonic-gate  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
157c478bd9Sstevel@tonic-gate  * If applicable, add the following below this CDDL HEADER, with the
167c478bd9Sstevel@tonic-gate  * fields enclosed by brackets "[]" replaced with your own identifying
177c478bd9Sstevel@tonic-gate  * information: Portions Copyright [yyyy] [name of copyright owner]
187c478bd9Sstevel@tonic-gate  *
197c478bd9Sstevel@tonic-gate  * CDDL HEADER END
207c478bd9Sstevel@tonic-gate  */
21*de81e71eSTim Marsland 
227c478bd9Sstevel@tonic-gate /*
23*de81e71eSTim Marsland  * Copyright 2009 Sun Microsystems, Inc.  All rights reserved.
247c478bd9Sstevel@tonic-gate  * Use is subject to license terms.
257c478bd9Sstevel@tonic-gate  */
267c478bd9Sstevel@tonic-gate 
277c478bd9Sstevel@tonic-gate /*	Copyright (c) 1984, 1986, 1987, 1988, 1989 AT&T	*/
287c478bd9Sstevel@tonic-gate /*	  All Rights Reserved  	*/
297c478bd9Sstevel@tonic-gate 
30ace1a5f1Sdp #include <stdio.h>
31ace1a5f1Sdp #include <stdlib.h>
32ace1a5f1Sdp #include <termio.h>
33ace1a5f1Sdp #include <sys/types.h>
34ace1a5f1Sdp #include <errno.h>
35ace1a5f1Sdp #include <signal.h>
36ace1a5f1Sdp #include <sys/times.h>
37ace1a5f1Sdp #include <string.h>
38ace1a5f1Sdp #include <limits.h>
397c478bd9Sstevel@tonic-gate #include <sys/prnio.h>
407c478bd9Sstevel@tonic-gate 
417c478bd9Sstevel@tonic-gate #include "lp.h"
427c478bd9Sstevel@tonic-gate 
437c478bd9Sstevel@tonic-gate #include <locale.h>
447c478bd9Sstevel@tonic-gate 
45*de81e71eSTim Marsland /*
46*de81e71eSTim Marsland  *	Begin Sun Additions for Parallel ports
47*de81e71eSTim Marsland  */
487c478bd9Sstevel@tonic-gate 
497c478bd9Sstevel@tonic-gate #include <string.h>
507c478bd9Sstevel@tonic-gate #include <stdarg.h>
517c478bd9Sstevel@tonic-gate #include <signal.h>
527c478bd9Sstevel@tonic-gate #include <unistd.h>
537c478bd9Sstevel@tonic-gate #include <sys/types.h>
547c478bd9Sstevel@tonic-gate #include <sys/ioccom.h>
557c478bd9Sstevel@tonic-gate #include <sys/ioctl.h>
567c478bd9Sstevel@tonic-gate 
577c478bd9Sstevel@tonic-gate #include <sys/bpp_io.h>
587c478bd9Sstevel@tonic-gate #include <sys/ecppsys.h>
597c478bd9Sstevel@tonic-gate #include <stropts.h>
607c478bd9Sstevel@tonic-gate 
617c478bd9Sstevel@tonic-gate /*
627c478bd9Sstevel@tonic-gate  * the parameter structure for the parallel port
637c478bd9Sstevel@tonic-gate  */
647c478bd9Sstevel@tonic-gate struct ppc_params_t {
657c478bd9Sstevel@tonic-gate 	int		flags;		/* same as above */
667c478bd9Sstevel@tonic-gate 	int		state;		/* status of the printer interface */
677c478bd9Sstevel@tonic-gate 	int		strobe_w;	/* strobe width, in uS */
687c478bd9Sstevel@tonic-gate 	int		data_setup;	/* data setup time, in uS */
697c478bd9Sstevel@tonic-gate 	int		ack_timeout;	/* ACK timeout, in secs */
707c478bd9Sstevel@tonic-gate 	int		error_timeout;	/* PAPER OUT, etc... timeout, in secs */
717c478bd9Sstevel@tonic-gate 	int		busy_timeout;	/* BUSY timeout, in seconds */
727c478bd9Sstevel@tonic-gate };
737c478bd9Sstevel@tonic-gate 
747c478bd9Sstevel@tonic-gate 
757c478bd9Sstevel@tonic-gate 
767c478bd9Sstevel@tonic-gate static void printer_info(char *fmt, ...);
777c478bd9Sstevel@tonic-gate 
787c478bd9Sstevel@tonic-gate /*	These are the routines avaliable to others for use 	*/
797c478bd9Sstevel@tonic-gate int is_a_parallel_bpp(int);
807c478bd9Sstevel@tonic-gate int bpp_state(int);
817c478bd9Sstevel@tonic-gate int parallel_comm(int, int());
827c478bd9Sstevel@tonic-gate int get_ecpp_status(int fd);
837c478bd9Sstevel@tonic-gate int is_a_prnio(int);
847c478bd9Sstevel@tonic-gate int prnio_state(int);
857c478bd9Sstevel@tonic-gate 
867c478bd9Sstevel@tonic-gate #define	PRINTER_ERROR_PAPER_OUT		1
877c478bd9Sstevel@tonic-gate #define	PRINTER_ERROR_OFFLINE		2
887c478bd9Sstevel@tonic-gate #define	PRINTER_ERROR_BUSY		3
897c478bd9Sstevel@tonic-gate #define	PRINTER_ERROR_ERROR		4
907c478bd9Sstevel@tonic-gate #define	PRINTER_ERROR_CABLE_POWER	5
917c478bd9Sstevel@tonic-gate #define	PRINTER_ERROR_UNKNOWN		6
927c478bd9Sstevel@tonic-gate #define	PRINTER_ERROR_TIMEOUT		7
937c478bd9Sstevel@tonic-gate #define	PRINTER_IO_ERROR		129
947c478bd9Sstevel@tonic-gate 
957c478bd9Sstevel@tonic-gate 
96*de81e71eSTim Marsland /*
977c478bd9Sstevel@tonic-gate  *	for BPP PARALLEL interfaces
98*de81e71eSTim Marsland  */
997c478bd9Sstevel@tonic-gate 
100*de81e71eSTim Marsland int
101*de81e71eSTim Marsland is_a_parallel_bpp(int fd)
1027c478bd9Sstevel@tonic-gate {
1037c478bd9Sstevel@tonic-gate 	if (ioctl(fd, BPPIOC_TESTIO) == 0 || errno == EIO)
1047c478bd9Sstevel@tonic-gate 		return (1);
1057c478bd9Sstevel@tonic-gate 	return (0);
1067c478bd9Sstevel@tonic-gate }
1077c478bd9Sstevel@tonic-gate 
1087c478bd9Sstevel@tonic-gate 
1097c478bd9Sstevel@tonic-gate #if defined(DEBUG) && defined(NOTDEF)
110*de81e71eSTim Marsland char *
111*de81e71eSTim Marsland BppState(int state)
1127c478bd9Sstevel@tonic-gate {
1137c478bd9Sstevel@tonic-gate 	static char buf[BUFSIZ];
1147c478bd9Sstevel@tonic-gate 
1157c478bd9Sstevel@tonic-gate 	memset(buf, 0, sizeof (buf));
1167c478bd9Sstevel@tonic-gate 	sprintf(buf, "State (0x%.4x) - (%s%s%s%s)\n", state,
1177c478bd9Sstevel@tonic-gate 	    ((state & BPP_SLCT_ERR) ?  "offline " : ""),
1187c478bd9Sstevel@tonic-gate 	    ((state & BPP_BUSY_ERR) ?  "busy " : ""),
1197c478bd9Sstevel@tonic-gate 	    ((state & BPP_PE_ERR) ?  "paper " : ""),
1207c478bd9Sstevel@tonic-gate 	    ((state & BPP_ERR_ERR) ?  "error " : ""));
1217c478bd9Sstevel@tonic-gate 
1227c478bd9Sstevel@tonic-gate 	return (buf);
1237c478bd9Sstevel@tonic-gate }
1247c478bd9Sstevel@tonic-gate #endif
1257c478bd9Sstevel@tonic-gate 
126*de81e71eSTim Marsland int
127*de81e71eSTim Marsland bpp_state(int fd)
1287c478bd9Sstevel@tonic-gate {
1297c478bd9Sstevel@tonic-gate 	if (ioctl(fd, BPPIOC_TESTIO)) {
1307c478bd9Sstevel@tonic-gate 		struct bpp_error_status  bpp_stat;
1317c478bd9Sstevel@tonic-gate 		int state;
1327c478bd9Sstevel@tonic-gate 
1337c478bd9Sstevel@tonic-gate 		if (ioctl(fd, BPPIOC_GETERR, &bpp_stat) < 0)
1347c478bd9Sstevel@tonic-gate 			exit(PRINTER_IO_ERROR);
1357c478bd9Sstevel@tonic-gate 		state = bpp_stat.pin_status;
1367c478bd9Sstevel@tonic-gate 
1377c478bd9Sstevel@tonic-gate #if defined(DEBUG) && defined(NOTDEF)
1387c478bd9Sstevel@tonic-gate 		logit("%s", BppState(state));
1397c478bd9Sstevel@tonic-gate #endif
1407c478bd9Sstevel@tonic-gate 
1417c478bd9Sstevel@tonic-gate 		if (state == (BPP_PE_ERR | BPP_ERR_ERR | BPP_SLCT_ERR)) {
1427c478bd9Sstevel@tonic-gate 			/* paper is out */
1437c478bd9Sstevel@tonic-gate 			return (PRINTER_ERROR_PAPER_OUT);
1447c478bd9Sstevel@tonic-gate 		} else if (state & BPP_BUSY_ERR) {
1457c478bd9Sstevel@tonic-gate 			/* printer is busy */
1467c478bd9Sstevel@tonic-gate 			return (PRINTER_ERROR_BUSY);
1477c478bd9Sstevel@tonic-gate 		} else if (state & BPP_SLCT_ERR) {
1487c478bd9Sstevel@tonic-gate 			/* printer is offline */
1497c478bd9Sstevel@tonic-gate 			return (PRINTER_ERROR_OFFLINE);
1507c478bd9Sstevel@tonic-gate 		} else if (state & BPP_ERR_ERR) {
1517c478bd9Sstevel@tonic-gate 			/* printer is errored */
1527c478bd9Sstevel@tonic-gate 			return (PRINTER_ERROR_ERROR);
1537c478bd9Sstevel@tonic-gate 		} else if (state == BPP_PE_ERR) {
1547c478bd9Sstevel@tonic-gate 			/* printer is off/unplugged */
1557c478bd9Sstevel@tonic-gate 			return (PRINTER_ERROR_CABLE_POWER);
1567c478bd9Sstevel@tonic-gate 		} else if (state) {
1577c478bd9Sstevel@tonic-gate 			return (PRINTER_ERROR_UNKNOWN);
1587c478bd9Sstevel@tonic-gate 		} else
1597c478bd9Sstevel@tonic-gate 			return (0);
1607c478bd9Sstevel@tonic-gate 	}
1617c478bd9Sstevel@tonic-gate 	return (0);
1627c478bd9Sstevel@tonic-gate }
1637c478bd9Sstevel@tonic-gate 
1647c478bd9Sstevel@tonic-gate /*
1657c478bd9Sstevel@tonic-gate  * For ecpp parallel port
1667c478bd9Sstevel@tonic-gate  */
1677c478bd9Sstevel@tonic-gate 
1687c478bd9Sstevel@tonic-gate int
1697c478bd9Sstevel@tonic-gate get_ecpp_status(int fd)
1707c478bd9Sstevel@tonic-gate {
1717c478bd9Sstevel@tonic-gate 	int state;
1727c478bd9Sstevel@tonic-gate 	struct ecpp_transfer_parms transfer_parms;
1737c478bd9Sstevel@tonic-gate 
1747c478bd9Sstevel@tonic-gate 
1757c478bd9Sstevel@tonic-gate 	if (ioctl(fd, ECPPIOC_GETPARMS, &transfer_parms) == -1) {
1767c478bd9Sstevel@tonic-gate 		return (-1);
1777c478bd9Sstevel@tonic-gate 	}
1787c478bd9Sstevel@tonic-gate 
1797c478bd9Sstevel@tonic-gate 	state = transfer_parms.mode;
1807c478bd9Sstevel@tonic-gate 	/*
1817c478bd9Sstevel@tonic-gate 	 * We don't know what all printers will return in
1827c478bd9Sstevel@tonic-gate 	 * nibble mode, therefore if we support nibble mode we will
1837c478bd9Sstevel@tonic-gate 	 * force the printer to be in CENTRONICS mode.
1847c478bd9Sstevel@tonic-gate 	 */
1857c478bd9Sstevel@tonic-gate 	if (state != ECPP_CENTRONICS) {
1867c478bd9Sstevel@tonic-gate 		transfer_parms.mode = ECPP_CENTRONICS;
1877c478bd9Sstevel@tonic-gate 		if (ioctl(fd, ECPPIOC_SETPARMS, &transfer_parms) == -1) {
1887c478bd9Sstevel@tonic-gate 			return (-1);
1897c478bd9Sstevel@tonic-gate 		} else {
1907c478bd9Sstevel@tonic-gate 			state = ECPP_CENTRONICS;
1917c478bd9Sstevel@tonic-gate 		}
1927c478bd9Sstevel@tonic-gate 	}
1937c478bd9Sstevel@tonic-gate 
1947c478bd9Sstevel@tonic-gate 
1957c478bd9Sstevel@tonic-gate 	return (state);
1967c478bd9Sstevel@tonic-gate }
1977c478bd9Sstevel@tonic-gate 
198*de81e71eSTim Marsland /*
1997c478bd9Sstevel@tonic-gate  * For prnio(7I) - generic printer interface
200*de81e71eSTim Marsland  */
201*de81e71eSTim Marsland int
202*de81e71eSTim Marsland is_a_prnio(int fd)
2037c478bd9Sstevel@tonic-gate {
2047c478bd9Sstevel@tonic-gate 	uint_t	cap;
2057c478bd9Sstevel@tonic-gate 
2067c478bd9Sstevel@tonic-gate 	/* check if device supports prnio */
2077c478bd9Sstevel@tonic-gate 	if (ioctl(fd, PRNIOC_GET_IFCAP, &cap) == -1) {
2087c478bd9Sstevel@tonic-gate 		return (0);
2097c478bd9Sstevel@tonic-gate 	}
2107c478bd9Sstevel@tonic-gate 	/* we will use 1284 status if available */
2117c478bd9Sstevel@tonic-gate 	if ((cap & PRN_1284_STATUS) == 0) {
2127c478bd9Sstevel@tonic-gate 		/* some devices may only support 1284 status in unidir. mode */
2137c478bd9Sstevel@tonic-gate 		if (cap & PRN_BIDI) {
2147c478bd9Sstevel@tonic-gate 			cap &= ~PRN_BIDI;
2157c478bd9Sstevel@tonic-gate 			(void) ioctl(fd, PRNIOC_SET_IFCAP, &cap);
2167c478bd9Sstevel@tonic-gate 		}
2177c478bd9Sstevel@tonic-gate 	}
2187c478bd9Sstevel@tonic-gate 	return (1);
2197c478bd9Sstevel@tonic-gate }
2207c478bd9Sstevel@tonic-gate 
221*de81e71eSTim Marsland int
222*de81e71eSTim Marsland prnio_state(int fd)
2237c478bd9Sstevel@tonic-gate {
2247c478bd9Sstevel@tonic-gate 	uint_t	status;
2257c478bd9Sstevel@tonic-gate 	uchar_t	pins;
2267c478bd9Sstevel@tonic-gate 
2277c478bd9Sstevel@tonic-gate 	if ((ioctl(fd, PRNIOC_GET_STATUS, &status) == 0) &&
2287c478bd9Sstevel@tonic-gate 	    (status & PRN_READY)) {
2297c478bd9Sstevel@tonic-gate 		return (0);
2307c478bd9Sstevel@tonic-gate 	}
2317c478bd9Sstevel@tonic-gate 
2327c478bd9Sstevel@tonic-gate 	if (ioctl(fd, PRNIOC_GET_1284_STATUS, &pins) != 0) {
2337c478bd9Sstevel@tonic-gate 		return (PRINTER_ERROR_UNKNOWN);
2347c478bd9Sstevel@tonic-gate 	}
2357c478bd9Sstevel@tonic-gate 
2367c478bd9Sstevel@tonic-gate 	if ((pins & ~PRN_1284_BUSY) == PRN_1284_PE) {
2377c478bd9Sstevel@tonic-gate 		/* paper is out */
2387c478bd9Sstevel@tonic-gate 		return (PRINTER_ERROR_PAPER_OUT);
2397c478bd9Sstevel@tonic-gate 	} else if (pins == (PRN_1284_PE | PRN_1284_SELECT |
2407c478bd9Sstevel@tonic-gate 	    PRN_1284_NOFAULT | PRN_1284_BUSY)) {
2417c478bd9Sstevel@tonic-gate 		/* printer is off/unplugged */
2427c478bd9Sstevel@tonic-gate 		return (PRINTER_ERROR_CABLE_POWER);
2437c478bd9Sstevel@tonic-gate 	} else if ((pins & PRN_1284_SELECT) == 0) {
2447c478bd9Sstevel@tonic-gate 		/* printer is offline */
2457c478bd9Sstevel@tonic-gate 		return (PRINTER_ERROR_OFFLINE);
2467c478bd9Sstevel@tonic-gate 	} else if ((pins & PRN_1284_NOFAULT) == 0) {
2477c478bd9Sstevel@tonic-gate 		/* printer is errored */
2487c478bd9Sstevel@tonic-gate 		return (PRINTER_ERROR_ERROR);
2497c478bd9Sstevel@tonic-gate 	} else if (pins & PRN_1284_PE) {
2507c478bd9Sstevel@tonic-gate 		/* paper is out */
2517c478bd9Sstevel@tonic-gate 		return (PRINTER_ERROR_PAPER_OUT);
2527c478bd9Sstevel@tonic-gate 	} else if (pins ^ (PRN_1284_SELECT | PRN_1284_NOFAULT)) {
2537c478bd9Sstevel@tonic-gate 		return (PRINTER_ERROR_UNKNOWN);
2547c478bd9Sstevel@tonic-gate 	}
2557c478bd9Sstevel@tonic-gate 
2567c478bd9Sstevel@tonic-gate 	return (0);
2577c478bd9Sstevel@tonic-gate }
2587c478bd9Sstevel@tonic-gate 
259*de81e71eSTim Marsland /*
2607c478bd9Sstevel@tonic-gate  *	Common routines
261*de81e71eSTim Marsland  */
2627c478bd9Sstevel@tonic-gate 
2637c478bd9Sstevel@tonic-gate /*ARGSUSED0*/
2647c478bd9Sstevel@tonic-gate static void
2657c478bd9Sstevel@tonic-gate ByeByeParallel(int sig)
2667c478bd9Sstevel@tonic-gate {
2677c478bd9Sstevel@tonic-gate 	/* try to shove out the EOT */
2687c478bd9Sstevel@tonic-gate 	(void) write(1, "\004", 1);
2697c478bd9Sstevel@tonic-gate 	exit(0);
2707c478bd9Sstevel@tonic-gate }
2717c478bd9Sstevel@tonic-gate 
2727c478bd9Sstevel@tonic-gate 
2737c478bd9Sstevel@tonic-gate /*ARGSUSED0*/
2747c478bd9Sstevel@tonic-gate static void
2757c478bd9Sstevel@tonic-gate printer_info(char *fmt, ...)
2767c478bd9Sstevel@tonic-gate {
2777c478bd9Sstevel@tonic-gate 	char mesg[BUFSIZ];
2787c478bd9Sstevel@tonic-gate 	va_list ap;
2797c478bd9Sstevel@tonic-gate 
2807c478bd9Sstevel@tonic-gate 	va_start(ap, fmt);
2817c478bd9Sstevel@tonic-gate 	vsprintf(mesg, fmt, ap);
2827c478bd9Sstevel@tonic-gate 	va_end(ap);
2837c478bd9Sstevel@tonic-gate /*
284*de81e71eSTim Marsland  *	fprintf(stderr,
285*de81e71eSTim Marsland  *		"%%%%[ PrinterError: %s; source: parallel ]%%%%\n",
286*de81e71eSTim Marsland  *		mesg);
2877c478bd9Sstevel@tonic-gate  */
2887c478bd9Sstevel@tonic-gate 	fprintf(stderr, "%s\n", mesg);
2897c478bd9Sstevel@tonic-gate 	fflush(stderr);
2907c478bd9Sstevel@tonic-gate 	fsync(2);
2917c478bd9Sstevel@tonic-gate 
2927c478bd9Sstevel@tonic-gate }
2937c478bd9Sstevel@tonic-gate 
2947c478bd9Sstevel@tonic-gate static void
2957c478bd9Sstevel@tonic-gate printer_error(int error)
2967c478bd9Sstevel@tonic-gate {
2977c478bd9Sstevel@tonic-gate 	switch (error) {
2987c478bd9Sstevel@tonic-gate 	case -1:
299ace1a5f1Sdp 		printer_info("ioctl(): %s", strerror(errno));
3007c478bd9Sstevel@tonic-gate 		break;
3017c478bd9Sstevel@tonic-gate 	case PRINTER_ERROR_PAPER_OUT:
3027c478bd9Sstevel@tonic-gate 		printer_info("out of paper");
3037c478bd9Sstevel@tonic-gate 		break;
3047c478bd9Sstevel@tonic-gate 	case PRINTER_ERROR_OFFLINE:
3057c478bd9Sstevel@tonic-gate 		printer_info("offline");
3067c478bd9Sstevel@tonic-gate 		break;
3077c478bd9Sstevel@tonic-gate 	case PRINTER_ERROR_BUSY:
3087c478bd9Sstevel@tonic-gate 		printer_info("busy");
3097c478bd9Sstevel@tonic-gate 		break;
3107c478bd9Sstevel@tonic-gate 	case PRINTER_ERROR_ERROR:
3117c478bd9Sstevel@tonic-gate 		printer_info("printer error");
3127c478bd9Sstevel@tonic-gate 		break;
3137c478bd9Sstevel@tonic-gate 	case PRINTER_ERROR_CABLE_POWER:
3147c478bd9Sstevel@tonic-gate 		printer_info("printer powered off or disconnected");
3157c478bd9Sstevel@tonic-gate 		break;
3167c478bd9Sstevel@tonic-gate 	case PRINTER_ERROR_UNKNOWN:
3177c478bd9Sstevel@tonic-gate 		printer_info("unknown error");
3187c478bd9Sstevel@tonic-gate 		break;
3197c478bd9Sstevel@tonic-gate 	case PRINTER_ERROR_TIMEOUT:
3207c478bd9Sstevel@tonic-gate 		printer_info("communications timeout");
3217c478bd9Sstevel@tonic-gate 		break;
3227c478bd9Sstevel@tonic-gate 	default:
3237c478bd9Sstevel@tonic-gate 		printer_info("get_status() failed");
3247c478bd9Sstevel@tonic-gate 	}
3257c478bd9Sstevel@tonic-gate }
3267c478bd9Sstevel@tonic-gate 
3277c478bd9Sstevel@tonic-gate 
3287c478bd9Sstevel@tonic-gate static void
3297c478bd9Sstevel@tonic-gate wait_state(int fd, int get_state())
3307c478bd9Sstevel@tonic-gate {
3317c478bd9Sstevel@tonic-gate 	int state;
3327c478bd9Sstevel@tonic-gate 	int was_faulted = 0;
3337c478bd9Sstevel@tonic-gate 
3347c478bd9Sstevel@tonic-gate 	while (state = get_state(fd)) {
3357c478bd9Sstevel@tonic-gate 		was_faulted = 1;
3367c478bd9Sstevel@tonic-gate 		printer_error(state);
3377c478bd9Sstevel@tonic-gate 		sleep(15);
3387c478bd9Sstevel@tonic-gate 	}
3397c478bd9Sstevel@tonic-gate 
3407c478bd9Sstevel@tonic-gate 	if (was_faulted) {
3417c478bd9Sstevel@tonic-gate 		fprintf(stderr, "printer ok\n");
3427c478bd9Sstevel@tonic-gate 		fflush(stderr);
3437c478bd9Sstevel@tonic-gate 		fsync(2);
3447c478bd9Sstevel@tonic-gate 	}
3457c478bd9Sstevel@tonic-gate }
3467c478bd9Sstevel@tonic-gate 
347*de81e71eSTim Marsland /*
348*de81e71eSTim Marsland  *  end of Sun Additions for parallel port
349*de81e71eSTim Marsland  */
3507c478bd9Sstevel@tonic-gate #define	IDENTICAL(A, B)	(A.st_dev == B.st_dev && A.st_ino == B.st_ino)
3517c478bd9Sstevel@tonic-gate #define	ISBLK(A)	((A.st_mode & S_IFMT) == S_IFBLK)
3527c478bd9Sstevel@tonic-gate #define	ISCHR(A)	((A.st_mode & S_IFMT) == S_IFCHR)
3537c478bd9Sstevel@tonic-gate 
3547c478bd9Sstevel@tonic-gate #define	E_SUCCESS	0
3557c478bd9Sstevel@tonic-gate #define	E_BAD_INPUT	1
3567c478bd9Sstevel@tonic-gate #define	E_BAD_OUTPUT	2
3577c478bd9Sstevel@tonic-gate #define	E_BAD_TERM	3
3587c478bd9Sstevel@tonic-gate #define	E_IDENTICAL	4
3597c478bd9Sstevel@tonic-gate #define	E_WRITE_FAILED	5
3607c478bd9Sstevel@tonic-gate #define	E_TIMEOUT	6
3617c478bd9Sstevel@tonic-gate #define	E_HANGUP	7
3627c478bd9Sstevel@tonic-gate #define	E_INTERRUPT	8
3637c478bd9Sstevel@tonic-gate 
3647c478bd9Sstevel@tonic-gate #define	SAFETY_FACTOR	2.0
3657c478bd9Sstevel@tonic-gate #define	R(F)		(int)((F) + .5)
3667c478bd9Sstevel@tonic-gate #define	DELAY(N, D)	R(SAFETY_FACTOR * ((N) / (double)(D)))
3677c478bd9Sstevel@tonic-gate 
3687c478bd9Sstevel@tonic-gate char			buffer[BUFSIZ];
3697c478bd9Sstevel@tonic-gate 
3707c478bd9Sstevel@tonic-gate void			sighup(),
3717c478bd9Sstevel@tonic-gate 			sigint(),
3727c478bd9Sstevel@tonic-gate 			sigquit(),
3737c478bd9Sstevel@tonic-gate 			sigpipe(),
3747c478bd9Sstevel@tonic-gate 			sigalrm(),
3757c478bd9Sstevel@tonic-gate 			sigterm();
3767c478bd9Sstevel@tonic-gate 
3777c478bd9Sstevel@tonic-gate #if	defined(baudrate)
3787c478bd9Sstevel@tonic-gate #undef	baudrate
3797c478bd9Sstevel@tonic-gate #endif
3807c478bd9Sstevel@tonic-gate 
3817c478bd9Sstevel@tonic-gate int baudrate();
3827c478bd9Sstevel@tonic-gate 
3837c478bd9Sstevel@tonic-gate 
384*de81e71eSTim Marsland int
385*de81e71eSTim Marsland nop(int fd)
386*de81e71eSTim Marsland {
387*de81e71eSTim Marsland 	return (0);
388*de81e71eSTim Marsland }
389*de81e71eSTim Marsland 
3907c478bd9Sstevel@tonic-gate int bpp_state(int);
3917c478bd9Sstevel@tonic-gate 
3927c478bd9Sstevel@tonic-gate 
393*de81e71eSTim Marsland /*
394*de81e71eSTim Marsland  * main()
395*de81e71eSTim Marsland  */
3967c478bd9Sstevel@tonic-gate 
397f928ce67Sceastha int
398f928ce67Sceastha main(int argc, char *argv[])
3997c478bd9Sstevel@tonic-gate {
400*de81e71eSTim Marsland 	int	nin, nout, effective_rate, max_delay = 0, n;
4017c478bd9Sstevel@tonic-gate 	int	report_rate;
4027c478bd9Sstevel@tonic-gate 	short	print_rate;
403*de81e71eSTim Marsland 	struct stat	in, out;
4047c478bd9Sstevel@tonic-gate 	struct tms	tms;
405*de81e71eSTim Marsland 	long	epoch_start, epoch_end;
4067c478bd9Sstevel@tonic-gate 	char	*TERM;
4077c478bd9Sstevel@tonic-gate 	int	(*func)(int fd);
4087c478bd9Sstevel@tonic-gate 
4097c478bd9Sstevel@tonic-gate 	/*
4107c478bd9Sstevel@tonic-gate 	 * The Spooler can hit us with SIGTERM for three reasons:
4117c478bd9Sstevel@tonic-gate 	 *
4127c478bd9Sstevel@tonic-gate 	 *	- the user's job has been canceled
4137c478bd9Sstevel@tonic-gate 	 *	- the printer has been disabled while we were printing
4147c478bd9Sstevel@tonic-gate 	 *	- the Spooler heard that the printer has a fault,
4157c478bd9Sstevel@tonic-gate 	 *	  and the fault recovery is wait or beginning
4167c478bd9Sstevel@tonic-gate 	 *
4177c478bd9Sstevel@tonic-gate 	 * We should exit cleanly for the first two cases,
4187c478bd9Sstevel@tonic-gate 	 * but we have to be careful with the last. If it was THIS
4197c478bd9Sstevel@tonic-gate 	 * PROGRAM that told the Spooler about the fault, we must
4207c478bd9Sstevel@tonic-gate 	 * exit consistently.
4217c478bd9Sstevel@tonic-gate 	 *
4227c478bd9Sstevel@tonic-gate 	 * The method of avoiding any problem is to turn off the
4237c478bd9Sstevel@tonic-gate 	 * trapping of SIGTERM before telling the Spooler about
4247c478bd9Sstevel@tonic-gate 	 * the fault.
4257c478bd9Sstevel@tonic-gate 	 *
4267c478bd9Sstevel@tonic-gate 	 * Faults that we can detect:
4277c478bd9Sstevel@tonic-gate 	 *	- hangup (drop of carrier)
4287c478bd9Sstevel@tonic-gate 	 *	- interrupt (printer sent a break or quit character)
4297c478bd9Sstevel@tonic-gate 	 *	- SIGPIPE (output port is a FIFO, and was closed early)
4307c478bd9Sstevel@tonic-gate 	 *	- failed or incomplete write()
4317c478bd9Sstevel@tonic-gate 	 *	- excess delay in write() (handled with SIGALRM later)
4327c478bd9Sstevel@tonic-gate 	 *
4337c478bd9Sstevel@tonic-gate 	 * Pseudo-faults (errors in use):
4347c478bd9Sstevel@tonic-gate 	 *	- No input/output, or strange input/output
4357c478bd9Sstevel@tonic-gate 	 *	- Input/output identical
4367c478bd9Sstevel@tonic-gate 	 *	- No TERM defined or trouble reading Terminfo database
4377c478bd9Sstevel@tonic-gate 	 */
4387c478bd9Sstevel@tonic-gate 	signal(SIGTERM, sigterm);
4397c478bd9Sstevel@tonic-gate 	signal(SIGHUP, sighup);
4407c478bd9Sstevel@tonic-gate 	signal(SIGINT, sigint);
4417c478bd9Sstevel@tonic-gate 	signal(SIGQUIT, sigint);
4427c478bd9Sstevel@tonic-gate 	signal(SIGPIPE, sigpipe);
4437c478bd9Sstevel@tonic-gate 
4447c478bd9Sstevel@tonic-gate 
4457c478bd9Sstevel@tonic-gate 	if (argc > 1 && STREQU(argv[1], "-r")) {
4467c478bd9Sstevel@tonic-gate 		report_rate = 1;
4477c478bd9Sstevel@tonic-gate 		argc--;
4487c478bd9Sstevel@tonic-gate 		argv++;
4497c478bd9Sstevel@tonic-gate 	} else
4507c478bd9Sstevel@tonic-gate 		report_rate = 0;
4517c478bd9Sstevel@tonic-gate 
4527c478bd9Sstevel@tonic-gate 	(void) setlocale(LC_ALL, "");
4537c478bd9Sstevel@tonic-gate #if !defined(TEXT_DOMAIN)
4547c478bd9Sstevel@tonic-gate #define	TEXT_DOMAIN "SYS_TEST"
4557c478bd9Sstevel@tonic-gate #endif
4567c478bd9Sstevel@tonic-gate 	(void) textdomain(TEXT_DOMAIN);
4577c478bd9Sstevel@tonic-gate 
4587c478bd9Sstevel@tonic-gate 	/*
4597c478bd9Sstevel@tonic-gate 	 * Stat the standard output to be sure it is defined.
4607c478bd9Sstevel@tonic-gate 	 */
4617c478bd9Sstevel@tonic-gate 	if (fstat(1, &out) < 0) {
4627c478bd9Sstevel@tonic-gate 		signal(SIGTERM, SIG_IGN);
463*de81e71eSTim Marsland 		fprintf(stderr, gettext("Can't stat output "
464*de81e71eSTim Marsland 		    "(%s);\nincorrect use of lp.cat!\n"), PERROR);
4657c478bd9Sstevel@tonic-gate 		exit(E_BAD_OUTPUT);
4667c478bd9Sstevel@tonic-gate 	}
4677c478bd9Sstevel@tonic-gate 
4687c478bd9Sstevel@tonic-gate 	/*
4697c478bd9Sstevel@tonic-gate 	 * Stat the standard input to be sure it is defined.
4707c478bd9Sstevel@tonic-gate 	 */
4717c478bd9Sstevel@tonic-gate 	if (fstat(0, &in) < 0) {
4727c478bd9Sstevel@tonic-gate 		signal(SIGTERM, SIG_IGN);
473*de81e71eSTim Marsland 		fprintf(stderr, gettext("Can't stat input "
474*de81e71eSTim Marsland 		    "(%s);\nincorrect use of lp.cat!\n"), PERROR);
4757c478bd9Sstevel@tonic-gate 		exit(E_BAD_INPUT);
4767c478bd9Sstevel@tonic-gate 	}
4777c478bd9Sstevel@tonic-gate 
4787c478bd9Sstevel@tonic-gate 	/*
4797c478bd9Sstevel@tonic-gate 	 * If the standard output is not a character special file or a
4807c478bd9Sstevel@tonic-gate 	 * block special file, make sure it is not identical to the
4817c478bd9Sstevel@tonic-gate 	 * standard input.
4827c478bd9Sstevel@tonic-gate 	 *
4837c478bd9Sstevel@tonic-gate 	 * If we are an ecpp parallel port in centronics mode treat
4847c478bd9Sstevel@tonic-gate 	 * ourselves as a bpp compatible device.
4857c478bd9Sstevel@tonic-gate 	 */
4867c478bd9Sstevel@tonic-gate 
4877c478bd9Sstevel@tonic-gate 	if (is_a_prnio(1)) {
4887c478bd9Sstevel@tonic-gate 		func = prnio_state;
4897c478bd9Sstevel@tonic-gate 	} else if (is_a_parallel_bpp(1) ||
4907c478bd9Sstevel@tonic-gate 	    (get_ecpp_status(1) == ECPP_CENTRONICS)) {
4917c478bd9Sstevel@tonic-gate 		func = bpp_state;
4927c478bd9Sstevel@tonic-gate 	} else if (isatty(1)) {
4937c478bd9Sstevel@tonic-gate 		/* serial connection (probably) - continue as usual */
4947c478bd9Sstevel@tonic-gate 		func = nop;
4957c478bd9Sstevel@tonic-gate 	} else {
4967c478bd9Sstevel@tonic-gate 		func = nop;
4977c478bd9Sstevel@tonic-gate 	}
4987c478bd9Sstevel@tonic-gate 
4997c478bd9Sstevel@tonic-gate 	if (!ISCHR(out) && !ISBLK(out) && IDENTICAL(out, in)) {
5007c478bd9Sstevel@tonic-gate 		signal(SIGTERM, SIG_IGN);
501*de81e71eSTim Marsland 		fprintf(stderr, gettext("Input and output are identical; "
502*de81e71eSTim Marsland 		    "incorrect use of lp.cat!\n"));
5037c478bd9Sstevel@tonic-gate 		exit(E_IDENTICAL);
5047c478bd9Sstevel@tonic-gate 	}
5057c478bd9Sstevel@tonic-gate 
5067c478bd9Sstevel@tonic-gate 	/*
5077c478bd9Sstevel@tonic-gate 	 * The effective data transfer rate is the lesser
5087c478bd9Sstevel@tonic-gate 	 * of the transmission rate and print rate. If an
5097c478bd9Sstevel@tonic-gate 	 * argument was passed to us, it should be a data
5107c478bd9Sstevel@tonic-gate 	 * rate and it may be lower still.
5117c478bd9Sstevel@tonic-gate 	 * Based on the effective data transfer rate,
5127c478bd9Sstevel@tonic-gate 	 * we can predict the maximum delay we should experience.
5137c478bd9Sstevel@tonic-gate 	 * But there are other factors that could introduce
5147c478bd9Sstevel@tonic-gate 	 * delay, so let's be generous; after all, we'd rather
5157c478bd9Sstevel@tonic-gate 	 * err in favor of waiting too long to detect a fault
5167c478bd9Sstevel@tonic-gate 	 * than err too often on false alarms.
5177c478bd9Sstevel@tonic-gate 	 */
5187c478bd9Sstevel@tonic-gate 
519*de81e71eSTim Marsland 	if (!(TERM = getenv("TERM")) || !*TERM) {
5207c478bd9Sstevel@tonic-gate 		signal(SIGTERM, SIG_IGN);
521*de81e71eSTim Marsland 		fprintf(stderr, gettext("No TERM variable defined! "
522*de81e71eSTim Marsland 		    "Trouble with the Spooler!\n"));
5237c478bd9Sstevel@tonic-gate 		exit(E_BAD_TERM);
5247c478bd9Sstevel@tonic-gate 	}
525*de81e71eSTim Marsland 	if (!STREQU(TERM, NAME_UNKNOWN) &&
526*de81e71eSTim Marsland 	    tidbit(TERM, "cps", &print_rate) == -1) {
5277c478bd9Sstevel@tonic-gate 		signal(SIGTERM, SIG_IGN);
528*de81e71eSTim Marsland 		fprintf(stderr, gettext("Trouble identifying printer "
529*de81e71eSTim Marsland 		    "type \"%s\"; check the Terminfo database.\n"), TERM);
5307c478bd9Sstevel@tonic-gate 		exit(E_BAD_TERM);
5317c478bd9Sstevel@tonic-gate 	}
5327c478bd9Sstevel@tonic-gate 	if (STREQU(TERM, NAME_UNKNOWN))
5337c478bd9Sstevel@tonic-gate 		print_rate = -1;
5347c478bd9Sstevel@tonic-gate 
5357c478bd9Sstevel@tonic-gate 	effective_rate = baudrate() / 10; /* okay for most bauds */
5367c478bd9Sstevel@tonic-gate 	if (print_rate != -1 && print_rate < effective_rate)
5377c478bd9Sstevel@tonic-gate 		effective_rate = print_rate;
5387c478bd9Sstevel@tonic-gate 	if (argc > 1 && (n = atoi(argv[1])) >= 0 && n < effective_rate)
5397c478bd9Sstevel@tonic-gate 		effective_rate = n;	  /* 0 means infinite delay */
5407c478bd9Sstevel@tonic-gate 	if (effective_rate)
5417c478bd9Sstevel@tonic-gate 		max_delay = DELAY(BUFSIZ, effective_rate);
5427c478bd9Sstevel@tonic-gate 
5437c478bd9Sstevel@tonic-gate 	/*
5447c478bd9Sstevel@tonic-gate 	 * We'll use the "alarm()" system call to keep us from
5457c478bd9Sstevel@tonic-gate 	 * waiting too long to write to a printer in trouble.
5467c478bd9Sstevel@tonic-gate 	 */
5477c478bd9Sstevel@tonic-gate 	if (max_delay)
5487c478bd9Sstevel@tonic-gate 		signal(SIGALRM, sigalrm);
5497c478bd9Sstevel@tonic-gate 
5507c478bd9Sstevel@tonic-gate 	/*
5517c478bd9Sstevel@tonic-gate 	 * While not end of standard input, copy blocks to
5527c478bd9Sstevel@tonic-gate 	 * standard output.
5537c478bd9Sstevel@tonic-gate 	 */
5547c478bd9Sstevel@tonic-gate 	while ((nin = read(0, buffer, BUFSIZ)) > 0) {
5557c478bd9Sstevel@tonic-gate 		char *ptr = buffer;
5567c478bd9Sstevel@tonic-gate 
5577c478bd9Sstevel@tonic-gate 		/*
5587c478bd9Sstevel@tonic-gate 		 * We should be safe from incomplete writes to a full
5597c478bd9Sstevel@tonic-gate 		 * pipe, as long as the size of the buffer we write is
5607c478bd9Sstevel@tonic-gate 		 * a even divisor of the pipe buffer limit. As long as
5617c478bd9Sstevel@tonic-gate 		 * we read from files or pipes (not communication devices)
5627c478bd9Sstevel@tonic-gate 		 * this should be true for all but the last buffer. The
5637c478bd9Sstevel@tonic-gate 		 * last will be smaller, and won't straddle the pipe max
5647c478bd9Sstevel@tonic-gate 		 * limit (think about it).
5657c478bd9Sstevel@tonic-gate 		 */
5667c478bd9Sstevel@tonic-gate #if	PIPE_BUF < BUFSIZ || (PIPE_MAX % BUFSIZ)
5677c478bd9Sstevel@tonic-gate 		this_wont_compile;
5687c478bd9Sstevel@tonic-gate #endif
5697c478bd9Sstevel@tonic-gate 		if (report_rate)
5707c478bd9Sstevel@tonic-gate 			epoch_start = times(&tms);
5717c478bd9Sstevel@tonic-gate 		do {
5727c478bd9Sstevel@tonic-gate 			wait_state(1, func);
5737c478bd9Sstevel@tonic-gate 
5747c478bd9Sstevel@tonic-gate 			if (max_delay)
5757c478bd9Sstevel@tonic-gate 				alarm(max_delay);
5767c478bd9Sstevel@tonic-gate 			nout = write(1, ptr, nin);
5777c478bd9Sstevel@tonic-gate 			alarm(0);
5787c478bd9Sstevel@tonic-gate 			if (nout < 0) {
579*de81e71eSTim Marsland 				fprintf(stderr, gettext("Write failed "
580*de81e71eSTim Marsland 				    "(%s);\nperhaps the printer has gone "
581*de81e71eSTim Marsland 				    "off-line.\n"), PERROR);
5827c478bd9Sstevel@tonic-gate 				fflush(stderr);
5837c478bd9Sstevel@tonic-gate 				if (errno != EINTR)
5847c478bd9Sstevel@tonic-gate 				/* I/O error on device, get lpcshed to retry */
5857c478bd9Sstevel@tonic-gate 					exit(PRINTER_IO_ERROR);
5867c478bd9Sstevel@tonic-gate 				else /* wait for printer to come back online */
5877c478bd9Sstevel@tonic-gate 					sleep(15);
5887c478bd9Sstevel@tonic-gate 			} else {
5897c478bd9Sstevel@tonic-gate 				nin -= nout;
5907c478bd9Sstevel@tonic-gate 				ptr += nout;
5917c478bd9Sstevel@tonic-gate 			}
5927c478bd9Sstevel@tonic-gate 		} while (nin > 0);
5937c478bd9Sstevel@tonic-gate 
5947c478bd9Sstevel@tonic-gate 		if (max_delay)
5957c478bd9Sstevel@tonic-gate 			alarm(0);
5967c478bd9Sstevel@tonic-gate 		else if (report_rate) {
5977c478bd9Sstevel@tonic-gate 			epoch_end = times(&tms);
5987c478bd9Sstevel@tonic-gate 			if (epoch_end - epoch_start > 0)
599*de81e71eSTim Marsland 				fprintf(stderr, "%d CPS\n",
600*de81e71eSTim Marsland 				    R((100 * BUFSIZ) /
601*de81e71eSTim Marsland 				    (double)(epoch_end - epoch_start)));
6027c478bd9Sstevel@tonic-gate 		}
6037c478bd9Sstevel@tonic-gate 
6047c478bd9Sstevel@tonic-gate 	}
6057c478bd9Sstevel@tonic-gate 
606f928ce67Sceastha 	return (E_SUCCESS);
6077c478bd9Sstevel@tonic-gate }
6087c478bd9Sstevel@tonic-gate 
609*de81e71eSTim Marsland /*
610*de81e71eSTim Marsland  * sighup() - CATCH A HANGUP (LOSS OF CARRIER)
611*de81e71eSTim Marsland  */
612*de81e71eSTim Marsland void
613*de81e71eSTim Marsland sighup()
6147c478bd9Sstevel@tonic-gate {
6157c478bd9Sstevel@tonic-gate 	signal(SIGTERM, SIG_IGN);
6167c478bd9Sstevel@tonic-gate 	signal(SIGHUP, SIG_IGN);
6177c478bd9Sstevel@tonic-gate 	fprintf(stderr, gettext(HANGUP_FAULT_LPCAT));
6187c478bd9Sstevel@tonic-gate 	exit(E_HANGUP);
6197c478bd9Sstevel@tonic-gate }
6207c478bd9Sstevel@tonic-gate 
621*de81e71eSTim Marsland /*
622*de81e71eSTim Marsland  * sigint() - CATCH AN INTERRUPT
623*de81e71eSTim Marsland  */
624*de81e71eSTim Marsland void
625*de81e71eSTim Marsland sigint()
6267c478bd9Sstevel@tonic-gate {
6277c478bd9Sstevel@tonic-gate 	signal(SIGTERM, SIG_IGN);
6287c478bd9Sstevel@tonic-gate 	signal(SIGINT, SIG_IGN);
6297c478bd9Sstevel@tonic-gate 	fprintf(stderr, gettext(INTERRUPT_FAULT));
6307c478bd9Sstevel@tonic-gate 	exit(E_INTERRUPT);
6317c478bd9Sstevel@tonic-gate }
6327c478bd9Sstevel@tonic-gate 
633*de81e71eSTim Marsland /*
634*de81e71eSTim Marsland  * sigpipe() - CATCH EARLY CLOSE OF PIPE
635*de81e71eSTim Marsland  */
636*de81e71eSTim Marsland void
637*de81e71eSTim Marsland sigpipe()
6387c478bd9Sstevel@tonic-gate {
6397c478bd9Sstevel@tonic-gate 	signal(SIGTERM, SIG_IGN);
6407c478bd9Sstevel@tonic-gate 	signal(SIGPIPE, SIG_IGN);
6417c478bd9Sstevel@tonic-gate 	fprintf(stderr, gettext(PIPE_FAULT));
6427c478bd9Sstevel@tonic-gate 	exit(E_INTERRUPT);
6437c478bd9Sstevel@tonic-gate }
6447c478bd9Sstevel@tonic-gate 
645*de81e71eSTim Marsland /*
646*de81e71eSTim Marsland  * sigalrm() - CATCH AN ALARM
647*de81e71eSTim Marsland  */
648*de81e71eSTim Marsland void
649*de81e71eSTim Marsland sigalrm()
6507c478bd9Sstevel@tonic-gate {
6517c478bd9Sstevel@tonic-gate 	signal(SIGTERM, SIG_IGN);
652*de81e71eSTim Marsland 	fprintf(stderr, gettext("Excessive write delay; "
653*de81e71eSTim Marsland 	    "perhaps the printer has gone off-line.\n"));
6547c478bd9Sstevel@tonic-gate 	exit(E_TIMEOUT);
6557c478bd9Sstevel@tonic-gate }
6567c478bd9Sstevel@tonic-gate 
657*de81e71eSTim Marsland /*
658*de81e71eSTim Marsland  * sigterm() - CATCH A TERMINATION SIGNAL
659*de81e71eSTim Marsland  */
660*de81e71eSTim Marsland void
661*de81e71eSTim Marsland sigterm()
6627c478bd9Sstevel@tonic-gate {
6637c478bd9Sstevel@tonic-gate 	signal(SIGTERM, SIG_IGN);
6647c478bd9Sstevel@tonic-gate 	/*
6657c478bd9Sstevel@tonic-gate 	 * try to flush the output queue in the case of ecpp port.
6667c478bd9Sstevel@tonic-gate 	 * ignore the return code as this may not be the ecpp.
6677c478bd9Sstevel@tonic-gate 	 */
6687c478bd9Sstevel@tonic-gate 	ioctl(1, I_FLUSH, FLUSHW);
6697c478bd9Sstevel@tonic-gate 	exit(E_SUCCESS);
6707c478bd9Sstevel@tonic-gate }
6717c478bd9Sstevel@tonic-gate 
672*de81e71eSTim Marsland /*
673*de81e71eSTim Marsland  * baudrate() - RETURN BAUD RATE OF OUTPUT LINE
674*de81e71eSTim Marsland  */
6757c478bd9Sstevel@tonic-gate 
6767c478bd9Sstevel@tonic-gate static int		baud_convert[] =
6777c478bd9Sstevel@tonic-gate {
6787c478bd9Sstevel@tonic-gate 	0, 50, 75, 110, 135, 150, 200, 300, 600, 1200,
6797c478bd9Sstevel@tonic-gate 	1800, 2400, 4800, 9600, 19200, 38400, 57600,
680*de81e71eSTim Marsland 	76800, 115200, 153600, 230400, 307200, 460800, 921600
6817c478bd9Sstevel@tonic-gate };
6827c478bd9Sstevel@tonic-gate 
683*de81e71eSTim Marsland int
684*de81e71eSTim Marsland baudrate()
6857c478bd9Sstevel@tonic-gate {
6867c478bd9Sstevel@tonic-gate 	struct termio		tm;
6877c478bd9Sstevel@tonic-gate 	struct termios		tms;
6887c478bd9Sstevel@tonic-gate 	int			speed;
6897c478bd9Sstevel@tonic-gate 
6907c478bd9Sstevel@tonic-gate 	if (ioctl(1, TCGETS, &tms) < 0) {
6917c478bd9Sstevel@tonic-gate 		if (ioctl(1, TCGETA, &tm) < 0)
6927c478bd9Sstevel@tonic-gate 			return (1200);
6937c478bd9Sstevel@tonic-gate 		else
6947c478bd9Sstevel@tonic-gate 			speed = tm.c_cflag&CBAUD;
6957c478bd9Sstevel@tonic-gate 	} else
6967c478bd9Sstevel@tonic-gate 		speed = cfgetospeed(&tms);
6977c478bd9Sstevel@tonic-gate 
6987c478bd9Sstevel@tonic-gate 	return (speed ? baud_convert[speed] : 1200);
6997c478bd9Sstevel@tonic-gate }
700