/* $OpenBSD: lpr.c,v 1.22 2001/11/23 03:58:18 deraadt Exp $ */ /* $NetBSD: lpr.c,v 1.10 1996/03/21 18:12:25 jtc Exp $ */ /* * Copyright (c) 1983, 1989, 1993 * The Regents of the University of California. All rights reserved. * (c) UNIX System Laboratories, Inc. * All or some portions of this file are derived from material licensed * to the University of California by American Telephone and Telegraph * Co. or Unix System Laboratories, Inc. and are reproduced herein with * the permission of UNIX System Laboratories, Inc. * * * 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. * 3. All advertising materials mentioning features or use of this software * must display the following acknowledgement: * This product includes software developed by the University of * California, Berkeley and its contributors. * 4. Neither the name of the University nor the names of its contributors * may be used to endorse or promote products derived from this software * without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE REGENTS 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 REGENTS 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. */ #ifndef lint static const char copyright[] = "@(#) Copyright (c) 1983, 1989, 1993\n\ The Regents of the University of California. All rights reserved.\n"; #endif /* not lint */ #ifndef lint #if 0 static const char sccsid[] = "@(#)lpr.c 8.4 (Berkeley) 4/28/95"; #else static const char rcsid[] = "$OpenBSD: lpr.c,v 1.22 2001/11/23 03:58:18 deraadt Exp $"; #endif #endif /* not lint */ /* * lpr -- off line print * * Allows multiple printers and printers on remote machines by * using information from a printer data base. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "lp.h" #include "lp.local.h" #include "pathnames.h" static char *cfname; /* daemon control files, linked from tf's */ static char *class = host; /* class title on header page */ static char *dfname; /* data files */ static char *fonts[4]; /* troff font names */ static char format = 'f'; /* format char for printing files */ static int hdr = 1; /* print header or not (default is yes) */ static int iflag; /* indentation wanted */ static int inchar; /* location to increment char in file names */ static int indent; /* amount to indent */ static char *jobname; /* job name on header page */ static int mailflg; /* send mail */ static int nact; /* number of jobs to act on */ static int ncopies = 1; /* # of copies to make */ static char *person; /* user name */ static int qflag; /* q job, but don't exec daemon */ static int rflag; /* remove files upon completion */ static int sflag; /* symbolic link flag */ static int tfd; /* control file descriptor */ static char *tfname; /* tmp copy of cf before linking */ static char *title; /* pr'ing title */ static int userid; /* user id */ static char *width; /* width for versatec printing */ static struct stat statb; volatile sig_atomic_t gotintr; static void card __P((int, char *)); static void chkprinter __P((char *)); static void cleanup __P((int)); static void copy __P((int, char [])); static void fatal2 __P((const char *, ...)); static char *linked __P((char *)); static char *lmktemp __P((char *, int, int)); static void mktemps __P((void)); static int nfile __P((char *)); static int test __P((char *)); static char *itoa __P((int)); uid_t uid, euid; int main(argc, argv) int argc; char *argv[]; { struct passwd *pw; struct group *gptr; char *arg, *cp; char buf[BUFSIZ]; int i, f; struct stat stb; euid = geteuid(); uid = getuid(); seteuid(uid); if (signal(SIGHUP, SIG_IGN) != SIG_IGN) signal(SIGHUP, cleanup); if (signal(SIGINT, SIG_IGN) != SIG_IGN) signal(SIGINT, cleanup); if (signal(SIGQUIT, SIG_IGN) != SIG_IGN) signal(SIGQUIT, cleanup); if (signal(SIGTERM, SIG_IGN) != SIG_IGN) signal(SIGTERM, cleanup); name = argv[0]; gethostname(host, sizeof (host)); openlog("lpd", 0, LOG_LPR); while (argc > 1 && argv[1][0] == '-') { argc--; arg = *++argv; switch (arg[1]) { case 'P': /* specifiy printer name */ if (arg[2]) printer = &arg[2]; else if (argc > 1) { argc--; printer = *++argv; } break; case 'C': /* classification spec */ hdr++; if (arg[2]) class = &arg[2]; else if (argc > 1) { argc--; class = *++argv; } break; case 'U': /* user name */ hdr++; if (arg[2]) person = &arg[2]; else if (argc > 1) { argc--; person = *++argv; } break; case 'J': /* job name */ hdr++; if (arg[2]) jobname = &arg[2]; else if (argc > 1) { argc--; jobname = *++argv; } break; case 'T': /* pr's title line */ if (arg[2]) title = &arg[2]; else if (argc > 1) { argc--; title = *++argv; } break; case 'l': /* literal output */ case 'p': /* print using ``pr'' */ case 't': /* print troff output (cat files) */ case 'n': /* print ditroff output */ case 'd': /* print tex output (dvi files) */ case 'g': /* print graph(1G) output */ case 'c': /* print cifplot output */ case 'v': /* print vplot output */ format = arg[1]; break; case 'f': /* print fortran output */ format = 'r'; break; case '4': /* troff fonts */ case '3': case '2': case '1': if (argc > 1) { argc--; fonts[arg[1] - '1'] = *++argv; } break; case 'w': /* versatec page width */ width = arg+2; break; case 'r': /* remove file when done */ rflag++; break; case 'm': /* send mail when done */ mailflg++; break; case 'h': /* toggle want of header page */ hdr = !hdr; break; case 's': /* try to link files */ sflag++; break; case 'q': /* just q job */ qflag++; break; case 'i': /* indent output */ iflag++; indent = arg[2] ? atoi(&arg[2]) : 8; break; case '#': /* n copies */ if (isdigit(arg[2])) { i = atoi(&arg[2]); if (i > 0) ncopies = i; } } } if (printer == NULL) { char *p; printer = DEFLP; if ((p = getenv("PRINTER")) != NULL) printer = p; } chkprinter(printer); if (SC && ncopies > 1) fatal2("multiple copies are not allowed"); if (MC > 0 && ncopies > MC) fatal2("only %d copies are allowed", MC); /* * Get the identity of the person doing the lpr using the same * algorithm as lprm. */ userid = getuid(); if (userid != DU || person == 0) { if ((pw = getpwuid(userid)) == NULL) fatal2("Who are you?"); person = pw->pw_name; } /* * Check for restricted group access. */ if (RG != NULL && userid != DU) { if ((gptr = getgrnam(RG)) == NULL) fatal2("Restricted group specified incorrectly"); if (gptr->gr_gid != getgid()) { while (*gptr->gr_mem != NULL) { if ((strcmp(person, *gptr->gr_mem)) == 0) break; gptr->gr_mem++; } if (*gptr->gr_mem == NULL) fatal2("Not a member of the restricted group"); } } /* * Check to make sure queuing is enabled if userid is not root. */ (void) snprintf(buf, sizeof(buf), "%s/%s", SD, LO); if (userid && stat(buf, &stb) == 0 && (stb.st_mode & 010)) fatal2("Printer queue is disabled"); /* * Initialize the control file. */ mktemps(); tfd = nfile(tfname); seteuid(euid); (void) fchown(tfd, DU, -1); /* owned by daemon for protection */ seteuid(uid); card('H', host); card('P', person); if (hdr) { if (jobname == NULL) { if (argc == 1) jobname = "stdin"; else jobname = (arg = strrchr(argv[1], '/')) ? arg+1 : argv[1]; } card('J', jobname); card('C', class); card('L', person); } if (iflag) card('I', itoa(indent)); if (mailflg) card('M', person); if (format == 't' || format == 'n' || format == 'd') for (i = 0; i < 4; i++) if (fonts[i] != NULL) card('1'+i, fonts[i]); if (width != NULL) card('W', width); /* * Read the files and spool them. */ if (argc == 1) copy(0, " "); else while (--argc) { if ((f = test(arg = *++argv)) < 0) continue; /* file unreasonable */ if (sflag && (cp = linked(arg)) != NULL) { (void) snprintf(buf, sizeof(buf), "%d %d", statb.st_dev, statb.st_ino); card('S', buf); if (format == 'p') card('T', title ? title : arg); for (i = 0; i < ncopies; i++) card(format, &dfname[inchar-2]); card('U', &dfname[inchar-2]); if (f) card('U', cp); card('N', arg); dfname[inchar]++; nact++; continue; } if (sflag) printf("%s: %s: not linked, copying instead\n", name, arg); if ((i = open(arg, O_RDONLY)) < 0) { printf("%s: cannot open %s\n", name, arg); } else { copy(i, arg); (void) close(i); if (f && unlink(arg) < 0) printf("%s: %s: not removed\n", name, arg); } } if (nact) { (void) close(tfd); tfname[inchar]--; /* * Touch the control file to fix position in the queue. */ seteuid(euid); if ((tfd = open(tfname, O_RDWR)) >= 0) { char c; if (read(tfd, &c, 1) == 1 && lseek(tfd, (off_t)0, 0) == 0 && write(tfd, &c, 1) != 1) { printf("%s: cannot touch %s\n", name, tfname); tfname[inchar]++; cleanup(0); } (void) close(tfd); } if (link(tfname, cfname) < 0) { printf("%s: cannot rename %s\n", name, cfname); tfname[inchar]++; cleanup(0); } unlink(tfname); seteuid(uid); if (qflag) /* just q things up */ exit(0); if (!startdaemon(printer)) printf("jobs queued, but cannot start daemon.\n"); exit(0); } cleanup(0); return (1); /* NOTREACHED */ } /* * Create the file n and copy from file descriptor f. */ static void copy(f, n) int f; char n[]; { int fd, i, nr, nc; char buf[BUFSIZ]; if (format == 'p') card('T', title ? title : n); for (i = 0; i < ncopies; i++) card(format, &dfname[inchar-2]); card('U', &dfname[inchar-2]); card('N', n); fd = nfile(dfname); nr = nc = 0; while ((i = read(f, buf, BUFSIZ)) > 0) { if (write(fd, buf, i) != i) { printf("%s: %s: temp file write error\n", name, n); break; } nc += i; if (nc >= BUFSIZ) { nc -= BUFSIZ; nr++; if (MX > 0 && nr > MX) { printf("%s: %s: copy file is too large\n", name, n); break; } } } (void) close(fd); if (nc==0 && nr==0) printf("%s: %s: empty input file\n", name, f ? n : "stdin"); else nact++; } /* * Try and link the file to dfname. Return a pointer to the full * path name if successful. */ static char * linked(file) char *file; { char *cp; static char nfile[MAXPATHLEN]; int ret; if (*file != '/') { if (getcwd(nfile, sizeof(nfile)) == NULL) return(NULL); while (file[0] == '.') { switch (file[1]) { case '/': file += 2; continue; case '.': if (file[2] == '/') { if ((cp = strrchr(nfile, '/')) != NULL) *cp = '\0'; file += 3; continue; } } break; } strncat(nfile, "/", sizeof(nfile) - strlen(nfile) - 1); strncat(nfile, file, sizeof(nfile) - strlen(nfile) - 1); file = nfile; } seteuid(euid); ret = symlink(file, dfname); seteuid(uid); return(ret ? NULL : file); } /* * Put a line into the control file. */ static void card(c, p2) int c; char *p2; { char buf[BUFSIZ]; char *p1 = buf; int len = 2; *p1++ = c; while ((c = *p2++) != '\0' && len < sizeof(buf)) { *p1++ = (c == '\n') ? ' ' : c; len++; } *p1++ = '\n'; write(tfd, buf, len); } /* * Create a new file in the spool directory. */ static int nfile(n) char *n; { int f; int oldumask = umask(0); /* should block signals */ seteuid(euid); f = open(n, O_WRONLY | O_EXCL | O_CREAT, FILMOD); (void) umask(oldumask); if (f < 0) { printf("%s: cannot create %s\n", name, n); cleanup(0); } if (fchown(f, userid, -1) < 0) { printf("%s: cannot chown %s\n", name, n); cleanup(0); /* cleanup does exit */ } seteuid(uid); if (++n[inchar] > 'z') { if (++n[inchar-2] == 't') { printf("too many files - break up the job\n"); cleanup(0); } n[inchar] = 'A'; } else if (n[inchar] == '[') n[inchar] = 'a'; return(f); } /* * Cleanup after interrupts and errors. */ static void cleanup(signo) int signo; { int i; signal(SIGHUP, SIG_IGN); signal(SIGINT, SIG_IGN); signal(SIGQUIT, SIG_IGN); signal(SIGTERM, SIG_IGN); i = inchar; seteuid(euid); if (tfname) do unlink(tfname); while (tfname[i]-- != 'A'); if (cfname) do unlink(cfname); while (cfname[i]-- != 'A'); if (dfname) do { do unlink(dfname); while (dfname[i]-- != 'A'); dfname[i] = 'z'; } while (dfname[i-2]-- != 'd'); _exit(1); } /* * Test to see if this is a printable file. * Return -1 if it is not, 0 if its printable, and 1 if * we should remove it after printing. */ static int test(file) char *file; { struct exec execb; int fd; char *cp; if ((fd = open(file, O_RDONLY)) < 0) { printf("%s: cannot open %s\n", name, file); goto bad; } if (fstat(fd, &statb) < 0) { printf("%s: cannot stat %s\n", name, file); goto bad; } if ((statb.st_mode & S_IFMT) == S_IFDIR) { printf("%s: %s is a directory\n", name, file); goto bad; } if (statb.st_size == 0) { printf("%s: %s is an empty file\n", name, file); goto bad; } if (read(fd, &execb, sizeof(execb)) == sizeof(execb) && !N_BADMAG(execb)) { printf("%s: %s is an executable program and is unprintable", name, file); (void) close(fd); goto bad; } (void) close(fd); if (rflag) { if ((cp = strrchr(file, '/')) == NULL) { if (access(".", 2) == 0) return(1); } else { if (cp == file) { fd = access("/", 2); } else { *cp = '\0'; fd = access(file, 2); *cp = '/'; } if (fd == 0) return(1); } printf("%s: %s: is not removable by you\n", name, file); } return(0); bad: return(-1); } /* * itoa - integer to string conversion */ static char * itoa(i) int i; { static char b[10] = "########"; char *p; p = &b[8]; do *p-- = i%10 + '0'; while (i /= 10); return(++p); } /* * Perform lookup for printer name or abbreviation -- */ static void chkprinter(s) char *s; { int status; if ((status = cgetent(&bp, printcapdb, s)) == -2) fatal2("cannot open printer description file"); else if (status == -1) fatal2("%s: unknown printer", s); if (cgetstr(bp, "sd", &SD) == -1) SD = _PATH_DEFSPOOL; if (cgetstr(bp, "lo", &LO) == -1) LO = DEFLOCK; cgetstr(bp, "rg", &RG); if (cgetnum(bp, "mx", &MX) < 0) MX = DEFMX; if (cgetnum(bp,"mc", &MC) < 0) MC = DEFMAXCOPIES; if (cgetnum(bp, "du", &DU) < 0) DU = DEFUID; SC = (cgetcap(bp, "sc", ':') != NULL); } /* * Make the temp files. */ static void mktemps() { int len, fd, n; char *cp; char buf[BUFSIZ]; char *lmktemp(); (void) snprintf(buf, sizeof(buf), "%s/.seq", SD); seteuid(euid); if ((fd = open(buf, O_RDWR|O_CREAT, 0661)) < 0) { printf("%s: cannot create %s\n", name, buf); exit(1); } if (flock(fd, LOCK_EX)) { printf("%s: cannot lock %s\n", name, buf); exit(1); } seteuid(uid); n = 0; if ((len = read(fd, buf, sizeof(buf))) > 0) { for (cp = buf; len--; ) { if (*cp < '0' || *cp > '9') break; n = n * 10 + (*cp++ - '0'); } } len = strlen(SD) + strlen(host) + 8; tfname = lmktemp("tf", n, len); cfname = lmktemp("cf", n, len); dfname = lmktemp("df", n, len); inchar = strlen(SD) + 3; n = (n + 1) % 1000; (void) lseek(fd, (off_t)0, 0); snprintf(buf, sizeof(buf), "%03d\n", n); (void) write(fd, buf, strlen(buf)); (void) close(fd); /* unlocks as well */ } /* * Make a temp file name. */ static char * lmktemp(id, num, len) char *id; int num, len; { char *s; if ((s = malloc(len)) == NULL) fatal2("out of memory"); (void) snprintf(s, len, "%s/%sA%03d%s", SD, id, num, host); return(s); } #ifdef __STDC__ #include #else #include #endif static void #ifdef __STDC__ fatal2(const char *msg, ...) #else fatal2(msg, va_alist) char *msg; va_dcl #endif { va_list ap; #ifdef __STDC__ va_start(ap, msg); #else va_start(ap); #endif printf("%s: ", name); vprintf(msg, ap); putchar('\n'); va_end(ap); exit(1); }