xref: /netbsd-src/external/bsd/less/dist/lesskey_parse.c (revision 838f5788460f0f133b15d706e644d692a9d4d6ec)
1*838f5788Ssimonb /*	$NetBSD: lesskey_parse.c,v 1.2 2023/10/06 05:49:49 simonb Exp $	*/
2*838f5788Ssimonb 
3e4a6e799Ssimonb /*
4e4a6e799Ssimonb  * Copyright (C) 1984-2023  Mark Nudelman
5e4a6e799Ssimonb  *
6e4a6e799Ssimonb  * You may distribute under the terms of either the GNU General Public
7e4a6e799Ssimonb  * License or the Less License, as specified in the README file.
8e4a6e799Ssimonb  *
9e4a6e799Ssimonb  * For more information, see the README file.
10e4a6e799Ssimonb  */
11e4a6e799Ssimonb 
12e4a6e799Ssimonb #include "defines.h"
13e4a6e799Ssimonb #include <stdio.h>
14e4a6e799Ssimonb #include <string.h>
15e4a6e799Ssimonb #include <stdlib.h>
16e4a6e799Ssimonb #include "lesskey.h"
17e4a6e799Ssimonb #include "cmd.h"
18e4a6e799Ssimonb #include "xbuf.h"
19e4a6e799Ssimonb 
20e4a6e799Ssimonb #define CONTROL(c)      ((c)&037)
21e4a6e799Ssimonb #define ESC             CONTROL('[')
22e4a6e799Ssimonb 
23e4a6e799Ssimonb extern void lesskey_parse_error(char *msg);
24e4a6e799Ssimonb extern char *homefile(char *filename);
25e4a6e799Ssimonb extern void *ecalloc(int count, unsigned int size);
26e4a6e799Ssimonb extern int lstrtoi(char *str, char **end, int radix);
27e4a6e799Ssimonb extern char version[];
28e4a6e799Ssimonb 
29e4a6e799Ssimonb static int linenum;
30e4a6e799Ssimonb static int errors;
31e4a6e799Ssimonb static int less_version = 0;
32e4a6e799Ssimonb static char *lesskey_file;
33e4a6e799Ssimonb 
34e4a6e799Ssimonb static struct lesskey_cmdname cmdnames[] =
35e4a6e799Ssimonb {
36e4a6e799Ssimonb 	{ "back-bracket",         A_B_BRACKET },
37e4a6e799Ssimonb 	{ "back-line",            A_B_LINE },
38e4a6e799Ssimonb 	{ "back-line-force",      A_BF_LINE },
39e4a6e799Ssimonb 	{ "back-screen",          A_B_SCREEN },
40e4a6e799Ssimonb 	{ "back-scroll",          A_B_SCROLL },
41e4a6e799Ssimonb 	{ "back-search",          A_B_SEARCH },
42e4a6e799Ssimonb 	{ "back-window",          A_B_WINDOW },
43e4a6e799Ssimonb 	{ "clear-mark",           A_CLRMARK },
44e4a6e799Ssimonb 	{ "debug",                A_DEBUG },
45e4a6e799Ssimonb 	{ "digit",                A_DIGIT },
46e4a6e799Ssimonb 	{ "display-flag",         A_DISP_OPTION },
47e4a6e799Ssimonb 	{ "display-option",       A_DISP_OPTION },
48e4a6e799Ssimonb 	{ "end",                  A_GOEND },
49e4a6e799Ssimonb 	{ "end-scroll",           A_RRSHIFT },
50e4a6e799Ssimonb 	{ "examine",              A_EXAMINE },
51e4a6e799Ssimonb 	{ "filter",               A_FILTER },
52e4a6e799Ssimonb 	{ "first-cmd",            A_FIRSTCMD },
53e4a6e799Ssimonb 	{ "firstcmd",             A_FIRSTCMD },
54e4a6e799Ssimonb 	{ "flush-repaint",        A_FREPAINT },
55e4a6e799Ssimonb 	{ "forw-bracket",         A_F_BRACKET },
56e4a6e799Ssimonb 	{ "forw-forever",         A_F_FOREVER },
57e4a6e799Ssimonb 	{ "forw-until-hilite",    A_F_UNTIL_HILITE },
58e4a6e799Ssimonb 	{ "forw-line",            A_F_LINE },
59e4a6e799Ssimonb 	{ "forw-line-force",      A_FF_LINE },
60e4a6e799Ssimonb 	{ "forw-screen",          A_F_SCREEN },
61e4a6e799Ssimonb 	{ "forw-screen-force",    A_FF_SCREEN },
62e4a6e799Ssimonb 	{ "forw-scroll",          A_F_SCROLL },
63e4a6e799Ssimonb 	{ "forw-search",          A_F_SEARCH },
64e4a6e799Ssimonb 	{ "forw-window",          A_F_WINDOW },
65e4a6e799Ssimonb 	{ "goto-end",             A_GOEND },
66e4a6e799Ssimonb 	{ "goto-end-buffered",    A_GOEND_BUF },
67e4a6e799Ssimonb 	{ "goto-line",            A_GOLINE },
68e4a6e799Ssimonb 	{ "goto-mark",            A_GOMARK },
69e4a6e799Ssimonb 	{ "help",                 A_HELP },
70e4a6e799Ssimonb 	{ "index-file",           A_INDEX_FILE },
71e4a6e799Ssimonb 	{ "invalid",              A_UINVALID },
72e4a6e799Ssimonb 	{ "left-scroll",          A_LSHIFT },
73e4a6e799Ssimonb 	{ "next-file",            A_NEXT_FILE },
74e4a6e799Ssimonb 	{ "next-tag",             A_NEXT_TAG },
75e4a6e799Ssimonb 	{ "noaction",             A_NOACTION },
76e4a6e799Ssimonb 	{ "no-scroll",            A_LLSHIFT },
77e4a6e799Ssimonb 	{ "percent",              A_PERCENT },
78e4a6e799Ssimonb 	{ "pipe",                 A_PIPE },
79e4a6e799Ssimonb 	{ "prev-file",            A_PREV_FILE },
80e4a6e799Ssimonb 	{ "prev-tag",             A_PREV_TAG },
81e4a6e799Ssimonb 	{ "quit",                 A_QUIT },
82e4a6e799Ssimonb 	{ "remove-file",          A_REMOVE_FILE },
83e4a6e799Ssimonb 	{ "repaint",              A_REPAINT },
84e4a6e799Ssimonb 	{ "repaint-flush",        A_FREPAINT },
85e4a6e799Ssimonb 	{ "repeat-search",        A_AGAIN_SEARCH },
86e4a6e799Ssimonb 	{ "repeat-search-all",    A_T_AGAIN_SEARCH },
87e4a6e799Ssimonb 	{ "reverse-search",       A_REVERSE_SEARCH },
88e4a6e799Ssimonb 	{ "reverse-search-all",   A_T_REVERSE_SEARCH },
89e4a6e799Ssimonb 	{ "right-scroll",         A_RSHIFT },
90e4a6e799Ssimonb 	{ "set-mark",             A_SETMARK },
91e4a6e799Ssimonb 	{ "set-mark-bottom",      A_SETMARKBOT },
92e4a6e799Ssimonb 	{ "shell",                A_SHELL },
93e4a6e799Ssimonb 	{ "pshell",               A_PSHELL },
94e4a6e799Ssimonb 	{ "status",               A_STAT },
95e4a6e799Ssimonb 	{ "toggle-flag",          A_OPT_TOGGLE },
96e4a6e799Ssimonb 	{ "toggle-option",        A_OPT_TOGGLE },
97e4a6e799Ssimonb 	{ "undo-hilite",          A_UNDO_SEARCH },
98e4a6e799Ssimonb 	{ "clear-search",         A_CLR_SEARCH },
99e4a6e799Ssimonb 	{ "version",              A_VERSION },
100e4a6e799Ssimonb 	{ "visual",               A_VISUAL },
101e4a6e799Ssimonb 	{ NULL,   0 }
102e4a6e799Ssimonb };
103e4a6e799Ssimonb 
104e4a6e799Ssimonb static struct lesskey_cmdname editnames[] =
105e4a6e799Ssimonb {
106e4a6e799Ssimonb 	{ "back-complete",      EC_B_COMPLETE },
107e4a6e799Ssimonb 	{ "backspace",          EC_BACKSPACE },
108e4a6e799Ssimonb 	{ "delete",             EC_DELETE },
109e4a6e799Ssimonb 	{ "down",               EC_DOWN },
110e4a6e799Ssimonb 	{ "end",                EC_END },
111e4a6e799Ssimonb 	{ "expand",             EC_EXPAND },
112e4a6e799Ssimonb 	{ "forw-complete",      EC_F_COMPLETE },
113e4a6e799Ssimonb 	{ "home",               EC_HOME },
114e4a6e799Ssimonb 	{ "insert",             EC_INSERT },
115e4a6e799Ssimonb 	{ "invalid",            EC_UINVALID },
116e4a6e799Ssimonb 	{ "kill-line",          EC_LINEKILL },
117e4a6e799Ssimonb 	{ "abort",              EC_ABORT },
118e4a6e799Ssimonb 	{ "left",               EC_LEFT },
119e4a6e799Ssimonb 	{ "literal",            EC_LITERAL },
120e4a6e799Ssimonb 	{ "right",              EC_RIGHT },
121e4a6e799Ssimonb 	{ "up",                 EC_UP },
122e4a6e799Ssimonb 	{ "word-backspace",     EC_W_BACKSPACE },
123e4a6e799Ssimonb 	{ "word-delete",        EC_W_DELETE },
124e4a6e799Ssimonb 	{ "word-left",          EC_W_LEFT },
125e4a6e799Ssimonb 	{ "word-right",         EC_W_RIGHT },
126e4a6e799Ssimonb 	{ NULL, 0 }
127e4a6e799Ssimonb };
128e4a6e799Ssimonb 
129e4a6e799Ssimonb /*
130e4a6e799Ssimonb  * Print a parse error message.
131e4a6e799Ssimonb  */
parse_error(char * fmt,char * arg1)132e4a6e799Ssimonb static void parse_error(char *fmt, char *arg1)
133e4a6e799Ssimonb {
134e4a6e799Ssimonb 	char buf[1024];
135e4a6e799Ssimonb 	int n = snprintf(buf, sizeof(buf), "%s: line %d: ", lesskey_file, linenum);
136e4a6e799Ssimonb 	if (n >= 0 && n < sizeof(buf))
137e4a6e799Ssimonb 		snprintf(buf+n, sizeof(buf)-n, fmt, arg1);
138e4a6e799Ssimonb 	++errors;
139e4a6e799Ssimonb 	lesskey_parse_error(buf);
140e4a6e799Ssimonb }
141e4a6e799Ssimonb 
142e4a6e799Ssimonb /*
143e4a6e799Ssimonb  * Initialize lesskey_tables.
144e4a6e799Ssimonb  */
init_tables(struct lesskey_tables * tables)145e4a6e799Ssimonb static void init_tables(struct lesskey_tables *tables)
146e4a6e799Ssimonb {
147e4a6e799Ssimonb 	tables->currtable = &tables->cmdtable;
148e4a6e799Ssimonb 
149e4a6e799Ssimonb 	tables->cmdtable.names = cmdnames;
150e4a6e799Ssimonb 	tables->cmdtable.is_var = 0;
151e4a6e799Ssimonb 	xbuf_init(&tables->cmdtable.buf);
152e4a6e799Ssimonb 
153e4a6e799Ssimonb 	tables->edittable.names = editnames;
154e4a6e799Ssimonb 	tables->edittable.is_var = 0;
155e4a6e799Ssimonb 	xbuf_init(&tables->edittable.buf);
156e4a6e799Ssimonb 
157e4a6e799Ssimonb 	tables->vartable.names = NULL;
158e4a6e799Ssimonb 	tables->vartable.is_var = 1;
159e4a6e799Ssimonb 	xbuf_init(&tables->vartable.buf);
160e4a6e799Ssimonb }
161e4a6e799Ssimonb 
162e4a6e799Ssimonb #define CHAR_STRING_LEN 8
163e4a6e799Ssimonb 
char_string(char * buf,int ch,int lit)164e4a6e799Ssimonb static char * char_string(char *buf, int ch, int lit)
165e4a6e799Ssimonb {
166e4a6e799Ssimonb 	if (lit || (ch >= 0x20 && ch < 0x7f))
167e4a6e799Ssimonb 	{
168e4a6e799Ssimonb 		buf[0] = ch;
169e4a6e799Ssimonb 		buf[1] = '\0';
170e4a6e799Ssimonb 	} else
171e4a6e799Ssimonb 	{
172e4a6e799Ssimonb 		snprintf(buf, CHAR_STRING_LEN, "\\x%02x", ch);
173e4a6e799Ssimonb 	}
174e4a6e799Ssimonb 	return buf;
175e4a6e799Ssimonb }
176e4a6e799Ssimonb 
177e4a6e799Ssimonb /*
178e4a6e799Ssimonb  * Increment char pointer by one up to terminating nul byte.
179e4a6e799Ssimonb  */
increment_pointer(char * p)180e4a6e799Ssimonb static char * increment_pointer(char *p)
181e4a6e799Ssimonb {
182e4a6e799Ssimonb 	if (*p == '\0')
183e4a6e799Ssimonb 		return p;
184e4a6e799Ssimonb 	return p+1;
185e4a6e799Ssimonb }
186e4a6e799Ssimonb 
187e4a6e799Ssimonb /*
188e4a6e799Ssimonb  * Parse one character of a string.
189e4a6e799Ssimonb  */
tstr(char ** pp,int xlate)190e4a6e799Ssimonb static char * tstr(char **pp, int xlate)
191e4a6e799Ssimonb {
192e4a6e799Ssimonb 	char *p;
193e4a6e799Ssimonb 	char ch;
194e4a6e799Ssimonb 	int i;
195e4a6e799Ssimonb 	static char buf[CHAR_STRING_LEN];
196e4a6e799Ssimonb 	static char tstr_control_k[] =
197e4a6e799Ssimonb 		{ SK_SPECIAL_KEY, SK_CONTROL_K, 6, 1, 1, 1, '\0' };
198e4a6e799Ssimonb 
199e4a6e799Ssimonb 	p = *pp;
200e4a6e799Ssimonb 	switch (*p)
201e4a6e799Ssimonb 	{
202e4a6e799Ssimonb 	case '\\':
203e4a6e799Ssimonb 		++p;
204e4a6e799Ssimonb 		switch (*p)
205e4a6e799Ssimonb 		{
206e4a6e799Ssimonb 		case '0': case '1': case '2': case '3':
207e4a6e799Ssimonb 		case '4': case '5': case '6': case '7':
208e4a6e799Ssimonb 			/*
209e4a6e799Ssimonb 			 * Parse an octal number.
210e4a6e799Ssimonb 			 */
211e4a6e799Ssimonb 			ch = 0;
212e4a6e799Ssimonb 			i = 0;
213e4a6e799Ssimonb 			do
214e4a6e799Ssimonb 				ch = 8*ch + (*p - '0');
215e4a6e799Ssimonb 			while (*++p >= '0' && *p <= '7' && ++i < 3);
216e4a6e799Ssimonb 			*pp = p;
217e4a6e799Ssimonb 			if (xlate && ch == CONTROL('K'))
218e4a6e799Ssimonb 				return tstr_control_k;
219e4a6e799Ssimonb 			return char_string(buf, ch, 1);
220e4a6e799Ssimonb 		case 'b':
221e4a6e799Ssimonb 			*pp = p+1;
222e4a6e799Ssimonb 			return ("\b");
223e4a6e799Ssimonb 		case 'e':
224e4a6e799Ssimonb 			*pp = p+1;
225e4a6e799Ssimonb 			return char_string(buf, ESC, 1);
226e4a6e799Ssimonb 		case 'n':
227e4a6e799Ssimonb 			*pp = p+1;
228e4a6e799Ssimonb 			return ("\n");
229e4a6e799Ssimonb 		case 'r':
230e4a6e799Ssimonb 			*pp = p+1;
231e4a6e799Ssimonb 			return ("\r");
232e4a6e799Ssimonb 		case 't':
233e4a6e799Ssimonb 			*pp = p+1;
234e4a6e799Ssimonb 			return ("\t");
235e4a6e799Ssimonb 		case 'k':
236e4a6e799Ssimonb 			if (xlate)
237e4a6e799Ssimonb 			{
238e4a6e799Ssimonb 				switch (*++p)
239e4a6e799Ssimonb 				{
240e4a6e799Ssimonb 				case 'b': ch = SK_BACKSPACE; break;
241e4a6e799Ssimonb 				case 'B': ch = SK_CTL_BACKSPACE; break;
242e4a6e799Ssimonb 				case 'd': ch = SK_DOWN_ARROW; break;
243e4a6e799Ssimonb 				case 'D': ch = SK_PAGE_DOWN; break;
244e4a6e799Ssimonb 				case 'e': ch = SK_END; break;
245e4a6e799Ssimonb 				case 'h': ch = SK_HOME; break;
246e4a6e799Ssimonb 				case 'i': ch = SK_INSERT; break;
247e4a6e799Ssimonb 				case 'l': ch = SK_LEFT_ARROW; break;
248e4a6e799Ssimonb 				case 'L': ch = SK_CTL_LEFT_ARROW; break;
249e4a6e799Ssimonb 				case 'r': ch = SK_RIGHT_ARROW; break;
250e4a6e799Ssimonb 				case 'R': ch = SK_CTL_RIGHT_ARROW; break;
251e4a6e799Ssimonb 				case 't': ch = SK_BACKTAB; break;
252e4a6e799Ssimonb 				case 'u': ch = SK_UP_ARROW; break;
253e4a6e799Ssimonb 				case 'U': ch = SK_PAGE_UP; break;
254e4a6e799Ssimonb 				case 'x': ch = SK_DELETE; break;
255e4a6e799Ssimonb 				case 'X': ch = SK_CTL_DELETE; break;
256e4a6e799Ssimonb 				case '1': ch = SK_F1; break;
257e4a6e799Ssimonb 				default:
258e4a6e799Ssimonb 					parse_error("invalid escape sequence \"\\k%s\"", char_string(buf, *p, 0));
259e4a6e799Ssimonb 					*pp = increment_pointer(p);
260e4a6e799Ssimonb 					return ("");
261e4a6e799Ssimonb 				}
262e4a6e799Ssimonb 				*pp = p+1;
263e4a6e799Ssimonb 				buf[0] = SK_SPECIAL_KEY;
264e4a6e799Ssimonb 				buf[1] = ch;
265e4a6e799Ssimonb 				buf[2] = 6;
266e4a6e799Ssimonb 				buf[3] = 1;
267e4a6e799Ssimonb 				buf[4] = 1;
268e4a6e799Ssimonb 				buf[5] = 1;
269e4a6e799Ssimonb 				buf[6] = '\0';
270e4a6e799Ssimonb 				return (buf);
271e4a6e799Ssimonb 			}
272e4a6e799Ssimonb 			/* FALLTHRU */
273e4a6e799Ssimonb 		default:
274e4a6e799Ssimonb 			/*
275e4a6e799Ssimonb 			 * Backslash followed by any other char
276e4a6e799Ssimonb 			 * just means that char.
277e4a6e799Ssimonb 			 */
278e4a6e799Ssimonb 			*pp = increment_pointer(p);
279e4a6e799Ssimonb 			char_string(buf, *p, 1);
280e4a6e799Ssimonb 			if (xlate && buf[0] == CONTROL('K'))
281e4a6e799Ssimonb 				return tstr_control_k;
282e4a6e799Ssimonb 			return (buf);
283e4a6e799Ssimonb 		}
284e4a6e799Ssimonb 	case '^':
285e4a6e799Ssimonb 		/*
286e4a6e799Ssimonb 		 * Caret means CONTROL.
287e4a6e799Ssimonb 		 */
288e4a6e799Ssimonb 		*pp = increment_pointer(p+1);
289e4a6e799Ssimonb 		char_string(buf, CONTROL(p[1]), 1);
290e4a6e799Ssimonb 		if (xlate && buf[0] == CONTROL('K'))
291e4a6e799Ssimonb 			return tstr_control_k;
292e4a6e799Ssimonb 		return (buf);
293e4a6e799Ssimonb 	}
294e4a6e799Ssimonb 	*pp = increment_pointer(p);
295e4a6e799Ssimonb 	char_string(buf, *p, 1);
296e4a6e799Ssimonb 	if (xlate && buf[0] == CONTROL('K'))
297e4a6e799Ssimonb 		return tstr_control_k;
298e4a6e799Ssimonb 	return (buf);
299e4a6e799Ssimonb }
300e4a6e799Ssimonb 
issp(char ch)301e4a6e799Ssimonb static int issp(char ch)
302e4a6e799Ssimonb {
303e4a6e799Ssimonb 	return (ch == ' ' || ch == '\t');
304e4a6e799Ssimonb }
305e4a6e799Ssimonb 
306e4a6e799Ssimonb /*
307e4a6e799Ssimonb  * Skip leading spaces in a string.
308e4a6e799Ssimonb  */
skipsp(char * s)309e4a6e799Ssimonb static char * skipsp(char *s)
310e4a6e799Ssimonb {
311e4a6e799Ssimonb 	while (issp(*s))
312e4a6e799Ssimonb 		s++;
313e4a6e799Ssimonb 	return (s);
314e4a6e799Ssimonb }
315e4a6e799Ssimonb 
316e4a6e799Ssimonb /*
317e4a6e799Ssimonb  * Skip non-space characters in a string.
318e4a6e799Ssimonb  */
skipnsp(char * s)319e4a6e799Ssimonb static char * skipnsp(char *s)
320e4a6e799Ssimonb {
321e4a6e799Ssimonb 	while (*s != '\0' && !issp(*s))
322e4a6e799Ssimonb 		s++;
323e4a6e799Ssimonb 	return (s);
324e4a6e799Ssimonb }
325e4a6e799Ssimonb 
326e4a6e799Ssimonb /*
327e4a6e799Ssimonb  * Clean up an input line:
328e4a6e799Ssimonb  * strip off the trailing newline & any trailing # comment.
329e4a6e799Ssimonb  */
clean_line(char * s)330e4a6e799Ssimonb static char * clean_line(char *s)
331e4a6e799Ssimonb {
332e4a6e799Ssimonb 	int i;
333e4a6e799Ssimonb 
334e4a6e799Ssimonb 	s = skipsp(s);
335e4a6e799Ssimonb 	for (i = 0;  s[i] != '\0' && s[i] != '\n' && s[i] != '\r';  i++)
336e4a6e799Ssimonb 		if (s[i] == '#' && (i == 0 || s[i-1] != '\\'))
337e4a6e799Ssimonb 			break;
338e4a6e799Ssimonb 	s[i] = '\0';
339e4a6e799Ssimonb 	return (s);
340e4a6e799Ssimonb }
341e4a6e799Ssimonb 
342e4a6e799Ssimonb /*
343e4a6e799Ssimonb  * Add a byte to the output command table.
344e4a6e799Ssimonb  */
add_cmd_char(unsigned char c,struct lesskey_tables * tables)345e4a6e799Ssimonb static void add_cmd_char(unsigned char c, struct lesskey_tables *tables)
346e4a6e799Ssimonb {
347e4a6e799Ssimonb 	xbuf_add_byte(&tables->currtable->buf, c);
348e4a6e799Ssimonb }
349e4a6e799Ssimonb 
erase_cmd_char(struct lesskey_tables * tables)350e4a6e799Ssimonb static void erase_cmd_char(struct lesskey_tables *tables)
351e4a6e799Ssimonb {
352e4a6e799Ssimonb 	xbuf_pop(&tables->currtable->buf);
353e4a6e799Ssimonb }
354e4a6e799Ssimonb 
355e4a6e799Ssimonb /*
356e4a6e799Ssimonb  * Add a string to the output command table.
357e4a6e799Ssimonb  */
add_cmd_str(char * s,struct lesskey_tables * tables)358e4a6e799Ssimonb static void add_cmd_str(char *s, struct lesskey_tables *tables)
359e4a6e799Ssimonb {
360e4a6e799Ssimonb 	for ( ;  *s != '\0';  s++)
361e4a6e799Ssimonb 		add_cmd_char(*s, tables);
362e4a6e799Ssimonb }
363e4a6e799Ssimonb 
364e4a6e799Ssimonb /*
365e4a6e799Ssimonb  * Does a given version number match the running version?
366e4a6e799Ssimonb  * Operator compares the running version to the given version.
367e4a6e799Ssimonb  */
match_version(char op,int ver)368e4a6e799Ssimonb static int match_version(char op, int ver)
369e4a6e799Ssimonb {
370e4a6e799Ssimonb 	switch (op)
371e4a6e799Ssimonb 	{
372e4a6e799Ssimonb 	case '>': return less_version > ver;
373e4a6e799Ssimonb 	case '<': return less_version < ver;
374e4a6e799Ssimonb 	case '+': return less_version >= ver;
375e4a6e799Ssimonb 	case '-': return less_version <= ver;
376e4a6e799Ssimonb 	case '=': return less_version == ver;
377e4a6e799Ssimonb 	case '!': return less_version != ver;
378e4a6e799Ssimonb 	default: return 0; /* cannot happen */
379e4a6e799Ssimonb 	}
380e4a6e799Ssimonb }
381e4a6e799Ssimonb 
382e4a6e799Ssimonb /*
383e4a6e799Ssimonb  * Handle a #version line.
384e4a6e799Ssimonb  * If the version matches, return the part of the line that should be executed.
385e4a6e799Ssimonb  * Otherwise, return NULL.
386e4a6e799Ssimonb  */
version_line(char * s,struct lesskey_tables * tables)387e4a6e799Ssimonb static char * version_line(char *s, struct lesskey_tables *tables)
388e4a6e799Ssimonb {
389e4a6e799Ssimonb 	char op;
390e4a6e799Ssimonb 	int ver;
391e4a6e799Ssimonb 	char *e;
392e4a6e799Ssimonb 	char buf[CHAR_STRING_LEN];
393e4a6e799Ssimonb 
394e4a6e799Ssimonb 	s += strlen("#version");
395e4a6e799Ssimonb 	s = skipsp(s);
396e4a6e799Ssimonb 	op = *s++;
397e4a6e799Ssimonb 	/* Simplify 2-char op to one char. */
398e4a6e799Ssimonb 	switch (op)
399e4a6e799Ssimonb 	{
400e4a6e799Ssimonb 	case '<': if (*s == '=') { s++; op = '-'; } break;
401e4a6e799Ssimonb 	case '>': if (*s == '=') { s++; op = '+'; } break;
402e4a6e799Ssimonb 	case '=': if (*s == '=') { s++; } break;
403e4a6e799Ssimonb 	case '!': if (*s == '=') { s++; } break;
404e4a6e799Ssimonb 	default:
405e4a6e799Ssimonb 		parse_error("invalid operator '%s' in #version line", char_string(buf, op, 0));
406e4a6e799Ssimonb 		return (NULL);
407e4a6e799Ssimonb 	}
408e4a6e799Ssimonb 	s = skipsp(s);
409e4a6e799Ssimonb 	ver = lstrtoi(s, &e, 10);
410e4a6e799Ssimonb 	if (e == s)
411e4a6e799Ssimonb 	{
412e4a6e799Ssimonb 		parse_error("non-numeric version number in #version line", "");
413e4a6e799Ssimonb 		return (NULL);
414e4a6e799Ssimonb 	}
415e4a6e799Ssimonb 	if (!match_version(op, ver))
416e4a6e799Ssimonb 		return (NULL);
417e4a6e799Ssimonb 	return (e);
418e4a6e799Ssimonb }
419e4a6e799Ssimonb 
420e4a6e799Ssimonb /*
421e4a6e799Ssimonb  * See if we have a special "control" line.
422e4a6e799Ssimonb  */
control_line(char * s,struct lesskey_tables * tables)423e4a6e799Ssimonb static char * control_line(char *s, struct lesskey_tables *tables)
424e4a6e799Ssimonb {
425e4a6e799Ssimonb #define PREFIX(str,pat) (strncmp(str,pat,strlen(pat)) == 0)
426e4a6e799Ssimonb 
427e4a6e799Ssimonb 	if (PREFIX(s, "#line-edit"))
428e4a6e799Ssimonb 	{
429e4a6e799Ssimonb 		tables->currtable = &tables->edittable;
430e4a6e799Ssimonb 		return (NULL);
431e4a6e799Ssimonb 	}
432e4a6e799Ssimonb 	if (PREFIX(s, "#command"))
433e4a6e799Ssimonb 	{
434e4a6e799Ssimonb 		tables->currtable = &tables->cmdtable;
435e4a6e799Ssimonb 		return (NULL);
436e4a6e799Ssimonb 	}
437e4a6e799Ssimonb 	if (PREFIX(s, "#env"))
438e4a6e799Ssimonb 	{
439e4a6e799Ssimonb 		tables->currtable = &tables->vartable;
440e4a6e799Ssimonb 		return (NULL);
441e4a6e799Ssimonb 	}
442e4a6e799Ssimonb 	if (PREFIX(s, "#stop"))
443e4a6e799Ssimonb 	{
444e4a6e799Ssimonb 		add_cmd_char('\0', tables);
445e4a6e799Ssimonb 		add_cmd_char(A_END_LIST, tables);
446e4a6e799Ssimonb 		return (NULL);
447e4a6e799Ssimonb 	}
448e4a6e799Ssimonb 	if (PREFIX(s, "#version"))
449e4a6e799Ssimonb 	{
450e4a6e799Ssimonb 		return (version_line(s, tables));
451e4a6e799Ssimonb 	}
452e4a6e799Ssimonb 	return (s);
453e4a6e799Ssimonb }
454e4a6e799Ssimonb 
455e4a6e799Ssimonb /*
456e4a6e799Ssimonb  * Find an action, given the name of the action.
457e4a6e799Ssimonb  */
findaction(char * actname,struct lesskey_tables * tables)458e4a6e799Ssimonb static int findaction(char *actname, struct lesskey_tables *tables)
459e4a6e799Ssimonb {
460e4a6e799Ssimonb 	int i;
461e4a6e799Ssimonb 
462e4a6e799Ssimonb 	for (i = 0;  tables->currtable->names[i].cn_name != NULL;  i++)
463e4a6e799Ssimonb 		if (strcmp(tables->currtable->names[i].cn_name, actname) == 0)
464e4a6e799Ssimonb 			return (tables->currtable->names[i].cn_action);
465e4a6e799Ssimonb 	parse_error("unknown action: \"%s\"", actname);
466e4a6e799Ssimonb 	return (A_INVALID);
467e4a6e799Ssimonb }
468e4a6e799Ssimonb 
469e4a6e799Ssimonb /*
470e4a6e799Ssimonb  * Parse a line describing one key binding, of the form
471e4a6e799Ssimonb  *  KEY ACTION [EXTRA]
472e4a6e799Ssimonb  * where KEY is the user key sequence, ACTION is the
473e4a6e799Ssimonb  * resulting less action, and EXTRA is an "extra" user
474e4a6e799Ssimonb  * key sequence injected after the action.
475e4a6e799Ssimonb  */
parse_cmdline(char * p,struct lesskey_tables * tables)476e4a6e799Ssimonb static void parse_cmdline(char *p, struct lesskey_tables *tables)
477e4a6e799Ssimonb {
478e4a6e799Ssimonb 	char *actname;
479e4a6e799Ssimonb 	int action;
480e4a6e799Ssimonb 	char *s;
481e4a6e799Ssimonb 	char c;
482e4a6e799Ssimonb 
483e4a6e799Ssimonb 	/*
484e4a6e799Ssimonb 	 * Parse the command string and store it in the current table.
485e4a6e799Ssimonb 	 */
486e4a6e799Ssimonb 	do
487e4a6e799Ssimonb 	{
488e4a6e799Ssimonb 		s = tstr(&p, 1);
489e4a6e799Ssimonb 		add_cmd_str(s, tables);
490e4a6e799Ssimonb 	} while (*p != '\0' && !issp(*p));
491e4a6e799Ssimonb 	/*
492e4a6e799Ssimonb 	 * Terminate the command string with a null byte.
493e4a6e799Ssimonb 	 */
494e4a6e799Ssimonb 	add_cmd_char('\0', tables);
495e4a6e799Ssimonb 
496e4a6e799Ssimonb 	/*
497e4a6e799Ssimonb 	 * Skip white space between the command string
498e4a6e799Ssimonb 	 * and the action name.
499e4a6e799Ssimonb 	 * Terminate the action name with a null byte.
500e4a6e799Ssimonb 	 */
501e4a6e799Ssimonb 	p = skipsp(p);
502e4a6e799Ssimonb 	if (*p == '\0')
503e4a6e799Ssimonb 	{
504e4a6e799Ssimonb 		parse_error("missing action", "");
505e4a6e799Ssimonb 		return;
506e4a6e799Ssimonb 	}
507e4a6e799Ssimonb 	actname = p;
508e4a6e799Ssimonb 	p = skipnsp(p);
509e4a6e799Ssimonb 	c = *p;
510e4a6e799Ssimonb 	*p = '\0';
511e4a6e799Ssimonb 
512e4a6e799Ssimonb 	/*
513e4a6e799Ssimonb 	 * Parse the action name and store it in the current table.
514e4a6e799Ssimonb 	 */
515e4a6e799Ssimonb 	action = findaction(actname, tables);
516e4a6e799Ssimonb 
517e4a6e799Ssimonb 	/*
518e4a6e799Ssimonb 	 * See if an extra string follows the action name.
519e4a6e799Ssimonb 	 */
520e4a6e799Ssimonb 	*p = c;
521e4a6e799Ssimonb 	p = skipsp(p);
522e4a6e799Ssimonb 	if (*p == '\0')
523e4a6e799Ssimonb 	{
524e4a6e799Ssimonb 		add_cmd_char((unsigned char) action, tables);
525e4a6e799Ssimonb 	} else
526e4a6e799Ssimonb 	{
527e4a6e799Ssimonb 		/*
528e4a6e799Ssimonb 		 * OR the special value A_EXTRA into the action byte.
529e4a6e799Ssimonb 		 * Put the extra string after the action byte.
530e4a6e799Ssimonb 		 */
531e4a6e799Ssimonb 		add_cmd_char((unsigned char) (action | A_EXTRA), tables);
532e4a6e799Ssimonb 		while (*p != '\0')
533e4a6e799Ssimonb 			add_cmd_str(tstr(&p, 0), tables);
534e4a6e799Ssimonb 		add_cmd_char('\0', tables);
535e4a6e799Ssimonb 	}
536e4a6e799Ssimonb }
537e4a6e799Ssimonb 
538e4a6e799Ssimonb /*
539e4a6e799Ssimonb  * Parse a variable definition line, of the form
540e4a6e799Ssimonb  *  NAME = VALUE
541e4a6e799Ssimonb  */
parse_varline(char * line,struct lesskey_tables * tables)542e4a6e799Ssimonb static void parse_varline(char *line, struct lesskey_tables *tables)
543e4a6e799Ssimonb {
544e4a6e799Ssimonb 	char *s;
545e4a6e799Ssimonb 	char *p = line;
546e4a6e799Ssimonb 	char *eq;
547e4a6e799Ssimonb 
548e4a6e799Ssimonb 	eq = strchr(line, '=');
549e4a6e799Ssimonb 	if (eq != NULL && eq > line && eq[-1] == '+')
550e4a6e799Ssimonb 	{
551e4a6e799Ssimonb 		/*
552e4a6e799Ssimonb 		 * Rather ugly way of handling a += line.
553e4a6e799Ssimonb 		 * {{ Note that we ignore the variable name and
554e4a6e799Ssimonb 		 *    just append to the previously defined variable. }}
555e4a6e799Ssimonb 		 */
556e4a6e799Ssimonb 		erase_cmd_char(tables); /* backspace over the final null */
557e4a6e799Ssimonb 		p = eq+1;
558e4a6e799Ssimonb 	} else
559e4a6e799Ssimonb 	{
560e4a6e799Ssimonb 		do
561e4a6e799Ssimonb 		{
562e4a6e799Ssimonb 			s = tstr(&p, 0);
563e4a6e799Ssimonb 			add_cmd_str(s, tables);
564e4a6e799Ssimonb 		} while (*p != '\0' && !issp(*p) && *p != '=');
565e4a6e799Ssimonb 		/*
566e4a6e799Ssimonb 		 * Terminate the variable name with a null byte.
567e4a6e799Ssimonb 		 */
568e4a6e799Ssimonb 		add_cmd_char('\0', tables);
569e4a6e799Ssimonb 		p = skipsp(p);
570e4a6e799Ssimonb 		if (*p++ != '=')
571e4a6e799Ssimonb 		{
572e4a6e799Ssimonb 			parse_error("missing = in variable definition", "");
573e4a6e799Ssimonb 			return;
574e4a6e799Ssimonb 		}
575e4a6e799Ssimonb 		add_cmd_char(EV_OK|A_EXTRA, tables);
576e4a6e799Ssimonb 	}
577e4a6e799Ssimonb 	p = skipsp(p);
578e4a6e799Ssimonb 	while (*p != '\0')
579e4a6e799Ssimonb 	{
580e4a6e799Ssimonb 		s = tstr(&p, 0);
581e4a6e799Ssimonb 		add_cmd_str(s, tables);
582e4a6e799Ssimonb 	}
583e4a6e799Ssimonb 	add_cmd_char('\0', tables);
584e4a6e799Ssimonb }
585e4a6e799Ssimonb 
586e4a6e799Ssimonb /*
587e4a6e799Ssimonb  * Parse a line from the lesskey file.
588e4a6e799Ssimonb  */
parse_line(char * line,struct lesskey_tables * tables)589e4a6e799Ssimonb static void parse_line(char *line, struct lesskey_tables *tables)
590e4a6e799Ssimonb {
591e4a6e799Ssimonb 	char *p;
592e4a6e799Ssimonb 
593e4a6e799Ssimonb 	/*
594e4a6e799Ssimonb 	 * See if it is a control line.
595e4a6e799Ssimonb 	 */
596e4a6e799Ssimonb 	p = control_line(line, tables);
597e4a6e799Ssimonb 	if (p == NULL)
598e4a6e799Ssimonb 		return;
599e4a6e799Ssimonb 	/*
600e4a6e799Ssimonb 	 * Skip leading white space.
601e4a6e799Ssimonb 	 * Replace the final newline with a null byte.
602e4a6e799Ssimonb 	 * Ignore blank lines and comments.
603e4a6e799Ssimonb 	 */
604e4a6e799Ssimonb 	p = clean_line(p);
605e4a6e799Ssimonb 	if (*p == '\0')
606e4a6e799Ssimonb 		return;
607e4a6e799Ssimonb 
608e4a6e799Ssimonb 	if (tables->currtable->is_var)
609e4a6e799Ssimonb 		parse_varline(p, tables);
610e4a6e799Ssimonb 	else
611e4a6e799Ssimonb 		parse_cmdline(p, tables);
612e4a6e799Ssimonb }
613e4a6e799Ssimonb 
614e4a6e799Ssimonb /*
615e4a6e799Ssimonb  * Parse a lesskey source file and store result in tables.
616e4a6e799Ssimonb  */
parse_lesskey(char * infile,struct lesskey_tables * tables)617e4a6e799Ssimonb int parse_lesskey(char *infile, struct lesskey_tables *tables)
618e4a6e799Ssimonb {
619e4a6e799Ssimonb 	FILE *desc;
620e4a6e799Ssimonb 	char line[1024];
621e4a6e799Ssimonb 
622e4a6e799Ssimonb 	if (infile == NULL)
623e4a6e799Ssimonb 		infile = homefile(DEF_LESSKEYINFILE);
624e4a6e799Ssimonb 	lesskey_file = infile;
625e4a6e799Ssimonb 
626e4a6e799Ssimonb 	init_tables(tables);
627e4a6e799Ssimonb 	errors = 0;
628e4a6e799Ssimonb 	linenum = 0;
629e4a6e799Ssimonb 	if (less_version == 0)
630e4a6e799Ssimonb 		less_version = lstrtoi(version, NULL, 10);
631e4a6e799Ssimonb 
632e4a6e799Ssimonb 	/*
633e4a6e799Ssimonb 	 * Open the input file.
634e4a6e799Ssimonb 	 */
635e4a6e799Ssimonb 	if (strcmp(infile, "-") == 0)
636e4a6e799Ssimonb 		desc = stdin;
637e4a6e799Ssimonb 	else if ((desc = fopen(infile, "r")) == NULL)
638e4a6e799Ssimonb 	{
639e4a6e799Ssimonb 		/* parse_error("cannot open lesskey file %s", infile); */
640e4a6e799Ssimonb 		return (-1);
641e4a6e799Ssimonb 	}
642e4a6e799Ssimonb 
643e4a6e799Ssimonb 	/*
644e4a6e799Ssimonb 	 * Read and parse the input file, one line at a time.
645e4a6e799Ssimonb 	 */
646e4a6e799Ssimonb 	while (fgets(line, sizeof(line), desc) != NULL)
647e4a6e799Ssimonb 	{
648e4a6e799Ssimonb 		++linenum;
649e4a6e799Ssimonb 		parse_line(line, tables);
650e4a6e799Ssimonb 	}
651e4a6e799Ssimonb 	fclose(desc);
652e4a6e799Ssimonb 	return (errors);
653e4a6e799Ssimonb }
654