Patrick Welche <prlw1@cam.ac.uk>
[netbsd-mini2440.git] / usr.sbin / lpr / filters / lpf.c
bloba1d7cd8a2d0dbe167df364b1a7b173160884249d
1 /* $NetBSD: lpf.c,v 1.12 2008/05/07 17:44:44 christos Exp $ */
2 /*
3 * Copyright (c) 1983, 1993
4 * The Regents of the University of California. All rights reserved.
6 * Redistribution and use in source and binary forms, with or without
7 * modification, are permitted provided that the following conditions
8 * are met:
9 * 1. Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * 2. Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * 3. Neither the name of the University nor the names of its contributors
15 * may be used to endorse or promote products derived from this software
16 * without specific prior written permission.
18 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
19 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
22 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
24 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
25 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
26 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
27 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
28 * SUCH DAMAGE.
31 #include <sys/cdefs.h>
32 #ifndef lint
33 __COPYRIGHT("@(#) Copyright (c) 1983, 1993\
34 The Regents of the University of California. All rights reserved.");
35 #if 0
36 static char sccsid[] = "@(#)lpf.c 8.1 (Berkeley) 6/6/93";
37 #else
38 __RCSID("$NetBSD: lpf.c,v 1.12 2008/05/07 17:44:44 christos Exp $");
39 #endif
40 #endif /* not lint */
43 * filter which reads the output of nroff and converts lines
44 * with ^H's to overwritten lines. Thus this works like 'ul'
45 * but is much better: it can handle more than 2 overwrites
46 * and it is written with some style.
47 * modified by kls to use register references instead of arrays
48 * to try to gain a little speed.
51 #include <signal.h>
52 #include <string.h>
53 #include <unistd.h>
54 #include <stdlib.h>
55 #include <stdio.h>
57 #define MAXWIDTH 132
58 #define MAXREP 10
60 char buf[MAXREP][MAXWIDTH];
61 int maxcol[MAXREP] = {-1};
62 int lineno;
63 int width = 132; /* default line length */
64 int length = 66; /* page length */
65 int indent; /* indentation length */
66 int npages = 1;
67 int literal; /* print control characters */
68 char *name; /* user's login name */
69 char *host; /* user's machine name */
70 char *acctfile; /* accounting information file */
71 int crnl; /* \n -> \r\n */
72 int need_cr;
74 int main(int, char *[]);
75 void usage(void);
77 int
78 main(int argc, char *argv[])
80 FILE *p = stdin, *o = stdout;
81 int i, col;
82 char *cp;
83 int done, linedone, maxrep, ch, prch;
84 char *limit;
86 while ((ch = getopt(argc, argv, "cfh:i:j:l:n:w:")) != -1)
87 switch (ch) {
88 case 'n':
89 name = optarg;
90 break;
91 case 'h':
92 host = optarg;
93 break;
94 case 'w':
95 if ((i = atoi(optarg)) > 0 && i <= MAXWIDTH)
96 width = i;
97 break;
98 case 'l':
99 length = atoi(optarg);
100 break;
101 case 'i':
102 indent = atoi(optarg);
103 break;
104 case 'c': /* Print control chars */
105 literal++;
106 break;
107 case 'f': /* Fix missing carriage returns */
108 crnl++;
109 break;
110 case 'j': /* ignore job name */
111 break;
112 default:
113 usage();
115 argc -= optind;
116 argv += optind;
117 if (argc)
118 acctfile = *argv;
120 memset(buf, ' ', sizeof(buf));
121 done = 0;
123 while (!done) {
124 col = indent;
125 maxrep = -1;
126 linedone = 0;
127 prch = ch = 0;
128 need_cr = 0;
129 while (!linedone) {
130 prch = ch;
131 switch (ch = getc(p)) {
132 case EOF:
133 linedone = done = 1;
134 ch = '\n';
135 break;
137 case '\f':
138 lineno = length;
139 case '\n':
140 if (crnl && prch != '\r')
141 need_cr = 1;
142 if (maxrep < 0)
143 maxrep = 0;
144 linedone = 1;
145 break;
147 case '\b':
148 if (--col < indent)
149 col = indent;
150 break;
152 case '\r':
153 col = indent;
154 break;
156 case '\t':
157 col = ((col - indent) | 07) + indent + 1;
158 break;
160 case '\031':
162 * lpd needs to use a different filter to
163 * print data so stop what we are doing and
164 * wait for lpd to restart us.
166 if ((ch = getchar()) == '\1') {
167 fflush(stdout);
168 kill(getpid(), SIGSTOP);
169 break;
170 } else {
171 ungetc(ch, stdin);
172 ch = '\031';
175 default:
176 if (col >= width || (!literal && ch < ' ')) {
177 col++;
178 break;
180 cp = &buf[0][col];
181 for (i = 0; i < MAXREP; i++) {
182 if (i > maxrep)
183 maxrep = i;
184 if (*cp == ' ') {
185 *cp = ch;
186 if (col > maxcol[i])
187 maxcol[i] = col;
188 break;
190 cp += MAXWIDTH;
192 col++;
193 break;
197 /* print out lines */
198 for (i = 0; i <= maxrep; i++) {
199 for (cp = buf[i], limit = cp+maxcol[i]; cp <= limit;) {
200 putc(*cp, o);
201 *cp++ = ' ';
203 if (i < maxrep)
204 putc('\r', o);
205 else {
206 if (need_cr)
207 putc('\r', o);
208 putc(ch, o);
210 if (++lineno >= length) {
211 fflush(o);
212 npages++;
213 lineno = 0;
215 maxcol[i] = -1;
218 if (lineno) { /* be sure to end on a page boundary */
219 putchar('\f');
220 npages++;
222 if (name && acctfile && access(acctfile, 02) >= 0 &&
223 freopen(acctfile, "a", stdout) != NULL) {
224 printf("%7.2f\t%s:%s\n", (float)npages, host, name);
226 exit(0);
229 void
230 usage(void)
232 fprintf(stderr,
233 "usage: lpf [-c] [-f] [-h host] [-i indent] [-l length] [-n name] [-w width] [acctfile]\n");
234 exit(1);