root/src/simh/linehistory.c

/* [previous][next][first][last][top][bottom][index][help] */

DEFINITIONS

This source file includes following definitions.
  1. linenoiseMaskModeEnable
  2. linenoiseMaskModeDisable
  3. linenoiseSetMultiLine
  4. isUnsupportedTerm
  5. enableRawMode
  6. disableRawMode
  7. getCursorPosition
  8. getColumns
  9. linenoiseClearScreen
  10. linenoiseBeep
  11. freeCompletions
  12. completeLine
  13. linenoiseSetCompletionCallback
  14. linenoiseSetHintsCallback
  15. linenoiseSetFreeHintsCallback
  16. linenoiseAddCompletion
  17. abInit
  18. abAppend
  19. abFree
  20. refreshShowHints
  21. refreshSingleLine
  22. refreshMultiLine
  23. refreshLine
  24. linenoiseEditInsert
  25. linenoiseEditMoveLeft
  26. linenoiseEditMoveRight
  27. linenoiseEditMoveWordEnd
  28. linenoiseEditMoveWordStart
  29. linenoiseEditMoveHome
  30. linenoiseEditMoveEnd
  31. linenoiseEditHistoryNext
  32. linenoiseSearchInHistory
  33. linenoiseEditDelete
  34. linenoiseEditBackspace
  35. linenoiseEditDeletePrevWord
  36. linenoiseEditDeleteNextWord
  37. linenoiseEdit
  38. linenoiseRaw
  39. linenoiseNoTTY
  40. linenoise
  41. linenoiseFree
  42. freeHistory
  43. linenoiseAtExit
  44. linenoiseHistoryAdd
  45. linenoiseHistorySetMaxLen
  46. pstrlen

   1 /*
   2  * vim: filetype=c:tabstop=4:ai:expandtab
   3  * SPDX-License-Identifier: BSD-2-Clause
   4  * scspell-id: 676f2224-f62a-11ec-baf3-80ee73e9b8e7
   5  *
   6  * -------------------------------------------------------------------------
   7  *
   8  * linehistory.c
   9  *
  10  * linehistory is forked from linenoise; the original version
  11  * is available from https://github.com/antirez/linenoise
  12  *
  13  * ------------------------------------------------------------------------
  14  *
  15  * Copyright (c) 2010-2014 Salvatore Sanfilippo <antirez@gmail.com>
  16  * Copyright (c) 2010-2013 Pieter Noordhuis <pcnoordhuis@gmail.com>
  17  * Copyright (c) 2020-2021 Jeffrey H. Johnson <trnsz@pobox.com>
  18  * Copyright (c) 2021-2025 The DPS8M Development Team
  19  *
  20  * All rights reserved.
  21  *
  22  * Redistribution and use in source and binary forms, with or without
  23  * modification, are permitted provided that the following conditions are
  24  * met:
  25  *
  26  *  *  Redistributions of source code must retain the above copyright
  27  *     notice, this list of conditions and the following disclaimer.
  28  *
  29  *  *  Redistributions in binary form must reproduce the above copyright
  30  *     notice, this list of conditions and the following disclaimer in the
  31  *     documentation and/or other materials provided with the distribution.
  32  *
  33  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  34  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  35  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
  36  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
  37  * HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
  38  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  39  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
  40  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
  41  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  42  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
  43  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  44  *
  45  * -------------------------------------------------------------------------
  46  */
  47 
  48 #if !defined(_POSIX_C_SOURCE)
  49 # define _POSIX_C_SOURCE 200809L
  50 #endif /* if !defined(_POSIX_C_SOURCE) */
  51 
  52 #if _POSIX_C_SOURCE < 200809L
  53 # undef _POSIX_C_SOURCE
  54 # define _POSIX_C_SOURCE 200809L
  55 #endif /* if _POSIX_C_SOURCE < 200809L */
  56 
  57 #if !defined(__MINGW32__) && !defined(CROSS_MINGW32) && !defined(CROSS_MINGW64) && \
  58     !defined(__MINGW64__) && !defined(_MSC_VER) && !defined(_MSC_BUILD)
  59 
  60 # if defined(__OpenBSD__)
  61 #  if !defined(_BSD_SOURCE)
  62 #   define _BSD_SOURCE
  63 #  endif
  64 # endif
  65 
  66 # if defined(__sun) && defined(__SVR4)
  67 #  if !defined(__EXTENSIONS__)
  68 #   define __EXTENSIONS__ 1
  69 #  endif /* if !defined(__EXTENSIONS__) */
  70 # endif /* if defined(__sun) && defined(__SVR4) */
  71 # include <termios.h>
  72 # if defined(__sun) && defined(__SVR4)
  73 #  include <sys/termiox.h>
  74 # endif /* if defined(__sun) && defined(__SVR4) */
  75 # if defined(__FreeBSD__) || defined(__DragonFly__)
  76 #  if !defined(__BSD_VISIBLE)
  77 #   define __BSD_VISIBLE 1
  78 #  endif
  79 #  include <sys/types.h>
  80 #  include <sys/sysctl.h>
  81 # endif
  82 # include "linehistory.h"
  83 # if !defined(__NetBSD__)
  84 #  include "../dps8/dps8.h"
  85 # endif /* if !defined(__NetBSD__) */
  86 # include <ctype.h>
  87 # include <errno.h>
  88 # include <stdio.h>
  89 # include <stdlib.h>
  90 # include <string.h>
  91 # include <strings.h>
  92 # include <sys/ioctl.h>
  93 # include <sys/stat.h>
  94 # include <sys/types.h>
  95 # include <unistd.h>
  96 
  97 # if defined(FREE)
  98 #  undef FREE
  99 # endif /* if defined(FREE) */
 100 # define FREE(p) do  \
 101   {                  \
 102     free((p));       \
 103     (p) = NULL;      \
 104   } while(0)
 105 
 106 # define LINENOISE_DEFAULT_HISTORY_MAX_LEN  100
 107 # define LINENOISE_MAX_LINE                4096
 108 
 109 static char *unsupported_term[] = {
 110   "dumb", "cons25", "emacs", NULL
 111 };
 112 
 113 # if defined(LH_COMPLETION)
 114 static linenoiseCompletionCallback *completionCallback = NULL;
 115 # endif /* if defined(LH_COMPLETION) */
 116 
 117 # if defined(LH_HINTS)
 118 static linenoiseHintsCallback *hintsCallback           = NULL;
 119 static linenoiseFreeHintsCallback *freeHintsCallback   = NULL;
 120 # endif /* if defined(LH_HINTS) */
 121 
 122 static struct termios orig_termios; /* In order to restore at exit.*/
 123 
 124 # if defined(LH_MASKMODE)
 125 static int maskmode          = 0;   /* Show "**" instead of input for passwords */
 126 # endif /* if defined(LH_MASKMODE) */
 127 
 128 static int rawmode           = 0;
 129 static int mlmode            = 0;   /* Multi line mode. Default is single line. */
 130 static int atexit_registered = 0;   /* Register atexit just 1 time.             */
 131 static int history_max_len   = LINENOISE_DEFAULT_HISTORY_MAX_LEN;
 132 static int history_len       = 0;
 133 static char **history        = NULL;
 134 
 135 /*
 136  * The linenoiseState structure represents the state during line editing.
 137  * We pass this state to functions implementing specific editing
 138  * functionalities.
 139  */
 140 
 141 struct linenoiseState
 142 {
 143   int    ifd;            /* Terminal stdin file descriptor.                  */
 144   int    ofd;            /* Terminal stdout file descriptor.                 */
 145   char   *buf;           /* Edited line buffer.                              */
 146   size_t buflen;         /* Edited line buffer size.                         */
 147   const  char *prompt;   /* Prompt to display.                               */
 148   size_t plen;           /* Prompt length.                                   */
 149   size_t pos;            /* Current cursor position.                         */
 150   size_t oldpos;         /* Previous refresh cursor position.                */
 151   size_t len;            /* Current edited line length.                      */
 152   size_t cols;           /* Number of columns in terminal.                   */
 153   size_t maxrows;        /* Maximum num of rows used so far (multiline mode) */
 154   int    history_index;  /* The history index we are currently editing.      */
 155 };
 156 
 157 enum KEY_ACTION
 158 {
 159   KEY_NULL  = 0,  /* NULL      */
 160   CTRL_A    = 1,  /* Ctrl-A    */
 161   CTRL_B    = 2,  /* Ctrl-B    */
 162   CTRL_C    = 3,  /* Ctrl-C    */
 163   CTRL_D    = 4,  /* Ctrl-D    */
 164   CTRL_E    = 5,  /* Ctrl-E    */
 165   CTRL_F    = 6,  /* Ctrl-F    */
 166   CTRL_H    = 8,  /* Ctrl-H    */
 167   TAB       = 9,  /* Tab       */
 168   CTRL_K    = 11, /* Ctrl-K    */
 169   CTRL_L    = 12, /* Ctrl-L    */
 170   ENTER     = 13, /* Enter     */
 171   CTRL_N    = 14, /* Ctrl-N    */
 172   CTRL_P    = 16, /* Ctrl-P    */
 173   CTRL_T    = 20, /* Ctrl-T    */
 174   CTRL_U    = 21, /* Ctrl-U    */
 175   CTRL_W    = 23, /* Ctrl-W    */
 176   ESC       = 27, /* Escape    */
 177   BACKSPACE = 127 /* Backspace */
 178 };
 179 
 180 static void linenoiseAtExit(void);
 181 static void refreshLine(struct linenoiseState *l);
 182 size_t pstrlen(const char *s);
 183 
 184 # if defined(LH_MASKMODE)
 185 
 186 /*
 187  * Enable "mask mode". When it is enabled, instead of the input that
 188  * the user is typing, the terminal will just display a corresponding
 189  * number of asterisks, like "****". This is useful for passwords and other
 190  * secrets that should not be displayed.
 191  */
 192 
 193 void
 194 linenoiseMaskModeEnable(void)
     /* [previous][next][first][last][top][bottom][index][help] */
 195 {
 196   maskmode = 1;
 197 }
 198 
 199 /* Disable mask mode. */
 200 void
 201 linenoiseMaskModeDisable(void)
     /* [previous][next][first][last][top][bottom][index][help] */
 202 {
 203   maskmode = 0;
 204 }
 205 # endif /* if defined(LH_MASKMODE) */
 206 
 207 /* Set if to use or not the multi line mode. */
 208 void
 209 linenoiseSetMultiLine(int ml)
     /* [previous][next][first][last][top][bottom][index][help] */
 210 {
 211   mlmode = ml;
 212 }
 213 
 214 /*
 215  * Return true if the terminal name is in the list of terminals we know are
 216  * not able to understand basic escape sequences.
 217  */
 218 
 219 static int
 220 isUnsupportedTerm(void)
     /* [previous][next][first][last][top][bottom][index][help] */
 221 {
 222   char *term = getenv("TERM");
 223   int j;
 224 
 225   if (term == NULL)
 226   {
 227     return ( 0 );
 228   }
 229 
 230   for (j = 0; unsupported_term[j]; j++)
 231   {
 232     if (!strcasecmp(term, unsupported_term[j]))
 233     {
 234       return ( 1 );
 235     }
 236   }
 237 
 238   return ( 0 );
 239 }
 240 
 241 /* Raw mode */
 242 static int
 243 enableRawMode(int fd)
     /* [previous][next][first][last][top][bottom][index][help] */
 244 {
 245   struct termios raw;
 246 
 247   if (!isatty(STDIN_FILENO))
 248   {
 249     goto fatal;
 250   }
 251 
 252   if (!atexit_registered)
 253   {
 254     atexit(linenoiseAtExit);
 255     atexit_registered = 1;
 256   }
 257 
 258   if (tcgetattr(fd, &orig_termios) == -1)
 259   {
 260     goto fatal;
 261   }
 262 
 263   raw = orig_termios; /* modify the original mode */
 264   /* input modes: no break, no CR to NL, no parity check, no strip char,
 265    * no start/stop output control. */
 266   raw.c_iflag &= ~( BRKINT | ICRNL | INPCK | ISTRIP | IXON );
 267   /* control modes - set 8 bit chars */
 268   raw.c_cflag |= ( CS8 );
 269   /* local modes - echoing off, canonical off, no extended functions,
 270    * no signal chars (^Z,^C) */
 271   raw.c_lflag &= ~( ECHO | ICANON | IEXTEN | ISIG );
 272   /* control chars - set return condition: min number of bytes and timer.
 273    * We want read to return every single byte, without timeout. */
 274   raw.c_cc[VMIN]  = 1;
 275   raw.c_cc[VTIME] = 0; /* 1 byte, no timer */
 276 
 277   /* put terminal in raw mode after flushing */
 278 # if defined(__ANDROID__)
 279 #  define TCSA_TYPE TCSANOW
 280   (void)fflush(stdout);
 281   (void)fflush(stderr);
 282 # else
 283 #  define TCSA_TYPE TCSAFLUSH
 284 # endif
 285   if (tcsetattr(fd, TCSA_TYPE, &raw) < 0)
 286   {
 287     goto fatal;
 288   }
 289 
 290   rawmode = 1;
 291   return ( 0 );
 292 
 293 fatal:
 294   errno = ENOTTY;
 295   return ( -1 );
 296 }
 297 
 298 static void
 299 disableRawMode(int fd)
     /* [previous][next][first][last][top][bottom][index][help] */
 300 {
 301   /* Don't even check the return value as it's too late. */
 302 # if defined(__ANDROID__)
 303   (void)fflush(stdout);
 304   (void)fflush(stderr);
 305 # endif
 306   if (rawmode && tcsetattr(fd, TCSA_TYPE, &orig_termios) != -1)
 307   {
 308     rawmode = 0;
 309   }
 310 }
 311 
 312 /*
 313  * Use the ESC [6n escape sequence to query the horizontal cursor position
 314  * and return it. On error -1 is returned, on success the position of the
 315  * cursor.
 316  */
 317 
 318 static int
 319 getCursorPosition(int ifd, int ofd)
     /* [previous][next][first][last][top][bottom][index][help] */
 320 {
 321   char     buf[32];
 322   int      cols, rows;
 323   unsigned int i = 0;
 324 
 325   /* Report cursor location */
 326   if (write(ofd, "\x1b[6n", 4) != 4)
 327   {
 328     return ( -1 );
 329   }
 330 
 331   /* Read the response: ESC [ rows ; cols R */
 332   while (i < sizeof ( buf ) - 1)
 333   {
 334     if (read(ifd, buf + i, 1) != 1)
 335     {
 336       break;
 337     }
 338 
 339     if (buf[i] == 'R')
 340     {
 341       break;
 342     }
 343 
 344     i++;
 345   }
 346   buf[i] = '\0';
 347 
 348   /* Parse it. */
 349   if (buf[0] != ESC || buf[1] != '[')
 350   {
 351     return ( -1 );
 352   }
 353 
 354   if (sscanf(buf + 2, "%d;%d", &rows, &cols) != 2)
 355   {
 356     return ( -1 );
 357   }
 358 
 359   return ( cols );
 360 }
 361 
 362 /* Try to get the number of columns in terminal, or assume 80 */
 363 static int
 364 getColumns(int ifd, int ofd)
     /* [previous][next][first][last][top][bottom][index][help] */
 365 {
 366   struct winsize ws;
 367 
 368   if (ioctl(1, TIOCGWINSZ, &ws) == -1 || ws.ws_col == 0)
 369   {
 370     /* ioctl() failed. Try to query the terminal itself. */
 371     int start, cols;
 372 
 373     /* Get the initial position so we can restore it later. */
 374     start = getCursorPosition(ifd, ofd);
 375     if (start == -1)
 376     {
 377       goto failed;
 378     }
 379 
 380     /* Go to right margin and get position. */
 381     if (write(ofd, "\x1b[999C", 6) != 6)
 382     {
 383       goto failed;
 384     }
 385 
 386     cols = getCursorPosition(ifd, ofd);
 387     if (cols == -1)
 388     {
 389       goto failed;
 390     }
 391 
 392     /* Restore position. */
 393     if (cols > start)
 394     {
 395       char seq[32];
 396       (void)snprintf(seq, sizeof ( seq ), "\x1b[%dD", cols - start);
 397       if (write(ofd, seq, strlen(seq)) == -1) { /* Can't recover... */ }
 398     }
 399 
 400     return ( cols );
 401   }
 402   else
 403   {
 404     return ( ws.ws_col );
 405   }
 406 
 407 failed:
 408   return ( 80 );
 409 }
 410 
 411 /* Clear the screen. Used to handle ctrl+l */
 412 void
 413 linenoiseClearScreen(void)
     /* [previous][next][first][last][top][bottom][index][help] */
 414 {
 415   if (write(STDOUT_FILENO, "\x1b[H\x1b[2J", 7) <= 0)
 416   { /* nothing to do, just to avoid warning. */
 417   }
 418 }
 419 
 420 # if defined(LH_COMPLETION)
 421 
 422 /*
 423  * Beep, used for completion when there is nothing to complete or when all
 424  * the choices were already shown.
 425  */
 426 
 427 static void
 428 linenoiseBeep(void)
     /* [previous][next][first][last][top][bottom][index][help] */
 429 {
 430   (void)fflush(stdout);
 431   (void)fprintf(stderr, "\x7");
 432   (void)fflush(stderr);
 433 }
 434 
 435 /* Free a list of completion option populated by linenoiseAddCompletion(). */
 436 static void
 437 freeCompletions(const linenoiseCompletions *lc)
     /* [previous][next][first][last][top][bottom][index][help] */
 438 {
 439   size_t i;
 440 
 441   for (i = 0; i < lc->len; i++)
 442   {
 443     FREE(lc->cvec[i]);
 444   }
 445 
 446   if (lc->cvec != NULL)
 447   {
 448     FREE(lc->cvec);
 449   }
 450 }
 451 
 452 /*
 453  * This is an helper function for linenoiseEdit() and is called when the
 454  * user types the <tab> key in order to complete the string currently in the
 455  * input.
 456  *
 457  * The state of the editing is encapsulated into the pointed linenoiseState
 458  * structure as described in the structure definition.
 459  */
 460 
 461 static int
 462 completeLine(struct linenoiseState *ls)
     /* [previous][next][first][last][top][bottom][index][help] */
 463 {
 464   linenoiseCompletions lc = {
 465     0, NULL
 466   };
 467   int nread, nwritten;
 468   char c = 0;
 469 
 470   completionCallback(ls->buf, &lc);
 471   if (lc.len == 0)
 472   {
 473     linenoiseBeep();
 474   }
 475   else
 476   {
 477     size_t stop = 0, i = 0;
 478 
 479     while (!stop)
 480     {
 481       /* Show completion or original buffer */
 482       if (i < lc.len)
 483       {
 484         struct linenoiseState saved = *ls;
 485 
 486         ls->len = ls->pos = strlen(lc.cvec[i]);
 487         ls->buf = lc.cvec[i];
 488         refreshLine(ls);
 489         ls->len = saved.len;
 490         ls->pos = saved.pos;
 491         ls->buf = saved.buf;
 492       }
 493       else
 494       {
 495         refreshLine(ls);
 496       }
 497 
 498       nread = read(ls->ifd, &c, 1);
 499       if (nread <= 0)
 500       {
 501         freeCompletions(&lc);
 502         return ( -1 );
 503       }
 504 
 505       switch (c)
 506       {
 507       case 9: /* Tab */
 508         i = ( i + 1 ) % ( lc.len + 1 );
 509         if (i == lc.len)
 510         {
 511           linenoiseBeep();
 512         }
 513 
 514         stop = 1;
 515         break;
 516 
 517       default:
 518         /* Update buffer and return */
 519         if (i < lc.len)
 520         {
 521           nwritten = snprintf(ls->buf, ls->buflen, "%s", lc.cvec[i]);
 522           ls->len  = ls->pos = nwritten;
 523         }
 524 
 525         stop = 1;
 526         break;
 527       }
 528     }
 529   }
 530 
 531   freeCompletions(&lc);
 532   return ( c ); /* Return last read character */
 533 }
 534 
 535 /* Register a callback function to be called for tab-completion. */
 536 void
 537 linenoiseSetCompletionCallback(linenoiseCompletionCallback *fn)
     /* [previous][next][first][last][top][bottom][index][help] */
 538 {
 539   completionCallback = fn;
 540 }
 541 
 542 # endif /* if defined(LH_COMPLETION) */
 543 
 544 # if defined(LH_HINTS)
 545 
 546 /*
 547  * Register a hits function to be called to show hits to the user at the
 548  * right of the prompt.
 549  */
 550 
 551 void
 552 linenoiseSetHintsCallback(linenoiseHintsCallback *fn)
     /* [previous][next][first][last][top][bottom][index][help] */
 553 {
 554   hintsCallback = fn;
 555 }
 556 
 557 /*
 558  * Register a function to free the hints returned by the hints callback
 559  * registered with linenoiseSetHintsCallback().
 560  */
 561 
 562 void
 563 linenoiseSetFreeHintsCallback(linenoiseFreeHintsCallback *fn)
     /* [previous][next][first][last][top][bottom][index][help] */
 564 {
 565   freeHintsCallback = fn;
 566 }
 567 
 568 # endif /* if defined(LH_HINTS) */
 569 
 570 # if defined(LH_COMPLETION)
 571 
 572 /*
 573  * This function is used by the callback function registered by the user
 574  * in order to add completion options given the input string when the
 575  * user typed <tab>.
 576  */
 577 
 578 void
 579 linenoiseAddCompletion(linenoiseCompletions *lc, const char *str)
     /* [previous][next][first][last][top][bottom][index][help] */
 580 {
 581   size_t len = strlen(str);
 582   char *copy, **cvec;
 583 
 584   copy = malloc(len + 1);
 585   if (copy == NULL)
 586   {
 587     return;
 588   }
 589 
 590   memcpy(copy, str, len + 1);
 591   cvec = realloc(lc->cvec, sizeof ( char * ) * ( lc->len + 1 ));
 592   if (cvec == NULL)
 593   {
 594     FREE(copy);
 595     return;
 596   }
 597 
 598   lc->cvec            = cvec;
 599   lc->cvec[lc->len++] = copy;
 600 }
 601 
 602 # endif /* if defined(LH_COMPLETION) */
 603 
 604 /*
 605  * We define a very simple "append buffer" structure, that is an heap
 606  * allocated string where we can append to. This is useful in order to
 607  * write all the escape sequences in a buffer and flush them to the standard
 608  * output in a single call, to avoid flickering effects.
 609  */
 610 
 611 struct abuf
 612 {
 613   char *b;
 614   int len;
 615 };
 616 
 617 static void
 618 abInit(struct abuf *ab)
     /* [previous][next][first][last][top][bottom][index][help] */
 619 {
 620   ab->b = NULL;
 621   ab->len = 0;
 622 }
 623 
 624 static void
 625 abAppend(struct abuf *ab, const char *s, int len)
     /* [previous][next][first][last][top][bottom][index][help] */
 626 {
 627   if (len <= 0)
 628   {
 629     return;
 630   }
 631 
 632   char *new = realloc(ab->b, 1 + ab->len + len);
 633 
 634   if (new == NULL)
 635   {
 636     return;
 637   }
 638 
 639   memcpy(new + ab->len, s, len);
 640   ab->b    = new;
 641   ab->len += len;
 642 }
 643 
 644 static void
 645 abFree(const struct abuf *ab)
     /* [previous][next][first][last][top][bottom][index][help] */
 646 {
 647   free(ab->b); /* X-LINTED: FREE */
 648 }
 649 
 650 # if defined(LH_HINTS)
 651 
 652 /*
 653  * Helper of refreshSingleLine() and refreshMultiLine() to show hints
 654  * to the right of the prompt.
 655  */
 656 
 657 static void
 658 refreshShowHints(struct abuf *ab, const struct linenoiseState *l, int plen)
     /* [previous][next][first][last][top][bottom][index][help] */
 659 {
 660   char seq[64];
 661 
 662   seq[0] = '\0';
 663   if (hintsCallback && plen + l->len < l->cols)
 664   {
 665     int color = -1, bold = 0;
 666     char *hint = hintsCallback(l->buf, &color, &bold);
 667     if (hint)
 668     {
 669       int hintlen    = strlen(hint);
 670       int hintmaxlen = l->cols - ( plen + l->len );
 671       if (hintlen > hintmaxlen)
 672       {
 673         hintlen = hintmaxlen;
 674       }
 675 
 676       if (bold == 1 && color == -1)
 677       {
 678         color = 37;
 679       }
 680 
 681       if (color != -1 || bold != 0)
 682       {
 683         (void)snprintf(seq, sizeof ( seq ), "\033[%d;%d;49m", bold, color);
 684       }
 685       else
 686       {
 687         seq[0] = '\0';
 688       }
 689 
 690       abAppend(ab, seq, strlen(seq));
 691       abAppend(ab, hint, hintlen);
 692       if (color != -1 || bold != 0)
 693       {
 694         abAppend(ab, "\033[0m", 4);
 695       }
 696 
 697       /* Call the function to free the hint returned. */
 698       if (freeHintsCallback)
 699       {
 700         freeHintsCallback(hint);
 701       }
 702     }
 703   }
 704 }
 705 
 706 # endif /* if defined(LH_HINTS) */
 707 
 708 /*
 709  * Single line low level line refresh.
 710  *
 711  * Rewrite the currently edited line accordingly to the buffer content,
 712  * cursor position, and number of columns of the terminal.
 713  */
 714 
 715 static void
 716 refreshSingleLine(const struct linenoiseState *l)
     /* [previous][next][first][last][top][bottom][index][help] */
 717 {
 718   char   seq[64];
 719   size_t plen = pstrlen(l->prompt);
 720   int    fd   = l->ofd;
 721   char   *buf = l->buf;
 722   size_t len  = l->len;
 723   size_t pos  = l->pos;
 724   struct abuf ab;
 725 
 726   while (( plen + pos ) >= l->cols)
 727   {
 728     buf++;
 729     len--;
 730     pos--;
 731   }
 732   while (plen + len > l->cols)
 733   {
 734     len--;
 735   }
 736 
 737   abInit(&ab);
 738   /* Cursor to left edge */
 739   (void)snprintf(seq, sizeof ( seq ), "\r");
 740   abAppend(&ab, seq, strlen(seq));
 741   /* Write the prompt and the current buffer content */
 742   abAppend(&ab, l->prompt, strlen(l->prompt));
 743 # if defined(LH_MASKMODE)
 744   if (maskmode == 1)
 745   {
 746     while (len--)
 747     {
 748       abAppend(&ab, "*", 1);
 749     }
 750   }
 751   else
 752   {
 753 # endif /* if defined(LH_MASKMODE) */
 754   abAppend(&ab, buf, len);
 755 # if defined(LH_MASKMODE)
 756 }
 757 # endif /* if defined(LH_MASKMODE) */
 758 # if defined(LH_HINTS)
 759   /* Show hits if any. */
 760   refreshShowHints(&ab, l, plen);
 761 # endif /* if defined(LH_HINTS) */
 762   /* Erase to right */
 763   (void)snprintf(seq, sizeof ( seq ), "\x1b[0K");
 764   abAppend(&ab, seq, strlen(seq));
 765   /* Move cursor to original position. */
 766   (void)snprintf(seq, sizeof ( seq ), "\r\x1b[%dC", (int)( pos + plen ));
 767   abAppend(&ab, seq, strlen(seq));
 768   if (write(fd, ab.b, ab.len) == -1)
 769   { /* Can't recover from write error. */
 770   }
 771 
 772   abFree(&ab);
 773 }
 774 
 775 /*
 776  * Multi line low level line refresh.
 777  *
 778  * Rewrite the currently edited line accordingly to the buffer content,
 779  * cursor position, and number of columns of the terminal.
 780  */
 781 
 782 static void
 783 refreshMultiLine(struct linenoiseState *l)
     /* [previous][next][first][last][top][bottom][index][help] */
 784 {
 785   char seq[64];
 786   int plen = strlen(l->prompt);
 787   int rows = ( plen + l->len + l->cols - 1 )
 788              / l->cols; /* rows used by current buf. */
 789   int rpos = ( plen + l->oldpos + l->cols ) / l->cols; /* cursor relative row. */
 790   int rpos2; /* rpos after refresh. */
 791   int col; /* column position, zero-based. */
 792   int old_rows = l->maxrows;
 793   int fd = l->ofd, j;
 794   struct abuf ab;
 795 
 796   /* Update maxrows if needed. */
 797   if (rows > (int)l->maxrows)
 798   {
 799     l->maxrows = rows;
 800   }
 801 
 802   /*
 803    * First step: clear all the lines used before. To do so start by
 804    * going to the last row.
 805    */
 806 
 807   abInit(&ab);
 808   if (old_rows - rpos > 0)
 809   {
 810     (void)snprintf(seq, sizeof ( seq ), "\x1b[%dB", old_rows - rpos);
 811     abAppend(&ab, seq, strlen(seq));
 812   }
 813 
 814   /* Now for every row clear it, go up. */
 815   for (j = 0; j < old_rows - 1; j++)
 816   {
 817     (void)snprintf(seq, sizeof ( seq ), "\r\x1b[0K\x1b[1A");
 818     abAppend(&ab, seq, strlen(seq));
 819   }
 820 
 821   /* Clean the top line. */
 822   (void)snprintf(seq, sizeof ( seq ), "\r\x1b[0K");
 823   abAppend(&ab, seq, strlen(seq));
 824 
 825   /* Write the prompt and the current buffer content */
 826   abAppend(&ab, l->prompt, strlen(l->prompt));
 827 # if defined(LH_MASKMODE)
 828   if (maskmode == 1)
 829   {
 830     unsigned int i;
 831     for (i = 0; i < l->len; i++)
 832     {
 833       abAppend(&ab, "*", 1);
 834     }
 835   }
 836   else
 837   {
 838 # endif /* if defined(LH_MASKMODE) */
 839   abAppend(&ab, l->buf, l->len);
 840 # if defined(LH_MASKMODE)
 841 }
 842 # endif /* if defined(LH_MASKMODE) */
 843 
 844 # if defined(LH_HINTS)
 845   /* Show hits if any. */
 846   refreshShowHints(&ab, l, plen);
 847 # endif /* if defined(LH_HINTS) */
 848 
 849   /*
 850    * If we are at the very end of the screen with our prompt, we need to
 851    * emit a newline and move the prompt to the first column.
 852    */
 853 
 854   if (l->pos && l->pos == l->len && ( l->pos + plen ) % l->cols == 0)
 855   {
 856     abAppend(&ab, "\n", 1); //-NLOK
 857     (void)snprintf(seq, sizeof ( seq ), "\r"); //-NLOK
 858     abAppend(&ab, seq, strlen(seq));
 859     rows++;
 860     if (rows > (int)l->maxrows)
 861     {
 862       l->maxrows = rows;
 863     }
 864   }
 865 
 866   /* Move cursor to right position. */
 867   rpos2 = ( plen + l->pos + l->cols ) / l->cols;
 868 
 869   /* Go up till we reach the expected position. */
 870   if (rows - rpos2 > 0)
 871   {
 872     (void)snprintf(seq, sizeof ( seq ), "\x1b[%dA", rows - rpos2);
 873     abAppend(&ab, seq, strlen(seq));
 874   }
 875 
 876   /* Set column. */
 877   col = ( plen + (int)l->pos ) % (int)l->cols;
 878   if (col)
 879   {
 880     (void)snprintf(seq, sizeof ( seq ), "\r\x1b[%dC", col);
 881   }
 882   else
 883   {
 884     (void)snprintf(seq, sizeof ( seq ), "\r");
 885   }
 886 
 887   abAppend(&ab, seq, strlen(seq));
 888 
 889   l->oldpos = l->pos;
 890 
 891   if (write(fd, ab.b, ab.len) == -1)
 892   { /* Can't recover from write error. */
 893   }
 894 
 895   abFree(&ab);
 896 }
 897 
 898 /*
 899  * Calls the two low level functions refreshSingleLine() or
 900  * refreshMultiLine() according to the selected mode.
 901  */
 902 
 903 static void
 904 refreshLine(struct linenoiseState *l)
     /* [previous][next][first][last][top][bottom][index][help] */
 905 {
 906   l->cols = getColumns(STDIN_FILENO, STDOUT_FILENO);
 907   if (mlmode)
 908   {
 909     refreshMultiLine(l);
 910   }
 911   else
 912   {
 913     refreshSingleLine(l);
 914   }
 915 }
 916 
 917 /*
 918  * Insert the character 'c' at cursor current position.
 919  * On error writing to the terminal -1 is returned, otherwise 0.
 920  */
 921 
 922 int
 923 linenoiseEditInsert(struct linenoiseState *l, char c)
     /* [previous][next][first][last][top][bottom][index][help] */
 924 {
 925   if (l->len < l->buflen)
 926   {
 927     if (l->len == l->pos)
 928     {
 929       l->buf[l->pos] = c;
 930       l->pos++;
 931       l->len++;
 932       l->buf[l->len] = '\0';
 933 # if defined(LH_MASKMODE) && defined(LH_HINTS)
 934       if (( !mlmode && l->plen + l->len < l->cols && !hintsCallback ))
 935       {
 936         /* Avoid a full update of the line in the trivial case. */
 937         char d = ( maskmode == 1 ) ? '*' : (char)c;
 938         if (write(l->ofd, &d, 1) == -1)
 939         {
 940           return ( -1 );
 941         }
 942       }
 943       else
 944       {
 945 # endif /* if defined(LH_MASKMODE) && defined(LH_HINTS) */
 946       refreshLine(l);
 947 # if defined(LH_MASKMODE) && defined(LH_HINTS)
 948       }
 949 # endif /* if defined(LH_MASKMODE) && defined(LH_HINTS) */
 950     }
 951     else
 952     {
 953       memmove(l->buf + l->pos + 1, l->buf + l->pos, l->len - l->pos);
 954       l->buf[l->pos] = c;
 955       l->len++;
 956       l->pos++;
 957       l->buf[l->len] = '\0';
 958       refreshLine(l);
 959     }
 960   }
 961 
 962   return ( 0 );
 963 }
 964 
 965 /* Move cursor on the left. */
 966 void
 967 linenoiseEditMoveLeft(struct linenoiseState *l)
     /* [previous][next][first][last][top][bottom][index][help] */
 968 {
 969   if (l->pos > 0)
 970   {
 971     l->pos--;
 972     refreshLine(l);
 973   }
 974 }
 975 
 976 /* Move cursor on the right. */
 977 void
 978 linenoiseEditMoveRight(struct linenoiseState *l)
     /* [previous][next][first][last][top][bottom][index][help] */
 979 {
 980   if (l->pos != l->len)
 981   {
 982     l->pos++;
 983     refreshLine(l);
 984   }
 985 }
 986 
 987 /* Move cursor to the end of the current word. */
 988 void
 989 linenoiseEditMoveWordEnd(struct linenoiseState *l)
     /* [previous][next][first][last][top][bottom][index][help] */
 990 {
 991   if (l->len == 0 || l->pos >= l->len)
 992   {
 993     return;
 994   }
 995 
 996   if (l->buf[l->pos] == ' ')
 997   {
 998     while (l->pos < l->len && l->buf[l->pos] == ' ')
 999     {
1000       ++l->pos;
1001     }
1002   }
1003 
1004   while (l->pos < l->len && l->buf[l->pos] != ' ')
1005   {
1006     ++l->pos;
1007   }
1008   refreshLine(l);
1009 }
1010 
1011 /* Move cursor to the start of the current word. */
1012 void
1013 linenoiseEditMoveWordStart(struct linenoiseState *l)
     /* [previous][next][first][last][top][bottom][index][help] */
1014 {
1015   if (l->len == 0)
1016   {
1017     return;
1018   }
1019 
1020   if (l->buf[l->pos - 1] == ' ')
1021   {
1022     --l->pos;
1023   }
1024 
1025   if (l->buf[l->pos] == ' ')
1026   {
1027     while (l->pos > 0 && l->buf[l->pos] == ' ')
1028     {
1029       --l->pos;
1030     }
1031   }
1032 
1033   while (l->pos > 0 && l->buf[l->pos - 1] != ' ')
1034   {
1035     --l->pos;
1036   }
1037   refreshLine(l);
1038 }
1039 
1040 /* Move cursor to the start of the line. */
1041 void
1042 linenoiseEditMoveHome(struct linenoiseState *l)
     /* [previous][next][first][last][top][bottom][index][help] */
1043 {
1044   if (l->pos != 0)
1045   {
1046     l->pos = 0;
1047     refreshLine(l);
1048   }
1049 }
1050 
1051 /* Move cursor to the end of the line. */
1052 void
1053 linenoiseEditMoveEnd(struct linenoiseState *l)
     /* [previous][next][first][last][top][bottom][index][help] */
1054 {
1055   if (l->pos != l->len)
1056   {
1057     l->pos = l->len;
1058     refreshLine(l);
1059   }
1060 }
1061 
1062 /*
1063  * Substitute the currently edited line with the next
1064  * or previous history entry as specified by 'dir'.
1065  */
1066 
1067 # define LINENOISE_HISTORY_NEXT 0
1068 # define LINENOISE_HISTORY_PREV 1
1069 void
1070 linenoiseEditHistoryNext(struct linenoiseState *l, int dir)
     /* [previous][next][first][last][top][bottom][index][help] */
1071 {
1072   if (history_len > 1)
1073   {
1074     /*
1075      * Update the current history entry before to
1076      * overwrite it with the next one.
1077      */
1078 
1079     FREE(history[history_len - 1 - l->history_index]);
1080     history[history_len - 1 - l->history_index] = strdup(l->buf);
1081     if (!history[history_len - 1 - l->history_index])
1082       {
1083         (void)fprintf (stderr, "\rFATAL: Out of memory! Aborting at %s[%s:%d]\r\n",
1084                        __func__, __FILE__, __LINE__);
1085 # if defined(USE_BACKTRACE)
1086 #  if defined(SIGUSR2)
1087         (void)raise(SIGUSR2);
1088         /*NOTREACHED*/ /* unreachable */
1089 #  endif /* if defined(SIGUSR2) */
1090 # endif /* if defined(USE_BACKTRACE) */
1091         abort();
1092       }
1093     /* Show the new entry */
1094     l->history_index += ( dir == LINENOISE_HISTORY_PREV ) ? 1 : -1;
1095     if (l->history_index < 0)
1096     {
1097       l->history_index = 0;
1098       return;
1099     }
1100     else if (l->history_index >= history_len)
1101     {
1102       l->history_index = history_len - 1;
1103       return;
1104     }
1105 
1106     strncpy(l->buf, history[history_len - 1 - l->history_index], l->buflen);
1107     l->buf[l->buflen - 1] = '\0';
1108     l->len = l->pos = strlen(l->buf);
1109     refreshLine(l);
1110   }
1111 }
1112 
1113 /*
1114  * Search a line in history that start with the same characters as the
1115  * currently edited line. Substitute the current line with this history
1116  */
1117 
1118 # define LINENOISE_SEARCH_HISTORY_FORWARD 0
1119 # define LINENOISE_SEARCH_HISTORY_REVERSE 1
1120 void
1121 linenoiseSearchInHistory(struct linenoiseState *l, int direction)
     /* [previous][next][first][last][top][bottom][index][help] */
1122 {
1123   if (history_len > 1)
1124   {
1125     /*
1126      * Update the current history entry before to
1127      * overwrite it with the next one.
1128      */
1129 
1130     FREE(history[history_len - 1 - l->history_index]);
1131     history[history_len - 1 - l->history_index] = strdup(l->buf);
1132     if (!history[history_len - 1 - l->history_index])
1133       {
1134         (void)fprintf (stderr, "\rFATAL: Out of memory! Aborting at %s[%s:%d]\r\n",
1135                        __func__, __FILE__, __LINE__);
1136 # if defined(USE_BACKTRACE)
1137 #  if defined(SIGUSR2)
1138         (void)raise(SIGUSR2);
1139         /*NOTREACHED*/ /* unreachable */
1140 #  endif /* if defined(SIGUSR2) */
1141 # endif /* if defined(USE_BACKTRACE) */
1142         abort();
1143       }
1144 
1145     /* Search new entry */
1146     int cnt;
1147     if (direction == LINENOISE_SEARCH_HISTORY_FORWARD)
1148     {
1149       cnt = history_len - 2 - l->history_index;
1150       for (; cnt >= 0; cnt--)
1151       {
1152         /*
1153          * Search a history entry that start with same
1154          * as the current line until the cursor position
1155          */
1156 
1157         if (strncmp(l->buf, history[cnt], l->pos) == 0)
1158         {
1159           strncpy(l->buf, history[cnt], l->buflen);
1160           l->buf[l->buflen - 1] = '\0';
1161           /* Don't change old cursor position */
1162           l->len = strlen(l->buf);
1163 
1164           /*
1165            * Set history index so that we can contiune
1166            * the search on this position
1167            */
1168 
1169           l->history_index = history_len - 1 - cnt;
1170           refreshLine(l);
1171           return;
1172         }
1173       }
1174     }
1175     else if (direction == LINENOISE_SEARCH_HISTORY_REVERSE)
1176     {
1177       cnt = history_len - l->history_index;
1178       for (; cnt < history_len; cnt++)
1179       {
1180         /*
1181          * Search a history entry that start with same
1182          * as the current line until the cursor position
1183          */
1184 
1185         if (strncmp(l->buf, history[cnt], l->pos) == 0)
1186         {
1187           strncpy(l->buf, history[cnt], l->buflen);
1188           l->buf[l->buflen - 1] = '\0';
1189           /* Don't change old cursor position */
1190           l->len = strlen(l->buf);
1191 
1192           /*
1193            * Set history index so that we can contiune
1194            * the search on this position
1195            */
1196 
1197           l->history_index = history_len - 1 - cnt;
1198           refreshLine(l);
1199           return;
1200         }
1201       }
1202     }
1203   }
1204 }
1205 
1206 /*
1207  * Delete the character at the right of the cursor without altering the cursor
1208  * position. Basically this is what happens with the "Delete" keyboard key.
1209  */
1210 
1211 void
1212 linenoiseEditDelete(struct linenoiseState *l)
     /* [previous][next][first][last][top][bottom][index][help] */
1213 {
1214   if (l->len > 0 && l->pos < l->len)
1215   {
1216     memmove(l->buf + l->pos, l->buf + l->pos + 1, l->len - l->pos - 1);
1217     l->len--;
1218     l->buf[l->len] = '\0';
1219     refreshLine(l);
1220   }
1221 }
1222 
1223 /* Backspace implementation. */
1224 void
1225 linenoiseEditBackspace(struct linenoiseState *l)
     /* [previous][next][first][last][top][bottom][index][help] */
1226 {
1227   if (l->pos > 0 && l->len > 0)
1228   {
1229     memmove(l->buf + l->pos - 1, l->buf + l->pos, l->len - l->pos);
1230     l->pos--;
1231     l->len--;
1232     l->buf[l->len] = '\0';
1233     refreshLine(l);
1234   }
1235 }
1236 
1237 /*
1238  * Delete the previous word, maintaining the
1239  * cursor at the start of the current word.
1240  */
1241 
1242 void
1243 linenoiseEditDeletePrevWord(struct linenoiseState *l)
     /* [previous][next][first][last][top][bottom][index][help] */
1244 {
1245   size_t old_pos = l->pos;
1246   size_t diff;
1247 
1248   while (l->pos > 0 && l->buf[l->pos - 1] == ' ')
1249   {
1250     l->pos--;
1251   }
1252   while (l->pos > 0 && l->buf[l->pos - 1] != ' ')
1253   {
1254     l->pos--;
1255   }
1256   diff = old_pos - l->pos;
1257   memmove(l->buf + l->pos, l->buf + old_pos, l->len - old_pos + 1);
1258   l->len -= diff;
1259   refreshLine(l);
1260 }
1261 
1262 /* Delete the next word, maintaining the cursor at the same position */
1263 void
1264 linenoiseEditDeleteNextWord(struct linenoiseState *l)
     /* [previous][next][first][last][top][bottom][index][help] */
1265 {
1266   size_t next_word_end = l->pos;
1267 
1268   while (next_word_end < l->len && l->buf[next_word_end] == ' ')
1269   {
1270     ++next_word_end;
1271   }
1272   while (next_word_end < l->len && l->buf[next_word_end] != ' ')
1273   {
1274     ++next_word_end;
1275   }
1276   memmove(l->buf + l->pos, l->buf + next_word_end, l->len - next_word_end);
1277   l->len -= next_word_end - l->pos;
1278   refreshLine(l);
1279 }
1280 
1281 /*
1282  * This function is the core of the line editing capability of linenoise.
1283  * It expects 'fd' to be already in "raw mode" so that every key pressed
1284  * will be returned ASAP to read().
1285  *
1286  * The resulting string is put into 'buf' when the user type enter, or
1287  * when ctrl+d is typed.
1288  *
1289  * The function returns the length of the current buffer.
1290  */
1291 
1292 static int
1293 linenoiseEdit(int stdin_fd, int stdout_fd, char *buf, size_t buflen,
     /* [previous][next][first][last][top][bottom][index][help] */
1294               const char *prompt)
1295 {
1296   struct linenoiseState l;
1297 
1298   /*
1299    * Populate the linenoise state that we pass to functions implementing
1300    * specific editing functionalities.
1301    */
1302 
1303   l.ifd           = stdin_fd;
1304   l.ofd           = stdout_fd;
1305   l.buf           = buf;
1306   l.buflen        = buflen;
1307   l.prompt        = prompt;
1308   l.plen          = pstrlen(prompt);
1309   l.oldpos        = l.pos = 0;
1310   l.len           = 0;
1311   l.cols          = getColumns(stdin_fd, stdout_fd);
1312   l.maxrows       = 0;
1313   l.history_index = 0;
1314 
1315   /* Buffer starts empty. */
1316   l.buf[0] = '\0';
1317   l.buflen--; /* Make sure there is always space for the nulterm */
1318 
1319   /*
1320    * The latest history entry is always our current
1321    * buffer, that initially is just an empty string.
1322    */
1323 
1324   (void)linenoiseHistoryAdd("");
1325 
1326   if (write(l.ofd, prompt, strlen(prompt)) == -1)
1327   {
1328     return ( -1 );
1329   }
1330 
1331   while (1)
1332   {
1333     signed char c;
1334     int    nread;
1335     char   seq[3];
1336 
1337     nread = read(l.ifd, &c, 1);
1338     if (nread <= 0)
1339     {
1340       return ( l.len );
1341     }
1342 
1343 # if defined(LH_COMPLETION)
1344 
1345     /*
1346      * Only autocomplete when the callback is set. It returns < 0 when
1347      * there was an error reading from fd. Otherwise it will return the
1348      * character that should be handled next.
1349      */
1350 
1351     if (c == 9 && completionCallback != NULL)
1352     {
1353       int cint = completeLine(&l);
1354       /* Return on errors */
1355       if (cint < 0)
1356       {
1357         return ( l.len );
1358       }
1359 
1360       /* Read next character when 0 */
1361       if (cint == 0)
1362       {
1363         continue;
1364       }
1365 
1366       c = (char)cint;
1367     }
1368 
1369 # endif /* if defined(LH_COMPLETION) */
1370 
1371     switch (c)
1372     {
1373     case 9:
1374       break;    /* johnsonjh - disable processing of tabs */
1375 
1376     case ENTER: /* Enter */
1377       history_len--;
1378       FREE(history[history_len]);
1379       if (mlmode)
1380       {
1381         linenoiseEditMoveEnd(&l);
1382       }
1383 
1384 # if defined(LH_HINTS)
1385       if (hintsCallback)
1386       {
1387         /*
1388          * Force a refresh without hints to leave the previous
1389          * line as the user typed it after a newline.
1390          */
1391 
1392         linenoiseHintsCallback *hc = hintsCallback;
1393         hintsCallback              = NULL;
1394         refreshLine(&l);
1395         hintsCallback              = hc;
1396       }
1397 
1398 # endif /* if defined(LH_HINTS) */
1399       return ((int)l.len );
1400 
1401     case CTRL_C: /* Ctrl-C */
1402       errno = EAGAIN;
1403       return ( -1 );
1404 
1405     case BACKSPACE: /* Backspace */
1406     case 8:         /* Ctrl-H */
1407       linenoiseEditBackspace(&l);
1408       break;
1409 
1410     case CTRL_D:     /* Ctrl-D, remove char at right of cursor, or */
1411       if (l.len > 0) /* if the line is empty, act as end-of-file.  */
1412       {
1413         linenoiseEditDelete(&l);
1414       }
1415       else
1416       {
1417         history_len--;
1418         FREE(history[history_len]);
1419         return ( -1 );
1420       }
1421 
1422       break;
1423 
1424     case CTRL_T: /* Ctrl-T */
1425       break;
1426 
1427     case CTRL_B: /* Ctrl-B */
1428       linenoiseEditMoveLeft(&l);
1429       break;
1430 
1431     case CTRL_F: /* Ctrl-F */
1432       linenoiseEditMoveRight(&l);
1433       break;
1434 
1435     case CTRL_P: /* Ctrl-P */
1436       linenoiseEditHistoryNext(&l, LINENOISE_HISTORY_PREV);
1437       break;
1438 
1439     case CTRL_N: /* Ctrl-N */
1440       linenoiseEditHistoryNext(&l, LINENOISE_HISTORY_NEXT);
1441       break;
1442 
1443     case ESC: /* Escape Sequence */
1444 
1445       /*
1446        * Read the next two bytes representing the escape sequence.
1447        * Use two calls to handle slow terminals returning the two
1448        * chars at different times.
1449        */
1450 
1451       if (read(l.ifd, seq, 1) == -1)
1452       {
1453         break;
1454       }
1455 
1456       if (read(l.ifd, seq + 1, 1) == -1)
1457       {
1458         break;
1459       }
1460 
1461       /* ESC [ sequences. */
1462       if (seq[0] == '[')
1463       {
1464         if (seq[1] >= '0' && seq[1] <= '9')
1465         {
1466           /* Extended escape, read additional byte. */
1467           if (read(l.ifd, seq + 2, 1) == -1)
1468           {
1469             break;
1470           }
1471 
1472           if (seq[2] == '~')
1473           {
1474             switch (seq[1])
1475             {
1476             case '3': /* Delete */
1477               linenoiseEditDelete(&l);
1478               break;
1479 
1480             case '6': /* Page Down */
1481               linenoiseSearchInHistory(
1482                 &l,
1483                 LINENOISE_SEARCH_HISTORY_REVERSE);
1484               break;
1485 
1486             case '5': /* Page Up */
1487               linenoiseSearchInHistory(
1488                 &l,
1489                 LINENOISE_SEARCH_HISTORY_FORWARD);
1490               break;
1491             }
1492           }
1493         }
1494         else
1495         {
1496           switch (seq[1])
1497           {
1498           case 'A': /* Up */
1499             linenoiseEditHistoryNext(&l, LINENOISE_HISTORY_PREV);
1500             break;
1501 
1502           case 'B': /* Down */
1503             linenoiseEditHistoryNext(&l, LINENOISE_HISTORY_NEXT);
1504             break;
1505 
1506           case 'C': /* Right */
1507             linenoiseEditMoveRight(&l);
1508             break;
1509 
1510           case 'D': /* Left */
1511             linenoiseEditMoveLeft(&l);
1512             break;
1513 
1514           case 'H': /* Home */
1515             linenoiseEditMoveHome(&l);
1516             break;
1517 
1518           case 'F': /* End */
1519             linenoiseEditMoveEnd(&l);
1520             break;
1521           }
1522         }
1523       }
1524 
1525       break;
1526 
1527     default:
1528       if (linenoiseEditInsert(&l, c))
1529       {
1530         return ( -1 );
1531       }
1532 
1533       break;
1534 
1535     case CTRL_U: /* Ctrl+U, delete the whole line. */
1536       buf[0] = '\0';
1537       l.pos  = l.len = 0;
1538       refreshLine(&l);
1539       break;
1540 
1541     case CTRL_K: /* Ctrl+K, delete from current to end of line. */
1542       buf[l.pos] = '\0';
1543       l.len      = l.pos;
1544       refreshLine(&l);
1545       break;
1546 
1547     case CTRL_A: /* Ctrl+A, go to the start of the line */
1548       linenoiseEditMoveHome(&l);
1549       break;
1550 
1551     case CTRL_E: /* Ctrl+E, go to the end of the line */
1552       linenoiseEditMoveEnd(&l);
1553       break;
1554 
1555     case CTRL_L: /* Ctrl+L, clear screen */
1556       linenoiseClearScreen();
1557       refreshLine(&l);
1558       break;
1559 
1560     case CTRL_W: /* Ctrl+W, delete previous word */
1561       linenoiseEditDeletePrevWord(&l);
1562       break;
1563     }
1564   }
1565 # if defined(SUNLINT) || !defined(__SUNPRO_C) && !defined(__SUNPRO_CC)
1566  /*NOTREACHED*/ /* unreachable */
1567  return ( -1 );
1568 # endif /* if defined(SUNLINT) || !defined(__SUNPRO_C) && !defined(__SUNPRO_CC) */
1569 }
1570 
1571 /*
1572  * This function calls the line editing function linenoiseEdit() using
1573  * the STDIN file descriptor set in raw mode.
1574  */
1575 
1576 static int
1577 linenoiseRaw(char *buf, size_t buflen, const char *prompt)
     /* [previous][next][first][last][top][bottom][index][help] */
1578 {
1579   int count;
1580 
1581   if (buflen == 0)
1582   {
1583     errno = EINVAL;
1584     return ( -1 );
1585   }
1586 
1587   if (enableRawMode(STDIN_FILENO) == -1)
1588   {
1589     return ( -1 );
1590   }
1591 
1592   count = linenoiseEdit(STDIN_FILENO, STDOUT_FILENO, buf, buflen, prompt);
1593   disableRawMode(STDIN_FILENO);
1594   (void)printf("\r\n");
1595   return ( count );
1596 }
1597 
1598 /*
1599  * This function is called when linenoise() is called with the standard
1600  * input file descriptor not attached to a TTY. So for example when the
1601  * program using linenoise is called in pipe or with a file redirected
1602  * to its standard input. In this case, we want to be able to return the
1603  * line regardless of its length (by default we are limited to 4k).
1604  */
1605 
1606 static char *
1607 linenoiseNoTTY(void)
     /* [previous][next][first][last][top][bottom][index][help] */
1608 {
1609   char *line = NULL;
1610   size_t len = 0, maxlen = 0;
1611 
1612   while (1)
1613   {
1614     if (len == maxlen)
1615     {
1616       if (maxlen == 0)
1617       {
1618         maxlen = 16;
1619       }
1620 
1621       maxlen *= 2;
1622       char *oldval = line;
1623       line = realloc(line, maxlen); //-V701
1624       if (line == NULL)
1625       {
1626         if (oldval)
1627         {
1628           FREE(oldval);
1629         }
1630 
1631         return ( NULL );
1632       }
1633     }
1634 
1635     int c = fgetc(stdin);
1636     if (c == EOF || c == '\n')
1637     {
1638       if (c == EOF && len == 0)
1639       {
1640         FREE(line);
1641         return ( NULL );
1642       }
1643       else
1644       {
1645         line[len] = '\0';
1646         return ( line );
1647       }
1648     }
1649     else
1650     {
1651       line[len] = c;
1652       len++;
1653     }
1654   }
1655 # if defined(SUNLINT) || !defined(__SUNPRO_C) && !defined(__SUNPRO_CC)
1656  /*NOTREACHED*/ /* unreachable */
1657  return ( NULL );
1658 # endif /* if defined(SUNLINT) || !defined(__SUNPRO_C) && !defined(__SUNPRO_CC) */
1659 }
1660 
1661 /*
1662  * The high level function that is the main API of the linenoise library.
1663  * This function checks if the terminal has basic capabilities, just checking
1664  * for a blacklist of stupid terminals, and later either calls the line
1665  * editing function or uses dummy fgets() so that you will be able to type
1666  * something even in the most desperate of the conditions.
1667  */
1668 
1669 char *
1670 linenoise(const char *prompt)
     /* [previous][next][first][last][top][bottom][index][help] */
1671 {
1672   char buf[LINENOISE_MAX_LINE];
1673   int count;
1674 
1675   if (!isatty(STDIN_FILENO))
1676   {
1677     /*
1678      * Not a tty: read from file / pipe. In this mode we don't want any
1679      * limit to the line size, so we call a function to handle that.
1680      */
1681 
1682     return ( linenoiseNoTTY());
1683   }
1684   else if (isUnsupportedTerm())
1685   {
1686     size_t len;
1687 
1688     (void)fflush(stderr);
1689     (void)printf("%s", prompt);
1690     (void)fflush(stdout);
1691     if (fgets(buf, LINENOISE_MAX_LINE, stdin) == NULL)
1692     {
1693       return ( NULL );
1694     }
1695 
1696     len = strlen(buf);
1697     while (len && ( buf[len - 1] == '\n' || buf[len - 1] == '\r' ))
1698     {
1699       len--;
1700       buf[len] = '\0';
1701     }
1702     return ( strdup(buf));
1703   }
1704   else
1705   {
1706     count = linenoiseRaw(buf, LINENOISE_MAX_LINE, prompt);
1707     if (count == -1)
1708     {
1709       return ( NULL );
1710     }
1711 
1712     return ( strdup(buf));
1713   }
1714 }
1715 
1716 /*
1717  * This is just a wrapper the user may want to call in order to make sure
1718  * the linenoise returned buffer is freed with the same allocator it was
1719  * created with. Useful when the main program is using an alternative
1720  * allocator.
1721  */
1722 
1723 void
1724 linenoiseFree(void *ptr)
     /* [previous][next][first][last][top][bottom][index][help] */
1725 {
1726   FREE(ptr);
1727 }
1728 
1729 /*
1730  * Free the history, but does not reset it. Only used when we have to
1731  * exit() to avoid memory leaks are reported by valgrind & co.
1732  */
1733 
1734 static void
1735 freeHistory(void)
     /* [previous][next][first][last][top][bottom][index][help] */
1736 {
1737   if (history)
1738   {
1739     int j;
1740 
1741     for (j = 0; j < history_len; j++)
1742     {
1743       FREE(history[j]);
1744     }
1745 
1746     FREE(history);
1747   }
1748 }
1749 
1750 /* At exit we'll try to fix the terminal to the initial conditions. */
1751 static void
1752 linenoiseAtExit(void)
     /* [previous][next][first][last][top][bottom][index][help] */
1753 {
1754   disableRawMode(STDIN_FILENO);
1755   freeHistory();
1756 }
1757 
1758 /*
1759  * This is the API call to add a new entry in the linenoise history.
1760  * It uses a fixed array of char pointers that are shifted (memmoved)
1761  * when the history max length is reached in order to remove the older
1762  * entry and make room for the new one, so it is not exactly suitable for huge
1763  * histories, but will work well for a few hundred of entries.
1764  */
1765 
1766 int
1767 linenoiseHistoryAdd(const char *line)
     /* [previous][next][first][last][top][bottom][index][help] */
1768 {
1769   char *linecopy;
1770 
1771   if (history_max_len == 0)
1772   {
1773     return ( 0 );
1774   }
1775 
1776   /* Initialization on first call. */
1777   if (history == NULL)
1778   {
1779     history = malloc(sizeof ( char * ) * history_max_len);
1780     if (history == NULL)
1781     {
1782       return ( 0 );
1783     }
1784 
1785     (void)memset(history, 0, ( sizeof ( char * ) * history_max_len ));
1786   }
1787 
1788   /* Don't add duplicated lines. */
1789   if (( history_len > 0 ) && ( !strcmp(history[history_len - 1], line)))
1790   {
1791     return ( 0 );
1792   }
1793 
1794   /*
1795    * Add an heap allocated copy of the line in the history.
1796    * If we reached the max length, remove the older line.
1797    */
1798 
1799   linecopy = strdup(line);
1800   if (!linecopy)
1801   {
1802     return ( 0 );
1803   }
1804 
1805   if (history_len == history_max_len)
1806   {
1807     FREE(history[0]);
1808     memmove(
1809       history,
1810       history + 1,
1811       sizeof ( char * ) * ( history_max_len - 1 ));
1812     history_len--;
1813   }
1814 
1815   history[history_len] = linecopy;
1816   history_len++;
1817   return ( 1 );
1818 }
1819 
1820 /*
1821  * Set the maximum length for the history. This function can be called even
1822  * if there is already some history, the function will make sure to retain
1823  * just the latest 'len' elements if the new history length value is smaller
1824  * than the amount of items already inside the history.
1825  */
1826 
1827 int
1828 linenoiseHistorySetMaxLen(int len)
     /* [previous][next][first][last][top][bottom][index][help] */
1829 {
1830   char **new;
1831 
1832   if (len < 1)
1833   {
1834     return ( 0 );
1835   }
1836 
1837   if (history)
1838   {
1839     int tocopy = history_len;
1840 
1841     new = malloc(sizeof ( char * ) * len);
1842     if (new == NULL)
1843     {
1844       return ( 0 );
1845     }
1846 
1847     /* If we can't copy everything, free the elements we'll not use. */
1848     if (len < tocopy)
1849     {
1850       int j;
1851 
1852       for (j = 0; j < tocopy - len; j++)
1853       {
1854         FREE(history[j]);
1855       }
1856 
1857       tocopy = len;
1858     }
1859 
1860     (void)memset(new, 0, sizeof ( char * ) * len);
1861     memcpy(
1862       new,
1863       history + ( history_len - tocopy ),
1864       sizeof ( char * ) * tocopy);
1865     FREE(history);
1866     history = new;
1867   }
1868 
1869   history_max_len = len;
1870   if (history_len > history_max_len)
1871   {
1872     history_len = history_max_len;
1873   }
1874 
1875   return ( 1 );
1876 }
1877 
1878 /* Calculate length of the prompt string as seen from a terminal. */
1879 size_t
1880 pstrlen(const char *s)
     /* [previous][next][first][last][top][bottom][index][help] */
1881 {
1882   size_t len = 0, i = 0;
1883 
1884   while (s[i] != '\0')
1885   {
1886     if (s[i] == '\033') //-V536
1887     {
1888       i = strpbrk(s + i, "m") - s + 1; //-V769
1889       continue;
1890     }
1891 
1892     len++;
1893     i++;
1894   }
1895   return ( len );
1896 }
1897 
1898 #endif

/* [previous][next][first][last][top][bottom][index][help] */