pktcdvd: don't set max_hw_sectors on the underlying device
authorChristoph Hellwig <hch@lst.de>
Thu, 29 Feb 2024 14:44:08 +0000 (06:44 -0800)
committerJens Axboe <axboe@kernel.dk>
Fri, 1 Mar 2024 16:08:00 +0000 (09:08 -0700)
pktcdvd sets max_hw_sectors on the queue of the underlying device that
it doesn't own (and doesn't reset it ever) since the driver was merged.
This can create all kinds of problems as the underlying driver doesn't
even know about it changing the limit.

As the state purpose is to not create I/Os larger than a single frame,
and pktcdvd never builds bios larger than that, just set REQ_NOMERGE
on the bios it submits so that largers I/Os never get built.

Note: I don't have packet writing hardware, so this is compile tested
only.

Signed-off-by: Christoph Hellwig <hch@lst.de>
Link: https://lore.kernel.org/r/20240229144408.1047967-1-hch@lst.de
Signed-off-by: Jens Axboe <axboe@kernel.dk>
drivers/block/pktcdvd.c

index 12fcc881b04f5494941f2a4ab923facc6e0138d0..9071c4ebc1b90115dc180eb594a63b80b809e56a 100644 (file)
@@ -828,6 +828,12 @@ static noinline_for_stack int pkt_set_speed(struct pktcdvd_device *pd,
  */
 static void pkt_queue_bio(struct pktcdvd_device *pd, struct bio *bio)
 {
+       /*
+        * Some CDRW drives can not handle writes larger than one packet,
+        * even if the size is a multiple of the packet size.
+        */
+       bio->bi_opf |= REQ_NOMERGE;
+
        spin_lock(&pd->iosched.lock);
        if (bio_data_dir(bio) == READ)
                bio_list_add(&pd->iosched.read_queue, bio);
@@ -2191,11 +2197,6 @@ static int pkt_open_dev(struct pktcdvd_device *pd, bool write)
                ret = pkt_open_write(pd);
                if (ret)
                        goto out_putdev;
-               /*
-                * Some CDRW drives can not handle writes larger than one packet,
-                * even if the size is a multiple of the packet size.
-                */
-               blk_queue_max_hw_sectors(q, pd->settings.size);
                set_bit(PACKET_WRITABLE, &pd->flags);
        } else {
                pkt_set_speed(pd, MAX_SPEED, MAX_SPEED);