aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/scsi
diff options
context:
space:
mode:
authorArnd Bergmann <arnd@arndb.de>2010-07-07 10:51:29 -0400
committerJens Axboe <jaxboe@fusionio.com>2010-08-07 12:26:08 -0400
commit409f3499a2cfcd1e9c2857c53af7fcce069f027f (patch)
tree7275165954c7f1143d219f4b141888d6c0a5a4ab /drivers/scsi
parent15392efb9d427482754f6d267262452878667499 (diff)
scsi/sd: remove big kernel lock
Every user of the BKL in the sd driver is the result of the pushdown from the block layer into the open/close/ioctl functions. The only place that used to rely on the BKL is the sdkp->openers variable, which gets converted into an atomic_t. Nothing else seems to rely on the BKL, since the functions do not touch global data without holding another lock, and the open/close functions are still protected from concurrent execution using the bdev->bd_mutex. Signed-off-by: Arnd Bergmann <arnd@arndb.de> Cc: linux-scsi@vger.kernel.org Cc: "James E.J. Bottomley" <James.Bottomley@suse.de> Acked-by: Christoph Hellwig <hch@infradead.org> Signed-off-by: Jens Axboe <jaxboe@fusionio.com>
Diffstat (limited to 'drivers/scsi')
-rw-r--r--drivers/scsi/sd.c17
-rw-r--r--drivers/scsi/sd.h2
2 files changed, 8 insertions, 11 deletions
diff --git a/drivers/scsi/sd.c b/drivers/scsi/sd.c
index 01680c7c8507..fc5d69a84af5 100644
--- a/drivers/scsi/sd.c
+++ b/drivers/scsi/sd.c
@@ -783,6 +783,8 @@ static int sd_prep_fn(struct request_queue *q, struct request *rq)
783 * or from within the kernel (e.g. as a result of a mount(1) ). 783 * or from within the kernel (e.g. as a result of a mount(1) ).
784 * In the latter case @inode and @filp carry an abridged amount 784 * In the latter case @inode and @filp carry an abridged amount
785 * of information as noted above. 785 * of information as noted above.
786 *
787 * Locking: called with bdev->bd_mutex held.
786 **/ 788 **/
787static int sd_open(struct block_device *bdev, fmode_t mode) 789static int sd_open(struct block_device *bdev, fmode_t mode)
788{ 790{
@@ -795,7 +797,6 @@ static int sd_open(struct block_device *bdev, fmode_t mode)
795 797
796 SCSI_LOG_HLQUEUE(3, sd_printk(KERN_INFO, sdkp, "sd_open\n")); 798 SCSI_LOG_HLQUEUE(3, sd_printk(KERN_INFO, sdkp, "sd_open\n"));
797 799
798 lock_kernel();
799 sdev = sdkp->device; 800 sdev = sdkp->device;
800 801
801 /* 802 /*
@@ -834,17 +835,15 @@ static int sd_open(struct block_device *bdev, fmode_t mode)
834 if (!scsi_device_online(sdev)) 835 if (!scsi_device_online(sdev))
835 goto error_out; 836 goto error_out;
836 837
837 if (!sdkp->openers++ && sdev->removable) { 838 if ((atomic_inc_return(&sdkp->openers) == 1) && sdev->removable) {
838 if (scsi_block_when_processing_errors(sdev)) 839 if (scsi_block_when_processing_errors(sdev))
839 scsi_set_medium_removal(sdev, SCSI_REMOVAL_PREVENT); 840 scsi_set_medium_removal(sdev, SCSI_REMOVAL_PREVENT);
840 } 841 }
841 842
842 unlock_kernel();
843 return 0; 843 return 0;
844 844
845error_out: 845error_out:
846 scsi_disk_put(sdkp); 846 scsi_disk_put(sdkp);
847 unlock_kernel();
848 return retval; 847 return retval;
849} 848}
850 849
@@ -858,6 +857,8 @@ error_out:
858 * 857 *
859 * Note: may block (uninterruptible) if error recovery is underway 858 * Note: may block (uninterruptible) if error recovery is underway
860 * on this disk. 859 * on this disk.
860 *
861 * Locking: called with bdev->bd_mutex held.
861 **/ 862 **/
862static int sd_release(struct gendisk *disk, fmode_t mode) 863static int sd_release(struct gendisk *disk, fmode_t mode)
863{ 864{
@@ -866,8 +867,7 @@ static int sd_release(struct gendisk *disk, fmode_t mode)
866 867
867 SCSI_LOG_HLQUEUE(3, sd_printk(KERN_INFO, sdkp, "sd_release\n")); 868 SCSI_LOG_HLQUEUE(3, sd_printk(KERN_INFO, sdkp, "sd_release\n"));
868 869
869 lock_kernel(); 870 if (atomic_dec_return(&sdkp->openers) && sdev->removable) {
870 if (!--sdkp->openers && sdev->removable) {
871 if (scsi_block_when_processing_errors(sdev)) 871 if (scsi_block_when_processing_errors(sdev))
872 scsi_set_medium_removal(sdev, SCSI_REMOVAL_ALLOW); 872 scsi_set_medium_removal(sdev, SCSI_REMOVAL_ALLOW);
873 } 873 }
@@ -877,7 +877,6 @@ static int sd_release(struct gendisk *disk, fmode_t mode)
877 * XXX is followed by a "rmmod sd_mod"? 877 * XXX is followed by a "rmmod sd_mod"?
878 */ 878 */
879 scsi_disk_put(sdkp); 879 scsi_disk_put(sdkp);
880 unlock_kernel();
881 return 0; 880 return 0;
882} 881}
883 882
@@ -930,7 +929,6 @@ static int sd_ioctl(struct block_device *bdev, fmode_t mode,
930 SCSI_LOG_IOCTL(1, printk("sd_ioctl: disk=%s, cmd=0x%x\n", 929 SCSI_LOG_IOCTL(1, printk("sd_ioctl: disk=%s, cmd=0x%x\n",
931 disk->disk_name, cmd)); 930 disk->disk_name, cmd));
932 931
933 lock_kernel();
934 /* 932 /*
935 * If we are in the middle of error recovery, don't let anyone 933 * If we are in the middle of error recovery, don't let anyone
936 * else try and use this device. Also, if error recovery fails, it 934 * else try and use this device. Also, if error recovery fails, it
@@ -960,7 +958,6 @@ static int sd_ioctl(struct block_device *bdev, fmode_t mode,
960 break; 958 break;
961 } 959 }
962out: 960out:
963 unlock_kernel();
964 return error; 961 return error;
965} 962}
966 963
@@ -2346,7 +2343,7 @@ static int sd_probe(struct device *dev)
2346 sdkp->driver = &sd_template; 2343 sdkp->driver = &sd_template;
2347 sdkp->disk = gd; 2344 sdkp->disk = gd;
2348 sdkp->index = index; 2345 sdkp->index = index;
2349 sdkp->openers = 0; 2346 atomic_set(&sdkp->openers, 0);
2350 sdkp->previous_state = 1; 2347 sdkp->previous_state = 1;
2351 2348
2352 if (!sdp->request_queue->rq_timeout) { 2349 if (!sdp->request_queue->rq_timeout) {
diff --git a/drivers/scsi/sd.h b/drivers/scsi/sd.h
index 43d3caf268ef..f81a9309e6de 100644
--- a/drivers/scsi/sd.h
+++ b/drivers/scsi/sd.h
@@ -47,7 +47,7 @@ struct scsi_disk {
47 struct scsi_device *device; 47 struct scsi_device *device;
48 struct device dev; 48 struct device dev;
49 struct gendisk *disk; 49 struct gendisk *disk;
50 unsigned int openers; /* protected by BKL for now, yuck */ 50 atomic_t openers;
51 sector_t capacity; /* size in 512-byte sectors */ 51 sector_t capacity; /* size in 512-byte sectors */
52 u32 index; 52 u32 index;
53 unsigned short hw_sector_size; 53 unsigned short hw_sector_size;