Home | History | Annotate | Download | only in posix
      1 /* df.c - report free disk space.
      2  *
      3  * Copyright 2006 Rob Landley <rob (at) landley.net>
      4  *
      5  * See http://opengroup.org/onlinepubs/9699919799/utilities/df.html
      6 
      7 USE_DF(NEWTOY(df, "HPkht*a[-HPkh]", TOYFLAG_SBIN))
      8 
      9 config DF
     10   bool "df"
     11   default y
     12   help
     13     usage: df [-HPkh] [-t type] [FILESYSTEM ...]
     14 
     15     The "disk free" command shows total/used/available disk space for
     16     each filesystem listed on the command line, or all currently mounted
     17     filesystems.
     18 
     19     -a	Show all (including /proc and friends)
     20     -P	The SUSv3 "Pedantic" option
     21     -k	Sets units back to 1024 bytes (the default without -P)
     22     -h	Human readable output (K=1024)
     23     -H	Human readable output (k=1000)
     24     -t type	Display only filesystems of this type
     25 
     26     Pedantic provides a slightly less useful output format dictated by Posix,
     27     and sets the units to 512 bytes instead of the default 1024 bytes.
     28 */
     29 
     30 #define FOR_df
     31 #include "toys.h"
     32 
     33 GLOBALS(
     34   struct arg_list *fstype;
     35 
     36   long units;
     37   int column_widths[5];
     38   int header_shown;
     39 )
     40 
     41 static void measure_column(int col, const char *s)
     42 {
     43   size_t len = strlen(s);
     44 
     45   if (TT.column_widths[col] < len) TT.column_widths[col] = len;
     46 }
     47 
     48 static void measure_numeric_column(int col, long long n)
     49 {
     50   snprintf(toybuf, sizeof(toybuf), "%lld", n);
     51   return measure_column(col, toybuf);
     52 }
     53 
     54 static void show_header()
     55 {
     56   TT.header_shown = 1;
     57 
     58   // The filesystem column is always at least this wide.
     59   if (TT.column_widths[0] < 14) TT.column_widths[0] = 14;
     60 
     61   if (toys.optflags & (FLAG_H|FLAG_h)) {
     62     xprintf("%-*s Size  Used Avail Use%% Mounted on\n",
     63             TT.column_widths[0], "Filesystem");
     64   } else {
     65     const char *blocks_label = TT.units == 512 ? "512-blocks" : "1K-blocks";
     66     const char *use_label = toys.optflags & FLAG_P ? "Capacity" : "Use%";
     67 
     68     measure_column(1, blocks_label);
     69     measure_column(2, "Used");
     70     measure_column(3, "Available");
     71     measure_column(4, use_label);
     72     xprintf("%-*s %*s %*s %*s %*s Mounted on\n",
     73             TT.column_widths[0], "Filesystem",
     74             TT.column_widths[1], blocks_label,
     75             TT.column_widths[2], "Used",
     76             TT.column_widths[3], "Available",
     77             TT.column_widths[4], use_label);
     78 
     79     // For the "Use%" column, the trailing % should be inside the column.
     80     TT.column_widths[4]--;
     81   }
     82 }
     83 
     84 static void show_mt(struct mtab_list *mt, int measuring)
     85 {
     86   long long size, used, avail, percent, block;
     87   char *device;
     88 
     89   // Return if it wasn't found (should never happen, but with /etc/mtab...)
     90   if (!mt) return;
     91 
     92   // If we have -t, skip other filesystem types
     93   if (TT.fstype) {
     94     struct arg_list *al;
     95 
     96     for (al = TT.fstype; al; al = al->next)
     97       if (!strcmp(mt->type, al->arg)) break;
     98 
     99     if (!al) return;
    100   }
    101 
    102   // If we don't have -a, skip synthetic filesystems
    103   if (!(toys.optflags & FLAG_a) && !mt->statvfs.f_blocks) return;
    104 
    105   // Figure out how much total/used/free space this filesystem has,
    106   // forcing 64-bit math because filesystems are big now.
    107   block = mt->statvfs.f_bsize ? mt->statvfs.f_bsize : 1;
    108   size = (block * mt->statvfs.f_blocks) / TT.units;
    109   used = (block * (mt->statvfs.f_blocks-mt->statvfs.f_bfree)) / TT.units;
    110   avail = (block*(getuid()?mt->statvfs.f_bavail:mt->statvfs.f_bfree))/TT.units;
    111   if (!(used+avail)) percent = 0;
    112   else {
    113     percent = (used*100)/(used+avail);
    114     if (used*100 != percent*(used+avail)) percent++;
    115   }
    116 
    117   device = *mt->device == '/' ? realpath(mt->device, NULL) : NULL;
    118   if (!device) device = mt->device;
    119 
    120   if (measuring) {
    121     measure_column(0, device);
    122     measure_numeric_column(1, size);
    123     measure_numeric_column(2, used);
    124     measure_numeric_column(3, avail);
    125   } else {
    126     if (!TT.header_shown) show_header();
    127 
    128     if (toys.optflags & (FLAG_H|FLAG_h)) {
    129       char *size_str = toybuf, *used_str = toybuf+64, *avail_str = toybuf+128;
    130       int hr_flags = (toys.optflags & FLAG_H) ? HR_1000 : 0;
    131 
    132       human_readable(size_str, size, hr_flags);
    133       human_readable(used_str, used, hr_flags);
    134       human_readable(avail_str, avail, hr_flags);
    135       xprintf("%-*s %4s  %4s  %4s % 3lld%% %s\n",
    136         TT.column_widths[0], device,
    137         size_str, used_str, avail_str, percent, mt->dir);
    138     } else xprintf("%-*s %*lld %*lld %*lld %*lld%% %s\n",
    139         TT.column_widths[0], device,
    140         TT.column_widths[1], size,
    141         TT.column_widths[2], used,
    142         TT.column_widths[3], avail,
    143         TT.column_widths[4], percent,
    144         mt->dir);
    145   }
    146 
    147   if (device != mt->device) free(device);
    148 }
    149 
    150 void df_main(void)
    151 {
    152   struct mtab_list *mt, *mtstart, *mtend;
    153   int measuring;
    154 
    155   if (toys.optflags & (FLAG_H|FLAG_h)) {
    156     TT.units = 1;
    157   } else {
    158     // Units are 512 bytes if you select "pedantic" without "kilobytes".
    159     TT.units = toys.optflags & FLAG_P ? 512 : 1024;
    160   }
    161 
    162   if (!(mtstart = xgetmountlist(0))) return;
    163   mtend = dlist_terminate(mtstart);
    164 
    165   // If we have a list of filesystems on the command line, loop through them.
    166   if (*toys.optargs) {
    167     // Measure the names then output the table.
    168     for (measuring = 1; measuring >= 0; --measuring) {
    169       char **next;
    170 
    171       for (next = toys.optargs; *next; next++) {
    172         struct stat st;
    173 
    174         // Stat it (complain if we can't).
    175         if (stat(*next, &st)) {
    176           perror_msg("'%s'", *next);
    177           continue;
    178         }
    179 
    180         // Find and display this filesystem.  Use _last_ hit in case of
    181         // overmounts (which is first hit in the reversed list).
    182         for (mt = mtend; mt; mt = mt->prev) {
    183           if (st.st_dev == mt->stat.st_dev
    184               || (st.st_rdev && (st.st_rdev == mt->stat.st_dev)))
    185           {
    186             show_mt(mt, measuring);
    187             break;
    188           }
    189         }
    190       }
    191     }
    192   } else {
    193     // Loop through mount list to filter out overmounts.
    194     for (mt = mtend; mt; mt = mt->prev) {
    195       struct mtab_list *mt2, *mt3;
    196 
    197       // 0:0 is LANANA null device
    198       if (!mt->stat.st_dev) continue;
    199 
    200       // Filter out overmounts.
    201       mt3 = mt;
    202       for (mt2 = mt->prev; mt2; mt2 = mt2->prev) {
    203         if (mt->stat.st_dev == mt2->stat.st_dev) {
    204           // For --bind mounts, show earliest mount
    205           if (!strcmp(mt->device, mt2->device)) {
    206             if (!(toys.optflags & FLAG_a)) mt3->stat.st_dev = 0;
    207             mt3 = mt2;
    208           } else mt2->stat.st_dev = 0;
    209         }
    210       }
    211     }
    212 
    213     // Measure the names then output the table.
    214     for (measuring = 1; measuring >= 0; --measuring) {
    215       // Cosmetic: show filesystems in creation order.
    216       for (mt = mtstart; mt; mt = mt->next) {
    217         if (mt->stat.st_dev) show_mt(mt, measuring);
    218       }
    219     }
    220   }
    221 
    222   if (CFG_TOYBOX_FREE) llist_traverse(mtstart, free);
    223 }
    224