From 99679fff629f4a624f4fe7a67335c6e90533afc2 Mon Sep 17 00:00:00 2001
From: Pádraig Brady
Date: Mon, 19 Apr 2010 08:41:50 +0100
Subject: df: fix alignment of columns
* src/df.c (alloc_table_row): A new function to allocate storage
for a row of strings.
(print_table): A new function to interate over all stored strings in
the table, and apply alignment honoring the max width of each column.
(get_header): Renamed from print_header, and adjusted accordingly.
(get_dev): Renamed from show_dev. Also we no longer wrap longer
device names over two lines, which can be an unexpected issue for
scripts parsing the output from df.
(get_disk): s/show_/get_/
(get_point): Likewise.
(get_entry): Likewise.
(get_all_entries): Likewise.
* NEWS: Mention the change.
---
NEWS | 3 +
src/df.c | 456 ++++++++++++++++++++++++++++++++++++++++-----------------------
2 files changed, 291 insertions(+), 168 deletions(-)
diff --git a/NEWS b/NEWS
index b2674c0e9..e93f01366 100644
--- a/NEWS
+++ b/NEWS
@@ -28,6 +28,9 @@ GNU coreutils NEWS -*- outline -*-
** Changes in behavior
+ df now aligns columns consistently, and no longer wraps entries
+ with longer device identifiers, over two lines.
+
test now accepts "==" as a synonym for "="
diff --git a/src/df.c b/src/df.c
index 4523c440f..357dca547 100644
--- a/src/df.c
+++ b/src/df.c
@@ -22,11 +22,14 @@
#include
#include
#include
+#include
#include "system.h"
#include "error.h"
#include "fsusage.h"
#include "human.h"
+#include "mbsalign.h"
+#include "mbswidth.h"
#include "mountlist.h"
#include "quote.h"
#include "find-mount-point.h"
@@ -112,6 +115,52 @@ static bool print_grand_total;
/* Grand total data. */
static struct fs_usage grand_fsu;
+/* Display modes. */
+enum { DEFAULT_MODE, INODES_MODE, HUMAN_MODE, POSIX_MODE, NMODES };
+static int header_mode = DEFAULT_MODE;
+
+/* Displayable fields. */
+enum
+{
+ DEV_FIELD, /* file system */
+ TYPE_FIELD, /* FS type */
+ TOTAL_FIELD, /* blocks or inodes */
+ USED_FIELD, /* ditto */
+ FREE_FIELD, /* ditto */
+ PCENT_FIELD, /* percent used */
+ MNT_FIELD, /* mount point */
+ NFIELDS
+};
+
+/* Header strings for the above fields in each mode.
+ NULL means to use the header for the default mode. */
+static const char *headers[NFIELDS][NMODES] = {
+/* DEFAULT_MODE INODES_MODE HUMAN_MODE POSIX_MODE */
+ { N_("Filesystem"), NULL, NULL, NULL },
+ { N_("Type"), NULL, NULL, NULL },
+ { N_("blocks"), N_("Inodes"), N_("Size"), NULL },
+ { N_("Used"), N_("IUsed"), NULL, NULL },
+ { N_("Available"), N_("IFree"), N_("Avail"), NULL },
+ { N_("Use%"), N_("IUse%"), NULL, N_("Capacity") },
+ { N_("Mounted on"), NULL, NULL, NULL }
+};
+
+/* Alignments for the 3 textual and 4 numeric fields. */
+static mbs_align_t alignments[NFIELDS] = {
+ MBS_ALIGN_LEFT, MBS_ALIGN_LEFT,
+ MBS_ALIGN_RIGHT, MBS_ALIGN_RIGHT, MBS_ALIGN_RIGHT, MBS_ALIGN_RIGHT,
+ MBS_ALIGN_LEFT
+};
+
+/* Auto adjusted (up) widths used to align columns. */
+static size_t widths[NFIELDS] = { 14, 4, 5, 5, 5, 4, 0 };
+
+/* Storage for pointers for each string (cell of table). */
+static char ***table;
+
+/* The current number of processed rows (including header). */
+static size_t nrows;
+
/* For long options that have no equivalent short option, use a
non-character as a pseudo short option, starting with CHAR_MAX + 1. */
enum
@@ -141,70 +190,133 @@ static struct option const long_options[] =
{NULL, 0, NULL, 0}
};
+/* Dynamically allocate a row of pointers in TABLE, which
+ can then be accessed with standard 2D array notation. */
+
static void
-print_header (void)
+alloc_table_row (void)
{
- char buf[MAX (LONGEST_HUMAN_READABLE + 1, INT_BUFSIZE_BOUND (uintmax_t))];
+ nrows++;
+ table = xnrealloc (table, nrows, sizeof (char *));
+ table[nrows-1] = xnmalloc (NFIELDS, sizeof (char *));
+}
- if (print_type)
- /* TRANSLATORS:
- For best results (df header/column alignment), ensure that
- your translation has the same length as the original. */
- fputs (_("Filesystem Type"), stdout);
- else
- fputs (_("Filesystem "), stdout);
+/* Output each cell in the table, accounting for the
+ alignment and max width of each column. */
- if (inode_format)
- /* TRANSLATORS:
- For best results (df header/column alignment), ensure that
- your translation has the same length as the original.
- Also, each column name translation should end at the same
- column as the corresponding original. */
- fputs (_(" Inodes IUsed IFree IUse%"), stdout);
- else if (human_output_opts & human_autoscale)
+static void
+print_table (void)
+{
+ size_t field, row;
+
+ for (row = 0; row < nrows; row ++)
{
- if (human_output_opts & human_base_1024)
- fputs (_(" Size Used Avail Use%"), stdout);
- else
- fputs (_(" Size Used Avail Use%"), stdout);
+ for (field = 0; field < NFIELDS; field++)
+ {
+ size_t width = widths[field];
+ char *cell = table[row][field];
+ if (!cell)
+ continue;
+
+ /* Note the DEV_FIELD used to be displayed on it's own line
+ if (!posix_format && mbswidth (cell) > 20), but that
+ functionality is probably more problematic than helpful. */
+ if (field != 0)
+ putchar (' ');
+ if (field == MNT_FIELD) /* The last one. */
+ fputs (cell, stdout);
+ else
+ {
+ cell = ambsalign (table[row][field], &width,
+ alignments[field], MBA_UNIBYTE_FALLBACK);
+ fputs (cell, stdout);
+ free (cell);
+ }
+ IF_LINT (free (table[row][field]));
+ }
+ putchar ('\n');
+ IF_LINT (free (table[row]));
}
- else if (posix_format)
- printf (_(" %s-blocks Used Available Capacity"),
- umaxtostr (output_block_size, buf));
- else
+
+ IF_LINT (free (table));
+}
+
+/* Optain the appropriate header entries. */
+
+static void
+get_header (void)
+{
+ size_t field;
+
+ alloc_table_row ();
+
+ for (field = 0; field < NFIELDS; field++)
{
- int opts = (human_suppress_point_zero
- | human_autoscale | human_SI
- | (human_output_opts
- & (human_group_digits | human_base_1024 | human_B)));
+ if (field == TYPE_FIELD && !print_type)
+ {
+ table[nrows-1][field] = NULL;
+ continue;
+ }
- /* Prefer the base that makes the human-readable value more exact,
- if there is a difference. */
+ char *cell = NULL;
+ char const *header = _(headers[field][header_mode]);
+ if (!header)
+ header = _(headers[field][DEFAULT_MODE]);
- uintmax_t q1000 = output_block_size;
- uintmax_t q1024 = output_block_size;
- bool divisible_by_1000;
- bool divisible_by_1024;
+ if (header_mode == DEFAULT_MODE && field == TOTAL_FIELD)
+ {
+ char buf[LONGEST_HUMAN_READABLE + 1];
- do
+ int opts = (human_suppress_point_zero
+ | human_autoscale | human_SI
+ | (human_output_opts
+ & (human_group_digits | human_base_1024 | human_B)));
+
+ /* Prefer the base that makes the human-readable value more exact,
+ if there is a difference. */
+
+ uintmax_t q1000 = output_block_size;
+ uintmax_t q1024 = output_block_size;
+ bool divisible_by_1000;
+ bool divisible_by_1024;
+
+ do
+ {
+ divisible_by_1000 = q1000 % 1000 == 0; q1000 /= 1000;
+ divisible_by_1024 = q1024 % 1024 == 0; q1024 /= 1024;
+ }
+ while (divisible_by_1000 & divisible_by_1024);
+
+ if (divisible_by_1000 < divisible_by_1024)
+ opts |= human_base_1024;
+ if (divisible_by_1024 < divisible_by_1000)
+ opts &= ~human_base_1024;
+ if (! (opts & human_base_1024))
+ opts |= human_B;
+
+ char *num = human_readable (output_block_size, buf, opts, 1, 1);
+
+ if (asprintf (&cell, "%s-%s", num, header) == -1)
+ cell = NULL;
+ }
+ else if (header_mode == POSIX_MODE && field == TOTAL_FIELD)
{
- divisible_by_1000 = q1000 % 1000 == 0; q1000 /= 1000;
- divisible_by_1024 = q1024 % 1024 == 0; q1024 /= 1024;
+ char buf[INT_BUFSIZE_BOUND (uintmax_t)];
+ char *num = umaxtostr (output_block_size, buf);
+
+ if (asprintf (&cell, "%s-%s", num, header) == -1)
+ cell = NULL;
}
- while (divisible_by_1000 & divisible_by_1024);
+ else
+ cell = strdup (header);
- if (divisible_by_1000 < divisible_by_1024)
- opts |= human_base_1024;
- if (divisible_by_1024 < divisible_by_1000)
- opts &= ~human_base_1024;
- if (! (opts & human_base_1024))
- opts |= human_B;
+ if (!cell)
+ xalloc_die ();
- printf (_(" %4s-blocks Used Available Use%%"),
- human_readable (output_block_size, buf, opts, 1, 1));
- }
+ table[nrows-1][field] = cell;
- fputs (_(" Mounted on\n"), stdout);
+ widths[field] = MAX (widths[field], mbswidth (cell, 0));
+ }
}
/* Is FSTYPE a type of file system that should be listed? */
@@ -305,7 +417,7 @@ add_uint_with_neg_flag (uintmax_t *dest, bool *dest_neg,
*dest = -*dest;
}
-/* Display a space listing for the disk device with absolute file name DISK.
+/* Optain a space listing for the disk device with absolute file name DISK.
If MOUNT_POINT is non-NULL, it is the name of the root of the
file system on DISK.
If STAT_FILE is non-null, it is the name of a file within the file
@@ -319,16 +431,13 @@ add_uint_with_neg_flag (uintmax_t *dest, bool *dest_neg,
ME_DUMMY and ME_REMOTE are the mount entry flags. */
static void
-show_dev (char const *disk, char const *mount_point,
- char const *stat_file, char const *fstype,
- bool me_dummy, bool me_remote,
- const struct fs_usage *force_fsu)
+get_dev (char const *disk, char const *mount_point,
+ char const *stat_file, char const *fstype,
+ bool me_dummy, bool me_remote,
+ const struct fs_usage *force_fsu)
{
struct fs_usage fsu;
- char buf[3][LONGEST_HUMAN_READABLE + 2];
- int width;
- int col1_adjustment = 0;
- int use_width;
+ char buf[LONGEST_HUMAN_READABLE + 2];
uintmax_t input_units;
uintmax_t output_units;
uintmax_t total;
@@ -338,6 +447,8 @@ show_dev (char const *disk, char const *mount_point,
uintmax_t used;
bool negate_used;
double pct = -1;
+ char* cell;
+ size_t field;
if (me_remote && show_local_fs)
return;
@@ -370,39 +481,18 @@ show_dev (char const *disk, char const *mount_point,
if (! file_systems_processed)
{
file_systems_processed = true;
- print_header ();
+ get_header ();
}
+ alloc_table_row ();
+
if (! disk)
disk = "-"; /* unknown */
if (! fstype)
fstype = "-"; /* unknown */
- /* df.c reserved 5 positions for fstype,
- but that does not suffice for type iso9660 */
- if (print_type)
- {
- size_t disk_name_len = strlen (disk);
- size_t fstype_len = strlen (fstype);
- if (disk_name_len + fstype_len < 18)
- printf ("%s%*s ", disk, 18 - (int) disk_name_len, fstype);
- else if (!posix_format)
- printf ("%s\n%18s ", disk, fstype);
- else
- printf ("%s %s", disk, fstype);
- }
- else
- {
- if (strlen (disk) > 20 && !posix_format)
- printf ("%s\n%20s", disk, "");
- else
- printf ("%-20s", disk);
- }
-
if (inode_format)
{
- width = 7;
- use_width = 5;
input_units = output_units = 1;
total = fsu.fsu_files;
available = fsu.fsu_ffree;
@@ -416,22 +506,6 @@ show_dev (char const *disk, char const *mount_point,
}
else
{
- if (human_output_opts & human_autoscale)
- width = 5 + ! (human_output_opts & human_base_1024);
- else
- {
- width = 9;
- if (posix_format)
- {
- uintmax_t b;
- col1_adjustment = -3;
- for (b = output_block_size; 9 < b; b /= 10)
- col1_adjustment++;
- }
- }
- use_width = ((posix_format
- && ! (human_output_opts & human_autoscale))
- ? 8 : 4);
input_units = fsu.fsu_blocksize;
output_units = output_block_size;
total = fsu.fsu_blocks;
@@ -458,73 +532,110 @@ show_dev (char const *disk, char const *mount_point,
negate_used = (total < available_to_root);
}
- printf (" %*s %*s %*s ",
- width + col1_adjustment,
- df_readable (false, total,
- buf[0], input_units, output_units),
- width, df_readable (negate_used, used,
- buf[1], input_units, output_units),
- width, df_readable (negate_available, available,
- buf[2], input_units, output_units));
-
- if (! known_value (used) || ! known_value (available))
- ;
- else if (!negate_used
- && used <= TYPE_MAXIMUM (uintmax_t) / 100
- && used + available != 0
- && (used + available < used) == negate_available)
- {
- uintmax_t u100 = used * 100;
- uintmax_t nonroot_total = used + available;
- pct = u100 / nonroot_total + (u100 % nonroot_total != 0);
- }
- else
+ for (field = 0; field < NFIELDS; field++)
{
- /* The calculation cannot be done easily with integer
- arithmetic. Fall back on floating point. This can suffer
- from minor rounding errors, but doing it exactly requires
- multiple precision arithmetic, and it's not worth the
- aggravation. */
- double u = negate_used ? - (double) - used : used;
- double a = negate_available ? - (double) - available : available;
- double nonroot_total = u + a;
- if (nonroot_total)
+ switch (field)
{
- long int lipct = pct = u * 100 / nonroot_total;
- double ipct = lipct;
+ case DEV_FIELD:
+ cell = xstrdup (disk);
+ break;
- /* Like `pct = ceil (dpct);', but avoid ceil so that
- the math library needn't be linked. */
- if (ipct - 1 < pct && pct <= ipct + 1)
- pct = ipct + (ipct < pct);
- }
- }
+ case TYPE_FIELD:
+ cell = print_type ? xstrdup (fstype) : NULL;
+ break;
- if (0 <= pct)
- printf ("%*.0f%%", use_width - 1, pct);
- else
- printf ("%*s", use_width, "- ");
+ case TOTAL_FIELD:
+ cell = xstrdup (df_readable (false, total, buf,
+ input_units, output_units));
+ break;
+ case USED_FIELD:
+ cell = xstrdup (df_readable (negate_used, used, buf,
+ input_units, output_units));
+ break;
+ case FREE_FIELD:
+ cell = xstrdup (df_readable (negate_available, available, buf,
+ input_units, output_units));
+ break;
- if (mount_point)
- {
+ case PCENT_FIELD:
+ if (! known_value (used) || ! known_value (available))
+ ;
+ else if (!negate_used
+ && used <= TYPE_MAXIMUM (uintmax_t) / 100
+ && used + available != 0
+ && (used + available < used) == negate_available)
+ {
+ uintmax_t u100 = used * 100;
+ uintmax_t nonroot_total = used + available;
+ pct = u100 / nonroot_total + (u100 % nonroot_total != 0);
+ }
+ else
+ {
+ /* The calculation cannot be done easily with integer
+ arithmetic. Fall back on floating point. This can suffer
+ from minor rounding errors, but doing it exactly requires
+ multiple precision arithmetic, and it's not worth the
+ aggravation. */
+ double u = negate_used ? - (double) - used : used;
+ double a = negate_available ? - (double) - available : available;
+ double nonroot_total = u + a;
+ if (nonroot_total)
+ {
+ long int lipct = pct = u * 100 / nonroot_total;
+ double ipct = lipct;
+
+ /* Like `pct = ceil (dpct);', but avoid ceil so that
+ the math library needn't be linked. */
+ if (ipct - 1 < pct && pct <= ipct + 1)
+ pct = ipct + (ipct < pct);
+ }
+ }
+
+ if (0 <= pct)
+ {
+ if (asprintf (&cell, "%.0f%%", pct) == -1)
+ cell = NULL;
+ }
+ else
+ cell = strdup ("-");
+
+ if (!cell)
+ xalloc_die ();
+
+ break;
+
+ case MNT_FIELD:
+ if (mount_point)
+ {
#ifdef HIDE_AUTOMOUNT_PREFIX
- /* Don't print the first directory name in MOUNT_POINT if it's an
- artifact of an automounter. This is a bit too aggressive to be
- the default. */
- if (strncmp ("/auto/", mount_point, 6) == 0)
- mount_point += 5;
- else if (strncmp ("/tmp_mnt/", mount_point, 9) == 0)
- mount_point += 8;
+ /* Don't print the first directory name in MOUNT_POINT if it's an
+ artifact of an automounter. This is a bit too aggressive to be
+ the default. */
+ if (strncmp ("/auto/", mount_point, 6) == 0)
+ mount_point += 5;
+ else if (strncmp ("/tmp_mnt/", mount_point, 9) == 0)
+ mount_point += 8;
#endif
- printf (" %s", mount_point);
+ cell = xstrdup (mount_point);
+ }
+ else
+ cell = NULL;
+ break;
+
+ default:
+ assert (!"unhandled field");
+ }
+
+ if (cell)
+ widths[field] = MAX (widths[field], mbswidth (cell, 0));
+ table[nrows-1][field] = cell;
}
- putchar ('\n');
}
/* If DISK corresponds to a mount point, show its usage
and return true. Otherwise, return false. */
static bool
-show_disk (char const *disk)
+get_disk (char const *disk)
{
struct mount_entry const *me;
struct mount_entry const *best_match = NULL;
@@ -535,9 +646,9 @@ show_disk (char const *disk)
if (best_match)
{
- show_dev (best_match->me_devname, best_match->me_mountdir, NULL,
- best_match->me_type, best_match->me_dummy,
- best_match->me_remote, NULL);
+ get_dev (best_match->me_devname, best_match->me_mountdir, NULL,
+ best_match->me_type, best_match->me_dummy,
+ best_match->me_remote, NULL);
return true;
}
@@ -548,7 +659,7 @@ show_disk (char const *disk)
and show its disk usage.
STATP must be the result of `stat (POINT, STATP)'. */
static void
-show_point (const char *point, const struct stat *statp)
+get_point (const char *point, const struct stat *statp)
{
struct stat disk_stats;
struct mount_entry *me;
@@ -621,9 +732,9 @@ show_point (const char *point, const struct stat *statp)
}
if (best_match)
- show_dev (best_match->me_devname, best_match->me_mountdir, point,
- best_match->me_type, best_match->me_dummy, best_match->me_remote,
- NULL);
+ get_dev (best_match->me_devname, best_match->me_mountdir, point,
+ best_match->me_type, best_match->me_dummy, best_match->me_remote,
+ NULL);
else
{
/* We couldn't find the mount entry corresponding to POINT. Go ahead and
@@ -634,7 +745,7 @@ show_point (const char *point, const struct stat *statp)
char *mp = find_mount_point (point, statp);
if (mp)
{
- show_dev (NULL, mp, NULL, NULL, false, false, NULL);
+ get_dev (NULL, mp, NULL, NULL, false, false, NULL);
free (mp);
}
}
@@ -644,25 +755,25 @@ show_point (const char *point, const struct stat *statp)
for it. STATP is the results of `stat' on NAME. */
static void
-show_entry (char const *name, struct stat const *statp)
+get_entry (char const *name, struct stat const *statp)
{
if ((S_ISBLK (statp->st_mode) || S_ISCHR (statp->st_mode))
- && show_disk (name))
+ && get_disk (name))
return;
- show_point (name, statp);
+ get_point (name, statp);
}
/* Show all mounted file systems, except perhaps those that are of
an unselected type or are empty. */
static void
-show_all_entries (void)
+get_all_entries (void)
{
struct mount_entry *me;
for (me = mount_list; me; me = me->me_next)
- show_dev (me->me_devname, me->me_mountdir, NULL, me->me_type,
+ get_dev (me->me_devname, me->me_mountdir, NULL, me->me_type,
me->me_dummy, me->me_remote, NULL);
}
@@ -862,6 +973,13 @@ main (int argc, char **argv)
&human_output_opts, &output_block_size);
}
+ if (inode_format)
+ header_mode = INODES_MODE;
+ else if (human_output_opts & human_autoscale)
+ header_mode = HUMAN_MODE;
+ else if (posix_format)
+ header_mode = POSIX_MODE;
+
/* Fail if the same file system type was both selected and excluded. */
{
bool match = false;
@@ -939,18 +1057,20 @@ main (int argc, char **argv)
for (i = optind; i < argc; ++i)
if (argv[i])
- show_entry (argv[i], &stats[i - optind]);
+ get_entry (argv[i], &stats[i - optind]);
}
else
- show_all_entries ();
+ get_all_entries ();
if (print_grand_total)
{
if (inode_format)
grand_fsu.fsu_blocks = 1;
- show_dev ("total", NULL, NULL, NULL, false, false, &grand_fsu);
+ get_dev ("total", NULL, NULL, NULL, false, false, &grand_fsu);
}
+ print_table ();
+
if (! file_systems_processed)
error (EXIT_FAILURE, 0, _("no file systems processed"));
--
cgit v1.2.3-70-g09d2