1 This file is hash.def
, from which is created hash.c.
2 It implements the builtin
"hash" in Bush.
4 Copyright (C
) 1987-2020 Free Software Foundation
, Inc.
6 This file is part of GNU Bush
, the Bourne Again SHell.
8 Bush 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.
13 Bush 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.
18 You should have received a copy of the GNU General Public License
19 along with Bush. If not
, see
<http
://www.gnu.org
/licenses
/>.
24 $FUNCTION hash_builtin
25 $SHORT_DOC hash
[-lr
] [-p pathname
] [-dt
] [name ...
]
26 Remember or display program locations.
28 Determine and remember the full pathname of each command NAME. If
29 no arguments are given
, information about remembered commands is displayed.
32 -d forget the remembered location of each NAME
33 -l display in a format that may be reused as input
34 -p pathname use PATHNAME as the full pathname of NAME
35 -r forget all remembered locations
36 -t print the remembered location of each NAME
, preceding
37 each location with the corresponding NAME if multiple
40 NAME Each NAME is searched for in $PATH and added to the list
41 of remembered commands.
44 Returns success unless NAME is not found or an invalid option is given.
51 #include
"../src/bushtypes.h"
53 #if
defined (HAVE_UNISTD_H
)
59 #include
"../src/bushansi.h"
60 #include
"../src/bushintl.h"
62 #include
"../src/shell.h"
63 #include
"../src/builtins.h"
64 #include
"../src/runner/execute_cmd.h"
65 #include
"../src/flags.h"
66 #include
"../src/impl/findcmd.h"
67 #include
"../src/hashcmd.h"
69 #include
"bushgetopt.h"
71 extern int dot_found_in_search
;
73 static int add_hashed_command
PARAMS((char
*, int
));
74 static int print_hash_info
PARAMS((BUCKET_CONTENTS *)
);
75 static int print_portable_hash_info
PARAMS((BUCKET_CONTENTS *)
);
76 static int print_hashed_commands
PARAMS((int
));
77 static int list_hashed_filename_targets
PARAMS((WORD_LIST
*, int
));
79 /* Print statistics on the current state of hashed commands. If LIST is
80 not empty
, then
rehash (or hash in the first place
) the specified
86 int expunge_hash_table
, list_targets
, list_portably
, delete
, opt
;
89 if (hashing_enabled
== 0)
91 builtin_error (_("hashing disabled"));
92 return (EXECUTION_FAILURE
);
95 expunge_hash_table
= list_targets
= list_portably
= delete
= 0;
96 pathname
= (char *)NULL
;
97 reset_internal_getopt ();
98 while ((opt
= internal_getopt (list
, "dlp:rt")) != -1)
109 pathname
= list_optarg
;
112 expunge_hash_table
= 1;
125 /* hash
-t requires at least one argument.
*/
126 if (list
== 0 && (delete || list_targets
))
128 sh_needarg (delete ?
"-d" : "-t");
129 return (EXECUTION_FAILURE
);
132 /* We want hash
-r to be silent
, but hash
-- to print hashing info
, so
133 we test expunge_hash_table.
*/
134 if (list
== 0 && expunge_hash_table
== 0)
136 opt
= print_hashed_commands (list_portably
);
137 if (opt
== 0 && posixly_correct
== 0 &&
138 (list_portably
== 0 || shell_compatibility_level
<= 50))
139 printf (_("%s: hash table empty\n"), this_command_name
);
141 return (EXECUTION_SUCCESS
);
144 if (expunge_hash_table
)
147 /* If someone runs `hash
-r
-t xyz
' he will be disappointed. */
149 return (list_hashed_filename_targets (list, list_portably));
151 #if defined (RESTRICTED_SHELL)
152 if (restricted && pathname)
154 if (strchr (pathname, '/'))
156 sh_restricted (pathname);
157 return (EXECUTION_FAILURE);
159 /* If we are changing the hash table in a restricted shell, make sure the
160 target pathname can be found using a $PATH search. */
161 w = find_user_command (pathname);
162 if (w == 0 || *w == 0 || executable_file (w) == 0)
164 sh_notfound (pathname);
166 return (EXECUTION_FAILURE);
172 for (opt = EXECUTION_SUCCESS; list; list = list->next)
174 /* Add, remove or rehash the specified commands. */
175 w = list->word->word;
176 if (absolute_program (w))
180 if (is_directory (pathname))
183 builtin_error ("%s: %s", pathname, strerror (EISDIR));
185 builtin_error (_("%s: is a directory"), pathname);
187 opt = EXECUTION_FAILURE;
190 phash_insert (w, pathname, 0, 0);
194 if (phash_remove (w))
197 opt = EXECUTION_FAILURE;
200 else if (add_hashed_command (w, 0))
201 opt = EXECUTION_FAILURE;
209 add_hashed_command (w, quiet)
217 if (find_function (w) == 0 && find_shell_builtin (w) == 0)
220 full_path = find_user_command (w);
221 if (full_path && executable_file (full_path))
222 phash_insert (w, full_path, dot_found_in_search, 0);
234 /* Print information about current hashed info. */
236 print_hash_info (item)
237 BUCKET_CONTENTS *item;
239 printf ("%4d\t%s\n", item->times_found, pathdata(item)->path);
244 print_portable_hash_info (item)
245 BUCKET_CONTENTS *item;
249 fp = printable_filename (pathdata(item)->path, 1);
250 fn = printable_filename (item->key, 1);
251 printf ("builtin hash -p %s %s\n", fp, fn);
252 if (fp != pathdata(item)->path)
260 print_hashed_commands (fmt)
263 if (hashed_filenames == 0 || HASH_ENTRIES (hashed_filenames) == 0)
267 printf (_("hits\tcommand\n"));
268 hash_walk (hashed_filenames, fmt ? print_portable_hash_info : print_hash_info);
273 list_hashed_filename_targets (list, fmt)
277 int all_found, multiple;
282 multiple = list->next != 0;
284 for (l = list; l; l = l->next)
286 target = phash_search (l->word->word);
290 sh_notfound (l->word->word);
294 printf ("builtin hash -p %s %s\n", target, l->word->word);
298 printf ("%s\t", l->word->word);
299 printf ("%s\n", target);
304 return (all_found ? EXECUTION_SUCCESS : EXECUTION_FAILURE);