summaryrefslogtreecommitdiff
path: root/sys/dev/ic/mfi.c
diff options
context:
space:
mode:
Diffstat (limited to 'sys/dev/ic/mfi.c')
-rw-r--r--sys/dev/ic/mfi.c211
1 files changed, 195 insertions, 16 deletions
diff --git a/sys/dev/ic/mfi.c b/sys/dev/ic/mfi.c
index bd0b505ae42..0e88e61de3f 100644
--- a/sys/dev/ic/mfi.c
+++ b/sys/dev/ic/mfi.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: mfi.c,v 1.58 2006/05/26 00:53:54 marco Exp $ */
+/* $OpenBSD: mfi.c,v 1.59 2006/06/19 19:05:45 marco Exp $ */
/*
* Copyright (c) 2006 Marco Peereboom <marco@peereboom.us>
*
@@ -79,8 +79,8 @@ void mfi_freemem(struct mfi_softc *, struct mfi_mem *);
int mfi_transition_firmware(struct mfi_softc *);
int mfi_initialize_firmware(struct mfi_softc *);
int mfi_get_info(struct mfi_softc *);
-u_int32_t mfi_read(struct mfi_softc *, bus_size_t);
-void mfi_write(struct mfi_softc *, bus_size_t, u_int32_t);
+uint32_t mfi_read(struct mfi_softc *, bus_size_t);
+void mfi_write(struct mfi_softc *, bus_size_t, uint32_t);
int mfi_poll(struct mfi_ccb *);
int mfi_despatch_cmd(struct mfi_ccb *);
int mfi_create_sgl(struct mfi_ccb *, int);
@@ -102,6 +102,7 @@ int mfi_ioctl_disk(struct mfi_softc *, struct bioc_disk *);
int mfi_ioctl_alarm(struct mfi_softc *, struct bioc_alarm *);
int mfi_ioctl_blink(struct mfi_softc *sc, struct bioc_blink *);
int mfi_ioctl_setstate(struct mfi_softc *, struct bioc_setstate *);
+int mfi_bio_hs(struct mfi_softc *, int, int, void *);
#endif /* NBIO > 0 */
struct mfi_ccb *
@@ -212,10 +213,10 @@ destroy:
return (1);
}
-u_int32_t
+uint32_t
mfi_read(struct mfi_softc *sc, bus_size_t r)
{
- u_int32_t rv;
+ uint32_t rv;
bus_space_barrier(sc->sc_iot, sc->sc_ioh, r, 4,
BUS_SPACE_BARRIER_READ);
@@ -226,7 +227,7 @@ mfi_read(struct mfi_softc *sc, bus_size_t r)
}
void
-mfi_write(struct mfi_softc *sc, bus_size_t r, u_int32_t v)
+mfi_write(struct mfi_softc *sc, bus_size_t r, uint32_t v)
{
DNPRINTF(MFI_D_RW, "%s: mw 0x%x 0x%08x", DEVNAME(sc), r, v);
@@ -1253,17 +1254,30 @@ mfi_ioctl(struct device *dev, u_long cmd, caddr_t addr)
int
mfi_ioctl_inq(struct mfi_softc *sc, struct bioc_inq *bi)
{
+ struct mfi_conf *cfg;
+ int rv = EINVAL;
+
+ DNPRINTF(MFI_D_IOCTL, "%s: mfi_ioctl_inq\n", DEVNAME(sc));
+
if (mfi_get_info(sc)) {
DNPRINTF(MFI_D_IOCTL, "%s: mfi_ioctl_inq failed\n",
DEVNAME(sc));
return (EIO);
}
+ /* get figures */
+ cfg = malloc(sizeof *cfg, M_DEVBUF, M_WAITOK);
+ if (mfi_mgmt(sc, MD_DCMD_CONF_GET, MFI_DATA_IN, sizeof *cfg, cfg, NULL))
+ goto freeme;
+
strlcpy(bi->bi_dev, DEVNAME(sc), sizeof(bi->bi_dev));
- bi->bi_novol = sc->sc_info.mci_lds_present;
+ bi->bi_novol = cfg->mfc_no_ld + cfg->mfc_no_hs;
bi->bi_nodisk = sc->sc_info.mci_pd_disks_present;
- return (0);
+ rv = 0;
+freeme:
+ free(cfg, M_DEVBUF);
+ return (rv);
}
int
@@ -1288,8 +1302,9 @@ mfi_ioctl_vol(struct mfi_softc *sc, struct bioc_vol *bv)
sizeof(sc->sc_ld_details), &sc->sc_ld_details, mbox))
goto done;
- if (bv->bv_volid > sc->sc_ld_list.mll_no_ld) {
- /* XXX go do hotspares */
+ if (bv->bv_volid >= sc->sc_ld_list.mll_no_ld) {
+ /* go do hotspares */
+ rv = mfi_bio_hs(sc, bv->bv_volid, MFI_MGMT_VD, bv);
goto done;
}
@@ -1401,8 +1416,9 @@ mfi_ioctl_disk(struct mfi_softc *sc, struct bioc_disk *bd)
vol = bd->bd_volid;
- if (vol > cfg->mfc_no_ld) {
- /* XXX do hotspares */
+ if (vol >= cfg->mfc_no_ld) {
+ /* do hotspares */
+ rv = mfi_bio_hs(sc, bd->bd_volid, MFI_MGMT_SD, bd);
goto freeme;
}
@@ -1422,7 +1438,7 @@ mfi_ioctl_disk(struct mfi_softc *sc, struct bioc_disk *bd)
bd->bd_status = BIOC_SDUNUSED;
break;
- case MFI_PD_HOTSPARE:
+ case MFI_PD_HOTSPARE: /* XXX dedicated hotspare part of array? */
bd->bd_status = BIOC_SDHOTSPARE;
break;
@@ -1530,6 +1546,9 @@ mfi_ioctl_blink(struct mfi_softc *sc, struct bioc_blink *bb)
uint32_t cmd;
struct mfi_pd_list *pd;
+ DNPRINTF(MFI_D_IOCTL, "%s: mfi_ioctl_blink %x\n", DEVNAME(sc),
+ bb->bb_status);
+
/* channel 0 means not in an enclosure so can't be blinked */
if (bb->bb_channel == 0)
return (EINVAL);
@@ -1552,14 +1571,14 @@ mfi_ioctl_blink(struct mfi_softc *sc, struct bioc_blink *bb)
memset(mbox, 0, sizeof mbox);
+ *((uint16_t *)&mbox) = pd->mpl_address[i].mpa_pd_id;;
+
switch (bb->bb_status) {
case BIOC_SBUNBLINK:
- *((uint16_t *)&mbox) = pd->mpl_address[i].mpa_pd_id;;
cmd = MR_DCMD_PD_UNBLINK;
break;
case BIOC_SBBLINK:
- *((uint16_t *)&mbox) = pd->mpl_address[i].mpa_pd_id;;
cmd = MR_DCMD_PD_BLINK;
break;
@@ -1583,6 +1602,166 @@ done:
int
mfi_ioctl_setstate(struct mfi_softc *sc, struct bioc_setstate *bs)
{
- return (ENOTTY); /* XXX not yet */
+ struct mfi_pd_list *pd;
+ int i, found, rv = EINVAL;
+ uint8_t mbox[MFI_MBOX_SIZE];
+ uint32_t cmd;
+
+ DNPRINTF(MFI_D_IOCTL, "%s: mfi_ioctl_setstate %x\n", DEVNAME(sc),
+ bs->bs_status);
+
+ pd = malloc(MFI_PD_LIST_SIZE, M_DEVBUF, M_WAITOK);
+
+ if (mfi_mgmt(sc, MR_DCMD_PD_GET_LIST, MFI_DATA_IN,
+ MFI_PD_LIST_SIZE, pd, NULL))
+ goto done;
+
+ for (i = 0, found = 0; i < pd->mpl_no_pd; i++)
+ if (bs->bs_channel == pd->mpl_address[i].mpa_enc_index &&
+ bs->bs_target == pd->mpl_address[i].mpa_enc_slot) {
+ found = 1;
+ break;
+ }
+
+ if (!found)
+ goto done;
+
+ memset(mbox, 0, sizeof mbox);
+
+ *((uint16_t *)&mbox) = pd->mpl_address[i].mpa_pd_id;;
+
+ switch (bs->bs_status) {
+ case BIOC_SSONLINE:
+ mbox[2] = MFI_PD_ONLINE;
+ cmd = MD_DCMD_PD_SET_STATE;
+ break;
+
+ case BIOC_SSOFFLINE:
+ mbox[2] = MFI_PD_OFFLINE;
+ cmd = MD_DCMD_PD_SET_STATE;
+ break;
+
+ case BIOC_SSHOTSPARE:
+ mbox[2] = MFI_PD_HOTSPARE;
+ cmd = MD_DCMD_PD_SET_STATE;
+ break;
+/*
+ case BIOC_SSREBUILD:
+ cmd = MD_DCMD_PD_REBUILD;
+ break;
+*/
+ default:
+ DNPRINTF(MFI_D_IOCTL, "%s: mfi_ioctl_setstate invalid "
+ "opcode %x\n", DEVNAME(sc), bs->bs_status);
+ goto done;
+ }
+
+
+ if (mfi_mgmt(sc, MD_DCMD_PD_SET_STATE, MFI_DATA_NONE, 0, NULL, mbox))
+ goto done;
+
+ rv = 0;
+done:
+ free(pd, M_DEVBUF);
+ return (rv);
}
+
+int
+mfi_bio_hs(struct mfi_softc *sc, int volid, int type, void *bio_hs)
+{
+ struct mfi_conf *cfg;
+ struct mfi_hotspare *hs;
+ struct mfi_pd_details *pd;
+ struct bioc_disk *sdhs;
+ struct bioc_vol *vdhs;
+ struct scsi_inquiry_data *inqbuf;
+ char vend[8+16+4+1];
+ int i, rv = EINVAL;
+ uint32_t size;
+ uint8_t mbox[MFI_MBOX_SIZE];
+
+ DNPRINTF(MFI_D_IOCTL, "%s: mfi_vol_hs %d\n", DEVNAME(sc), volid);
+
+ if (!bio_hs)
+ return (EINVAL);
+
+ pd = malloc(sizeof *pd, M_DEVBUF, M_WAITOK);
+
+ /* send single element command to retrieve size for full structure */
+ cfg = malloc(sizeof *cfg, M_DEVBUF, M_WAITOK);
+ if (mfi_mgmt(sc, MD_DCMD_CONF_GET, MFI_DATA_IN, sizeof *cfg, cfg, NULL))
+ goto freeme;
+
+ size = cfg->mfc_size;
+ free(cfg, M_DEVBUF);
+
+ /* memory for read config */
+ cfg = malloc(size, M_DEVBUF, M_WAITOK);
+ memset(cfg, 0, size);
+ if (mfi_mgmt(sc, MD_DCMD_CONF_GET, MFI_DATA_IN, size, cfg, NULL))
+ goto freeme;
+
+ /* calculate offset to hs structure */
+ hs = (struct mfi_hotspare *)(
+ ((uint8_t *)cfg) + offsetof(struct mfi_conf, mfc_array) +
+ cfg->mfc_array_size * cfg->mfc_no_array +
+ cfg->mfc_ld_size * cfg->mfc_no_ld);
+
+ if (volid < cfg->mfc_no_ld)
+ goto freeme; /* not a hotspare */
+
+ if (volid > (cfg->mfc_no_ld + cfg->mfc_no_hs))
+ goto freeme; /* not a hotspare */
+
+ /* offset into hotspare structure */
+ i = volid - cfg->mfc_no_ld;
+
+ DNPRINTF(MFI_D_IOCTL, "%s: mfi_vol_hs i %d volid %d no_ld %d no_hs %d "
+ "hs %p cfg %p id %02x\n", DEVNAME(sc), i, volid, cfg->mfc_no_ld,
+ cfg->mfc_no_hs, hs, cfg, hs[i].mhs_pd.mfp_id);
+
+ /* get pd fields */
+ memset(mbox, 0, sizeof mbox);
+ *((uint16_t *)&mbox) = hs[i].mhs_pd.mfp_id;
+ if (mfi_mgmt(sc, MR_DCMD_PD_GET_INFO, MFI_DATA_IN,
+ sizeof *pd, pd, mbox)) {
+ DNPRINTF(MFI_D_IOCTL, "%s: mfi_vol_hs illegal PD\n",
+ DEVNAME(sc));
+ goto freeme;
+ }
+
+ switch (type) {
+ case MFI_MGMT_VD:
+ vdhs = bio_hs;
+ vdhs->bv_status = BIOC_SVONLINE;
+ vdhs->bv_size = pd->mpd_size;
+ vdhs->bv_level = -1; /* hotspare */
+ vdhs->bv_nodisk = 1;
+ break;
+
+ case MFI_MGMT_SD:
+ sdhs = bio_hs;
+ sdhs->bd_status = BIOC_SDHOTSPARE;
+ sdhs->bd_size = pd->mpd_size;
+ sdhs->bd_channel = pd->mpd_enc_idx;
+ sdhs->bd_target = pd->mpd_enc_slot;
+ inqbuf = (struct scsi_inquiry_data *)&pd->mpd_inq_data;
+ memcpy(vend, inqbuf->vendor, sizeof vend - 1);
+ vend[sizeof vend - 1] = '\0';
+ strlcpy(sdhs->bd_vendor, vend, sizeof(sdhs->bd_vendor));
+ break;
+
+ default:
+ goto freeme;
+ }
+
+ DNPRINTF(MFI_D_IOCTL, "%s: mfi_vol_hs 6\n", DEVNAME(sc));
+ rv = 0;
+freeme:
+ free(pd, M_DEVBUF);
+ free(cfg, M_DEVBUF);
+
+ return (rv);
+}
+
#endif /* NBIO > 0 */