xref: /illumos-gate/usr/src/cmd/svc/svccfg/svccfg.l (revision 7b209c2c)
1 %{
2 /*
3  * CDDL HEADER START
4  *
5  * The contents of this file are subject to the terms of the
6  * Common Development and Distribution License (the "License").
7  * You may not use this file except in compliance with the License.
8  *
9  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
10  * or http://www.opensolaris.org/os/licensing.
11  * See the License for the specific language governing permissions
12  * and limitations under the License.
13  *
14  * When distributing Covered Code, include this CDDL HEADER in each
15  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
16  * If applicable, add the following below this CDDL HEADER, with the
17  * fields enclosed by brackets "[]" replaced with your own identifying
18  * information: Portions Copyright [yyyy] [name of copyright owner]
19  *
20  * CDDL HEADER END
21  */
22 
23 /*
24  * Copyright 2008 Sun Microsystems, Inc.  All rights reserved.
25  * Use is subject to license terms.
26  */
27 
28 #pragma ident	"%Z%%M%	%I%	%E% SMI"
29 
30 #pragma error_messages(off, E_BLOCK_DECL_UNUSED)
31 #pragma error_messages(off, E_EQUALITY_NOT_ASSIGNMENT)
32 #pragma error_messages(off, E_FUNC_RET_MAYBE_IGNORED2)
33 #pragma error_messages(off, E_STMT_NOT_REACHED)
34 
35 #include <libintl.h>
36 #include <string.h>
37 
38 #include "svccfg.h"
39 #include "svccfg_grammar.h"
40 
41 /*
42  * We need to undefine lex's input, unput, and output macros so that references
43  * to these call the functions we provide at the end of this source file,
44  * instead of the default versions based on libc's stdio.
45  */
46 #ifdef input
47 #undef input
48 #endif
49 
50 #ifdef unput
51 #undef unput
52 #endif
53 
54 #ifdef output
55 #undef output
56 #endif
57 
58 static int input(void);
59 static void unput(int);
60 static void output(int);
61 
62 int parens = 0;
63 
64 extern int yyerror(const char *);
65 
66 %}
67 
68 /*
69  * Since command tokens are only valid at the beginning of the command (or
70  * after help), we'll only return them in the INITIAL state, and report them
71  * as SCV_WORDs afterwards.
72  */
73 %Start	WORD
74 
75 %%
76 
77 #.*$			;	/* comments */
78 
79 <INITIAL>validate	{ BEGIN WORD; return (SCC_VALIDATE); }
80 <INITIAL>import		{ BEGIN WORD; return (SCC_IMPORT); }
81 <INITIAL>export		{ BEGIN WORD; return (SCC_EXPORT); }
82 <INITIAL>archive	{ BEGIN WORD; return (SCC_ARCHIVE); }
83 <INITIAL>restore	{ BEGIN WORD; return (SCC_RESTORE); }
84 <INITIAL>apply		{ BEGIN WORD; return (SCC_APPLY); }
85 <INITIAL>extract	{ BEGIN WORD; return (SCC_EXTRACT); }
86 <INITIAL>repository	{ BEGIN WORD; return (SCC_REPOSITORY); }
87 <INITIAL>inventory	{ BEGIN WORD; return (SCC_INVENTORY); }
88 <INITIAL>set		{ BEGIN WORD; return (SCC_SET); }
89 <INITIAL>end		{ BEGIN WORD; return (SCC_END); }
90 <INITIAL>exit		{ BEGIN WORD; return (SCC_END); }
91 <INITIAL>quit		{ BEGIN WORD; return (SCC_END); }
92 <INITIAL>help		{ return (SCC_HELP); }
93 
94 <INITIAL>list		{ BEGIN WORD; return (SCC_LIST); }
95 <INITIAL>add		{ BEGIN WORD; return (SCC_ADD); }
96 <INITIAL>delete		{ BEGIN WORD; return (SCC_DELETE); }
97 <INITIAL>select		{ BEGIN WORD; return (SCC_SELECT); }
98 <INITIAL>unselect	{ BEGIN WORD; return (SCC_UNSELECT); }
99 
100 <INITIAL>listpg		{ BEGIN WORD; return (SCC_LISTPG); }
101 <INITIAL>addpg		{ BEGIN WORD; return (SCC_ADDPG); }
102 <INITIAL>delpg		{ BEGIN WORD; return (SCC_DELPG); }
103 <INITIAL>listprop	{ BEGIN WORD; return (SCC_LISTPROP); }
104 <INITIAL>setprop	{ BEGIN WORD; return (SCC_SETPROP); }
105 <INITIAL>delprop	{ BEGIN WORD; return (SCC_DELPROP); }
106 <INITIAL>editprop	{ BEGIN WORD; return (SCC_EDITPROP); }
107 <INITIAL>addpropvalue	{ BEGIN WORD; return (SCC_ADDPROPVALUE); }
108 <INITIAL>delpropvalue	{ BEGIN WORD; return (SCC_DELPROPVALUE); }
109 <INITIAL>setenv		{ BEGIN WORD; return (SCC_SETENV); }
110 <INITIAL>unsetenv	{ BEGIN WORD; return (SCC_UNSETENV); }
111 
112 <INITIAL>listsnap	{ BEGIN WORD; return (SCC_LISTSNAP); }
113 <INITIAL>selectsnap	{ BEGIN WORD; return (SCC_SELECTSNAP); }
114 <INITIAL>revert		{ BEGIN WORD; return (SCC_REVERT); }
115 <INITIAL>refresh	{ BEGIN WORD; return (SCC_REFRESH); }
116 
117 [^ \t\n">=()]+		{
118 				if ((yylval.str = strdup(yytext)) == NULL) {
119 					yyerror(gettext("Out of memory"));
120 					exit(UU_EXIT_FATAL);
121 				}
122 
123 				return SCV_WORD;
124 			}
125 
126 \"([^"\\]|\\.)*\"	{
127 				/*
128 				 * double-quoted strings start at a
129 				 * double-quote, include characters other than
130 				 * double-quote and backslash, and
131 				 * backslashed-characters, and end with a
132 				 * double-quote.
133 				 */
134 
135 				char *str, *cp;
136 				int shift;
137 
138 				if ((str = strdup(yytext)) == NULL) {
139 					yyerror(gettext("Out of memory"));
140 					exit(UU_EXIT_FATAL);
141 				}
142 
143 				/* Strip out the backslashes. */
144 				for (cp = str, shift = 0; *cp != '\0'; ++cp) {
145 					if (*cp == '\\') {
146 						++cp;
147 
148 						/*
149 						 * This can't be null because
150 						 * the string always ends with
151 						 * a double-quote.
152 						 */
153 
154 						++shift;
155 						*(cp - shift) = *cp;
156 					} else if (shift != 0)
157 						*(cp - shift) = *cp;
158 				}
159 
160 				/* Nullify everything after trailing quote */
161 				*(cp - shift) = '\0';
162 
163 				yylval.str = str;
164 				return SCV_STRING;
165 			}
166 
167 \n			{
168 				est->sc_cmd_lineno++;
169 				BEGIN INITIAL;
170 				return (SCS_NEWLINE);
171 			}
172 
173 [ \t]+			;
174 
175 ">"			{ return SCS_REDIRECT; }
176 "="			{ return SCS_EQUALS; }
177 "("			{ ++parens; return SCS_LPAREN; }
178 ")"			{ --parens; return SCS_RPAREN; }
179 
180 .			{
181 				uu_die(gettext("unrecognized character %s\n"),
182 				    yytext);
183 			}
184 
185 %%
186 
187 int
188 yyerror(const char *s)
189 {
190 	return (0);
191 }
192 
193 static int
194 input(void)
195 {
196 	static int saw_eof = 0;
197 
198 	int c = engine_cmd_getc(est);
199 
200 	/*
201 	 * To ensure input is terminated, slip in a newline on EOF.
202 	 */
203 	if (c == EOF) {
204 		if (saw_eof)
205 			return (0);
206 
207 		saw_eof = 1;
208 		return ('\n');
209 	} else
210 		saw_eof = 0;
211 
212 	if (c == '\n')
213 		yylineno++;
214 
215 	return (c);
216 }
217 
218 static void
219 unput(int c)
220 {
221 	if (c == '\n')
222 		yylineno--;
223 
224 	(void) engine_cmd_ungetc(est, c == 0 ? EOF : c);
225 }
226 
227 static void
228 output(int c)
229 {
230 	char ch = c;
231 	engine_cmd_nputs(est, &ch, sizeof (ch));
232 }
233