xref: /freebsd/bin/stty/modes.c (revision eda14cbc264d6969b02f2b1994cef11148e914f1)
1 /*-
2  * Copyright (c) 1991, 1993, 1994
3  *	The Regents of the University of California.  All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  * 3. Neither the name of the University nor the names of its contributors
14  *    may be used to endorse or promote products derived from this software
15  *    without specific prior written permission.
16  *
17  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
18  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
21  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27  * SUCH DAMAGE.
28  */
29 
30 #ifndef lint
31 #if 0
32 static char sccsid[] = "@(#)modes.c	8.3 (Berkeley) 4/2/94";
33 #endif
34 #endif /* not lint */
35 #include <sys/cdefs.h>
36 __FBSDID("$FreeBSD$");
37 
38 #include <sys/types.h>
39 #include <stddef.h>
40 #include <string.h>
41 #include "stty.h"
42 
43 int msearch(char ***, struct info *);
44 
45 struct modes {
46 	const char *name;
47 	long set;
48 	long unset;
49 };
50 
51 /*
52  * The code in optlist() depends on minus options following regular
53  * options, i.e. "foo" must immediately precede "-foo".
54  */
55 static const struct modes cmodes[] = {
56 	{ "cs5",	CS5, CSIZE },
57 	{ "cs6",	CS6, CSIZE },
58 	{ "cs7",	CS7, CSIZE },
59 	{ "cs8",	CS8, CSIZE },
60 	{ "cstopb",	CSTOPB, 0 },
61 	{ "-cstopb",	0, CSTOPB },
62 	{ "cread",	CREAD, 0 },
63 	{ "-cread",	0, CREAD },
64 	{ "parenb",	PARENB, 0 },
65 	{ "-parenb",	0, PARENB },
66 	{ "parodd",	PARODD, 0 },
67 	{ "-parodd",	0, PARODD },
68 	{ "parity",	PARENB | CS7, PARODD | CSIZE },
69 	{ "-parity",	CS8, PARODD | PARENB | CSIZE },
70 	{ "evenp",	PARENB | CS7, PARODD | CSIZE },
71 	{ "-evenp",	CS8, PARODD | PARENB | CSIZE },
72 	{ "oddp",	PARENB | CS7 | PARODD, CSIZE },
73 	{ "-oddp",	CS8, PARODD | PARENB | CSIZE },
74 	{ "pass8",	CS8, PARODD | PARENB | CSIZE },
75 	{ "-pass8",	PARENB | CS7, PARODD | CSIZE },
76 	{ "hupcl",	HUPCL, 0 },
77 	{ "-hupcl",	0, HUPCL },
78 	{ "hup",	HUPCL, 0 },
79 	{ "-hup",	0, HUPCL },
80 	{ "clocal",	CLOCAL, 0 },
81 	{ "-clocal",	0, CLOCAL },
82 	{ "crtscts",	CRTSCTS, 0 },
83 	{ "-crtscts",	0, CRTSCTS },
84 	{ "ctsflow",	CCTS_OFLOW, 0 },
85 	{ "-ctsflow",	0, CCTS_OFLOW },
86 	{ "dsrflow",	CDSR_OFLOW, 0 },
87 	{ "-dsrflow",	0, CDSR_OFLOW },
88 	{ "dtrflow",	CDTR_IFLOW, 0 },
89 	{ "-dtrflow",	0, CDTR_IFLOW },
90 	{ "rtsflow",	CRTS_IFLOW, 0 },
91 	{ "-rtsflow",	0, CRTS_IFLOW },
92 	{ "mdmbuf",	MDMBUF, 0 },
93 	{ "-mdmbuf",	0, MDMBUF },
94 	{ "rtsdtr",	0, CNO_RTSDTR },
95 	{ "-rtsdtr",	CNO_RTSDTR, 0 },
96 	{ NULL,		0, 0 },
97 };
98 
99 static const struct modes imodes[] = {
100 	{ "ignbrk",	IGNBRK, 0 },
101 	{ "-ignbrk",	0, IGNBRK },
102 	{ "brkint",	BRKINT, 0 },
103 	{ "-brkint",	0, BRKINT },
104 	{ "ignpar",	IGNPAR, 0 },
105 	{ "-ignpar",	0, IGNPAR },
106 	{ "parmrk",	PARMRK, 0 },
107 	{ "-parmrk",	0, PARMRK },
108 	{ "inpck",	INPCK, 0 },
109 	{ "-inpck",	0, INPCK },
110 	{ "istrip",	ISTRIP, 0 },
111 	{ "-istrip",	0, ISTRIP },
112 	{ "inlcr",	INLCR, 0 },
113 	{ "-inlcr",	0, INLCR },
114 	{ "igncr",	IGNCR, 0 },
115 	{ "-igncr",	0, IGNCR },
116 	{ "icrnl",	ICRNL, 0 },
117 	{ "-icrnl",	0, ICRNL },
118 	{ "ixon",	IXON, 0 },
119 	{ "-ixon",	0, IXON },
120 	{ "flow",	IXON, 0 },
121 	{ "-flow",	0, IXON },
122 	{ "ixoff",	IXOFF, 0 },
123 	{ "-ixoff",	0, IXOFF },
124 	{ "tandem",	IXOFF, 0 },
125 	{ "-tandem",	0, IXOFF },
126 	{ "ixany",	IXANY, 0 },
127 	{ "-ixany",	0, IXANY },
128 	{ "decctlq",	0, IXANY },
129 	{ "-decctlq",	IXANY, 0 },
130 	{ "imaxbel",	IMAXBEL, 0 },
131 	{ "-imaxbel",	0, IMAXBEL },
132 	{ NULL,		0, 0 },
133 };
134 
135 static const struct modes lmodes[] = {
136 	{ "echo",	ECHO, 0 },
137 	{ "-echo",	0, ECHO },
138 	{ "echoe",	ECHOE, 0 },
139 	{ "-echoe",	0, ECHOE },
140 	{ "crterase",	ECHOE, 0 },
141 	{ "-crterase",	0, ECHOE },
142 	{ "crtbs",	ECHOE, 0 },	/* crtbs not supported, close enough */
143 	{ "-crtbs",	0, ECHOE },
144 	{ "echok",	ECHOK, 0 },
145 	{ "-echok",	0, ECHOK },
146 	{ "echoke",	ECHOKE, 0 },
147 	{ "-echoke",	0, ECHOKE },
148 	{ "crtkill",	ECHOKE, 0 },
149 	{ "-crtkill",	0, ECHOKE },
150 	{ "altwerase",	ALTWERASE, 0 },
151 	{ "-altwerase",	0, ALTWERASE },
152 	{ "iexten",	IEXTEN, 0 },
153 	{ "-iexten",	0, IEXTEN },
154 	{ "echonl",	ECHONL, 0 },
155 	{ "-echonl",	0, ECHONL },
156 	{ "echoctl",	ECHOCTL, 0 },
157 	{ "-echoctl",	0, ECHOCTL },
158 	{ "ctlecho",	ECHOCTL, 0 },
159 	{ "-ctlecho",	0, ECHOCTL },
160 	{ "echoprt",	ECHOPRT, 0 },
161 	{ "-echoprt",	0, ECHOPRT },
162 	{ "prterase",	ECHOPRT, 0 },
163 	{ "-prterase",	0, ECHOPRT },
164 	{ "isig",	ISIG, 0 },
165 	{ "-isig",	0, ISIG },
166 	{ "icanon",	ICANON, 0 },
167 	{ "-icanon",	0, ICANON },
168 	{ "noflsh",	NOFLSH, 0 },
169 	{ "-noflsh",	0, NOFLSH },
170 	{ "tostop",	TOSTOP, 0 },
171 	{ "-tostop",	0, TOSTOP },
172 	{ "flusho",	FLUSHO, 0 },
173 	{ "-flusho",	0, FLUSHO },
174 	{ "pendin",	PENDIN, 0 },
175 	{ "-pendin",	0, PENDIN },
176 	{ "crt",	ECHOE|ECHOKE|ECHOCTL, ECHOK|ECHOPRT },
177 	{ "-crt",	ECHOK, ECHOE|ECHOKE|ECHOCTL },
178 	{ "newcrt",	ECHOE|ECHOKE|ECHOCTL, ECHOK|ECHOPRT },
179 	{ "-newcrt",	ECHOK, ECHOE|ECHOKE|ECHOCTL },
180 	{ "nokerninfo",	NOKERNINFO, 0 },
181 	{ "-nokerninfo",0, NOKERNINFO },
182 	{ "kerninfo",	0, NOKERNINFO },
183 	{ "-kerninfo",	NOKERNINFO, 0 },
184 	{ NULL,		0, 0 },
185 };
186 
187 static const struct modes omodes[] = {
188 	{ "opost",	OPOST, 0 },
189 	{ "-opost",	0, OPOST },
190 	{ "litout",	0, OPOST },
191 	{ "-litout",	OPOST, 0 },
192 	{ "onlcr",	ONLCR, 0 },
193 	{ "-onlcr",	0, ONLCR },
194 	{ "ocrnl",	OCRNL, 0 },
195 	{ "-ocrnl",	0, OCRNL },
196 	{ "tabs",	TAB0, TABDLY },		/* "preserve" tabs */
197 	{ "-tabs",	TAB3, TABDLY },
198 	{ "oxtabs",	TAB3, TABDLY },
199 	{ "-oxtabs",	TAB0, TABDLY },
200 	{ "tab0",	TAB0, TABDLY },
201 	{ "tab3",	TAB3, TABDLY },
202 	{ "onocr",	ONOCR, 0 },
203 	{ "-onocr",	0, ONOCR },
204 	{ "onlret",	ONLRET, 0 },
205 	{ "-onlret",	0, ONLRET },
206 	{ NULL,		0, 0 },
207 };
208 
209 #define	CHK(s)	(*name == s[0] && !strcmp(name, s))
210 
211 int
212 msearch(char ***argvp, struct info *ip)
213 {
214 	const struct modes *mp;
215 	char *name;
216 
217 	name = **argvp;
218 
219 	for (mp = cmodes; mp->name; ++mp)
220 		if (CHK(mp->name)) {
221 			ip->t.c_cflag &= ~mp->unset;
222 			ip->t.c_cflag |= mp->set;
223 			ip->set = 1;
224 			return (1);
225 		}
226 	for (mp = imodes; mp->name; ++mp)
227 		if (CHK(mp->name)) {
228 			ip->t.c_iflag &= ~mp->unset;
229 			ip->t.c_iflag |= mp->set;
230 			ip->set = 1;
231 			return (1);
232 		}
233 	for (mp = lmodes; mp->name; ++mp)
234 		if (CHK(mp->name)) {
235 			ip->t.c_lflag &= ~mp->unset;
236 			ip->t.c_lflag |= mp->set;
237 			ip->set = 1;
238 			return (1);
239 		}
240 	for (mp = omodes; mp->name; ++mp)
241 		if (CHK(mp->name)) {
242 			ip->t.c_oflag &= ~mp->unset;
243 			ip->t.c_oflag |= mp->set;
244 			ip->set = 1;
245 			return (1);
246 		}
247 	return (0);
248 }
249