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