Source-Changes-HG archive
[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index][Old Index]
[src/trunk]: src/sys/dev/ata account for already transferred data (partially ...
details: https://anonhg.NetBSD.org/src/rev/4c507b1d1197
branches: trunk
changeset: 972372:4c507b1d1197
user: jdolecek <jdolecek%NetBSD.org@localhost>
date: Sun May 24 22:12:29 2020 +0000
description:
account for already transferred data (partially done I/O) when
retrying an xfer, to avoid reading/writing data from/to wrong offset,
and eventually beyond the end of data buffer
fixes data corruption under QEMU observed by Paul Ripke for emulated
IDE drives
diffstat:
sys/dev/ata/wd.c | 18 +++++++++++++-----
1 files changed, 13 insertions(+), 5 deletions(-)
diffs (54 lines):
diff -r 9eac4f5d8aef -r 4c507b1d1197 sys/dev/ata/wd.c
--- a/sys/dev/ata/wd.c Sun May 24 21:09:49 2020 +0000
+++ b/sys/dev/ata/wd.c Sun May 24 22:12:29 2020 +0000
@@ -1,4 +1,4 @@
-/* $NetBSD: wd.c,v 1.462 2020/04/13 10:49:34 jdolecek Exp $ */
+/* $NetBSD: wd.c,v 1.463 2020/05/24 22:12:29 jdolecek Exp $ */
/*
* Copyright (c) 1998, 2001 Manuel Bouyer. All rights reserved.
@@ -54,7 +54,7 @@
*/
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: wd.c,v 1.462 2020/04/13 10:49:34 jdolecek Exp $");
+__KERNEL_RCSID(0, "$NetBSD: wd.c,v 1.463 2020/05/24 22:12:29 jdolecek Exp $");
#include "opt_ata.h"
#include "opt_wd.h"
@@ -663,6 +663,7 @@
wdstart1(struct wd_softc *wd, struct buf *bp, struct ata_xfer *xfer)
{
struct dk_softc *dksc = &wd->sc_dksc;
+ const uint32_t secsize = dksc->sc_dkdev.dk_geom.dg_secsize;
KASSERT(bp == xfer->c_bio.bp || xfer->c_bio.bp == NULL);
KASSERT((xfer->c_flags & (C_WAITACT|C_FREE)) == 0);
@@ -680,6 +681,14 @@
xfer->c_bio.blkdone = 0;
xfer->c_bio.bp = bp;
+ /* Adjust blkno and bcount if xfer has been already partially done */
+ if (__predict_false(xfer->c_skip > 0)) {
+ KASSERT(xfer->c_skip < xfer->c_bio.bcount);
+ KASSERT((xfer->c_skip % secsize) == 0);
+ xfer->c_bio.bcount -= xfer->c_skip;
+ xfer->c_bio.blkno += xfer->c_skip / secsize;
+ }
+
#ifdef WD_CHAOS_MONKEY
/*
* Override blkno to be over device capacity to trigger error,
@@ -714,10 +723,9 @@
* and needed by transfer offset or size.
*/
if (wd->sc_flags & WDF_LBA48 &&
- (((xfer->c_bio.blkno +
- xfer->c_bio.bcount / dksc->sc_dkdev.dk_geom.dg_secsize) >
+ (((xfer->c_bio.blkno + xfer->c_bio.bcount / secsize) >
wd->sc_capacity28) ||
- ((xfer->c_bio.bcount / dksc->sc_dkdev.dk_geom.dg_secsize) > 128)))
+ ((xfer->c_bio.bcount / secsize) > 128)))
xfer->c_bio.flags |= ATA_LBA48;
/*
Home |
Main Index |
Thread Index |
Old Index