From 451e80f07d6682dbe1ce5310ce10d1c77281d028 Mon Sep 17 00:00:00 2001 From: mlelstv Date: Sun, 7 Jan 2018 11:37:30 +0000 Subject: [PATCH] Fix block address calculation for bad sectors. --- sys/dev/ata/wd.c | 17 +++++++++++++---- 1 file changed, 13 insertions(+), 4 deletions(-) diff --git a/sys/dev/ata/wd.c b/sys/dev/ata/wd.c index bf766c4b5d50..81629a7fa3b5 100644 --- a/sys/dev/ata/wd.c +++ b/sys/dev/ata/wd.c @@ -1,4 +1,4 @@ -/* $NetBSD: wd.c,v 1.437 2017/12/13 10:24:31 pgoyette Exp $ */ +/* $NetBSD: wd.c,v 1.438 2018/01/07 11:37:30 mlelstv Exp $ */ /* * Copyright (c) 1998, 2001 Manuel Bouyer. All rights reserved. @@ -54,7 +54,7 @@ */ #include -__KERNEL_RCSID(0, "$NetBSD: wd.c,v 1.437 2017/12/13 10:24:31 pgoyette Exp $"); +__KERNEL_RCSID(0, "$NetBSD: wd.c,v 1.438 2018/01/07 11:37:30 mlelstv Exp $"); #include "opt_ata.h" #include "opt_wd.h" @@ -585,9 +585,18 @@ wdstrategy(struct buf *bp) * up failing again. */ if (__predict_false(!SLIST_EMPTY(&wd->sc_bslist))) { + struct disklabel *lp = dksc->sc_dkdev.dk_label; struct disk_badsectors *dbs; - daddr_t maxblk = bp->b_rawblkno + - (bp->b_bcount / wd->sc_blksize) - 1; + daddr_t blkno, maxblk; + + /* convert the block number to absolute */ + if (lp->d_secsize >= DEV_BSIZE) + blkno = bp->b_blkno / (lp->d_secsize / DEV_BSIZE); + else + blkno = bp->b_blkno * (DEV_BSIZE / lp->d_secsize); + if (WDPART(bp->b_dev) != RAW_PART) + blkno += lp->d_partitions[WDPART(bp->b_dev)].p_offset; + maxblk = blkno + (bp->b_bcount / wd->sc_blksize) - 1; mutex_enter(&wd->sc_lock); SLIST_FOREACH(dbs, &wd->sc_bslist, dbs_next)