123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524 |
- BASH PATCH REPORT
- =================
- Bash-Release: 4.2
- Patch-ID: bash42-029
- Bug-Reported-by: "Michael Kalisz" <michael@kalisz.homelinux.net>
- Bug-Reference-ID: <50241.78.69.11.112.1298585641.squirrel@kalisz.homelinux.net>
- Bug-Reference-URL: http://lists.gnu.org/archive/html/bug-bash/2011-02/msg00274.html
- Bug-Description:
- Bash-4.2 tries to leave completed directory names as the user typed them,
- without expanding them to a full pathname. One effect of this is that
- shell variables used in pathnames being completed (e.g., $HOME) are left
- unchanged, but the `$' is quoted by readline because it is a special
- character to the shell.
- This patch introduces two things:
- 1. A new shell option, `direxpand', which, if set, attempts to emulate the
- bash-4.1 behavior of expanding words to full pathnames during
- completion;
- 2. A set of heuristics that reduce the number of times special characters
- such as `$' are quoted when the directory name is not expanded.
- Patch (apply with `patch -p0'):
- diff -NrC 2 ../bash-4.2-patched/bashline.c ./bashline.c
- *** ../bash-4.2-patched/bashline.c 2011-01-16 15:32:47.000000000 -0500
- --- ./bashline.c 2012-05-07 16:27:18.000000000 -0400
- ***************
- *** 122,125 ****
- --- 122,128 ----
- static int bash_push_line __P((void));
-
- + static rl_icppfunc_t *save_directory_hook __P((void));
- + static void reset_directory_hook __P((rl_icppfunc_t *));
- +
- static void cleanup_expansion_error __P((void));
- static void maybe_make_readline_line __P((char *));
- ***************
- *** 244,251 ****
- --- 247,261 ----
- int dircomplete_spelling = 0;
-
- + /* Expand directory names during word/filename completion. */
- + int dircomplete_expand = 0;
- + int dircomplete_expand_relpath = 0;
- +
- static char *bash_completer_word_break_characters = " \t\n\"'@><=;|&(:";
- static char *bash_nohostname_word_break_characters = " \t\n\"'><=;|&(:";
- /* )) */
-
- + static const char *default_filename_quote_characters = " \t\n\\\"'@<>=;|&()#$`?*[!:{~"; /*}*/
- + static char *custom_filename_quote_characters = 0;
- +
- static rl_hook_func_t *old_rl_startup_hook = (rl_hook_func_t *)NULL;
-
- ***************
- *** 502,506 ****
- /* Tell the completer that we might want to follow symbolic links or
- do other expansion on directory names. */
- ! rl_directory_rewrite_hook = bash_directory_completion_hook;
-
- rl_filename_rewrite_hook = bash_filename_rewrite_hook;
- --- 512,516 ----
- /* Tell the completer that we might want to follow symbolic links or
- do other expansion on directory names. */
- ! set_directory_hook ();
-
- rl_filename_rewrite_hook = bash_filename_rewrite_hook;
- ***************
- *** 530,534 ****
-
- /* characters that need to be quoted when appearing in filenames. */
- ! rl_filename_quote_characters = " \t\n\\\"'@<>=;|&()#$`?*[!:{~"; /*}*/
-
- rl_filename_quoting_function = bash_quote_filename;
- --- 540,544 ----
-
- /* characters that need to be quoted when appearing in filenames. */
- ! rl_filename_quote_characters = default_filename_quote_characters;
-
- rl_filename_quoting_function = bash_quote_filename;
- ***************
- *** 565,570 ****
- rl_attempted_completion_function = attempt_shell_completion;
- rl_completion_entry_function = NULL;
- - rl_directory_rewrite_hook = bash_directory_completion_hook;
- rl_ignore_some_completions_function = filename_completion_ignore;
- }
-
- --- 575,582 ----
- rl_attempted_completion_function = attempt_shell_completion;
- rl_completion_entry_function = NULL;
- rl_ignore_some_completions_function = filename_completion_ignore;
- + rl_filename_quote_characters = default_filename_quote_characters;
- +
- + set_directory_hook ();
- }
-
- ***************
- *** 1280,1283 ****
- --- 1292,1298 ----
- rl_ignore_some_completions_function = filename_completion_ignore;
-
- + rl_filename_quote_characters = default_filename_quote_characters;
- + set_directory_hook ();
- +
- /* Determine if this could be a command word. It is if it appears at
- the start of the line (ignoring preceding whitespace), or if it
- ***************
- *** 1592,1595 ****
- --- 1607,1616 ----
- else
- {
- + if (dircomplete_expand && dot_or_dotdot (filename_hint))
- + {
- + dircomplete_expand = 0;
- + set_directory_hook ();
- + dircomplete_expand = 1;
- + }
- mapping_over = 4;
- goto inner;
- ***************
- *** 1792,1795 ****
- --- 1813,1819 ----
- inner:
- val = rl_filename_completion_function (filename_hint, istate);
- + if (mapping_over == 4 && dircomplete_expand)
- + set_directory_hook ();
- +
- istate = 1;
-
- ***************
- *** 2694,2697 ****
- --- 2718,2767 ----
- }
-
- + /* Functions to save and restore the appropriate directory hook */
- + /* This is not static so the shopt code can call it */
- + void
- + set_directory_hook ()
- + {
- + if (dircomplete_expand)
- + {
- + rl_directory_completion_hook = bash_directory_completion_hook;
- + rl_directory_rewrite_hook = (rl_icppfunc_t *)0;
- + }
- + else
- + {
- + rl_directory_rewrite_hook = bash_directory_completion_hook;
- + rl_directory_completion_hook = (rl_icppfunc_t *)0;
- + }
- + }
- +
- + static rl_icppfunc_t *
- + save_directory_hook ()
- + {
- + rl_icppfunc_t *ret;
- +
- + if (dircomplete_expand)
- + {
- + ret = rl_directory_completion_hook;
- + rl_directory_completion_hook = (rl_icppfunc_t *)NULL;
- + }
- + else
- + {
- + ret = rl_directory_rewrite_hook;
- + rl_directory_rewrite_hook = (rl_icppfunc_t *)NULL;
- + }
- +
- + return ret;
- + }
- +
- + static void
- + restore_directory_hook (hookf)
- + rl_icppfunc_t *hookf;
- + {
- + if (dircomplete_expand)
- + rl_directory_completion_hook = hookf;
- + else
- + rl_directory_rewrite_hook = hookf;
- + }
- +
- /* Handle symbolic link references and other directory name
- expansions while hacking completion. This should return 1 if it modifies
- ***************
- *** 2703,2720 ****
- {
- char *local_dirname, *new_dirname, *t;
- ! int return_value, should_expand_dirname;
- WORD_LIST *wl;
- struct stat sb;
-
- ! return_value = should_expand_dirname = 0;
- local_dirname = *dirname;
-
- ! if (mbschr (local_dirname, '$'))
- ! should_expand_dirname = 1;
- else
- {
- t = mbschr (local_dirname, '`');
- if (t && unclosed_pair (local_dirname, strlen (local_dirname), "`") == 0)
- ! should_expand_dirname = 1;
- }
-
- --- 2773,2801 ----
- {
- char *local_dirname, *new_dirname, *t;
- ! int return_value, should_expand_dirname, nextch, closer;
- WORD_LIST *wl;
- struct stat sb;
-
- ! return_value = should_expand_dirname = nextch = closer = 0;
- local_dirname = *dirname;
-
- ! if (t = mbschr (local_dirname, '$'))
- ! {
- ! should_expand_dirname = '$';
- ! nextch = t[1];
- ! /* Deliberately does not handle the deprecated $[...] arithmetic
- ! expansion syntax */
- ! if (nextch == '(')
- ! closer = ')';
- ! else if (nextch == '{')
- ! closer = '}';
- ! else
- ! nextch = 0;
- ! }
- else
- {
- t = mbschr (local_dirname, '`');
- if (t && unclosed_pair (local_dirname, strlen (local_dirname), "`") == 0)
- ! should_expand_dirname = '`';
- }
-
- ***************
- *** 2740,2743 ****
- --- 2821,2841 ----
- dispose_words (wl);
- local_dirname = *dirname;
- + /* XXX - change rl_filename_quote_characters here based on
- + should_expand_dirname/nextch/closer. This is the only place
- + custom_filename_quote_characters is modified. */
- + if (rl_filename_quote_characters && *rl_filename_quote_characters)
- + {
- + int i, j, c;
- + i = strlen (default_filename_quote_characters);
- + custom_filename_quote_characters = xrealloc (custom_filename_quote_characters, i+1);
- + for (i = j = 0; c = default_filename_quote_characters[i]; i++)
- + {
- + if (c == should_expand_dirname || c == nextch || c == closer)
- + continue;
- + custom_filename_quote_characters[j++] = c;
- + }
- + custom_filename_quote_characters[j] = '\0';
- + rl_filename_quote_characters = custom_filename_quote_characters;
- + }
- }
- else
- ***************
- *** 2759,2762 ****
- --- 2857,2871 ----
- }
-
- + /* no_symbolic_links == 0 -> use (default) logical view of the file system.
- + local_dirname[0] == '.' && local_dirname[1] == '/' means files in the
- + current directory (./).
- + local_dirname[0] == '.' && local_dirname[1] == 0 means relative pathnames
- + in the current directory (e.g., lib/sh).
- + XXX - should we do spelling correction on these? */
- +
- + /* This is test as it was in bash-4.2: skip relative pathnames in current
- + directory. Change test to
- + (local_dirname[0] != '.' || (local_dirname[1] && local_dirname[1] != '/'))
- + if we want to skip paths beginning with ./ also. */
- if (no_symbolic_links == 0 && (local_dirname[0] != '.' || local_dirname[1]))
- {
- ***************
- *** 2764,2767 ****
- --- 2873,2885 ----
- int len1, len2;
-
- + /* If we have a relative path
- + (local_dirname[0] != '/' && local_dirname[0] != '.')
- + that is canonical after appending it to the current directory, then
- + temp1 = temp2+'/'
- + That is,
- + strcmp (temp1, temp2) == 0
- + after adding a slash to temp2 below. It should be safe to not
- + change those.
- + */
- t = get_working_directory ("symlink-hook");
- temp1 = make_absolute (local_dirname, t);
- ***************
- *** 2798,2802 ****
- }
- }
- ! return_value |= STREQ (local_dirname, temp2) == 0;
- free (local_dirname);
- *dirname = temp2;
- --- 2916,2928 ----
- }
- }
- !
- ! /* dircomplete_expand_relpath == 0 means we want to leave relative
- ! pathnames that are unchanged by canonicalization alone.
- ! *local_dirname != '/' && *local_dirname != '.' == relative pathname
- ! (consistent with general.c:absolute_pathname())
- ! temp1 == temp2 (after appending a slash to temp2) means the pathname
- ! is not changed by canonicalization as described above. */
- ! if (dircomplete_expand_relpath || ((local_dirname[0] != '/' && local_dirname[0] != '.') && STREQ (temp1, temp2) == 0))
- ! return_value |= STREQ (local_dirname, temp2) == 0;
- free (local_dirname);
- *dirname = temp2;
- ***************
- *** 3003,3012 ****
- orig_func = rl_completion_entry_function;
- orig_attempt_func = rl_attempted_completion_function;
- - orig_dir_func = rl_directory_rewrite_hook;
- orig_ignore_func = rl_ignore_some_completions_function;
- orig_rl_completer_word_break_characters = rl_completer_word_break_characters;
- rl_completion_entry_function = rl_filename_completion_function;
- rl_attempted_completion_function = (rl_completion_func_t *)NULL;
- - rl_directory_rewrite_hook = (rl_icppfunc_t *)NULL;
- rl_ignore_some_completions_function = filename_completion_ignore;
- rl_completer_word_break_characters = " \t\n\"\'";
- --- 3129,3139 ----
- orig_func = rl_completion_entry_function;
- orig_attempt_func = rl_attempted_completion_function;
- orig_ignore_func = rl_ignore_some_completions_function;
- orig_rl_completer_word_break_characters = rl_completer_word_break_characters;
- +
- + orig_dir_func = save_directory_hook ();
- +
- rl_completion_entry_function = rl_filename_completion_function;
- rl_attempted_completion_function = (rl_completion_func_t *)NULL;
- rl_ignore_some_completions_function = filename_completion_ignore;
- rl_completer_word_break_characters = " \t\n\"\'";
- ***************
- *** 3016,3023 ****
- rl_completion_entry_function = orig_func;
- rl_attempted_completion_function = orig_attempt_func;
- - rl_directory_rewrite_hook = orig_dir_func;
- rl_ignore_some_completions_function = orig_ignore_func;
- rl_completer_word_break_characters = orig_rl_completer_word_break_characters;
-
- return r;
- }
- --- 3143,3151 ----
- rl_completion_entry_function = orig_func;
- rl_attempted_completion_function = orig_attempt_func;
- rl_ignore_some_completions_function = orig_ignore_func;
- rl_completer_word_break_characters = orig_rl_completer_word_break_characters;
-
- + restore_directory_hook (orig_dir_func);
- +
- return r;
- }
- diff -NrC 2 ../bash-4.2-patched/bashline.h ./bashline.h
- *** ../bash-4.2-patched/bashline.h 2009-01-04 14:32:22.000000000 -0500
- --- ./bashline.h 2012-05-07 16:27:18.000000000 -0400
- ***************
- *** 34,41 ****
- --- 34,46 ----
- extern int bash_re_edit __P((char *));
-
- + extern void bashline_set_event_hook __P((void));
- + extern void bashline_reset_event_hook __P((void));
- +
- extern int bind_keyseq_to_unix_command __P((char *));
-
- extern char **bash_default_completion __P((const char *, int, int, int, int));
-
- + void set_directory_hook __P((void));
- +
- /* Used by programmable completion code. */
- extern char *command_word_completion_function __P((const char *, int));
- diff -NrC 2 ../bash-4.2-patched/builtins/shopt.def ./builtins/shopt.def
- *** ../bash-4.2-patched/builtins/shopt.def 2010-07-02 22:42:44.000000000 -0400
- --- ./builtins/shopt.def 2012-05-07 16:27:18.000000000 -0400
- ***************
- *** 62,65 ****
- --- 62,69 ----
- #include "bashgetopt.h"
-
- + #if defined (READLINE)
- + # include "../bashline.h"
- + #endif
- +
- #if defined (HISTORY)
- # include "../bashhist.h"
- ***************
- *** 95,99 ****
- extern int no_empty_command_completion;
- extern int force_fignore;
- ! extern int dircomplete_spelling;
-
- extern int enable_hostname_completion __P((int));
- --- 99,103 ----
- extern int no_empty_command_completion;
- extern int force_fignore;
- ! extern int dircomplete_spelling, dircomplete_expand;
-
- extern int enable_hostname_completion __P((int));
- ***************
- *** 122,125 ****
- --- 126,133 ----
- #endif
-
- + #if defined (READLINE)
- + static int shopt_set_complete_direxpand __P((char *, int));
- + #endif
- +
- static int shopt_login_shell;
- static int shopt_compat31;
- ***************
- *** 151,154 ****
- --- 159,163 ----
- { "compat41", &shopt_compat41, set_compatibility_level },
- #if defined (READLINE)
- + { "direxpand", &dircomplete_expand, shopt_set_complete_direxpand },
- { "dirspell", &dircomplete_spelling, (shopt_set_func_t *)NULL },
- #endif
- ***************
- *** 536,539 ****
- --- 545,559 ----
- }
-
- + #if defined (READLINE)
- + static int
- + shopt_set_complete_direxpand (option_name, mode)
- + char *option_name;
- + int mode;
- + {
- + set_directory_hook ();
- + return 0;
- + }
- + #endif
- +
- #if defined (RESTRICTED_SHELL)
- /* Don't allow the value of restricted_shell to be modified. */
- Binary files ../bash-4.2-patched/doc/._bashref.pdf and ./doc/._bashref.pdf differ
- diff -NrC 2 ../bash-4.2-patched/doc/bash.1 ./doc/bash.1
- *** ../bash-4.2-patched/doc/bash.1 2011-01-16 15:31:39.000000000 -0500
- --- ./doc/bash.1 2012-05-07 16:27:18.000000000 -0400
- ***************
- *** 8949,8952 ****
- --- 8949,8962 ----
- The default bash behavior remains as in previous versions.
- .TP 8
- + .B direxpand
- + If set,
- + .B bash
- + replaces directory names with the results of word expansion when performing
- + filename completion. This changes the contents of the readline editing
- + buffer.
- + If not set,
- + .B bash
- + attempts to preserve what the user typed.
- + .TP 8
- .B dirspell
- If set,
- diff -NrC 2 ../bash-4.2-patched/doc/bashref.texi ./doc/bashref.texi
- *** ../bash-4.2-patched/doc/bashref.texi 2011-01-16 15:31:57.000000000 -0500
- --- ./doc/bashref.texi 2012-05-07 16:27:18.000000000 -0400
- ***************
- *** 4536,4539 ****
- --- 4536,4546 ----
- The default Bash behavior remains as in previous versions.
-
- + @item direxpand
- + If set, Bash
- + replaces directory names with the results of word expansion when performing
- + filename completion. This changes the contents of the readline editing
- + buffer.
- + If not set, Bash attempts to preserve what the user typed.
- +
- @item dirspell
- If set, Bash
- diff -NrC 2 ../bash-4.2-patched/tests/shopt.right ./tests/shopt.right
- *** ../bash-4.2-patched/tests/shopt.right 2010-07-02 23:36:30.000000000 -0400
- --- ./tests/shopt.right 2012-05-07 16:27:18.000000000 -0400
- ***************
- *** 13,16 ****
- --- 13,17 ----
- shopt -u compat40
- shopt -u compat41
- + shopt -u direxpand
- shopt -u dirspell
- shopt -u dotglob
- ***************
- *** 69,72 ****
- --- 70,74 ----
- shopt -u compat40
- shopt -u compat41
- + shopt -u direxpand
- shopt -u dirspell
- shopt -u dotglob
- ***************
- *** 102,105 ****
- --- 104,108 ----
- compat40 off
- compat41 off
- + direxpand off
- dirspell off
- dotglob off
- *** ../bash-4.2-patched/patchlevel.h Sat Jun 12 20:14:48 2010
- --- ./patchlevel.h Thu Feb 24 21:41:34 2011
- ***************
- *** 26,30 ****
- looks for to find the patch level (for the sccs version string). */
-
- ! #define PATCHLEVEL 28
-
- #endif /* _PATCHLEVEL_H_ */
- --- 26,30 ----
- looks for to find the patch level (for the sccs version string). */
-
- ! #define PATCHLEVEL 29
-
- #endif /* _PATCHLEVEL_H_ */
|