Suppose that in a program I have an open file descriptor for a device,
and I want to find the /sys/block information for this device.
There is currently no direct way to do this.  I need to read
   /sys/block/*/dev, /sys/block/*/*/dev
and match major/minor numbers with the result from fstat.

I would like a more direct mechanism.

The following patch is a proposal for such a mechanism.

It provides an 'ioctl' which returns then 'name' of the device, as
generated by bdevname.  This is the same name that is used to create
entries in sysfs.
For a partition of a device, it returns 'device/partition'.

There is also a patch to 'blockdev' so you can test it.

dell:~# ./blockdev --name /dev/sda1
sda/sda1
dell:~# ./blockdev --name 
/dev/disk/by-uuid/f1394e26-6e0d-48bf-9fb7-1321e06efba3 
sde

My particular need for this is: given a device, find out what md/dm
array it is a member of.  Given this ioctl, I can then look up
 /sys/block/$DEVNAME/holders
and see what is in there.

Any objection to this becoming a new ioctl?  Is 'BLKGETNAME' and
adequate name?

Thanks,
NeilBrown

----------------
Subject: Allow mapping from block-device-file to sysfs entry.

New ioctl returns name provided by bdevname for non-partitions,
or "parentname/bdevname" for partitions.

Signed-off-by: Neil Brown <[EMAIL PROTECTED]>

### Diffstat output
 ./block/ioctl.c      |   13 +++++++++++++
 ./include/linux/fs.h |    6 +++---
 2 files changed, 16 insertions(+), 3 deletions(-)

diff .prev/block/ioctl.c ./block/ioctl.c
--- .prev/block/ioctl.c 2007-08-07 14:53:07.000000000 +1000
+++ ./block/ioctl.c     2007-08-07 15:38:46.000000000 +1000
@@ -223,8 +223,21 @@ int blkdev_ioctl(struct inode *inode, st
        struct block_device *bdev = inode->i_bdev;
        struct gendisk *disk = bdev->bd_disk;
        int ret, n;
+       char b[BDEVNAME_SIZE*2];
 
        switch(cmd) {
+       case BLKGETNAME:
+               memset(b, 0, sizeof(b));
+               bdevname(bdev->bd_contains, b);
+               if (bdev->bd_contains != bdev) {
+                       char *e = b + strlen(b);
+                       *e++ = '/';
+                       bdevname(bdev, e);
+               }
+               if (copy_to_user((char __user *)arg, b, BDEVNAME_SIZE*2))
+                       return -EFAULT;
+               return 0;
+
        case BLKFLSBUF:
                if (!capable(CAP_SYS_ADMIN))
                        return -EACCES;

diff .prev/include/linux/fs.h ./include/linux/fs.h
--- .prev/include/linux/fs.h    2007-08-07 14:51:08.000000000 +1000
+++ ./include/linux/fs.h        2007-08-07 15:38:49.000000000 +1000
@@ -223,6 +223,9 @@ extern int dir_notify_enable;
 #define BLKTRACESTOP _IO(0x12,117)
 #define BLKTRACETEARDOWN _IO(0x12,118)
 
+#define BDEVNAME_SIZE  32      /* Largest string for a blockdev identifier */
+#define BLKGETNAME _IOR(0x12, 119, char [BDEVNAME_SIZE*2])
+
 #define BMAP_IOCTL 1           /* obsolete - kept for compatibility */
 #define FIBMAP    _IO(0x00,1)  /* bmap access */
 #define FIGETBSZ   _IO(0x00,2) /* get the block size used for bmap */
@@ -1590,9 +1593,6 @@ extern void unregister_chrdev_region(dev
 extern int chrdev_open(struct inode *, struct file *);
 extern void chrdev_show(struct seq_file *,off_t);
 
-/* fs/block_dev.c */
-#define BDEVNAME_SIZE  32      /* Largest string for a blockdev identifier */
-
 #ifdef CONFIG_BLOCK
 #define BLKDEV_MAJOR_HASH_SIZE 255
 extern const char *__bdevname(dev_t, char *buffer);

--------------
--- blockdev.c.orig     2007-08-07 15:06:00.000000000 +1000
+++ blockdev.c  2007-08-07 15:39:04.000000000 +1000
@@ -29,6 +29,7 @@
 #define BLKBSZGET  _IOR(0x12,112,size_t)
 #define BLKBSZSET  _IOW(0x12,113,size_t)
 #define BLKGETSIZE64 _IOR(0x12,114,size_t)
+#define BLKGETNAME _IOR(0x12,119,char [64])
 #endif
 
 /* Maybe <linux/hdreg.h> could be included */
@@ -56,6 +57,7 @@
 #define ARGINTG        4
 #define ARGLINTG 5
 #define ARGLLINTG 6
+#define ARGSTR64 7
        long argval;
        char *argname;
        char *help;
@@ -101,6 +103,10 @@
        { "--rereadpt", "BLKRRPART", BLKRRPART, ARGNONE, 0, NULL,
          N_("reread partition table") },
 #endif
+#ifdef BLKGETNAME
+       { "--name", "BLKGETNAME", BLKGETNAME, ARGSTR64, 0, NULL,
+         N_("get device name") },
+#endif
 };
 
 #define SIZE(a)        (sizeof(a)/sizeof((a)[0]))
@@ -242,6 +248,7 @@
        int iarg;
        long larg;
        long long llarg;
+       char str64arg[64];
        int verbose = 0;
 
        for (i = 1; i < d; i++) {
@@ -306,6 +313,9 @@
                        llarg = bdcms[j].argval;
                        res = ioctl(fd, bdcms[j].ioc, &llarg);
                        break;
+               case ARGSTR64:
+                       res = ioctl(fd, bdcms[j].ioc, &str64arg);
+                       break;
                }
                if (res == -1) {
                        perror(bdcms[j].iocname);
@@ -332,6 +342,13 @@
                        else
                                printf("%lld\n", llarg);
                        break;
+               case ARGSTR64:
+                       if (verbose)
+                               printf("%s: %.64s\n", _(bdcms[j].help),
+                                      str64arg);
+                       else
+                               printf("%.64s\n", str64arg);
+                       break;
                default:
                        if (verbose)
                                printf(_("%s succeeded.\n"), _(bdcms[j].help));
-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to [EMAIL PROTECTED]
More majordomo info at  http://vger.kernel.org/majordomo-info.html
Please read the FAQ at  http://www.tux.org/lkml/

Reply via email to