10b561052SJoerg Wunsch /* 20b561052SJoerg Wunsch * Copyright (c) 1983, 1993 30b561052SJoerg Wunsch * The Regents of the University of California. All rights reserved. 40b561052SJoerg Wunsch * 50b561052SJoerg Wunsch * Redistribution and use in source and binary forms, with or without 60b561052SJoerg Wunsch * modification, are permitted provided that the following conditions 70b561052SJoerg Wunsch * are met: 80b561052SJoerg Wunsch * 1. Redistributions of source code must retain the above copyright 90b561052SJoerg Wunsch * notice, this list of conditions and the following disclaimer. 100b561052SJoerg Wunsch * 2. Redistributions in binary form must reproduce the above copyright 110b561052SJoerg Wunsch * notice, this list of conditions and the following disclaimer in the 120b561052SJoerg Wunsch * documentation and/or other materials provided with the distribution. 130b561052SJoerg Wunsch * 3. All advertising materials mentioning features or use of this software 140b561052SJoerg Wunsch * must display the following acknowledgement: 150b561052SJoerg Wunsch * This product includes software developed by the University of 160b561052SJoerg Wunsch * California, Berkeley and its contributors. 170b561052SJoerg Wunsch * 4. Neither the name of the University nor the names of its contributors 180b561052SJoerg Wunsch * may be used to endorse or promote products derived from this software 190b561052SJoerg Wunsch * without specific prior written permission. 200b561052SJoerg Wunsch * 210b561052SJoerg Wunsch * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 220b561052SJoerg Wunsch * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 230b561052SJoerg Wunsch * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 240b561052SJoerg Wunsch * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 250b561052SJoerg Wunsch * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 260b561052SJoerg Wunsch * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 270b561052SJoerg Wunsch * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 280b561052SJoerg Wunsch * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 290b561052SJoerg Wunsch * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 300b561052SJoerg Wunsch * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 310b561052SJoerg Wunsch * SUCH DAMAGE. 320b561052SJoerg Wunsch */ 330b561052SJoerg Wunsch 340b561052SJoerg Wunsch #ifndef lint 354a1a0dbeSGarrett Wollman /* 365458e2f4SJoerg Wunsch static char sccsid[] = "@(#)displayq.c 8.4 (Berkeley) 4/28/95"; 374a1a0dbeSGarrett Wollman */ 384a1a0dbeSGarrett Wollman static const char rcsid[] = 3997d92980SPeter Wemm "$FreeBSD$"; 400b561052SJoerg Wunsch #endif /* not lint */ 410b561052SJoerg Wunsch 420b561052SJoerg Wunsch #include <sys/param.h> 430b561052SJoerg Wunsch #include <sys/stat.h> 440b561052SJoerg Wunsch 454a1a0dbeSGarrett Wollman #include <ctype.h> 460b561052SJoerg Wunsch #include <dirent.h> 474a1a0dbeSGarrett Wollman #include <errno.h> 484a1a0dbeSGarrett Wollman #include <fcntl.h> 494a1a0dbeSGarrett Wollman #include <signal.h> 500b561052SJoerg Wunsch #include <stdio.h> 510b561052SJoerg Wunsch #include <stdlib.h> 520b561052SJoerg Wunsch #include <string.h> 534a1a0dbeSGarrett Wollman #define psignal foil_gcc_psignal 544a1a0dbeSGarrett Wollman #define sys_siglist foil_gcc_siglist 554a1a0dbeSGarrett Wollman #include <unistd.h> 564a1a0dbeSGarrett Wollman #undef psignal 574a1a0dbeSGarrett Wollman #undef sys_siglist 584a1a0dbeSGarrett Wollman 590b561052SJoerg Wunsch #include "lp.h" 600b561052SJoerg Wunsch #include "lp.local.h" 610b561052SJoerg Wunsch #include "pathnames.h" 620b561052SJoerg Wunsch 630b561052SJoerg Wunsch /* 640b561052SJoerg Wunsch * Routines to display the state of the queue. 650b561052SJoerg Wunsch */ 660b561052SJoerg Wunsch #define JOBCOL 40 /* column for job # in -l format */ 670b561052SJoerg Wunsch #define OWNCOL 7 /* start of Owner column in normal */ 680b561052SJoerg Wunsch #define SIZCOL 62 /* start of Size column in normal */ 690b561052SJoerg Wunsch 700b561052SJoerg Wunsch /* 710b561052SJoerg Wunsch * Stuff for handling job specifications 720b561052SJoerg Wunsch */ 73360d4ad5SWarner Losh extern uid_t uid, euid; 74360d4ad5SWarner Losh 750b561052SJoerg Wunsch static int col; /* column on screen */ 767b7fb4bbSKris Kennaway static char current[MAXNAMLEN+1]; /* current file being printed */ 777b7fb4bbSKris Kennaway static char file[MAXNAMLEN+1]; /* print file name */ 780b561052SJoerg Wunsch static int first; /* first file in ``files'' column? */ 790b561052SJoerg Wunsch static int garbage; /* # of garbage cf files */ 800b561052SJoerg Wunsch static int lflag; /* long output option */ 810b561052SJoerg Wunsch static int rank; /* order to be printed (-1=none, 0=active) */ 820b561052SJoerg Wunsch static long totsize; /* total print job size in bytes */ 830b561052SJoerg Wunsch 84ba7a1ad7SGarance A Drosehn static const char *head0 = "Rank Owner Job Files"; 85ba7a1ad7SGarance A Drosehn static const char *head1 = "Total Size\n"; 860b561052SJoerg Wunsch 87ba7a1ad7SGarance A Drosehn static void alarmhandler(int _signo); 88ba7a1ad7SGarance A Drosehn static void warn(const struct printer *_pp); 89334a9508SJoerg Wunsch 900b561052SJoerg Wunsch /* 910b561052SJoerg Wunsch * Display the current state of the queue. Format = 1 if long format. 920b561052SJoerg Wunsch */ 930b561052SJoerg Wunsch void 94ba7a1ad7SGarance A Drosehn displayq(struct printer *pp, int format) 950b561052SJoerg Wunsch { 9630b4b758SGarance A Drosehn register struct jobqueue *q; 97360d4ad5SWarner Losh register int i, nitems, fd, ret; 987b7fb4bbSKris Kennaway char *cp, *endp; 9930b4b758SGarance A Drosehn struct jobqueue **queue; 1000b561052SJoerg Wunsch struct stat statb; 1010b561052SJoerg Wunsch FILE *fp; 102334a9508SJoerg Wunsch void (*savealrm)(int); 1030b561052SJoerg Wunsch 1040b561052SJoerg Wunsch lflag = format; 1050b561052SJoerg Wunsch totsize = 0; 1060b561052SJoerg Wunsch rank = -1; 1074a1a0dbeSGarrett Wollman 1084a1a0dbeSGarrett Wollman if ((cp = checkremote(pp))) { 1090b561052SJoerg Wunsch printf("Warning: %s\n", cp); 1104a1a0dbeSGarrett Wollman free(cp); 1114a1a0dbeSGarrett Wollman } 1120b561052SJoerg Wunsch 1130b561052SJoerg Wunsch /* 1140b561052SJoerg Wunsch * Print out local queue 1150b561052SJoerg Wunsch * Find all the control files in the spooling directory 1160b561052SJoerg Wunsch */ 117360d4ad5SWarner Losh seteuid(euid); 1184a1a0dbeSGarrett Wollman if (chdir(pp->spool_dir) < 0) 1194a1a0dbeSGarrett Wollman fatal(pp, "cannot chdir to spooling directory: %s", 1204a1a0dbeSGarrett Wollman strerror(errno)); 121360d4ad5SWarner Losh seteuid(uid); 1224a1a0dbeSGarrett Wollman if ((nitems = getq(pp, &queue)) < 0) 1234a1a0dbeSGarrett Wollman fatal(pp, "cannot examine spooling area\n"); 124360d4ad5SWarner Losh seteuid(euid); 1254a1a0dbeSGarrett Wollman ret = stat(pp->lock_file, &statb); 126360d4ad5SWarner Losh seteuid(uid); 127360d4ad5SWarner Losh if (ret >= 0) { 1284a1a0dbeSGarrett Wollman if (statb.st_mode & LFM_PRINT_DIS) { 1294a1a0dbeSGarrett Wollman if (pp->remote) 130cc3fd56fSGarance A Drosehn printf("%s: ", local_host); 1314a1a0dbeSGarrett Wollman printf("Warning: %s is down: ", pp->printer); 132360d4ad5SWarner Losh seteuid(euid); 1334a1a0dbeSGarrett Wollman fd = open(pp->status_file, O_RDONLY|O_SHLOCK); 134360d4ad5SWarner Losh seteuid(uid); 1350b561052SJoerg Wunsch if (fd >= 0) { 1360b561052SJoerg Wunsch while ((i = read(fd, line, sizeof(line))) > 0) 1370b561052SJoerg Wunsch (void) fwrite(line, 1, i, stdout); 1380b561052SJoerg Wunsch (void) close(fd); /* unlocks as well */ 1390b561052SJoerg Wunsch } else 1400b561052SJoerg Wunsch putchar('\n'); 1410b561052SJoerg Wunsch } 1424a1a0dbeSGarrett Wollman if (statb.st_mode & LFM_QUEUE_DIS) { 1434a1a0dbeSGarrett Wollman if (pp->remote) 144cc3fd56fSGarance A Drosehn printf("%s: ", local_host); 1454a1a0dbeSGarrett Wollman printf("Warning: %s queue is turned off\n", 1464a1a0dbeSGarrett Wollman pp->printer); 1470b561052SJoerg Wunsch } 1480b561052SJoerg Wunsch } 1490b561052SJoerg Wunsch 1500b561052SJoerg Wunsch if (nitems) { 151360d4ad5SWarner Losh seteuid(euid); 1524a1a0dbeSGarrett Wollman fp = fopen(pp->lock_file, "r"); 153360d4ad5SWarner Losh seteuid(uid); 1540b561052SJoerg Wunsch if (fp == NULL) 1554a1a0dbeSGarrett Wollman warn(pp); 1560b561052SJoerg Wunsch else { 1570b561052SJoerg Wunsch /* get daemon pid */ 1580b561052SJoerg Wunsch cp = current; 1597b7fb4bbSKris Kennaway endp = cp + sizeof(current) - 1; 1607b7fb4bbSKris Kennaway while ((i = getc(fp)) != EOF && i != '\n') { 1617b7fb4bbSKris Kennaway if (cp < endp) 1625458e2f4SJoerg Wunsch *cp++ = i; 1637b7fb4bbSKris Kennaway } 1640b561052SJoerg Wunsch *cp = '\0'; 1650b561052SJoerg Wunsch i = atoi(current); 166360d4ad5SWarner Losh if (i <= 0) { 167360d4ad5SWarner Losh ret = -1; 168360d4ad5SWarner Losh } else { 169360d4ad5SWarner Losh seteuid(euid); 170360d4ad5SWarner Losh ret = kill(i, 0); 171360d4ad5SWarner Losh seteuid(uid); 172360d4ad5SWarner Losh } 173360d4ad5SWarner Losh if (ret < 0) { 1744a1a0dbeSGarrett Wollman warn(pp); 175360d4ad5SWarner Losh } else { 1760b561052SJoerg Wunsch /* read current file name */ 1770b561052SJoerg Wunsch cp = current; 1787b7fb4bbSKris Kennaway endp = cp + sizeof(current) - 1; 1797b7fb4bbSKris Kennaway while ((i = getc(fp)) != EOF && i != '\n') { 1807b7fb4bbSKris Kennaway if (cp < endp) 1815458e2f4SJoerg Wunsch *cp++ = i; 1827b7fb4bbSKris Kennaway } 1830b561052SJoerg Wunsch *cp = '\0'; 1840b561052SJoerg Wunsch /* 1850b561052SJoerg Wunsch * Print the status file. 1860b561052SJoerg Wunsch */ 1874a1a0dbeSGarrett Wollman if (pp->remote) 188cc3fd56fSGarance A Drosehn printf("%s: ", local_host); 189360d4ad5SWarner Losh seteuid(euid); 1904a1a0dbeSGarrett Wollman fd = open(pp->status_file, O_RDONLY|O_SHLOCK); 191360d4ad5SWarner Losh seteuid(uid); 1920b561052SJoerg Wunsch if (fd >= 0) { 1934a1a0dbeSGarrett Wollman while ((i = read(fd, line, 1944a1a0dbeSGarrett Wollman sizeof(line))) > 0) 1954a1a0dbeSGarrett Wollman fwrite(line, 1, i, stdout); 1964a1a0dbeSGarrett Wollman close(fd); /* unlocks as well */ 1970b561052SJoerg Wunsch } else 1980b561052SJoerg Wunsch putchar('\n'); 1990b561052SJoerg Wunsch } 2000b561052SJoerg Wunsch (void) fclose(fp); 2010b561052SJoerg Wunsch } 2020b561052SJoerg Wunsch /* 2030b561052SJoerg Wunsch * Now, examine the control files and print out the jobs to 2040b561052SJoerg Wunsch * be done for each user. 2050b561052SJoerg Wunsch */ 2060b561052SJoerg Wunsch if (!lflag) 2070b561052SJoerg Wunsch header(); 2080b561052SJoerg Wunsch for (i = 0; i < nitems; i++) { 2090b561052SJoerg Wunsch q = queue[i]; 21030b4b758SGarance A Drosehn inform(pp, q->job_cfname); 2110b561052SJoerg Wunsch free(q); 2120b561052SJoerg Wunsch } 2130b561052SJoerg Wunsch free(queue); 2140b561052SJoerg Wunsch } 2154a1a0dbeSGarrett Wollman if (!pp->remote) { 2160b561052SJoerg Wunsch if (nitems == 0) 2170b561052SJoerg Wunsch puts("no entries"); 2180b561052SJoerg Wunsch return; 2190b561052SJoerg Wunsch } 2200b561052SJoerg Wunsch 2210b561052SJoerg Wunsch /* 2220b561052SJoerg Wunsch * Print foreign queue 2230b561052SJoerg Wunsch * Note that a file in transit may show up in either queue. 2240b561052SJoerg Wunsch */ 2250b561052SJoerg Wunsch if (nitems) 2260b561052SJoerg Wunsch putchar('\n'); 2274a1a0dbeSGarrett Wollman (void) snprintf(line, sizeof(line), "%c%s", format ? '\4' : '\3', 2284a1a0dbeSGarrett Wollman pp->remote_queue); 2290b561052SJoerg Wunsch cp = line; 2306ee8b269SWarner Losh for (i = 0; i < requests && cp-line+10 < sizeof(line) - 1; i++) { 2310b561052SJoerg Wunsch cp += strlen(cp); 2320b561052SJoerg Wunsch (void) sprintf(cp, " %d", requ[i]); 2330b561052SJoerg Wunsch } 234d583a7c3SWarner Losh for (i = 0; i < users && cp - line + 1 + strlen(user[i]) < 2356ee8b269SWarner Losh sizeof(line) - 1; i++) { 2360b561052SJoerg Wunsch cp += strlen(cp); 2370b561052SJoerg Wunsch *cp++ = ' '; 2380b561052SJoerg Wunsch (void) strcpy(cp, user[i]); 2390b561052SJoerg Wunsch } 2400b561052SJoerg Wunsch strcat(line, "\n"); 241334a9508SJoerg Wunsch savealrm = signal(SIGALRM, alarmhandler); 2424a1a0dbeSGarrett Wollman alarm(pp->conn_timeout); 2434a1a0dbeSGarrett Wollman fd = getport(pp, pp->remote_host, 0); 2446d0727f4SJoerg Wunsch alarm(0); 245334a9508SJoerg Wunsch (void)signal(SIGALRM, savealrm); 2460b561052SJoerg Wunsch if (fd < 0) { 247cc3fd56fSGarance A Drosehn if (from_host != local_host) 248cc3fd56fSGarance A Drosehn printf("%s: ", local_host); 2494a1a0dbeSGarrett Wollman printf("connection to %s is down\n", pp->remote_host); 2500b561052SJoerg Wunsch } 2510b561052SJoerg Wunsch else { 2520b561052SJoerg Wunsch i = strlen(line); 2530b561052SJoerg Wunsch if (write(fd, line, i) != i) 2544a1a0dbeSGarrett Wollman fatal(pp, "Lost connection"); 2550b561052SJoerg Wunsch while ((i = read(fd, line, sizeof(line))) > 0) 2560b561052SJoerg Wunsch (void) fwrite(line, 1, i, stdout); 2570b561052SJoerg Wunsch (void) close(fd); 2580b561052SJoerg Wunsch } 2590b561052SJoerg Wunsch } 2600b561052SJoerg Wunsch 2610b561052SJoerg Wunsch /* 2620b561052SJoerg Wunsch * Print a warning message if there is no daemon present. 2630b561052SJoerg Wunsch */ 26436d0e2a3SJoerg Wunsch static void 265ba7a1ad7SGarance A Drosehn warn(const struct printer *pp) 2660b561052SJoerg Wunsch { 2674a1a0dbeSGarrett Wollman if (pp->remote) 268cc3fd56fSGarance A Drosehn printf("%s: ", local_host); 2690b561052SJoerg Wunsch puts("Warning: no daemon present"); 2700b561052SJoerg Wunsch current[0] = '\0'; 2710b561052SJoerg Wunsch } 2720b561052SJoerg Wunsch 2730b561052SJoerg Wunsch /* 2740b561052SJoerg Wunsch * Print the header for the short listing format 2750b561052SJoerg Wunsch */ 2760b561052SJoerg Wunsch void 277ba7a1ad7SGarance A Drosehn header(void) 2780b561052SJoerg Wunsch { 2796d39e1b7SGarance A Drosehn printf("%s", head0); 2800b561052SJoerg Wunsch col = strlen(head0)+1; 2810b561052SJoerg Wunsch blankfill(SIZCOL); 2826d39e1b7SGarance A Drosehn printf("%s", head1); 2830b561052SJoerg Wunsch } 2840b561052SJoerg Wunsch 2850b561052SJoerg Wunsch void 286ba7a1ad7SGarance A Drosehn inform(const struct printer *pp, char *cf) 2870b561052SJoerg Wunsch { 288ec6b8da5SGarance A Drosehn register int copycnt; 289ec6b8da5SGarance A Drosehn char savedname[MAXPATHLEN+1]; 2900b561052SJoerg Wunsch FILE *cfp; 2910b561052SJoerg Wunsch 2920b561052SJoerg Wunsch /* 2930b561052SJoerg Wunsch * There's a chance the control file has gone away 2940b561052SJoerg Wunsch * in the meantime; if this is the case just keep going 2950b561052SJoerg Wunsch */ 296360d4ad5SWarner Losh seteuid(euid); 2970b561052SJoerg Wunsch if ((cfp = fopen(cf, "r")) == NULL) 2980b561052SJoerg Wunsch return; 299360d4ad5SWarner Losh seteuid(uid); 3000b561052SJoerg Wunsch 3010b561052SJoerg Wunsch if (rank < 0) 3020b561052SJoerg Wunsch rank = 0; 3034a1a0dbeSGarrett Wollman if (pp->remote || garbage || strcmp(cf, current)) 3040b561052SJoerg Wunsch rank++; 305ec6b8da5SGarance A Drosehn 306ec6b8da5SGarance A Drosehn /* 307ec6b8da5SGarance A Drosehn * The cf-file may include commands to print more than one datafile 308ec6b8da5SGarance A Drosehn * from the user. For each datafile, the cf-file contains at least 309ec6b8da5SGarance A Drosehn * one line which starts with some format-specifier ('a'-'z'), and 310ec6b8da5SGarance A Drosehn * a second line ('N'ame) which indicates the original name the user 311ec6b8da5SGarance A Drosehn * specified for that file. There can be multiple format-spec lines 312ec6b8da5SGarance A Drosehn * for a single Name-line, if the user requested multiple copies of 313ec6b8da5SGarance A Drosehn * that file. Standard lpr puts the format-spec line(s) before the 314ec6b8da5SGarance A Drosehn * Name-line, while lprNG puts the Name-line before the format-spec 315ec6b8da5SGarance A Drosehn * line(s). This section needs to handle the lines in either order. 316ec6b8da5SGarance A Drosehn */ 317ec6b8da5SGarance A Drosehn copycnt = 0; 318ec6b8da5SGarance A Drosehn file[0] = '\0'; 319ec6b8da5SGarance A Drosehn savedname[0] = '\0'; 3200b561052SJoerg Wunsch while (getline(cfp)) { 3210b561052SJoerg Wunsch switch (line[0]) { 3220b561052SJoerg Wunsch case 'P': /* Was this file specified in the user's list? */ 3230b561052SJoerg Wunsch if (!inlist(line+1, cf)) { 3240b561052SJoerg Wunsch fclose(cfp); 3250b561052SJoerg Wunsch return; 3260b561052SJoerg Wunsch } 3270b561052SJoerg Wunsch if (lflag) { 3280b561052SJoerg Wunsch printf("\n%s: ", line+1); 3290b561052SJoerg Wunsch col = strlen(line+1) + 2; 3300b561052SJoerg Wunsch prank(rank); 3310b561052SJoerg Wunsch blankfill(JOBCOL); 3320b561052SJoerg Wunsch printf(" [job %s]\n", cf+3); 3330b561052SJoerg Wunsch } else { 3340b561052SJoerg Wunsch col = 0; 3350b561052SJoerg Wunsch prank(rank); 3360b561052SJoerg Wunsch blankfill(OWNCOL); 3370b561052SJoerg Wunsch printf("%-10s %-3d ", line+1, atoi(cf+3)); 3380b561052SJoerg Wunsch col += 16; 3390b561052SJoerg Wunsch first = 1; 3400b561052SJoerg Wunsch } 3410b561052SJoerg Wunsch continue; 3420b561052SJoerg Wunsch default: /* some format specifer and file name? */ 3430b561052SJoerg Wunsch if (line[0] < 'a' || line[0] > 'z') 344ec6b8da5SGarance A Drosehn break; 345ec6b8da5SGarance A Drosehn if (copycnt == 0 || strcmp(file, line+1) != 0) { 3465d7321f6SGarance A Drosehn strlcpy(file, line + 1, sizeof(file)); 347d583a7c3SWarner Losh } 348ec6b8da5SGarance A Drosehn copycnt++; 349ec6b8da5SGarance A Drosehn /* 350ec6b8da5SGarance A Drosehn * deliberately 'continue' to another getline(), so 351ec6b8da5SGarance A Drosehn * all format-spec lines for this datafile are read 352ec6b8da5SGarance A Drosehn * in and counted before calling show() 353ec6b8da5SGarance A Drosehn */ 3540b561052SJoerg Wunsch continue; 3550b561052SJoerg Wunsch case 'N': 3565d7321f6SGarance A Drosehn strlcpy(savedname, line + 1, sizeof(savedname)); 357ec6b8da5SGarance A Drosehn break; 358ec6b8da5SGarance A Drosehn } 359ec6b8da5SGarance A Drosehn if ((file[0] != '\0') && (savedname[0] != '\0')) { 360ec6b8da5SGarance A Drosehn show(savedname, file, copycnt); 361ec6b8da5SGarance A Drosehn copycnt = 0; 3620b561052SJoerg Wunsch file[0] = '\0'; 363ec6b8da5SGarance A Drosehn savedname[0] = '\0'; 3640b561052SJoerg Wunsch } 3650b561052SJoerg Wunsch } 3660b561052SJoerg Wunsch fclose(cfp); 367ec6b8da5SGarance A Drosehn /* check for a file which hasn't been shown yet */ 368ec6b8da5SGarance A Drosehn if (file[0] != '\0') { 369ec6b8da5SGarance A Drosehn if (savedname[0] == '\0') { 370ec6b8da5SGarance A Drosehn /* a safeguard in case the N-ame line is missing */ 3715d7321f6SGarance A Drosehn strlcpy(savedname, file, sizeof(savedname)); 372ec6b8da5SGarance A Drosehn } 373ec6b8da5SGarance A Drosehn show(savedname, file, copycnt); 374ec6b8da5SGarance A Drosehn } 3750b561052SJoerg Wunsch if (!lflag) { 3760b561052SJoerg Wunsch blankfill(SIZCOL); 3770b561052SJoerg Wunsch printf("%ld bytes\n", totsize); 3780b561052SJoerg Wunsch totsize = 0; 3790b561052SJoerg Wunsch } 3800b561052SJoerg Wunsch } 3810b561052SJoerg Wunsch 3820b561052SJoerg Wunsch int 383ba7a1ad7SGarance A Drosehn inlist(char *uname, char *cfile) 3840b561052SJoerg Wunsch { 3850b561052SJoerg Wunsch register int *r, n; 3860b561052SJoerg Wunsch register char **u, *cp; 3870b561052SJoerg Wunsch 3880b561052SJoerg Wunsch if (users == 0 && requests == 0) 3890b561052SJoerg Wunsch return(1); 3900b561052SJoerg Wunsch /* 3910b561052SJoerg Wunsch * Check to see if it's in the user list 3920b561052SJoerg Wunsch */ 3930b561052SJoerg Wunsch for (u = user; u < &user[users]; u++) 394ba7a1ad7SGarance A Drosehn if (!strcmp(*u, uname)) 3950b561052SJoerg Wunsch return(1); 3960b561052SJoerg Wunsch /* 3970b561052SJoerg Wunsch * Check the request list 3980b561052SJoerg Wunsch */ 399ba7a1ad7SGarance A Drosehn for (n = 0, cp = cfile+3; isdigit(*cp); ) 4000b561052SJoerg Wunsch n = n * 10 + (*cp++ - '0'); 4010b561052SJoerg Wunsch for (r = requ; r < &requ[requests]; r++) 402cc3fd56fSGarance A Drosehn if (*r == n && !strcmp(cp, from_host)) 4030b561052SJoerg Wunsch return(1); 4040b561052SJoerg Wunsch return(0); 4050b561052SJoerg Wunsch } 4060b561052SJoerg Wunsch 4070b561052SJoerg Wunsch void 408ba7a1ad7SGarance A Drosehn show(const char *nfile, const char *datafile, int copies) 4090b561052SJoerg Wunsch { 4100b561052SJoerg Wunsch if (strcmp(nfile, " ") == 0) 4110b561052SJoerg Wunsch nfile = "(standard input)"; 4120b561052SJoerg Wunsch if (lflag) 413ba7a1ad7SGarance A Drosehn ldump(nfile, datafile, copies); 4140b561052SJoerg Wunsch else 415ba7a1ad7SGarance A Drosehn dump(nfile, datafile, copies); 4160b561052SJoerg Wunsch } 4170b561052SJoerg Wunsch 4180b561052SJoerg Wunsch /* 4190b561052SJoerg Wunsch * Fill the line with blanks to the specified column 4200b561052SJoerg Wunsch */ 4210b561052SJoerg Wunsch void 422ba7a1ad7SGarance A Drosehn blankfill(int tocol) 4230b561052SJoerg Wunsch { 424ba7a1ad7SGarance A Drosehn while (col++ < tocol) 4250b561052SJoerg Wunsch putchar(' '); 4260b561052SJoerg Wunsch } 4270b561052SJoerg Wunsch 4280b561052SJoerg Wunsch /* 4290b561052SJoerg Wunsch * Give the abbreviated dump of the file names 4300b561052SJoerg Wunsch */ 4310b561052SJoerg Wunsch void 432ba7a1ad7SGarance A Drosehn dump(const char *nfile, const char *datafile, int copies) 4330b561052SJoerg Wunsch { 4340b561052SJoerg Wunsch struct stat lbuf; 435be794da7SGarance A Drosehn const char etctmpl[] = ", ..."; 436be794da7SGarance A Drosehn char etc[sizeof(etctmpl)]; 437be794da7SGarance A Drosehn char *lastsep; 438be794da7SGarance A Drosehn short fill, nlen; 439be794da7SGarance A Drosehn short rem, remetc; 4400b561052SJoerg Wunsch 4410b561052SJoerg Wunsch /* 442be794da7SGarance A Drosehn * Print as many filenames as will fit 443be794da7SGarance A Drosehn * (leaving room for the 'total size' field) 4440b561052SJoerg Wunsch */ 4450b561052SJoerg Wunsch fill = first ? 0 : 2; /* fill space for ``, '' */ 446be794da7SGarance A Drosehn nlen = strlen(nfile); 447be794da7SGarance A Drosehn rem = SIZCOL - 1 - col; 448be794da7SGarance A Drosehn if (nlen + fill > rem) { 449be794da7SGarance A Drosehn if (first) { 450be794da7SGarance A Drosehn /* print the right-most part of the name */ 451be794da7SGarance A Drosehn printf("...%s ", &nfile[3+nlen-rem]); 452be794da7SGarance A Drosehn col = SIZCOL; 453be794da7SGarance A Drosehn } else if (rem > 0) { 454be794da7SGarance A Drosehn /* fit as much of the etc-string as we can */ 455be794da7SGarance A Drosehn remetc = rem; 456be794da7SGarance A Drosehn if (rem > strlen(etctmpl)) 457be794da7SGarance A Drosehn remetc = strlen(etctmpl); 458be794da7SGarance A Drosehn etc[0] = '\0'; 459be794da7SGarance A Drosehn strncat(etc, etctmpl, remetc); 4606d39e1b7SGarance A Drosehn printf("%s", etc); 461be794da7SGarance A Drosehn col += remetc; 462be794da7SGarance A Drosehn rem -= remetc; 463be794da7SGarance A Drosehn /* room for the last segment of this filename? */ 464be794da7SGarance A Drosehn lastsep = strrchr(nfile, '/'); 465be794da7SGarance A Drosehn if ((lastsep != NULL) && (rem > strlen(lastsep))) { 466be794da7SGarance A Drosehn /* print the right-most part of this name */ 467be794da7SGarance A Drosehn printf("%s", lastsep); 468be794da7SGarance A Drosehn col += strlen(lastsep); 469be794da7SGarance A Drosehn } else { 470be794da7SGarance A Drosehn /* do not pack any more names in here */ 4710b561052SJoerg Wunsch blankfill(SIZCOL); 4720b561052SJoerg Wunsch } 473be794da7SGarance A Drosehn } 4740b561052SJoerg Wunsch } else { 475be794da7SGarance A Drosehn if (!first) 4760b561052SJoerg Wunsch printf(", "); 4770b561052SJoerg Wunsch printf("%s", nfile); 478be794da7SGarance A Drosehn col += nlen + fill; 4790b561052SJoerg Wunsch } 480be794da7SGarance A Drosehn first = 0; 481be794da7SGarance A Drosehn 482360d4ad5SWarner Losh seteuid(euid); 483ba7a1ad7SGarance A Drosehn if (*datafile && !stat(datafile, &lbuf)) 4840b561052SJoerg Wunsch totsize += copies * lbuf.st_size; 485360d4ad5SWarner Losh seteuid(uid); 4860b561052SJoerg Wunsch } 4870b561052SJoerg Wunsch 4880b561052SJoerg Wunsch /* 4890b561052SJoerg Wunsch * Print the long info about the file 4900b561052SJoerg Wunsch */ 4910b561052SJoerg Wunsch void 492ba7a1ad7SGarance A Drosehn ldump(const char *nfile, const char *datafile, int copies) 4930b561052SJoerg Wunsch { 4940b561052SJoerg Wunsch struct stat lbuf; 4950b561052SJoerg Wunsch 4960b561052SJoerg Wunsch putchar('\t'); 4970b561052SJoerg Wunsch if (copies > 1) 4980b561052SJoerg Wunsch printf("%-2d copies of %-19s", copies, nfile); 4990b561052SJoerg Wunsch else 5000b561052SJoerg Wunsch printf("%-32s", nfile); 501ba7a1ad7SGarance A Drosehn if (*datafile && !stat(datafile, &lbuf)) 502ef1c4c53SJohn Birrell printf(" %qd bytes", (long long) lbuf.st_size); 5030b561052SJoerg Wunsch else 5040b561052SJoerg Wunsch printf(" ??? bytes"); 5050b561052SJoerg Wunsch putchar('\n'); 5060b561052SJoerg Wunsch } 5070b561052SJoerg Wunsch 5080b561052SJoerg Wunsch /* 5090b561052SJoerg Wunsch * Print the job's rank in the queue, 5100b561052SJoerg Wunsch * update col for screen management 5110b561052SJoerg Wunsch */ 5120b561052SJoerg Wunsch void 513ba7a1ad7SGarance A Drosehn prank(int n) 5140b561052SJoerg Wunsch { 5150b561052SJoerg Wunsch char rline[100]; 516ba7a1ad7SGarance A Drosehn static const char *r[] = { 5170b561052SJoerg Wunsch "th", "st", "nd", "rd", "th", "th", "th", "th", "th", "th" 5180b561052SJoerg Wunsch }; 5190b561052SJoerg Wunsch 5200b561052SJoerg Wunsch if (n == 0) { 5210b561052SJoerg Wunsch printf("active"); 5220b561052SJoerg Wunsch col += 6; 5230b561052SJoerg Wunsch return; 5240b561052SJoerg Wunsch } 5250b561052SJoerg Wunsch if ((n/10)%10 == 1) 5260b561052SJoerg Wunsch (void)snprintf(rline, sizeof(rline), "%dth", n); 5270b561052SJoerg Wunsch else 5280b561052SJoerg Wunsch (void)snprintf(rline, sizeof(rline), "%d%s", n, r[n%10]); 5290b561052SJoerg Wunsch col += strlen(rline); 5300b561052SJoerg Wunsch printf("%s", rline); 5310b561052SJoerg Wunsch } 532334a9508SJoerg Wunsch 533334a9508SJoerg Wunsch void 534ba7a1ad7SGarance A Drosehn alarmhandler(int signo __unused) 535334a9508SJoerg Wunsch { 536ba7a1ad7SGarance A Drosehn /* the signal is ignored */ 537ba7a1ad7SGarance A Drosehn /* (the '__unused' is just to avoid a compile-time warning) */ 538334a9508SJoerg Wunsch } 539