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 } 581