NetBSD/sys/dev/ata/wd.c

2414 lines
61 KiB
C
Raw Normal View History

/* $NetBSD: wd.c,v 1.432 2017/10/14 13:20:32 jdolecek Exp $ */
1998-10-12 20:09:10 +04:00
/*
2001-12-03 03:20:22 +03:00
* Copyright (c) 1998, 2001 Manuel Bouyer. All rights reserved.
1998-10-12 20:09:10 +04:00
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
1998-10-12 20:09:10 +04:00
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
1994-10-27 07:14:23 +03:00
1998-08-15 14:10:47 +04:00
/*-
2004-06-22 23:20:56 +04:00
* Copyright (c) 1998, 2003, 2004 The NetBSD Foundation, Inc.
1998-08-15 14:10:47 +04:00
* All rights reserved.
*
1998-08-15 14:10:47 +04:00
* This code is derived from software contributed to The NetBSD Foundation
* by Charles M. Hannum and by Onno van der Linden.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
1998-08-15 14:10:47 +04:00
* THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
1993-03-21 12:45:37 +03:00
*/
2001-11-13 15:51:12 +03:00
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: wd.c,v 1.432 2017/10/14 13:20:32 jdolecek Exp $");
2001-11-13 15:51:12 +03:00
#include "opt_ata.h"
#include "opt_wd.h"
1998-10-12 20:09:10 +04:00
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/kernel.h>
#include <sys/conf.h>
#include <sys/file.h>
#include <sys/stat.h>
#include <sys/ioctl.h>
#include <sys/buf.h>
#include <sys/bufq.h>
#include <sys/uio.h>
#include <sys/malloc.h>
#include <sys/device.h>
1994-10-20 17:08:07 +03:00
#include <sys/disklabel.h>
#include <sys/disk.h>
#include <sys/syslog.h>
1996-04-29 23:50:47 +04:00
#include <sys/proc.h>
#include <sys/reboot.h>
#include <sys/vnode.h>
#include <sys/rndsource.h>
#include <sys/intr.h>
#include <sys/bus.h>
1998-10-12 20:09:10 +04:00
#include <dev/ata/atareg.h>
#include <dev/ata/atavar.h>
#include <dev/ata/wdvar.h>
#include <dev/ic/wdcreg.h>
#include <sys/ataio.h>
#include "locators.h"
1993-03-21 12:45:37 +03:00
#include <prop/proplib.h>
#define WDIORETRIES_SINGLE 4 /* number of retries for single-sector */
1998-10-12 20:09:10 +04:00
#define WDIORETRIES 5 /* number of retries before giving up */
#define RECOVERYTIME hz/2 /* time to wait before retrying a cmd */
1998-10-12 20:09:10 +04:00
#define WDUNIT(dev) DISKUNIT(dev)
#define WDPART(dev) DISKPART(dev)
#define WDMINOR(unit, part) DISKMINOR(unit, part)
#define MAKEWDDEV(maj, unit, part) MAKEDISKDEV(maj, unit, part)
#define WDLABELDEV(dev) (MAKEWDDEV(major(dev), WDUNIT(dev), RAW_PART))
1993-03-21 12:45:37 +03:00
1998-10-12 20:09:10 +04:00
#define DEBUG_INTR 0x01
#define DEBUG_XFERS 0x02
#define DEBUG_STATUS 0x04
#define DEBUG_FUNCS 0x08
#define DEBUG_PROBE 0x10
2004-08-13 08:10:49 +04:00
#ifdef ATADEBUG
2005-02-27 03:26:58 +03:00
int wdcdebug_wd_mask = 0x0;
2004-08-13 08:10:49 +04:00
#define ATADEBUG_PRINT(args, level) \
1998-10-12 20:09:10 +04:00
if (wdcdebug_wd_mask & (level)) \
printf args
#else
2004-08-13 08:10:49 +04:00
#define ATADEBUG_PRINT(args, level)
#endif
int wdprobe(device_t, cfdata_t, void *);
void wdattach(device_t, device_t, void *);
int wddetach(device_t, int);
int wdprint(void *, char *);
void wdperror(const struct wd_softc *, struct ata_xfer *);
static void wdminphys(struct buf *);
2009-05-20 03:43:44 +04:00
static int wdlastclose(device_t);
static bool wd_suspend(device_t, const pmf_qual_t *);
2007-12-11 05:02:31 +03:00
static int wd_standby(struct wd_softc *, int);
2007-12-09 23:27:42 +03:00
During shutdown, detach devices in an orderly fashion. Call the detach routine for every device in the device tree, starting with the leaves and moving toward the root, expecting that each (pseudo-)device driver will use the opportunity to gracefully commit outstandings transactions to the underlying (pseudo-)device and to relinquish control of the hardware to the system BIOS. Detaching devices is not suitable for every shutdown: in an emergency, or if the system state is inconsistent, we should resort to a fast, simple shutdown that uses only the pmf(9) shutdown hooks and the (deprecated) shutdownhooks. For now, if the flag RB_NOSYNC is set in boothowto, opt for the fast, simple shutdown. Add a device flag, DVF_DETACH_SHUTDOWN, that indicates by its presence that it is safe to detach a device during shutdown. Introduce macros CFATTACH_DECL3() and CFATTACH_DECL3_NEW() for creating autoconf attachments with default device flags. Add DVF_DETACH_SHUTDOWN to configuration attachments for atabus(4), atw(4) at cardbus(4), cardbus(4), cardslot(4), com(4) at isa(4), elanpar(4), elanpex(4), elansc(4), gpio(4), npx(4) at isa(4), nsphyter(4), pci(4), pcib(4), pcmcia(4), ppb(4), sip(4), wd(4), and wdc(4) at isa(4). Add a device-detachment "reason" flag, DETACH_SHUTDOWN, that tells the autoconf code and a device driver that the reason for detachment is system shutdown. Add a sysctl, kern.detachall, that tells the system to try to detach every device at shutdown, regardless of any device's DVF_DETACH_SHUTDOWN flag. The default for kern.detachall is 0. SET IT TO 1, PLEASE, TO HELP TEST AND DEBUG DEVICE DETACHMENT AT SHUTDOWN. This is a work in progress. In future work, I aim to treat pseudo-devices more thoroughly, and to gracefully tear down a stack of (pseudo-)disk drivers and filesystems, including cgd(4), vnd(4), and raid(4) instances at shutdown. Also commit some changes that are not easily untangled from the rest: (1) begin to simplify device_t locking: rename struct pmf_private to device_lock, and incorporate device_lock into struct device. (2) #include <sys/device.h> in sys/pmf.h in order to get some definitions that it needs. Stop unnecessarily #including <sys/device.h> in sys/arch/x86/include/pic.h to keep the amd64, xen, and i386 releases building.
2009-04-02 04:09:32 +04:00
CFATTACH_DECL3_NEW(wd, sizeof(struct wd_softc),
wdprobe, wdattach, wddetach, NULL, NULL, NULL, DVF_DETACH_SHUTDOWN);
1998-01-12 12:39:57 +03:00
extern struct cfdriver wd_cd;
dev_type_open(wdopen);
dev_type_close(wdclose);
dev_type_read(wdread);
dev_type_write(wdwrite);
dev_type_ioctl(wdioctl);
dev_type_strategy(wdstrategy);
dev_type_dump(wddump);
dev_type_size(wdsize);
2014-07-25 12:22:08 +04:00
static dev_type_discard(wddiscard);
const struct bdevsw wd_bdevsw = {
.d_open = wdopen,
.d_close = wdclose,
.d_strategy = wdstrategy,
.d_ioctl = wdioctl,
.d_dump = wddump,
.d_psize = wdsize,
2014-07-25 12:22:08 +04:00
.d_discard = wddiscard,
.d_flag = D_DISK
};
const struct cdevsw wd_cdevsw = {
.d_open = wdopen,
.d_close = wdclose,
.d_read = wdread,
.d_write = wdwrite,
.d_ioctl = wdioctl,
.d_stop = nostop,
.d_tty = notty,
.d_poll = nopoll,
.d_mmap = nommap,
.d_kqfilter = nokqfilter,
2014-07-25 12:22:08 +04:00
.d_discard = wddiscard,
.d_flag = D_DISK
};
/* #define WD_DUMP_NOT_TRUSTED if you just want to watch */
static int wddoingadump = 0;
static int wddumprecalibrated = 0;
/*
* Glue necessary to hook WDCIOCCOMMAND into physio
*/
struct wd_ioctl {
LIST_ENTRY(wd_ioctl) wi_list;
struct buf wi_bp;
struct uio wi_uio;
struct iovec wi_iov;
atareq_t wi_atareq;
struct wd_softc *wi_softc;
};
struct wd_ioctl *wi_find(struct buf *);
void wi_free(struct wd_ioctl *);
struct wd_ioctl *wi_get(struct wd_softc *);
void wdioctlstrategy(struct buf *);
void wdgetdefaultlabel(struct wd_softc *, struct disklabel *);
void wdgetdisklabel(struct wd_softc *);
void wdstart(device_t);
void wdstart1(struct wd_softc *, struct buf *, struct ata_xfer *);
static void wdbiorestart(void *);
void wddone(device_t, struct ata_xfer *);
static void wd_params_to_properties(struct wd_softc *);
2014-09-10 11:04:48 +04:00
int wd_get_params(struct wd_softc *, uint8_t, struct ataparams *);
int wd_flushcache(struct wd_softc *, int, bool);
2014-07-25 12:22:08 +04:00
int wd_trim(struct wd_softc *, int, daddr_t, long);
bool wd_shutdown(device_t, int);
int wd_getcache(struct wd_softc *, int *);
int wd_setcache(struct wd_softc *, int);
static void wd_sysctl_attach(struct wd_softc *);
static void wd_sysctl_detach(struct wd_softc *);
struct dkdriver wddkdriver = {
.d_strategy = wdstrategy,
.d_minphys = wdminphys
};
1994-03-10 22:57:20 +03:00
#ifdef HAS_BAD144_HANDLING
static void bad144intern(struct wd_softc *);
#endif
1994-03-10 23:05:30 +03:00
#define WD_QUIRK_SPLIT_MOD15_WRITE 0x0001 /* must split certain writes */
#define WD_QUIRK_FMT "\20\1SPLIT_MOD15_WRITE\2FORCE_LBA48"
/*
* Quirk table for IDE drives. Put more-specific matches first, since
* a simple globing routine is used for matching.
*/
static const struct wd_quirk {
const char *wdq_match; /* inquiry pattern to match */
int wdq_quirks; /* drive quirks */
} wd_quirk_table[] = {
/*
* Some Seagate S-ATA drives have a PHY which can get confused
* with the way data is packetized by some S-ATA controllers.
*
* The work-around is to split in two any write transfer whose
* sector count % 15 == 1 (assuming 512 byte sectors).
*
* XXX This is an incomplete list. There are at least a couple
* XXX more model numbers. If you have trouble with such transfers
* XXX (8K is the most common) on Seagate S-ATA drives, please
* XXX notify thorpej@NetBSD.org.
*
* The ST360015AS has not yet been confirmed to have this
* issue, however, it is the only other drive in the
* Seagate Barracuda Serial ATA V family.
*
*/
{ "ST3120023AS",
WD_QUIRK_SPLIT_MOD15_WRITE },
{ "ST380023AS",
WD_QUIRK_SPLIT_MOD15_WRITE },
{ "ST360015AS",
WD_QUIRK_SPLIT_MOD15_WRITE },
{ NULL,
0 }
};
static const struct wd_quirk *
wd_lookup_quirks(const char *name)
{
const struct wd_quirk *wdq;
const char *estr;
for (wdq = wd_quirk_table; wdq->wdq_match != NULL; wdq++) {
/*
* We only want exact matches (which include matches
* against globbing characters).
*/
if (pmatch(name, wdq->wdq_match, &estr) == 2)
return (wdq);
}
return (NULL);
}
1993-03-21 12:45:37 +03:00
int
wdprobe(device_t parent, cfdata_t match, void *aux)
1993-03-21 12:45:37 +03:00
{
struct ata_device *adev = aux;
1998-01-12 12:39:57 +03:00
if (adev == NULL)
return 0;
if (adev->adev_bustype->bustype_type != SCSIPI_BUSTYPE_ATA)
return 0;
if (match->cf_loc[ATA_HLCF_DRIVE] != ATA_HLCF_DRIVE_DEFAULT &&
match->cf_loc[ATA_HLCF_DRIVE] != adev->adev_drv_data->drive)
1998-10-12 20:09:10 +04:00
return 0;
1994-03-29 08:35:37 +04:00
return 1;
}
1994-03-29 08:35:37 +04:00
void
wdattach(device_t parent, device_t self, void *aux)
1994-03-29 08:35:37 +04:00
{
struct wd_softc *wd = device_private(self);
struct ata_device *adev= aux;
1994-03-29 08:35:37 +04:00
int i, blank;
char tbuf[41], pbuf[9], c, *p, *q;
const struct wd_quirk *wdq;
wd->sc_dev = self;
2007-11-07 11:59:03 +03:00
ATADEBUG_PRINT(("wdattach\n"), DEBUG_FUNCS | DEBUG_PROBE);
mutex_init(&wd->sc_lock, MUTEX_DEFAULT, IPL_BIO);
bufq_alloc(&wd->sc_q, BUFQ_DISK_DEFAULT_STRAT, BUFQ_SORT_RAWBLOCK);
#ifdef WD_SOFTBADSECT
SLIST_INIT(&wd->sc_bslist);
#endif
wd->atabus = adev->adev_bustype;
2003-01-20 08:29:53 +03:00
wd->drvp = adev->adev_drv_data;
wd->drvp->drv_openings = 1;
wd->drvp->drv_start = wdstart;
wd->drvp->drv_done = wddone;
wd->drvp->drv_softc = wd->sc_dev; /* done in atabusconfig_thread()
but too late */
1994-11-05 02:30:15 +03:00
aprint_naive("\n");
aprint_normal("\n");
1998-10-12 20:09:10 +04:00
/* read our drive info */
if (wd_get_params(wd, AT_WAIT, &wd->sc_params) != 0) {
aprint_error_dev(self, "IDENTIFY failed\n");
goto out;
1998-10-12 20:09:10 +04:00
}
for (blank = 0, p = wd->sc_params.atap_model, q = tbuf, i = 0;
1998-10-12 20:09:10 +04:00
i < sizeof(wd->sc_params.atap_model); i++) {
c = *p++;
if (c == '\0')
break;
if (c != ' ') {
if (blank) {
*q++ = ' ';
blank = 0;
}
*q++ = c;
} else
blank = 1;
1999-09-22 13:51:03 +04:00
}
*q++ = '\0';
aprint_normal_dev(self, "<%s>\n", tbuf);
wdq = wd_lookup_quirks(tbuf);
if (wdq != NULL)
wd->sc_quirks = wdq->wdq_quirks;
if (wd->sc_quirks != 0) {
char sbuf[sizeof(WD_QUIRK_FMT) + 64];
snprintb(sbuf, sizeof(sbuf), WD_QUIRK_FMT, wd->sc_quirks);
aprint_normal_dev(self, "quirks %s\n", sbuf);
if (wd->sc_quirks & WD_QUIRK_SPLIT_MOD15_WRITE) {
aprint_error_dev(self, "drive corrupts write transfers with certain controllers, consider replacing\n");
}
}
1998-10-12 20:09:10 +04:00
if ((wd->sc_params.atap_multi & 0xff) > 1) {
wd->drvp->multi = wd->sc_params.atap_multi & 0xff;
} else {
wd->drvp->multi = 1;
}
aprint_verbose_dev(self, "drive supports %d-sector PIO transfers,",
wd->drvp->multi);
/* 48-bit LBA addressing */
if ((wd->sc_params.atap_cmd2_en & ATA_CMD2_LBA48) != 0)
wd->sc_flags |= WDF_LBA48;
/* Prior to ATA-4, LBA was optional. */
1998-10-12 20:09:10 +04:00
if ((wd->sc_params.atap_capabilities1 & WDC_CAP_LBA) != 0)
wd->sc_flags |= WDF_LBA;
#if 0
/* ATA-4 requires LBA. */
1998-10-12 20:09:10 +04:00
if (wd->sc_params.atap_ataversion != 0xffff &&
wd->sc_params.atap_ataversion >= WDC_VER_ATA4)
wd->sc_flags |= WDF_LBA;
#endif
if ((wd->sc_flags & WDF_LBA48) != 0) {
2007-02-10 00:55:00 +03:00
aprint_verbose(" LBA48 addressing\n");
wd->sc_capacity =
2014-09-10 11:04:48 +04:00
((uint64_t) wd->sc_params.atap_max_lba[3] << 48) |
((uint64_t) wd->sc_params.atap_max_lba[2] << 32) |
((uint64_t) wd->sc_params.atap_max_lba[1] << 16) |
((uint64_t) wd->sc_params.atap_max_lba[0] << 0);
wd->sc_capacity28 =
(wd->sc_params.atap_capacity[1] << 16) |
wd->sc_params.atap_capacity[0];
} else if ((wd->sc_flags & WDF_LBA) != 0) {
2007-02-10 00:55:00 +03:00
aprint_verbose(" LBA addressing\n");
wd->sc_capacity28 = wd->sc_capacity =
(wd->sc_params.atap_capacity[1] << 16) |
1998-10-12 20:09:10 +04:00
wd->sc_params.atap_capacity[0];
} else {
2007-02-10 00:55:00 +03:00
aprint_verbose(" chs addressing\n");
wd->sc_capacity28 = wd->sc_capacity =
1998-10-12 20:09:10 +04:00
wd->sc_params.atap_cylinders *
wd->sc_params.atap_heads *
wd->sc_params.atap_sectors;
}
if ((wd->sc_params.atap_secsz & ATA_SECSZ_VALID_MASK) == ATA_SECSZ_VALID
&& ((wd->sc_params.atap_secsz & ATA_SECSZ_LLS) != 0)) {
wd->sc_blksize = 2ULL *
((uint32_t)((wd->sc_params.atap_lls_secsz[1] << 16) |
wd->sc_params.atap_lls_secsz[0]));
} else {
wd->sc_blksize = 512;
}
wd->sc_capacity512 = (wd->sc_capacity * wd->sc_blksize) / DEV_BSIZE;
format_bytes(pbuf, sizeof(pbuf), wd->sc_capacity * wd->sc_blksize);
aprint_normal_dev(self, "%s, %d cyl, %d head, %d sec, "
"%d bytes/sect x %llu sectors\n",
pbuf,
(wd->sc_flags & WDF_LBA) ? (int)(wd->sc_capacity /
(wd->sc_params.atap_heads * wd->sc_params.atap_sectors)) :
wd->sc_params.atap_cylinders,
wd->sc_params.atap_heads, wd->sc_params.atap_sectors,
wd->sc_blksize, (unsigned long long)wd->sc_capacity);
2004-08-13 08:10:49 +04:00
ATADEBUG_PRINT(("%s: atap_dmatiming_mimi=%d, atap_dmatiming_recom=%d\n",
device_xname(self), wd->sc_params.atap_dmatiming_mimi,
1998-10-12 20:09:10 +04:00
wd->sc_params.atap_dmatiming_recom), DEBUG_PROBE);
if (wd->sc_blksize <= 0 || !powerof2(wd->sc_blksize) ||
wd->sc_blksize < DEV_BSIZE || wd->sc_blksize > MAXPHYS) {
aprint_normal_dev(self, "WARNING: block size %u "
"might not actually work\n", wd->sc_blksize);
}
out:
1998-10-12 20:09:10 +04:00
/*
* Initialize and attach the disk structure.
*/
/* we fill in dk_info later */
disk_init(&wd->sc_dk, device_xname(wd->sc_dev), &wddkdriver);
1998-10-12 20:09:10 +04:00
disk_attach(&wd->sc_dk);
wd->drvp->lp = wd->sc_dk.dk_label;
wd_params_to_properties(wd);
rnd_attach_source(&wd->rnd_source, device_xname(wd->sc_dev),
RND_TYPE_DISK, RND_FLAG_DEFAULT);
/* Discover wedges on this disk. */
dkwedge_discover(&wd->sc_dk);
2007-12-09 23:27:42 +03:00
if (!pmf_device_register1(self, wd_suspend, NULL, wd_shutdown))
2007-12-09 23:27:42 +03:00
aprint_error_dev(self, "couldn't establish power handler\n");
wd_sysctl_attach(wd);
2007-12-09 23:27:42 +03:00
}
static bool
wd_suspend(device_t dv, const pmf_qual_t *qual)
2007-12-09 23:27:42 +03:00
{
struct wd_softc *sc = device_private(dv);
/* the adapter needs to be enabled */
if (sc->atabus->ata_addref(sc->drvp))
return true; /* no need to complain */
wd_flushcache(sc, AT_WAIT, false);
wd_standby(sc, AT_WAIT);
sc->atabus->ata_delref(sc->drvp);
2007-12-09 23:27:42 +03:00
return true;
1993-03-21 12:45:37 +03:00
}
int
wddetach(device_t self, int flags)
{
struct wd_softc *sc = device_private(self);
int bmaj, cmaj, i, mn, rc;
if ((rc = disk_begindetach(&sc->sc_dk, wdlastclose, self, flags)) != 0)
return rc;
/* locate the major number */
bmaj = bdevsw_lookup_major(&wd_bdevsw);
cmaj = cdevsw_lookup_major(&wd_cdevsw);
/* Nuke the vnodes for any open instances. */
for (i = 0; i < MAXPARTITIONS; i++) {
2006-03-28 21:38:24 +04:00
mn = WDMINOR(device_unit(self), i);
vdevgone(bmaj, mn, mn, VBLK);
vdevgone(cmaj, mn, mn, VCHR);
}
/* Delete all of our wedges. */
dkwedge_delall(&sc->sc_dk);
mutex_enter(&sc->sc_lock);
/* Kill off any queued buffers. */
bufq_drain(sc->sc_q);
sc->atabus->ata_killpending(sc->drvp);
mutex_exit(&sc->sc_lock);
if (flags & DETACH_POWEROFF)
wd_standby(sc, AT_POLL);
bufq_free(sc->sc_q);
/* Detach disk. */
disk_detach(&sc->sc_dk);
disk_destroy(&sc->sc_dk);
#ifdef WD_SOFTBADSECT
/* Clean out the bad sector list */
while (!SLIST_EMPTY(&sc->sc_bslist)) {
void *head = SLIST_FIRST(&sc->sc_bslist);
SLIST_REMOVE_HEAD(&sc->sc_bslist, dbs_next);
free(head, M_TEMP);
}
sc->sc_bscount = 0;
#endif
2007-12-09 23:27:42 +03:00
pmf_device_deregister(self);
wd_sysctl_detach(sc);
/* Unhook the entropy source. */
rnd_detach_source(&sc->rnd_source);
mutex_destroy(&sc->sc_lock);
sc->drvp->drive_type = ATA_DRIVET_NONE; /* no drive any more here */
sc->drvp->drive_flags = 0;
return (0);
}
/*
* Read/write routine for a buffer. Validates the arguments and schedules the
* transfer. Does not wait for the transfer to complete.
1993-03-21 12:45:37 +03:00
*/
void
wdstrategy(struct buf *bp)
1993-03-21 12:45:37 +03:00
{
struct wd_softc *wd =
device_lookup_private(&wd_cd, WDUNIT(bp->b_dev));
struct disklabel *lp = wd->sc_dk.dk_label;
daddr_t blkno;
ATADEBUG_PRINT(("wdstrategy (%s)\n", device_xname(wd->sc_dev)),
DEBUG_XFERS);
/* Valid request? */
if (bp->b_blkno < 0 ||
(bp->b_bcount % lp->d_secsize) != 0 ||
(bp->b_bcount / lp->d_secsize) >= (1 << NBBY)) {
1993-03-21 12:45:37 +03:00
bp->b_error = EINVAL;
goto done;
1993-03-21 12:45:37 +03:00
}
/* If device invalidated (e.g. media change, door open,
* device detachment), then error.
*/
if ((wd->sc_flags & WDF_LOADED) == 0 ||
!device_is_enabled(wd->sc_dev)) {
bp->b_error = EIO;
goto done;
}
/* If it's a null transfer, return immediately. */
if (bp->b_bcount == 0)
goto done;
1995-01-13 13:46:32 +03:00
/*
* Do bounds checking, adjust transfer. if error, process.
* If end of partition, just return.
*/
if (WDPART(bp->b_dev) == RAW_PART) {
if (bounds_check_with_mediasize(bp, DEV_BSIZE,
wd->sc_capacity512) <= 0)
goto done;
} else {
if (bounds_check_with_label(&wd->sc_dk, bp,
(wd->sc_flags & (WDF_WLABEL|WDF_LABELLING)) != 0) <= 0)
goto done;
}
/*
* Now convert the block number to absolute and put it in
* terms of the device's logical block size.
*/
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;
bp->b_rawblkno = blkno;
#ifdef WD_SOFTBADSECT
/*
* If the transfer about to be attempted contains only a block that
* is known to be bad then return an error for the transfer without
* even attempting to start a transfer up under the premis that we
* will just end up doing more retries for a transfer that will end
* up failing again.
*/
if (__predict_false(!SLIST_EMPTY(&wd->sc_bslist))) {
struct disk_badsectors *dbs;
daddr_t maxblk = blkno + (bp->b_bcount / wd->sc_blksize) - 1;
mutex_enter(&wd->sc_lock);
SLIST_FOREACH(dbs, &wd->sc_bslist, dbs_next)
if ((dbs->dbs_min <= blkno && blkno <= dbs->dbs_max) ||
(dbs->dbs_min <= maxblk && maxblk <= dbs->dbs_max)){
bp->b_error = EIO;
mutex_exit(&wd->sc_lock);
goto done;
}
mutex_exit(&wd->sc_lock);
}
#endif
/* Queue transfer on drive, activate drive and controller if idle. */
mutex_enter(&wd->sc_lock);
disk_wait(&wd->sc_dk);
bufq_put(wd->sc_q, bp);
mutex_exit(&wd->sc_lock);
/* Try to queue on the current drive only */
wdstart(wd->sc_dev);
return;
1993-03-21 12:45:37 +03:00
done:
/* Toss transfer; we're done early. */
bp->b_resid = bp->b_bcount;
1993-03-21 12:45:37 +03:00
biodone(bp);
}
/*
* Queue a drive for I/O.
1993-03-21 12:45:37 +03:00
*/
void
wdstart(device_t self)
{
struct wd_softc *wd = device_private(self);
struct buf *bp;
struct ata_xfer *xfer;
ATADEBUG_PRINT(("wdstart %s\n", device_xname(wd->sc_dev)),
DEBUG_XFERS);
if (!device_is_active(wd->sc_dev))
return;
mutex_enter(&wd->sc_lock);
/*
* Do not queue any transfers until flush is finished, so that
* once flush is pending, it will get handled as soon as xfer
* is available.
*/
if (ISSET(wd->sc_flags, WDF_FLUSH_PEND))
goto out;
while (bufq_peek(wd->sc_q) != NULL) {
/* First try to get xfer. Limit to drive openings iff NCQ. */
xfer = ata_get_xfer_ext(wd->drvp->chnl_softc, 0,
WD_USE_NCQ(wd) ? WD_MAX_OPENINGS(wd) : 0);
if (xfer == NULL)
break;
/* There is got to be a buf for us */
bp = bufq_get(wd->sc_q);
KASSERT(bp != NULL);
1998-10-12 20:09:10 +04:00
xfer->c_retries = 0;
wdstart1(wd, bp, xfer);
}
out:
mutex_exit(&wd->sc_lock);
}
1998-10-12 20:09:10 +04:00
void
wdstart1(struct wd_softc *wd, struct buf *bp, struct ata_xfer *xfer)
1998-10-12 20:09:10 +04:00
{
/* must be locked on entry */
KASSERT(mutex_owned(&wd->sc_lock));
KASSERT(bp == xfer->c_bio.bp || xfer->c_bio.bp == NULL);
KASSERT((xfer->c_flags & (C_WAITACT|C_FREE)) == 0);
/* Reset state, so that retries don't use stale info */
if (__predict_false(xfer->c_retries > 0)) {
xfer->c_flags = 0;
memset(&xfer->c_bio, 0, sizeof(xfer->c_bio));
}
xfer->c_bio.blkno = bp->b_rawblkno;
xfer->c_bio.bcount = bp->b_bcount;
xfer->c_bio.databuf = bp->b_data;
xfer->c_bio.blkdone = 0;
xfer->c_bio.bp = bp;
#ifdef WD_CHAOS_MONKEY
/*
* Override blkno to be over device capacity to trigger error,
* but only if it's read, to avoid trashing disk contents should
* the command be clipped, or otherwise misinterpreted, by the
* driver or controller.
*/
if (BUF_ISREAD(bp) && xfer->c_retries == 0 && wd->drv_chaos_freq > 0 &&
(++wd->drv_chaos_cnt % wd->drv_chaos_freq) == 0) {
aprint_normal_dev(wd->sc_dev, "%s: chaos xfer %d\n",
__func__, xfer->c_slot);
xfer->c_bio.blkno = 7777777 + wd->sc_capacity;
xfer->c_flags |= C_CHAOS;
}
#endif
1998-10-12 20:09:10 +04:00
/*
* If we're retrying, retry in single-sector mode. This will give us
* the sector number of the problem, and will eventually allow the
* transfer to succeed. If FUA is requested, we can't actually
* do this, as ATA_SINGLE is usually executed as PIO transfer by drivers
* which support it, and that isn't compatible with NCQ/FUA.
1998-10-12 20:09:10 +04:00
*/
if (xfer->c_retries >= WDIORETRIES_SINGLE &&
(bp->b_flags & B_MEDIA_FUA) == 0)
xfer->c_bio.flags = ATA_SINGLE;
1998-10-12 20:09:10 +04:00
else
xfer->c_bio.flags = 0;
if (wd->sc_flags & WDF_LBA48 &&
(((xfer->c_bio.blkno +
xfer->c_bio.bcount / wd->sc_dk.dk_label->d_secsize) >
wd->sc_capacity28) ||
((xfer->c_bio.bcount / wd->sc_dk.dk_label->d_secsize) > 128)))
xfer->c_bio.flags |= ATA_LBA48;
/*
* If NCQ was negotiated, always use it for the first several attempts.
* Since device cancels all outstanding requests on error, downgrade
* to non-NCQ on retry, so that the retried transfer would not cause
* cascade failure for the other transfers if it fails again.
* If FUA was requested, we can't downgrade, as that would violate
* the semantics - FUA would not be honored. In that case, continue
* retrying with NCQ.
*/
if (WD_USE_NCQ(wd) && (xfer->c_retries < WDIORETRIES_SINGLE ||
(bp->b_flags & B_MEDIA_FUA) != 0)) {
xfer->c_bio.flags |= ATA_LBA48;
xfer->c_flags |= C_NCQ;
if (WD_USE_NCQ_PRIO(wd) &&
BIO_GETPRIO(bp) == BPRIO_TIMECRITICAL)
xfer->c_bio.flags |= ATA_PRIO_HIGH;
}
1998-10-12 20:09:10 +04:00
if (wd->sc_flags & WDF_LBA)
xfer->c_bio.flags |= ATA_LBA;
1998-10-12 20:09:10 +04:00
if (bp->b_flags & B_READ)
xfer->c_bio.flags |= ATA_READ;
if (bp->b_flags & B_MEDIA_FUA) {
/* If not using NCQ, the command WRITE DMA FUA EXT is LBA48 */
KASSERT((wd->sc_flags & WDF_LBA48) != 0);
if ((xfer->c_flags & C_NCQ) == 0)
xfer->c_bio.flags |= ATA_LBA48;
xfer->c_bio.flags |= ATA_FUA;
}
1998-10-12 20:09:10 +04:00
/* Instrumentation. */
if (xfer->c_retries == 0)
disk_busy(&wd->sc_dk);
switch (wd->atabus->ata_bio(wd->drvp, xfer)) {
case ATACMD_TRY_AGAIN:
panic("wdstart1: try again");
1998-10-12 20:09:10 +04:00
break;
case ATACMD_QUEUED:
case ATACMD_COMPLETE:
1998-10-12 20:09:10 +04:00
break;
default:
panic("wdstart1: bad return code from ata_bio()");
1998-10-12 20:09:10 +04:00
}
}
void
wddone(device_t self, struct ata_xfer *xfer)
1998-10-12 20:09:10 +04:00
{
struct wd_softc *wd = device_private(self);
const char *errmsg;
int do_perror = 0;
struct buf *bp;
1998-10-12 20:09:10 +04:00
ATADEBUG_PRINT(("wddone %s\n", device_xname(wd->sc_dev)),
2007-11-07 11:56:41 +03:00
DEBUG_XFERS);
if (__predict_false(wddoingadump)) {
/* just drop it to the floor */
ata_free_xfer(wd->drvp->chnl_softc, xfer);
return;
}
bp = xfer->c_bio.bp;
KASSERT(bp != NULL);
bp->b_resid = xfer->c_bio.bcount;
switch (xfer->c_bio.error) {
1998-10-12 20:09:10 +04:00
case ERR_DMA:
errmsg = "DMA error";
1998-10-12 20:09:10 +04:00
goto retry;
case ERR_DF:
errmsg = "device fault";
1998-10-12 20:09:10 +04:00
goto retry;
case TIMEOUT:
errmsg = "device timeout";
1998-10-12 20:09:10 +04:00
goto retry;
case REQUEUE:
errmsg = "requeue";
goto retry2;
case ERR_RESET:
errmsg = "channel reset";
goto retry2;
1998-10-12 20:09:10 +04:00
case ERROR:
/* Don't care about media change bits */
if (xfer->c_bio.r_error != 0 &&
(xfer->c_bio.r_error & ~(WDCE_MC | WDCE_MCR)) == 0)
1998-10-12 20:09:10 +04:00
goto noerror;
errmsg = "error";
do_perror = 1;
1998-10-12 20:09:10 +04:00
retry: /* Just reset and retry. Can we do more ? */
if ((xfer->c_flags & C_RECOVERED) == 0) {
int wflags = (xfer->c_flags & C_POLL) ? AT_POLL : 0;
(*wd->atabus->ata_reset_drive)(wd->drvp, wflags, NULL);
}
retry2:
mutex_enter(&wd->sc_lock);
diskerr(bp, "wd", errmsg, LOG_PRINTF,
xfer->c_bio.blkdone, wd->sc_dk.dk_label);
if (xfer->c_retries < WDIORETRIES)
printf(", slot %d, retry %d", xfer->c_slot,
xfer->c_retries + 1);
printf("\n");
if (do_perror)
wdperror(wd, xfer);
if (xfer->c_retries < WDIORETRIES) {
xfer->c_retries++;
/* Rerun ASAP if just requeued */
callout_reset(&xfer->c_retry_callout,
(xfer->c_bio.error == REQUEUE) ? 1 : RECOVERYTIME,
wdbiorestart, xfer);
mutex_exit(&wd->sc_lock);
1998-10-12 20:09:10 +04:00
return;
}
mutex_exit(&wd->sc_lock);
#ifdef WD_SOFTBADSECT
/*
* Not all errors indicate a failed block but those that do,
* put the block on the bad-block list for the device. Only
* do this for reads because the drive should do it for writes,
* itself, according to Manuel.
*/
2003-04-15 21:42:44 +04:00
if ((bp->b_flags & B_READ) &&
((wd->drvp->ata_vers >= 4 && xfer->c_bio.r_error & 64) ||
(wd->drvp->ata_vers < 4 && xfer->c_bio.r_error & 192))) {
struct disk_badsectors *dbs;
dbs = malloc(sizeof *dbs, M_TEMP, M_NOWAIT);
if (dbs == NULL) {
aprint_error_dev(wd->sc_dev,
"failed to add bad block to list\n");
goto out;
}
dbs->dbs_min = bp->b_rawblkno;
dbs->dbs_max = dbs->dbs_min +
(bp->b_bcount /wd->sc_blksize) - 1;
microtime(&dbs->dbs_failedat);
mutex_enter(&wd->sc_lock);
SLIST_INSERT_HEAD(&wd->sc_bslist, dbs, dbs_next);
wd->sc_bscount++;
mutex_exit(&wd->sc_lock);
}
out:
#endif
1998-10-12 20:09:10 +04:00
bp->b_error = EIO;
break;
case NOERROR:
noerror: if ((xfer->c_bio.flags & ATA_CORR) || xfer->c_retries > 0)
aprint_error_dev(wd->sc_dev,
"soft error (corrected) slot %d\n", xfer->c_slot);
#ifdef WD_CHAOS_MONKEY
KASSERT((xfer->c_flags & C_CHAOS) == 0);
#endif
break;
case ERR_NODEV:
bp->b_error = EIO;
break;
1998-10-12 20:09:10 +04:00
}
if (__predict_false(bp->b_error != 0) && bp->b_resid == 0) {
/*
* the disk or controller sometimes report a complete
* xfer, when there has been an error. This is wrong,
* assume nothing got transfered in this case
*/
bp->b_resid = bp->b_bcount;
}
disk_unbusy(&wd->sc_dk, (bp->b_bcount - bp->b_resid),
(bp->b_flags & B_READ));
1998-10-12 20:09:10 +04:00
rnd_add_uint32(&wd->rnd_source, bp->b_blkno);
ata_free_xfer(wd->drvp->chnl_softc, xfer);
biodone(bp);
ata_channel_start(wd->drvp->chnl_softc, wd->drvp->drive);
1998-10-12 20:09:10 +04:00
}
static void
wdbiorestart(void *v)
1998-10-12 20:09:10 +04:00
{
struct ata_xfer *xfer = v;
struct buf *bp = xfer->c_bio.bp;
struct wd_softc *wd = device_lookup_private(&wd_cd, WDUNIT(bp->b_dev));
2007-11-07 11:59:03 +03:00
ATADEBUG_PRINT(("wdrestart %s\n", device_xname(wd->sc_dev)),
DEBUG_XFERS);
mutex_enter(&wd->sc_lock);
wdstart1(wd, bp, xfer);
mutex_exit(&wd->sc_lock);
1998-10-12 20:09:10 +04:00
}
static void
wdminphys(struct buf *bp)
{
const struct wd_softc * const wd =
device_lookup_private(&wd_cd, WDUNIT(bp->b_dev));
uint32_t maxsectors;
/*
* The limit is actually 65536 for LBA48 and 256 for non-LBA48,
* but that requires to set the count for the ATA command
* to 0, which is somewhat error prone, so better stay safe.
*/
if (wd->sc_flags & WDF_LBA48)
maxsectors = 65535;
else
maxsectors = 128;
if (bp->b_bcount > (wd->sc_blksize * maxsectors))
bp->b_bcount = (wd->sc_blksize * maxsectors);
minphys(bp);
}
int
wdread(dev_t dev, struct uio *uio, int flags)
{
2004-08-13 08:10:49 +04:00
ATADEBUG_PRINT(("wdread\n"), DEBUG_XFERS);
return (physio(wdstrategy, NULL, dev, B_READ, wdminphys, uio));
}
int
wdwrite(dev_t dev, struct uio *uio, int flags)
{
2004-08-13 08:10:49 +04:00
ATADEBUG_PRINT(("wdwrite\n"), DEBUG_XFERS);
return (physio(wdstrategy, NULL, dev, B_WRITE, wdminphys, uio));
}
1993-03-21 12:45:37 +03:00
int
wdopen(dev_t dev, int flag, int fmt, struct lwp *l)
1993-03-21 12:45:37 +03:00
{
struct wd_softc *wd;
2000-07-06 04:43:04 +04:00
int part, error;
2004-08-13 08:10:49 +04:00
ATADEBUG_PRINT(("wdopen\n"), DEBUG_FUNCS);
wd = device_lookup_private(&wd_cd, WDUNIT(dev));
if (wd == NULL)
2000-07-06 04:43:04 +04:00
return (ENXIO);
if (! device_is_active(wd->sc_dev))
return (ENODEV);
if (wd->sc_capacity == 0)
return (ENODEV);
part = WDPART(dev);
2007-07-21 23:51:47 +04:00
mutex_enter(&wd->sc_dk.dk_openlock);
/*
* If there are wedges, and this is not RAW_PART, then we
* need to fail.
*/
if (wd->sc_dk.dk_nwedges != 0 && part != RAW_PART) {
error = EBUSY;
goto bad1;
}
/*
* If this is the first open of this device, add a reference
* to the adapter.
*/
if (wd->sc_dk.dk_openmask == 0 &&
(error = wd->atabus->ata_addref(wd->drvp)) != 0)
goto bad1;
1994-11-22 06:23:49 +03:00
if (wd->sc_dk.dk_openmask != 0) {
/*
* If any partition is open, but the disk has been invalidated,
* disallow further opens.
*/
1998-10-12 20:09:10 +04:00
if ((wd->sc_flags & WDF_LOADED) == 0) {
1995-04-01 14:29:41 +04:00
error = EIO;
goto bad2;
1995-04-01 14:29:41 +04:00
}
1994-11-22 06:23:49 +03:00
} else {
1998-10-12 20:09:10 +04:00
if ((wd->sc_flags & WDF_LOADED) == 0) {
/* Load the physical device parameters. */
if (wd_get_params(wd, AT_WAIT, &wd->sc_params) != 0) {
aprint_error_dev(wd->sc_dev,
"IDENTIFY failed\n");
error = EIO;
goto bad2;
}
wd->sc_flags |= WDF_LOADED;
/* Load the partition info if not already loaded. */
wdgetdisklabel(wd);
}
}
1994-11-22 06:23:49 +03:00
/* Check that the partition exists. */
if (part != RAW_PART &&
(part >= wd->sc_dk.dk_label->d_npartitions ||
wd->sc_dk.dk_label->d_partitions[part].p_fstype == FS_UNUSED)) {
error = ENXIO;
goto bad2;
}
/* Insure only one open at a time. */
switch (fmt) {
case S_IFCHR:
1994-10-20 17:08:07 +03:00
wd->sc_dk.dk_copenmask |= (1 << part);
break;
case S_IFBLK:
1994-10-20 17:08:07 +03:00
wd->sc_dk.dk_bopenmask |= (1 << part);
break;
}
1998-10-12 20:09:10 +04:00
wd->sc_dk.dk_openmask =
wd->sc_dk.dk_copenmask | wd->sc_dk.dk_bopenmask;
2007-07-21 23:51:47 +04:00
mutex_exit(&wd->sc_dk.dk_openlock);
return 0;
bad2:
if (wd->sc_dk.dk_openmask == 0)
wd->atabus->ata_delref(wd->drvp);
bad1:
2007-07-21 23:51:47 +04:00
mutex_exit(&wd->sc_dk.dk_openlock);
return error;
1993-03-21 12:45:37 +03:00
}
/*
* Caller must hold wd->sc_dk.dk_openlock.
*/
2009-05-20 03:43:44 +04:00
static int
wdlastclose(device_t self)
{
2009-05-20 03:43:44 +04:00
struct wd_softc *wd = device_private(self);
wd_flushcache(wd, AT_WAIT, false);
if (! (wd->sc_flags & WDF_KLABEL))
wd->sc_flags &= ~WDF_LOADED;
wd->atabus->ata_delref(wd->drvp);
2009-05-20 03:43:44 +04:00
return 0;
}
int
wdclose(dev_t dev, int flag, int fmt, struct lwp *l)
{
struct wd_softc *wd =
device_lookup_private(&wd_cd, WDUNIT(dev));
int part = WDPART(dev);
2004-08-13 08:10:49 +04:00
ATADEBUG_PRINT(("wdclose\n"), DEBUG_FUNCS);
2007-07-21 23:51:47 +04:00
mutex_enter(&wd->sc_dk.dk_openlock);
switch (fmt) {
case S_IFCHR:
wd->sc_dk.dk_copenmask &= ~(1 << part);
break;
case S_IFBLK:
wd->sc_dk.dk_bopenmask &= ~(1 << part);
break;
}
1998-10-12 20:09:10 +04:00
wd->sc_dk.dk_openmask =
wd->sc_dk.dk_copenmask | wd->sc_dk.dk_bopenmask;
if (wd->sc_dk.dk_openmask == 0)
2009-05-20 03:43:44 +04:00
wdlastclose(wd->sc_dev);
2007-07-21 23:51:47 +04:00
mutex_exit(&wd->sc_dk.dk_openlock);
return 0;
}
void
wdgetdefaultlabel(struct wd_softc *wd, struct disklabel *lp)
{
2004-08-13 08:10:49 +04:00
ATADEBUG_PRINT(("wdgetdefaultlabel\n"), DEBUG_FUNCS);
1998-10-12 20:09:10 +04:00
memset(lp, 0, sizeof(struct disklabel));
lp->d_secsize = wd->sc_blksize;
1998-10-12 20:09:10 +04:00
lp->d_ntracks = wd->sc_params.atap_heads;
lp->d_nsectors = wd->sc_params.atap_sectors;
lp->d_ncylinders = (wd->sc_flags & WDF_LBA) ? wd->sc_capacity /
(wd->sc_params.atap_heads * wd->sc_params.atap_sectors) :
wd->sc_params.atap_cylinders;
lp->d_secpercyl = lp->d_ntracks * lp->d_nsectors;
if (strcmp(wd->sc_params.atap_model, "ST506") == 0)
lp->d_type = DKTYPE_ST506;
else
lp->d_type = DKTYPE_ESDI;
1998-10-12 20:09:10 +04:00
strncpy(lp->d_typename, wd->sc_params.atap_model, 16);
strncpy(lp->d_packname, "fictitious", 16);
if (wd->sc_capacity > UINT32_MAX)
lp->d_secperunit = UINT32_MAX;
else
lp->d_secperunit = wd->sc_capacity;
lp->d_rpm = 3600;
lp->d_interleave = 1;
lp->d_flags = 0;
lp->d_partitions[RAW_PART].p_offset = 0;
lp->d_partitions[RAW_PART].p_size = lp->d_secperunit;
lp->d_partitions[RAW_PART].p_fstype = FS_UNUSED;
lp->d_npartitions = RAW_PART + 1;
lp->d_magic = DISKMAGIC;
lp->d_magic2 = DISKMAGIC;
lp->d_checksum = dkcksum(lp);
1997-10-09 03:05:22 +04:00
}
/*
* Fabricate a default disk label, and try to read the correct one.
*/
void
wdgetdisklabel(struct wd_softc *wd)
1997-10-09 03:05:22 +04:00
{
struct disklabel *lp = wd->sc_dk.dk_label;
const char *errstring;
1997-10-09 03:05:22 +04:00
2004-08-13 08:10:49 +04:00
ATADEBUG_PRINT(("wdgetdisklabel\n"), DEBUG_FUNCS);
1997-10-09 03:05:22 +04:00
1998-10-12 20:09:10 +04:00
memset(wd->sc_dk.dk_cpulabel, 0, sizeof(struct cpu_disklabel));
1997-10-09 03:05:22 +04:00
wdgetdefaultlabel(wd, lp);
wd->drvp->badsect[0] = -1;
if (wd->drvp->state > RESET) {
mutex_enter(&wd->sc_lock);
wd->drvp->drive_flags |= ATA_DRIVE_RESET;
mutex_exit(&wd->sc_lock);
}
errstring = readdisklabel(MAKEWDDEV(0, device_unit(wd->sc_dev),
2006-03-28 21:38:24 +04:00
RAW_PART), wdstrategy, lp,
wd->sc_dk.dk_cpulabel);
if (errstring) {
/*
* This probably happened because the drive's default
* geometry doesn't match the DOS geometry. We
* assume the DOS geometry is now in the label and try
* again. XXX This is a kluge.
*/
if (wd->drvp->state > RESET) {
mutex_enter(&wd->sc_lock);
wd->drvp->drive_flags |= ATA_DRIVE_RESET;
mutex_exit(&wd->sc_lock);
}
errstring = readdisklabel(MAKEWDDEV(0, device_unit(wd->sc_dev),
1998-10-12 20:09:10 +04:00
RAW_PART), wdstrategy, lp, wd->sc_dk.dk_cpulabel);
}
if (errstring) {
aprint_error_dev(wd->sc_dev, "%s\n", errstring);
return;
}
if (wd->drvp->state > RESET) {
mutex_enter(&wd->sc_lock);
wd->drvp->drive_flags |= ATA_DRIVE_RESET;
mutex_exit(&wd->sc_lock);
}
#ifdef HAS_BAD144_HANDLING
if ((lp->d_flags & D_BADSECT) != 0)
bad144intern(wd);
#endif
}
void
wdperror(const struct wd_softc *wd, struct ata_xfer *xfer)
{
static const char *const errstr0_3[] = {"address mark not found",
"track 0 not found", "aborted command", "media change requested",
"id not found", "media changed", "uncorrectable data error",
"bad block detected"};
static const char *const errstr4_5[] = {
"obsolete (address mark not found)",
"no media/write protected", "aborted command",
"media change requested", "id not found", "media changed",
"uncorrectable data error", "interface CRC error"};
const char *const *errstr;
int i;
const char *sep = "";
const char *devname = device_xname(wd->sc_dev);
struct ata_drive_datas *drvp = wd->drvp;
int errno = xfer->c_bio.r_error;
if (drvp->ata_vers >= 4)
errstr = errstr4_5;
else
errstr = errstr0_3;
printf("%s: (", devname);
if (errno == 0)
printf("error not notified");
for (i = 0; i < 8; i++) {
if (errno & (1 << i)) {
printf("%s%s", sep, errstr[i]);
sep = ", ";
}
}
printf(")\n");
}
1993-03-21 12:45:37 +03:00
int
wdioctl(dev_t dev, u_long xfer, void *addr, int flag, struct lwp *l)
1993-03-21 12:45:37 +03:00
{
struct wd_softc *wd =
device_lookup_private(&wd_cd, WDUNIT(dev));
int error;
#ifdef __HAVE_OLD_DISKLABEL
2002-12-15 04:56:02 +03:00
struct disklabel *newlabel = NULL;
#endif
2004-08-13 08:10:49 +04:00
ATADEBUG_PRINT(("wdioctl\n"), DEBUG_FUNCS);
1998-10-12 20:09:10 +04:00
if ((wd->sc_flags & WDF_LOADED) == 0)
return EIO;
error = disk_ioctl(&wd->sc_dk, dev, xfer, addr, flag, l);
if (error != EPASSTHROUGH)
return error;
error = 0;
switch (xfer) {
#ifdef HAS_BAD144_HANDLING
1993-03-21 12:45:37 +03:00
case DIOCSBAD:
if ((flag & FWRITE) == 0)
return EBADF;
wd->sc_dk.dk_cpulabel->bad = *(struct dkbad *)addr;
wd->sc_dk.dk_label->d_flags |= D_BADSECT;
bad144intern(wd);
return 0;
#endif
#ifdef WD_SOFTBADSECT
case DIOCBSLIST :
{
2014-09-10 11:04:48 +04:00
uint32_t count, missing, skip;
struct disk_badsecinfo dbsi;
struct disk_badsectors *dbs;
size_t available;
2007-09-05 09:36:19 +04:00
uint8_t *laddr;
dbsi = *(struct disk_badsecinfo *)addr;
missing = wd->sc_bscount;
count = 0;
available = dbsi.dbsi_bufsize;
skip = dbsi.dbsi_skip;
2007-09-05 09:36:19 +04:00
laddr = (uint8_t *)dbsi.dbsi_buffer;
/*
* We start this loop with the expectation that all of the
* entries will be missed and decrement this counter each
* time we either skip over one (already copied out) or
* we actually copy it back to user space. The structs
* holding the bad sector information are copied directly
* back to user space whilst the summary is returned via
* the struct passed in via the ioctl.
*/
SLIST_FOREACH(dbs, &wd->sc_bslist, dbs_next) {
if (skip > 0) {
missing--;
skip--;
continue;
}
if (available < sizeof(*dbs))
break;
available -= sizeof(*dbs);
copyout(dbs, laddr, sizeof(*dbs));
laddr += sizeof(*dbs);
missing--;
count++;
}
dbsi.dbsi_left = missing;
dbsi.dbsi_copied = count;
*(struct disk_badsecinfo *)addr = dbsi;
return 0;
}
case DIOCBSFLUSH :
/* Clean out the bad sector list */
while (!SLIST_EMPTY(&wd->sc_bslist)) {
void *head = SLIST_FIRST(&wd->sc_bslist);
SLIST_REMOVE_HEAD(&wd->sc_bslist, dbs_next);
free(head, M_TEMP);
}
wd->sc_bscount = 0;
return 0;
#endif
case DIOCWDINFO:
case DIOCSDINFO:
#ifdef __HAVE_OLD_DISKLABEL
case ODIOCWDINFO:
case ODIOCSDINFO:
#endif
{
struct disklabel *lp;
2002-12-15 04:56:02 +03:00
if ((flag & FWRITE) == 0)
return EBADF;
#ifdef __HAVE_OLD_DISKLABEL
if (xfer == ODIOCSDINFO || xfer == ODIOCWDINFO) {
2011-11-25 17:55:40 +04:00
newlabel = malloc(sizeof *newlabel, M_TEMP,
M_WAITOK | M_ZERO);
2002-12-15 04:56:02 +03:00
if (newlabel == NULL)
return EIO;
memcpy(newlabel, addr, sizeof (struct olddisklabel));
lp = newlabel;
} else
#endif
lp = (struct disklabel *)addr;
2007-07-21 23:51:47 +04:00
mutex_enter(&wd->sc_dk.dk_openlock);
1998-10-12 20:09:10 +04:00
wd->sc_flags |= WDF_LABELLING;
error = setdisklabel(wd->sc_dk.dk_label,
lp, /*wd->sc_dk.dk_openmask : */0,
wd->sc_dk.dk_cpulabel);
if (error == 0) {
if (wd->drvp->state > RESET) {
mutex_enter(&wd->sc_lock);
wd->drvp->drive_flags |= ATA_DRIVE_RESET;
mutex_exit(&wd->sc_lock);
}
if (xfer == DIOCWDINFO
#ifdef __HAVE_OLD_DISKLABEL
|| xfer == ODIOCWDINFO
#endif
)
error = writedisklabel(WDLABELDEV(dev),
wdstrategy, wd->sc_dk.dk_label,
wd->sc_dk.dk_cpulabel);
1993-03-21 12:45:37 +03:00
}
1998-10-12 20:09:10 +04:00
wd->sc_flags &= ~WDF_LABELLING;
2007-07-21 23:51:47 +04:00
mutex_exit(&wd->sc_dk.dk_openlock);
2002-12-15 04:56:02 +03:00
#ifdef __HAVE_OLD_DISKLABEL
if (newlabel != NULL)
free(newlabel, M_TEMP);
#endif
return error;
}
case DIOCKLABEL:
if (*(int *)addr)
wd->sc_flags |= WDF_KLABEL;
else
wd->sc_flags &= ~WDF_KLABEL;
return 0;
1994-02-26 03:00:17 +03:00
case DIOCWLABEL:
if ((flag & FWRITE) == 0)
return EBADF;
if (*(int *)addr)
1998-10-12 20:09:10 +04:00
wd->sc_flags |= WDF_WLABEL;
else
1998-10-12 20:09:10 +04:00
wd->sc_flags &= ~WDF_WLABEL;
return 0;
1997-10-09 03:05:22 +04:00
case DIOCGDEFLABEL:
wdgetdefaultlabel(wd, (struct disklabel *)addr);
return 0;
#ifdef __HAVE_OLD_DISKLABEL
case ODIOCGDEFLABEL:
2002-12-15 04:56:02 +03:00
newlabel = malloc(sizeof *newlabel, M_TEMP, M_WAITOK);
if (newlabel == NULL)
return EIO;
wdgetdefaultlabel(wd, newlabel);
if (newlabel->d_npartitions <= OLDMAXPARTITIONS)
memcpy(addr, &newlabel, sizeof (struct olddisklabel));
else
error = ENOTTY;
free(newlabel, M_TEMP);
return error;
#endif
1997-10-09 03:05:22 +04:00
1993-03-21 12:45:37 +03:00
#ifdef notyet
case DIOCWFORMAT:
if ((flag & FWRITE) == 0)
return EBADF;
1998-10-12 20:09:10 +04:00
{
register struct format_op *fop;
struct iovec aiov;
struct uio auio;
fop = (struct format_op *)addr;
aiov.iov_base = fop->df_buf;
aiov.iov_len = fop->df_count;
auio.uio_iov = &aiov;
auio.uio_iovcnt = 1;
auio.uio_resid = fop->df_count;
auio.uio_offset =
1998-10-12 20:09:10 +04:00
fop->df_startblk * wd->sc_dk.dk_label->d_secsize;
auio.uio_vmspace = l->l_proc->p_vmspace;
error = physio(wdformat, NULL, dev, B_WRITE, wdminphys,
&auio);
fop->df_count -= auio.uio_resid;
fop->df_reg[0] = wdc->sc_status;
fop->df_reg[1] = wdc->sc_error;
return error;
1998-10-12 20:09:10 +04:00
}
1993-03-21 12:45:37 +03:00
#endif
case DIOCGCACHE:
return wd_getcache(wd, (int *)addr);
case DIOCSCACHE:
return wd_setcache(wd, *(int *)addr);
case DIOCCACHESYNC:
return wd_flushcache(wd, AT_WAIT, true);
case ATAIOCCOMMAND:
/*
* Make sure this command is (relatively) safe first
*/
if ((((atareq_t *) addr)->flags & ATACMD_READ) == 0 &&
(flag & FWRITE) == 0)
return (EBADF);
{
struct wd_ioctl *wi;
atareq_t *atareq = (atareq_t *) addr;
int error1;
wi = wi_get(wd);
wi->wi_atareq = *atareq;
if (atareq->datalen && atareq->flags &
(ATACMD_READ | ATACMD_WRITE)) {
void *tbuf;
if (atareq->datalen < DEV_BSIZE
&& atareq->command == WDCC_IDENTIFY) {
tbuf = malloc(DEV_BSIZE, M_TEMP, M_WAITOK);
wi->wi_iov.iov_base = tbuf;
wi->wi_iov.iov_len = DEV_BSIZE;
UIO_SETUP_SYSSPACE(&wi->wi_uio);
} else {
tbuf = NULL;
wi->wi_iov.iov_base = atareq->databuf;
wi->wi_iov.iov_len = atareq->datalen;
wi->wi_uio.uio_vmspace = l->l_proc->p_vmspace;
}
wi->wi_uio.uio_iov = &wi->wi_iov;
wi->wi_uio.uio_iovcnt = 1;
wi->wi_uio.uio_resid = atareq->datalen;
wi->wi_uio.uio_offset = 0;
wi->wi_uio.uio_rw =
(atareq->flags & ATACMD_READ) ? B_READ : B_WRITE;
error1 = physio(wdioctlstrategy, &wi->wi_bp, dev,
(atareq->flags & ATACMD_READ) ? B_READ : B_WRITE,
wdminphys, &wi->wi_uio);
if (tbuf != NULL && error1 == 0) {
error1 = copyout(tbuf, atareq->databuf,
atareq->datalen);
free(tbuf, M_TEMP);
}
} else {
/* No need to call physio if we don't have any
user data */
wi->wi_bp.b_flags = 0;
wi->wi_bp.b_data = 0;
wi->wi_bp.b_bcount = 0;
wi->wi_bp.b_dev = dev;
2005-12-11 15:16:03 +03:00
wi->wi_bp.b_proc = l->l_proc;
wdioctlstrategy(&wi->wi_bp);
error1 = wi->wi_bp.b_error;
}
*atareq = wi->wi_atareq;
wi_free(wi);
return(error1);
}
case DIOCGSTRATEGY:
{
struct disk_strategy *dks = (void *)addr;
mutex_enter(&wd->sc_lock);
strlcpy(dks->dks_name, bufq_getstrategyname(wd->sc_q),
sizeof(dks->dks_name));
mutex_exit(&wd->sc_lock);
dks->dks_paramlen = 0;
return 0;
}
case DIOCSSTRATEGY:
{
struct disk_strategy *dks = (void *)addr;
struct bufq_state *new;
struct bufq_state *old;
if ((flag & FWRITE) == 0) {
return EBADF;
}
if (dks->dks_param != NULL) {
return EINVAL;
}
dks->dks_name[sizeof(dks->dks_name) - 1] = 0; /* ensure term */
error = bufq_alloc(&new, dks->dks_name,
BUFQ_EXACT|BUFQ_SORT_RAWBLOCK);
if (error) {
return error;
}
mutex_enter(&wd->sc_lock);
old = wd->sc_q;
bufq_move(new, old);
wd->sc_q = new;
mutex_exit(&wd->sc_lock);
bufq_free(old);
return 0;
}
1993-03-21 12:45:37 +03:00
default:
return ENOTTY;
1993-03-21 12:45:37 +03:00
}
#ifdef DIAGNOSTIC
panic("wdioctl: impossible");
#endif
1993-03-21 12:45:37 +03:00
}
2014-07-25 12:22:08 +04:00
static int
wddiscard(dev_t dev, off_t pos, off_t len)
{
struct wd_softc *wd = device_lookup_private(&wd_cd, WDUNIT(dev));
daddr_t bno;
long size, done;
long maxatonce, amount;
int result;
if (!(wd->sc_params.atap_ata_major & WDC_VER_ATA7)
|| !(wd->sc_params.support_dsm & ATA_SUPPORT_DSM_TRIM)) {
/* not supported; ignore request */
ATADEBUG_PRINT(("wddiscard (unsupported)\n"), DEBUG_FUNCS);
return 0;
}
maxatonce = 0xffff; /*wd->sc_params.max_dsm_blocks*/
ATADEBUG_PRINT(("wddiscard\n"), DEBUG_FUNCS);
if ((wd->sc_flags & WDF_LOADED) == 0)
return EIO;
/* round the start up and the end down */
bno = (pos + wd->sc_blksize - 1) / wd->sc_blksize;
size = ((pos + len) / wd->sc_blksize) - bno;
2014-07-25 12:22:08 +04:00
done = 0;
while (done < size) {
amount = size - done;
if (amount > maxatonce) {
amount = maxatonce;
}
result = wd_trim(wd, WDPART(dev), bno + done, amount);
if (result) {
return result;
}
done += amount;
}
return 0;
}
1994-02-26 03:00:17 +03:00
#ifdef B_FORMAT
1993-03-21 12:45:37 +03:00
int
wdformat(struct buf *bp)
{
1994-02-26 03:00:17 +03:00
1993-03-21 12:45:37 +03:00
bp->b_flags |= B_FORMAT;
return wdstrategy(bp);
1993-03-21 12:45:37 +03:00
}
#endif
int
wdsize(dev_t dev)
1993-03-21 12:45:37 +03:00
{
struct wd_softc *wd;
2000-07-06 04:43:04 +04:00
int part, omask;
int size;
2004-08-13 08:10:49 +04:00
ATADEBUG_PRINT(("wdsize\n"), DEBUG_FUNCS);
wd = device_lookup_private(&wd_cd, WDUNIT(dev));
if (wd == NULL)
return (-1);
part = WDPART(dev);
omask = wd->sc_dk.dk_openmask & (1 << part);
if (omask == 0 && wdopen(dev, 0, S_IFBLK, NULL) != 0)
return (-1);
if (wd->sc_dk.dk_label->d_partitions[part].p_fstype != FS_SWAP)
size = -1;
else
size = wd->sc_dk.dk_label->d_partitions[part].p_size *
1998-10-12 20:09:10 +04:00
(wd->sc_dk.dk_label->d_secsize / DEV_BSIZE);
if (omask == 0 && wdclose(dev, 0, S_IFBLK, NULL) != 0)
return (-1);
return (size);
1993-03-21 12:45:37 +03:00
}
/*
* Dump core after a system crash.
*/
1993-03-21 12:45:37 +03:00
int
wddump(dev_t dev, daddr_t blkno, void *va, size_t size)
1993-03-21 12:45:37 +03:00
{
struct wd_softc *wd; /* disk unit to do the I/O */
1998-10-12 20:09:10 +04:00
struct disklabel *lp; /* disk's disklabel */
2000-07-06 04:43:04 +04:00
int part, err;
1998-10-12 20:09:10 +04:00
int nblks; /* total number of sectors left to write */
struct ata_xfer *xfer;
/* Check if recursive dump; if so, punt. */
if (wddoingadump)
return EFAULT;
wddoingadump = 1;
wd = device_lookup_private(&wd_cd, WDUNIT(dev));
2000-07-06 04:43:04 +04:00
if (wd == NULL)
return (ENXIO);
part = WDPART(dev);
1998-10-12 20:09:10 +04:00
/* Convert to disk sectors. Request must be a multiple of size. */
lp = wd->sc_dk.dk_label;
if ((size % lp->d_secsize) != 0)
return EFAULT;
nblks = size / lp->d_secsize;
blkno = blkno / (lp->d_secsize / DEV_BSIZE);
/* Check transfer bounds against partition size. */
if ((blkno < 0) || ((blkno + nblks) > lp->d_partitions[part].p_size))
return EINVAL;
/* Offset block number to start of partition. */
blkno += lp->d_partitions[part].p_offset;
/* Recalibrate, if first dump transfer. */
if (wddumprecalibrated == 0) {
wddumprecalibrated = 1;
(*wd->atabus->ata_reset_drive)(wd->drvp,
AT_POLL | AT_RST_EMERG, NULL);
wd->drvp->state = RESET;
}
xfer = ata_get_xfer_ext(wd->drvp->chnl_softc, 0, 0);
if (xfer == NULL) {
printf("%s: no xfer\n", __func__);
return EAGAIN;
}
xfer->c_bio.blkno = blkno;
xfer->c_bio.flags = ATA_POLL;
if (wd->sc_flags & WDF_LBA48 &&
(xfer->c_bio.blkno + nblks) > wd->sc_capacity28)
xfer->c_bio.flags |= ATA_LBA48;
if (wd->sc_flags & WDF_LBA)
xfer->c_bio.flags |= ATA_LBA;
xfer->c_bio.bcount = nblks * lp->d_secsize;
xfer->c_bio.databuf = va;
#ifndef WD_DUMP_NOT_TRUSTED
switch (err = wd->atabus->ata_bio(wd->drvp, xfer)) {
case ATACMD_TRY_AGAIN:
panic("wddump: try again");
break;
case ATACMD_QUEUED:
panic("wddump: polled command has been queued");
break;
case ATACMD_COMPLETE:
break;
2008-12-13 22:38:20 +03:00
default:
panic("wddump: unknown atacmd code %d", err);
}
switch(err = xfer->c_bio.error) {
case TIMEOUT:
printf("wddump: device timed out");
err = EIO;
break;
case ERR_DF:
printf("wddump: drive fault");
err = EIO;
break;
case ERR_DMA:
printf("wddump: DMA error");
err = EIO;
break;
case ERROR:
printf("wddump: ");
wdperror(wd, xfer);
err = EIO;
break;
case NOERROR:
err = 0;
break;
default:
panic("wddump: unknown error type %d", err);
}
if (err != 0) {
printf("\n");
return err;
}
#else /* WD_DUMP_NOT_TRUSTED */
/* Let's just talk about this first... */
printf("wd%d: dump addr 0x%x, cylin %d, head %d, sector %d\n",
unit, va, cylin, head, sector);
delay(500 * 1000); /* half a second */
#endif
1993-03-21 12:45:37 +03:00
wddoingadump = 0;
return 0;
1993-03-21 12:45:37 +03:00
}
#ifdef HAS_BAD144_HANDLING
/*
* Internalize the bad sector table.
*/
void
bad144intern(struct wd_softc *wd)
{
struct dkbad *bt = &wd->sc_dk.dk_cpulabel->bad;
struct disklabel *lp = wd->sc_dk.dk_label;
int i = 0;
2004-08-13 08:10:49 +04:00
ATADEBUG_PRINT(("bad144intern\n"), DEBUG_XFERS);
for (; i < NBT_BAD; i++) {
if (bt->bt_bad[i].bt_cyl == 0xffff)
break;
wd->drvp->badsect[i] =
bt->bt_bad[i].bt_cyl * lp->d_secpercyl +
(bt->bt_bad[i].bt_trksec >> 8) * lp->d_nsectors +
(bt->bt_bad[i].bt_trksec & 0xff);
}
for (; i < NBT_BAD+1; i++)
wd->drvp->badsect[i] = -1;
}
#endif
static void
wd_params_to_properties(struct wd_softc *wd)
{
struct disk_geom *dg = &wd->sc_dk.dk_geom;
memset(dg, 0, sizeof(*dg));
dg->dg_secperunit = wd->sc_capacity;
dg->dg_secsize = wd->sc_blksize;
dg->dg_nsectors = wd->sc_params.atap_sectors;
dg->dg_ntracks = wd->sc_params.atap_heads;
if ((wd->sc_flags & WDF_LBA) == 0)
dg->dg_ncylinders = wd->sc_params.atap_cylinders;
/* XXX Should have a case for ATA here, too. */
const char *cp = strcmp(wd->sc_params.atap_model, "ST506") ?
"ST506" : "ESDI";
disk_set_info(wd->sc_dev, &wd->sc_dk, cp);
}
1998-10-12 20:09:10 +04:00
int
2014-09-10 11:04:48 +04:00
wd_get_params(struct wd_softc *wd, uint8_t flags, struct ataparams *params)
{
2007-11-07 11:59:03 +03:00
switch (wd->atabus->ata_get_params(wd->drvp, flags, params)) {
1998-10-12 20:09:10 +04:00
case CMD_AGAIN:
return 1;
case CMD_ERR:
if (wd->drvp->drive_type != ATA_DRIVET_OLD)
return 1;
1998-10-12 20:09:10 +04:00
/*
* We `know' there's a drive here; just assume it's old.
* This geometry is only used to read the MBR and print a
* (false) attach message.
*/
strncpy(params->atap_model, "ST506",
sizeof params->atap_model);
params->atap_config = ATA_CFG_FIXED;
params->atap_cylinders = 1024;
params->atap_heads = 8;
params->atap_sectors = 17;
params->atap_multi = 1;
params->atap_capabilities1 = params->atap_capabilities2 = 0;
wd->drvp->ata_vers = -1; /* Mark it as pre-ATA */
/* FALLTHROUGH */
1998-10-12 20:09:10 +04:00
case CMD_OK:
return 0;
default:
panic("wd_get_params: bad return code from ata_get_params");
/* NOTREACHED */
}
}
int
wd_getcache(struct wd_softc *wd, int *bitsp)
{
struct ataparams params;
if (wd_get_params(wd, AT_WAIT, &params) != 0)
return EIO;
if (params.atap_cmd_set1 == 0x0000 ||
params.atap_cmd_set1 == 0xffff ||
(params.atap_cmd_set1 & WDC_CMD1_CACHE) == 0) {
*bitsp = 0;
return 0;
}
*bitsp = DKCACHE_WCHANGE | DKCACHE_READ;
if (params.atap_cmd1_en & WDC_CMD1_CACHE)
*bitsp |= DKCACHE_WRITE;
if (WD_USE_NCQ(wd) || (wd->drvp->drive_flags & ATA_DRIVE_WFUA))
*bitsp |= DKCACHE_FUA;
return 0;
}
const char at_errbits[] = "\20\10ERROR\11TIMEOU\12DF";
int
wd_setcache(struct wd_softc *wd, int bits)
{
struct ataparams params;
struct ata_xfer *xfer;
int error;
if (wd_get_params(wd, AT_WAIT, &params) != 0)
return EIO;
if (params.atap_cmd_set1 == 0x0000 ||
2003-04-27 18:33:20 +04:00
params.atap_cmd_set1 == 0xffff ||
(params.atap_cmd_set1 & WDC_CMD1_CACHE) == 0)
return EOPNOTSUPP;
if ((bits & DKCACHE_READ) == 0 ||
(bits & DKCACHE_SAVE) != 0)
return EOPNOTSUPP;
xfer = ata_get_xfer(wd->drvp->chnl_softc);
if (xfer == NULL)
return EINTR;
xfer->c_ata_c.r_command = SET_FEATURES;
xfer->c_ata_c.r_st_bmask = 0;
xfer->c_ata_c.r_st_pmask = 0;
xfer->c_ata_c.timeout = 30000; /* 30s timeout */
xfer->c_ata_c.flags = AT_WAIT;
if (bits & DKCACHE_WRITE)
xfer->c_ata_c.r_features = WDSF_WRITE_CACHE_EN;
else
xfer->c_ata_c.r_features = WDSF_WRITE_CACHE_DS;
if (wd->atabus->ata_exec_command(wd->drvp, xfer) != ATACMD_COMPLETE) {
aprint_error_dev(wd->sc_dev,
"wd_setcache command not complete\n");
error = EIO;
goto out;
}
if (xfer->c_ata_c.flags & (AT_ERROR | AT_TIMEOU | AT_DF)) {
char sbuf[sizeof(at_errbits) + 64];
snprintb(sbuf, sizeof(sbuf), at_errbits, xfer->c_ata_c.flags);
aprint_error_dev(wd->sc_dev, "wd_setcache: status=%s\n", sbuf);
error = EIO;
goto out;
}
error = 0;
out:
ata_free_xfer(wd->drvp->chnl_softc, xfer);
ata_channel_start(wd->drvp->chnl_softc, wd->drvp->drive);
return error;
}
2007-12-09 23:27:42 +03:00
static int
wd_standby(struct wd_softc *wd, int flags)
{
struct ata_xfer *xfer;
int error;
xfer = ata_get_xfer(wd->drvp->chnl_softc);
if (xfer == NULL)
return EINTR;
xfer->c_ata_c.r_command = WDCC_STANDBY_IMMED;
xfer->c_ata_c.r_st_bmask = WDCS_DRDY;
xfer->c_ata_c.r_st_pmask = WDCS_DRDY;
xfer->c_ata_c.flags = flags;
xfer->c_ata_c.timeout = 30000; /* 30s timeout */
if (wd->atabus->ata_exec_command(wd->drvp, xfer) != ATACMD_COMPLETE) {
aprint_error_dev(wd->sc_dev,
2007-12-09 23:27:42 +03:00
"standby immediate command didn't complete\n");
error = EIO;
goto out;
}
if (xfer->c_ata_c.flags & AT_ERROR) {
if (xfer->c_ata_c.r_error == WDCE_ABRT) {
/* command not supported */
error = ENODEV;
goto out;
}
}
if (xfer->c_ata_c.flags & (AT_ERROR | AT_TIMEOU | AT_DF)) {
char sbuf[sizeof(at_errbits) + 64];
snprintb(sbuf, sizeof(sbuf), at_errbits, xfer->c_ata_c.flags);
aprint_error_dev(wd->sc_dev, "wd_standby: status=%s\n", sbuf);
error = EIO;
goto out;
}
error = 0;
out:
ata_free_xfer(wd->drvp->chnl_softc, xfer);
/* drive is supposed to go idle, do not call ata_channel_start() */
return error;
}
int
wd_flushcache(struct wd_softc *wd, int flags, bool start)
{
struct ata_xfer *xfer;
int error;
/*
* WDCC_FLUSHCACHE is here since ATA-4, but some drives report
* only ATA-2 and still support it.
*/
if (wd->drvp->ata_vers < 4 &&
((wd->sc_params.atap_cmd_set2 & WDC_CMD2_FC) == 0 ||
wd->sc_params.atap_cmd_set2 == 0xffff))
return ENODEV;
mutex_enter(&wd->sc_lock);
SET(wd->sc_flags, WDF_FLUSH_PEND);
mutex_exit(&wd->sc_lock);
xfer = ata_get_xfer(wd->drvp->chnl_softc);
mutex_enter(&wd->sc_lock);
CLR(wd->sc_flags, WDF_FLUSH_PEND);
mutex_exit(&wd->sc_lock);
if (xfer == NULL) {
error = EINTR;
goto out;
}
if ((wd->sc_params.atap_cmd2_en & ATA_CMD2_LBA48) != 0 &&
(wd->sc_params.atap_cmd2_en & ATA_CMD2_FCE) != 0) {
xfer->c_ata_c.r_command = WDCC_FLUSHCACHE_EXT;
flags |= AT_LBA48;
} else
xfer->c_ata_c.r_command = WDCC_FLUSHCACHE;
xfer->c_ata_c.r_st_bmask = WDCS_DRDY;
xfer->c_ata_c.r_st_pmask = WDCS_DRDY;
xfer->c_ata_c.flags = flags | AT_READREG;
xfer->c_ata_c.timeout = 300000; /* 5m timeout */
if (wd->atabus->ata_exec_command(wd->drvp, xfer) != ATACMD_COMPLETE) {
aprint_error_dev(wd->sc_dev,
"flush cache command didn't complete\n");
error = EIO;
goto out_xfer;
}
if (xfer->c_ata_c.flags & AT_ERROR) {
if (xfer->c_ata_c.r_error == WDCE_ABRT) {
/* command not supported */
error = ENODEV;
goto out_xfer;
}
}
if (xfer->c_ata_c.flags & (AT_ERROR | AT_TIMEOU | AT_DF)) {
char sbuf[sizeof(at_errbits) + 64];
snprintb(sbuf, sizeof(sbuf), at_errbits, xfer->c_ata_c.flags);
aprint_error_dev(wd->sc_dev, "wd_flushcache: status=%s\n",
sbuf);
error = EIO;
goto out_xfer;
}
error = 0;
out_xfer:
ata_free_xfer(wd->drvp->chnl_softc, xfer);
out:
/* kick queue processing blocked while waiting for flush xfer */
if (start)
ata_channel_start(wd->drvp->chnl_softc, wd->drvp->drive);
return error;
}
int
2014-07-25 12:22:08 +04:00
wd_trim(struct wd_softc *wd, int part, daddr_t bno, long size)
{
struct ata_xfer *xfer;
int error;
unsigned char *req;
if (part != RAW_PART)
bno += wd->sc_dk.dk_label->d_partitions[part].p_offset;;
xfer = ata_get_xfer(wd->drvp->chnl_softc);
if (xfer == NULL)
return EINTR;
req = kmem_zalloc(512, KM_SLEEP);
req[0] = bno & 0xff;
req[1] = (bno >> 8) & 0xff;
req[2] = (bno >> 16) & 0xff;
req[3] = (bno >> 24) & 0xff;
req[4] = (bno >> 32) & 0xff;
req[5] = (bno >> 40) & 0xff;
2014-07-25 12:22:08 +04:00
req[6] = size & 0xff;
req[7] = (size >> 8) & 0xff;
xfer->c_ata_c.r_command = ATA_DATA_SET_MANAGEMENT;
xfer->c_ata_c.r_count = 1;
xfer->c_ata_c.r_features = ATA_SUPPORT_DSM_TRIM;
xfer->c_ata_c.r_st_bmask = WDCS_DRDY;
xfer->c_ata_c.r_st_pmask = WDCS_DRDY;
xfer->c_ata_c.timeout = 30000; /* 30s timeout */
xfer->c_ata_c.data = req;
xfer->c_ata_c.bcount = 512;
xfer->c_ata_c.flags |= AT_WRITE | AT_WAIT;
if (wd->atabus->ata_exec_command(wd->drvp, xfer) != ATACMD_COMPLETE) {
aprint_error_dev(wd->sc_dev,
"trim command didn't complete\n");
kmem_free(req, 512);
error = EIO;
goto out;
}
kmem_free(req, 512);
if (xfer->c_ata_c.flags & AT_ERROR) {
if (xfer->c_ata_c.r_error == WDCE_ABRT) {
/* command not supported */
error = ENODEV;
goto out;
}
}
if (xfer->c_ata_c.flags & (AT_ERROR | AT_TIMEOU | AT_DF)) {
char sbuf[sizeof(at_errbits) + 64];
snprintb(sbuf, sizeof(sbuf), at_errbits, xfer->c_ata_c.flags);
aprint_error_dev(wd->sc_dev, "wd_trim: status=%s\n",
sbuf);
error = EIO;
goto out;
}
error = 0;
out:
ata_free_xfer(wd->drvp->chnl_softc, xfer);
ata_channel_start(wd->drvp->chnl_softc, wd->drvp->drive);
return error;
}
bool
wd_shutdown(device_t dev, int how)
{
struct wd_softc *wd = device_private(dev);
/* the adapter needs to be enabled */
if (wd->atabus->ata_addref(wd->drvp))
return true; /* no need to complain */
wd_flushcache(wd, AT_POLL, false);
if ((how & RB_POWERDOWN) == RB_POWERDOWN)
wd_standby(wd, AT_POLL);
return true;
}
/*
* Allocate space for a ioctl queue structure. Mostly taken from
* scsipi_ioctl.c
*/
struct wd_ioctl *
wi_get(struct wd_softc *wd)
{
struct wd_ioctl *wi;
wi = malloc(sizeof(struct wd_ioctl), M_TEMP, M_WAITOK|M_ZERO);
wi->wi_softc = wd;
2008-01-02 14:48:20 +03:00
buf_init(&wi->wi_bp);
return (wi);
}
/*
* Free an ioctl structure and remove it from our list
*/
void
wi_free(struct wd_ioctl *wi)
{
2008-01-02 14:48:20 +03:00
buf_destroy(&wi->wi_bp);
free(wi, M_TEMP);
}
/*
* Find a wd_ioctl structure based on the struct buf.
*/
struct wd_ioctl *
wi_find(struct buf *bp)
{
return container_of(bp, struct wd_ioctl, wi_bp);
}
static uint
wi_sector_size(const struct wd_ioctl * const wi)
{
switch (wi->wi_atareq.command) {
case WDCC_READ:
case WDCC_WRITE:
case WDCC_READMULTI:
case WDCC_WRITEMULTI:
case WDCC_READDMA:
case WDCC_WRITEDMA:
case WDCC_READ_EXT:
case WDCC_WRITE_EXT:
case WDCC_READMULTI_EXT:
case WDCC_WRITEMULTI_EXT:
case WDCC_READDMA_EXT:
case WDCC_WRITEDMA_EXT:
case WDCC_READ_FPDMA_QUEUED:
case WDCC_WRITE_FPDMA_QUEUED:
return wi->wi_softc->sc_blksize;
default:
return 512;
}
}
/*
* Ioctl pseudo strategy routine
*
* This is mostly stolen from scsipi_ioctl.c:scsistrategy(). What
* happens here is:
*
* - wdioctl() queues a wd_ioctl structure.
*
* - wdioctl() calls physio/wdioctlstrategy based on whether or not
* user space I/O is required. If physio() is called, physio() eventually
* calls wdioctlstrategy().
*
* - In either case, wdioctlstrategy() calls wd->atabus->ata_exec_command()
* to perform the actual command
*
* The reason for the use of the pseudo strategy routine is because
* when doing I/O to/from user space, physio _really_ wants to be in
* the loop. We could put the entire buffer into the ioctl request
* structure, but that won't scale if we want to do things like download
* microcode.
*/
void
wdioctlstrategy(struct buf *bp)
{
struct wd_ioctl *wi;
struct ata_xfer *xfer;
int error = 0;
wi = wi_find(bp);
if (wi == NULL) {
printf("wdioctlstrategy: "
"No matching ioctl request found in queue\n");
error = EINVAL;
goto out2;
}
xfer = ata_get_xfer(wi->wi_softc->drvp->chnl_softc);
if (xfer == NULL) {
error = EINTR;
goto out2;
}
/*
* Abort if physio broke up the transfer
*/
if (bp->b_bcount != wi->wi_atareq.datalen) {
printf("physio split wd ioctl request... cannot proceed\n");
error = EIO;
goto out;
}
/*
* Abort if we didn't get a buffer size that was a multiple of
* our sector size (or overflows CHS/LBA28 sector count)
*/
if ((bp->b_bcount % wi_sector_size(wi)) != 0 ||
(bp->b_bcount / wi_sector_size(wi)) >=
(1 << NBBY)) {
error = EINVAL;
goto out;
}
/*
* Make sure a timeout was supplied in the ioctl request
*/
if (wi->wi_atareq.timeout == 0) {
error = EINVAL;
goto out;
}
if (wi->wi_atareq.flags & ATACMD_READ)
xfer->c_ata_c.flags |= AT_READ;
else if (wi->wi_atareq.flags & ATACMD_WRITE)
xfer->c_ata_c.flags |= AT_WRITE;
if (wi->wi_atareq.flags & ATACMD_READREG)
xfer->c_ata_c.flags |= AT_READREG;
if ((wi->wi_atareq.flags & ATACMD_LBA) != 0)
xfer->c_ata_c.flags |= AT_LBA;
xfer->c_ata_c.flags |= AT_WAIT;
xfer->c_ata_c.timeout = wi->wi_atareq.timeout;
xfer->c_ata_c.r_command = wi->wi_atareq.command;
xfer->c_ata_c.r_lba = ((wi->wi_atareq.head & 0x0f) << 24) |
(wi->wi_atareq.cylinder << 8) |
wi->wi_atareq.sec_num;
xfer->c_ata_c.r_count = wi->wi_atareq.sec_count;
xfer->c_ata_c.r_features = wi->wi_atareq.features;
xfer->c_ata_c.r_st_bmask = WDCS_DRDY;
xfer->c_ata_c.r_st_pmask = WDCS_DRDY;
xfer->c_ata_c.data = wi->wi_bp.b_data;
xfer->c_ata_c.bcount = wi->wi_bp.b_bcount;
if (wi->wi_softc->atabus->ata_exec_command(wi->wi_softc->drvp, xfer)
!= ATACMD_COMPLETE) {
wi->wi_atareq.retsts = ATACMD_ERROR;
error = EIO;
goto out;
}
if (xfer->c_ata_c.flags & (AT_ERROR | AT_TIMEOU | AT_DF)) {
if (xfer->c_ata_c.flags & AT_ERROR) {
wi->wi_atareq.retsts = ATACMD_ERROR;
wi->wi_atareq.error = xfer->c_ata_c.r_error;
} else if (xfer->c_ata_c.flags & AT_DF)
wi->wi_atareq.retsts = ATACMD_DF;
else
wi->wi_atareq.retsts = ATACMD_TIMEOUT;
} else {
wi->wi_atareq.retsts = ATACMD_OK;
if (wi->wi_atareq.flags & ATACMD_READREG) {
wi->wi_atareq.command = xfer->c_ata_c.r_status;
wi->wi_atareq.features = xfer->c_ata_c.r_error;
wi->wi_atareq.sec_count = xfer->c_ata_c.r_count;
wi->wi_atareq.sec_num = xfer->c_ata_c.r_lba & 0xff;
wi->wi_atareq.head = (xfer->c_ata_c.r_device & 0xf0) |
((xfer->c_ata_c.r_lba >> 24) & 0x0f);
wi->wi_atareq.cylinder =
(xfer->c_ata_c.r_lba >> 8) & 0xffff;
wi->wi_atareq.error = xfer->c_ata_c.r_error;
}
}
out:
ata_free_xfer(wi->wi_softc->drvp->chnl_softc, xfer);
ata_channel_start(wi->wi_softc->drvp->chnl_softc,
wi->wi_softc->drvp->drive);
out2:
bp->b_error = error;
if (error)
bp->b_resid = bp->b_bcount;
biodone(bp);
}
static void
wd_sysctl_attach(struct wd_softc *wd)
{
const struct sysctlnode *node;
int error;
/* sysctl set-up */
if (sysctl_createv(&wd->nodelog, 0, NULL, &node,
0, CTLTYPE_NODE, device_xname(wd->sc_dev),
SYSCTL_DESCR("wd driver settings"),
NULL, 0, NULL, 0,
CTL_HW, CTL_CREATE, CTL_EOL) != 0) {
aprint_error_dev(wd->sc_dev,
"could not create %s.%s sysctl node\n",
"hw", device_xname(wd->sc_dev));
return;
}
wd->drv_max_tags = ATA_MAX_OPENINGS;
if ((error = sysctl_createv(&wd->nodelog, 0, NULL, NULL,
CTLFLAG_READWRITE, CTLTYPE_INT, "max_tags",
SYSCTL_DESCR("max number of NCQ tags to use"),
NULL, 0, &wd->drv_max_tags, 0,
CTL_HW, node->sysctl_num, CTL_CREATE, CTL_EOL))
!= 0) {
aprint_error_dev(wd->sc_dev,
"could not create %s.%s.max_tags sysctl - error %d\n",
"hw", device_xname(wd->sc_dev), error);
return;
}
wd->drv_ncq = true;
if ((error = sysctl_createv(&wd->nodelog, 0, NULL, NULL,
CTLFLAG_READWRITE, CTLTYPE_BOOL, "use_ncq",
SYSCTL_DESCR("use NCQ if supported"),
NULL, 0, &wd->drv_ncq, 0,
CTL_HW, node->sysctl_num, CTL_CREATE, CTL_EOL))
!= 0) {
aprint_error_dev(wd->sc_dev,
"could not create %s.%s.use_ncq sysctl - error %d\n",
"hw", device_xname(wd->sc_dev), error);
return;
}
wd->drv_ncq_prio = false;
if ((error = sysctl_createv(&wd->nodelog, 0, NULL, NULL,
CTLFLAG_READWRITE, CTLTYPE_BOOL, "use_ncq_prio",
SYSCTL_DESCR("use NCQ PRIORITY if supported"),
NULL, 0, &wd->drv_ncq_prio, 0,
CTL_HW, node->sysctl_num, CTL_CREATE, CTL_EOL))
!= 0) {
aprint_error_dev(wd->sc_dev,
"could not create %s.%s.use_ncq_prio sysctl - error %d\n",
"hw", device_xname(wd->sc_dev), error);
return;
}
#ifdef WD_CHAOS_MONKEY
wd->drv_chaos_freq = 0;
if ((error = sysctl_createv(&wd->nodelog, 0, NULL, NULL,
CTLFLAG_READWRITE, CTLTYPE_INT, "chaos_freq",
SYSCTL_DESCR("simulated bio read error rate"),
NULL, 0, &wd->drv_chaos_freq, 0,
CTL_HW, node->sysctl_num, CTL_CREATE, CTL_EOL))
!= 0) {
aprint_error_dev(wd->sc_dev,
"could not create %s.%s.chaos_freq sysctl - error %d\n",
"hw", device_xname(wd->sc_dev), error);
return;
}
wd->drv_chaos_cnt = 0;
if ((error = sysctl_createv(&wd->nodelog, 0, NULL, NULL,
CTLFLAG_READONLY, CTLTYPE_INT, "chaos_cnt",
SYSCTL_DESCR("number of processed bio reads"),
NULL, 0, &wd->drv_chaos_cnt, 0,
CTL_HW, node->sysctl_num, CTL_CREATE, CTL_EOL))
!= 0) {
aprint_error_dev(wd->sc_dev,
"could not create %s.%s.chaos_cnt sysctl - error %d\n",
"hw", device_xname(wd->sc_dev), error);
return;
}
#endif
}
static void
wd_sysctl_detach(struct wd_softc *wd)
{
sysctl_teardown(&wd->nodelog);
}