1 /* search.c - search devices based on a file or a filesystem label */
3 * GRUB -- GRand Unified Bootloader
4 * Copyright (C) 2005,2007,2008,2009 Free Software Foundation, Inc.
6 * GRUB is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
11 * GRUB is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with GRUB. If not, see <http://www.gnu.org/licenses/>.
20 #include <grub/types.h>
21 #include <grub/misc.h>
25 #include <grub/device.h>
26 #include <grub/file.h>
28 #include <grub/command.h>
29 #include <grub/search.h>
30 #include <grub/i18n.h>
31 #include <grub/disk.h>
32 #include <grub/partition.h>
34 GRUB_MOD_LICENSE ("GPLv3+");
38 struct cache_entry
*next
;
43 static struct cache_entry
*cache
;
46 FUNC_NAME (const char *key
, const char *var
, int no_floppy
,
47 char **hints
, unsigned nhints
)
51 grub_fs_autoload_hook_t saved_autoload
;
53 auto int iterate_device (const char *name
);
54 int iterate_device (const char *name
)
58 /* Skip floppy drives when requested. */
60 name
[0] == 'f' && name
[1] == 'd' && name
[2] >= '0' && name
[2] <= '9')
63 #ifdef DO_SEARCH_FS_UUID
64 #define compare_fn grub_strcasecmp
66 #define compare_fn grub_strcmp
74 buf
= grub_xasprintf ("(%s)%s", name
, key
);
78 grub_file_filter_disable_compression ();
79 file
= grub_file_open (buf
);
83 grub_file_close (file
);
89 /* SEARCH_FS_UUID or SEARCH_LABEL */
94 dev
= grub_device_open (name
);
97 fs
= grub_fs_probe (dev
);
99 #ifdef DO_SEARCH_FS_UUID
102 #define read_fn label
105 if (fs
&& fs
->read_fn
)
107 fs
->read_fn (dev
, &quid
);
109 if (grub_errno
== GRUB_ERR_NONE
&& quid
)
111 if (compare_fn (quid
, key
) == 0)
118 grub_device_close (dev
);
123 if (!is_cache
&& found
&& count
== 0)
125 struct cache_entry
*cache_ent
;
126 cache_ent
= grub_malloc (sizeof (*cache_ent
));
129 cache_ent
->key
= grub_strdup (key
);
130 cache_ent
->value
= grub_strdup (name
);
131 if (cache_ent
->value
&& cache_ent
->key
)
133 cache_ent
->next
= cache
;
138 grub_free (cache_ent
->value
);
139 grub_free (cache_ent
->key
);
140 grub_free (cache_ent
);
141 grub_errno
= GRUB_ERR_NONE
;
145 grub_errno
= GRUB_ERR_NONE
;
152 grub_env_set (var
, name
);
154 grub_printf (" %s", name
);
157 grub_errno
= GRUB_ERR_NONE
;
158 return (found
&& var
);
161 auto int part_hook (grub_disk_t disk
, const grub_partition_t partition
);
162 int part_hook (grub_disk_t disk
, const grub_partition_t partition
)
164 char *partition_name
, *devname
;
167 partition_name
= grub_partition_get_name (partition
);
168 if (! partition_name
)
171 devname
= grub_xasprintf ("%s,%s", disk
->name
, partition_name
);
172 grub_free (partition_name
);
175 ret
= iterate_device (devname
);
181 auto void try (void);
185 struct cache_entry
**prev
;
186 struct cache_entry
*cache_ent
;
188 for (prev
= &cache
, cache_ent
= *prev
; cache_ent
;
189 prev
= &cache_ent
->next
, cache_ent
= *prev
)
190 if (compare_fn (cache_ent
->key
, key
) == 0)
195 if (iterate_device (cache_ent
->value
))
201 /* Cache entry was outdated. Remove it. */
204 grub_free (cache_ent
->key
);
205 grub_free (cache_ent
->value
);
206 grub_free (cache_ent
);
207 *prev
= cache_ent
->next
;
211 for (i
= 0; i
< nhints
; i
++)
216 end
= hints
[i
] + grub_strlen (hints
[i
]) - 1;
219 if (iterate_device (hints
[i
]))
229 dev
= grub_device_open (hints
[i
]);
238 grub_device_close (dev
);
243 ret
= grub_partition_iterate (dev
->disk
, part_hook
);
246 grub_device_close (dev
);
251 grub_device_iterate (iterate_device
);
254 /* First try without autoloading if we're setting variable. */
257 saved_autoload
= grub_fs_autoload_hook
;
258 grub_fs_autoload_hook
= 0;
261 /* Restore autoload hook. */
262 grub_fs_autoload_hook
= saved_autoload
;
264 /* Retry with autoload if nothing found. */
265 if (grub_errno
== GRUB_ERR_NONE
&& count
== 0)
271 if (grub_errno
== GRUB_ERR_NONE
&& count
== 0)
272 grub_error (GRUB_ERR_FILE_NOT_FOUND
, "no such device: %s", key
);
276 grub_cmd_do_search (grub_command_t cmd
__attribute__ ((unused
)), int argc
,
280 return grub_error (GRUB_ERR_BAD_ARGUMENT
, N_("one argument expected"));
282 FUNC_NAME (args
[0], argc
== 1 ? 0 : args
[1], 0, (args
+ 2),
283 argc
> 2 ? argc
- 2 : 0);
288 static grub_command_t cmd
;
290 #ifdef DO_SEARCH_FILE
291 GRUB_MOD_INIT(search_fs_file
)
292 #elif defined (DO_SEARCH_FS_UUID)
293 GRUB_MOD_INIT(search_fs_uuid
)
295 GRUB_MOD_INIT(search_label
)
299 grub_register_command (COMMAND_NAME
, grub_cmd_do_search
,
300 N_("NAME [VARIABLE] [HINTS]"),
304 #ifdef DO_SEARCH_FILE
305 GRUB_MOD_FINI(search_fs_file
)
306 #elif defined (DO_SEARCH_FS_UUID)
307 GRUB_MOD_FINI(search_fs_uuid
)
309 GRUB_MOD_FINI(search_label
)
312 grub_unregister_command (cmd
);