Annotation of embedaddon/readline/search.c, revision 1.1
1.1 ! misho 1: /* search.c - code for non-incremental searching in emacs and vi modes. */
! 2:
! 3: /* Copyright (C) 1992-2013 Free Software Foundation, Inc.
! 4:
! 5: This file is part of the GNU Readline Library (Readline), a library
! 6: for reading lines of text with interactive input and history editing.
! 7:
! 8: Readline is free software: you can redistribute it and/or modify
! 9: it under the terms of the GNU General Public License as published by
! 10: the Free Software Foundation, either version 3 of the License, or
! 11: (at your option) any later version.
! 12:
! 13: Readline is distributed in the hope that it will be useful,
! 14: but WITHOUT ANY WARRANTY; without even the implied warranty of
! 15: MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
! 16: GNU General Public License for more details.
! 17:
! 18: You should have received a copy of the GNU General Public License
! 19: along with Readline. If not, see <http://www.gnu.org/licenses/>.
! 20: */
! 21:
! 22: #define READLINE_LIBRARY
! 23:
! 24: #if defined (HAVE_CONFIG_H)
! 25: # include <config.h>
! 26: #endif
! 27:
! 28: #include <sys/types.h>
! 29: #include <stdio.h>
! 30:
! 31: #if defined (HAVE_UNISTD_H)
! 32: # include <unistd.h>
! 33: #endif
! 34:
! 35: #if defined (HAVE_STDLIB_H)
! 36: # include <stdlib.h>
! 37: #else
! 38: # include "ansi_stdlib.h"
! 39: #endif
! 40:
! 41: #include "rldefs.h"
! 42: #include "rlmbutil.h"
! 43:
! 44: #include "readline.h"
! 45: #include "history.h"
! 46: #include "histlib.h"
! 47:
! 48: #include "rlprivate.h"
! 49: #include "xmalloc.h"
! 50:
! 51: #ifdef abs
! 52: # undef abs
! 53: #endif
! 54: #define abs(x) (((x) >= 0) ? (x) : -(x))
! 55:
! 56: _rl_search_cxt *_rl_nscxt = 0;
! 57:
! 58: extern HIST_ENTRY *_rl_saved_line_for_history;
! 59:
! 60: /* Functions imported from the rest of the library. */
! 61: extern int _rl_free_history_entry PARAMS((HIST_ENTRY *));
! 62:
! 63: static char *noninc_search_string = (char *) NULL;
! 64: static int noninc_history_pos;
! 65:
! 66: static char *prev_line_found = (char *) NULL;
! 67:
! 68: static int rl_history_search_len;
! 69: static int rl_history_search_pos;
! 70: static int rl_history_search_flags;
! 71:
! 72: static char *history_search_string;
! 73: static int history_string_size;
! 74:
! 75: static void make_history_line_current PARAMS((HIST_ENTRY *));
! 76: static int noninc_search_from_pos PARAMS((char *, int, int));
! 77: static int noninc_dosearch PARAMS((char *, int));
! 78: static int noninc_search PARAMS((int, int));
! 79: static int rl_history_search_internal PARAMS((int, int));
! 80: static void rl_history_search_reinit PARAMS((int));
! 81:
! 82: static _rl_search_cxt *_rl_nsearch_init PARAMS((int, int));
! 83: static int _rl_nsearch_cleanup PARAMS((_rl_search_cxt *, int));
! 84: static void _rl_nsearch_abort PARAMS((_rl_search_cxt *));
! 85: static int _rl_nsearch_dispatch PARAMS((_rl_search_cxt *, int));
! 86:
! 87: /* Make the data from the history entry ENTRY be the contents of the
! 88: current line. This doesn't do anything with rl_point; the caller
! 89: must set it. */
! 90: static void
! 91: make_history_line_current (entry)
! 92: HIST_ENTRY *entry;
! 93: {
! 94: _rl_replace_text (entry->line, 0, rl_end);
! 95: _rl_fix_point (1);
! 96: #if defined (VI_MODE)
! 97: if (rl_editing_mode == vi_mode)
! 98: /* POSIX.2 says that the `U' command doesn't affect the copy of any
! 99: command lines to the edit line. We're going to implement that by
! 100: making the undo list start after the matching line is copied to the
! 101: current editing buffer. */
! 102: rl_free_undo_list ();
! 103: #endif
! 104:
! 105: if (_rl_saved_line_for_history)
! 106: _rl_free_history_entry (_rl_saved_line_for_history);
! 107: _rl_saved_line_for_history = (HIST_ENTRY *)NULL;
! 108: }
! 109:
! 110: /* Search the history list for STRING starting at absolute history position
! 111: POS. If STRING begins with `^', the search must match STRING at the
! 112: beginning of a history line, otherwise a full substring match is performed
! 113: for STRING. DIR < 0 means to search backwards through the history list,
! 114: DIR >= 0 means to search forward. */
! 115: static int
! 116: noninc_search_from_pos (string, pos, dir)
! 117: char *string;
! 118: int pos, dir;
! 119: {
! 120: int ret, old;
! 121:
! 122: if (pos < 0)
! 123: return -1;
! 124:
! 125: old = where_history ();
! 126: if (history_set_pos (pos) == 0)
! 127: return -1;
! 128:
! 129: RL_SETSTATE(RL_STATE_SEARCH);
! 130: if (*string == '^')
! 131: ret = history_search_prefix (string + 1, dir);
! 132: else
! 133: ret = history_search (string, dir);
! 134: RL_UNSETSTATE(RL_STATE_SEARCH);
! 135:
! 136: if (ret != -1)
! 137: ret = where_history ();
! 138:
! 139: history_set_pos (old);
! 140: return (ret);
! 141: }
! 142:
! 143: /* Search for a line in the history containing STRING. If DIR is < 0, the
! 144: search is backwards through previous entries, else through subsequent
! 145: entries. Returns 1 if the search was successful, 0 otherwise. */
! 146: static int
! 147: noninc_dosearch (string, dir)
! 148: char *string;
! 149: int dir;
! 150: {
! 151: int oldpos, pos;
! 152: HIST_ENTRY *entry;
! 153:
! 154: if (string == 0 || *string == '\0' || noninc_history_pos < 0)
! 155: {
! 156: rl_ding ();
! 157: return 0;
! 158: }
! 159:
! 160: pos = noninc_search_from_pos (string, noninc_history_pos + dir, dir);
! 161: if (pos == -1)
! 162: {
! 163: /* Search failed, current history position unchanged. */
! 164: rl_maybe_unsave_line ();
! 165: rl_clear_message ();
! 166: rl_point = 0;
! 167: rl_ding ();
! 168: return 0;
! 169: }
! 170:
! 171: noninc_history_pos = pos;
! 172:
! 173: oldpos = where_history ();
! 174: history_set_pos (noninc_history_pos);
! 175: entry = current_history (); /* will never be NULL after successful search */
! 176:
! 177: #if defined (VI_MODE)
! 178: if (rl_editing_mode != vi_mode)
! 179: #endif
! 180: history_set_pos (oldpos);
! 181:
! 182: make_history_line_current (entry);
! 183:
! 184: rl_point = 0;
! 185: rl_mark = rl_end;
! 186:
! 187: rl_clear_message ();
! 188: return 1;
! 189: }
! 190:
! 191: static _rl_search_cxt *
! 192: _rl_nsearch_init (dir, pchar)
! 193: int dir, pchar;
! 194: {
! 195: _rl_search_cxt *cxt;
! 196: char *p;
! 197:
! 198: cxt = _rl_scxt_alloc (RL_SEARCH_NSEARCH, 0);
! 199: if (dir < 0)
! 200: cxt->sflags |= SF_REVERSE; /* not strictly needed */
! 201:
! 202: cxt->direction = dir;
! 203: cxt->history_pos = cxt->save_line;
! 204:
! 205: rl_maybe_save_line ();
! 206:
! 207: /* Clear the undo list, since reading the search string should create its
! 208: own undo list, and the whole list will end up being freed when we
! 209: finish reading the search string. */
! 210: rl_undo_list = 0;
! 211:
! 212: /* Use the line buffer to read the search string. */
! 213: rl_line_buffer[0] = 0;
! 214: rl_end = rl_point = 0;
! 215:
! 216: p = _rl_make_prompt_for_search (pchar ? pchar : ':');
! 217: rl_message ("%s", p);
! 218: xfree (p);
! 219:
! 220: RL_SETSTATE(RL_STATE_NSEARCH);
! 221:
! 222: _rl_nscxt = cxt;
! 223:
! 224: return cxt;
! 225: }
! 226:
! 227: static int
! 228: _rl_nsearch_cleanup (cxt, r)
! 229: _rl_search_cxt *cxt;
! 230: int r;
! 231: {
! 232: _rl_scxt_dispose (cxt, 0);
! 233: _rl_nscxt = 0;
! 234:
! 235: RL_UNSETSTATE(RL_STATE_NSEARCH);
! 236:
! 237: return (r != 1);
! 238: }
! 239:
! 240: static void
! 241: _rl_nsearch_abort (cxt)
! 242: _rl_search_cxt *cxt;
! 243: {
! 244: rl_maybe_unsave_line ();
! 245: rl_clear_message ();
! 246: rl_point = cxt->save_point;
! 247: rl_mark = cxt->save_mark;
! 248: rl_restore_prompt ();
! 249:
! 250: RL_UNSETSTATE (RL_STATE_NSEARCH);
! 251: }
! 252:
! 253: /* Process just-read character C according to search context CXT. Return -1
! 254: if the caller should abort the search, 0 if we should break out of the
! 255: loop, and 1 if we should continue to read characters. */
! 256: static int
! 257: _rl_nsearch_dispatch (cxt, c)
! 258: _rl_search_cxt *cxt;
! 259: int c;
! 260: {
! 261: switch (c)
! 262: {
! 263: case CTRL('W'):
! 264: rl_unix_word_rubout (1, c);
! 265: break;
! 266:
! 267: case CTRL('U'):
! 268: rl_unix_line_discard (1, c);
! 269: break;
! 270:
! 271: case RETURN:
! 272: case NEWLINE:
! 273: return 0;
! 274:
! 275: case CTRL('H'):
! 276: case RUBOUT:
! 277: if (rl_point == 0)
! 278: {
! 279: _rl_nsearch_abort (cxt);
! 280: return -1;
! 281: }
! 282: _rl_rubout_char (1, c);
! 283: break;
! 284:
! 285: case CTRL('C'):
! 286: case CTRL('G'):
! 287: rl_ding ();
! 288: _rl_nsearch_abort (cxt);
! 289: return -1;
! 290:
! 291: default:
! 292: #if defined (HANDLE_MULTIBYTE)
! 293: if (MB_CUR_MAX > 1 && rl_byte_oriented == 0)
! 294: rl_insert_text (cxt->mb);
! 295: else
! 296: #endif
! 297: _rl_insert_char (1, c);
! 298: break;
! 299: }
! 300:
! 301: (*rl_redisplay_function) ();
! 302: return 1;
! 303: }
! 304:
! 305: /* Perform one search according to CXT, using NONINC_SEARCH_STRING. Return
! 306: -1 if the search should be aborted, any other value means to clean up
! 307: using _rl_nsearch_cleanup (). Returns 1 if the search was successful,
! 308: 0 otherwise. */
! 309: static int
! 310: _rl_nsearch_dosearch (cxt)
! 311: _rl_search_cxt *cxt;
! 312: {
! 313: rl_mark = cxt->save_mark;
! 314:
! 315: /* If rl_point == 0, we want to re-use the previous search string and
! 316: start from the saved history position. If there's no previous search
! 317: string, punt. */
! 318: if (rl_point == 0)
! 319: {
! 320: if (noninc_search_string == 0)
! 321: {
! 322: rl_ding ();
! 323: rl_restore_prompt ();
! 324: RL_UNSETSTATE (RL_STATE_NSEARCH);
! 325: return -1;
! 326: }
! 327: }
! 328: else
! 329: {
! 330: /* We want to start the search from the current history position. */
! 331: noninc_history_pos = cxt->save_line;
! 332: FREE (noninc_search_string);
! 333: noninc_search_string = savestring (rl_line_buffer);
! 334:
! 335: /* If we don't want the subsequent undo list generated by the search
! 336: matching a history line to include the contents of the search string,
! 337: we need to clear rl_line_buffer here. For now, we just clear the
! 338: undo list generated by reading the search string. (If the search
! 339: fails, the old undo list will be restored by rl_maybe_unsave_line.) */
! 340: rl_free_undo_list ();
! 341: }
! 342:
! 343: rl_restore_prompt ();
! 344: return (noninc_dosearch (noninc_search_string, cxt->direction));
! 345: }
! 346:
! 347: /* Search non-interactively through the history list. DIR < 0 means to
! 348: search backwards through the history of previous commands; otherwise
! 349: the search is for commands subsequent to the current position in the
! 350: history list. PCHAR is the character to use for prompting when reading
! 351: the search string; if not specified (0), it defaults to `:'. */
! 352: static int
! 353: noninc_search (dir, pchar)
! 354: int dir;
! 355: int pchar;
! 356: {
! 357: _rl_search_cxt *cxt;
! 358: int c, r;
! 359:
! 360: cxt = _rl_nsearch_init (dir, pchar);
! 361:
! 362: if (RL_ISSTATE (RL_STATE_CALLBACK))
! 363: return (0);
! 364:
! 365: /* Read the search string. */
! 366: r = 0;
! 367: while (1)
! 368: {
! 369: c = _rl_search_getchar (cxt);
! 370:
! 371: if (c == 0)
! 372: break;
! 373:
! 374: r = _rl_nsearch_dispatch (cxt, c);
! 375: if (r < 0)
! 376: return 1;
! 377: else if (r == 0)
! 378: break;
! 379: }
! 380:
! 381: r = _rl_nsearch_dosearch (cxt);
! 382: return ((r >= 0) ? _rl_nsearch_cleanup (cxt, r) : (r != 1));
! 383: }
! 384:
! 385: /* Search forward through the history list for a string. If the vi-mode
! 386: code calls this, KEY will be `?'. */
! 387: int
! 388: rl_noninc_forward_search (count, key)
! 389: int count, key;
! 390: {
! 391: return noninc_search (1, (key == '?') ? '?' : 0);
! 392: }
! 393:
! 394: /* Reverse search the history list for a string. If the vi-mode code
! 395: calls this, KEY will be `/'. */
! 396: int
! 397: rl_noninc_reverse_search (count, key)
! 398: int count, key;
! 399: {
! 400: return noninc_search (-1, (key == '/') ? '/' : 0);
! 401: }
! 402:
! 403: /* Search forward through the history list for the last string searched
! 404: for. If there is no saved search string, abort. */
! 405: int
! 406: rl_noninc_forward_search_again (count, key)
! 407: int count, key;
! 408: {
! 409: int r;
! 410:
! 411: if (!noninc_search_string)
! 412: {
! 413: rl_ding ();
! 414: return (-1);
! 415: }
! 416: r = noninc_dosearch (noninc_search_string, 1);
! 417: return (r != 1);
! 418: }
! 419:
! 420: /* Reverse search in the history list for the last string searched
! 421: for. If there is no saved search string, abort. */
! 422: int
! 423: rl_noninc_reverse_search_again (count, key)
! 424: int count, key;
! 425: {
! 426: int r;
! 427:
! 428: if (!noninc_search_string)
! 429: {
! 430: rl_ding ();
! 431: return (-1);
! 432: }
! 433: r = noninc_dosearch (noninc_search_string, -1);
! 434: return (r != 1);
! 435: }
! 436:
! 437: #if defined (READLINE_CALLBACKS)
! 438: int
! 439: _rl_nsearch_callback (cxt)
! 440: _rl_search_cxt *cxt;
! 441: {
! 442: int c, r;
! 443:
! 444: c = _rl_search_getchar (cxt);
! 445: r = _rl_nsearch_dispatch (cxt, c);
! 446: if (r != 0)
! 447: return 1;
! 448:
! 449: r = _rl_nsearch_dosearch (cxt);
! 450: return ((r >= 0) ? _rl_nsearch_cleanup (cxt, r) : (r != 1));
! 451: }
! 452: #endif
! 453:
! 454: static int
! 455: rl_history_search_internal (count, dir)
! 456: int count, dir;
! 457: {
! 458: HIST_ENTRY *temp;
! 459: int ret, oldpos;
! 460: char *t;
! 461:
! 462: rl_maybe_save_line ();
! 463: temp = (HIST_ENTRY *)NULL;
! 464:
! 465: /* Search COUNT times through the history for a line matching
! 466: history_search_string. If history_search_string[0] == '^', the
! 467: line must match from the start; otherwise any substring can match.
! 468: When this loop finishes, TEMP, if non-null, is the history line to
! 469: copy into the line buffer. */
! 470: while (count)
! 471: {
! 472: RL_CHECK_SIGNALS ();
! 473: ret = noninc_search_from_pos (history_search_string, rl_history_search_pos + dir, dir);
! 474: if (ret == -1)
! 475: break;
! 476:
! 477: /* Get the history entry we found. */
! 478: rl_history_search_pos = ret;
! 479: oldpos = where_history ();
! 480: history_set_pos (rl_history_search_pos);
! 481: temp = current_history (); /* will never be NULL after successful search */
! 482: history_set_pos (oldpos);
! 483:
! 484: /* Don't find multiple instances of the same line. */
! 485: if (prev_line_found && STREQ (prev_line_found, temp->line))
! 486: continue;
! 487: prev_line_found = temp->line;
! 488: count--;
! 489: }
! 490:
! 491: /* If we didn't find anything at all, return. */
! 492: if (temp == 0)
! 493: {
! 494: rl_maybe_unsave_line ();
! 495: rl_ding ();
! 496: /* If you don't want the saved history line (last match) to show up
! 497: in the line buffer after the search fails, change the #if 0 to
! 498: #if 1 */
! 499: #if 0
! 500: if (rl_point > rl_history_search_len)
! 501: {
! 502: rl_point = rl_end = rl_history_search_len;
! 503: rl_line_buffer[rl_end] = '\0';
! 504: rl_mark = 0;
! 505: }
! 506: #else
! 507: rl_point = rl_history_search_len; /* rl_maybe_unsave_line changes it */
! 508: rl_mark = rl_end;
! 509: #endif
! 510: return 1;
! 511: }
! 512:
! 513: /* Copy the line we found into the current line buffer. */
! 514: make_history_line_current (temp);
! 515:
! 516: if (rl_history_search_flags & ANCHORED_SEARCH)
! 517: rl_point = rl_history_search_len; /* easy case */
! 518: else
! 519: {
! 520: t = strstr (rl_line_buffer, history_search_string);
! 521: rl_point = t ? (int)(t - rl_line_buffer) + rl_history_search_len : rl_end;
! 522: }
! 523: rl_mark = rl_end;
! 524:
! 525: return 0;
! 526: }
! 527:
! 528: static void
! 529: rl_history_search_reinit (flags)
! 530: int flags;
! 531: {
! 532: int sind;
! 533:
! 534: rl_history_search_pos = where_history ();
! 535: rl_history_search_len = rl_point;
! 536: rl_history_search_flags = flags;
! 537:
! 538: prev_line_found = (char *)NULL;
! 539: if (rl_point)
! 540: {
! 541: /* Allocate enough space for anchored and non-anchored searches */
! 542: if (rl_history_search_len >= history_string_size - 2)
! 543: {
! 544: history_string_size = rl_history_search_len + 2;
! 545: history_search_string = (char *)xrealloc (history_search_string, history_string_size);
! 546: }
! 547: sind = 0;
! 548: if (flags & ANCHORED_SEARCH)
! 549: history_search_string[sind++] = '^';
! 550: strncpy (history_search_string + sind, rl_line_buffer, rl_point);
! 551: history_search_string[rl_point + sind] = '\0';
! 552: }
! 553: _rl_free_saved_history_line ();
! 554: }
! 555:
! 556: /* Search forward in the history for the string of characters
! 557: from the start of the line to rl_point. This is a non-incremental
! 558: search. The search is anchored to the beginning of the history line. */
! 559: int
! 560: rl_history_search_forward (count, ignore)
! 561: int count, ignore;
! 562: {
! 563: if (count == 0)
! 564: return (0);
! 565:
! 566: if (rl_last_func != rl_history_search_forward &&
! 567: rl_last_func != rl_history_search_backward)
! 568: rl_history_search_reinit (ANCHORED_SEARCH);
! 569:
! 570: if (rl_history_search_len == 0)
! 571: return (rl_get_next_history (count, ignore));
! 572: return (rl_history_search_internal (abs (count), (count > 0) ? 1 : -1));
! 573: }
! 574:
! 575: /* Search backward through the history for the string of characters
! 576: from the start of the line to rl_point. This is a non-incremental
! 577: search. */
! 578: int
! 579: rl_history_search_backward (count, ignore)
! 580: int count, ignore;
! 581: {
! 582: if (count == 0)
! 583: return (0);
! 584:
! 585: if (rl_last_func != rl_history_search_forward &&
! 586: rl_last_func != rl_history_search_backward)
! 587: rl_history_search_reinit (ANCHORED_SEARCH);
! 588:
! 589: if (rl_history_search_len == 0)
! 590: return (rl_get_previous_history (count, ignore));
! 591: return (rl_history_search_internal (abs (count), (count > 0) ? -1 : 1));
! 592: }
! 593:
! 594: /* Search forward in the history for the string of characters
! 595: from the start of the line to rl_point. This is a non-incremental
! 596: search. The search succeeds if the search string is present anywhere
! 597: in the history line. */
! 598: int
! 599: rl_history_substr_search_forward (count, ignore)
! 600: int count, ignore;
! 601: {
! 602: if (count == 0)
! 603: return (0);
! 604:
! 605: if (rl_last_func != rl_history_substr_search_forward &&
! 606: rl_last_func != rl_history_substr_search_backward)
! 607: rl_history_search_reinit (NON_ANCHORED_SEARCH);
! 608:
! 609: if (rl_history_search_len == 0)
! 610: return (rl_get_next_history (count, ignore));
! 611: return (rl_history_search_internal (abs (count), (count > 0) ? 1 : -1));
! 612: }
! 613:
! 614: /* Search backward through the history for the string of characters
! 615: from the start of the line to rl_point. This is a non-incremental
! 616: search. */
! 617: int
! 618: rl_history_substr_search_backward (count, ignore)
! 619: int count, ignore;
! 620: {
! 621: if (count == 0)
! 622: return (0);
! 623:
! 624: if (rl_last_func != rl_history_substr_search_forward &&
! 625: rl_last_func != rl_history_substr_search_backward)
! 626: rl_history_search_reinit (NON_ANCHORED_SEARCH);
! 627:
! 628: if (rl_history_search_len == 0)
! 629: return (rl_get_previous_history (count, ignore));
! 630: return (rl_history_search_internal (abs (count), (count > 0) ? -1 : 1));
! 631: }
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>