diff options
-rw-r--r-- | sys/dev/ic/mfi.c | 90 | ||||
-rw-r--r-- | sys/dev/ic/mfireg.h | 121 | ||||
-rw-r--r-- | sys/dev/ic/mfivar.h | 3 |
3 files changed, 204 insertions, 10 deletions
diff --git a/sys/dev/ic/mfi.c b/sys/dev/ic/mfi.c index dd2c7d686f8..9aabe9ad265 100644 --- a/sys/dev/ic/mfi.c +++ b/sys/dev/ic/mfi.c @@ -1,4 +1,4 @@ -/* $OpenBSD: mfi.c,v 1.53 2006/05/23 03:55:41 deraadt Exp $ */ +/* $OpenBSD: mfi.c,v 1.54 2006/05/25 00:21:31 marco Exp $ */ /* * Copyright (c) 2006 Marco Peereboom <marco@peereboom.us> * @@ -671,9 +671,6 @@ mfi_attach(struct mfi_softc *sc) /* enable interrupts */ mfi_write(sc, MFI_OMSK, MFI_ENABLE_INTR); - /* memory for physical disk map */ - sc->sc_pd_list = malloc(MFI_PD_LIST_SIZE, M_DEVBUF, M_WAITOK); - return (0); noinit: mfi_freemem(sc, sc->sc_sense); @@ -1333,15 +1330,94 @@ done: int mfi_ioctl_disk(struct mfi_softc *sc, struct bioc_disk *bd) { - int rv = EINVAL; + struct mfi_conf *cfg; + struct mfi_array *ar; + struct mfi_ld_cfg *ld; + int i, rv = EINVAL; + int arr, vol, disk; + uint32_t size; DNPRINTF(MFI_D_IOCTL, "%s: mfi_ioctl_disk %#x\n", DEVNAME(sc), bd->bd_diskid); - if (mfi_mgmt(sc, MR_DCMD_PD_GET_LIST, MFI_DATA_IN, - MFI_PD_LIST_SIZE, sc->sc_pd_list, NULL)) + memset(&cfg, 0, sizeof cfg); + + /* 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 done; + + 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; + + ar = cfg->mfc_array; + + /* calculate offset to ld structure */ + ld = (struct mfi_ld_cfg *)( + ((uint8_t *)cfg) + offsetof(struct mfi_conf, mfc_array) + + cfg->mfc_array_size * cfg->mfc_no_array); + + vol = bd->bd_volid; + + if (vol > cfg->mfc_no_ld) { + /* XXX do hotspares */ + goto freeme; + } + + /* find corresponding array for ld */ + for (i = 0, arr = 0; i < vol; i++) + arr += ld[i].mlc_parm.mpa_span_depth; + + /* offset disk into pd list */ + disk = bd->bd_diskid % ld[vol].mlc_parm.mpa_no_drv_per_span; + + /* offset array index into the next spans */ + arr += bd->bd_diskid / ld[vol].mlc_parm.mpa_no_drv_per_span; + + bd->bd_target = ar[arr].pd[disk].mar_enc_slot; + + switch (ar[arr].pd[disk].mar_pd_state){ + case MFI_PD_UNCONFIG_GOOD: + bd->bd_status = BIOC_SDUNUSED; + break; + + case MFI_PD_HOTSPARE: + bd->bd_status = BIOC_SDHOTSPARE; + break; + + case MFI_PD_OFFLINE: + bd->bd_status = BIOC_SDOFFLINE; + break; + + case MFI_PD_FAILED: + bd->bd_status = BIOC_SDFAILED; + break; + + case MFI_PD_REBUILD: + bd->bd_status = BIOC_SDREBUILD; + break; + + case MFI_PD_ONLINE: + bd->bd_status = BIOC_SDONLINE; + break; + + case MFI_PD_UNCONFIG_BAD: /* XXX define new state in bio */ + default: + bd->bd_status = BIOC_SDINVALID; + break; + + } + rv = 0; +freeme: + free(cfg, M_DEVBUF); + cfg = NULL; done: return (rv); } diff --git a/sys/dev/ic/mfireg.h b/sys/dev/ic/mfireg.h index fddbbad7008..fa2a78c9384 100644 --- a/sys/dev/ic/mfireg.h +++ b/sys/dev/ic/mfireg.h @@ -1,4 +1,4 @@ -/* $OpenBSD: mfireg.h,v 1.19 2006/05/23 00:49:26 marco Exp $ */ +/* $OpenBSD: mfireg.h,v 1.20 2006/05/25 00:21:31 marco Exp $ */ /* * Copyright (c) 2006 Marco Peereboom <marco@peereboom.us> * @@ -97,6 +97,7 @@ #define MR_DCMD_LD_GET_LIST 0x03010000 #define MR_DCMD_LD_GET_INFO 0x03020000 #define MR_DCMD_LD_GET_PROPERTIES 0x03030000 +#define MD_DCMD_CONF_GET 0x04010000 #define MR_DCMD_CLUSTER 0x08000000 #define MR_DCMD_CLUSTER_RESET_ALL 0x08010100 #define MR_DCMD_CLUSTER_RESET_LD 0x08010200 @@ -220,6 +221,7 @@ typedef enum { /* driver definitions */ #define MFI_MAX_PD_CHANNELS 2 +#define MFI_MAX_PD_ARRAY 32 #define MFI_MAX_LD_CHANNELS 2 #define MFI_MAX_CHANNELS (MFI_MAX_PD_CHANNELS + MFI_MAX_LD_CHANNELS) #define MFI_MAX_CHANNEL_DEVS 128 @@ -227,6 +229,7 @@ typedef enum { #define MFI_MAX_LUN 8 #define MFI_MAX_LD 64 #define MFI_MAX_SPAN 8 +#define MFI_MAX_ARRAY_DEDICATED 16 /* sense buffer */ struct mfi_sense { @@ -787,3 +790,119 @@ struct mfi_pd_list { struct mfi_pd_address mpl_address[1]; } __packed; #define MFI_PD_LIST_SIZE (256 * sizeof(struct mfi_pd_address) + 8) + +struct mfi_pd { + uint16_t mfp_id; + uint16_t mfp_seq; +} __packed; + +struct mfi_pd_progress { + uint32_t mfp_in_prog; +#define MFI_PD_PROG_RBLD 0x01 +#define MFI_PD_PROG_PR 0x02 +#define MFI_PD_PROG_CLEAR 0x04 + struct mfi_progress mfp_rebuild; + struct mfi_progress mfp_patrol_read; + struct mfi_progress mfp_clear; + struct mfi_progress mfp_res[4]; +} __packed; + +struct mfi_pd_details { + struct mfi_pd mpd_pd; + uint8_t mpd_inq_data[96]; + uint8_t mpd_inq_page83[64]; + uint8_t mpd_no_support; + uint8_t mpd_scsy_type; + uint8_t mpd_port; + uint8_t mpd_speed; + uint32_t mpd_mediaerr_cnt; + uint32_t mpd_othererr_cnt; + uint32_t mpd_predfail_cnt; + uint32_t mpd_last_pred_event; + uint16_t mpd_fw_state; + uint8_t mpd_rdy_for_remove; + uint8_t mpd_link_speed; + uint32_t mpd_ddf_state; +#define MFI_DDF_GUID_FORCED 0x01 +#define MFI_DDF_PART_OF_VD 0x02 +#define MFI_DDF_GLOB_HOTSPARE 0x04 +#define MFI_DDF_HOTSPARE 0x08 +#define MFI_DDF_FOREIGN 0x10 +#define MFI_DDF_TYPE_MASK 0xf000 +#define MFI_DDF_TYPE_UNKNOWN 0x0000 +#define MFI_DDF_TYPE_PAR_SCSI 0x1000 +#define MFI_DDF_TYPE_SAS 0x2000 +#define MFI_DDF_TYPE_SATA 0x3000 +#define MFI_DDF_TYPE_FC 0x4000 + struct { + uint8_t mpp_cnt; + uint8_t mpp_severed; + uint8_t mpp_res[6]; + u_quad_t mpp_sas_addr[4]; + } __packed mpd_path; + u_quad_t mpd_size; + u_quad_t mpd_no_coerce_size; + u_quad_t mpd_coerce_size; + uint16_t mpd_enc_id; + uint8_t mpd_enc_idx; + uint8_t mpd_slot; + struct mfi_pd_progress mpd_progress; + uint8_t mpd_bblock_full; + uint8_t mpd_unusable; + uint8_t mpd_res[218]; /* size is 512 */ +} __packed; + +/* array configuration from MD_DCMD_CONF_GET */ +struct mfi_array { + u_quad_t mar_smallest_pd; + uint8_t mar_no_disk; + uint8_t mar_res1; + uint16_t mar_array_ref; + uint8_t mar_res2[20]; + struct { + struct mfi_pd mar_pd; + uint16_t mar_pd_state; +#define MFI_PD_UNCONFIG_GOOD 0x00 +#define MFI_PD_UNCONFIG_BAD 0x01 +#define MFI_PD_HOTSPARE 0x02 +#define MFI_PD_OFFLINE 0x10 +#define MFI_PD_FAILED 0x11 +#define MFI_PD_REBUILD 0x14 +#define MFI_PD_ONLINE 0x18 + uint8_t mar_enc_pd; + uint8_t mar_enc_slot; + } pd[MFI_MAX_PD_ARRAY]; +} __packed; + +struct mfi_hotspare { + struct mfi_pd mhs_pd; + uint8_t mhs_type; +#define MFI_PD_HS_DEDICATED 0x01 +#define MFI_PD_HS_REVERTIBLE 0x02 +#define MFI_PD_HS_ENC_AFFINITY 0x04 + uint8_t mhs_res[2]; + uint8_t mhs_array_max; + uint16_t mhs_array_ref[MFI_MAX_ARRAY_DEDICATED]; +} __packed; + +struct mfi_conf { + uint32_t mfc_size; + uint16_t mfc_no_array; + uint16_t mfc_array_size; + uint16_t mfc_no_ld; + uint16_t mfc_ld_size; + uint16_t mfc_no_hs; + uint16_t mfc_hs_size; + uint8_t mfc_res[16]; + /* + * XXX this is a ridiculous hack and does not reflect reality + * Structures are actually indexed and therefore need pointer + * math to reach. We need the size of this structure first so + * call it with the size of this structure and then use the returned + * values to allocate memory and do the transfer of the whole structure + * then calculate pointers to each of these structures. + */ + struct mfi_array mfc_array[1]; + struct mfi_ld_cfg mfc_ld[1]; + struct mfi_hotspare mfc_hs[1]; +} __packed; diff --git a/sys/dev/ic/mfivar.h b/sys/dev/ic/mfivar.h index 25ecfe38015..0a1b7f9c310 100644 --- a/sys/dev/ic/mfivar.h +++ b/sys/dev/ic/mfivar.h @@ -1,4 +1,4 @@ -/* $OpenBSD: mfivar.h,v 1.24 2006/05/23 00:43:57 marco Exp $ */ +/* $OpenBSD: mfivar.h,v 1.25 2006/05/25 00:21:31 marco Exp $ */ /* * Copyright (c) 2006 Marco Peereboom <marco@peereboom.us> * @@ -125,7 +125,6 @@ struct mfi_softc { struct mfi_ctrl_info sc_info; struct mfi_ld_list sc_ld_list; struct mfi_ld_details sc_ld_details; - struct mfi_pd_list *sc_pd_list; /* all commands */ struct mfi_ccb *sc_ccb; |