summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorSamuel Thibault <samuel.thibault@ens-lyon.org>2013-06-24 15:02:09 +0200
committerSamuel Thibault <samuel.thibault@ens-lyon.org>2013-06-24 15:02:09 +0200
commitf925d0efa2161aef215f858347a9036f1153c4b2 (patch)
tree0d3a846c72c1ea7f655d6bc65b65af2ae484a28c
parenteb2dc3e7ed87ef45dc2dafba22bbe7a3fd7ca388 (diff)
Add partitioning reload support to ahci
* linux/dev/drivers/block/ahci.c (port): Add `gd' field. (ahci_ioctl): New function. (ahci_fops): Fill `ioctl' field with `ahci_ioctl'. (ahci_probe_pci): Fill `gd' field with `gd'.
-rw-r--r--linux/dev/drivers/block/ahci.c35
1 files changed, 33 insertions, 2 deletions
diff --git a/linux/dev/drivers/block/ahci.c b/linux/dev/drivers/block/ahci.c
index ceded7c..2c573ac 100644
--- a/linux/dev/drivers/block/ahci.c
+++ b/linux/dev/drivers/block/ahci.c
@@ -248,6 +248,7 @@ static struct port {
unsigned lba48; /* Whether LBA48 is supported */
unsigned identify; /* Whether we are just identifying
at boot */
+ struct gendisk *gd;
} ports[MAX_PORTS];
@@ -471,6 +472,34 @@ static void ahci_interrupt (int irq, void *host, struct pt_regs *regs)
/* unlock */
}
+static int ahci_ioctl (struct inode *inode, struct file *file,
+ unsigned int cmd, unsigned long arg)
+{
+ int major, unit;
+
+ if (!inode || !inode->i_rdev)
+ return -EINVAL;
+
+ major = MAJOR(inode->i_rdev);
+ if (major != MAJOR_NR)
+ return -ENOTTY;
+
+ unit = DEVICE_NR(inode->i_rdev);
+ if (unit >= MAX_PORTS)
+ return -EINVAL;
+
+ switch (cmd) {
+ case BLKRRPART:
+ if (!suser()) return -EACCES;
+ if (!ports[unit].gd)
+ return -EINVAL;
+ resetup_one_dev(ports[unit].gd, unit);
+ return 0;
+ default:
+ return -EPERM;
+ }
+}
+
static int ahci_open (struct inode *inode, struct file *file)
{
int target;
@@ -504,7 +533,7 @@ static struct file_operations ahci_fops = {
.write = block_write,
.readdir = NULL,
.select = NULL,
- .ioctl = NULL,
+ .ioctl = ahci_ioctl,
.mmap = NULL,
.open = ahci_open,
.release = ahci_release,
@@ -875,8 +904,10 @@ void ahci_probe_pci(void)
memset(gd->part, 0, nminors * sizeof(*gd->part));
- for (unit = 0; unit < nports; unit++)
+ for (unit = 0; unit < nports; unit++) {
+ ports[unit].gd = gd;
ports[unit].part = &gd->part[unit << PARTN_BITS];
+ }
gd->major = MAJOR_NR;
gd->major_name = "sd";