On Wed, 2009-03-18 at 14:18 -0700, bugme-daemon@xxxxxxxxxxxxxxxxxxx wrote: > http://bugzilla.kernel.org/show_bug.cgi?id=12893 > > Summary: Race condition can cause two devices to get assigned the > same device minor number. > Product: IO/Storage > Version: 2.5 > KernelVersion: 2.6.27 > Platform: All > OS/Version: Linux > Tree: Mainline > Status: NEW > Severity: normal > Priority: P1 > Component: SCSI > AssignedTo: linux-scsi@xxxxxxxxxxxxxxx > ReportedBy: tdefeo@xxxxxxxxxxxx > > > Latest working kernel version: > Earliest failing kernel version: > Distribution: > Hardware Environment: x86 > Software Environment: > Problem Description: > There is a race condition in scsi/sd.c caused bu the call to ida_get_new() not > being protected by a spinlock. If two devices appear at the same time (which > can happen when booting with multiple USB flash drives installed), occasionally > the timing will be just right such that two devices will get assigned the same > device minor number and hence the same device inode (i.e. /dev/sda). This > causes the scsi subsystem to crash. > > Steps to reproduce: I can reproduce this by booting a system off of a USB flash > drive, with one or more other USB flash drives plugged in. It is sporadic, > depending on the timing, but it will eventually hang up when booting. > > Here is a patch to add the proper locking and fix the problem: > > --- ./linux-2.6.27.orignal/drivers/scsi/sd.c 2008-10-09 17:13:53.000000000 > -0 > 500 > +++ ./linux-2.6.27/drivers/scsi/sd.c 2009-03-18 14:19:42.000000000 -0600 > @@ -99,6 +99,7 @@ > static void sd_print_sense_hdr(struct scsi_disk *, struct scsi_sense_hdr *); > static void sd_print_result(struct scsi_disk *, int); > > +static DEFINE_SPINLOCK(sda_index_lock); // tpd - 3/18/09 - Added. > static DEFINE_IDA(sd_index_ida); > > /* This semaphore is used to mediate the 0->1 reference get in the > @@ -1808,8 +1809,9 @@ > do { > if (!ida_pre_get(&sd_index_ida, GFP_KERNEL)) > goto out_put; > - > + spin_lock(&sda_index_lock); // tpd - 3/18/09 - Added. > error = ida_get_new(&sd_index_ida, &index); > + spin_unlock(&sda_index_lock); // tpd - 3/18/09 - Added. > } while (error == -EAGAIN); > > if (error) > @@ -1883,7 +1885,9 @@ > return 0; > > out_free_index: > + spin_lock(&sda_index_lock); // tpd - 3/18/09 - Added. > ida_remove(&sd_index_ida, index); > + spin_unlock(&sda_index_lock); // tpd - 3/18/09 - Added. > out_put: > put_disk(gd); > out_free: > @@ -1933,7 +1937,9 @@ > struct scsi_disk *sdkp = to_scsi_disk(dev); > struct gendisk *disk = sdkp->disk; > > + spin_lock(&sda_index_lock); // tpd - 3/18/09 - Added. > ida_remove(&sd_index_ida, sdkp->index); > + spin_unlock(&sda_index_lock); // tpd - 3/18/09 - Added. > > disk->private_data = NULL; > put_disk(disk); Very similar patch already upstream and in both stable kernels: http://git.kernel.org/?p=linux/kernel/git/torvalds/linux-2.6.git;a=commit;h=4034cc68157bfa0b6622efe368488d3d3e20f4e6 James -- 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