123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227 |
- /* search.c - search devices based on a file or a filesystem label */
- /*
- * GRUB -- GRand Unified Bootloader
- * Copyright (C) 2005,2007,2008,2009 Free Software Foundation, Inc.
- *
- * GRUB is free software: you can redistribute it and/or modify
- * it under the terms of the GNU General Public License as published by
- * the Free Software Foundation, either version 3 of the License, or
- * (at your option) any later version.
- *
- * GRUB is distributed in the hope that it will be useful,
- * but WITHOUT ANY WARRANTY; without even the implied warranty of
- * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- * GNU General Public License for more details.
- *
- * You should have received a copy of the GNU General Public License
- * along with GRUB. If not, see <http://www.gnu.org/licenses/>.
- */
- #include <grub/types.h>
- #include <grub/misc.h>
- #include <grub/mm.h>
- #include <grub/err.h>
- #include <grub/dl.h>
- #include <grub/env.h>
- #include <grub/extcmd.h>
- #include <grub/search.h>
- #include <grub/i18n.h>
- GRUB_MOD_LICENSE ("GPLv3+");
- static const struct grub_arg_option options[] =
- {
- {"file", 'f', 0, N_("Search devices by a file."), 0, 0},
- {"label", 'l', 0, N_("Search devices by a filesystem label."),
- 0, 0},
- {"fs-uuid", 'u', 0, N_("Search devices by a filesystem UUID."),
- 0, 0},
- {"set", 's', GRUB_ARG_OPTION_OPTIONAL,
- N_("Set a variable to the first device found."), N_("VARNAME"),
- ARG_TYPE_STRING},
- {"no-floppy", 'n', 0, N_("Do not probe any floppy drive."), 0, 0},
- {"efidisk-only", 0, 0, N_("Only probe EFI disks."), 0, 0},
- {"hint", 'h', GRUB_ARG_OPTION_REPEATABLE,
- N_("First try the device HINT. If HINT ends in comma, "
- "also try subpartitions"), N_("HINT"), ARG_TYPE_STRING},
- {"hint-ieee1275", 0, GRUB_ARG_OPTION_REPEATABLE,
- N_("First try the device HINT if currently running on IEEE1275. "
- "If HINT ends in comma, also try subpartitions"),
- N_("HINT"), ARG_TYPE_STRING},
- {"hint-bios", 0, GRUB_ARG_OPTION_REPEATABLE,
- N_("First try the device HINT if currently running on BIOS. "
- "If HINT ends in comma, also try subpartitions"),
- N_("HINT"), ARG_TYPE_STRING},
- {"hint-baremetal", 0, GRUB_ARG_OPTION_REPEATABLE,
- N_("First try the device HINT if direct hardware access is supported. "
- "If HINT ends in comma, also try subpartitions"),
- N_("HINT"), ARG_TYPE_STRING},
- {"hint-efi", 0, GRUB_ARG_OPTION_REPEATABLE,
- N_("First try the device HINT if currently running on EFI. "
- "If HINT ends in comma, also try subpartitions"),
- N_("HINT"), ARG_TYPE_STRING},
- {"hint-arc", 0, GRUB_ARG_OPTION_REPEATABLE,
- N_("First try the device HINT if currently running on ARC."
- " If HINT ends in comma, also try subpartitions"),
- N_("HINT"), ARG_TYPE_STRING},
- {0, 0, 0, 0, 0, 0}
- };
- enum options
- {
- SEARCH_FILE,
- SEARCH_LABEL,
- SEARCH_FS_UUID,
- SEARCH_SET,
- SEARCH_NO_FLOPPY,
- SEARCH_EFIDISK_ONLY,
- SEARCH_HINT,
- SEARCH_HINT_IEEE1275,
- SEARCH_HINT_BIOS,
- SEARCH_HINT_BAREMETAL,
- SEARCH_HINT_EFI,
- SEARCH_HINT_ARC,
- };
- static grub_err_t
- grub_cmd_search (grub_extcmd_context_t ctxt, int argc, char **args)
- {
- struct grub_arg_list *state = ctxt->state;
- const char *var = 0;
- const char *id = 0;
- int i = 0, j = 0, nhints = 0;
- char **hints = NULL;
- enum search_flags flags = SEARCH_FLAGS_NONE;
- if (state[SEARCH_HINT].set)
- for (i = 0; state[SEARCH_HINT].args[i]; i++)
- nhints++;
- #ifdef GRUB_MACHINE_IEEE1275
- if (state[SEARCH_HINT_IEEE1275].set)
- for (i = 0; state[SEARCH_HINT_IEEE1275].args[i]; i++)
- nhints++;
- #endif
- #ifdef GRUB_MACHINE_EFI
- if (state[SEARCH_HINT_EFI].set)
- for (i = 0; state[SEARCH_HINT_EFI].args[i]; i++)
- nhints++;
- #endif
- #ifdef GRUB_MACHINE_PCBIOS
- if (state[SEARCH_HINT_BIOS].set)
- for (i = 0; state[SEARCH_HINT_BIOS].args[i]; i++)
- nhints++;
- #endif
- #ifdef GRUB_MACHINE_ARC
- if (state[SEARCH_HINT_ARC].set)
- for (i = 0; state[SEARCH_HINT_ARC].args[i]; i++)
- nhints++;
- #endif
- if (state[SEARCH_HINT_BAREMETAL].set)
- for (i = 0; state[SEARCH_HINT_BAREMETAL].args[i]; i++)
- nhints++;
- hints = grub_calloc (nhints, sizeof (hints[0]));
- if (!hints)
- return grub_errno;
- j = 0;
- if (state[SEARCH_HINT].set)
- for (i = 0; state[SEARCH_HINT].args[i]; i++)
- hints[j++] = state[SEARCH_HINT].args[i];
- #ifdef GRUB_MACHINE_IEEE1275
- if (state[SEARCH_HINT_IEEE1275].set)
- for (i = 0; state[SEARCH_HINT_IEEE1275].args[i]; i++)
- hints[j++] = state[SEARCH_HINT_IEEE1275].args[i];
- #endif
- #ifdef GRUB_MACHINE_EFI
- if (state[SEARCH_HINT_EFI].set)
- for (i = 0; state[SEARCH_HINT_EFI].args[i]; i++)
- hints[j++] = state[SEARCH_HINT_EFI].args[i];
- #endif
- #ifdef GRUB_MACHINE_ARC
- if (state[SEARCH_HINT_ARC].set)
- for (i = 0; state[SEARCH_HINT_ARC].args[i]; i++)
- hints[j++] = state[SEARCH_HINT_ARC].args[i];
- #endif
- #ifdef GRUB_MACHINE_PCBIOS
- if (state[SEARCH_HINT_BIOS].set)
- for (i = 0; state[SEARCH_HINT_BIOS].args[i]; i++)
- hints[j++] = state[SEARCH_HINT_BIOS].args[i];
- #endif
- if (state[SEARCH_HINT_BAREMETAL].set)
- for (i = 0; state[SEARCH_HINT_BAREMETAL].args[i]; i++)
- hints[j++] = state[SEARCH_HINT_BAREMETAL].args[i];
- /* Skip hints for future platforms. */
- for (j = 0; j < argc; j++)
- if (grub_memcmp (args[j], "--hint-", sizeof ("--hint-") - 1) != 0)
- break;
- if (state[SEARCH_SET].set)
- var = state[SEARCH_SET].arg ? state[SEARCH_SET].arg : "root";
- if (argc != j)
- id = args[j];
- else if (state[SEARCH_SET].set && state[SEARCH_SET].arg)
- {
- id = state[SEARCH_SET].arg;
- var = "root";
- }
- else
- {
- grub_error (GRUB_ERR_BAD_ARGUMENT, N_("one argument expected"));
- goto out;
- }
- if (state[SEARCH_NO_FLOPPY].set)
- flags |= SEARCH_FLAGS_NO_FLOPPY;
- if (state[SEARCH_EFIDISK_ONLY].set)
- flags |= SEARCH_FLAGS_EFIDISK_ONLY;
- if (state[SEARCH_LABEL].set)
- grub_search_label (id, var, flags, hints, nhints);
- else if (state[SEARCH_FS_UUID].set)
- grub_search_fs_uuid (id, var, flags, hints, nhints);
- else if (state[SEARCH_FILE].set)
- grub_search_fs_file (id, var, flags, hints, nhints);
- else
- grub_error (GRUB_ERR_INVALID_COMMAND, "unspecified search type");
- out:
- grub_free (hints);
- return grub_errno;
- }
- static grub_extcmd_t cmd;
- GRUB_MOD_INIT(search)
- {
- cmd =
- grub_register_extcmd ("search", grub_cmd_search,
- GRUB_COMMAND_FLAG_EXTRACTOR | GRUB_COMMAND_ACCEPT_DASH,
- N_("[-f|-l|-u|-s|-n] [--hint HINT [--hint HINT] ...]"
- " NAME"),
- N_("Search devices by file, filesystem label"
- " or filesystem UUID."
- " If --set is specified, the first device found is"
- " set to a variable. If no variable name is"
- " specified, `root' is used."),
- options);
- }
- GRUB_MOD_FINI(search)
- {
- grub_unregister_extcmd (cmd);
- }
|