usr.sbin/makefs/hammer2: Remove redundant hammer2_inode_modify()
[dragonfly.git] / contrib / less / lesskey_parse.c
blob7042b6e12708ff177a1c56173cfd6fbd696f4f97
1 /*
2 * Copyright (C) 1984-2022 Mark Nudelman
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.
7 * For more information, see the README file.
8 */
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"
18 #define CONTROL(c) ((c)&037)
19 #define ESC CONTROL('[')
21 extern void lesskey_parse_error(char *msg);
22 extern char *homefile(char *filename);
23 extern void *ecalloc(int count, unsigned int size);
24 extern int lstrtoi(char *str, char **end);
25 extern char version[];
27 static int linenum;
28 static int errors;
29 static int less_version = 0;
30 static char *lesskey_file;
32 static struct lesskey_cmdname cmdnames[] =
34 { "back-bracket", A_B_BRACKET },
35 { "back-line", A_B_LINE },
36 { "back-line-force", A_BF_LINE },
37 { "back-screen", A_B_SCREEN },
38 { "back-scroll", A_B_SCROLL },
39 { "back-search", A_B_SEARCH },
40 { "back-window", A_B_WINDOW },
41 { "clear-mark", A_CLRMARK },
42 { "debug", A_DEBUG },
43 { "digit", A_DIGIT },
44 { "display-flag", A_DISP_OPTION },
45 { "display-option", A_DISP_OPTION },
46 { "end", A_GOEND },
47 { "end-scroll", A_RRSHIFT },
48 { "examine", A_EXAMINE },
49 { "filter", A_FILTER },
50 { "first-cmd", A_FIRSTCMD },
51 { "firstcmd", A_FIRSTCMD },
52 { "flush-repaint", A_FREPAINT },
53 { "forw-bracket", A_F_BRACKET },
54 { "forw-forever", A_F_FOREVER },
55 { "forw-until-hilite", A_F_UNTIL_HILITE },
56 { "forw-line", A_F_LINE },
57 { "forw-line-force", A_FF_LINE },
58 { "forw-screen", A_F_SCREEN },
59 { "forw-screen-force", A_FF_SCREEN },
60 { "forw-scroll", A_F_SCROLL },
61 { "forw-search", A_F_SEARCH },
62 { "forw-window", A_F_WINDOW },
63 { "goto-end", A_GOEND },
64 { "goto-end-buffered", A_GOEND_BUF },
65 { "goto-line", A_GOLINE },
66 { "goto-mark", A_GOMARK },
67 { "help", A_HELP },
68 { "index-file", A_INDEX_FILE },
69 { "invalid", A_UINVALID },
70 { "left-scroll", A_LSHIFT },
71 { "next-file", A_NEXT_FILE },
72 { "next-tag", A_NEXT_TAG },
73 { "noaction", A_NOACTION },
74 { "no-scroll", A_LLSHIFT },
75 { "percent", A_PERCENT },
76 { "pipe", A_PIPE },
77 { "prev-file", A_PREV_FILE },
78 { "prev-tag", A_PREV_TAG },
79 { "quit", A_QUIT },
80 { "remove-file", A_REMOVE_FILE },
81 { "repaint", A_REPAINT },
82 { "repaint-flush", A_FREPAINT },
83 { "repeat-search", A_AGAIN_SEARCH },
84 { "repeat-search-all", A_T_AGAIN_SEARCH },
85 { "reverse-search", A_REVERSE_SEARCH },
86 { "reverse-search-all", A_T_REVERSE_SEARCH },
87 { "right-scroll", A_RSHIFT },
88 { "set-mark", A_SETMARK },
89 { "set-mark-bottom", A_SETMARKBOT },
90 { "shell", A_SHELL },
91 { "status", A_STAT },
92 { "toggle-flag", A_OPT_TOGGLE },
93 { "toggle-option", A_OPT_TOGGLE },
94 { "undo-hilite", A_UNDO_SEARCH },
95 { "clear-search", A_CLR_SEARCH },
96 { "version", A_VERSION },
97 { "visual", A_VISUAL },
98 { NULL, 0 }
101 static struct lesskey_cmdname editnames[] =
103 { "back-complete", EC_B_COMPLETE },
104 { "backspace", EC_BACKSPACE },
105 { "delete", EC_DELETE },
106 { "down", EC_DOWN },
107 { "end", EC_END },
108 { "expand", EC_EXPAND },
109 { "forw-complete", EC_F_COMPLETE },
110 { "home", EC_HOME },
111 { "insert", EC_INSERT },
112 { "invalid", EC_UINVALID },
113 { "kill-line", EC_LINEKILL },
114 { "abort", EC_ABORT },
115 { "left", EC_LEFT },
116 { "literal", EC_LITERAL },
117 { "right", EC_RIGHT },
118 { "up", EC_UP },
119 { "word-backspace", EC_W_BACKSPACE },
120 { "word-delete", EC_W_DELETE },
121 { "word-left", EC_W_LEFT },
122 { "word-right", EC_W_RIGHT },
123 { NULL, 0 }
127 * Print a parse error message.
129 static void
130 parse_error(fmt, arg1)
131 char *fmt;
132 char *arg1;
134 char buf[1024];
135 int n = snprintf(buf, sizeof(buf), "%s: line %d: ", lesskey_file, linenum);
136 if (n >= 0 && n < sizeof(buf))
137 snprintf(buf+n, sizeof(buf)-n, fmt, arg1);
138 ++errors;
139 lesskey_parse_error(buf);
143 * Initialize lesskey_tables.
145 static void
146 init_tables(tables)
147 struct lesskey_tables *tables;
149 tables->currtable = &tables->cmdtable;
151 tables->cmdtable.names = cmdnames;
152 tables->cmdtable.is_var = 0;
153 xbuf_init(&tables->cmdtable.buf);
155 tables->edittable.names = editnames;
156 tables->edittable.is_var = 0;
157 xbuf_init(&tables->edittable.buf);
159 tables->vartable.names = NULL;
160 tables->vartable.is_var = 1;
161 xbuf_init(&tables->vartable.buf);
164 #define CHAR_STRING_LEN 8
166 static char *
167 char_string(buf, ch, lit)
168 char *buf;
169 int ch;
170 int lit;
172 if (lit || (ch >= 0x20 && ch < 0x7f))
174 buf[0] = ch;
175 buf[1] = '\0';
176 } else
178 snprintf(buf, CHAR_STRING_LEN, "\\x%02x", ch);
180 return buf;
184 * Increment char pointer by one up to terminating nul byte.
186 static char *
187 increment_pointer(p)
188 char *p;
190 if (*p == '\0')
191 return p;
192 return p+1;
196 * Parse one character of a string.
198 static char *
199 tstr(pp, xlate)
200 char **pp;
201 int xlate;
203 char *p;
204 char ch;
205 int i;
206 static char buf[CHAR_STRING_LEN];
207 static char tstr_control_k[] =
208 { SK_SPECIAL_KEY, SK_CONTROL_K, 6, 1, 1, 1, '\0' };
210 p = *pp;
211 switch (*p)
213 case '\\':
214 ++p;
215 switch (*p)
217 case '0': case '1': case '2': case '3':
218 case '4': case '5': case '6': case '7':
220 * Parse an octal number.
222 ch = 0;
223 i = 0;
225 ch = 8*ch + (*p - '0');
226 while (*++p >= '0' && *p <= '7' && ++i < 3);
227 *pp = p;
228 if (xlate && ch == CONTROL('K'))
229 return tstr_control_k;
230 return char_string(buf, ch, 1);
231 case 'b':
232 *pp = p+1;
233 return ("\b");
234 case 'e':
235 *pp = p+1;
236 return char_string(buf, ESC, 1);
237 case 'n':
238 *pp = p+1;
239 return ("\n");
240 case 'r':
241 *pp = p+1;
242 return ("\r");
243 case 't':
244 *pp = p+1;
245 return ("\t");
246 case 'k':
247 if (xlate)
249 switch (*++p)
251 case 'b': ch = SK_BACKSPACE; break;
252 case 'B': ch = SK_CTL_BACKSPACE; break;
253 case 'd': ch = SK_DOWN_ARROW; break;
254 case 'D': ch = SK_PAGE_DOWN; break;
255 case 'e': ch = SK_END; break;
256 case 'h': ch = SK_HOME; break;
257 case 'i': ch = SK_INSERT; break;
258 case 'l': ch = SK_LEFT_ARROW; break;
259 case 'L': ch = SK_CTL_LEFT_ARROW; break;
260 case 'r': ch = SK_RIGHT_ARROW; break;
261 case 'R': ch = SK_CTL_RIGHT_ARROW; break;
262 case 't': ch = SK_BACKTAB; break;
263 case 'u': ch = SK_UP_ARROW; break;
264 case 'U': ch = SK_PAGE_UP; break;
265 case 'x': ch = SK_DELETE; break;
266 case 'X': ch = SK_CTL_DELETE; break;
267 case '1': ch = SK_F1; break;
268 default:
269 parse_error("invalid escape sequence \"\\k%s\"", char_string(buf, *p, 0));
270 *pp = increment_pointer(p);
271 return ("");
273 *pp = p+1;
274 buf[0] = SK_SPECIAL_KEY;
275 buf[1] = ch;
276 buf[2] = 6;
277 buf[3] = 1;
278 buf[4] = 1;
279 buf[5] = 1;
280 buf[6] = '\0';
281 return (buf);
283 /* FALLTHRU */
284 default:
286 * Backslash followed by any other char
287 * just means that char.
289 *pp = increment_pointer(p);
290 char_string(buf, *p, 1);
291 if (xlate && buf[0] == CONTROL('K'))
292 return tstr_control_k;
293 return (buf);
295 case '^':
297 * Caret means CONTROL.
299 *pp = increment_pointer(p+1);
300 char_string(buf, CONTROL(p[1]), 1);
301 if (xlate && buf[0] == CONTROL('K'))
302 return tstr_control_k;
303 return (buf);
305 *pp = increment_pointer(p);
306 char_string(buf, *p, 1);
307 if (xlate && buf[0] == CONTROL('K'))
308 return tstr_control_k;
309 return (buf);
312 static int
313 issp(ch)
314 char ch;
316 return (ch == ' ' || ch == '\t');
320 * Skip leading spaces in a string.
322 static char *
323 skipsp(s)
324 char *s;
326 while (issp(*s))
327 s++;
328 return (s);
332 * Skip non-space characters in a string.
334 static char *
335 skipnsp(s)
336 char *s;
338 while (*s != '\0' && !issp(*s))
339 s++;
340 return (s);
344 * Clean up an input line:
345 * strip off the trailing newline & any trailing # comment.
347 static char *
348 clean_line(s)
349 char *s;
351 int i;
353 s = skipsp(s);
354 for (i = 0; s[i] != '\0' && s[i] != '\n' && s[i] != '\r'; i++)
355 if (s[i] == '#' && (i == 0 || s[i-1] != '\\'))
356 break;
357 s[i] = '\0';
358 return (s);
362 * Add a byte to the output command table.
364 static void
365 add_cmd_char(c, tables)
366 int c;
367 struct lesskey_tables *tables;
369 xbuf_add(&tables->currtable->buf, c);
372 static void
373 erase_cmd_char(tables)
374 struct lesskey_tables *tables;
376 xbuf_pop(&tables->currtable->buf);
380 * Add a string to the output command table.
382 static void
383 add_cmd_str(s, tables)
384 char *s;
385 struct lesskey_tables *tables;
387 for ( ; *s != '\0'; s++)
388 add_cmd_char(*s, tables);
392 * Does a given version number match the running version?
393 * Operator compares the running version to the given version.
395 static int
396 match_version(op, ver)
397 char op;
398 int ver;
400 switch (op)
402 case '>': return less_version > ver;
403 case '<': return less_version < ver;
404 case '+': return less_version >= ver;
405 case '-': return less_version <= ver;
406 case '=': return less_version == ver;
407 case '!': return less_version != ver;
408 default: return 0; /* cannot happen */
413 * Handle a #version line.
414 * If the version matches, return the part of the line that should be executed.
415 * Otherwise, return NULL.
417 static char *
418 version_line(s, tables)
419 char *s;
420 struct lesskey_tables *tables;
422 char op;
423 int ver;
424 char *e;
425 char buf[CHAR_STRING_LEN];
427 s += strlen("#version");
428 s = skipsp(s);
429 op = *s++;
430 /* Simplify 2-char op to one char. */
431 switch (op)
433 case '<': if (*s == '=') { s++; op = '-'; } break;
434 case '>': if (*s == '=') { s++; op = '+'; } break;
435 case '=': if (*s == '=') { s++; } break;
436 case '!': if (*s == '=') { s++; } break;
437 default:
438 parse_error("invalid operator '%s' in #version line", char_string(buf, op, 0));
439 return (NULL);
441 s = skipsp(s);
442 ver = lstrtoi(s, &e);
443 if (e == s)
445 parse_error("non-numeric version number in #version line", "");
446 return (NULL);
448 if (!match_version(op, ver))
449 return (NULL);
450 return (e);
454 * See if we have a special "control" line.
456 static char *
457 control_line(s, tables)
458 char *s;
459 struct lesskey_tables *tables;
461 #define PREFIX(str,pat) (strncmp(str,pat,strlen(pat)) == 0)
463 if (PREFIX(s, "#line-edit"))
465 tables->currtable = &tables->edittable;
466 return (NULL);
468 if (PREFIX(s, "#command"))
470 tables->currtable = &tables->cmdtable;
471 return (NULL);
473 if (PREFIX(s, "#env"))
475 tables->currtable = &tables->vartable;
476 return (NULL);
478 if (PREFIX(s, "#stop"))
480 add_cmd_char('\0', tables);
481 add_cmd_char(A_END_LIST, tables);
482 return (NULL);
484 if (PREFIX(s, "#version"))
486 return (version_line(s, tables));
488 return (s);
492 * Find an action, given the name of the action.
494 static int
495 findaction(actname, tables)
496 char *actname;
497 struct lesskey_tables *tables;
499 int i;
501 for (i = 0; tables->currtable->names[i].cn_name != NULL; i++)
502 if (strcmp(tables->currtable->names[i].cn_name, actname) == 0)
503 return (tables->currtable->names[i].cn_action);
504 parse_error("unknown action: \"%s\"", actname);
505 return (A_INVALID);
509 * Parse a line describing one key binding, of the form
510 * KEY ACTION [EXTRA]
511 * where KEY is the user key sequence, ACTION is the
512 * resulting less action, and EXTRA is an "extra" user
513 * key sequence injected after the action.
515 static void
516 parse_cmdline(p, tables)
517 char *p;
518 struct lesskey_tables *tables;
520 char *actname;
521 int action;
522 char *s;
523 char c;
526 * Parse the command string and store it in the current table.
530 s = tstr(&p, 1);
531 add_cmd_str(s, tables);
532 } while (*p != '\0' && !issp(*p));
534 * Terminate the command string with a null byte.
536 add_cmd_char('\0', tables);
539 * Skip white space between the command string
540 * and the action name.
541 * Terminate the action name with a null byte.
543 p = skipsp(p);
544 if (*p == '\0')
546 parse_error("missing action", "");
547 return;
549 actname = p;
550 p = skipnsp(p);
551 c = *p;
552 *p = '\0';
555 * Parse the action name and store it in the current table.
557 action = findaction(actname, tables);
560 * See if an extra string follows the action name.
562 *p = c;
563 p = skipsp(p);
564 if (*p == '\0')
566 add_cmd_char(action, tables);
567 } else
570 * OR the special value A_EXTRA into the action byte.
571 * Put the extra string after the action byte.
573 add_cmd_char(action | A_EXTRA, tables);
574 while (*p != '\0')
575 add_cmd_str(tstr(&p, 0), tables);
576 add_cmd_char('\0', tables);
581 * Parse a variable definition line, of the form
582 * NAME = VALUE
584 static void
585 parse_varline(line, tables)
586 char *line;
587 struct lesskey_tables *tables;
589 char *s;
590 char *p = line;
591 char *eq;
593 eq = strchr(line, '=');
594 if (eq != NULL && eq > line && eq[-1] == '+')
597 * Rather ugly way of handling a += line.
598 * {{ Note that we ignore the variable name and
599 * just append to the previously defined variable. }}
601 erase_cmd_char(tables); /* backspace over the final null */
602 p = eq+1;
603 } else
607 s = tstr(&p, 0);
608 add_cmd_str(s, tables);
609 } while (*p != '\0' && !issp(*p) && *p != '=');
611 * Terminate the variable name with a null byte.
613 add_cmd_char('\0', tables);
614 p = skipsp(p);
615 if (*p++ != '=')
617 parse_error("missing = in variable definition", "");
618 return;
620 add_cmd_char(EV_OK|A_EXTRA, tables);
622 p = skipsp(p);
623 while (*p != '\0')
625 s = tstr(&p, 0);
626 add_cmd_str(s, tables);
628 add_cmd_char('\0', tables);
632 * Parse a line from the lesskey file.
634 static void
635 parse_line(line, tables)
636 char *line;
637 struct lesskey_tables *tables;
639 char *p;
642 * See if it is a control line.
644 p = control_line(line, tables);
645 if (p == NULL)
646 return;
648 * Skip leading white space.
649 * Replace the final newline with a null byte.
650 * Ignore blank lines and comments.
652 p = clean_line(p);
653 if (*p == '\0')
654 return;
656 if (tables->currtable->is_var)
657 parse_varline(p, tables);
658 else
659 parse_cmdline(p, tables);
663 * Parse a lesskey source file and store result in tables.
666 parse_lesskey(infile, tables)
667 char *infile;
668 struct lesskey_tables *tables;
670 FILE *desc;
671 char line[1024];
673 if (infile == NULL)
674 infile = homefile(DEF_LESSKEYINFILE);
675 lesskey_file = infile;
677 init_tables(tables);
678 errors = 0;
679 linenum = 0;
680 if (less_version == 0)
681 less_version = lstrtoi(version, NULL);
684 * Open the input file.
686 if (strcmp(infile, "-") == 0)
687 desc = stdin;
688 else if ((desc = fopen(infile, "r")) == NULL)
690 /* parse_error("cannot open lesskey file %s", infile); */
691 return (-1);
695 * Read and parse the input file, one line at a time.
697 while (fgets(line, sizeof(line), desc) != NULL)
699 ++linenum;
700 parse_line(line, tables);
702 fclose(desc);
703 return (errors);