This patch allows the user to set an "alias name" of the disk via sysfs interface. A raw device name of a disk does not always point a same disk each boot-up time. Therefore, users have to use persistent device names, which udev create to always access the same disk. However, kernel messages still display device names. This patch adds a new attribute "alias name" in gendisk structure. And if users set their preferred name to an "alias name" of the disk, it would be appeared in kernel messages. A disk can have an "alias name" which length is up to 255bytes. Users can use alphabets, numbers, '-' and '_' in alias name. Suggested-by: James Bottomley <James.Bottomley@xxxxxxxxxxxxxxxxxxxxx> Suggested-by: Jon Masters <jcm@xxxxxxxxxx> Signed-off-by: Nao Nishijima <nao.nishijima.xt@xxxxxxxxxxx> --- Documentation/ABI/testing/sysfs-block | 15 ++++++ block/genhd.c | 85 +++++++++++++++++++++++++++++++++ include/linux/genhd.h | 4 ++ 3 files changed, 104 insertions(+), 0 deletions(-) diff --git a/Documentation/ABI/testing/sysfs-block b/Documentation/ABI/testing/sysfs-block index c1eb41c..07c4fcd 100644 --- a/Documentation/ABI/testing/sysfs-block +++ b/Documentation/ABI/testing/sysfs-block @@ -206,3 +206,18 @@ Description: when a discarded area is read the discard_zeroes_data parameter will be set to one. Otherwise it will be 0 and the result of reading a discarded area is undefined. + +What: /sys/block/<disk>/alias_name +Date: June 2011 +Contact: Nao Nishijima <nao.nishijima.xt@xxxxxxxxxxx> +Description: + A raw device name of a disk does not always point a same disk + each boot-up time. Therefore, users have to use persistent + device names, which udev creates when the kernel finds a disk, + instead of raw device name. However, kernel doesn't show those + persistent names on its message (e.g. dmesg). + This file can store an alias name of the disk and it would be + appeared in kernel messages if it is set. A disk can have an + alias name which length is up to 255bytes. Users can use + use alphabets, numbers, "-" and "_" in alias name and can + change it anytime. diff --git a/block/genhd.c b/block/genhd.c index 3608289..4363947 100644 --- a/block/genhd.c +++ b/block/genhd.c @@ -19,6 +19,7 @@ #include <linux/mutex.h> #include <linux/idr.h> #include <linux/log2.h> +#include <linux/ctype.h> #include "blk.h" @@ -909,6 +910,87 @@ static int __init genhd_device_init(void) subsys_initcall(genhd_device_init); +static ssize_t alias_name_show(struct device *dev, + struct device_attribute *attr, char *buf) +{ + struct gendisk *disk = dev_to_disk(dev); + ssize_t ret = 0; + + if (disk->alias_name) + ret = snprintf(buf, ALIAS_NAME_LEN + 1, "%s\n", + disk->alias_name); + return ret; +} + +static ssize_t alias_name_store(struct device *dev, + struct device_attribute *attr, const char *buf, + size_t count) +{ + struct gendisk *disk = dev_to_disk(dev); + struct disk_part_iter piter; + struct hd_struct *part; + + char *new_alias_name; + char *envp[] = { NULL, NULL }; + unsigned char c; + int idx; + ssize_t ret = count; + + if (!count) + return -EINVAL; + + if (count >= ALIAS_NAME_LEN) { + printk(KERN_ERR "alias_name: alias name is too long\n"); + return -EINVAL; + } + + /* Validation check */ + for (idx = 0; idx < count; idx++) { + c = buf[idx]; + if (idx == count - 1 && c == '\n') + break; + if (!isalnum(c) && c != '_' && c != '-') { + printk(KERN_ERR "alias_name: invalid alias name\n"); + return -EINVAL; + } + } + + new_alias_name = kasprintf(GFP_KERNEL, "%s", buf); + if (!new_alias_name) + return -ENOMEM; + + if (new_alias_name[count - 1] == '\n') + new_alias_name[count - 1] = '\0'; + + envp[0] = kasprintf(GFP_KERNEL, "ALIAS=%s", new_alias_name); + if (!envp[0]) { + kfree(new_alias_name); + return -ENOMEM; + } + + if (disk->alias_name[0] != '\0') + printk(KERN_INFO "alias_name: assigned %s to %s (previous " + "alias_name was %s)\n", + new_alias_name, disk->disk_name, disk->alias_name); + else + printk(KERN_INFO "alias_name: assigned %s to %s\n", + new_alias_name, disk->disk_name); + + snprintf(disk->alias_name, ALIAS_NAME_LEN, "%s", new_alias_name); + + kobject_uevent_env(&dev->kobj, KOBJ_CHANGE, envp); + + /* announce possible partitions */ + disk_part_iter_init(&piter, disk, 0); + while ((part = disk_part_iter_next(&piter))) + kobject_uevent_env(&part_to_dev(part)->kobj, KOBJ_CHANGE, envp); + disk_part_iter_exit(&piter); + + kfree(envp[0]); + kfree(new_alias_name); + return ret; +} + static ssize_t disk_range_show(struct device *dev, struct device_attribute *attr, char *buf) { @@ -968,6 +1050,8 @@ static ssize_t disk_discard_alignment_show(struct device *dev, return sprintf(buf, "%d\n", queue_discard_alignment(disk->queue)); } +static DEVICE_ATTR(alias_name, S_IRUGO|S_IWUSR, alias_name_show, + alias_name_store); static DEVICE_ATTR(range, S_IRUGO, disk_range_show, NULL); static DEVICE_ATTR(ext_range, S_IRUGO, disk_ext_range_show, NULL); static DEVICE_ATTR(removable, S_IRUGO, disk_removable_show, NULL); @@ -990,6 +1074,7 @@ static struct device_attribute dev_attr_fail_timeout = #endif static struct attribute *disk_attrs[] = { + &dev_attr_alias_name.attr, &dev_attr_range.attr, &dev_attr_ext_range.attr, &dev_attr_removable.attr, diff --git a/include/linux/genhd.h b/include/linux/genhd.h index 300d758..a3c219d 100644 --- a/include/linux/genhd.h +++ b/include/linux/genhd.h @@ -21,6 +21,8 @@ #define dev_to_part(device) container_of((device), struct hd_struct, __dev) #define disk_to_dev(disk) (&(disk)->part0.__dev) #define part_to_dev(part) (&((part)->__dev)) +#define alias_name(disk) ((disk)->alias_name[0] != '\0' \ + ? (disk)->alias_name : (disk)->disk_name) extern struct device_type part_type; extern struct kobject *block_depr; @@ -58,6 +60,7 @@ enum { #define DISK_MAX_PARTS 256 #define DISK_NAME_LEN 32 +#define ALIAS_NAME_LEN 256 #include <linux/major.h> #include <linux/device.h> @@ -162,6 +165,7 @@ struct gendisk { * disks that can't be partitioned. */ char disk_name[DISK_NAME_LEN]; /* name of major driver */ + char alias_name[ALIAS_NAME_LEN];/* alias name of disk */ char *(*devnode)(struct gendisk *gd, mode_t *mode); unsigned int events; /* supported events */ -- To unsubscribe from this list: send the line "unsubscribe linux-scsi" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html