/Mikael
diff -ruN linux-2.5.19/drivers/block/floppy.c linux-2.5.19.fix-floppy/drivers/block/floppy.c
--- linux-2.5.19/drivers/block/floppy.c Thu May 30 17:51:50 2002
+++ linux-2.5.19.fix-floppy/drivers/block/floppy.c Fri May 31 01:09:52 2002
@@ -3700,6 +3700,10 @@
UDRS->fd_ref = 0;
}
floppy_release_irq_and_dma();
+
+ /* undo ++bd_openers in floppy_open() */
+ --inode->i_bdev->bd_openers;
+
return 0;
}
@@ -3792,6 +3796,35 @@
invalidate_buffers(mk_kdev(FLOPPY_MAJOR,old_dev));
}
+ /* Problems:
+ * 1. floppy_open() triggers a call to submit_bio(), but
+ * bdev->bd_queue may be NULL since it is not set up until
+ * after floppy_open() has returned. Prior to 2.5.18, NULL
+ * queues were initialised when needed.
+ * 2. fs/block_dev.c:do_open() changes bdev's block size
+ * after floppy_open() has returned. For some reason, this
+ * causes data corruption durings writes.
+ * Workarounds:
+ * 1. bdev->bd_queue is initialised here.
+ * 2. a) Set bdev block size equal to the hardsect/queue size;
+ * this seems to cure the data corruption problem.
+ * b) ++bdev->bd_openers to bypass do_open()'s block size
+ * change. floppy_release() does a --bdev->bd_openers.
+ */
+ {
+ struct block_device *bdev = inode->i_bdev;
+ kdev_t dev = inode->i_rdev;
+ struct blk_dev_struct *p = blk_dev + major(dev);
+
+ if (p->queue)
+ bdev->bd_queue = p->queue(dev);
+ else
+ bdev->bd_queue = &p->request_queue;
+ bdev->bd_block_size = bdev_hardsect_size(bdev);
+ bdev->bd_inode->i_blkbits = blksize_bits(block_size(bdev));
+ ++bdev->bd_openers;
+ }
+
/* Allow ioctls if we have write-permissions even if read-only open.
* Needed so that programs such as fdrawcmd still can work on write
* protected disks */
@@ -4229,8 +4262,6 @@
{
int i,unit,drive;
- register_sys_device(&device_floppy);
-
raw_cmd = NULL;
devfs_handle = devfs_mk_dir (NULL, "floppy", NULL);
@@ -4354,6 +4385,9 @@
register_disk(NULL, mk_kdev(MAJOR_NR,TOMINOR(drive)+i*4),
1, &floppy_fops, 0);
}
+
+ register_sys_device(&device_floppy);
+
return have_no_fdc;
}
@@ -4536,6 +4570,7 @@
{
int dummy;
+ unregister_sys_device(&device_floppy);
devfs_unregister (devfs_handle);
devfs_unregister_blkdev(MAJOR_NR, "fd");
-
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/