| /* misc.c - miscellaneous functions */ |
| /* |
| * 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/normal.h> |
| #include <grub/disk.h> |
| #include <grub/fs.h> |
| #include <grub/err.h> |
| #include <grub/misc.h> |
| #include <grub/mm.h> |
| #include <grub/datetime.h> |
| #include <grub/term.h> |
| #include <grub/i18n.h> |
| #include <grub/partition.h> |
| |
| static const char *grub_human_sizes[3][6] = |
| { |
| /* This algorithm in reality would work only up to (2^64) / 100 B = 81 PiB. |
| Put here all possible suffixes it can produce so no array bounds check |
| is needed. |
| */ |
| /* TRANSLATORS: that's the list of binary unit prefixes. */ |
| { N_("B"), N_("KiB"), N_("MiB"), N_("GiB"), N_("TiB"), N_("PiB")}, |
| /* TRANSLATORS: that's the list of binary unit prefixes. */ |
| { "", N_("K"), N_("M"), N_("G"), N_("T"), N_("P") }, |
| /* TRANSLATORS: that's the list of binary unit prefixes. */ |
| { N_("B/s"), N_("KiB/s"), N_("MiB/s"), N_("GiB/s"), N_("TiB/s"), N_("PiB/s"), }, |
| }; |
| |
| const char * |
| grub_get_human_size (grub_uint64_t size, enum grub_human_size_type type) |
| { |
| grub_uint64_t fsize; |
| unsigned units = 0; |
| static char buf[30]; |
| const char *umsg; |
| |
| if (type != GRUB_HUMAN_SIZE_SPEED) |
| fsize = size * 100ULL; |
| else |
| fsize = size; |
| |
| /* Since 2^64 / 1024^5 < 102400, this can give at most 5 iterations. |
| So units <=5, so impossible to go past the end of array. |
| */ |
| while (fsize >= 102400) |
| { |
| fsize = (fsize + 512) / 1024; |
| units++; |
| } |
| |
| umsg = _(grub_human_sizes[type][units]); |
| |
| if (units || type == GRUB_HUMAN_SIZE_SPEED) |
| { |
| grub_uint64_t whole, fraction; |
| |
| whole = grub_divmod64 (fsize, 100, &fraction); |
| grub_snprintf (buf, sizeof (buf), |
| "%" PRIuGRUB_UINT64_T |
| ".%02" PRIuGRUB_UINT64_T "%s", whole, fraction, |
| umsg); |
| } |
| else |
| grub_snprintf (buf, sizeof (buf), "%llu%s", (unsigned long long) size, |
| umsg); |
| return buf; |
| } |
| |
| /* Print the information on the device NAME. */ |
| grub_err_t |
| grub_normal_print_device_info (const char *name) |
| { |
| grub_device_t dev; |
| char *p; |
| |
| p = grub_strchr (name, ','); |
| if (p) |
| { |
| grub_xputs ("\t"); |
| grub_printf_ (N_("Partition %s:"), name); |
| grub_xputs (" "); |
| } |
| else |
| { |
| grub_printf_ (N_("Device %s:"), name); |
| grub_xputs (" "); |
| } |
| |
| dev = grub_device_open (name); |
| if (! dev) |
| grub_printf ("%s", _("Filesystem cannot be accessed")); |
| else if (dev->disk) |
| { |
| grub_fs_t fs; |
| |
| fs = grub_fs_probe (dev); |
| /* Ignore all errors. */ |
| grub_errno = 0; |
| |
| if (fs) |
| { |
| const char *fsname = fs->name; |
| if (grub_strcmp (fsname, "ext2") == 0) |
| fsname = "ext*"; |
| grub_printf_ (N_("Filesystem type %s"), fsname); |
| if (fs->label) |
| { |
| char *label; |
| (fs->label) (dev, &label); |
| if (grub_errno == GRUB_ERR_NONE) |
| { |
| if (label && grub_strlen (label)) |
| { |
| grub_xputs (" "); |
| grub_printf_ (N_("- Label `%s'"), label); |
| } |
| grub_free (label); |
| } |
| grub_errno = GRUB_ERR_NONE; |
| } |
| if (fs->mtime) |
| { |
| grub_int32_t tm; |
| struct grub_datetime datetime; |
| (fs->mtime) (dev, &tm); |
| if (grub_errno == GRUB_ERR_NONE) |
| { |
| grub_unixtime2datetime (tm, &datetime); |
| grub_xputs (" "); |
| /* TRANSLATORS: Arguments are year, month, day, hour, minute, |
| second, day of the week (translated). */ |
| grub_printf_ (N_("- Last modification time %d-%02d-%02d " |
| "%02d:%02d:%02d %s"), |
| datetime.year, datetime.month, datetime.day, |
| datetime.hour, datetime.minute, datetime.second, |
| grub_get_weekday_name (&datetime)); |
| |
| } |
| grub_errno = GRUB_ERR_NONE; |
| } |
| if (fs->uuid) |
| { |
| char *uuid; |
| (fs->uuid) (dev, &uuid); |
| if (grub_errno == GRUB_ERR_NONE) |
| { |
| if (uuid && grub_strlen (uuid)) |
| grub_printf (", UUID %s", uuid); |
| grub_free (uuid); |
| } |
| grub_errno = GRUB_ERR_NONE; |
| } |
| } |
| else |
| grub_printf ("%s", _("No known filesystem detected")); |
| |
| if (dev->disk->partition) |
| grub_printf (_(" - Partition start at %llu%sKiB"), |
| (unsigned long long) (grub_partition_get_start (dev->disk->partition) >> 1), |
| (grub_partition_get_start (dev->disk->partition) & 1) ? ".5" : "" ); |
| else |
| grub_printf_ (N_(" - Sector size %uB"), 1 << dev->disk->log_sector_size); |
| if (grub_disk_get_size (dev->disk) == GRUB_DISK_SIZE_UNKNOWN) |
| grub_puts_ (N_(" - Total size unknown")); |
| else |
| grub_printf (_(" - Total size %llu%sKiB"), |
| (unsigned long long) (grub_disk_get_size (dev->disk) >> 1), |
| /* TRANSLATORS: Replace dot with appropriate decimal separator for |
| your language. */ |
| (grub_disk_get_size (dev->disk) & 1) ? _(".5") : ""); |
| } |
| |
| if (dev) |
| grub_device_close (dev); |
| |
| grub_xputs ("\n"); |
| return grub_errno; |
| } |