14b88c807SRodney W. Grimes /*- 24b88c807SRodney W. Grimes * Copyright (c) 1990, 1993, 1994 34b88c807SRodney W. Grimes * The Regents of the University of California. All rights reserved. 44b88c807SRodney W. Grimes * 54b88c807SRodney W. Grimes * Redistribution and use in source and binary forms, with or without 64b88c807SRodney W. Grimes * modification, are permitted provided that the following conditions 74b88c807SRodney W. Grimes * are met: 84b88c807SRodney W. Grimes * 1. Redistributions of source code must retain the above copyright 94b88c807SRodney W. Grimes * notice, this list of conditions and the following disclaimer. 104b88c807SRodney W. Grimes * 2. Redistributions in binary form must reproduce the above copyright 114b88c807SRodney W. Grimes * notice, this list of conditions and the following disclaimer in the 124b88c807SRodney W. Grimes * documentation and/or other materials provided with the distribution. 134b88c807SRodney W. Grimes * 4. Neither the name of the University nor the names of its contributors 144b88c807SRodney W. Grimes * may be used to endorse or promote products derived from this software 154b88c807SRodney W. Grimes * without specific prior written permission. 164b88c807SRodney W. Grimes * 174b88c807SRodney W. Grimes * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 184b88c807SRodney W. Grimes * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 194b88c807SRodney W. Grimes * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 204b88c807SRodney W. Grimes * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 214b88c807SRodney W. Grimes * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 224b88c807SRodney W. Grimes * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 234b88c807SRodney W. Grimes * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 244b88c807SRodney W. Grimes * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 254b88c807SRodney W. Grimes * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 264b88c807SRodney W. Grimes * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 274b88c807SRodney W. Grimes * SUCH DAMAGE. 284b88c807SRodney W. Grimes */ 294b88c807SRodney W. Grimes 30aeb7f2b6SSteve Price #if 0 31871e8d8cSMark Murray #ifndef lint 32aeb7f2b6SSteve Price static char sccsid[] = "@(#)keyword.c 8.5 (Berkeley) 4/2/94"; 334b88c807SRodney W. Grimes #endif /* not lint */ 34871e8d8cSMark Murray #endif 352749b141SDavid E. O'Brien #include <sys/cdefs.h> 362749b141SDavid E. O'Brien __FBSDID("$FreeBSD$"); 374b88c807SRodney W. Grimes 384b88c807SRodney W. Grimes #include <sys/param.h> 394b88c807SRodney W. Grimes #include <sys/time.h> 404b88c807SRodney W. Grimes #include <sys/resource.h> 414b88c807SRodney W. Grimes #include <sys/proc.h> 42c1cee2f6SSteve Price #include <sys/sysctl.h> 43c1cee2f6SSteve Price #include <sys/user.h> 444b88c807SRodney W. Grimes 454b88c807SRodney W. Grimes #include <err.h> 464b88c807SRodney W. Grimes #include <stddef.h> 474b88c807SRodney W. Grimes #include <stdio.h> 484b88c807SRodney W. Grimes #include <stdlib.h> 494b88c807SRodney W. Grimes #include <string.h> 504b88c807SRodney W. Grimes 514b88c807SRodney W. Grimes #include "ps.h" 524b88c807SRodney W. Grimes 5378b1878aSJuli Mallett static VAR *findvar(char *, int, char **header); 5446251ddeSWarner Losh static int vcmp(const void *, const void *); 554b88c807SRodney W. Grimes 564b88c807SRodney W. Grimes /* Compute offset in common structures. */ 571f7d2501SKirk McKusick #define KOFF(x) offsetof(struct kinfo_proc, x) 584b88c807SRodney W. Grimes #define ROFF(x) offsetof(struct rusage, x) 594b88c807SRodney W. Grimes 6015b87b53SGarance A Drosehn #define EMULLEN 13 /* enough for "FreeBSD ELF32" */ 6176adc1fdSGarance A Drosehn #define LWPFMT "d" 6276adc1fdSGarance A Drosehn #define LWPLEN 6 6376adc1fdSGarance A Drosehn #define NLWPFMT "d" 6476adc1fdSGarance A Drosehn #define NLWPLEN 4 654b88c807SRodney W. Grimes #define UIDFMT "u" 664b88c807SRodney W. Grimes #define UIDLEN 5 674b88c807SRodney W. Grimes #define PIDFMT "d" 684b88c807SRodney W. Grimes #define PIDLEN 5 69a755f1c9SRobert Drehmel #define USERLEN (MAXLOGNAME - 1) 704b88c807SRodney W. Grimes 716f15bc16SMaxim Sobolev /* PLEASE KEEP THE TABLE BELOW SORTED ALPHABETICALLY!!! */ 72871e8d8cSMark Murray static VAR var[] = { 73871e8d8cSMark Murray {"%cpu", "%CPU", NULL, 0, pcpu, NULL, 4, 0, CHAR, NULL, 0}, 74871e8d8cSMark Murray {"%mem", "%MEM", NULL, 0, pmem, NULL, 4, 0, CHAR, NULL, 0}, 75871e8d8cSMark Murray {"acflag", "ACFLG", NULL, 0, kvar, NULL, 3, KOFF(ki_acflag), USHORT, 76871e8d8cSMark Murray "x", 0}, 77871e8d8cSMark Murray {"acflg", "", "acflag", 0, NULL, NULL, 0, 0, CHAR, NULL, 0}, 7803334017SJuli Mallett {"args", "COMMAND", NULL, COMM|LJUST|USER, arguments, NULL, 16, 0, 7903334017SJuli Mallett CHAR, NULL, 0}, 80871e8d8cSMark Murray {"blocked", "", "sigmask", 0, NULL, NULL, 0, 0, CHAR, NULL, 0}, 81871e8d8cSMark Murray {"caught", "", "sigcatch", 0, NULL, NULL, 0, 0, CHAR, NULL, 0}, 8203334017SJuli Mallett {"comm", "COMMAND", NULL, LJUST, ucomm, NULL, MAXCOMLEN, 0, CHAR, 8303334017SJuli Mallett NULL, 0}, 8403334017SJuli Mallett {"command", "COMMAND", NULL, COMM|LJUST|USER, command, NULL, 16, 0, 85d9cd71b6SJuli Mallett CHAR, NULL, 0}, 86871e8d8cSMark Murray {"cpu", "CPU", NULL, 0, kvar, NULL, 3, KOFF(ki_estcpu), UINT, "d", 87871e8d8cSMark Murray 0}, 88871e8d8cSMark Murray {"cputime", "", "time", 0, NULL, NULL, 0, 0, CHAR, NULL, 0}, 8915b87b53SGarance A Drosehn {"emul", "EMUL", NULL, LJUST, emulname, NULL, EMULLEN, 0, CHAR, 9015b87b53SGarance A Drosehn NULL, 0}, 91eaed5652SPhilippe Charnier {"etime", "ELAPSED", NULL, USER, elapsed, NULL, 12, 0, CHAR, NULL, 0}, 92871e8d8cSMark Murray {"f", "F", NULL, 0, kvar, NULL, 7, KOFF(ki_flag), INT, "x", 0}, 93871e8d8cSMark Murray {"flags", "", "f", 0, NULL, NULL, 0, 0, CHAR, NULL, 0}, 94871e8d8cSMark Murray {"ignored", "", "sigignore", 0, NULL, NULL, 0, 0, CHAR, NULL, 0}, 95871e8d8cSMark Murray {"inblk", "INBLK", NULL, USER, rvar, NULL, 4, ROFF(ru_inblock), LONG, 96871e8d8cSMark Murray "ld", 0}, 97871e8d8cSMark Murray {"inblock", "", "inblk", 0, NULL, NULL, 0, 0, CHAR, NULL, 0}, 98a990344fSPawel Jakub Dawidek {"jid", "JID", NULL, 0, kvar, NULL, 6, KOFF(ki_jid), INT, "d", 0}, 99871e8d8cSMark Murray {"jobc", "JOBC", NULL, 0, kvar, NULL, 4, KOFF(ki_jobc), SHORT, "d", 100871e8d8cSMark Murray 0}, 101871e8d8cSMark Murray {"ktrace", "KTRACE", NULL, 0, kvar, NULL, 8, KOFF(ki_traceflag), INT, 102871e8d8cSMark Murray "x", 0}, 1032af538ebSRobert Watson {"label", "LABEL", NULL, LJUST|DSIZ, label, s_label, SHRT_MAX, 0, CHAR, 1042af538ebSRobert Watson NULL, 0}, 105871e8d8cSMark Murray {"lim", "LIM", NULL, 0, maxrss, NULL, 5, 0, CHAR, NULL, 0}, 1066f15bc16SMaxim Sobolev {"lockname", "LOCK", NULL, LJUST, lockname, NULL, 6, 0, CHAR, NULL, 1076f15bc16SMaxim Sobolev 0}, 108871e8d8cSMark Murray {"login", "LOGIN", NULL, LJUST, logname, NULL, MAXLOGNAME-1, 0, CHAR, 109871e8d8cSMark Murray NULL, 0}, 110871e8d8cSMark Murray {"logname", "", "login", 0, NULL, NULL, 0, 0, CHAR, NULL, 0}, 111871e8d8cSMark Murray {"lstart", "STARTED", NULL, LJUST|USER, lstarted, NULL, 28, 0, CHAR, 112871e8d8cSMark Murray NULL, 0}, 11376adc1fdSGarance A Drosehn {"lwp", "LWP", NULL, 0, kvar, NULL, LWPLEN, KOFF(ki_tid), UINT, 11476adc1fdSGarance A Drosehn LWPFMT, 0}, 115871e8d8cSMark Murray {"majflt", "MAJFLT", NULL, USER, rvar, NULL, 4, ROFF(ru_majflt), 116871e8d8cSMark Murray LONG, "ld", 0}, 117871e8d8cSMark Murray {"minflt", "MINFLT", NULL, USER, rvar, NULL, 4, ROFF(ru_minflt), 118871e8d8cSMark Murray LONG, "ld", 0}, 119871e8d8cSMark Murray {"msgrcv", "MSGRCV", NULL, USER, rvar, NULL, 4, ROFF(ru_msgrcv), 120871e8d8cSMark Murray LONG, "ld", 0}, 121871e8d8cSMark Murray {"msgsnd", "MSGSND", NULL, USER, rvar, NULL, 4, ROFF(ru_msgsnd), 122871e8d8cSMark Murray LONG, "ld", 0}, 123d9a5f890SMatthew Dillon {"mwchan", "MWCHAN", NULL, LJUST, mwchan, NULL, 6, 0, CHAR, NULL, 0}, 124871e8d8cSMark Murray {"ni", "", "nice", 0, NULL, NULL, 0, 0, CHAR, NULL, 0}, 125871e8d8cSMark Murray {"nice", "NI", NULL, 0, kvar, NULL, 2, KOFF(ki_nice), CHAR, "d", 126871e8d8cSMark Murray 0}, 127871e8d8cSMark Murray {"nivcsw", "NIVCSW", NULL, USER, rvar, NULL, 5, ROFF(ru_nivcsw), 128871e8d8cSMark Murray LONG, "ld", 0}, 12976adc1fdSGarance A Drosehn {"nlwp", "NLWP", NULL, 0, kvar, NULL, NLWPLEN, KOFF(ki_numthreads), 13076adc1fdSGarance A Drosehn UINT, NLWPFMT, 0}, 131871e8d8cSMark Murray {"nsignals", "", "nsigs", 0, NULL, NULL, 0, 0, CHAR, NULL, 0}, 132871e8d8cSMark Murray {"nsigs", "NSIGS", NULL, USER, rvar, NULL, 4, ROFF(ru_nsignals), 133871e8d8cSMark Murray LONG, "ld", 0}, 134871e8d8cSMark Murray {"nswap", "NSWAP", NULL, USER, rvar, NULL, 4, ROFF(ru_nswap), 135871e8d8cSMark Murray LONG, "ld", 0}, 136871e8d8cSMark Murray {"nvcsw", "NVCSW", NULL, USER, rvar, NULL, 5, ROFF(ru_nvcsw), 137871e8d8cSMark Murray LONG, "ld", 0}, 138de244df7SHartmut Brandt {"nwchan", "NWCHAN", NULL, LJUST, nwchan, NULL, 8, 0, CHAR, NULL, 0}, 139871e8d8cSMark Murray {"oublk", "OUBLK", NULL, USER, rvar, NULL, 4, ROFF(ru_oublock), 140871e8d8cSMark Murray LONG, "ld", 0}, 141871e8d8cSMark Murray {"oublock", "", "oublk", 0, NULL, NULL, 0, 0, CHAR, NULL, 0}, 142871e8d8cSMark Murray {"paddr", "PADDR", NULL, 0, kvar, NULL, 8, KOFF(ki_paddr), KPTR, 143871e8d8cSMark Murray "lx", 0}, 144871e8d8cSMark Murray {"pagein", "PAGEIN", NULL, USER, pagein, NULL, 6, 0, CHAR, NULL, 0}, 145871e8d8cSMark Murray {"pcpu", "", "%cpu", 0, NULL, NULL, 0, 0, CHAR, NULL, 0}, 146871e8d8cSMark Murray {"pending", "", "sig", 0, NULL, NULL, 0, 0, CHAR, NULL, 0}, 147871e8d8cSMark Murray {"pgid", "PGID", NULL, 0, kvar, NULL, PIDLEN, KOFF(ki_pgid), UINT, 148871e8d8cSMark Murray PIDFMT, 0}, 149871e8d8cSMark Murray {"pid", "PID", NULL, 0, kvar, NULL, PIDLEN, KOFF(ki_pid), UINT, 150871e8d8cSMark Murray PIDFMT, 0}, 151871e8d8cSMark Murray {"pmem", "", "%mem", 0, NULL, NULL, 0, 0, CHAR, NULL, 0}, 152871e8d8cSMark Murray {"ppid", "PPID", NULL, 0, kvar, NULL, PIDLEN, KOFF(ki_ppid), UINT, 153871e8d8cSMark Murray PIDFMT, 0}, 154871e8d8cSMark Murray {"pri", "PRI", NULL, 0, pri, NULL, 3, 0, CHAR, NULL, 0}, 155e2c9ac69STim J. Robbins {"re", "RE", NULL, INF127, kvar, NULL, 3, KOFF(ki_swtime), UINT, "d", 156871e8d8cSMark Murray 0}, 1571f7d2501SKirk McKusick {"rgid", "RGID", NULL, 0, kvar, NULL, UIDLEN, KOFF(ki_rgid), 158871e8d8cSMark Murray UINT, UIDFMT, 0}, 159e8eef4bbSJuli Mallett {"rgroup", "RGROUP", NULL, LJUST|DSIZ, rgroupname, s_rgroupname, 160e8eef4bbSJuli Mallett USERLEN, 0, CHAR, NULL, 0}, 161237d4cefSChristian S.J. Peron {"rss", "RSS", NULL, 0, kvar, NULL, 5, KOFF(ki_rssize), PGTOK, "ld", 0}, 162871e8d8cSMark Murray {"rtprio", "RTPRIO", NULL, 0, priorityr, NULL, 7, KOFF(ki_pri), CHAR, 163871e8d8cSMark Murray NULL, 0}, 1641f7d2501SKirk McKusick {"ruid", "RUID", NULL, 0, kvar, NULL, UIDLEN, KOFF(ki_ruid), 165871e8d8cSMark Murray UINT, UIDFMT, 0}, 166871e8d8cSMark Murray {"ruser", "RUSER", NULL, LJUST|DSIZ, runame, s_runame, USERLEN, 167871e8d8cSMark Murray 0, CHAR, NULL, 0}, 168ff1b168bSBrian Somers {"sid", "SID", NULL, 0, kvar, NULL, PIDLEN, KOFF(ki_sid), UINT, 169871e8d8cSMark Murray PIDFMT, 0}, 170871e8d8cSMark Murray {"sig", "PENDING", NULL, 0, kvar, NULL, 8, KOFF(ki_siglist), INT, 171871e8d8cSMark Murray "x", 0}, 172871e8d8cSMark Murray {"sigcatch", "CAUGHT", NULL, 0, kvar, NULL, 8, KOFF(ki_sigcatch), 173871e8d8cSMark Murray UINT, "x", 0}, 174871e8d8cSMark Murray {"sigignore", "IGNORED", NULL, 0, kvar, NULL, 8, KOFF(ki_sigignore), 175871e8d8cSMark Murray UINT, "x", 0}, 176871e8d8cSMark Murray {"sigmask", "BLOCKED", NULL, 0, kvar, NULL, 8, KOFF(ki_sigmask), 177871e8d8cSMark Murray UINT, "x", 0}, 178e2c9ac69STim J. Robbins {"sl", "SL", NULL, INF127, kvar, NULL, 3, KOFF(ki_slptime), UINT, "d", 179871e8d8cSMark Murray 0}, 180871e8d8cSMark Murray {"start", "STARTED", NULL, LJUST|USER, started, NULL, 7, 0, CHAR, NULL, 181871e8d8cSMark Murray 0}, 182871e8d8cSMark Murray {"stat", "", "state", 0, NULL, NULL, 0, 0, CHAR, NULL, 0}, 183871e8d8cSMark Murray {"state", "STAT", NULL, 0, state, NULL, 4, 0, CHAR, NULL, 0}, 184871e8d8cSMark Murray {"svgid", "SVGID", NULL, 0, kvar, NULL, UIDLEN, KOFF(ki_svgid), 185871e8d8cSMark Murray UINT, UIDFMT, 0}, 186871e8d8cSMark Murray {"svuid", "SVUID", NULL, 0, kvar, NULL, UIDLEN, KOFF(ki_svuid), 187871e8d8cSMark Murray UINT, UIDFMT, 0}, 188871e8d8cSMark Murray {"tdev", "TDEV", NULL, 0, tdev, NULL, 4, 0, CHAR, NULL, 0}, 189871e8d8cSMark Murray {"time", "TIME", NULL, USER, cputime, NULL, 9, 0, CHAR, NULL, 0}, 190871e8d8cSMark Murray {"tpgid", "TPGID", NULL, 0, kvar, NULL, 4, KOFF(ki_tpgid), UINT, 191871e8d8cSMark Murray PIDFMT, 0}, 192ff1b168bSBrian Somers {"tsid", "TSID", NULL, 0, kvar, NULL, PIDLEN, KOFF(ki_tsid), UINT, 193871e8d8cSMark Murray PIDFMT, 0}, 194362d62baSJuli Mallett {"tsiz", "TSIZ", NULL, 0, kvar, NULL, 4, KOFF(ki_tsize), PGTOK, "ld", 0}, 195871e8d8cSMark Murray {"tt", "TT ", NULL, 0, tname, NULL, 4, 0, CHAR, NULL, 0}, 196871e8d8cSMark Murray {"tty", "TTY", NULL, LJUST, longtname, NULL, 8, 0, CHAR, NULL, 0}, 197871e8d8cSMark Murray {"ucomm", "UCOMM", NULL, LJUST, ucomm, NULL, MAXCOMLEN, 0, CHAR, NULL, 198871e8d8cSMark Murray 0}, 199871e8d8cSMark Murray {"uid", "UID", NULL, 0, kvar, NULL, UIDLEN, KOFF(ki_uid), UINT, 200871e8d8cSMark Murray UIDFMT, 0}, 201820434b2SGarance A Drosehn {"upr", "UPR", NULL, 0, upr, NULL, 3, 0, CHAR, NULL, 0}, 2027cb9663bSJuli Mallett {"uprocp", "UPROCP", NULL, 0, kvar, NULL, 8, KOFF(ki_paddr), KPTR, 2037cb9663bSJuli Mallett "lx", 0}, 204871e8d8cSMark Murray {"user", "USER", NULL, LJUST|DSIZ, uname, s_uname, USERLEN, 0, CHAR, 205871e8d8cSMark Murray NULL, 0}, 206871e8d8cSMark Murray {"usrpri", "", "upr", 0, NULL, NULL, 0, 0, CHAR, NULL, 0}, 207871e8d8cSMark Murray {"vsize", "", "vsz", 0, NULL, NULL, 0, 0, CHAR, NULL, 0}, 208871e8d8cSMark Murray {"vsz", "VSZ", NULL, 0, vsize, NULL, 5, 0, CHAR, NULL, 0}, 209871e8d8cSMark Murray {"wchan", "WCHAN", NULL, LJUST, wchan, NULL, 6, 0, CHAR, NULL, 0}, 210871e8d8cSMark Murray {"xstat", "XSTAT", NULL, 0, kvar, NULL, 4, KOFF(ki_xstat), USHORT, 211871e8d8cSMark Murray "x", 0}, 212871e8d8cSMark Murray {"", NULL, NULL, 0, NULL, NULL, 0, 0, CHAR, NULL, 0}, 2134b88c807SRodney W. Grimes }; 2144b88c807SRodney W. Grimes 2154b88c807SRodney W. Grimes void 21646251ddeSWarner Losh showkey(void) 2174b88c807SRodney W. Grimes { 2184b88c807SRodney W. Grimes VAR *v; 2194b88c807SRodney W. Grimes int i; 220871e8d8cSMark Murray const char *p, *sep; 2214b88c807SRodney W. Grimes 2224b88c807SRodney W. Grimes i = 0; 2234b88c807SRodney W. Grimes sep = ""; 2244b88c807SRodney W. Grimes for (v = var; *(p = v->name); ++v) { 2254b88c807SRodney W. Grimes int len = strlen(p); 2264b88c807SRodney W. Grimes if (termwidth && (i += len + 1) > termwidth) { 2274b88c807SRodney W. Grimes i = len; 2284b88c807SRodney W. Grimes sep = "\n"; 2294b88c807SRodney W. Grimes } 2304b88c807SRodney W. Grimes (void) printf("%s%s", sep, p); 2314b88c807SRodney W. Grimes sep = " "; 2324b88c807SRodney W. Grimes } 2334b88c807SRodney W. Grimes (void) printf("\n"); 2344b88c807SRodney W. Grimes } 2354b88c807SRodney W. Grimes 2364b88c807SRodney W. Grimes void 237fde411d5SJuli Mallett parsefmt(const char *p, int user) 2384b88c807SRodney W. Grimes { 239871e8d8cSMark Murray char *tempstr, *tempstr1; 2404b88c807SRodney W. Grimes 2412dda9fe9SJuli Mallett #define FMTSEP " \t,\n" 242871e8d8cSMark Murray tempstr1 = tempstr = strdup(p); 243871e8d8cSMark Murray while (tempstr && *tempstr) { 24478b1878aSJuli Mallett char *cp, *hp; 2454b88c807SRodney W. Grimes VAR *v; 2464b88c807SRodney W. Grimes struct varent *vent; 2474b88c807SRodney W. Grimes 24840fea9c3SJuli Mallett /* 2492dda9fe9SJuli Mallett * If an item contains an equals sign, it specifies a column 2502dda9fe9SJuli Mallett * header, may contain embedded separator characters and 2512dda9fe9SJuli Mallett * is always the last item. 25240fea9c3SJuli Mallett */ 2532dda9fe9SJuli Mallett if (tempstr[strcspn(tempstr, "="FMTSEP)] != '=') 2542dda9fe9SJuli Mallett while ((cp = strsep(&tempstr, FMTSEP)) != NULL && 2552dda9fe9SJuli Mallett *cp == '\0') 2564b88c807SRodney W. Grimes /* void */; 2572dda9fe9SJuli Mallett else { 2582dda9fe9SJuli Mallett cp = tempstr; 2592dda9fe9SJuli Mallett tempstr = NULL; 2602dda9fe9SJuli Mallett } 26178b1878aSJuli Mallett if (cp == NULL || !(v = findvar(cp, user, &hp))) 2624b88c807SRodney W. Grimes continue; 263fde411d5SJuli Mallett if (!user) { 264fde411d5SJuli Mallett /* 265fde411d5SJuli Mallett * If the user is NOT adding this field manually, 266fde411d5SJuli Mallett * get on with our lives if this VAR is already 267fde411d5SJuli Mallett * represented in the list. 268fde411d5SJuli Mallett */ 269fde411d5SJuli Mallett vent = find_varentry(v); 270fde411d5SJuli Mallett if (vent != NULL) 271fde411d5SJuli Mallett continue; 272fde411d5SJuli Mallett } 2734b88c807SRodney W. Grimes if ((vent = malloc(sizeof(struct varent))) == NULL) 2744fa7d788SJuli Mallett errx(1, "malloc failed"); 27578b1878aSJuli Mallett vent->header = v->header; 27678b1878aSJuli Mallett if (hp) { 27778b1878aSJuli Mallett hp = strdup(hp); 27878b1878aSJuli Mallett if (hp) 27978b1878aSJuli Mallett vent->header = hp; 28078b1878aSJuli Mallett } 2816d041cc8SJuli Mallett vent->var = malloc(sizeof(*vent->var)); 2826d041cc8SJuli Mallett if (vent->var == NULL) 2834fa7d788SJuli Mallett errx(1, "malloc failed"); 2846d041cc8SJuli Mallett memcpy(vent->var, v, sizeof(*vent->var)); 285bdf8ab46SGarance A Drosehn STAILQ_INSERT_TAIL(&varlist, vent, next_ve); 2864b88c807SRodney W. Grimes } 287871e8d8cSMark Murray free(tempstr1); 288bdf8ab46SGarance A Drosehn if (STAILQ_EMPTY(&varlist)) { 2898dd2eb05SJuli Mallett warnx("no valid keywords; valid keywords:"); 2908dd2eb05SJuli Mallett showkey(); 2918dd2eb05SJuli Mallett exit(1); 2928dd2eb05SJuli Mallett } 2934b88c807SRodney W. Grimes } 2944b88c807SRodney W. Grimes 2954b88c807SRodney W. Grimes static VAR * 29678b1878aSJuli Mallett findvar(char *p, int user, char **header) 2974b88c807SRodney W. Grimes { 298ec716487SGarance A Drosehn size_t rflen; 2994b88c807SRodney W. Grimes VAR *v, key; 300ec716487SGarance A Drosehn char *hp, *realfmt; 3014b88c807SRodney W. Grimes 3024b88c807SRodney W. Grimes hp = strchr(p, '='); 3034b88c807SRodney W. Grimes if (hp) 3044b88c807SRodney W. Grimes *hp++ = '\0'; 3054b88c807SRodney W. Grimes 3064b88c807SRodney W. Grimes key.name = p; 3074b88c807SRodney W. Grimes v = bsearch(&key, var, sizeof(var)/sizeof(VAR) - 1, sizeof(VAR), vcmp); 3084b88c807SRodney W. Grimes 3094b88c807SRodney W. Grimes if (v && v->alias) { 310ec716487SGarance A Drosehn /* 3116331f11eSGarance A Drosehn * If the user specified an alternate-header for this 3126331f11eSGarance A Drosehn * (aliased) format-name, then we need to copy that 3136331f11eSGarance A Drosehn * alternate-header when making the recursive call to 3146331f11eSGarance A Drosehn * process the alias. 3156331f11eSGarance A Drosehn */ 3166331f11eSGarance A Drosehn if (hp == NULL) 3176331f11eSGarance A Drosehn parsefmt(v->alias, user); 3186331f11eSGarance A Drosehn else { 3196331f11eSGarance A Drosehn /* 3206331f11eSGarance A Drosehn * XXX - This processing will not be correct for 3216331f11eSGarance A Drosehn * any alias which expands into a list of format 3226331f11eSGarance A Drosehn * keywords. Presently there are no aliases 3236331f11eSGarance A Drosehn * which do that. 324ec716487SGarance A Drosehn */ 325ec716487SGarance A Drosehn rflen = strlen(v->alias) + strlen(hp) + 2; 326ec716487SGarance A Drosehn realfmt = malloc(rflen); 32788985aedSGarance A Drosehn snprintf(realfmt, rflen, "%s=%s", v->alias, hp); 328ec716487SGarance A Drosehn parsefmt(realfmt, user); 3296331f11eSGarance A Drosehn } 3304b88c807SRodney W. Grimes return ((VAR *)NULL); 3314b88c807SRodney W. Grimes } 3324b88c807SRodney W. Grimes if (!v) { 3334b88c807SRodney W. Grimes warnx("%s: keyword not found", p); 3344b88c807SRodney W. Grimes eval = 1; 33578b1878aSJuli Mallett } 33678b1878aSJuli Mallett if (header) 33778b1878aSJuli Mallett *header = hp; 3384b88c807SRodney W. Grimes return (v); 3394b88c807SRodney W. Grimes } 3404b88c807SRodney W. Grimes 3414b88c807SRodney W. Grimes static int 34246251ddeSWarner Losh vcmp(const void *a, const void *b) 3434b88c807SRodney W. Grimes { 344871e8d8cSMark Murray return (strcmp(((const VAR *)a)->name, ((const VAR *)b)->name)); 3454b88c807SRodney W. Grimes } 346