OSDN Git Service

blkid: Add new option -L which pretty-prints the device list
authorTheodore Ts'o <tytso@mit.edu>
Sun, 13 Jul 2008 02:06:30 +0000 (22:06 -0400)
committerTheodore Ts'o <tytso@mit.edu>
Sun, 13 Jul 2008 20:06:47 +0000 (16:06 -0400)
Also accessed via -o list, this new output format is much more
user-friendly and lists whether or not a particular device is mounted.

Addresses-Debian-Bug #490527

Signed-off-by: "Theodore Ts'o" <tytso@mit.edu>
configure
configure.in
misc/Makefile.in
misc/blkid.8.in
misc/blkid.c

index 39d1ca8..0b7a05e 100755 (executable)
--- a/configure
+++ b/configure
@@ -11820,7 +11820,8 @@ fi
 
 
 
-for ac_header in dirent.h errno.h getopt.h malloc.h mntent.h paths.h setjmp.h signal.h stdarg.h stdint.h stdlib.h termios.h unistd.h utime.h linux/fd.h linux/major.h net/if_dl.h netinet/in.h sys/disklabel.h sys/ioctl.h sys/mkdev.h sys/mman.h sys/prctl.h sys/queue.h sys/resource.h sys/select.h sys/socket.h sys/sockio.h sys/stat.h sys/syscall.h sys/sysmacros.h sys/time.h sys/types.h sys/un.h sys/wait.h
+
+for ac_header in dirent.h errno.h getopt.h malloc.h mntent.h paths.h setjmp.h signal.h stdarg.h stdint.h stdlib.h termios.h termio.h unistd.h utime.h linux/fd.h linux/major.h net/if_dl.h netinet/in.h sys/disklabel.h sys/ioctl.h sys/mkdev.h sys/mman.h sys/prctl.h sys/queue.h sys/resource.h sys/select.h sys/socket.h sys/sockio.h sys/stat.h sys/syscall.h sys/sysmacros.h sys/time.h sys/types.h sys/un.h sys/wait.h
 do
 as_ac_Header=`echo "ac_cv_header_$ac_header" | $as_tr_sh`
 if { as_var=$as_ac_Header; eval "test \"\${$as_var+set}\" = set"; }; then
index f8a7edf..4d84eeb 100644 (file)
@@ -624,7 +624,7 @@ if test $cross_compiling = no; then
 else
   AC_CHECK_PROGS(BUILD_CC, gcc cc)
 fi
-AC_CHECK_HEADERS(dirent.h errno.h getopt.h malloc.h mntent.h paths.h setjmp.h signal.h stdarg.h stdint.h stdlib.h termios.h unistd.h utime.h linux/fd.h linux/major.h net/if_dl.h netinet/in.h sys/disklabel.h sys/ioctl.h sys/mkdev.h sys/mman.h sys/prctl.h sys/queue.h sys/resource.h sys/select.h sys/socket.h sys/sockio.h sys/stat.h sys/syscall.h sys/sysmacros.h sys/time.h sys/types.h sys/un.h sys/wait.h)
+AC_CHECK_HEADERS(dirent.h errno.h getopt.h malloc.h mntent.h paths.h setjmp.h signal.h stdarg.h stdint.h stdlib.h termios.h termio.h unistd.h utime.h linux/fd.h linux/major.h net/if_dl.h netinet/in.h sys/disklabel.h sys/ioctl.h sys/mkdev.h sys/mman.h sys/prctl.h sys/queue.h sys/resource.h sys/select.h sys/socket.h sys/sockio.h sys/stat.h sys/syscall.h sys/sysmacros.h sys/time.h sys/types.h sys/un.h sys/wait.h)
 AC_CHECK_HEADERS(sys/disk.h sys/mount.h,,,
 [[
 #if HAVE_SYS_QUEUE_H
index 7232c15..e48a276 100644 (file)
@@ -61,8 +61,8 @@ DEPLIBS= $(LIBEXT2FS) $(LIBCOM_ERR)
 STATIC_LIBS= $(STATIC_LIBEXT2FS) $(STATIC_LIBCOM_ERR) 
 STATIC_DEPLIBS= $(STATIC_LIBEXT2FS) $(STATIC_LIBCOM_ERR) 
 
-LIBS_BLKID= $(LIBBLKID) $(LIBUUID)
-DEPLIBS_BLKID= $(DEPLIBBLKID) $(DEPLIBUUID)
+LIBS_BLKID= $(LIBBLKID) $(LIBUUID) $(LIBEXT2FS) 
+DEPLIBS_BLKID= $(DEPLIBBLKID) $(DEPLIBUUID) $(LIBEXT2FS) 
 
 LIBS_E2P= $(LIBE2P) $(LIBCOM_ERR) 
 DEPLIBS_E2P= $(LIBE2P) $(LIBCOM_ERR) 
index 8a4e971..408dbc9 100644 (file)
@@ -11,7 +11,7 @@ blkid \- command\-line utility to locate/print block device attributes
 .SH SYNOPSIS
 .B blkid 
 [
-.B \-ghlv
+.B \-ghlLv
 ]
 [
 [
@@ -90,10 +90,16 @@ parameter may be
 .IR  value ,
 (only print the value of any tags printed by 
 .BR blkid)
+.IR list ,
+(print the devices in a user friendly format)
 or 
 .I device
 (only print the device name).
 .TP
+.B \-L
+Print the devices in a user-friendly list format.  This is the
+equivalent of using the option \fB-o list\fR.
+.TP
 .B \-s
 Show only the tags for each (specified) device that match
 .IR tag .
index c0fda73..d64f4ce 100644 (file)
 
 #include <stdio.h>
 #include <stdlib.h>
+#include <unistd.h>
 #include <string.h>
+#ifdef HAVE_TERMIOS_H
+#include <termios.h>
+#endif
+#ifdef HAVE_TERMIO_H
+#include <termio.h>
+#endif
+#ifdef HAVE_SYS_IOCTL_H
+#include <sys/ioctl.h>
+#endif
 #ifdef HAVE_GETOPT_H
 #include <getopt.h>
 #else
@@ -22,7 +32,9 @@ extern int optind;
 
 #define OUTPUT_VALUE_ONLY      0x0001
 #define OUTPUT_DEVICE_ONLY     0x0002
+#define OUTPUT_PRETTY_LIST     0x0004
 
+#include "ext2fs/ext2fs.h"
 #include "blkid/blkid.h"
 
 const char *progname = "blkid";
@@ -38,8 +50,8 @@ static void usage(int error)
 
        print_version(out);
        fprintf(out,
-               "usage:\t%s [-c <file>] [-ghl] [-o format] "
-               "[-s <tag>] [-t <token>]\n    [-v] [-w <file>] [dev ...]\n"
+               "usage:\t%s [-c <file>] [-ghlLv] [-o format] "
+               "[-s <tag>] [-t <token>]\n    [-w <file>] [dev ...]\n"
                "\t-c\tcache file (default: /etc/blkid.tab, /dev/null = none)\n"
                "\t-h\tprint this usage message and exit\n"
                "\t-g\tgarbage collect the blkid cache\n"
@@ -78,6 +90,133 @@ static void safe_print(const char *cp, int len)
        }
 }
 
+static int get_terminal_width(void)
+{
+#ifdef TIOCGSIZE
+       struct ttysize  win;
+#endif
+#ifdef TIOCGWINSZ
+       struct winsize  win;
+#endif
+        const char     *cp;
+
+#ifdef TIOCGSIZE
+       if (ioctl (0, TIOCGSIZE, &win) == 0)
+               return (win.ts_cols);
+#endif
+#ifdef TIOCGWINSZ
+       if (ioctl (0, TIOCGWINSZ, &win) == 0)
+               return (win.ws_col);
+#endif
+        cp = getenv("COLUMNS");
+       if (cp)
+               return strtol(cp, NULL, 10);
+       return 80;
+}
+
+static int pretty_print_word(const char *str, int max_len,
+                            int left_len, int overflow_nl)
+{
+       int len = strlen(str) + left_len;
+       int ret = 0;
+
+       fputs(str, stdout);
+       if (overflow_nl && len > max_len) {
+               fputc('\n', stdout);
+               len = 0;
+       } else if (len > max_len)
+               ret = len - max_len;
+       do
+               fputc(' ', stdout);
+       while (len++ < max_len);
+       return ret;
+}
+
+static void pretty_print_line(const char *device, const char *fs_type,
+                             const char *label, const char *mtpt,
+                             const char *uuid)
+{
+       static int device_len = 10, fs_type_len = 7;
+       static int label_len = 8, mtpt_len = 14;
+       static int term_width = -1;
+       int len, w;
+
+       if (term_width < 0)
+               term_width = get_terminal_width();
+
+       if (term_width > 80) {
+               term_width -= 80;
+               w = term_width / 10;
+               if (w > 8)
+                       w = 8;
+               term_width -= 2*w;
+               label_len += w;
+               fs_type_len += w;
+               w = term_width/2;
+               device_len += w;
+               mtpt_len +=w;
+       }
+
+       len = pretty_print_word(device, device_len, 0, 1);
+       len = pretty_print_word(fs_type, fs_type_len, len, 0);
+       len = pretty_print_word(label, label_len, len, 0);
+       len = pretty_print_word(mtpt, mtpt_len, len, 0);
+       fputs(uuid, stdout);
+       fputc('\n', stdout);
+}
+
+static void pretty_print_dev(blkid_dev dev)
+{
+       blkid_tag_iterate       iter;
+       const char              *type, *value, *devname;
+       const char              *uuid = "", *fs_type = "", *label = "";
+       char                    *cp;
+       int                     len, mount_flags;
+       char                    mtpt[80];
+       errcode_t               retval;
+
+       if (dev == NULL) {
+               pretty_print_line("device", "fs_type", "label",
+                                 "mount point", "UUID");
+               for (len=get_terminal_width()-1; len > 0; len--)
+                       fputc('-', stdout);
+               fputc('\n', stdout);
+               return;
+       }
+
+       devname = blkid_dev_devname(dev);
+       if (access(devname, F_OK))
+               return;
+
+       /* Get the uuid, label, type */
+       iter = blkid_tag_iterate_begin(dev);
+       while (blkid_tag_next(iter, &type, &value) == 0) {
+               if (!strcmp(type, "UUID"))
+                       uuid = value;
+               if (!strcmp(type, "TYPE"))
+                       fs_type = value;
+               if (!strcmp(type, "LABEL"))
+                       label = value;
+       }
+       blkid_tag_iterate_end(iter);
+
+       /* Get the mount point */
+       mtpt[0] = 0;
+       retval = ext2fs_check_mount_point(devname, &mount_flags,
+                                         mtpt, sizeof(mtpt));
+       if (retval == 0) {
+               if (mount_flags & EXT2_MF_MOUNTED) {
+                       if (!mtpt[0])
+                               strcpy(mtpt, "(mounted, mtpt unknown)");
+               } else if (mount_flags & EXT2_MF_BUSY)
+                       strcpy(mtpt, "(in use)");
+               else
+                       strcpy(mtpt, "(not mounted)");
+       }
+
+       pretty_print_line(devname, fs_type, label, mtpt, uuid);
+}
+
 static void print_tags(blkid_dev dev, char *show[], int numtag, int output)
 {
        blkid_tag_iterate       iter;
@@ -87,6 +226,11 @@ static void print_tags(blkid_dev dev, char *show[], int numtag, int output)
        if (!dev)
                return;
 
+       if (output & OUTPUT_PRETTY_LIST) {
+               pretty_print_dev(dev);
+               return;
+       }
+
        if (output & OUTPUT_DEVICE_ONLY) {
                printf("%s\n", blkid_dev_devname(dev));
                return;
@@ -137,7 +281,7 @@ int main(int argc, char **argv)
        int lookup = 0, gc = 0;
        int c;
 
-       while ((c = getopt (argc, argv, "c:f:ghlo:s:t:w:v")) != EOF)
+       while ((c = getopt (argc, argv, "c:f:ghlLo:s:t:w:v")) != EOF)
                switch (c) {
                case 'c':
                        if (optarg && !*optarg)
@@ -150,6 +294,9 @@ int main(int argc, char **argv)
                case 'l':
                        lookup++;
                        break;
+               case 'L':
+                       output_format = OUTPUT_PRETTY_LIST;
+                       break;
                case 'g':
                        gc = 1;
                        break;
@@ -158,10 +305,14 @@ int main(int argc, char **argv)
                                output_format = OUTPUT_VALUE_ONLY;
                        else if (!strcmp(optarg, "device"))
                                output_format = OUTPUT_DEVICE_ONLY;
+                       else if (!strcmp(optarg, "list"))
+                               output_format = OUTPUT_PRETTY_LIST;
                        else if (!strcmp(optarg, "full"))
                                output_format = 0;
                        else {
-                               fprintf(stderr, "Invalid output format %s.  Chose from value, device, or full\n", optarg);
+                               fprintf(stderr, "Invalid output format %s. "
+                                       "Choose from value,\n\t"
+                                       "device, list, or full\n", optarg);
                                exit(1);
                        }
                        break;
@@ -214,7 +365,12 @@ int main(int argc, char **argv)
        err = 2;
        if (gc) {
                blkid_gc_cache(cache);
-       } else if (lookup) {
+               goto exit;
+       }
+       if (output_format & OUTPUT_PRETTY_LIST)
+               pretty_print_dev(NULL);
+
+       if (lookup) {
                blkid_dev dev;
 
                if (!search_type) {