6603c22f31
Found by: Coverity scan.
194 lines
5.4 KiB
C
194 lines
5.4 KiB
C
/* 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;
|
|
}
|