Commit 3e5edf7c authored by Christoph Hellwig's avatar Christoph Hellwig

[PATCH] fix sd medium removal handling

sdev->access_count is incremented not only by sd but also e.g. by
sg.  Now if sd opens first, then sg, then sd closes and sg last
medium removal will still be prevented.

Add a counter in scsi_disk for doorlocking instead.
parent 9a0754e7
...@@ -76,6 +76,7 @@ struct scsi_disk { ...@@ -76,6 +76,7 @@ struct scsi_disk {
struct scsi_driver *driver; /* always &sd_template */ struct scsi_driver *driver; /* always &sd_template */
struct scsi_device *device; struct scsi_device *device;
struct gendisk *disk; struct gendisk *disk;
unsigned int openers; /* protected by BKL for now, yuck */
sector_t capacity; /* size in 512-byte sectors */ sector_t capacity; /* size in 512-byte sectors */
u32 index; u32 index;
u8 media_present; u8 media_present;
...@@ -396,9 +397,10 @@ static int sd_open(struct inode *inode, struct file *filp) ...@@ -396,9 +397,10 @@ static int sd_open(struct inode *inode, struct file *filp)
if (!sdev->online) if (!sdev->online)
goto error_out; goto error_out;
if (sdev->removable && sdev->access_count == 1) if (!sdkp->openers++ && sdev->removable) {
if (scsi_block_when_processing_errors(sdev)) if (scsi_block_when_processing_errors(sdev))
scsi_set_medium_removal(sdev, SCSI_REMOVAL_PREVENT); scsi_set_medium_removal(sdev, SCSI_REMOVAL_PREVENT);
}
return 0; return 0;
...@@ -421,13 +423,15 @@ static int sd_open(struct inode *inode, struct file *filp) ...@@ -421,13 +423,15 @@ static int sd_open(struct inode *inode, struct file *filp)
static int sd_release(struct inode *inode, struct file *filp) static int sd_release(struct inode *inode, struct file *filp)
{ {
struct gendisk *disk = inode->i_bdev->bd_disk; struct gendisk *disk = inode->i_bdev->bd_disk;
struct scsi_device *sdev = scsi_disk(disk)->device; struct scsi_disk *sdkp = scsi_disk(disk);
struct scsi_device *sdev = sdkp->device;
SCSI_LOG_HLQUEUE(3, printk("sd_release: disk=%s\n", disk->disk_name)); SCSI_LOG_HLQUEUE(3, printk("sd_release: disk=%s\n", disk->disk_name));
if (sdev->removable && sdev->access_count == 1) if (!--sdkp->openers && sdev->removable) {
if (scsi_block_when_processing_errors(sdev)) if (scsi_block_when_processing_errors(sdev))
scsi_set_medium_removal(sdev, SCSI_REMOVAL_ALLOW); scsi_set_medium_removal(sdev, SCSI_REMOVAL_ALLOW);
}
/* /*
* XXX and what if there are packets in flight and this close() * XXX and what if there are packets in flight and this close()
...@@ -1331,6 +1335,7 @@ static int sd_probe(struct device *dev) ...@@ -1331,6 +1335,7 @@ static int sd_probe(struct device *dev)
sdkp->driver = &sd_template; sdkp->driver = &sd_template;
sdkp->disk = gd; sdkp->disk = gd;
sdkp->index = index; sdkp->index = index;
sdkp->openers = 0;
gd->major = sd_major(index >> 4); gd->major = sd_major(index >> 4);
gd->first_minor = (index & 15) << 4; gd->first_minor = (index & 15) << 4;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment