123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251 |
- /*
- * GRUB -- GRand Unified Bootloader
- * Copyright (C) 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/device.h>
- #include <grub/disk.h>
- #include <grub/partition.h>
- #include <grub/gpt_partition.h>
- #include <grub/net.h>
- #include <grub/fs.h>
- #include <grub/file.h>
- #include <grub/misc.h>
- #include <grub/env.h>
- #include <grub/extcmd.h>
- #include <grub/i18n.h>
- #include <grub/i386/pc/boot.h>
- GRUB_MOD_LICENSE ("GPLv3+");
- static const struct grub_arg_option options[] =
- {
- {"set", 's', 0,
- N_("Set a variable to return value."), N_("VARNAME"), ARG_TYPE_STRING},
- /* TRANSLATORS: It's a driver that is currently in use to access
- the diven disk. */
- {"driver", 'd', 0, N_("Determine driver."), 0, 0},
- {"partmap", 'p', 0, N_("Determine partition map type."), 0, 0},
- {"fs", 'f', 0, N_("Determine filesystem type."), 0, 0},
- {"fs-uuid", 'u', 0, N_("Determine filesystem UUID."), 0, 0},
- {"label", 'l', 0, N_("Determine filesystem label."), 0, 0},
- {"part-uuid", 0, 0, N_("Determine partition UUID."), 0, 0},
- {0, 0, 0, 0, 0, 0}
- };
- static grub_err_t
- grub_cmd_probe (grub_extcmd_context_t ctxt, int argc, char **args)
- {
- struct grub_arg_list *state = ctxt->state;
- grub_device_t dev;
- grub_fs_t fs;
- char *ptr;
- grub_err_t err;
- if (argc < 1)
- return grub_error (GRUB_ERR_BAD_ARGUMENT, "device name required");
- ptr = args[0] + grub_strlen (args[0]) - 1;
- if (args[0][0] == '(' && *ptr == ')')
- {
- *ptr = 0;
- dev = grub_device_open (args[0] + 1);
- *ptr = ')';
- }
- else
- dev = grub_device_open (args[0]);
- if (! dev)
- return grub_errno;
- if (state[1].set)
- {
- const char *val = "none";
- if (dev->net)
- val = dev->net->protocol->name;
- if (dev->disk)
- val = dev->disk->dev->name;
- if (state[0].set)
- grub_env_set (state[0].arg, val);
- else
- grub_printf ("%s", val);
- grub_device_close (dev);
- return GRUB_ERR_NONE;
- }
- if (state[2].set)
- {
- const char *val = "none";
- if (dev->disk && dev->disk->partition)
- val = dev->disk->partition->partmap->name;
- if (state[0].set)
- grub_env_set (state[0].arg, val);
- else
- grub_printf ("%s", val);
- grub_device_close (dev);
- return GRUB_ERR_NONE;
- }
- if (state[6].set)
- {
- /* AAAABBBB-CCCC-DDDD-EEEE-FFFFFFFFFFFF + null terminator */
- char val[37] = "none";
- if (dev->disk && dev->disk->partition)
- {
- struct grub_partition *p = dev->disk->partition;
- grub_disk_t disk = grub_disk_open(dev->disk->name);
- if (!disk)
- {
- grub_device_close (dev);
- return grub_errno;
- }
- if (grub_strcmp(dev->disk->partition->partmap->name, "gpt") == 0)
- {
- struct grub_gpt_partentry entry;
- grub_guid_t *guid;
- if (grub_disk_read(disk, p->offset, p->index, sizeof(entry), &entry))
- {
- grub_error_push ();
- grub_disk_close (disk);
- grub_device_close (dev);
- grub_error_pop ();
- return grub_errno;
- }
- guid = &entry.guid;
- guid->data1 = grub_le_to_cpu32 (guid->data1);
- guid->data2 = grub_le_to_cpu16 (guid->data2);
- guid->data3 = grub_le_to_cpu16 (guid->data3);
- grub_snprintf (val, sizeof(val), "%pG", guid);
- }
- else if (grub_strcmp(dev->disk->partition->partmap->name, "msdos") == 0)
- {
- grub_uint32_t nt_disk_sig;
- if (grub_disk_read(disk, 0, GRUB_BOOT_MACHINE_WINDOWS_NT_MAGIC,
- sizeof(nt_disk_sig), &nt_disk_sig) == 0)
- grub_snprintf (val, sizeof(val), "%08x-%02x",
- grub_le_to_cpu32(nt_disk_sig), 1 + p->number);
- }
- grub_disk_close(disk);
- }
- if (state[0].set)
- grub_env_set (state[0].arg, val);
- else
- grub_printf ("%s", val);
- grub_device_close (dev);
- return GRUB_ERR_NONE;
- }
- fs = grub_fs_probe (dev);
- if (! fs)
- {
- grub_error_push ();
- grub_device_close (dev);
- grub_error_pop ();
- return grub_errno;
- }
- if (state[3].set)
- {
- if (state[0].set)
- grub_env_set (state[0].arg, fs->name);
- else
- grub_printf ("%s", fs->name);
- grub_device_close (dev);
- return GRUB_ERR_NONE;
- }
- if (state[4].set)
- {
- char *uuid;
- if (! fs->fs_uuid)
- {
- grub_device_close (dev);
- return grub_error (GRUB_ERR_NOT_IMPLEMENTED_YET,
- N_("%s does not support UUIDs"), fs->name);
- }
- err = fs->fs_uuid (dev, &uuid);
- if (err)
- {
- grub_device_close (dev);
- return err;
- }
- if (! uuid)
- {
- grub_device_close (dev);
- return grub_error (GRUB_ERR_NOT_IMPLEMENTED_YET,
- N_("%s does not support UUIDs"), fs->name);
- }
- if (state[0].set)
- grub_env_set (state[0].arg, uuid);
- else
- grub_printf ("%s", uuid);
- grub_free (uuid);
- grub_device_close (dev);
- return GRUB_ERR_NONE;
- }
- if (state[5].set)
- {
- char *label;
- if (! fs->fs_label)
- {
- grub_device_close (dev);
- return grub_error (GRUB_ERR_NOT_IMPLEMENTED_YET,
- N_("filesystem `%s' does not support labels"),
- fs->name);
- }
- err = fs->fs_label (dev, &label);
- if (err)
- {
- grub_device_close (dev);
- return err;
- }
- if (! label)
- {
- grub_device_close (dev);
- return grub_error (GRUB_ERR_NOT_IMPLEMENTED_YET,
- N_("filesystem `%s' does not support labels"),
- fs->name);
- }
- if (state[0].set)
- grub_env_set (state[0].arg, label);
- else
- grub_printf ("%s", label);
- grub_free (label);
- grub_device_close (dev);
- return GRUB_ERR_NONE;
- }
- grub_device_close (dev);
- return grub_error (GRUB_ERR_BAD_ARGUMENT, "unrecognised target");
- }
- static grub_extcmd_t cmd;
- GRUB_MOD_INIT (probe)
- {
- cmd = grub_register_extcmd ("probe", grub_cmd_probe, 0, N_("DEVICE"),
- N_("Retrieve device info."), options);
- }
- GRUB_MOD_FINI (probe)
- {
- grub_unregister_extcmd (cmd);
- }
|