14b88c807SRodney W. Grimes /*-
24b88c807SRodney W. Grimes * Copyright (c) 1991, 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.
13*fbbd9655SWarner Losh * 3. 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
304b88c807SRodney W. Grimes #include <sys/types.h>
314b88c807SRodney W. Grimes
324b88c807SRodney W. Grimes #include <err.h>
334b88c807SRodney W. Grimes #include <limits.h>
344b88c807SRodney W. Grimes #include <stdlib.h>
354b88c807SRodney W. Grimes #include <string.h>
364b88c807SRodney W. Grimes
374b88c807SRodney W. Grimes #include "stty.h"
384b88c807SRodney W. Grimes #include "extern.h"
394b88c807SRodney W. Grimes
405134c3f7SWarner Losh static int c_cchar(const void *, const void *);
415e5a5667SKris Kennaway
424b88c807SRodney W. Grimes /*
434b88c807SRodney W. Grimes * Special control characters.
444b88c807SRodney W. Grimes *
454b88c807SRodney W. Grimes * Cchars1 are the standard names, cchars2 are the old aliases.
464b88c807SRodney W. Grimes * The first are displayed, but both are recognized on the
474b88c807SRodney W. Grimes * command line.
484b88c807SRodney W. Grimes */
494b88c807SRodney W. Grimes struct cchar cchars1[] = {
504b88c807SRodney W. Grimes { "discard", VDISCARD, CDISCARD },
514b88c807SRodney W. Grimes { "dsusp", VDSUSP, CDSUSP },
524b88c807SRodney W. Grimes { "eof", VEOF, CEOF },
534b88c807SRodney W. Grimes { "eol", VEOL, CEOL },
544b88c807SRodney W. Grimes { "eol2", VEOL2, CEOL },
554b88c807SRodney W. Grimes { "erase", VERASE, CERASE },
564660b141SJordan K. Hubbard { "erase2", VERASE2, CERASE2 },
574b88c807SRodney W. Grimes { "intr", VINTR, CINTR },
584b88c807SRodney W. Grimes { "kill", VKILL, CKILL },
594b88c807SRodney W. Grimes { "lnext", VLNEXT, CLNEXT },
604b88c807SRodney W. Grimes { "min", VMIN, CMIN },
614b88c807SRodney W. Grimes { "quit", VQUIT, CQUIT },
624b88c807SRodney W. Grimes { "reprint", VREPRINT, CREPRINT },
634b88c807SRodney W. Grimes { "start", VSTART, CSTART },
644b88c807SRodney W. Grimes { "status", VSTATUS, CSTATUS },
654b88c807SRodney W. Grimes { "stop", VSTOP, CSTOP },
664b88c807SRodney W. Grimes { "susp", VSUSP, CSUSP },
674b88c807SRodney W. Grimes { "time", VTIME, CTIME },
684b88c807SRodney W. Grimes { "werase", VWERASE, CWERASE },
695e5a5667SKris Kennaway { NULL, 0, 0},
704b88c807SRodney W. Grimes };
714b88c807SRodney W. Grimes
724b88c807SRodney W. Grimes struct cchar cchars2[] = {
734b88c807SRodney W. Grimes { "brk", VEOL, CEOL },
744b88c807SRodney W. Grimes { "flush", VDISCARD, CDISCARD },
754b88c807SRodney W. Grimes { "rprnt", VREPRINT, CREPRINT },
765e5a5667SKris Kennaway { NULL, 0, 0 },
774b88c807SRodney W. Grimes };
784b88c807SRodney W. Grimes
794b88c807SRodney W. Grimes static int
c_cchar(const void * a,const void * b)805134c3f7SWarner Losh c_cchar(const void *a, const void *b)
814b88c807SRodney W. Grimes {
824b88c807SRodney W. Grimes
835e5a5667SKris Kennaway return (strcmp(((const struct cchar *)a)->name, ((const struct cchar *)b)->name));
844b88c807SRodney W. Grimes }
854b88c807SRodney W. Grimes
864b88c807SRodney W. Grimes int
csearch(char *** argvp,struct info * ip)875134c3f7SWarner Losh csearch(char ***argvp, struct info *ip)
884b88c807SRodney W. Grimes {
894b88c807SRodney W. Grimes struct cchar *cp, tmp;
904b88c807SRodney W. Grimes long val;
914b88c807SRodney W. Grimes char *arg, *ep, *name;
924b88c807SRodney W. Grimes
934b88c807SRodney W. Grimes name = **argvp;
944b88c807SRodney W. Grimes
954b88c807SRodney W. Grimes tmp.name = name;
964b88c807SRodney W. Grimes if (!(cp = (struct cchar *)bsearch(&tmp, cchars1,
974b88c807SRodney W. Grimes sizeof(cchars1)/sizeof(struct cchar) - 1, sizeof(struct cchar),
980b51c95dSAndrey A. Chernov c_cchar)) && !(cp = (struct cchar *)bsearch(&tmp, cchars2,
990b51c95dSAndrey A. Chernov sizeof(cchars2)/sizeof(struct cchar) - 1, sizeof(struct cchar),
1004b88c807SRodney W. Grimes c_cchar)))
1014b88c807SRodney W. Grimes return (0);
1024b88c807SRodney W. Grimes
1034b88c807SRodney W. Grimes arg = *++*argvp;
1044b88c807SRodney W. Grimes if (!arg) {
1054b88c807SRodney W. Grimes warnx("option requires an argument -- %s", name);
1064b88c807SRodney W. Grimes usage();
1074b88c807SRodney W. Grimes }
1084b88c807SRodney W. Grimes
1094b88c807SRodney W. Grimes #define CHK(s) (*arg == s[0] && !strcmp(arg, s))
1104b88c807SRodney W. Grimes if (CHK("undef") || CHK("<undef>"))
1114b88c807SRodney W. Grimes ip->t.c_cc[cp->sub] = _POSIX_VDISABLE;
1124b88c807SRodney W. Grimes else if (cp->sub == VMIN || cp->sub == VTIME) {
1134b88c807SRodney W. Grimes val = strtol(arg, &ep, 10);
1144b88c807SRodney W. Grimes if (val > UCHAR_MAX) {
1154b88c807SRodney W. Grimes warnx("maximum option value is %d -- %s",
1164b88c807SRodney W. Grimes UCHAR_MAX, name);
1174b88c807SRodney W. Grimes usage();
1184b88c807SRodney W. Grimes }
1194b88c807SRodney W. Grimes if (*ep != '\0') {
1204b88c807SRodney W. Grimes warnx("option requires a numeric argument -- %s", name);
1214b88c807SRodney W. Grimes usage();
1224b88c807SRodney W. Grimes }
1234b88c807SRodney W. Grimes ip->t.c_cc[cp->sub] = val;
1244b88c807SRodney W. Grimes } else if (arg[0] == '^')
1254b88c807SRodney W. Grimes ip->t.c_cc[cp->sub] = (arg[1] == '?') ? 0177 :
1264b88c807SRodney W. Grimes (arg[1] == '-') ? _POSIX_VDISABLE : arg[1] & 037;
1274b88c807SRodney W. Grimes else
1284b88c807SRodney W. Grimes ip->t.c_cc[cp->sub] = arg[0];
1294b88c807SRodney W. Grimes ip->set = 1;
1304b88c807SRodney W. Grimes return (1);
1314b88c807SRodney W. Grimes }
132