1665484d8SDoug Ambrisko /* 2665484d8SDoug Ambrisko * Copyright (c) 2014, LSI Corp. 3665484d8SDoug Ambrisko * All rights reserved. 4665484d8SDoug Ambrisko * Author: Marian Choy 5665484d8SDoug Ambrisko * Support: freebsdraid@lsi.com 6665484d8SDoug Ambrisko * 7665484d8SDoug Ambrisko * Redistribution and use in source and binary forms, with or without 8665484d8SDoug Ambrisko * modification, are permitted provided that the following conditions 9665484d8SDoug Ambrisko * are met: 10665484d8SDoug Ambrisko * 11665484d8SDoug Ambrisko * 1. Redistributions of source code must retain the above copyright 12665484d8SDoug Ambrisko * notice, this list of conditions and the following disclaimer. 13665484d8SDoug Ambrisko * 2. Redistributions in binary form must reproduce the above copyright 14665484d8SDoug Ambrisko * notice, this list of conditions and the following disclaimer in 15665484d8SDoug Ambrisko * the documentation and/or other materials provided with the 16665484d8SDoug Ambrisko * distribution. 17665484d8SDoug Ambrisko * 3. Neither the name of the <ORGANIZATION> nor the names of its 18665484d8SDoug Ambrisko * contributors may be used to endorse or promote products derived 19665484d8SDoug Ambrisko * from this software without specific prior written permission. 20665484d8SDoug Ambrisko * 21665484d8SDoug Ambrisko * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS 22665484d8SDoug Ambrisko * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT 23665484d8SDoug Ambrisko * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS 24665484d8SDoug Ambrisko * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE 25665484d8SDoug Ambrisko * COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, 26665484d8SDoug Ambrisko * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, 27665484d8SDoug Ambrisko * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; 28665484d8SDoug Ambrisko * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER 29665484d8SDoug Ambrisko * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 30665484d8SDoug Ambrisko * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN 31665484d8SDoug Ambrisko * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 32665484d8SDoug Ambrisko * POSSIBILITY OF SUCH DAMAGE. 33665484d8SDoug Ambrisko * 34665484d8SDoug Ambrisko * The views and conclusions contained in the software and documentation 35665484d8SDoug Ambrisko * are those of the authors and should not be interpreted as representing 36665484d8SDoug Ambrisko * official policies,either expressed or implied, of the FreeBSD Project. 37665484d8SDoug Ambrisko * 38665484d8SDoug Ambrisko * Send feedback to: <megaraidfbsd@lsi.com> 39665484d8SDoug Ambrisko * Mail to: LSI Corporation, 1621 Barber Lane, Milpitas, CA 95035 40665484d8SDoug Ambrisko * ATTN: MegaRaid FreeBSD 41665484d8SDoug Ambrisko * 42665484d8SDoug Ambrisko */ 43665484d8SDoug Ambrisko 44665484d8SDoug Ambrisko #include <sys/cdefs.h> 45665484d8SDoug Ambrisko __FBSDID("$FreeBSD$"); 46665484d8SDoug Ambrisko 47665484d8SDoug Ambrisko #include <dev/mrsas/mrsas.h> 48665484d8SDoug Ambrisko #include <dev/mrsas/mrsas_ioctl.h> 49665484d8SDoug Ambrisko 50665484d8SDoug Ambrisko #include <cam/cam.h> 51665484d8SDoug Ambrisko #include <cam/cam_ccb.h> 52665484d8SDoug Ambrisko 53665484d8SDoug Ambrisko #include <sys/sysctl.h> 54665484d8SDoug Ambrisko #include <sys/types.h> 55665484d8SDoug Ambrisko #include <sys/kthread.h> 56665484d8SDoug Ambrisko #include <sys/taskqueue.h> 57665484d8SDoug Ambrisko 58665484d8SDoug Ambrisko 59665484d8SDoug Ambrisko /* 60665484d8SDoug Ambrisko * Function prototypes 61665484d8SDoug Ambrisko */ 62665484d8SDoug Ambrisko static d_open_t mrsas_open; 63665484d8SDoug Ambrisko static d_close_t mrsas_close; 64665484d8SDoug Ambrisko static d_read_t mrsas_read; 65665484d8SDoug Ambrisko static d_write_t mrsas_write; 66665484d8SDoug Ambrisko static d_ioctl_t mrsas_ioctl; 67665484d8SDoug Ambrisko 68665484d8SDoug Ambrisko static struct mrsas_ident *mrsas_find_ident(device_t); 69665484d8SDoug Ambrisko static void mrsas_shutdown_ctlr(struct mrsas_softc *sc, u_int32_t opcode); 70665484d8SDoug Ambrisko static void mrsas_flush_cache(struct mrsas_softc *sc); 71665484d8SDoug Ambrisko static void mrsas_reset_reply_desc(struct mrsas_softc *sc); 72665484d8SDoug Ambrisko static void mrsas_ocr_thread(void *arg); 73665484d8SDoug Ambrisko static int mrsas_get_map_info(struct mrsas_softc *sc); 74665484d8SDoug Ambrisko static int mrsas_get_ld_map_info(struct mrsas_softc *sc); 75665484d8SDoug Ambrisko static int mrsas_sync_map_info(struct mrsas_softc *sc); 76665484d8SDoug Ambrisko static int mrsas_get_pd_list(struct mrsas_softc *sc); 77665484d8SDoug Ambrisko static int mrsas_get_ld_list(struct mrsas_softc *sc); 78665484d8SDoug Ambrisko static int mrsas_setup_irq(struct mrsas_softc *sc); 79665484d8SDoug Ambrisko static int mrsas_alloc_mem(struct mrsas_softc *sc); 80665484d8SDoug Ambrisko static int mrsas_init_fw(struct mrsas_softc *sc); 81665484d8SDoug Ambrisko static int mrsas_setup_raidmap(struct mrsas_softc *sc); 82665484d8SDoug Ambrisko static int mrsas_complete_cmd(struct mrsas_softc *sc); 83665484d8SDoug Ambrisko static int mrsas_clear_intr(struct mrsas_softc *sc); 84665484d8SDoug Ambrisko static int mrsas_get_ctrl_info(struct mrsas_softc *sc, 85665484d8SDoug Ambrisko struct mrsas_ctrl_info *ctrl_info); 86665484d8SDoug Ambrisko static int mrsas_issue_blocked_abort_cmd(struct mrsas_softc *sc, 87665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd_to_abort); 88665484d8SDoug Ambrisko u_int32_t mrsas_read_reg(struct mrsas_softc *sc, int offset); 89665484d8SDoug Ambrisko u_int8_t mrsas_build_mptmfi_passthru(struct mrsas_softc *sc, 90665484d8SDoug Ambrisko struct mrsas_mfi_cmd *mfi_cmd); 91665484d8SDoug Ambrisko int mrsas_transition_to_ready(struct mrsas_softc *sc, int ocr); 92665484d8SDoug Ambrisko int mrsas_init_adapter(struct mrsas_softc *sc); 93665484d8SDoug Ambrisko int mrsas_alloc_mpt_cmds(struct mrsas_softc *sc); 94665484d8SDoug Ambrisko int mrsas_alloc_ioc_cmd(struct mrsas_softc *sc); 95665484d8SDoug Ambrisko int mrsas_alloc_ctlr_info_cmd(struct mrsas_softc *sc); 96665484d8SDoug Ambrisko int mrsas_ioc_init(struct mrsas_softc *sc); 97665484d8SDoug Ambrisko int mrsas_bus_scan(struct mrsas_softc *sc); 98665484d8SDoug Ambrisko int mrsas_issue_dcmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 99665484d8SDoug Ambrisko int mrsas_issue_polled(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 100665484d8SDoug Ambrisko int mrsas_reset_ctrl(struct mrsas_softc *sc); 101665484d8SDoug Ambrisko int mrsas_wait_for_outstanding(struct mrsas_softc *sc); 102665484d8SDoug Ambrisko int mrsas_issue_blocked_cmd(struct mrsas_softc *sc, 103665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd); 104665484d8SDoug Ambrisko int mrsas_alloc_tmp_dcmd(struct mrsas_softc *sc, struct mrsas_tmp_dcmd *tcmd, 105665484d8SDoug Ambrisko int size); 106665484d8SDoug Ambrisko void mrsas_release_mfi_cmd(struct mrsas_mfi_cmd *cmd); 107665484d8SDoug Ambrisko void mrsas_wakeup(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 108665484d8SDoug Ambrisko void mrsas_complete_aen(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 109665484d8SDoug Ambrisko void mrsas_complete_abort(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 110665484d8SDoug Ambrisko void mrsas_disable_intr(struct mrsas_softc *sc); 111665484d8SDoug Ambrisko void mrsas_enable_intr(struct mrsas_softc *sc); 112665484d8SDoug Ambrisko void mrsas_free_ioc_cmd(struct mrsas_softc *sc); 113665484d8SDoug Ambrisko void mrsas_free_mem(struct mrsas_softc *sc); 114665484d8SDoug Ambrisko void mrsas_free_tmp_dcmd(struct mrsas_tmp_dcmd *tmp); 115665484d8SDoug Ambrisko void mrsas_isr(void *arg); 116665484d8SDoug Ambrisko void mrsas_teardown_intr(struct mrsas_softc *sc); 117665484d8SDoug Ambrisko void mrsas_addr_cb(void *arg, bus_dma_segment_t *segs, int nsegs, int error); 118665484d8SDoug Ambrisko void mrsas_kill_hba (struct mrsas_softc *sc); 119665484d8SDoug Ambrisko void mrsas_aen_handler(struct mrsas_softc *sc); 120665484d8SDoug Ambrisko void mrsas_write_reg(struct mrsas_softc *sc, int offset, 121665484d8SDoug Ambrisko u_int32_t value); 122665484d8SDoug Ambrisko void mrsas_fire_cmd(struct mrsas_softc *sc, u_int32_t req_desc_lo, 123665484d8SDoug Ambrisko u_int32_t req_desc_hi); 124665484d8SDoug Ambrisko void mrsas_free_ctlr_info_cmd(struct mrsas_softc *sc); 125665484d8SDoug Ambrisko void mrsas_complete_mptmfi_passthru(struct mrsas_softc *sc, 126665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd, u_int8_t status); 127665484d8SDoug Ambrisko void mrsas_map_mpt_cmd_status(struct mrsas_mpt_cmd *cmd, u_int8_t status, 128665484d8SDoug Ambrisko u_int8_t extStatus); 129665484d8SDoug Ambrisko struct mrsas_mfi_cmd* mrsas_get_mfi_cmd(struct mrsas_softc *sc); 130665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION * mrsas_build_mpt_cmd(struct mrsas_softc *sc, 131665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd); 132665484d8SDoug Ambrisko 133665484d8SDoug Ambrisko extern int mrsas_cam_attach(struct mrsas_softc *sc); 134665484d8SDoug Ambrisko extern void mrsas_cam_detach(struct mrsas_softc *sc); 135665484d8SDoug Ambrisko extern void mrsas_cmd_done(struct mrsas_softc *sc, struct mrsas_mpt_cmd *cmd); 136665484d8SDoug Ambrisko extern void mrsas_free_frame(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 137665484d8SDoug Ambrisko extern int mrsas_alloc_mfi_cmds(struct mrsas_softc *sc); 138665484d8SDoug Ambrisko extern void mrsas_release_mpt_cmd(struct mrsas_mpt_cmd *cmd); 139665484d8SDoug Ambrisko extern struct mrsas_mpt_cmd *mrsas_get_mpt_cmd(struct mrsas_softc *sc); 140665484d8SDoug Ambrisko extern int mrsas_passthru(struct mrsas_softc *sc, void *arg); 141665484d8SDoug Ambrisko extern uint8_t MR_ValidateMapInfo(struct mrsas_softc *sc); 142*4799d485SKashyap D Desai extern u_int16_t MR_GetLDTgtId(u_int32_t ld, MR_DRV_RAID_MAP_ALL *map); 143*4799d485SKashyap D Desai extern MR_LD_RAID *MR_LdRaidGet(u_int32_t ld, MR_DRV_RAID_MAP_ALL *map); 144665484d8SDoug Ambrisko extern void mrsas_xpt_freeze(struct mrsas_softc *sc); 145665484d8SDoug Ambrisko extern void mrsas_xpt_release(struct mrsas_softc *sc); 146665484d8SDoug Ambrisko extern MRSAS_REQUEST_DESCRIPTOR_UNION *mrsas_get_request_desc(struct mrsas_softc *sc, 147665484d8SDoug Ambrisko u_int16_t index); 148665484d8SDoug Ambrisko extern int mrsas_bus_scan_sim(struct mrsas_softc *sc, struct cam_sim *sim); 149665484d8SDoug Ambrisko static int mrsas_alloc_evt_log_info_cmd(struct mrsas_softc *sc); 150665484d8SDoug Ambrisko static void mrsas_free_evt_log_info_cmd(struct mrsas_softc *sc); 151665484d8SDoug Ambrisko SYSCTL_NODE(_hw, OID_AUTO, mrsas, CTLFLAG_RD, 0, "MRSAS Driver Parameters"); 152665484d8SDoug Ambrisko 153665484d8SDoug Ambrisko 154665484d8SDoug Ambrisko /** 155665484d8SDoug Ambrisko * PCI device struct and table 156665484d8SDoug Ambrisko * 157665484d8SDoug Ambrisko */ 158665484d8SDoug Ambrisko typedef struct mrsas_ident { 159665484d8SDoug Ambrisko uint16_t vendor; 160665484d8SDoug Ambrisko uint16_t device; 161665484d8SDoug Ambrisko uint16_t subvendor; 162665484d8SDoug Ambrisko uint16_t subdevice; 163665484d8SDoug Ambrisko const char *desc; 164665484d8SDoug Ambrisko } MRSAS_CTLR_ID; 165665484d8SDoug Ambrisko 166665484d8SDoug Ambrisko MRSAS_CTLR_ID device_table[] = { 167665484d8SDoug Ambrisko {0x1000, MRSAS_TBOLT, 0xffff, 0xffff, "LSI Thunderbolt SAS Controller"}, 168665484d8SDoug Ambrisko {0x1000, MRSAS_INVADER, 0xffff, 0xffff, "LSI Invader SAS Controller"}, 169665484d8SDoug Ambrisko {0x1000, MRSAS_FURY, 0xffff, 0xffff, "LSI Fury SAS Controller"}, 170665484d8SDoug Ambrisko {0, 0, 0, 0, NULL} 171665484d8SDoug Ambrisko }; 172665484d8SDoug Ambrisko 173665484d8SDoug Ambrisko /** 174665484d8SDoug Ambrisko * Character device entry points 175665484d8SDoug Ambrisko * 176665484d8SDoug Ambrisko */ 177665484d8SDoug Ambrisko static struct cdevsw mrsas_cdevsw = { 178665484d8SDoug Ambrisko .d_version = D_VERSION, 179665484d8SDoug Ambrisko .d_open = mrsas_open, 180665484d8SDoug Ambrisko .d_close = mrsas_close, 181665484d8SDoug Ambrisko .d_read = mrsas_read, 182665484d8SDoug Ambrisko .d_write = mrsas_write, 183665484d8SDoug Ambrisko .d_ioctl = mrsas_ioctl, 184665484d8SDoug Ambrisko .d_name = "mrsas", 185665484d8SDoug Ambrisko }; 186665484d8SDoug Ambrisko 187665484d8SDoug Ambrisko MALLOC_DEFINE(M_MRSAS, "mrsasbuf", "Buffers for the MRSAS driver"); 188665484d8SDoug Ambrisko 189665484d8SDoug Ambrisko /** 190665484d8SDoug Ambrisko * In the cdevsw routines, we find our softc by using the si_drv1 member 191665484d8SDoug Ambrisko * of struct cdev. We set this variable to point to our softc in our 192665484d8SDoug Ambrisko * attach routine when we create the /dev entry. 193665484d8SDoug Ambrisko */ 194665484d8SDoug Ambrisko int 195665484d8SDoug Ambrisko mrsas_open(struct cdev *dev, int oflags, int devtype, d_thread_t *td) 196665484d8SDoug Ambrisko { 197665484d8SDoug Ambrisko struct mrsas_softc *sc; 198665484d8SDoug Ambrisko 199665484d8SDoug Ambrisko sc = dev->si_drv1; 200665484d8SDoug Ambrisko return (0); 201665484d8SDoug Ambrisko } 202665484d8SDoug Ambrisko 203665484d8SDoug Ambrisko int 204665484d8SDoug Ambrisko mrsas_close(struct cdev *dev, int fflag, int devtype, d_thread_t *td) 205665484d8SDoug Ambrisko { 206665484d8SDoug Ambrisko struct mrsas_softc *sc; 207665484d8SDoug Ambrisko 208665484d8SDoug Ambrisko sc = dev->si_drv1; 209665484d8SDoug Ambrisko return (0); 210665484d8SDoug Ambrisko } 211665484d8SDoug Ambrisko 212665484d8SDoug Ambrisko int 213665484d8SDoug Ambrisko mrsas_read(struct cdev *dev, struct uio *uio, int ioflag) 214665484d8SDoug Ambrisko { 215665484d8SDoug Ambrisko struct mrsas_softc *sc; 216665484d8SDoug Ambrisko 217665484d8SDoug Ambrisko sc = dev->si_drv1; 218665484d8SDoug Ambrisko return (0); 219665484d8SDoug Ambrisko } 220665484d8SDoug Ambrisko int 221665484d8SDoug Ambrisko mrsas_write(struct cdev *dev, struct uio *uio, int ioflag) 222665484d8SDoug Ambrisko { 223665484d8SDoug Ambrisko struct mrsas_softc *sc; 224665484d8SDoug Ambrisko 225665484d8SDoug Ambrisko sc = dev->si_drv1; 226665484d8SDoug Ambrisko return (0); 227665484d8SDoug Ambrisko } 228665484d8SDoug Ambrisko 229665484d8SDoug Ambrisko /** 230665484d8SDoug Ambrisko * Register Read/Write Functions 231665484d8SDoug Ambrisko * 232665484d8SDoug Ambrisko */ 233665484d8SDoug Ambrisko void 234665484d8SDoug Ambrisko mrsas_write_reg(struct mrsas_softc *sc, int offset, 235665484d8SDoug Ambrisko u_int32_t value) 236665484d8SDoug Ambrisko { 237665484d8SDoug Ambrisko bus_space_tag_t bus_tag = sc->bus_tag; 238665484d8SDoug Ambrisko bus_space_handle_t bus_handle = sc->bus_handle; 239665484d8SDoug Ambrisko 240665484d8SDoug Ambrisko bus_space_write_4(bus_tag, bus_handle, offset, value); 241665484d8SDoug Ambrisko } 242665484d8SDoug Ambrisko 243665484d8SDoug Ambrisko u_int32_t 244665484d8SDoug Ambrisko mrsas_read_reg(struct mrsas_softc *sc, int offset) 245665484d8SDoug Ambrisko { 246665484d8SDoug Ambrisko bus_space_tag_t bus_tag = sc->bus_tag; 247665484d8SDoug Ambrisko bus_space_handle_t bus_handle = sc->bus_handle; 248665484d8SDoug Ambrisko 249665484d8SDoug Ambrisko return((u_int32_t)bus_space_read_4(bus_tag, bus_handle, offset)); 250665484d8SDoug Ambrisko } 251665484d8SDoug Ambrisko 252665484d8SDoug Ambrisko 253665484d8SDoug Ambrisko /** 254665484d8SDoug Ambrisko * Interrupt Disable/Enable/Clear Functions 255665484d8SDoug Ambrisko * 256665484d8SDoug Ambrisko */ 257665484d8SDoug Ambrisko void mrsas_disable_intr(struct mrsas_softc *sc) 258665484d8SDoug Ambrisko { 259665484d8SDoug Ambrisko u_int32_t mask = 0xFFFFFFFF; 260665484d8SDoug Ambrisko u_int32_t status; 261665484d8SDoug Ambrisko 262665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask), mask); 263665484d8SDoug Ambrisko /* Dummy read to force pci flush */ 264665484d8SDoug Ambrisko status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask)); 265665484d8SDoug Ambrisko } 266665484d8SDoug Ambrisko 267665484d8SDoug Ambrisko void mrsas_enable_intr(struct mrsas_softc *sc) 268665484d8SDoug Ambrisko { 269665484d8SDoug Ambrisko u_int32_t mask = MFI_FUSION_ENABLE_INTERRUPT_MASK; 270665484d8SDoug Ambrisko u_int32_t status; 271665484d8SDoug Ambrisko 272665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status), ~0); 273665484d8SDoug Ambrisko status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status)); 274665484d8SDoug Ambrisko 275665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask), ~mask); 276665484d8SDoug Ambrisko status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask)); 277665484d8SDoug Ambrisko } 278665484d8SDoug Ambrisko 279665484d8SDoug Ambrisko static int mrsas_clear_intr(struct mrsas_softc *sc) 280665484d8SDoug Ambrisko { 281665484d8SDoug Ambrisko u_int32_t status, fw_status, fw_state; 282665484d8SDoug Ambrisko 283665484d8SDoug Ambrisko /* Read received interrupt */ 284665484d8SDoug Ambrisko status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status)); 285665484d8SDoug Ambrisko 286665484d8SDoug Ambrisko /* If FW state change interrupt is received, write to it again to clear */ 287665484d8SDoug Ambrisko if (status & MRSAS_FW_STATE_CHNG_INTERRUPT) { 288665484d8SDoug Ambrisko fw_status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 289665484d8SDoug Ambrisko outbound_scratch_pad)); 290665484d8SDoug Ambrisko fw_state = fw_status & MFI_STATE_MASK; 291665484d8SDoug Ambrisko if (fw_state == MFI_STATE_FAULT) { 292665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "FW is in FAULT state!\n"); 293665484d8SDoug Ambrisko if(sc->ocr_thread_active) 294665484d8SDoug Ambrisko wakeup(&sc->ocr_chan); 295665484d8SDoug Ambrisko } 296665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status), status); 297665484d8SDoug Ambrisko mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status)); 298665484d8SDoug Ambrisko return(1); 299665484d8SDoug Ambrisko } 300665484d8SDoug Ambrisko 301665484d8SDoug Ambrisko /* Not our interrupt, so just return */ 302665484d8SDoug Ambrisko if (!(status & MFI_FUSION_ENABLE_INTERRUPT_MASK)) 303665484d8SDoug Ambrisko return(0); 304665484d8SDoug Ambrisko 305665484d8SDoug Ambrisko /* We got a reply interrupt */ 306665484d8SDoug Ambrisko return(1); 307665484d8SDoug Ambrisko } 308665484d8SDoug Ambrisko 309665484d8SDoug Ambrisko /** 310665484d8SDoug Ambrisko * PCI Support Functions 311665484d8SDoug Ambrisko * 312665484d8SDoug Ambrisko */ 313665484d8SDoug Ambrisko static struct mrsas_ident * mrsas_find_ident(device_t dev) 314665484d8SDoug Ambrisko { 315665484d8SDoug Ambrisko struct mrsas_ident *pci_device; 316665484d8SDoug Ambrisko 317665484d8SDoug Ambrisko for (pci_device=device_table; pci_device->vendor != 0; pci_device++) 318665484d8SDoug Ambrisko { 319665484d8SDoug Ambrisko if ((pci_device->vendor == pci_get_vendor(dev)) && 320665484d8SDoug Ambrisko (pci_device->device == pci_get_device(dev)) && 321665484d8SDoug Ambrisko ((pci_device->subvendor == pci_get_subvendor(dev)) || 322665484d8SDoug Ambrisko (pci_device->subvendor == 0xffff)) && 323665484d8SDoug Ambrisko ((pci_device->subdevice == pci_get_subdevice(dev)) || 324665484d8SDoug Ambrisko (pci_device->subdevice == 0xffff))) 325665484d8SDoug Ambrisko return (pci_device); 326665484d8SDoug Ambrisko } 327665484d8SDoug Ambrisko return (NULL); 328665484d8SDoug Ambrisko } 329665484d8SDoug Ambrisko 330665484d8SDoug Ambrisko static int mrsas_probe(device_t dev) 331665484d8SDoug Ambrisko { 332665484d8SDoug Ambrisko static u_int8_t first_ctrl = 1; 333665484d8SDoug Ambrisko struct mrsas_ident *id; 334665484d8SDoug Ambrisko 335665484d8SDoug Ambrisko if ((id = mrsas_find_ident(dev)) != NULL) { 336665484d8SDoug Ambrisko if (first_ctrl) { 337665484d8SDoug Ambrisko printf("LSI MegaRAID SAS FreeBSD mrsas driver version: %s\n", MRSAS_VERSION); 338665484d8SDoug Ambrisko first_ctrl = 0; 339665484d8SDoug Ambrisko } 340665484d8SDoug Ambrisko device_set_desc(dev, id->desc); 341665484d8SDoug Ambrisko /* between BUS_PROBE_DEFAULT and BUS_PROBE_LOW_PRIORITY */ 342665484d8SDoug Ambrisko return (-30); 343665484d8SDoug Ambrisko } 344665484d8SDoug Ambrisko return (ENXIO); 345665484d8SDoug Ambrisko } 346665484d8SDoug Ambrisko 347665484d8SDoug Ambrisko /** 348665484d8SDoug Ambrisko * mrsas_setup_sysctl: setup sysctl values for mrsas 349665484d8SDoug Ambrisko * input: Adapter instance soft state 350665484d8SDoug Ambrisko * 351665484d8SDoug Ambrisko * Setup sysctl entries for mrsas driver. 352665484d8SDoug Ambrisko */ 353665484d8SDoug Ambrisko static void 354665484d8SDoug Ambrisko mrsas_setup_sysctl(struct mrsas_softc *sc) 355665484d8SDoug Ambrisko { 356665484d8SDoug Ambrisko struct sysctl_ctx_list *sysctl_ctx = NULL; 357665484d8SDoug Ambrisko struct sysctl_oid *sysctl_tree = NULL; 358665484d8SDoug Ambrisko char tmpstr[80], tmpstr2[80]; 359665484d8SDoug Ambrisko 360665484d8SDoug Ambrisko /* 361665484d8SDoug Ambrisko * Setup the sysctl variable so the user can change the debug level 362665484d8SDoug Ambrisko * on the fly. 363665484d8SDoug Ambrisko */ 364665484d8SDoug Ambrisko snprintf(tmpstr, sizeof(tmpstr), "MRSAS controller %d", 365665484d8SDoug Ambrisko device_get_unit(sc->mrsas_dev)); 366665484d8SDoug Ambrisko snprintf(tmpstr2, sizeof(tmpstr2), "%d", device_get_unit(sc->mrsas_dev)); 367665484d8SDoug Ambrisko 368665484d8SDoug Ambrisko sysctl_ctx = device_get_sysctl_ctx(sc->mrsas_dev); 369665484d8SDoug Ambrisko if (sysctl_ctx != NULL) 370665484d8SDoug Ambrisko sysctl_tree = device_get_sysctl_tree(sc->mrsas_dev); 371665484d8SDoug Ambrisko 372665484d8SDoug Ambrisko if (sysctl_tree == NULL) { 373665484d8SDoug Ambrisko sysctl_ctx_init(&sc->sysctl_ctx); 374665484d8SDoug Ambrisko sc->sysctl_tree = SYSCTL_ADD_NODE(&sc->sysctl_ctx, 375665484d8SDoug Ambrisko SYSCTL_STATIC_CHILDREN(_hw_mrsas), OID_AUTO, tmpstr2, 376665484d8SDoug Ambrisko CTLFLAG_RD, 0, tmpstr); 377665484d8SDoug Ambrisko if (sc->sysctl_tree == NULL) 378665484d8SDoug Ambrisko return; 379665484d8SDoug Ambrisko sysctl_ctx = &sc->sysctl_ctx; 380665484d8SDoug Ambrisko sysctl_tree = sc->sysctl_tree; 381665484d8SDoug Ambrisko } 382665484d8SDoug Ambrisko SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 383665484d8SDoug Ambrisko OID_AUTO, "disable_ocr", CTLFLAG_RW, &sc->disableOnlineCtrlReset, 0, 384665484d8SDoug Ambrisko "Disable the use of OCR"); 385665484d8SDoug Ambrisko 386665484d8SDoug Ambrisko SYSCTL_ADD_STRING(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 387665484d8SDoug Ambrisko OID_AUTO, "driver_version", CTLFLAG_RD, MRSAS_VERSION, 388665484d8SDoug Ambrisko strlen(MRSAS_VERSION), "driver version"); 389665484d8SDoug Ambrisko 390665484d8SDoug Ambrisko SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 391665484d8SDoug Ambrisko OID_AUTO, "reset_count", CTLFLAG_RD, 392665484d8SDoug Ambrisko &sc->reset_count, 0, "number of ocr from start of the day"); 393665484d8SDoug Ambrisko 394665484d8SDoug Ambrisko SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 395665484d8SDoug Ambrisko OID_AUTO, "fw_outstanding", CTLFLAG_RD, 396665484d8SDoug Ambrisko &sc->fw_outstanding, 0, "FW outstanding commands"); 397665484d8SDoug Ambrisko 398665484d8SDoug Ambrisko SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 399665484d8SDoug Ambrisko OID_AUTO, "io_cmds_highwater", CTLFLAG_RD, 400665484d8SDoug Ambrisko &sc->io_cmds_highwater, 0, "Max FW outstanding commands"); 401665484d8SDoug Ambrisko 402665484d8SDoug Ambrisko SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 403665484d8SDoug Ambrisko OID_AUTO, "mrsas_debug", CTLFLAG_RW, &sc->mrsas_debug, 0, 404665484d8SDoug Ambrisko "Driver debug level"); 405665484d8SDoug Ambrisko 406665484d8SDoug Ambrisko SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 407665484d8SDoug Ambrisko OID_AUTO, "mrsas_io_timeout", CTLFLAG_RW, &sc->mrsas_io_timeout, 408665484d8SDoug Ambrisko 0, "Driver IO timeout value in mili-second."); 409665484d8SDoug Ambrisko 410665484d8SDoug Ambrisko SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 411665484d8SDoug Ambrisko OID_AUTO, "mrsas_fw_fault_check_delay", CTLFLAG_RW, 412665484d8SDoug Ambrisko &sc->mrsas_fw_fault_check_delay, 413665484d8SDoug Ambrisko 0, "FW fault check thread delay in seconds. <default is 1 sec>"); 414665484d8SDoug Ambrisko 415665484d8SDoug Ambrisko SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 416665484d8SDoug Ambrisko OID_AUTO, "reset_in_progress", CTLFLAG_RD, 417665484d8SDoug Ambrisko &sc->reset_in_progress, 0, "ocr in progress status"); 418665484d8SDoug Ambrisko 419665484d8SDoug Ambrisko } 420665484d8SDoug Ambrisko 421665484d8SDoug Ambrisko /** 422665484d8SDoug Ambrisko * mrsas_get_tunables: get tunable parameters. 423665484d8SDoug Ambrisko * input: Adapter instance soft state 424665484d8SDoug Ambrisko * 425665484d8SDoug Ambrisko * Get tunable parameters. This will help to debug driver at boot time. 426665484d8SDoug Ambrisko */ 427665484d8SDoug Ambrisko static void 428665484d8SDoug Ambrisko mrsas_get_tunables(struct mrsas_softc *sc) 429665484d8SDoug Ambrisko { 430665484d8SDoug Ambrisko char tmpstr[80]; 431665484d8SDoug Ambrisko 432665484d8SDoug Ambrisko /* XXX default to some debugging for now */ 433665484d8SDoug Ambrisko sc->mrsas_debug = MRSAS_FAULT; 434665484d8SDoug Ambrisko sc->mrsas_io_timeout = MRSAS_IO_TIMEOUT; 435665484d8SDoug Ambrisko sc->mrsas_fw_fault_check_delay = 1; 436665484d8SDoug Ambrisko sc->reset_count = 0; 437665484d8SDoug Ambrisko sc->reset_in_progress = 0; 438665484d8SDoug Ambrisko 439665484d8SDoug Ambrisko /* 440665484d8SDoug Ambrisko * Grab the global variables. 441665484d8SDoug Ambrisko */ 442665484d8SDoug Ambrisko TUNABLE_INT_FETCH("hw.mrsas.debug_level", &sc->mrsas_debug); 443665484d8SDoug Ambrisko 444665484d8SDoug Ambrisko /* Grab the unit-instance variables */ 445665484d8SDoug Ambrisko snprintf(tmpstr, sizeof(tmpstr), "dev.mrsas.%d.debug_level", 446665484d8SDoug Ambrisko device_get_unit(sc->mrsas_dev)); 447665484d8SDoug Ambrisko TUNABLE_INT_FETCH(tmpstr, &sc->mrsas_debug); 448665484d8SDoug Ambrisko } 449665484d8SDoug Ambrisko 450665484d8SDoug Ambrisko /** 451665484d8SDoug Ambrisko * mrsas_alloc_evt_log_info cmd: Allocates memory to get event log information. 452665484d8SDoug Ambrisko * Used to get sequence number at driver load time. 453665484d8SDoug Ambrisko * input: Adapter soft state 454665484d8SDoug Ambrisko * 455665484d8SDoug Ambrisko * Allocates DMAable memory for the event log info internal command. 456665484d8SDoug Ambrisko */ 457665484d8SDoug Ambrisko int mrsas_alloc_evt_log_info_cmd(struct mrsas_softc *sc) 458665484d8SDoug Ambrisko { 459665484d8SDoug Ambrisko int el_info_size; 460665484d8SDoug Ambrisko 461665484d8SDoug Ambrisko /* Allocate get event log info command */ 462665484d8SDoug Ambrisko el_info_size = sizeof(struct mrsas_evt_log_info); 463665484d8SDoug Ambrisko if (bus_dma_tag_create( sc->mrsas_parent_tag, // parent 464665484d8SDoug Ambrisko 1, 0, // algnmnt, boundary 465665484d8SDoug Ambrisko BUS_SPACE_MAXADDR_32BIT,// lowaddr 466665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, // highaddr 467665484d8SDoug Ambrisko NULL, NULL, // filter, filterarg 468665484d8SDoug Ambrisko el_info_size, // maxsize 469665484d8SDoug Ambrisko 1, // msegments 470665484d8SDoug Ambrisko el_info_size, // maxsegsize 471665484d8SDoug Ambrisko BUS_DMA_ALLOCNOW, // flags 472665484d8SDoug Ambrisko NULL, NULL, // lockfunc, lockarg 473665484d8SDoug Ambrisko &sc->el_info_tag)) { 474665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate event log info tag\n"); 475665484d8SDoug Ambrisko return (ENOMEM); 476665484d8SDoug Ambrisko } 477665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->el_info_tag, (void **)&sc->el_info_mem, 478665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->el_info_dmamap)) { 479665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate event log info cmd mem\n"); 480665484d8SDoug Ambrisko return (ENOMEM); 481665484d8SDoug Ambrisko } 482665484d8SDoug Ambrisko if (bus_dmamap_load(sc->el_info_tag, sc->el_info_dmamap, 483665484d8SDoug Ambrisko sc->el_info_mem, el_info_size, mrsas_addr_cb, 484665484d8SDoug Ambrisko &sc->el_info_phys_addr, BUS_DMA_NOWAIT)) { 485665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load event log info cmd mem\n"); 486665484d8SDoug Ambrisko return (ENOMEM); 487665484d8SDoug Ambrisko } 488665484d8SDoug Ambrisko 489665484d8SDoug Ambrisko memset(sc->el_info_mem, 0, el_info_size); 490665484d8SDoug Ambrisko return (0); 491665484d8SDoug Ambrisko } 492665484d8SDoug Ambrisko 493665484d8SDoug Ambrisko /** 494665484d8SDoug Ambrisko * mrsas_free_evt_info_cmd: Free memory for Event log info command 495665484d8SDoug Ambrisko * input: Adapter soft state 496665484d8SDoug Ambrisko * 497665484d8SDoug Ambrisko * Deallocates memory for the event log info internal command. 498665484d8SDoug Ambrisko */ 499665484d8SDoug Ambrisko void mrsas_free_evt_log_info_cmd(struct mrsas_softc *sc) 500665484d8SDoug Ambrisko { 501665484d8SDoug Ambrisko if (sc->el_info_phys_addr) 502665484d8SDoug Ambrisko bus_dmamap_unload(sc->el_info_tag, sc->el_info_dmamap); 503665484d8SDoug Ambrisko if (sc->el_info_mem != NULL) 504665484d8SDoug Ambrisko bus_dmamem_free(sc->el_info_tag, sc->el_info_mem, sc->el_info_dmamap); 505665484d8SDoug Ambrisko if (sc->el_info_tag != NULL) 506665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->el_info_tag); 507665484d8SDoug Ambrisko } 508665484d8SDoug Ambrisko 509665484d8SDoug Ambrisko /** 510665484d8SDoug Ambrisko * mrsas_get_seq_num: Get latest event sequence number 511665484d8SDoug Ambrisko * @sc: Adapter soft state 512665484d8SDoug Ambrisko * @eli: Firmware event log sequence number information. 513665484d8SDoug Ambrisko * Firmware maintains a log of all events in a non-volatile area. 514665484d8SDoug Ambrisko * Driver get the sequence number using DCMD 515665484d8SDoug Ambrisko * "MR_DCMD_CTRL_EVENT_GET_INFO" at driver load time. 516665484d8SDoug Ambrisko */ 517665484d8SDoug Ambrisko 518665484d8SDoug Ambrisko static int 519665484d8SDoug Ambrisko mrsas_get_seq_num(struct mrsas_softc *sc, 520665484d8SDoug Ambrisko struct mrsas_evt_log_info *eli) 521665484d8SDoug Ambrisko { 522665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 523665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 524665484d8SDoug Ambrisko 525665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 526665484d8SDoug Ambrisko 527665484d8SDoug Ambrisko if (!cmd) { 528665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Failed to get a free cmd\n"); 529665484d8SDoug Ambrisko return -ENOMEM; 530665484d8SDoug Ambrisko } 531665484d8SDoug Ambrisko 532665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 533665484d8SDoug Ambrisko 534665484d8SDoug Ambrisko if (mrsas_alloc_evt_log_info_cmd(sc) != SUCCESS) { 535665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate evt log info cmd\n"); 536665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 537665484d8SDoug Ambrisko return -ENOMEM; 538665484d8SDoug Ambrisko } 539665484d8SDoug Ambrisko 540665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 541665484d8SDoug Ambrisko 542665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 543665484d8SDoug Ambrisko dcmd->cmd_status = 0x0; 544665484d8SDoug Ambrisko dcmd->sge_count = 1; 545665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 546665484d8SDoug Ambrisko dcmd->timeout = 0; 547665484d8SDoug Ambrisko dcmd->pad_0 = 0; 548665484d8SDoug Ambrisko dcmd->data_xfer_len = sizeof(struct mrsas_evt_log_info); 549665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_CTRL_EVENT_GET_INFO; 550665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = sc->el_info_phys_addr; 551665484d8SDoug Ambrisko dcmd->sgl.sge32[0].length = sizeof(struct mrsas_evt_log_info); 552665484d8SDoug Ambrisko 553665484d8SDoug Ambrisko mrsas_issue_blocked_cmd(sc, cmd); 554665484d8SDoug Ambrisko 555665484d8SDoug Ambrisko /* 556665484d8SDoug Ambrisko * Copy the data back into callers buffer 557665484d8SDoug Ambrisko */ 558665484d8SDoug Ambrisko memcpy(eli, sc->el_info_mem, sizeof(struct mrsas_evt_log_info)); 559665484d8SDoug Ambrisko mrsas_free_evt_log_info_cmd(sc); 560665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 561665484d8SDoug Ambrisko 562665484d8SDoug Ambrisko return 0; 563665484d8SDoug Ambrisko } 564665484d8SDoug Ambrisko 565665484d8SDoug Ambrisko 566665484d8SDoug Ambrisko /** 567665484d8SDoug Ambrisko * mrsas_register_aen: Register for asynchronous event notification 568665484d8SDoug Ambrisko * @sc: Adapter soft state 569665484d8SDoug Ambrisko * @seq_num: Starting sequence number 570665484d8SDoug Ambrisko * @class_locale: Class of the event 571665484d8SDoug Ambrisko * This function subscribes for events beyond the @seq_num 572665484d8SDoug Ambrisko * and type @class_locale. 573665484d8SDoug Ambrisko * 574665484d8SDoug Ambrisko * */ 575665484d8SDoug Ambrisko static int 576665484d8SDoug Ambrisko mrsas_register_aen(struct mrsas_softc *sc, u_int32_t seq_num, 577665484d8SDoug Ambrisko u_int32_t class_locale_word) 578665484d8SDoug Ambrisko { 579665484d8SDoug Ambrisko int ret_val; 580665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 581665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 582665484d8SDoug Ambrisko union mrsas_evt_class_locale curr_aen; 583665484d8SDoug Ambrisko union mrsas_evt_class_locale prev_aen; 584665484d8SDoug Ambrisko 585665484d8SDoug Ambrisko /* 586665484d8SDoug Ambrisko * If there an AEN pending already (aen_cmd), check if the 587665484d8SDoug Ambrisko * class_locale of that pending AEN is inclusive of the new 588665484d8SDoug Ambrisko * AEN request we currently have. If it is, then we don't have 589665484d8SDoug Ambrisko * to do anything. In other words, whichever events the current 590665484d8SDoug Ambrisko * AEN request is subscribing to, have already been subscribed 591665484d8SDoug Ambrisko * to. 592665484d8SDoug Ambrisko * If the old_cmd is _not_ inclusive, then we have to abort 593665484d8SDoug Ambrisko * that command, form a class_locale that is superset of both 594665484d8SDoug Ambrisko * old and current and re-issue to the FW 595665484d8SDoug Ambrisko * */ 596665484d8SDoug Ambrisko 597665484d8SDoug Ambrisko curr_aen.word = class_locale_word; 598665484d8SDoug Ambrisko 599665484d8SDoug Ambrisko if (sc->aen_cmd) { 600665484d8SDoug Ambrisko 601665484d8SDoug Ambrisko prev_aen.word = sc->aen_cmd->frame->dcmd.mbox.w[1]; 602665484d8SDoug Ambrisko 603665484d8SDoug Ambrisko /* 604665484d8SDoug Ambrisko * A class whose enum value is smaller is inclusive of all 605665484d8SDoug Ambrisko * higher values. If a PROGRESS (= -1) was previously 606665484d8SDoug Ambrisko * registered, then a new registration requests for higher 607665484d8SDoug Ambrisko * classes need not be sent to FW. They are automatically 608665484d8SDoug Ambrisko * included. 609665484d8SDoug Ambrisko * Locale numbers don't have such hierarchy. They are bitmap values 610665484d8SDoug Ambrisko */ 611665484d8SDoug Ambrisko if ((prev_aen.members.class <= curr_aen.members.class) && 612665484d8SDoug Ambrisko !((prev_aen.members.locale & curr_aen.members.locale) ^ 613665484d8SDoug Ambrisko curr_aen.members.locale)) { 614665484d8SDoug Ambrisko /* 615665484d8SDoug Ambrisko * Previously issued event registration includes 616665484d8SDoug Ambrisko * current request. Nothing to do. 617665484d8SDoug Ambrisko */ 618665484d8SDoug Ambrisko return 0; 619665484d8SDoug Ambrisko } else { 620665484d8SDoug Ambrisko curr_aen.members.locale |= prev_aen.members.locale; 621665484d8SDoug Ambrisko 622665484d8SDoug Ambrisko if (prev_aen.members.class < curr_aen.members.class) 623665484d8SDoug Ambrisko curr_aen.members.class = prev_aen.members.class; 624665484d8SDoug Ambrisko 625665484d8SDoug Ambrisko sc->aen_cmd->abort_aen = 1; 626665484d8SDoug Ambrisko ret_val = mrsas_issue_blocked_abort_cmd(sc, 627665484d8SDoug Ambrisko sc->aen_cmd); 628665484d8SDoug Ambrisko 629665484d8SDoug Ambrisko if (ret_val) { 630665484d8SDoug Ambrisko printf("mrsas: Failed to abort " 631665484d8SDoug Ambrisko "previous AEN command\n"); 632665484d8SDoug Ambrisko return ret_val; 633665484d8SDoug Ambrisko } 634665484d8SDoug Ambrisko } 635665484d8SDoug Ambrisko } 636665484d8SDoug Ambrisko 637665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 638665484d8SDoug Ambrisko 639665484d8SDoug Ambrisko if (!cmd) 640665484d8SDoug Ambrisko return -ENOMEM; 641665484d8SDoug Ambrisko 642665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 643665484d8SDoug Ambrisko 644665484d8SDoug Ambrisko memset(sc->evt_detail_mem, 0, sizeof(struct mrsas_evt_detail)); 645665484d8SDoug Ambrisko 646665484d8SDoug Ambrisko /* 647665484d8SDoug Ambrisko * Prepare DCMD for aen registration 648665484d8SDoug Ambrisko */ 649665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 650665484d8SDoug Ambrisko 651665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 652665484d8SDoug Ambrisko dcmd->cmd_status = 0x0; 653665484d8SDoug Ambrisko dcmd->sge_count = 1; 654665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 655665484d8SDoug Ambrisko dcmd->timeout = 0; 656665484d8SDoug Ambrisko dcmd->pad_0 = 0; 657665484d8SDoug Ambrisko dcmd->data_xfer_len = sizeof(struct mrsas_evt_detail); 658665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_CTRL_EVENT_WAIT; 659665484d8SDoug Ambrisko dcmd->mbox.w[0] = seq_num; 660665484d8SDoug Ambrisko sc->last_seq_num = seq_num; 661665484d8SDoug Ambrisko dcmd->mbox.w[1] = curr_aen.word; 662665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = (u_int32_t) sc->evt_detail_phys_addr; 663665484d8SDoug Ambrisko dcmd->sgl.sge32[0].length = sizeof(struct mrsas_evt_detail); 664665484d8SDoug Ambrisko 665665484d8SDoug Ambrisko if (sc->aen_cmd != NULL) { 666665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 667665484d8SDoug Ambrisko return 0; 668665484d8SDoug Ambrisko } 669665484d8SDoug Ambrisko 670665484d8SDoug Ambrisko /* 671665484d8SDoug Ambrisko * Store reference to the cmd used to register for AEN. When an 672665484d8SDoug Ambrisko * application wants us to register for AEN, we have to abort this 673665484d8SDoug Ambrisko * cmd and re-register with a new EVENT LOCALE supplied by that app 674665484d8SDoug Ambrisko */ 675665484d8SDoug Ambrisko sc->aen_cmd = cmd; 676665484d8SDoug Ambrisko 677665484d8SDoug Ambrisko /* 678665484d8SDoug Ambrisko Issue the aen registration frame 679665484d8SDoug Ambrisko */ 680665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)){ 681665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot issue AEN DCMD command.\n"); 682665484d8SDoug Ambrisko return(1); 683665484d8SDoug Ambrisko } 684665484d8SDoug Ambrisko 685665484d8SDoug Ambrisko return 0; 686665484d8SDoug Ambrisko } 687665484d8SDoug Ambrisko /** 688665484d8SDoug Ambrisko * mrsas_start_aen - Subscribes to AEN during driver load time 689665484d8SDoug Ambrisko * @instance: Adapter soft state 690665484d8SDoug Ambrisko */ 691665484d8SDoug Ambrisko static int mrsas_start_aen(struct mrsas_softc *sc) 692665484d8SDoug Ambrisko { 693665484d8SDoug Ambrisko struct mrsas_evt_log_info eli; 694665484d8SDoug Ambrisko union mrsas_evt_class_locale class_locale; 695665484d8SDoug Ambrisko 696665484d8SDoug Ambrisko 697665484d8SDoug Ambrisko /* Get the latest sequence number from FW*/ 698665484d8SDoug Ambrisko 699665484d8SDoug Ambrisko memset(&eli, 0, sizeof(eli)); 700665484d8SDoug Ambrisko 701665484d8SDoug Ambrisko if (mrsas_get_seq_num(sc, &eli)) 702665484d8SDoug Ambrisko return -1; 703665484d8SDoug Ambrisko 704665484d8SDoug Ambrisko /* Register AEN with FW for latest sequence number plus 1*/ 705665484d8SDoug Ambrisko class_locale.members.reserved = 0; 706665484d8SDoug Ambrisko class_locale.members.locale = MR_EVT_LOCALE_ALL; 707665484d8SDoug Ambrisko class_locale.members.class = MR_EVT_CLASS_DEBUG; 708665484d8SDoug Ambrisko 709665484d8SDoug Ambrisko return mrsas_register_aen(sc, eli.newest_seq_num + 1, 710665484d8SDoug Ambrisko class_locale.word); 711665484d8SDoug Ambrisko } 712665484d8SDoug Ambrisko 713665484d8SDoug Ambrisko /** 714665484d8SDoug Ambrisko * mrsas_attach: PCI entry point 715665484d8SDoug Ambrisko * input: device struct pointer 716665484d8SDoug Ambrisko * 717665484d8SDoug Ambrisko * Performs setup of PCI and registers, initializes mutexes and 718665484d8SDoug Ambrisko * linked lists, registers interrupts and CAM, and initializes 719665484d8SDoug Ambrisko * the adapter/controller to its proper state. 720665484d8SDoug Ambrisko */ 721665484d8SDoug Ambrisko static int mrsas_attach(device_t dev) 722665484d8SDoug Ambrisko { 723665484d8SDoug Ambrisko struct mrsas_softc *sc = device_get_softc(dev); 724665484d8SDoug Ambrisko uint32_t cmd, bar, error; 725665484d8SDoug Ambrisko 726665484d8SDoug Ambrisko /* Look up our softc and initialize its fields. */ 727665484d8SDoug Ambrisko sc->mrsas_dev = dev; 728665484d8SDoug Ambrisko sc->device_id = pci_get_device(dev); 729665484d8SDoug Ambrisko 730665484d8SDoug Ambrisko mrsas_get_tunables(sc); 731665484d8SDoug Ambrisko 732665484d8SDoug Ambrisko /* 733665484d8SDoug Ambrisko * Set up PCI and registers 734665484d8SDoug Ambrisko */ 735665484d8SDoug Ambrisko cmd = pci_read_config(dev, PCIR_COMMAND, 2); 736665484d8SDoug Ambrisko if ( (cmd & PCIM_CMD_PORTEN) == 0) { 737665484d8SDoug Ambrisko return (ENXIO); 738665484d8SDoug Ambrisko } 739665484d8SDoug Ambrisko /* Force the busmaster enable bit on. */ 740665484d8SDoug Ambrisko cmd |= PCIM_CMD_BUSMASTEREN; 741665484d8SDoug Ambrisko pci_write_config(dev, PCIR_COMMAND, cmd, 2); 742665484d8SDoug Ambrisko 743665484d8SDoug Ambrisko //bar = pci_read_config(dev, MRSAS_PCI_BAR0, 4); 744665484d8SDoug Ambrisko bar = pci_read_config(dev, MRSAS_PCI_BAR1, 4); 745665484d8SDoug Ambrisko 746665484d8SDoug Ambrisko sc->reg_res_id = MRSAS_PCI_BAR1; /* BAR1 offset */ 747665484d8SDoug Ambrisko if ((sc->reg_res = bus_alloc_resource(dev, SYS_RES_MEMORY, 748665484d8SDoug Ambrisko &(sc->reg_res_id), 0, ~0, 1, RF_ACTIVE)) 749665484d8SDoug Ambrisko == NULL) { 750665484d8SDoug Ambrisko device_printf(dev, "Cannot allocate PCI registers\n"); 751665484d8SDoug Ambrisko goto attach_fail; 752665484d8SDoug Ambrisko } 753665484d8SDoug Ambrisko sc->bus_tag = rman_get_bustag(sc->reg_res); 754665484d8SDoug Ambrisko sc->bus_handle = rman_get_bushandle(sc->reg_res); 755665484d8SDoug Ambrisko 756665484d8SDoug Ambrisko /* Intialize mutexes */ 757665484d8SDoug Ambrisko mtx_init(&sc->sim_lock, "mrsas_sim_lock", NULL, MTX_DEF); 758665484d8SDoug Ambrisko mtx_init(&sc->pci_lock, "mrsas_pci_lock", NULL, MTX_DEF); 759665484d8SDoug Ambrisko mtx_init(&sc->io_lock, "mrsas_io_lock", NULL, MTX_DEF); 760665484d8SDoug Ambrisko mtx_init(&sc->aen_lock, "mrsas_aen_lock", NULL, MTX_DEF); 761665484d8SDoug Ambrisko mtx_init(&sc->ioctl_lock, "mrsas_ioctl_lock", NULL, MTX_SPIN); 762665484d8SDoug Ambrisko mtx_init(&sc->mpt_cmd_pool_lock, "mrsas_mpt_cmd_pool_lock", NULL, MTX_DEF); 763665484d8SDoug Ambrisko mtx_init(&sc->mfi_cmd_pool_lock, "mrsas_mfi_cmd_pool_lock", NULL, MTX_DEF); 764665484d8SDoug Ambrisko mtx_init(&sc->raidmap_lock, "mrsas_raidmap_lock", NULL, MTX_DEF); 765665484d8SDoug Ambrisko 766665484d8SDoug Ambrisko /* Intialize linked list */ 767665484d8SDoug Ambrisko TAILQ_INIT(&sc->mrsas_mpt_cmd_list_head); 768665484d8SDoug Ambrisko TAILQ_INIT(&sc->mrsas_mfi_cmd_list_head); 769665484d8SDoug Ambrisko 770665484d8SDoug Ambrisko atomic_set(&sc->fw_outstanding,0); 771665484d8SDoug Ambrisko 772665484d8SDoug Ambrisko sc->io_cmds_highwater = 0; 773665484d8SDoug Ambrisko 774665484d8SDoug Ambrisko /* Create a /dev entry for this device. */ 775665484d8SDoug Ambrisko sc->mrsas_cdev = make_dev(&mrsas_cdevsw, device_get_unit(dev), UID_ROOT, 776665484d8SDoug Ambrisko GID_OPERATOR, (S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP), "mrsas%u", 777665484d8SDoug Ambrisko device_get_unit(dev)); 778665484d8SDoug Ambrisko if (sc->mrsas_cdev) 779665484d8SDoug Ambrisko sc->mrsas_cdev->si_drv1 = sc; 780665484d8SDoug Ambrisko 781665484d8SDoug Ambrisko sc->adprecovery = MRSAS_HBA_OPERATIONAL; 782665484d8SDoug Ambrisko sc->UnevenSpanSupport = 0; 783665484d8SDoug Ambrisko 784665484d8SDoug Ambrisko /* Initialize Firmware */ 785665484d8SDoug Ambrisko if (mrsas_init_fw(sc) != SUCCESS) { 786665484d8SDoug Ambrisko goto attach_fail_fw; 787665484d8SDoug Ambrisko } 788665484d8SDoug Ambrisko 789665484d8SDoug Ambrisko /* Register SCSI mid-layer */ 790665484d8SDoug Ambrisko if ((mrsas_cam_attach(sc) != SUCCESS)) { 791665484d8SDoug Ambrisko goto attach_fail_cam; 792665484d8SDoug Ambrisko } 793665484d8SDoug Ambrisko 794665484d8SDoug Ambrisko /* Register IRQs */ 795665484d8SDoug Ambrisko if (mrsas_setup_irq(sc) != SUCCESS) { 796665484d8SDoug Ambrisko goto attach_fail_irq; 797665484d8SDoug Ambrisko } 798665484d8SDoug Ambrisko 799665484d8SDoug Ambrisko /* Enable Interrupts */ 800665484d8SDoug Ambrisko mrsas_enable_intr(sc); 801665484d8SDoug Ambrisko 802665484d8SDoug Ambrisko error = mrsas_kproc_create(mrsas_ocr_thread, sc, 803665484d8SDoug Ambrisko &sc->ocr_thread, 0, 0, "mrsas_ocr%d", 804665484d8SDoug Ambrisko device_get_unit(sc->mrsas_dev)); 805665484d8SDoug Ambrisko if (error) { 806665484d8SDoug Ambrisko printf("Error %d starting rescan thread\n", error); 807665484d8SDoug Ambrisko goto attach_fail_irq; 808665484d8SDoug Ambrisko } 809665484d8SDoug Ambrisko 810665484d8SDoug Ambrisko mrsas_setup_sysctl(sc); 811665484d8SDoug Ambrisko 812665484d8SDoug Ambrisko /* Initiate AEN (Asynchronous Event Notification)*/ 813665484d8SDoug Ambrisko 814665484d8SDoug Ambrisko if (mrsas_start_aen(sc)) { 815665484d8SDoug Ambrisko printf("Error: start aen failed\n"); 816665484d8SDoug Ambrisko goto fail_start_aen; 817665484d8SDoug Ambrisko } 818665484d8SDoug Ambrisko 819665484d8SDoug Ambrisko return (0); 820665484d8SDoug Ambrisko 821665484d8SDoug Ambrisko fail_start_aen: 822665484d8SDoug Ambrisko attach_fail_irq: 823665484d8SDoug Ambrisko mrsas_teardown_intr(sc); 824665484d8SDoug Ambrisko attach_fail_cam: 825665484d8SDoug Ambrisko mrsas_cam_detach(sc); 826665484d8SDoug Ambrisko attach_fail_fw: 827665484d8SDoug Ambrisko //attach_fail_raidmap: 828665484d8SDoug Ambrisko mrsas_free_mem(sc); 829665484d8SDoug Ambrisko mtx_destroy(&sc->sim_lock); 830665484d8SDoug Ambrisko mtx_destroy(&sc->aen_lock); 831665484d8SDoug Ambrisko mtx_destroy(&sc->pci_lock); 832665484d8SDoug Ambrisko mtx_destroy(&sc->io_lock); 833665484d8SDoug Ambrisko mtx_destroy(&sc->ioctl_lock); 834665484d8SDoug Ambrisko mtx_destroy(&sc->mpt_cmd_pool_lock); 835665484d8SDoug Ambrisko mtx_destroy(&sc->mfi_cmd_pool_lock); 836665484d8SDoug Ambrisko mtx_destroy(&sc->raidmap_lock); 837665484d8SDoug Ambrisko attach_fail: 838665484d8SDoug Ambrisko destroy_dev(sc->mrsas_cdev); 839665484d8SDoug Ambrisko if (sc->reg_res){ 840665484d8SDoug Ambrisko bus_release_resource(sc->mrsas_dev, SYS_RES_MEMORY, 841665484d8SDoug Ambrisko sc->reg_res_id, sc->reg_res); 842665484d8SDoug Ambrisko } 843665484d8SDoug Ambrisko return (ENXIO); 844665484d8SDoug Ambrisko } 845665484d8SDoug Ambrisko 846665484d8SDoug Ambrisko /** 847665484d8SDoug Ambrisko * mrsas_detach: De-allocates and teardown resources 848665484d8SDoug Ambrisko * input: device struct pointer 849665484d8SDoug Ambrisko * 850665484d8SDoug Ambrisko * This function is the entry point for device disconnect and detach. It 851665484d8SDoug Ambrisko * performs memory de-allocations, shutdown of the controller and various 852665484d8SDoug Ambrisko * teardown and destroy resource functions. 853665484d8SDoug Ambrisko */ 854665484d8SDoug Ambrisko static int mrsas_detach(device_t dev) 855665484d8SDoug Ambrisko { 856665484d8SDoug Ambrisko struct mrsas_softc *sc; 857665484d8SDoug Ambrisko int i = 0; 858665484d8SDoug Ambrisko 859665484d8SDoug Ambrisko sc = device_get_softc(dev); 860665484d8SDoug Ambrisko sc->remove_in_progress = 1; 861665484d8SDoug Ambrisko if(sc->ocr_thread_active) 862665484d8SDoug Ambrisko wakeup(&sc->ocr_chan); 863665484d8SDoug Ambrisko while(sc->reset_in_progress){ 864665484d8SDoug Ambrisko i++; 865665484d8SDoug Ambrisko if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) { 866665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_INFO, 867665484d8SDoug Ambrisko "[%2d]waiting for ocr to be finished\n",i); 868665484d8SDoug Ambrisko } 869665484d8SDoug Ambrisko pause("mr_shutdown", hz); 870665484d8SDoug Ambrisko } 871665484d8SDoug Ambrisko i = 0; 872665484d8SDoug Ambrisko while(sc->ocr_thread_active){ 873665484d8SDoug Ambrisko i++; 874665484d8SDoug Ambrisko if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) { 875665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_INFO, 876665484d8SDoug Ambrisko "[%2d]waiting for " 877665484d8SDoug Ambrisko "mrsas_ocr thread to quit ocr %d\n",i, 878665484d8SDoug Ambrisko sc->ocr_thread_active); 879665484d8SDoug Ambrisko } 880665484d8SDoug Ambrisko pause("mr_shutdown", hz); 881665484d8SDoug Ambrisko } 882665484d8SDoug Ambrisko mrsas_flush_cache(sc); 883665484d8SDoug Ambrisko mrsas_shutdown_ctlr(sc, MR_DCMD_CTRL_SHUTDOWN); 884665484d8SDoug Ambrisko mrsas_disable_intr(sc); 885665484d8SDoug Ambrisko mrsas_cam_detach(sc); 886665484d8SDoug Ambrisko mrsas_teardown_intr(sc); 887665484d8SDoug Ambrisko mrsas_free_mem(sc); 888665484d8SDoug Ambrisko mtx_destroy(&sc->sim_lock); 889665484d8SDoug Ambrisko mtx_destroy(&sc->aen_lock); 890665484d8SDoug Ambrisko mtx_destroy(&sc->pci_lock); 891665484d8SDoug Ambrisko mtx_destroy(&sc->io_lock); 892665484d8SDoug Ambrisko mtx_destroy(&sc->ioctl_lock); 893665484d8SDoug Ambrisko mtx_destroy(&sc->mpt_cmd_pool_lock); 894665484d8SDoug Ambrisko mtx_destroy(&sc->mfi_cmd_pool_lock); 895665484d8SDoug Ambrisko mtx_destroy(&sc->raidmap_lock); 896665484d8SDoug Ambrisko if (sc->reg_res){ 897665484d8SDoug Ambrisko bus_release_resource(sc->mrsas_dev, 898665484d8SDoug Ambrisko SYS_RES_MEMORY, sc->reg_res_id, sc->reg_res); 899665484d8SDoug Ambrisko } 900665484d8SDoug Ambrisko destroy_dev(sc->mrsas_cdev); 901665484d8SDoug Ambrisko if (sc->sysctl_tree != NULL) 902665484d8SDoug Ambrisko sysctl_ctx_free(&sc->sysctl_ctx); 903665484d8SDoug Ambrisko return (0); 904665484d8SDoug Ambrisko } 905665484d8SDoug Ambrisko 906665484d8SDoug Ambrisko /** 907665484d8SDoug Ambrisko * mrsas_free_mem: Frees allocated memory 908665484d8SDoug Ambrisko * input: Adapter instance soft state 909665484d8SDoug Ambrisko * 910665484d8SDoug Ambrisko * This function is called from mrsas_detach() to free previously allocated 911665484d8SDoug Ambrisko * memory. 912665484d8SDoug Ambrisko */ 913665484d8SDoug Ambrisko void mrsas_free_mem(struct mrsas_softc *sc) 914665484d8SDoug Ambrisko { 915665484d8SDoug Ambrisko int i; 916665484d8SDoug Ambrisko u_int32_t max_cmd; 917665484d8SDoug Ambrisko struct mrsas_mfi_cmd *mfi_cmd; 918665484d8SDoug Ambrisko struct mrsas_mpt_cmd *mpt_cmd; 919665484d8SDoug Ambrisko 920665484d8SDoug Ambrisko /* 921665484d8SDoug Ambrisko * Free RAID map memory 922665484d8SDoug Ambrisko */ 923665484d8SDoug Ambrisko for (i=0; i < 2; i++) 924665484d8SDoug Ambrisko { 925665484d8SDoug Ambrisko if (sc->raidmap_phys_addr[i]) 926665484d8SDoug Ambrisko bus_dmamap_unload(sc->raidmap_tag[i], sc->raidmap_dmamap[i]); 927665484d8SDoug Ambrisko if (sc->raidmap_mem[i] != NULL) 928665484d8SDoug Ambrisko bus_dmamem_free(sc->raidmap_tag[i], sc->raidmap_mem[i], sc->raidmap_dmamap[i]); 929665484d8SDoug Ambrisko if (sc->raidmap_tag[i] != NULL) 930665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->raidmap_tag[i]); 931*4799d485SKashyap D Desai 932*4799d485SKashyap D Desai if (sc->ld_drv_map[i] != NULL) 933*4799d485SKashyap D Desai free(sc->ld_drv_map[i], M_MRSAS); 934665484d8SDoug Ambrisko } 935665484d8SDoug Ambrisko 936665484d8SDoug Ambrisko /* 937665484d8SDoug Ambrisko * Free version buffer memroy 938665484d8SDoug Ambrisko */ 939665484d8SDoug Ambrisko if (sc->verbuf_phys_addr) 940665484d8SDoug Ambrisko bus_dmamap_unload(sc->verbuf_tag, sc->verbuf_dmamap); 941665484d8SDoug Ambrisko if (sc->verbuf_mem != NULL) 942665484d8SDoug Ambrisko bus_dmamem_free(sc->verbuf_tag, sc->verbuf_mem, sc->verbuf_dmamap); 943665484d8SDoug Ambrisko if (sc->verbuf_tag != NULL) 944665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->verbuf_tag); 945665484d8SDoug Ambrisko 946665484d8SDoug Ambrisko 947665484d8SDoug Ambrisko /* 948665484d8SDoug Ambrisko * Free sense buffer memory 949665484d8SDoug Ambrisko */ 950665484d8SDoug Ambrisko if (sc->sense_phys_addr) 951665484d8SDoug Ambrisko bus_dmamap_unload(sc->sense_tag, sc->sense_dmamap); 952665484d8SDoug Ambrisko if (sc->sense_mem != NULL) 953665484d8SDoug Ambrisko bus_dmamem_free(sc->sense_tag, sc->sense_mem, sc->sense_dmamap); 954665484d8SDoug Ambrisko if (sc->sense_tag != NULL) 955665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->sense_tag); 956665484d8SDoug Ambrisko 957665484d8SDoug Ambrisko /* 958665484d8SDoug Ambrisko * Free chain frame memory 959665484d8SDoug Ambrisko */ 960665484d8SDoug Ambrisko if (sc->chain_frame_phys_addr) 961665484d8SDoug Ambrisko bus_dmamap_unload(sc->chain_frame_tag, sc->chain_frame_dmamap); 962665484d8SDoug Ambrisko if (sc->chain_frame_mem != NULL) 963665484d8SDoug Ambrisko bus_dmamem_free(sc->chain_frame_tag, sc->chain_frame_mem, sc->chain_frame_dmamap); 964665484d8SDoug Ambrisko if (sc->chain_frame_tag != NULL) 965665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->chain_frame_tag); 966665484d8SDoug Ambrisko 967665484d8SDoug Ambrisko /* 968665484d8SDoug Ambrisko * Free IO Request memory 969665484d8SDoug Ambrisko */ 970665484d8SDoug Ambrisko if (sc->io_request_phys_addr) 971665484d8SDoug Ambrisko bus_dmamap_unload(sc->io_request_tag, sc->io_request_dmamap); 972665484d8SDoug Ambrisko if (sc->io_request_mem != NULL) 973665484d8SDoug Ambrisko bus_dmamem_free(sc->io_request_tag, sc->io_request_mem, sc->io_request_dmamap); 974665484d8SDoug Ambrisko if (sc->io_request_tag != NULL) 975665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->io_request_tag); 976665484d8SDoug Ambrisko 977665484d8SDoug Ambrisko /* 978665484d8SDoug Ambrisko * Free Reply Descriptor memory 979665484d8SDoug Ambrisko */ 980665484d8SDoug Ambrisko if (sc->reply_desc_phys_addr) 981665484d8SDoug Ambrisko bus_dmamap_unload(sc->reply_desc_tag, sc->reply_desc_dmamap); 982665484d8SDoug Ambrisko if (sc->reply_desc_mem != NULL) 983665484d8SDoug Ambrisko bus_dmamem_free(sc->reply_desc_tag, sc->reply_desc_mem, sc->reply_desc_dmamap); 984665484d8SDoug Ambrisko if (sc->reply_desc_tag != NULL) 985665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->reply_desc_tag); 986665484d8SDoug Ambrisko 987665484d8SDoug Ambrisko /* 988665484d8SDoug Ambrisko * Free event detail memory 989665484d8SDoug Ambrisko */ 990665484d8SDoug Ambrisko if (sc->evt_detail_phys_addr) 991665484d8SDoug Ambrisko bus_dmamap_unload(sc->evt_detail_tag, sc->evt_detail_dmamap); 992665484d8SDoug Ambrisko if (sc->evt_detail_mem != NULL) 993665484d8SDoug Ambrisko bus_dmamem_free(sc->evt_detail_tag, sc->evt_detail_mem, sc->evt_detail_dmamap); 994665484d8SDoug Ambrisko if (sc->evt_detail_tag != NULL) 995665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->evt_detail_tag); 996665484d8SDoug Ambrisko 997665484d8SDoug Ambrisko /* 998665484d8SDoug Ambrisko * Free MFI frames 999665484d8SDoug Ambrisko */ 1000665484d8SDoug Ambrisko if (sc->mfi_cmd_list) { 1001665484d8SDoug Ambrisko for (i = 0; i < MRSAS_MAX_MFI_CMDS; i++) { 1002665484d8SDoug Ambrisko mfi_cmd = sc->mfi_cmd_list[i]; 1003665484d8SDoug Ambrisko mrsas_free_frame(sc, mfi_cmd); 1004665484d8SDoug Ambrisko } 1005665484d8SDoug Ambrisko } 1006665484d8SDoug Ambrisko if (sc->mficmd_frame_tag != NULL) 1007665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->mficmd_frame_tag); 1008665484d8SDoug Ambrisko 1009665484d8SDoug Ambrisko /* 1010665484d8SDoug Ambrisko * Free MPT internal command list 1011665484d8SDoug Ambrisko */ 1012665484d8SDoug Ambrisko max_cmd = sc->max_fw_cmds; 1013665484d8SDoug Ambrisko if (sc->mpt_cmd_list) { 1014665484d8SDoug Ambrisko for (i = 0; i < max_cmd; i++) { 1015665484d8SDoug Ambrisko mpt_cmd = sc->mpt_cmd_list[i]; 1016665484d8SDoug Ambrisko bus_dmamap_destroy(sc->data_tag, mpt_cmd->data_dmamap); 1017665484d8SDoug Ambrisko free(sc->mpt_cmd_list[i], M_MRSAS); 1018665484d8SDoug Ambrisko } 1019665484d8SDoug Ambrisko free(sc->mpt_cmd_list, M_MRSAS); 1020665484d8SDoug Ambrisko sc->mpt_cmd_list = NULL; 1021665484d8SDoug Ambrisko } 1022665484d8SDoug Ambrisko 1023665484d8SDoug Ambrisko /* 1024665484d8SDoug Ambrisko * Free MFI internal command list 1025665484d8SDoug Ambrisko */ 1026665484d8SDoug Ambrisko 1027665484d8SDoug Ambrisko if (sc->mfi_cmd_list) { 1028665484d8SDoug Ambrisko for (i = 0; i < MRSAS_MAX_MFI_CMDS; i++) { 1029665484d8SDoug Ambrisko free(sc->mfi_cmd_list[i], M_MRSAS); 1030665484d8SDoug Ambrisko } 1031665484d8SDoug Ambrisko free(sc->mfi_cmd_list, M_MRSAS); 1032665484d8SDoug Ambrisko sc->mfi_cmd_list = NULL; 1033665484d8SDoug Ambrisko } 1034665484d8SDoug Ambrisko 1035665484d8SDoug Ambrisko /* 1036665484d8SDoug Ambrisko * Free request descriptor memory 1037665484d8SDoug Ambrisko */ 1038665484d8SDoug Ambrisko free(sc->req_desc, M_MRSAS); 1039665484d8SDoug Ambrisko sc->req_desc = NULL; 1040665484d8SDoug Ambrisko 1041665484d8SDoug Ambrisko /* 1042665484d8SDoug Ambrisko * Destroy parent tag 1043665484d8SDoug Ambrisko */ 1044665484d8SDoug Ambrisko if (sc->mrsas_parent_tag != NULL) 1045665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->mrsas_parent_tag); 1046665484d8SDoug Ambrisko } 1047665484d8SDoug Ambrisko 1048665484d8SDoug Ambrisko /** 1049665484d8SDoug Ambrisko * mrsas_teardown_intr: Teardown interrupt 1050665484d8SDoug Ambrisko * input: Adapter instance soft state 1051665484d8SDoug Ambrisko * 1052665484d8SDoug Ambrisko * This function is called from mrsas_detach() to teardown and release 1053665484d8SDoug Ambrisko * bus interrupt resourse. 1054665484d8SDoug Ambrisko */ 1055665484d8SDoug Ambrisko void mrsas_teardown_intr(struct mrsas_softc *sc) 1056665484d8SDoug Ambrisko { 1057665484d8SDoug Ambrisko if (sc->intr_handle) 1058665484d8SDoug Ambrisko bus_teardown_intr(sc->mrsas_dev, sc->mrsas_irq, sc->intr_handle); 1059665484d8SDoug Ambrisko if (sc->mrsas_irq != NULL) 1060665484d8SDoug Ambrisko bus_release_resource(sc->mrsas_dev, SYS_RES_IRQ, sc->irq_id, sc->mrsas_irq); 1061665484d8SDoug Ambrisko sc->intr_handle = NULL; 1062665484d8SDoug Ambrisko } 1063665484d8SDoug Ambrisko 1064665484d8SDoug Ambrisko /** 1065665484d8SDoug Ambrisko * mrsas_suspend: Suspend entry point 1066665484d8SDoug Ambrisko * input: Device struct pointer 1067665484d8SDoug Ambrisko * 1068665484d8SDoug Ambrisko * This function is the entry point for system suspend from the OS. 1069665484d8SDoug Ambrisko */ 1070665484d8SDoug Ambrisko static int mrsas_suspend(device_t dev) 1071665484d8SDoug Ambrisko { 1072665484d8SDoug Ambrisko struct mrsas_softc *sc; 1073665484d8SDoug Ambrisko 1074665484d8SDoug Ambrisko sc = device_get_softc(dev); 1075665484d8SDoug Ambrisko return (0); 1076665484d8SDoug Ambrisko } 1077665484d8SDoug Ambrisko 1078665484d8SDoug Ambrisko /** 1079665484d8SDoug Ambrisko * mrsas_resume: Resume entry point 1080665484d8SDoug Ambrisko * input: Device struct pointer 1081665484d8SDoug Ambrisko * 1082665484d8SDoug Ambrisko * This function is the entry point for system resume from the OS. 1083665484d8SDoug Ambrisko */ 1084665484d8SDoug Ambrisko static int mrsas_resume(device_t dev) 1085665484d8SDoug Ambrisko { 1086665484d8SDoug Ambrisko struct mrsas_softc *sc; 1087665484d8SDoug Ambrisko 1088665484d8SDoug Ambrisko sc = device_get_softc(dev); 1089665484d8SDoug Ambrisko return (0); 1090665484d8SDoug Ambrisko } 1091665484d8SDoug Ambrisko 1092665484d8SDoug Ambrisko /** 1093665484d8SDoug Ambrisko * mrsas_ioctl: IOCtl commands entry point. 1094665484d8SDoug Ambrisko * 1095665484d8SDoug Ambrisko * This function is the entry point for IOCtls from the OS. It calls the 1096665484d8SDoug Ambrisko * appropriate function for processing depending on the command received. 1097665484d8SDoug Ambrisko */ 1098665484d8SDoug Ambrisko static int 1099665484d8SDoug Ambrisko mrsas_ioctl(struct cdev *dev, u_long cmd, caddr_t arg, int flag, d_thread_t *td) 1100665484d8SDoug Ambrisko { 1101665484d8SDoug Ambrisko struct mrsas_softc *sc; 1102665484d8SDoug Ambrisko int ret = 0, i = 0; 1103665484d8SDoug Ambrisko 1104665484d8SDoug Ambrisko sc = (struct mrsas_softc *)(dev->si_drv1); 1105665484d8SDoug Ambrisko 1106665484d8SDoug Ambrisko if (sc->remove_in_progress) { 1107665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_INFO, 1108665484d8SDoug Ambrisko "Driver remove or shutdown called.\n"); 1109665484d8SDoug Ambrisko return ENOENT; 1110665484d8SDoug Ambrisko } 1111665484d8SDoug Ambrisko 1112665484d8SDoug Ambrisko mtx_lock_spin(&sc->ioctl_lock); 1113665484d8SDoug Ambrisko if (!sc->reset_in_progress) { 1114665484d8SDoug Ambrisko mtx_unlock_spin(&sc->ioctl_lock); 1115665484d8SDoug Ambrisko goto do_ioctl; 1116665484d8SDoug Ambrisko } 1117665484d8SDoug Ambrisko 1118665484d8SDoug Ambrisko /* Release ioclt_lock, and wait for OCR 1119665484d8SDoug Ambrisko * to be finished */ 1120665484d8SDoug Ambrisko mtx_unlock_spin(&sc->ioctl_lock); 1121665484d8SDoug Ambrisko while(sc->reset_in_progress){ 1122665484d8SDoug Ambrisko i++; 1123665484d8SDoug Ambrisko if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) { 1124665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_INFO, 1125665484d8SDoug Ambrisko "[%2d]waiting for " 1126665484d8SDoug Ambrisko "OCR to be finished %d\n",i, 1127665484d8SDoug Ambrisko sc->ocr_thread_active); 1128665484d8SDoug Ambrisko } 1129665484d8SDoug Ambrisko pause("mr_ioctl", hz); 1130665484d8SDoug Ambrisko } 1131665484d8SDoug Ambrisko 1132665484d8SDoug Ambrisko do_ioctl: 1133665484d8SDoug Ambrisko switch (cmd) { 1134665484d8SDoug Ambrisko case MRSAS_IOC_FIRMWARE_PASS_THROUGH: 1135665484d8SDoug Ambrisko ret = mrsas_passthru(sc, (void *)arg); 1136665484d8SDoug Ambrisko break; 1137665484d8SDoug Ambrisko case MRSAS_IOC_SCAN_BUS: 1138665484d8SDoug Ambrisko ret = mrsas_bus_scan(sc); 1139665484d8SDoug Ambrisko break; 1140665484d8SDoug Ambrisko } 1141665484d8SDoug Ambrisko 1142665484d8SDoug Ambrisko return (ret); 1143665484d8SDoug Ambrisko } 1144665484d8SDoug Ambrisko 1145665484d8SDoug Ambrisko /** 1146665484d8SDoug Ambrisko * mrsas_setup_irq: Set up interrupt. 1147665484d8SDoug Ambrisko * input: Adapter instance soft state 1148665484d8SDoug Ambrisko * 1149665484d8SDoug Ambrisko * This function sets up interrupts as a bus resource, with flags indicating 1150665484d8SDoug Ambrisko * resource permitting contemporaneous sharing and for resource to activate 1151665484d8SDoug Ambrisko * atomically. 1152665484d8SDoug Ambrisko */ 1153665484d8SDoug Ambrisko static int mrsas_setup_irq(struct mrsas_softc *sc) 1154665484d8SDoug Ambrisko { 1155665484d8SDoug Ambrisko sc->irq_id = 0; 1156665484d8SDoug Ambrisko sc->mrsas_irq = bus_alloc_resource_any(sc->mrsas_dev, SYS_RES_IRQ, 1157665484d8SDoug Ambrisko &sc->irq_id, RF_SHAREABLE | RF_ACTIVE); 1158665484d8SDoug Ambrisko if (sc->mrsas_irq == NULL){ 1159665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate interrupt\n"); 1160665484d8SDoug Ambrisko return (FAIL); 1161665484d8SDoug Ambrisko } 1162665484d8SDoug Ambrisko if (bus_setup_intr(sc->mrsas_dev, sc->mrsas_irq, INTR_MPSAFE|INTR_TYPE_CAM, 1163665484d8SDoug Ambrisko NULL, mrsas_isr, sc, &sc->intr_handle)) { 1164665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot set up interrupt\n"); 1165665484d8SDoug Ambrisko return (FAIL); 1166665484d8SDoug Ambrisko } 1167665484d8SDoug Ambrisko 1168665484d8SDoug Ambrisko return (0); 1169665484d8SDoug Ambrisko } 1170665484d8SDoug Ambrisko 1171665484d8SDoug Ambrisko /* 1172665484d8SDoug Ambrisko * mrsas_isr: ISR entry point 1173665484d8SDoug Ambrisko * input: argument pointer 1174665484d8SDoug Ambrisko * 1175665484d8SDoug Ambrisko * This function is the interrupt service routine entry point. There 1176665484d8SDoug Ambrisko * are two types of interrupts, state change interrupt and response 1177665484d8SDoug Ambrisko * interrupt. If an interrupt is not ours, we just return. 1178665484d8SDoug Ambrisko */ 1179665484d8SDoug Ambrisko void mrsas_isr(void *arg) 1180665484d8SDoug Ambrisko { 1181665484d8SDoug Ambrisko struct mrsas_softc *sc = (struct mrsas_softc *)arg; 1182665484d8SDoug Ambrisko int status; 1183665484d8SDoug Ambrisko 1184665484d8SDoug Ambrisko /* Clear FW state change interrupt */ 1185665484d8SDoug Ambrisko status = mrsas_clear_intr(sc); 1186665484d8SDoug Ambrisko 1187665484d8SDoug Ambrisko /* Not our interrupt */ 1188665484d8SDoug Ambrisko if (!status) 1189665484d8SDoug Ambrisko return; 1190665484d8SDoug Ambrisko 1191665484d8SDoug Ambrisko /* If we are resetting, bail */ 1192665484d8SDoug Ambrisko if (test_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags)) { 1193665484d8SDoug Ambrisko printf(" Entered into ISR when OCR is going active. \n"); 1194665484d8SDoug Ambrisko mrsas_clear_intr(sc); 1195665484d8SDoug Ambrisko return; 1196665484d8SDoug Ambrisko } 1197665484d8SDoug Ambrisko /* Process for reply request and clear response interrupt */ 1198665484d8SDoug Ambrisko if (mrsas_complete_cmd(sc) != SUCCESS) 1199665484d8SDoug Ambrisko mrsas_clear_intr(sc); 1200665484d8SDoug Ambrisko 1201665484d8SDoug Ambrisko return; 1202665484d8SDoug Ambrisko } 1203665484d8SDoug Ambrisko 1204665484d8SDoug Ambrisko /* 1205665484d8SDoug Ambrisko * mrsas_complete_cmd: Process reply request 1206665484d8SDoug Ambrisko * input: Adapter instance soft state 1207665484d8SDoug Ambrisko * 1208665484d8SDoug Ambrisko * This function is called from mrsas_isr() to process reply request and 1209665484d8SDoug Ambrisko * clear response interrupt. Processing of the reply request entails 1210665484d8SDoug Ambrisko * walking through the reply descriptor array for the command request 1211665484d8SDoug Ambrisko * pended from Firmware. We look at the Function field to determine 1212665484d8SDoug Ambrisko * the command type and perform the appropriate action. Before we 1213665484d8SDoug Ambrisko * return, we clear the response interrupt. 1214665484d8SDoug Ambrisko */ 1215665484d8SDoug Ambrisko static int mrsas_complete_cmd(struct mrsas_softc *sc) 1216665484d8SDoug Ambrisko { 1217665484d8SDoug Ambrisko Mpi2ReplyDescriptorsUnion_t *desc; 1218665484d8SDoug Ambrisko MPI2_SCSI_IO_SUCCESS_REPLY_DESCRIPTOR *reply_desc; 1219665484d8SDoug Ambrisko MRSAS_RAID_SCSI_IO_REQUEST *scsi_io_req; 1220665484d8SDoug Ambrisko struct mrsas_mpt_cmd *cmd_mpt; 1221665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd_mfi; 1222665484d8SDoug Ambrisko u_int8_t arm, reply_descript_type; 1223665484d8SDoug Ambrisko u_int16_t smid, num_completed; 1224665484d8SDoug Ambrisko u_int8_t status, extStatus; 1225665484d8SDoug Ambrisko union desc_value desc_val; 1226665484d8SDoug Ambrisko PLD_LOAD_BALANCE_INFO lbinfo; 1227665484d8SDoug Ambrisko u_int32_t device_id; 1228665484d8SDoug Ambrisko int threshold_reply_count = 0; 1229665484d8SDoug Ambrisko 1230665484d8SDoug Ambrisko 1231665484d8SDoug Ambrisko /* If we have a hardware error, not need to continue */ 1232665484d8SDoug Ambrisko if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) 1233665484d8SDoug Ambrisko return (DONE); 1234665484d8SDoug Ambrisko 1235665484d8SDoug Ambrisko desc = sc->reply_desc_mem; 1236665484d8SDoug Ambrisko desc += sc->last_reply_idx; 1237665484d8SDoug Ambrisko 1238665484d8SDoug Ambrisko reply_desc = (MPI2_SCSI_IO_SUCCESS_REPLY_DESCRIPTOR *)desc; 1239665484d8SDoug Ambrisko 1240665484d8SDoug Ambrisko desc_val.word = desc->Words; 1241665484d8SDoug Ambrisko num_completed = 0; 1242665484d8SDoug Ambrisko 1243665484d8SDoug Ambrisko reply_descript_type = reply_desc->ReplyFlags & MPI2_RPY_DESCRIPT_FLAGS_TYPE_MASK; 1244665484d8SDoug Ambrisko 1245665484d8SDoug Ambrisko /* Find our reply descriptor for the command and process */ 1246665484d8SDoug Ambrisko while((desc_val.u.low != 0xFFFFFFFF) && (desc_val.u.high != 0xFFFFFFFF)) 1247665484d8SDoug Ambrisko { 1248665484d8SDoug Ambrisko smid = reply_desc->SMID; 1249665484d8SDoug Ambrisko cmd_mpt = sc->mpt_cmd_list[smid -1]; 1250665484d8SDoug Ambrisko scsi_io_req = (MRSAS_RAID_SCSI_IO_REQUEST *)cmd_mpt->io_request; 1251665484d8SDoug Ambrisko 1252665484d8SDoug Ambrisko status = scsi_io_req->RaidContext.status; 1253665484d8SDoug Ambrisko extStatus = scsi_io_req->RaidContext.exStatus; 1254665484d8SDoug Ambrisko 1255665484d8SDoug Ambrisko switch (scsi_io_req->Function) 1256665484d8SDoug Ambrisko { 1257665484d8SDoug Ambrisko case MPI2_FUNCTION_SCSI_IO_REQUEST : /*Fast Path IO.*/ 1258665484d8SDoug Ambrisko device_id = cmd_mpt->ccb_ptr->ccb_h.target_id; 1259665484d8SDoug Ambrisko lbinfo = &sc->load_balance_info[device_id]; 1260665484d8SDoug Ambrisko if (cmd_mpt->load_balance == MRSAS_LOAD_BALANCE_FLAG) { 1261665484d8SDoug Ambrisko arm = lbinfo->raid1DevHandle[0] == scsi_io_req->DevHandle ? 0 : 1; 1262665484d8SDoug Ambrisko atomic_dec(&lbinfo->scsi_pending_cmds[arm]); 1263665484d8SDoug Ambrisko cmd_mpt->load_balance &= ~MRSAS_LOAD_BALANCE_FLAG; 1264665484d8SDoug Ambrisko } 1265665484d8SDoug Ambrisko //Fall thru and complete IO 1266665484d8SDoug Ambrisko case MRSAS_MPI2_FUNCTION_LD_IO_REQUEST: 1267665484d8SDoug Ambrisko mrsas_map_mpt_cmd_status(cmd_mpt, status, extStatus); 1268665484d8SDoug Ambrisko mrsas_cmd_done(sc, cmd_mpt); 1269665484d8SDoug Ambrisko scsi_io_req->RaidContext.status = 0; 1270665484d8SDoug Ambrisko scsi_io_req->RaidContext.exStatus = 0; 1271665484d8SDoug Ambrisko atomic_dec(&sc->fw_outstanding); 1272665484d8SDoug Ambrisko break; 1273665484d8SDoug Ambrisko case MRSAS_MPI2_FUNCTION_PASSTHRU_IO_REQUEST: /*MFI command */ 1274665484d8SDoug Ambrisko cmd_mfi = sc->mfi_cmd_list[cmd_mpt->sync_cmd_idx]; 1275665484d8SDoug Ambrisko mrsas_complete_mptmfi_passthru(sc, cmd_mfi, status); 1276665484d8SDoug Ambrisko cmd_mpt->flags = 0; 1277665484d8SDoug Ambrisko mrsas_release_mpt_cmd(cmd_mpt); 1278665484d8SDoug Ambrisko break; 1279665484d8SDoug Ambrisko } 1280665484d8SDoug Ambrisko 1281665484d8SDoug Ambrisko sc->last_reply_idx++; 1282665484d8SDoug Ambrisko if (sc->last_reply_idx >= sc->reply_q_depth) 1283665484d8SDoug Ambrisko sc->last_reply_idx = 0; 1284665484d8SDoug Ambrisko 1285665484d8SDoug Ambrisko desc->Words = ~((uint64_t)0x00); /* set it back to all 0xFFFFFFFFs */ 1286665484d8SDoug Ambrisko num_completed++; 1287665484d8SDoug Ambrisko threshold_reply_count++; 1288665484d8SDoug Ambrisko 1289665484d8SDoug Ambrisko /* Get the next reply descriptor */ 1290665484d8SDoug Ambrisko if (!sc->last_reply_idx) 1291665484d8SDoug Ambrisko desc = sc->reply_desc_mem; 1292665484d8SDoug Ambrisko else 1293665484d8SDoug Ambrisko desc++; 1294665484d8SDoug Ambrisko 1295665484d8SDoug Ambrisko reply_desc = (MPI2_SCSI_IO_SUCCESS_REPLY_DESCRIPTOR *)desc; 1296665484d8SDoug Ambrisko desc_val.word = desc->Words; 1297665484d8SDoug Ambrisko 1298665484d8SDoug Ambrisko reply_descript_type = reply_desc->ReplyFlags & MPI2_RPY_DESCRIPT_FLAGS_TYPE_MASK; 1299665484d8SDoug Ambrisko 1300665484d8SDoug Ambrisko if(reply_descript_type == MPI2_RPY_DESCRIPT_FLAGS_UNUSED) 1301665484d8SDoug Ambrisko break; 1302665484d8SDoug Ambrisko 1303665484d8SDoug Ambrisko /* 1304665484d8SDoug Ambrisko * Write to reply post index after completing threshold reply count 1305665484d8SDoug Ambrisko * and still there are more replies in reply queue pending to be 1306665484d8SDoug Ambrisko * completed. 1307665484d8SDoug Ambrisko */ 1308665484d8SDoug Ambrisko if (threshold_reply_count >= THRESHOLD_REPLY_COUNT) { 1309665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, reply_post_host_index), 1310665484d8SDoug Ambrisko sc->last_reply_idx); 1311665484d8SDoug Ambrisko threshold_reply_count = 0; 1312665484d8SDoug Ambrisko } 1313665484d8SDoug Ambrisko } 1314665484d8SDoug Ambrisko 1315665484d8SDoug Ambrisko /* No match, just return */ 1316665484d8SDoug Ambrisko if (num_completed == 0) 1317665484d8SDoug Ambrisko return (DONE); 1318665484d8SDoug Ambrisko 1319665484d8SDoug Ambrisko /* Clear response interrupt */ 1320665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, reply_post_host_index),sc->last_reply_idx); 1321665484d8SDoug Ambrisko 1322665484d8SDoug Ambrisko return(0); 1323665484d8SDoug Ambrisko } 1324665484d8SDoug Ambrisko 1325665484d8SDoug Ambrisko /* 1326665484d8SDoug Ambrisko * mrsas_map_mpt_cmd_status: Allocate DMAable memory. 1327665484d8SDoug Ambrisko * input: Adapter instance soft state 1328665484d8SDoug Ambrisko * 1329665484d8SDoug Ambrisko * This function is called from mrsas_complete_cmd(), for LD IO and FastPath IO. 1330665484d8SDoug Ambrisko * It checks the command status and maps the appropriate CAM status for the CCB. 1331665484d8SDoug Ambrisko */ 1332665484d8SDoug Ambrisko void mrsas_map_mpt_cmd_status(struct mrsas_mpt_cmd *cmd, u_int8_t status, u_int8_t extStatus) 1333665484d8SDoug Ambrisko { 1334665484d8SDoug Ambrisko struct mrsas_softc *sc = cmd->sc; 1335665484d8SDoug Ambrisko u_int8_t *sense_data; 1336665484d8SDoug Ambrisko 1337665484d8SDoug Ambrisko switch (status) { 1338665484d8SDoug Ambrisko case MFI_STAT_OK: 1339665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status = CAM_REQ_CMP; 1340665484d8SDoug Ambrisko break; 1341665484d8SDoug Ambrisko case MFI_STAT_SCSI_IO_FAILED: 1342665484d8SDoug Ambrisko case MFI_STAT_SCSI_DONE_WITH_ERROR: 1343665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status = CAM_SCSI_STATUS_ERROR; 1344665484d8SDoug Ambrisko sense_data = (u_int8_t *)&cmd->ccb_ptr->csio.sense_data; 1345665484d8SDoug Ambrisko if (sense_data) { 1346665484d8SDoug Ambrisko /* For now just copy 18 bytes back */ 1347665484d8SDoug Ambrisko memcpy(sense_data, cmd->sense, 18); 1348665484d8SDoug Ambrisko cmd->ccb_ptr->csio.sense_len = 18; 1349665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status |= CAM_AUTOSNS_VALID; 1350665484d8SDoug Ambrisko } 1351665484d8SDoug Ambrisko break; 1352665484d8SDoug Ambrisko case MFI_STAT_LD_OFFLINE: 1353665484d8SDoug Ambrisko case MFI_STAT_DEVICE_NOT_FOUND: 1354665484d8SDoug Ambrisko if (cmd->ccb_ptr->ccb_h.target_lun) 1355665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status |= CAM_LUN_INVALID; 1356665484d8SDoug Ambrisko else 1357665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status |= CAM_DEV_NOT_THERE; 1358665484d8SDoug Ambrisko break; 1359665484d8SDoug Ambrisko case MFI_STAT_CONFIG_SEQ_MISMATCH: 1360665484d8SDoug Ambrisko /*send status to CAM layer to retry sending command without 1361665484d8SDoug Ambrisko * decrementing retry counter*/ 1362665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status |= CAM_REQUEUE_REQ; 1363665484d8SDoug Ambrisko break; 1364665484d8SDoug Ambrisko default: 1365665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "FW cmd complete status %x\n", status); 1366665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status = CAM_REQ_CMP_ERR; 1367665484d8SDoug Ambrisko cmd->ccb_ptr->csio.scsi_status = status; 1368665484d8SDoug Ambrisko } 1369665484d8SDoug Ambrisko return; 1370665484d8SDoug Ambrisko } 1371665484d8SDoug Ambrisko 1372665484d8SDoug Ambrisko /* 1373665484d8SDoug Ambrisko * mrsas_alloc_mem: Allocate DMAable memory. 1374665484d8SDoug Ambrisko * input: Adapter instance soft state 1375665484d8SDoug Ambrisko * 1376665484d8SDoug Ambrisko * This function creates the parent DMA tag and allocates DMAable memory. 1377665484d8SDoug Ambrisko * DMA tag describes constraints of DMA mapping. Memory allocated is mapped 1378665484d8SDoug Ambrisko * into Kernel virtual address. Callback argument is physical memory address. 1379665484d8SDoug Ambrisko */ 1380665484d8SDoug Ambrisko static int mrsas_alloc_mem(struct mrsas_softc *sc) 1381665484d8SDoug Ambrisko { 1382665484d8SDoug Ambrisko u_int32_t verbuf_size, io_req_size, reply_desc_size, sense_size, 1383665484d8SDoug Ambrisko chain_frame_size, evt_detail_size; 1384665484d8SDoug Ambrisko 1385665484d8SDoug Ambrisko /* 1386665484d8SDoug Ambrisko * Allocate parent DMA tag 1387665484d8SDoug Ambrisko */ 1388665484d8SDoug Ambrisko if (bus_dma_tag_create(NULL, /* parent */ 1389665484d8SDoug Ambrisko 1, /* alignment */ 1390665484d8SDoug Ambrisko 0, /* boundary */ 1391665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, /* lowaddr */ 1392665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, /* highaddr */ 1393665484d8SDoug Ambrisko NULL, NULL, /* filter, filterarg */ 1394665484d8SDoug Ambrisko MRSAS_MAX_IO_SIZE,/* maxsize */ 1395665484d8SDoug Ambrisko MRSAS_MAX_SGL, /* nsegments */ 1396665484d8SDoug Ambrisko MRSAS_MAX_IO_SIZE,/* maxsegsize */ 1397665484d8SDoug Ambrisko 0, /* flags */ 1398665484d8SDoug Ambrisko NULL, NULL, /* lockfunc, lockarg */ 1399665484d8SDoug Ambrisko &sc->mrsas_parent_tag /* tag */ 1400665484d8SDoug Ambrisko )) { 1401665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate parent DMA tag\n"); 1402665484d8SDoug Ambrisko return(ENOMEM); 1403665484d8SDoug Ambrisko } 1404665484d8SDoug Ambrisko 1405665484d8SDoug Ambrisko /* 1406665484d8SDoug Ambrisko * Allocate for version buffer 1407665484d8SDoug Ambrisko */ 1408665484d8SDoug Ambrisko verbuf_size = MRSAS_MAX_NAME_LENGTH*(sizeof(bus_addr_t)); 1409665484d8SDoug Ambrisko if (bus_dma_tag_create(sc->mrsas_parent_tag, // parent 1410665484d8SDoug Ambrisko 1, 0, // algnmnt, boundary 1411665484d8SDoug Ambrisko BUS_SPACE_MAXADDR_32BIT,// lowaddr 1412665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, // highaddr 1413665484d8SDoug Ambrisko NULL, NULL, // filter, filterarg 1414665484d8SDoug Ambrisko verbuf_size, // maxsize 1415665484d8SDoug Ambrisko 1, // msegments 1416665484d8SDoug Ambrisko verbuf_size, // maxsegsize 1417665484d8SDoug Ambrisko BUS_DMA_ALLOCNOW, // flags 1418665484d8SDoug Ambrisko NULL, NULL, // lockfunc, lockarg 1419665484d8SDoug Ambrisko &sc->verbuf_tag)) { 1420665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate verbuf DMA tag\n"); 1421665484d8SDoug Ambrisko return (ENOMEM); 1422665484d8SDoug Ambrisko } 1423665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->verbuf_tag, (void **)&sc->verbuf_mem, 1424665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->verbuf_dmamap)) { 1425665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate verbuf memory\n"); 1426665484d8SDoug Ambrisko return (ENOMEM); 1427665484d8SDoug Ambrisko } 1428665484d8SDoug Ambrisko bzero(sc->verbuf_mem, verbuf_size); 1429665484d8SDoug Ambrisko if (bus_dmamap_load(sc->verbuf_tag, sc->verbuf_dmamap, sc->verbuf_mem, 1430665484d8SDoug Ambrisko verbuf_size, mrsas_addr_cb, &sc->verbuf_phys_addr, BUS_DMA_NOWAIT)){ 1431665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load verbuf DMA map\n"); 1432665484d8SDoug Ambrisko return(ENOMEM); 1433665484d8SDoug Ambrisko } 1434665484d8SDoug Ambrisko 1435665484d8SDoug Ambrisko /* 1436665484d8SDoug Ambrisko * Allocate IO Request Frames 1437665484d8SDoug Ambrisko */ 1438665484d8SDoug Ambrisko io_req_size = sc->io_frames_alloc_sz; 1439665484d8SDoug Ambrisko if (bus_dma_tag_create( sc->mrsas_parent_tag, // parent 1440665484d8SDoug Ambrisko 16, 0, // algnmnt, boundary 1441665484d8SDoug Ambrisko BUS_SPACE_MAXADDR_32BIT,// lowaddr 1442665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, // highaddr 1443665484d8SDoug Ambrisko NULL, NULL, // filter, filterarg 1444665484d8SDoug Ambrisko io_req_size, // maxsize 1445665484d8SDoug Ambrisko 1, // msegments 1446665484d8SDoug Ambrisko io_req_size, // maxsegsize 1447665484d8SDoug Ambrisko BUS_DMA_ALLOCNOW, // flags 1448665484d8SDoug Ambrisko NULL, NULL, // lockfunc, lockarg 1449665484d8SDoug Ambrisko &sc->io_request_tag)) { 1450665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create IO request tag\n"); 1451665484d8SDoug Ambrisko return (ENOMEM); 1452665484d8SDoug Ambrisko } 1453665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->io_request_tag, (void **)&sc->io_request_mem, 1454665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->io_request_dmamap)) { 1455665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc IO request memory\n"); 1456665484d8SDoug Ambrisko return (ENOMEM); 1457665484d8SDoug Ambrisko } 1458665484d8SDoug Ambrisko bzero(sc->io_request_mem, io_req_size); 1459665484d8SDoug Ambrisko if (bus_dmamap_load(sc->io_request_tag, sc->io_request_dmamap, 1460665484d8SDoug Ambrisko sc->io_request_mem, io_req_size, mrsas_addr_cb, 1461665484d8SDoug Ambrisko &sc->io_request_phys_addr, BUS_DMA_NOWAIT)) { 1462665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load IO request memory\n"); 1463665484d8SDoug Ambrisko return (ENOMEM); 1464665484d8SDoug Ambrisko } 1465665484d8SDoug Ambrisko 1466665484d8SDoug Ambrisko /* 1467665484d8SDoug Ambrisko * Allocate Chain Frames 1468665484d8SDoug Ambrisko */ 1469665484d8SDoug Ambrisko chain_frame_size = sc->chain_frames_alloc_sz; 1470665484d8SDoug Ambrisko if (bus_dma_tag_create( sc->mrsas_parent_tag, // parent 1471665484d8SDoug Ambrisko 4, 0, // algnmnt, boundary 1472665484d8SDoug Ambrisko BUS_SPACE_MAXADDR_32BIT,// lowaddr 1473665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, // highaddr 1474665484d8SDoug Ambrisko NULL, NULL, // filter, filterarg 1475665484d8SDoug Ambrisko chain_frame_size, // maxsize 1476665484d8SDoug Ambrisko 1, // msegments 1477665484d8SDoug Ambrisko chain_frame_size, // maxsegsize 1478665484d8SDoug Ambrisko BUS_DMA_ALLOCNOW, // flags 1479665484d8SDoug Ambrisko NULL, NULL, // lockfunc, lockarg 1480665484d8SDoug Ambrisko &sc->chain_frame_tag)) { 1481665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create chain frame tag\n"); 1482665484d8SDoug Ambrisko return (ENOMEM); 1483665484d8SDoug Ambrisko } 1484665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->chain_frame_tag, (void **)&sc->chain_frame_mem, 1485665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->chain_frame_dmamap)) { 1486665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc chain frame memory\n"); 1487665484d8SDoug Ambrisko return (ENOMEM); 1488665484d8SDoug Ambrisko } 1489665484d8SDoug Ambrisko bzero(sc->chain_frame_mem, chain_frame_size); 1490665484d8SDoug Ambrisko if (bus_dmamap_load(sc->chain_frame_tag, sc->chain_frame_dmamap, 1491665484d8SDoug Ambrisko sc->chain_frame_mem, chain_frame_size, mrsas_addr_cb, 1492665484d8SDoug Ambrisko &sc->chain_frame_phys_addr, BUS_DMA_NOWAIT)) { 1493665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load chain frame memory\n"); 1494665484d8SDoug Ambrisko return (ENOMEM); 1495665484d8SDoug Ambrisko } 1496665484d8SDoug Ambrisko 1497665484d8SDoug Ambrisko /* 1498665484d8SDoug Ambrisko * Allocate Reply Descriptor Array 1499665484d8SDoug Ambrisko */ 1500665484d8SDoug Ambrisko reply_desc_size = sc->reply_alloc_sz; 1501665484d8SDoug Ambrisko if (bus_dma_tag_create( sc->mrsas_parent_tag, // parent 1502665484d8SDoug Ambrisko 16, 0, // algnmnt, boundary 1503665484d8SDoug Ambrisko BUS_SPACE_MAXADDR_32BIT,// lowaddr 1504665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, // highaddr 1505665484d8SDoug Ambrisko NULL, NULL, // filter, filterarg 1506665484d8SDoug Ambrisko reply_desc_size, // maxsize 1507665484d8SDoug Ambrisko 1, // msegments 1508665484d8SDoug Ambrisko reply_desc_size, // maxsegsize 1509665484d8SDoug Ambrisko BUS_DMA_ALLOCNOW, // flags 1510665484d8SDoug Ambrisko NULL, NULL, // lockfunc, lockarg 1511665484d8SDoug Ambrisko &sc->reply_desc_tag)) { 1512665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create reply descriptor tag\n"); 1513665484d8SDoug Ambrisko return (ENOMEM); 1514665484d8SDoug Ambrisko } 1515665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->reply_desc_tag, (void **)&sc->reply_desc_mem, 1516665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->reply_desc_dmamap)) { 1517665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc reply descriptor memory\n"); 1518665484d8SDoug Ambrisko return (ENOMEM); 1519665484d8SDoug Ambrisko } 1520665484d8SDoug Ambrisko if (bus_dmamap_load(sc->reply_desc_tag, sc->reply_desc_dmamap, 1521665484d8SDoug Ambrisko sc->reply_desc_mem, reply_desc_size, mrsas_addr_cb, 1522665484d8SDoug Ambrisko &sc->reply_desc_phys_addr, BUS_DMA_NOWAIT)) { 1523665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load reply descriptor memory\n"); 1524665484d8SDoug Ambrisko return (ENOMEM); 1525665484d8SDoug Ambrisko } 1526665484d8SDoug Ambrisko 1527665484d8SDoug Ambrisko /* 1528665484d8SDoug Ambrisko * Allocate Sense Buffer Array. Keep in lower 4GB 1529665484d8SDoug Ambrisko */ 1530665484d8SDoug Ambrisko sense_size = sc->max_fw_cmds * MRSAS_SENSE_LEN; 1531665484d8SDoug Ambrisko if (bus_dma_tag_create(sc->mrsas_parent_tag, // parent 1532665484d8SDoug Ambrisko 64, 0, // algnmnt, boundary 1533665484d8SDoug Ambrisko BUS_SPACE_MAXADDR_32BIT,// lowaddr 1534665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, // highaddr 1535665484d8SDoug Ambrisko NULL, NULL, // filter, filterarg 1536665484d8SDoug Ambrisko sense_size, // maxsize 1537665484d8SDoug Ambrisko 1, // nsegments 1538665484d8SDoug Ambrisko sense_size, // maxsegsize 1539665484d8SDoug Ambrisko BUS_DMA_ALLOCNOW, // flags 1540665484d8SDoug Ambrisko NULL, NULL, // lockfunc, lockarg 1541665484d8SDoug Ambrisko &sc->sense_tag)) { 1542665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate sense buf tag\n"); 1543665484d8SDoug Ambrisko return (ENOMEM); 1544665484d8SDoug Ambrisko } 1545665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->sense_tag, (void **)&sc->sense_mem, 1546665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->sense_dmamap)) { 1547665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate sense buf memory\n"); 1548665484d8SDoug Ambrisko return (ENOMEM); 1549665484d8SDoug Ambrisko } 1550665484d8SDoug Ambrisko if (bus_dmamap_load(sc->sense_tag, sc->sense_dmamap, 1551665484d8SDoug Ambrisko sc->sense_mem, sense_size, mrsas_addr_cb, &sc->sense_phys_addr, 1552665484d8SDoug Ambrisko BUS_DMA_NOWAIT)){ 1553665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load sense buf memory\n"); 1554665484d8SDoug Ambrisko return (ENOMEM); 1555665484d8SDoug Ambrisko } 1556665484d8SDoug Ambrisko 1557665484d8SDoug Ambrisko /* 1558665484d8SDoug Ambrisko * Allocate for Event detail structure 1559665484d8SDoug Ambrisko */ 1560665484d8SDoug Ambrisko evt_detail_size = sizeof(struct mrsas_evt_detail); 1561665484d8SDoug Ambrisko if (bus_dma_tag_create( sc->mrsas_parent_tag, // parent 1562665484d8SDoug Ambrisko 1, 0, // algnmnt, boundary 1563665484d8SDoug Ambrisko BUS_SPACE_MAXADDR_32BIT,// lowaddr 1564665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, // highaddr 1565665484d8SDoug Ambrisko NULL, NULL, // filter, filterarg 1566665484d8SDoug Ambrisko evt_detail_size, // maxsize 1567665484d8SDoug Ambrisko 1, // msegments 1568665484d8SDoug Ambrisko evt_detail_size, // maxsegsize 1569665484d8SDoug Ambrisko BUS_DMA_ALLOCNOW, // flags 1570665484d8SDoug Ambrisko NULL, NULL, // lockfunc, lockarg 1571665484d8SDoug Ambrisko &sc->evt_detail_tag)) { 1572665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create Event detail tag\n"); 1573665484d8SDoug Ambrisko return (ENOMEM); 1574665484d8SDoug Ambrisko } 1575665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->evt_detail_tag, (void **)&sc->evt_detail_mem, 1576665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->evt_detail_dmamap)) { 1577665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc Event detail buffer memory\n"); 1578665484d8SDoug Ambrisko return (ENOMEM); 1579665484d8SDoug Ambrisko } 1580665484d8SDoug Ambrisko bzero(sc->evt_detail_mem, evt_detail_size); 1581665484d8SDoug Ambrisko if (bus_dmamap_load(sc->evt_detail_tag, sc->evt_detail_dmamap, 1582665484d8SDoug Ambrisko sc->evt_detail_mem, evt_detail_size, mrsas_addr_cb, 1583665484d8SDoug Ambrisko &sc->evt_detail_phys_addr, BUS_DMA_NOWAIT)) { 1584665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load Event detail buffer memory\n"); 1585665484d8SDoug Ambrisko return (ENOMEM); 1586665484d8SDoug Ambrisko } 1587665484d8SDoug Ambrisko 1588665484d8SDoug Ambrisko 1589665484d8SDoug Ambrisko /* 1590665484d8SDoug Ambrisko * Create a dma tag for data buffers; size will be the maximum 1591665484d8SDoug Ambrisko * possible I/O size (280kB). 1592665484d8SDoug Ambrisko */ 1593665484d8SDoug Ambrisko if (bus_dma_tag_create(sc->mrsas_parent_tag, // parent 1594665484d8SDoug Ambrisko 1, // alignment 1595665484d8SDoug Ambrisko 0, // boundary 1596665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, // lowaddr 1597665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, // highaddr 1598665484d8SDoug Ambrisko NULL, NULL, // filter, filterarg 1599665484d8SDoug Ambrisko MRSAS_MAX_IO_SIZE, // maxsize 1600665484d8SDoug Ambrisko MRSAS_MAX_SGL, // nsegments 1601665484d8SDoug Ambrisko MRSAS_MAX_IO_SIZE, // maxsegsize 1602665484d8SDoug Ambrisko BUS_DMA_ALLOCNOW, // flags 1603665484d8SDoug Ambrisko busdma_lock_mutex, // lockfunc 1604665484d8SDoug Ambrisko &sc->io_lock, // lockfuncarg 1605665484d8SDoug Ambrisko &sc->data_tag)) { 1606665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create data dma tag\n"); 1607665484d8SDoug Ambrisko return(ENOMEM); 1608665484d8SDoug Ambrisko } 1609665484d8SDoug Ambrisko 1610665484d8SDoug Ambrisko return(0); 1611665484d8SDoug Ambrisko } 1612665484d8SDoug Ambrisko 1613665484d8SDoug Ambrisko /* 1614665484d8SDoug Ambrisko * mrsas_addr_cb: Callback function of bus_dmamap_load() 1615665484d8SDoug Ambrisko * input: callback argument, 1616665484d8SDoug Ambrisko * machine dependent type that describes DMA segments, 1617665484d8SDoug Ambrisko * number of segments, 1618665484d8SDoug Ambrisko * error code. 1619665484d8SDoug Ambrisko * 1620665484d8SDoug Ambrisko * This function is for the driver to receive mapping information resultant 1621665484d8SDoug Ambrisko * of the bus_dmamap_load(). The information is actually not being used, 1622665484d8SDoug Ambrisko * but the address is saved anyway. 1623665484d8SDoug Ambrisko */ 1624665484d8SDoug Ambrisko void 1625665484d8SDoug Ambrisko mrsas_addr_cb(void *arg, bus_dma_segment_t *segs, int nsegs, int error) 1626665484d8SDoug Ambrisko { 1627665484d8SDoug Ambrisko bus_addr_t *addr; 1628665484d8SDoug Ambrisko 1629665484d8SDoug Ambrisko addr = arg; 1630665484d8SDoug Ambrisko *addr = segs[0].ds_addr; 1631665484d8SDoug Ambrisko } 1632665484d8SDoug Ambrisko 1633665484d8SDoug Ambrisko /* 1634665484d8SDoug Ambrisko * mrsas_setup_raidmap: Set up RAID map. 1635665484d8SDoug Ambrisko * input: Adapter instance soft state 1636665484d8SDoug Ambrisko * 1637665484d8SDoug Ambrisko * Allocate DMA memory for the RAID maps and perform setup. 1638665484d8SDoug Ambrisko */ 1639665484d8SDoug Ambrisko static int mrsas_setup_raidmap(struct mrsas_softc *sc) 1640665484d8SDoug Ambrisko { 1641*4799d485SKashyap D Desai int i; 1642*4799d485SKashyap D Desai 1643*4799d485SKashyap D Desai sc->drv_supported_vd_count = 1644*4799d485SKashyap D Desai MRSAS_MAX_LD_CHANNELS * MRSAS_MAX_DEV_PER_CHANNEL; 1645*4799d485SKashyap D Desai sc->drv_supported_pd_count = 1646*4799d485SKashyap D Desai MRSAS_MAX_PD_CHANNELS * MRSAS_MAX_DEV_PER_CHANNEL; 1647*4799d485SKashyap D Desai 1648*4799d485SKashyap D Desai if(sc->max256vdSupport) { 1649*4799d485SKashyap D Desai sc->fw_supported_vd_count = MAX_LOGICAL_DRIVES_EXT; 1650*4799d485SKashyap D Desai sc->fw_supported_pd_count = MAX_PHYSICAL_DEVICES; 1651*4799d485SKashyap D Desai } else { 1652*4799d485SKashyap D Desai sc->fw_supported_vd_count = MAX_LOGICAL_DRIVES; 1653*4799d485SKashyap D Desai sc->fw_supported_pd_count = MAX_PHYSICAL_DEVICES; 1654*4799d485SKashyap D Desai } 1655*4799d485SKashyap D Desai 1656*4799d485SKashyap D Desai #if VD_EXT_DEBUG 1657*4799d485SKashyap D Desai device_printf(sc->mrsas_dev, "FW supports: max256vdSupport = %s\n", 1658*4799d485SKashyap D Desai sc->max256vdSupport ? "YES":"NO"); 1659*4799d485SKashyap D Desai device_printf(sc->mrsas_dev, "FW supports %dVDs %dPDs\n" 1660*4799d485SKashyap D Desai "DRIVER supports %dVDs %dPDs \n", 1661*4799d485SKashyap D Desai sc->fw_supported_vd_count, sc->fw_supported_pd_count, 1662*4799d485SKashyap D Desai sc->drv_supported_vd_count, sc->drv_supported_pd_count); 1663*4799d485SKashyap D Desai #endif 1664*4799d485SKashyap D Desai 1665*4799d485SKashyap D Desai sc->old_map_sz = sizeof(MR_FW_RAID_MAP) + 1666*4799d485SKashyap D Desai (sizeof(MR_LD_SPAN_MAP) * (sc->fw_supported_vd_count - 1)); 1667*4799d485SKashyap D Desai sc->new_map_sz = sizeof(MR_FW_RAID_MAP_EXT); 1668*4799d485SKashyap D Desai sc->drv_map_sz = sizeof(MR_DRV_RAID_MAP) + 1669*4799d485SKashyap D Desai (sizeof(MR_LD_SPAN_MAP) * (sc->drv_supported_vd_count-1)); 1670*4799d485SKashyap D Desai 1671*4799d485SKashyap D Desai for (i = 0; i < 2; i++) { 1672*4799d485SKashyap D Desai sc->ld_drv_map[i] = 1673*4799d485SKashyap D Desai (void*) malloc(sc->drv_map_sz, M_MRSAS, M_NOWAIT); 1674*4799d485SKashyap D Desai /* Do Error handling */ 1675*4799d485SKashyap D Desai if (!sc->ld_drv_map[i]) { 1676*4799d485SKashyap D Desai device_printf(sc->mrsas_dev, "Could not allocate memory for local map"); 1677*4799d485SKashyap D Desai 1678*4799d485SKashyap D Desai if (i == 1) 1679*4799d485SKashyap D Desai free (sc->ld_drv_map[0], M_MRSAS); 1680*4799d485SKashyap D Desai //ABORT driver initialization 1681*4799d485SKashyap D Desai goto ABORT; 1682*4799d485SKashyap D Desai } 1683*4799d485SKashyap D Desai } 1684*4799d485SKashyap D Desai 1685*4799d485SKashyap D Desai sc->max_map_sz = max(sc->old_map_sz, sc->new_map_sz); 1686*4799d485SKashyap D Desai 1687*4799d485SKashyap D Desai if(sc->max256vdSupport) 1688*4799d485SKashyap D Desai sc->current_map_sz = sc->new_map_sz; 1689*4799d485SKashyap D Desai else 1690*4799d485SKashyap D Desai sc->current_map_sz = sc->old_map_sz; 1691*4799d485SKashyap D Desai 1692665484d8SDoug Ambrisko 1693665484d8SDoug Ambrisko for (int i=0; i < 2; i++) 1694665484d8SDoug Ambrisko { 1695665484d8SDoug Ambrisko if (bus_dma_tag_create(sc->mrsas_parent_tag, // parent 1696665484d8SDoug Ambrisko 4, 0, // algnmnt, boundary 1697665484d8SDoug Ambrisko BUS_SPACE_MAXADDR_32BIT,// lowaddr 1698665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, // highaddr 1699665484d8SDoug Ambrisko NULL, NULL, // filter, filterarg 1700*4799d485SKashyap D Desai sc->max_map_sz, // maxsize 1701665484d8SDoug Ambrisko 1, // nsegments 1702*4799d485SKashyap D Desai sc->max_map_sz, // maxsegsize 1703665484d8SDoug Ambrisko BUS_DMA_ALLOCNOW, // flags 1704665484d8SDoug Ambrisko NULL, NULL, // lockfunc, lockarg 1705665484d8SDoug Ambrisko &sc->raidmap_tag[i])) { 1706*4799d485SKashyap D Desai device_printf(sc->mrsas_dev, 1707*4799d485SKashyap D Desai "Cannot allocate raid map tag.\n"); 1708665484d8SDoug Ambrisko return (ENOMEM); 1709665484d8SDoug Ambrisko } 1710*4799d485SKashyap D Desai if (bus_dmamem_alloc(sc->raidmap_tag[i], 1711*4799d485SKashyap D Desai (void **)&sc->raidmap_mem[i], 1712665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->raidmap_dmamap[i])) { 1713*4799d485SKashyap D Desai device_printf(sc->mrsas_dev, 1714*4799d485SKashyap D Desai "Cannot allocate raidmap memory.\n"); 1715665484d8SDoug Ambrisko return (ENOMEM); 1716665484d8SDoug Ambrisko } 1717*4799d485SKashyap D Desai 1718*4799d485SKashyap D Desai bzero (sc->raidmap_mem[i], sc->max_map_sz); 1719*4799d485SKashyap D Desai 1720665484d8SDoug Ambrisko if (bus_dmamap_load(sc->raidmap_tag[i], sc->raidmap_dmamap[i], 1721*4799d485SKashyap D Desai sc->raidmap_mem[i], sc->max_map_sz, 1722*4799d485SKashyap D Desai mrsas_addr_cb, &sc->raidmap_phys_addr[i], 1723665484d8SDoug Ambrisko BUS_DMA_NOWAIT)){ 1724665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load raidmap memory.\n"); 1725665484d8SDoug Ambrisko return (ENOMEM); 1726665484d8SDoug Ambrisko } 1727665484d8SDoug Ambrisko if (!sc->raidmap_mem[i]) { 1728*4799d485SKashyap D Desai device_printf(sc->mrsas_dev, 1729*4799d485SKashyap D Desai "Cannot allocate memory for raid map.\n"); 1730665484d8SDoug Ambrisko return (ENOMEM); 1731665484d8SDoug Ambrisko } 1732665484d8SDoug Ambrisko } 1733665484d8SDoug Ambrisko 1734665484d8SDoug Ambrisko if (!mrsas_get_map_info(sc)) 1735665484d8SDoug Ambrisko mrsas_sync_map_info(sc); 1736665484d8SDoug Ambrisko 1737665484d8SDoug Ambrisko return (0); 1738*4799d485SKashyap D Desai 1739*4799d485SKashyap D Desai ABORT: 1740*4799d485SKashyap D Desai return (1); 1741665484d8SDoug Ambrisko } 1742665484d8SDoug Ambrisko 1743665484d8SDoug Ambrisko /** 1744665484d8SDoug Ambrisko * mrsas_init_fw: Initialize Firmware 1745665484d8SDoug Ambrisko * input: Adapter soft state 1746665484d8SDoug Ambrisko * 1747665484d8SDoug Ambrisko * Calls transition_to_ready() to make sure Firmware is in operational 1748665484d8SDoug Ambrisko * state and calls mrsas_init_adapter() to send IOC_INIT command to 1749665484d8SDoug Ambrisko * Firmware. It issues internal commands to get the controller info 1750665484d8SDoug Ambrisko * after the IOC_INIT command response is received by Firmware. 1751665484d8SDoug Ambrisko * Note: code relating to get_pdlist, get_ld_list and max_sectors 1752665484d8SDoug Ambrisko * are currently not being used, it is left here as placeholder. 1753665484d8SDoug Ambrisko */ 1754665484d8SDoug Ambrisko static int mrsas_init_fw(struct mrsas_softc *sc) 1755665484d8SDoug Ambrisko { 1756665484d8SDoug Ambrisko u_int32_t max_sectors_1; 1757665484d8SDoug Ambrisko u_int32_t max_sectors_2; 1758665484d8SDoug Ambrisko u_int32_t tmp_sectors; 1759665484d8SDoug Ambrisko struct mrsas_ctrl_info *ctrl_info; 1760665484d8SDoug Ambrisko 1761665484d8SDoug Ambrisko int ret, ocr = 0; 1762665484d8SDoug Ambrisko 1763665484d8SDoug Ambrisko 1764665484d8SDoug Ambrisko /* Make sure Firmware is ready */ 1765665484d8SDoug Ambrisko ret = mrsas_transition_to_ready(sc, ocr); 1766665484d8SDoug Ambrisko if (ret != SUCCESS) { 1767665484d8SDoug Ambrisko return(ret); 1768665484d8SDoug Ambrisko } 1769665484d8SDoug Ambrisko 1770665484d8SDoug Ambrisko /* Get operational params, sge flags, send init cmd to ctlr */ 1771665484d8SDoug Ambrisko if (mrsas_init_adapter(sc) != SUCCESS){ 1772665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Adapter initialize Fail.\n"); 1773665484d8SDoug Ambrisko return(1); 1774665484d8SDoug Ambrisko } 1775665484d8SDoug Ambrisko 1776665484d8SDoug Ambrisko /* Allocate internal commands for pass-thru */ 1777665484d8SDoug Ambrisko if (mrsas_alloc_mfi_cmds(sc) != SUCCESS){ 1778665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Allocate MFI cmd failed.\n"); 1779665484d8SDoug Ambrisko return(1); 1780665484d8SDoug Ambrisko } 1781665484d8SDoug Ambrisko 1782*4799d485SKashyap D Desai /* 1783*4799d485SKashyap D Desai * Get the controller info from FW, so that 1784*4799d485SKashyap D Desai * the MAX VD support availability can be decided. 1785*4799d485SKashyap D Desai */ 1786*4799d485SKashyap D Desai ctrl_info = malloc(sizeof(struct mrsas_ctrl_info), M_MRSAS, M_NOWAIT); 1787*4799d485SKashyap D Desai if (!ctrl_info) 1788*4799d485SKashyap D Desai device_printf(sc->mrsas_dev, "Malloc for ctrl_info failed.\n"); 1789*4799d485SKashyap D Desai 1790*4799d485SKashyap D Desai if (mrsas_get_ctrl_info(sc, ctrl_info)) { 1791*4799d485SKashyap D Desai device_printf(sc->mrsas_dev, "Unable to get FW ctrl_info.\n"); 1792*4799d485SKashyap D Desai } 1793*4799d485SKashyap D Desai 1794*4799d485SKashyap D Desai sc->max256vdSupport = 1795*4799d485SKashyap D Desai (u_int8_t) ctrl_info->adapterOperations3.supportMaxExtLDs; 1796*4799d485SKashyap D Desai 1797*4799d485SKashyap D Desai if (ctrl_info->max_lds > 64){ 1798*4799d485SKashyap D Desai sc->max256vdSupport = 1; 1799*4799d485SKashyap D Desai } 1800*4799d485SKashyap D Desai 1801665484d8SDoug Ambrisko if (mrsas_setup_raidmap(sc) != SUCCESS) { 1802665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Set up RAID map failed.\n"); 1803665484d8SDoug Ambrisko return(1); 1804665484d8SDoug Ambrisko } 1805665484d8SDoug Ambrisko 1806665484d8SDoug Ambrisko /* For pass-thru, get PD/LD list and controller info */ 1807*4799d485SKashyap D Desai memset(sc->pd_list, 0, 1808*4799d485SKashyap D Desai MRSAS_MAX_PD * sizeof(struct mrsas_pd_list)); 1809665484d8SDoug Ambrisko mrsas_get_pd_list(sc); 1810665484d8SDoug Ambrisko 1811*4799d485SKashyap D Desai memset(sc->ld_ids, 0xff, MRSAS_MAX_LD_IDS); 1812665484d8SDoug Ambrisko mrsas_get_ld_list(sc); 1813665484d8SDoug Ambrisko 1814665484d8SDoug Ambrisko /* 1815665484d8SDoug Ambrisko * Compute the max allowed sectors per IO: The controller info has two 1816665484d8SDoug Ambrisko * limits on max sectors. Driver should use the minimum of these two. 1817665484d8SDoug Ambrisko * 1818665484d8SDoug Ambrisko * 1 << stripe_sz_ops.min = max sectors per strip 1819665484d8SDoug Ambrisko * 1820665484d8SDoug Ambrisko * Note that older firmwares ( < FW ver 30) didn't report information 1821665484d8SDoug Ambrisko * to calculate max_sectors_1. So the number ended up as zero always. 1822665484d8SDoug Ambrisko */ 1823665484d8SDoug Ambrisko tmp_sectors = 0; 1824665484d8SDoug Ambrisko max_sectors_1 = (1 << ctrl_info->stripe_sz_ops.min) * 1825665484d8SDoug Ambrisko ctrl_info->max_strips_per_io; 1826665484d8SDoug Ambrisko max_sectors_2 = ctrl_info->max_request_size; 1827665484d8SDoug Ambrisko tmp_sectors = min(max_sectors_1 , max_sectors_2); 1828*4799d485SKashyap D Desai sc->max_sectors_per_req = sc->max_num_sge * MRSAS_PAGE_SIZE / 512; 1829*4799d485SKashyap D Desai 1830*4799d485SKashyap D Desai if (tmp_sectors && (sc->max_sectors_per_req > tmp_sectors)) 1831*4799d485SKashyap D Desai sc->max_sectors_per_req = tmp_sectors; 1832*4799d485SKashyap D Desai 1833665484d8SDoug Ambrisko sc->disableOnlineCtrlReset = 1834665484d8SDoug Ambrisko ctrl_info->properties.OnOffProperties.disableOnlineCtrlReset; 1835665484d8SDoug Ambrisko sc->UnevenSpanSupport = 1836665484d8SDoug Ambrisko ctrl_info->adapterOperations2.supportUnevenSpans; 1837665484d8SDoug Ambrisko if(sc->UnevenSpanSupport) { 1838*4799d485SKashyap D Desai printf("FW supports: UnevenSpanSupport=%x\n\n", 1839665484d8SDoug Ambrisko sc->UnevenSpanSupport); 1840*4799d485SKashyap D Desai 1841665484d8SDoug Ambrisko if (MR_ValidateMapInfo(sc)) 1842665484d8SDoug Ambrisko sc->fast_path_io = 1; 1843665484d8SDoug Ambrisko else 1844665484d8SDoug Ambrisko sc->fast_path_io = 0; 1845665484d8SDoug Ambrisko } 1846665484d8SDoug Ambrisko 1847665484d8SDoug Ambrisko if (ctrl_info) 1848665484d8SDoug Ambrisko free(ctrl_info, M_MRSAS); 1849665484d8SDoug Ambrisko 1850665484d8SDoug Ambrisko return(0); 1851665484d8SDoug Ambrisko } 1852665484d8SDoug Ambrisko 1853665484d8SDoug Ambrisko /** 1854665484d8SDoug Ambrisko * mrsas_init_adapter: Initializes the adapter/controller 1855665484d8SDoug Ambrisko * input: Adapter soft state 1856665484d8SDoug Ambrisko * 1857665484d8SDoug Ambrisko * Prepares for the issuing of the IOC Init cmd to FW for initializing the 1858665484d8SDoug Ambrisko * ROC/controller. The FW register is read to determined the number of 1859665484d8SDoug Ambrisko * commands that is supported. All memory allocations for IO is based on 1860665484d8SDoug Ambrisko * max_cmd. Appropriate calculations are performed in this function. 1861665484d8SDoug Ambrisko */ 1862665484d8SDoug Ambrisko int mrsas_init_adapter(struct mrsas_softc *sc) 1863665484d8SDoug Ambrisko { 1864665484d8SDoug Ambrisko uint32_t status; 1865665484d8SDoug Ambrisko u_int32_t max_cmd; 1866665484d8SDoug Ambrisko int ret; 1867665484d8SDoug Ambrisko 1868665484d8SDoug Ambrisko /* Read FW status register */ 1869665484d8SDoug Ambrisko status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad)); 1870665484d8SDoug Ambrisko 1871665484d8SDoug Ambrisko /* Get operational params from status register */ 1872665484d8SDoug Ambrisko sc->max_fw_cmds = status & MRSAS_FWSTATE_MAXCMD_MASK; 1873665484d8SDoug Ambrisko 1874665484d8SDoug Ambrisko /* Decrement the max supported by 1, to correlate with FW */ 1875665484d8SDoug Ambrisko sc->max_fw_cmds = sc->max_fw_cmds-1; 1876665484d8SDoug Ambrisko max_cmd = sc->max_fw_cmds; 1877665484d8SDoug Ambrisko 1878665484d8SDoug Ambrisko /* Determine allocation size of command frames */ 1879665484d8SDoug Ambrisko sc->reply_q_depth = ((max_cmd *2 +1 +15)/16*16); 1880665484d8SDoug Ambrisko sc->request_alloc_sz = sizeof(MRSAS_REQUEST_DESCRIPTOR_UNION) * max_cmd; 1881665484d8SDoug Ambrisko sc->reply_alloc_sz = sizeof(MPI2_REPLY_DESCRIPTORS_UNION) * (sc->reply_q_depth); 1882665484d8SDoug Ambrisko sc->io_frames_alloc_sz = MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE + (MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE * (max_cmd + 1)); 1883665484d8SDoug Ambrisko sc->chain_frames_alloc_sz = 1024 * max_cmd; 1884665484d8SDoug Ambrisko sc->max_sge_in_main_msg = (MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE - 1885665484d8SDoug Ambrisko offsetof(MRSAS_RAID_SCSI_IO_REQUEST, SGL))/16; 1886665484d8SDoug Ambrisko 1887665484d8SDoug Ambrisko sc->max_sge_in_chain = MRSAS_MAX_SZ_CHAIN_FRAME / sizeof(MPI2_SGE_IO_UNION); 1888665484d8SDoug Ambrisko sc->max_num_sge = sc->max_sge_in_main_msg + sc->max_sge_in_chain - 2; 1889665484d8SDoug Ambrisko 1890665484d8SDoug Ambrisko /* Used for pass thru MFI frame (DCMD) */ 1891665484d8SDoug Ambrisko sc->chain_offset_mfi_pthru = offsetof(MRSAS_RAID_SCSI_IO_REQUEST, SGL)/16; 1892665484d8SDoug Ambrisko 1893665484d8SDoug Ambrisko sc->chain_offset_io_request = (MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE - 1894665484d8SDoug Ambrisko sizeof(MPI2_SGE_IO_UNION))/16; 1895665484d8SDoug Ambrisko 1896665484d8SDoug Ambrisko sc->last_reply_idx = 0; 1897665484d8SDoug Ambrisko 1898665484d8SDoug Ambrisko ret = mrsas_alloc_mem(sc); 1899665484d8SDoug Ambrisko if (ret != SUCCESS) 1900665484d8SDoug Ambrisko return(ret); 1901665484d8SDoug Ambrisko 1902665484d8SDoug Ambrisko ret = mrsas_alloc_mpt_cmds(sc); 1903665484d8SDoug Ambrisko if (ret != SUCCESS) 1904665484d8SDoug Ambrisko return(ret); 1905665484d8SDoug Ambrisko 1906665484d8SDoug Ambrisko ret = mrsas_ioc_init(sc); 1907665484d8SDoug Ambrisko if (ret != SUCCESS) 1908665484d8SDoug Ambrisko return(ret); 1909665484d8SDoug Ambrisko 1910665484d8SDoug Ambrisko 1911665484d8SDoug Ambrisko return(0); 1912665484d8SDoug Ambrisko } 1913665484d8SDoug Ambrisko 1914665484d8SDoug Ambrisko /** 1915665484d8SDoug Ambrisko * mrsas_alloc_ioc_cmd: Allocates memory for IOC Init command 1916665484d8SDoug Ambrisko * input: Adapter soft state 1917665484d8SDoug Ambrisko * 1918665484d8SDoug Ambrisko * Allocates for the IOC Init cmd to FW to initialize the ROC/controller. 1919665484d8SDoug Ambrisko */ 1920665484d8SDoug Ambrisko int mrsas_alloc_ioc_cmd(struct mrsas_softc *sc) 1921665484d8SDoug Ambrisko { 1922665484d8SDoug Ambrisko int ioc_init_size; 1923665484d8SDoug Ambrisko 1924665484d8SDoug Ambrisko /* Allocate IOC INIT command */ 1925665484d8SDoug Ambrisko ioc_init_size = 1024 + sizeof(MPI2_IOC_INIT_REQUEST); 1926665484d8SDoug Ambrisko if (bus_dma_tag_create( sc->mrsas_parent_tag, // parent 1927665484d8SDoug Ambrisko 1, 0, // algnmnt, boundary 1928665484d8SDoug Ambrisko BUS_SPACE_MAXADDR_32BIT,// lowaddr 1929665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, // highaddr 1930665484d8SDoug Ambrisko NULL, NULL, // filter, filterarg 1931665484d8SDoug Ambrisko ioc_init_size, // maxsize 1932665484d8SDoug Ambrisko 1, // msegments 1933665484d8SDoug Ambrisko ioc_init_size, // maxsegsize 1934665484d8SDoug Ambrisko BUS_DMA_ALLOCNOW, // flags 1935665484d8SDoug Ambrisko NULL, NULL, // lockfunc, lockarg 1936665484d8SDoug Ambrisko &sc->ioc_init_tag)) { 1937665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate ioc init tag\n"); 1938665484d8SDoug Ambrisko return (ENOMEM); 1939665484d8SDoug Ambrisko } 1940665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->ioc_init_tag, (void **)&sc->ioc_init_mem, 1941665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->ioc_init_dmamap)) { 1942665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate ioc init cmd mem\n"); 1943665484d8SDoug Ambrisko return (ENOMEM); 1944665484d8SDoug Ambrisko } 1945665484d8SDoug Ambrisko bzero(sc->ioc_init_mem, ioc_init_size); 1946665484d8SDoug Ambrisko if (bus_dmamap_load(sc->ioc_init_tag, sc->ioc_init_dmamap, 1947665484d8SDoug Ambrisko sc->ioc_init_mem, ioc_init_size, mrsas_addr_cb, 1948665484d8SDoug Ambrisko &sc->ioc_init_phys_mem, BUS_DMA_NOWAIT)) { 1949665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load ioc init cmd mem\n"); 1950665484d8SDoug Ambrisko return (ENOMEM); 1951665484d8SDoug Ambrisko } 1952665484d8SDoug Ambrisko 1953665484d8SDoug Ambrisko return (0); 1954665484d8SDoug Ambrisko } 1955665484d8SDoug Ambrisko 1956665484d8SDoug Ambrisko /** 1957665484d8SDoug Ambrisko * mrsas_free_ioc_cmd: Allocates memory for IOC Init command 1958665484d8SDoug Ambrisko * input: Adapter soft state 1959665484d8SDoug Ambrisko * 1960665484d8SDoug Ambrisko * Deallocates memory of the IOC Init cmd. 1961665484d8SDoug Ambrisko */ 1962665484d8SDoug Ambrisko void mrsas_free_ioc_cmd(struct mrsas_softc *sc) 1963665484d8SDoug Ambrisko { 1964665484d8SDoug Ambrisko if (sc->ioc_init_phys_mem) 1965665484d8SDoug Ambrisko bus_dmamap_unload(sc->ioc_init_tag, sc->ioc_init_dmamap); 1966665484d8SDoug Ambrisko if (sc->ioc_init_mem != NULL) 1967665484d8SDoug Ambrisko bus_dmamem_free(sc->ioc_init_tag, sc->ioc_init_mem, sc->ioc_init_dmamap); 1968665484d8SDoug Ambrisko if (sc->ioc_init_tag != NULL) 1969665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->ioc_init_tag); 1970665484d8SDoug Ambrisko } 1971665484d8SDoug Ambrisko 1972665484d8SDoug Ambrisko /** 1973665484d8SDoug Ambrisko * mrsas_ioc_init: Sends IOC Init command to FW 1974665484d8SDoug Ambrisko * input: Adapter soft state 1975665484d8SDoug Ambrisko * 1976665484d8SDoug Ambrisko * Issues the IOC Init cmd to FW to initialize the ROC/controller. 1977665484d8SDoug Ambrisko */ 1978665484d8SDoug Ambrisko int mrsas_ioc_init(struct mrsas_softc *sc) 1979665484d8SDoug Ambrisko { 1980665484d8SDoug Ambrisko struct mrsas_init_frame *init_frame; 1981665484d8SDoug Ambrisko pMpi2IOCInitRequest_t IOCInitMsg; 1982665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION req_desc; 1983665484d8SDoug Ambrisko u_int8_t max_wait = MRSAS_IOC_INIT_WAIT_TIME; 1984665484d8SDoug Ambrisko bus_addr_t phys_addr; 1985665484d8SDoug Ambrisko int i, retcode = 0; 1986665484d8SDoug Ambrisko 1987665484d8SDoug Ambrisko /* Allocate memory for the IOC INIT command */ 1988665484d8SDoug Ambrisko if (mrsas_alloc_ioc_cmd(sc)) { 1989665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate IOC command.\n"); 1990665484d8SDoug Ambrisko return(1); 1991665484d8SDoug Ambrisko } 1992665484d8SDoug Ambrisko 1993665484d8SDoug Ambrisko IOCInitMsg = (pMpi2IOCInitRequest_t)(((char *)sc->ioc_init_mem) +1024); 1994665484d8SDoug Ambrisko IOCInitMsg->Function = MPI2_FUNCTION_IOC_INIT; 1995665484d8SDoug Ambrisko IOCInitMsg->WhoInit = MPI2_WHOINIT_HOST_DRIVER; 1996665484d8SDoug Ambrisko IOCInitMsg->MsgVersion = MPI2_VERSION; 1997665484d8SDoug Ambrisko IOCInitMsg->HeaderVersion = MPI2_HEADER_VERSION; 1998665484d8SDoug Ambrisko IOCInitMsg->SystemRequestFrameSize = MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE / 4; 1999665484d8SDoug Ambrisko IOCInitMsg->ReplyDescriptorPostQueueDepth = sc->reply_q_depth; 2000665484d8SDoug Ambrisko IOCInitMsg->ReplyDescriptorPostQueueAddress = sc->reply_desc_phys_addr; 2001665484d8SDoug Ambrisko IOCInitMsg->SystemRequestFrameBaseAddress = sc->io_request_phys_addr; 2002665484d8SDoug Ambrisko 2003665484d8SDoug Ambrisko init_frame = (struct mrsas_init_frame *)sc->ioc_init_mem; 2004665484d8SDoug Ambrisko init_frame->cmd = MFI_CMD_INIT; 2005665484d8SDoug Ambrisko init_frame->cmd_status = 0xFF; 2006665484d8SDoug Ambrisko init_frame->flags |= MFI_FRAME_DONT_POST_IN_REPLY_QUEUE; 2007665484d8SDoug Ambrisko 2008665484d8SDoug Ambrisko if (sc->verbuf_mem) { 2009665484d8SDoug Ambrisko snprintf((char *)sc->verbuf_mem, strlen(MRSAS_VERSION)+2,"%s\n", 2010665484d8SDoug Ambrisko MRSAS_VERSION); 2011665484d8SDoug Ambrisko init_frame->driver_ver_lo = (bus_addr_t)sc->verbuf_phys_addr; 2012665484d8SDoug Ambrisko init_frame->driver_ver_hi = 0; 2013665484d8SDoug Ambrisko } 2014665484d8SDoug Ambrisko 2015*4799d485SKashyap D Desai init_frame->driver_operations.mfi_capabilities.support_max_255lds = 1; 2016665484d8SDoug Ambrisko phys_addr = (bus_addr_t)sc->ioc_init_phys_mem + 1024; 2017665484d8SDoug Ambrisko init_frame->queue_info_new_phys_addr_lo = phys_addr; 2018665484d8SDoug Ambrisko init_frame->data_xfer_len = sizeof(Mpi2IOCInitRequest_t); 2019665484d8SDoug Ambrisko 2020665484d8SDoug Ambrisko req_desc.addr.Words = (bus_addr_t)sc->ioc_init_phys_mem; 2021665484d8SDoug Ambrisko req_desc.MFAIo.RequestFlags = 2022665484d8SDoug Ambrisko (MRSAS_REQ_DESCRIPT_FLAGS_MFA << MRSAS_REQ_DESCRIPT_FLAGS_TYPE_SHIFT); 2023665484d8SDoug Ambrisko 2024665484d8SDoug Ambrisko mrsas_disable_intr(sc); 2025665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "Issuing IOC INIT command to FW.\n"); 2026665484d8SDoug Ambrisko //device_printf(sc->mrsas_dev, "Issuing IOC INIT command to FW.\n");del? 2027665484d8SDoug Ambrisko mrsas_fire_cmd(sc, req_desc.addr.u.low, req_desc.addr.u.high); 2028665484d8SDoug Ambrisko 2029665484d8SDoug Ambrisko /* 2030665484d8SDoug Ambrisko * Poll response timer to wait for Firmware response. While this 2031665484d8SDoug Ambrisko * timer with the DELAY call could block CPU, the time interval for 2032665484d8SDoug Ambrisko * this is only 1 millisecond. 2033665484d8SDoug Ambrisko */ 2034665484d8SDoug Ambrisko if (init_frame->cmd_status == 0xFF) { 2035665484d8SDoug Ambrisko for (i=0; i < (max_wait * 1000); i++){ 2036665484d8SDoug Ambrisko if (init_frame->cmd_status == 0xFF) 2037665484d8SDoug Ambrisko DELAY(1000); 2038665484d8SDoug Ambrisko else 2039665484d8SDoug Ambrisko break; 2040665484d8SDoug Ambrisko } 2041665484d8SDoug Ambrisko } 2042665484d8SDoug Ambrisko 2043665484d8SDoug Ambrisko if (init_frame->cmd_status == 0) 2044665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2045665484d8SDoug Ambrisko "IOC INIT response received from FW.\n"); 2046665484d8SDoug Ambrisko //device_printf(sc->mrsas_dev, "IOC INIT response received from FW.\n");del? 2047665484d8SDoug Ambrisko else 2048665484d8SDoug Ambrisko { 2049665484d8SDoug Ambrisko if (init_frame->cmd_status == 0xFF) 2050665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "IOC Init timed out after %d seconds.\n", max_wait); 2051665484d8SDoug Ambrisko else 2052665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "IOC Init failed, status = 0x%x\n", init_frame->cmd_status); 2053665484d8SDoug Ambrisko retcode = 1; 2054665484d8SDoug Ambrisko } 2055665484d8SDoug Ambrisko 2056665484d8SDoug Ambrisko mrsas_free_ioc_cmd(sc); 2057665484d8SDoug Ambrisko return (retcode); 2058665484d8SDoug Ambrisko } 2059665484d8SDoug Ambrisko 2060665484d8SDoug Ambrisko /** 2061665484d8SDoug Ambrisko * mrsas_alloc_mpt_cmds: Allocates the command packets 2062665484d8SDoug Ambrisko * input: Adapter instance soft state 2063665484d8SDoug Ambrisko * 2064665484d8SDoug Ambrisko * This function allocates the internal commands for IOs. Each command that is 2065665484d8SDoug Ambrisko * issued to FW is wrapped in a local data structure called mrsas_mpt_cmd. 2066665484d8SDoug Ambrisko * An array is allocated with mrsas_mpt_cmd context. The free commands are 2067665484d8SDoug Ambrisko * maintained in a linked list (cmd pool). SMID value range is from 1 to 2068665484d8SDoug Ambrisko * max_fw_cmds. 2069665484d8SDoug Ambrisko */ 2070665484d8SDoug Ambrisko int mrsas_alloc_mpt_cmds(struct mrsas_softc *sc) 2071665484d8SDoug Ambrisko { 2072665484d8SDoug Ambrisko int i, j; 2073665484d8SDoug Ambrisko u_int32_t max_cmd; 2074665484d8SDoug Ambrisko struct mrsas_mpt_cmd *cmd; 2075665484d8SDoug Ambrisko pMpi2ReplyDescriptorsUnion_t reply_desc; 2076665484d8SDoug Ambrisko u_int32_t offset, chain_offset, sense_offset; 2077665484d8SDoug Ambrisko bus_addr_t io_req_base_phys, chain_frame_base_phys, sense_base_phys; 2078665484d8SDoug Ambrisko u_int8_t *io_req_base, *chain_frame_base, *sense_base; 2079665484d8SDoug Ambrisko 2080665484d8SDoug Ambrisko max_cmd = sc->max_fw_cmds; 2081665484d8SDoug Ambrisko 2082665484d8SDoug Ambrisko sc->req_desc = malloc(sc->request_alloc_sz, M_MRSAS, M_NOWAIT); 2083665484d8SDoug Ambrisko if (!sc->req_desc) { 2084665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Out of memory, cannot alloc req desc\n"); 2085665484d8SDoug Ambrisko return(ENOMEM); 2086665484d8SDoug Ambrisko } 2087665484d8SDoug Ambrisko memset(sc->req_desc, 0, sc->request_alloc_sz); 2088665484d8SDoug Ambrisko 2089665484d8SDoug Ambrisko /* 2090665484d8SDoug Ambrisko * sc->mpt_cmd_list is an array of struct mrsas_mpt_cmd pointers. Allocate the 2091665484d8SDoug Ambrisko * dynamic array first and then allocate individual commands. 2092665484d8SDoug Ambrisko */ 2093665484d8SDoug Ambrisko sc->mpt_cmd_list = malloc(sizeof(struct mrsas_mpt_cmd*)*max_cmd, M_MRSAS, M_NOWAIT); 2094665484d8SDoug Ambrisko if (!sc->mpt_cmd_list) { 2095665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc memory for mpt_cmd_list.\n"); 2096665484d8SDoug Ambrisko return(ENOMEM); 2097665484d8SDoug Ambrisko } 2098665484d8SDoug Ambrisko memset(sc->mpt_cmd_list, 0, sizeof(struct mrsas_mpt_cmd *)*max_cmd); 2099665484d8SDoug Ambrisko for (i = 0; i < max_cmd; i++) { 2100665484d8SDoug Ambrisko sc->mpt_cmd_list[i] = malloc(sizeof(struct mrsas_mpt_cmd), 2101665484d8SDoug Ambrisko M_MRSAS, M_NOWAIT); 2102665484d8SDoug Ambrisko if (!sc->mpt_cmd_list[i]) { 2103665484d8SDoug Ambrisko for (j = 0; j < i; j++) 2104665484d8SDoug Ambrisko free(sc->mpt_cmd_list[j],M_MRSAS); 2105665484d8SDoug Ambrisko free(sc->mpt_cmd_list, M_MRSAS); 2106665484d8SDoug Ambrisko sc->mpt_cmd_list = NULL; 2107665484d8SDoug Ambrisko return(ENOMEM); 2108665484d8SDoug Ambrisko } 2109665484d8SDoug Ambrisko } 2110665484d8SDoug Ambrisko 2111665484d8SDoug Ambrisko io_req_base = (u_int8_t*)sc->io_request_mem + MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE; 2112665484d8SDoug Ambrisko io_req_base_phys = (bus_addr_t)sc->io_request_phys_addr + MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE; 2113665484d8SDoug Ambrisko chain_frame_base = (u_int8_t*)sc->chain_frame_mem; 2114665484d8SDoug Ambrisko chain_frame_base_phys = (bus_addr_t)sc->chain_frame_phys_addr; 2115665484d8SDoug Ambrisko sense_base = (u_int8_t*)sc->sense_mem; 2116665484d8SDoug Ambrisko sense_base_phys = (bus_addr_t)sc->sense_phys_addr; 2117665484d8SDoug Ambrisko for (i = 0; i < max_cmd; i++) { 2118665484d8SDoug Ambrisko cmd = sc->mpt_cmd_list[i]; 2119665484d8SDoug Ambrisko offset = MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE * i; 2120665484d8SDoug Ambrisko chain_offset = 1024 * i; 2121665484d8SDoug Ambrisko sense_offset = MRSAS_SENSE_LEN * i; 2122665484d8SDoug Ambrisko memset(cmd, 0, sizeof(struct mrsas_mpt_cmd)); 2123665484d8SDoug Ambrisko cmd->index = i + 1; 2124665484d8SDoug Ambrisko cmd->ccb_ptr = NULL; 2125665484d8SDoug Ambrisko callout_init(&cmd->cm_callout, 0); 2126665484d8SDoug Ambrisko cmd->sync_cmd_idx = (u_int32_t)MRSAS_ULONG_MAX; 2127665484d8SDoug Ambrisko cmd->sc = sc; 2128665484d8SDoug Ambrisko cmd->io_request = (MRSAS_RAID_SCSI_IO_REQUEST *) (io_req_base + offset); 2129665484d8SDoug Ambrisko memset(cmd->io_request, 0, sizeof(MRSAS_RAID_SCSI_IO_REQUEST)); 2130665484d8SDoug Ambrisko cmd->io_request_phys_addr = io_req_base_phys + offset; 2131665484d8SDoug Ambrisko cmd->chain_frame = (MPI2_SGE_IO_UNION *) (chain_frame_base + chain_offset); 2132665484d8SDoug Ambrisko cmd->chain_frame_phys_addr = chain_frame_base_phys + chain_offset; 2133665484d8SDoug Ambrisko cmd->sense = sense_base + sense_offset; 2134665484d8SDoug Ambrisko cmd->sense_phys_addr = sense_base_phys + sense_offset; 2135665484d8SDoug Ambrisko if (bus_dmamap_create(sc->data_tag, 0, &cmd->data_dmamap)) { 2136665484d8SDoug Ambrisko return(FAIL); 2137665484d8SDoug Ambrisko } 2138665484d8SDoug Ambrisko TAILQ_INSERT_TAIL(&(sc->mrsas_mpt_cmd_list_head), cmd, next); 2139665484d8SDoug Ambrisko } 2140665484d8SDoug Ambrisko 2141665484d8SDoug Ambrisko /* Initialize reply descriptor array to 0xFFFFFFFF */ 2142665484d8SDoug Ambrisko reply_desc = sc->reply_desc_mem; 2143665484d8SDoug Ambrisko for (i = 0; i < sc->reply_q_depth; i++, reply_desc++) { 2144665484d8SDoug Ambrisko reply_desc->Words = MRSAS_ULONG_MAX; 2145665484d8SDoug Ambrisko } 2146665484d8SDoug Ambrisko return(0); 2147665484d8SDoug Ambrisko } 2148665484d8SDoug Ambrisko 2149665484d8SDoug Ambrisko /** 2150665484d8SDoug Ambrisko * mrsas_fire_cmd: Sends command to FW 2151665484d8SDoug Ambrisko * input: Adapter soft state 2152665484d8SDoug Ambrisko * request descriptor address low 2153665484d8SDoug Ambrisko * request descriptor address high 2154665484d8SDoug Ambrisko * 2155665484d8SDoug Ambrisko * This functions fires the command to Firmware by writing to the 2156665484d8SDoug Ambrisko * inbound_low_queue_port and inbound_high_queue_port. 2157665484d8SDoug Ambrisko */ 2158665484d8SDoug Ambrisko void mrsas_fire_cmd(struct mrsas_softc *sc, u_int32_t req_desc_lo, 2159665484d8SDoug Ambrisko u_int32_t req_desc_hi) 2160665484d8SDoug Ambrisko { 2161665484d8SDoug Ambrisko mtx_lock(&sc->pci_lock); 2162665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, inbound_low_queue_port), 2163665484d8SDoug Ambrisko req_desc_lo); 2164665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, inbound_high_queue_port), 2165665484d8SDoug Ambrisko req_desc_hi); 2166665484d8SDoug Ambrisko mtx_unlock(&sc->pci_lock); 2167665484d8SDoug Ambrisko } 2168665484d8SDoug Ambrisko 2169665484d8SDoug Ambrisko /** 2170665484d8SDoug Ambrisko * mrsas_transition_to_ready: Move FW to Ready state 2171665484d8SDoug Ambrisko * input: Adapter instance soft state 2172665484d8SDoug Ambrisko * 2173665484d8SDoug Ambrisko * During the initialization, FW passes can potentially be in any one of 2174665484d8SDoug Ambrisko * several possible states. If the FW in operational, waiting-for-handshake 2175665484d8SDoug Ambrisko * states, driver must take steps to bring it to ready state. Otherwise, it 2176665484d8SDoug Ambrisko * has to wait for the ready state. 2177665484d8SDoug Ambrisko */ 2178665484d8SDoug Ambrisko int mrsas_transition_to_ready(struct mrsas_softc *sc, int ocr) 2179665484d8SDoug Ambrisko { 2180665484d8SDoug Ambrisko int i; 2181665484d8SDoug Ambrisko u_int8_t max_wait; 2182665484d8SDoug Ambrisko u_int32_t val, fw_state; 2183665484d8SDoug Ambrisko u_int32_t cur_state; 2184665484d8SDoug Ambrisko u_int32_t abs_state, curr_abs_state; 2185665484d8SDoug Ambrisko 2186665484d8SDoug Ambrisko val = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad)); 2187665484d8SDoug Ambrisko fw_state = val & MFI_STATE_MASK; 2188665484d8SDoug Ambrisko max_wait = MRSAS_RESET_WAIT_TIME; 2189665484d8SDoug Ambrisko 2190665484d8SDoug Ambrisko if (fw_state != MFI_STATE_READY) 2191665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Waiting for FW to come to ready state\n"); 2192665484d8SDoug Ambrisko 2193665484d8SDoug Ambrisko while (fw_state != MFI_STATE_READY) { 2194665484d8SDoug Ambrisko abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad)); 2195665484d8SDoug Ambrisko switch (fw_state) { 2196665484d8SDoug Ambrisko case MFI_STATE_FAULT: 2197665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "FW is in FAULT state!!\n"); 2198665484d8SDoug Ambrisko if (ocr) { 2199665484d8SDoug Ambrisko cur_state = MFI_STATE_FAULT; 2200665484d8SDoug Ambrisko break; 2201665484d8SDoug Ambrisko } 2202665484d8SDoug Ambrisko else 2203665484d8SDoug Ambrisko return -ENODEV; 2204665484d8SDoug Ambrisko case MFI_STATE_WAIT_HANDSHAKE: 2205665484d8SDoug Ambrisko /* Set the CLR bit in inbound doorbell */ 2206665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell), 2207665484d8SDoug Ambrisko MFI_INIT_CLEAR_HANDSHAKE|MFI_INIT_HOTPLUG); 2208665484d8SDoug Ambrisko cur_state = MFI_STATE_WAIT_HANDSHAKE; 2209665484d8SDoug Ambrisko break; 2210665484d8SDoug Ambrisko case MFI_STATE_BOOT_MESSAGE_PENDING: 2211665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell), 2212665484d8SDoug Ambrisko MFI_INIT_HOTPLUG); 2213665484d8SDoug Ambrisko cur_state = MFI_STATE_BOOT_MESSAGE_PENDING; 2214665484d8SDoug Ambrisko break; 2215665484d8SDoug Ambrisko case MFI_STATE_OPERATIONAL: 2216665484d8SDoug Ambrisko /* Bring it to READY state; assuming max wait 10 secs */ 2217665484d8SDoug Ambrisko mrsas_disable_intr(sc); 2218665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell), MFI_RESET_FLAGS); 2219665484d8SDoug Ambrisko for (i=0; i < max_wait * 1000; i++) { 2220665484d8SDoug Ambrisko if (mrsas_read_reg(sc, offsetof(mrsas_reg_set, doorbell)) & 1) 2221665484d8SDoug Ambrisko DELAY(1000); 2222665484d8SDoug Ambrisko else 2223665484d8SDoug Ambrisko break; 2224665484d8SDoug Ambrisko } 2225665484d8SDoug Ambrisko cur_state = MFI_STATE_OPERATIONAL; 2226665484d8SDoug Ambrisko break; 2227665484d8SDoug Ambrisko case MFI_STATE_UNDEFINED: 2228665484d8SDoug Ambrisko /* This state should not last for more than 2 seconds */ 2229665484d8SDoug Ambrisko cur_state = MFI_STATE_UNDEFINED; 2230665484d8SDoug Ambrisko break; 2231665484d8SDoug Ambrisko case MFI_STATE_BB_INIT: 2232665484d8SDoug Ambrisko cur_state = MFI_STATE_BB_INIT; 2233665484d8SDoug Ambrisko break; 2234665484d8SDoug Ambrisko case MFI_STATE_FW_INIT: 2235665484d8SDoug Ambrisko cur_state = MFI_STATE_FW_INIT; 2236665484d8SDoug Ambrisko break; 2237665484d8SDoug Ambrisko case MFI_STATE_FW_INIT_2: 2238665484d8SDoug Ambrisko cur_state = MFI_STATE_FW_INIT_2; 2239665484d8SDoug Ambrisko break; 2240665484d8SDoug Ambrisko case MFI_STATE_DEVICE_SCAN: 2241665484d8SDoug Ambrisko cur_state = MFI_STATE_DEVICE_SCAN; 2242665484d8SDoug Ambrisko break; 2243665484d8SDoug Ambrisko case MFI_STATE_FLUSH_CACHE: 2244665484d8SDoug Ambrisko cur_state = MFI_STATE_FLUSH_CACHE; 2245665484d8SDoug Ambrisko break; 2246665484d8SDoug Ambrisko default: 2247665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Unknown state 0x%x\n", fw_state); 2248665484d8SDoug Ambrisko return -ENODEV; 2249665484d8SDoug Ambrisko } 2250665484d8SDoug Ambrisko 2251665484d8SDoug Ambrisko /* 2252665484d8SDoug Ambrisko * The cur_state should not last for more than max_wait secs 2253665484d8SDoug Ambrisko */ 2254665484d8SDoug Ambrisko for (i = 0; i < (max_wait * 1000); i++) { 2255665484d8SDoug Ambrisko fw_state = (mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2256665484d8SDoug Ambrisko outbound_scratch_pad))& MFI_STATE_MASK); 2257665484d8SDoug Ambrisko curr_abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2258665484d8SDoug Ambrisko outbound_scratch_pad)); 2259665484d8SDoug Ambrisko if (abs_state == curr_abs_state) 2260665484d8SDoug Ambrisko DELAY(1000); 2261665484d8SDoug Ambrisko else 2262665484d8SDoug Ambrisko break; 2263665484d8SDoug Ambrisko } 2264665484d8SDoug Ambrisko 2265665484d8SDoug Ambrisko /* 2266665484d8SDoug Ambrisko * Return error if fw_state hasn't changed after max_wait 2267665484d8SDoug Ambrisko */ 2268665484d8SDoug Ambrisko if (curr_abs_state == abs_state) { 2269665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "FW state [%d] hasn't changed " 2270665484d8SDoug Ambrisko "in %d secs\n", fw_state, max_wait); 2271665484d8SDoug Ambrisko return -ENODEV; 2272665484d8SDoug Ambrisko } 2273665484d8SDoug Ambrisko } 2274665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "FW now in Ready state\n"); 2275665484d8SDoug Ambrisko //device_printf(sc->mrsas_dev, "FW now in Ready state\n");del? 2276665484d8SDoug Ambrisko return 0; 2277665484d8SDoug Ambrisko } 2278665484d8SDoug Ambrisko 2279665484d8SDoug Ambrisko /** 2280665484d8SDoug Ambrisko * mrsas_get_mfi_cmd: Get a cmd from free command pool 2281665484d8SDoug Ambrisko * input: Adapter soft state 2282665484d8SDoug Ambrisko * 2283665484d8SDoug Ambrisko * This function removes an MFI command from the command list. 2284665484d8SDoug Ambrisko */ 2285665484d8SDoug Ambrisko struct mrsas_mfi_cmd* mrsas_get_mfi_cmd(struct mrsas_softc *sc) 2286665484d8SDoug Ambrisko { 2287665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd = NULL; 2288665484d8SDoug Ambrisko 2289665484d8SDoug Ambrisko mtx_lock(&sc->mfi_cmd_pool_lock); 2290665484d8SDoug Ambrisko if (!TAILQ_EMPTY(&sc->mrsas_mfi_cmd_list_head)){ 2291665484d8SDoug Ambrisko cmd = TAILQ_FIRST(&sc->mrsas_mfi_cmd_list_head); 2292665484d8SDoug Ambrisko TAILQ_REMOVE(&sc->mrsas_mfi_cmd_list_head, cmd, next); 2293665484d8SDoug Ambrisko } 2294665484d8SDoug Ambrisko mtx_unlock(&sc->mfi_cmd_pool_lock); 2295665484d8SDoug Ambrisko 2296665484d8SDoug Ambrisko return cmd; 2297665484d8SDoug Ambrisko } 2298665484d8SDoug Ambrisko 2299665484d8SDoug Ambrisko /** 2300665484d8SDoug Ambrisko * mrsas_ocr_thread Thread to handle OCR/Kill Adapter. 2301665484d8SDoug Ambrisko * input: Adapter Context. 2302665484d8SDoug Ambrisko * 2303665484d8SDoug Ambrisko * This function will check FW status register and flag 2304665484d8SDoug Ambrisko * do_timeout_reset flag. It will do OCR/Kill adapter if 2305665484d8SDoug Ambrisko * FW is in fault state or IO timed out has trigger reset. 2306665484d8SDoug Ambrisko */ 2307665484d8SDoug Ambrisko static void 2308665484d8SDoug Ambrisko mrsas_ocr_thread(void *arg) 2309665484d8SDoug Ambrisko { 2310665484d8SDoug Ambrisko struct mrsas_softc *sc; 2311665484d8SDoug Ambrisko u_int32_t fw_status, fw_state; 2312665484d8SDoug Ambrisko 2313665484d8SDoug Ambrisko sc = (struct mrsas_softc *)arg; 2314665484d8SDoug Ambrisko 2315665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_TRACE, "%s\n", __func__); 2316665484d8SDoug Ambrisko 2317665484d8SDoug Ambrisko sc->ocr_thread_active = 1; 2318665484d8SDoug Ambrisko mtx_lock(&sc->sim_lock); 2319665484d8SDoug Ambrisko for (;;) { 2320665484d8SDoug Ambrisko /* Sleep for 1 second and check the queue status*/ 2321665484d8SDoug Ambrisko msleep(&sc->ocr_chan, &sc->sim_lock, PRIBIO, 2322665484d8SDoug Ambrisko "mrsas_ocr", sc->mrsas_fw_fault_check_delay * hz); 2323665484d8SDoug Ambrisko if (sc->remove_in_progress) { 2324665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2325665484d8SDoug Ambrisko "Exit due to shutdown from %s\n", __func__); 2326665484d8SDoug Ambrisko break; 2327665484d8SDoug Ambrisko } 2328665484d8SDoug Ambrisko fw_status = mrsas_read_reg(sc, 2329665484d8SDoug Ambrisko offsetof(mrsas_reg_set, outbound_scratch_pad)); 2330665484d8SDoug Ambrisko fw_state = fw_status & MFI_STATE_MASK; 2331665484d8SDoug Ambrisko if (fw_state == MFI_STATE_FAULT || sc->do_timedout_reset) { 2332665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "OCR started due to %s!\n", 2333665484d8SDoug Ambrisko sc->do_timedout_reset?"IO Timeout": 2334665484d8SDoug Ambrisko "FW fault detected"); 2335665484d8SDoug Ambrisko mtx_lock_spin(&sc->ioctl_lock); 2336665484d8SDoug Ambrisko sc->reset_in_progress = 1; 2337665484d8SDoug Ambrisko sc->reset_count++; 2338665484d8SDoug Ambrisko mtx_unlock_spin(&sc->ioctl_lock); 2339665484d8SDoug Ambrisko mrsas_xpt_freeze(sc); 2340665484d8SDoug Ambrisko mrsas_reset_ctrl(sc); 2341665484d8SDoug Ambrisko mrsas_xpt_release(sc); 2342665484d8SDoug Ambrisko sc->reset_in_progress = 0; 2343665484d8SDoug Ambrisko sc->do_timedout_reset = 0; 2344665484d8SDoug Ambrisko } 2345665484d8SDoug Ambrisko } 2346665484d8SDoug Ambrisko mtx_unlock(&sc->sim_lock); 2347665484d8SDoug Ambrisko sc->ocr_thread_active = 0; 2348665484d8SDoug Ambrisko mrsas_kproc_exit(0); 2349665484d8SDoug Ambrisko } 2350665484d8SDoug Ambrisko 2351665484d8SDoug Ambrisko /** 2352665484d8SDoug Ambrisko * mrsas_reset_reply_desc Reset Reply descriptor as part of OCR. 2353665484d8SDoug Ambrisko * input: Adapter Context. 2354665484d8SDoug Ambrisko * 2355665484d8SDoug Ambrisko * This function will clear reply descriptor so that post OCR 2356665484d8SDoug Ambrisko * driver and FW will lost old history. 2357665484d8SDoug Ambrisko */ 2358665484d8SDoug Ambrisko void mrsas_reset_reply_desc(struct mrsas_softc *sc) 2359665484d8SDoug Ambrisko { 2360665484d8SDoug Ambrisko int i; 2361665484d8SDoug Ambrisko pMpi2ReplyDescriptorsUnion_t reply_desc; 2362665484d8SDoug Ambrisko 2363665484d8SDoug Ambrisko sc->last_reply_idx = 0; 2364665484d8SDoug Ambrisko reply_desc = sc->reply_desc_mem; 2365665484d8SDoug Ambrisko for (i = 0; i < sc->reply_q_depth; i++, reply_desc++) { 2366665484d8SDoug Ambrisko reply_desc->Words = MRSAS_ULONG_MAX; 2367665484d8SDoug Ambrisko } 2368665484d8SDoug Ambrisko } 2369665484d8SDoug Ambrisko 2370665484d8SDoug Ambrisko /** 2371665484d8SDoug Ambrisko * mrsas_reset_ctrl Core function to OCR/Kill adapter. 2372665484d8SDoug Ambrisko * input: Adapter Context. 2373665484d8SDoug Ambrisko * 2374665484d8SDoug Ambrisko * This function will run from thread context so that it can sleep. 2375665484d8SDoug Ambrisko * 1. Do not handle OCR if FW is in HW critical error. 2376665484d8SDoug Ambrisko * 2. Wait for outstanding command to complete for 180 seconds. 2377665484d8SDoug Ambrisko * 3. If #2 does not find any outstanding command Controller is in working 2378665484d8SDoug Ambrisko * state, so skip OCR. 2379665484d8SDoug Ambrisko * Otherwise, do OCR/kill Adapter based on flag disableOnlineCtrlReset. 2380665484d8SDoug Ambrisko * 4. Start of the OCR, return all SCSI command back to CAM layer which has 2381665484d8SDoug Ambrisko * ccb_ptr. 2382665484d8SDoug Ambrisko * 5. Post OCR, Re-fire Managment command and move Controller to Operation 2383665484d8SDoug Ambrisko * state. 2384665484d8SDoug Ambrisko */ 2385665484d8SDoug Ambrisko int mrsas_reset_ctrl(struct mrsas_softc *sc) 2386665484d8SDoug Ambrisko { 2387665484d8SDoug Ambrisko int retval = SUCCESS, i, j, retry = 0; 2388665484d8SDoug Ambrisko u_int32_t host_diag, abs_state, status_reg, reset_adapter; 2389665484d8SDoug Ambrisko union ccb *ccb; 2390665484d8SDoug Ambrisko struct mrsas_mfi_cmd *mfi_cmd; 2391665484d8SDoug Ambrisko struct mrsas_mpt_cmd *mpt_cmd; 2392665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION *req_desc; 2393665484d8SDoug Ambrisko 2394665484d8SDoug Ambrisko if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) { 2395665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, 2396665484d8SDoug Ambrisko "mrsas: Hardware critical error, returning FAIL.\n"); 2397665484d8SDoug Ambrisko return FAIL; 2398665484d8SDoug Ambrisko } 2399665484d8SDoug Ambrisko 2400665484d8SDoug Ambrisko set_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags); 2401665484d8SDoug Ambrisko sc->adprecovery = MRSAS_ADPRESET_SM_INFAULT; 2402665484d8SDoug Ambrisko mrsas_disable_intr(sc); 2403665484d8SDoug Ambrisko DELAY(1000 * 1000); 2404665484d8SDoug Ambrisko 2405665484d8SDoug Ambrisko /* First try waiting for commands to complete */ 2406665484d8SDoug Ambrisko if (mrsas_wait_for_outstanding(sc)) { 2407665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2408665484d8SDoug Ambrisko "resetting adapter from %s.\n", 2409665484d8SDoug Ambrisko __func__); 2410665484d8SDoug Ambrisko /* Now return commands back to the CAM layer */ 2411665484d8SDoug Ambrisko for (i = 0 ; i < sc->max_fw_cmds; i++) { 2412665484d8SDoug Ambrisko mpt_cmd = sc->mpt_cmd_list[i]; 2413665484d8SDoug Ambrisko if (mpt_cmd->ccb_ptr) { 2414665484d8SDoug Ambrisko ccb = (union ccb *)(mpt_cmd->ccb_ptr); 2415665484d8SDoug Ambrisko ccb->ccb_h.status = CAM_SCSI_BUS_RESET; 2416665484d8SDoug Ambrisko mrsas_cmd_done(sc, mpt_cmd); 2417665484d8SDoug Ambrisko atomic_dec(&sc->fw_outstanding); 2418665484d8SDoug Ambrisko } 2419665484d8SDoug Ambrisko } 2420665484d8SDoug Ambrisko 2421665484d8SDoug Ambrisko status_reg = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2422665484d8SDoug Ambrisko outbound_scratch_pad)); 2423665484d8SDoug Ambrisko abs_state = status_reg & MFI_STATE_MASK; 2424665484d8SDoug Ambrisko reset_adapter = status_reg & MFI_RESET_ADAPTER; 2425665484d8SDoug Ambrisko if (sc->disableOnlineCtrlReset || 2426665484d8SDoug Ambrisko (abs_state == MFI_STATE_FAULT && !reset_adapter)) { 2427665484d8SDoug Ambrisko /* Reset not supported, kill adapter */ 2428665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR,"Reset not supported, killing adapter.\n"); 2429665484d8SDoug Ambrisko mrsas_kill_hba(sc); 2430665484d8SDoug Ambrisko sc->adprecovery = MRSAS_HW_CRITICAL_ERROR; 2431665484d8SDoug Ambrisko retval = FAIL; 2432665484d8SDoug Ambrisko goto out; 2433665484d8SDoug Ambrisko } 2434665484d8SDoug Ambrisko 2435665484d8SDoug Ambrisko /* Now try to reset the chip */ 2436665484d8SDoug Ambrisko for (i = 0; i < MRSAS_FUSION_MAX_RESET_TRIES; i++) { 2437665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 2438665484d8SDoug Ambrisko MPI2_WRSEQ_FLUSH_KEY_VALUE); 2439665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 2440665484d8SDoug Ambrisko MPI2_WRSEQ_1ST_KEY_VALUE); 2441665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 2442665484d8SDoug Ambrisko MPI2_WRSEQ_2ND_KEY_VALUE); 2443665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 2444665484d8SDoug Ambrisko MPI2_WRSEQ_3RD_KEY_VALUE); 2445665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 2446665484d8SDoug Ambrisko MPI2_WRSEQ_4TH_KEY_VALUE); 2447665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 2448665484d8SDoug Ambrisko MPI2_WRSEQ_5TH_KEY_VALUE); 2449665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 2450665484d8SDoug Ambrisko MPI2_WRSEQ_6TH_KEY_VALUE); 2451665484d8SDoug Ambrisko 2452665484d8SDoug Ambrisko /* Check that the diag write enable (DRWE) bit is on */ 2453665484d8SDoug Ambrisko host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2454665484d8SDoug Ambrisko fusion_host_diag)); 2455665484d8SDoug Ambrisko retry = 0; 2456665484d8SDoug Ambrisko while (!(host_diag & HOST_DIAG_WRITE_ENABLE)) { 2457665484d8SDoug Ambrisko DELAY(100 * 1000); 2458665484d8SDoug Ambrisko host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2459665484d8SDoug Ambrisko fusion_host_diag)); 2460665484d8SDoug Ambrisko if (retry++ == 100) { 2461665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2462665484d8SDoug Ambrisko "Host diag unlock failed!\n"); 2463665484d8SDoug Ambrisko break; 2464665484d8SDoug Ambrisko } 2465665484d8SDoug Ambrisko } 2466665484d8SDoug Ambrisko if (!(host_diag & HOST_DIAG_WRITE_ENABLE)) 2467665484d8SDoug Ambrisko continue; 2468665484d8SDoug Ambrisko 2469665484d8SDoug Ambrisko /* Send chip reset command */ 2470665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_host_diag), 2471665484d8SDoug Ambrisko host_diag | HOST_DIAG_RESET_ADAPTER); 2472665484d8SDoug Ambrisko DELAY(3000 * 1000); 2473665484d8SDoug Ambrisko 2474665484d8SDoug Ambrisko /* Make sure reset adapter bit is cleared */ 2475665484d8SDoug Ambrisko host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2476665484d8SDoug Ambrisko fusion_host_diag)); 2477665484d8SDoug Ambrisko retry = 0; 2478665484d8SDoug Ambrisko while (host_diag & HOST_DIAG_RESET_ADAPTER) { 2479665484d8SDoug Ambrisko DELAY(100 * 1000); 2480665484d8SDoug Ambrisko host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2481665484d8SDoug Ambrisko fusion_host_diag)); 2482665484d8SDoug Ambrisko if (retry++ == 1000) { 2483665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2484665484d8SDoug Ambrisko "Diag reset adapter never cleared!\n"); 2485665484d8SDoug Ambrisko break; 2486665484d8SDoug Ambrisko } 2487665484d8SDoug Ambrisko } 2488665484d8SDoug Ambrisko if (host_diag & HOST_DIAG_RESET_ADAPTER) 2489665484d8SDoug Ambrisko continue; 2490665484d8SDoug Ambrisko 2491665484d8SDoug Ambrisko abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2492665484d8SDoug Ambrisko outbound_scratch_pad)) & MFI_STATE_MASK; 2493665484d8SDoug Ambrisko retry = 0; 2494665484d8SDoug Ambrisko 2495665484d8SDoug Ambrisko while ((abs_state <= MFI_STATE_FW_INIT) && (retry++ < 1000)) { 2496665484d8SDoug Ambrisko DELAY(100 * 1000); 2497665484d8SDoug Ambrisko abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2498665484d8SDoug Ambrisko outbound_scratch_pad)) & MFI_STATE_MASK; 2499665484d8SDoug Ambrisko } 2500665484d8SDoug Ambrisko if (abs_state <= MFI_STATE_FW_INIT) { 2501665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "firmware state < MFI_STATE_FW_INIT," 2502665484d8SDoug Ambrisko " state = 0x%x\n", abs_state); 2503665484d8SDoug Ambrisko continue; 2504665484d8SDoug Ambrisko } 2505665484d8SDoug Ambrisko 2506665484d8SDoug Ambrisko /* Wait for FW to become ready */ 2507665484d8SDoug Ambrisko if (mrsas_transition_to_ready(sc, 1)) { 2508665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2509665484d8SDoug Ambrisko "mrsas: Failed to transition controller to ready.\n"); 2510665484d8SDoug Ambrisko continue; 2511665484d8SDoug Ambrisko } 2512665484d8SDoug Ambrisko 2513665484d8SDoug Ambrisko mrsas_reset_reply_desc(sc); 2514665484d8SDoug Ambrisko if (mrsas_ioc_init(sc)) { 2515665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "mrsas_ioc_init() failed!\n"); 2516665484d8SDoug Ambrisko continue; 2517665484d8SDoug Ambrisko } 2518665484d8SDoug Ambrisko 2519665484d8SDoug Ambrisko clear_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags); 2520665484d8SDoug Ambrisko mrsas_enable_intr(sc); 2521665484d8SDoug Ambrisko sc->adprecovery = MRSAS_HBA_OPERATIONAL; 2522665484d8SDoug Ambrisko 2523665484d8SDoug Ambrisko /* Re-fire management commands */ 2524665484d8SDoug Ambrisko for (j = 0 ; j < sc->max_fw_cmds; j++) { 2525665484d8SDoug Ambrisko mpt_cmd = sc->mpt_cmd_list[j]; 2526665484d8SDoug Ambrisko if (mpt_cmd->sync_cmd_idx != (u_int32_t)MRSAS_ULONG_MAX) { 2527665484d8SDoug Ambrisko mfi_cmd = sc->mfi_cmd_list[mpt_cmd->sync_cmd_idx]; 2528665484d8SDoug Ambrisko if (mfi_cmd->frame->dcmd.opcode == 2529665484d8SDoug Ambrisko MR_DCMD_LD_MAP_GET_INFO) { 2530665484d8SDoug Ambrisko mrsas_release_mfi_cmd(mfi_cmd); 2531665484d8SDoug Ambrisko mrsas_release_mpt_cmd(mpt_cmd); 2532665484d8SDoug Ambrisko } else { 2533665484d8SDoug Ambrisko req_desc = mrsas_get_request_desc(sc, 2534665484d8SDoug Ambrisko mfi_cmd->cmd_id.context.smid - 1); 2535665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2536665484d8SDoug Ambrisko "Re-fire command DCMD opcode 0x%x index %d\n ", 2537665484d8SDoug Ambrisko mfi_cmd->frame->dcmd.opcode, j); 2538665484d8SDoug Ambrisko if (!req_desc) 2539665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, 2540665484d8SDoug Ambrisko "Cannot build MPT cmd.\n"); 2541665484d8SDoug Ambrisko else 2542665484d8SDoug Ambrisko mrsas_fire_cmd(sc, req_desc->addr.u.low, 2543665484d8SDoug Ambrisko req_desc->addr.u.high); 2544665484d8SDoug Ambrisko } 2545665484d8SDoug Ambrisko } 2546665484d8SDoug Ambrisko } 2547665484d8SDoug Ambrisko 2548665484d8SDoug Ambrisko /* Reset load balance info */ 2549665484d8SDoug Ambrisko memset(sc->load_balance_info, 0, 2550*4799d485SKashyap D Desai sizeof(LD_LOAD_BALANCE_INFO) * MAX_LOGICAL_DRIVES_EXT); 2551665484d8SDoug Ambrisko 2552665484d8SDoug Ambrisko if (!mrsas_get_map_info(sc)) 2553665484d8SDoug Ambrisko mrsas_sync_map_info(sc); 2554665484d8SDoug Ambrisko 2555665484d8SDoug Ambrisko /* Adapter reset completed successfully */ 2556665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Reset successful\n"); 2557665484d8SDoug Ambrisko retval = SUCCESS; 2558665484d8SDoug Ambrisko goto out; 2559665484d8SDoug Ambrisko } 2560665484d8SDoug Ambrisko /* Reset failed, kill the adapter */ 2561665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Reset failed, killing adapter.\n"); 2562665484d8SDoug Ambrisko mrsas_kill_hba(sc); 2563665484d8SDoug Ambrisko retval = FAIL; 2564665484d8SDoug Ambrisko } else { 2565665484d8SDoug Ambrisko clear_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags); 2566665484d8SDoug Ambrisko mrsas_enable_intr(sc); 2567665484d8SDoug Ambrisko sc->adprecovery = MRSAS_HBA_OPERATIONAL; 2568665484d8SDoug Ambrisko } 2569665484d8SDoug Ambrisko out: 2570665484d8SDoug Ambrisko clear_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags); 2571665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2572665484d8SDoug Ambrisko "Reset Exit with %d.\n", retval); 2573665484d8SDoug Ambrisko return retval; 2574665484d8SDoug Ambrisko } 2575665484d8SDoug Ambrisko 2576665484d8SDoug Ambrisko /** 2577665484d8SDoug Ambrisko * mrsas_kill_hba Kill HBA when OCR is not supported. 2578665484d8SDoug Ambrisko * input: Adapter Context. 2579665484d8SDoug Ambrisko * 2580665484d8SDoug Ambrisko * This function will kill HBA when OCR is not supported. 2581665484d8SDoug Ambrisko */ 2582665484d8SDoug Ambrisko void mrsas_kill_hba (struct mrsas_softc *sc) 2583665484d8SDoug Ambrisko { 2584665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "%s\n", __func__); 2585665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell), 2586665484d8SDoug Ambrisko MFI_STOP_ADP); 2587665484d8SDoug Ambrisko /* Flush */ 2588665484d8SDoug Ambrisko mrsas_read_reg(sc, offsetof(mrsas_reg_set, doorbell)); 2589665484d8SDoug Ambrisko } 2590665484d8SDoug Ambrisko 2591665484d8SDoug Ambrisko /** 2592665484d8SDoug Ambrisko * mrsas_wait_for_outstanding Wait for outstanding commands 2593665484d8SDoug Ambrisko * input: Adapter Context. 2594665484d8SDoug Ambrisko * 2595665484d8SDoug Ambrisko * This function will wait for 180 seconds for outstanding 2596665484d8SDoug Ambrisko * commands to be completed. 2597665484d8SDoug Ambrisko */ 2598665484d8SDoug Ambrisko int mrsas_wait_for_outstanding(struct mrsas_softc *sc) 2599665484d8SDoug Ambrisko { 2600665484d8SDoug Ambrisko int i, outstanding, retval = 0; 2601665484d8SDoug Ambrisko u_int32_t fw_state; 2602665484d8SDoug Ambrisko 2603665484d8SDoug Ambrisko for (i = 0; i < MRSAS_RESET_WAIT_TIME; i++) { 2604665484d8SDoug Ambrisko if (sc->remove_in_progress) { 2605665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2606665484d8SDoug Ambrisko "Driver remove or shutdown called.\n"); 2607665484d8SDoug Ambrisko retval = 1; 2608665484d8SDoug Ambrisko goto out; 2609665484d8SDoug Ambrisko } 2610665484d8SDoug Ambrisko /* Check if firmware is in fault state */ 2611665484d8SDoug Ambrisko fw_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2612665484d8SDoug Ambrisko outbound_scratch_pad)) & MFI_STATE_MASK; 2613665484d8SDoug Ambrisko if (fw_state == MFI_STATE_FAULT) { 2614665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2615665484d8SDoug Ambrisko "Found FW in FAULT state, will reset adapter.\n"); 2616665484d8SDoug Ambrisko retval = 1; 2617665484d8SDoug Ambrisko goto out; 2618665484d8SDoug Ambrisko } 2619665484d8SDoug Ambrisko outstanding = atomic_read(&sc->fw_outstanding); 2620665484d8SDoug Ambrisko if (!outstanding) 2621665484d8SDoug Ambrisko goto out; 2622665484d8SDoug Ambrisko 2623665484d8SDoug Ambrisko if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) { 2624665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "[%2d]waiting for %d " 2625665484d8SDoug Ambrisko "commands to complete\n",i,outstanding); 2626665484d8SDoug Ambrisko mrsas_complete_cmd(sc); 2627665484d8SDoug Ambrisko } 2628665484d8SDoug Ambrisko DELAY(1000 * 1000); 2629665484d8SDoug Ambrisko } 2630665484d8SDoug Ambrisko 2631665484d8SDoug Ambrisko if (atomic_read(&sc->fw_outstanding)) { 2632665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2633665484d8SDoug Ambrisko " pending commands remain after waiting," 2634665484d8SDoug Ambrisko " will reset adapter.\n"); 2635665484d8SDoug Ambrisko retval = 1; 2636665484d8SDoug Ambrisko } 2637665484d8SDoug Ambrisko out: 2638665484d8SDoug Ambrisko return retval; 2639665484d8SDoug Ambrisko } 2640665484d8SDoug Ambrisko 2641665484d8SDoug Ambrisko /** 2642665484d8SDoug Ambrisko * mrsas_release_mfi_cmd: Return a cmd to free command pool 2643665484d8SDoug Ambrisko * input: Command packet for return to free cmd pool 2644665484d8SDoug Ambrisko * 2645665484d8SDoug Ambrisko * This function returns the MFI command to the command list. 2646665484d8SDoug Ambrisko */ 2647665484d8SDoug Ambrisko void mrsas_release_mfi_cmd(struct mrsas_mfi_cmd *cmd) 2648665484d8SDoug Ambrisko { 2649665484d8SDoug Ambrisko struct mrsas_softc *sc = cmd->sc; 2650665484d8SDoug Ambrisko 2651665484d8SDoug Ambrisko mtx_lock(&sc->mfi_cmd_pool_lock); 2652665484d8SDoug Ambrisko cmd->ccb_ptr = NULL; 2653665484d8SDoug Ambrisko cmd->cmd_id.frame_count = 0; 2654665484d8SDoug Ambrisko TAILQ_INSERT_TAIL(&(sc->mrsas_mfi_cmd_list_head), cmd, next); 2655665484d8SDoug Ambrisko mtx_unlock(&sc->mfi_cmd_pool_lock); 2656665484d8SDoug Ambrisko 2657665484d8SDoug Ambrisko return; 2658665484d8SDoug Ambrisko } 2659665484d8SDoug Ambrisko 2660665484d8SDoug Ambrisko /** 2661665484d8SDoug Ambrisko * mrsas_get_controller_info - Returns FW's controller structure 2662665484d8SDoug Ambrisko * input: Adapter soft state 2663665484d8SDoug Ambrisko * Controller information structure 2664665484d8SDoug Ambrisko * 2665665484d8SDoug Ambrisko * Issues an internal command (DCMD) to get the FW's controller structure. 2666665484d8SDoug Ambrisko * This information is mainly used to find out the maximum IO transfer per 2667665484d8SDoug Ambrisko * command supported by the FW. 2668665484d8SDoug Ambrisko */ 2669665484d8SDoug Ambrisko static int mrsas_get_ctrl_info(struct mrsas_softc *sc, 2670665484d8SDoug Ambrisko struct mrsas_ctrl_info *ctrl_info) 2671665484d8SDoug Ambrisko { 2672665484d8SDoug Ambrisko int retcode = 0; 2673665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 2674665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 2675665484d8SDoug Ambrisko 2676665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 2677665484d8SDoug Ambrisko 2678665484d8SDoug Ambrisko if (!cmd) { 2679665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Failed to get a free cmd\n"); 2680665484d8SDoug Ambrisko return -ENOMEM; 2681665484d8SDoug Ambrisko } 2682665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 2683665484d8SDoug Ambrisko 2684665484d8SDoug Ambrisko if (mrsas_alloc_ctlr_info_cmd(sc) != SUCCESS) { 2685665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate get ctlr info cmd\n"); 2686665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 2687665484d8SDoug Ambrisko return -ENOMEM; 2688665484d8SDoug Ambrisko } 2689665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 2690665484d8SDoug Ambrisko 2691665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 2692665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 2693665484d8SDoug Ambrisko dcmd->sge_count = 1; 2694665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 2695665484d8SDoug Ambrisko dcmd->timeout = 0; 2696665484d8SDoug Ambrisko dcmd->pad_0 = 0; 2697665484d8SDoug Ambrisko dcmd->data_xfer_len = sizeof(struct mrsas_ctrl_info); 2698665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_CTRL_GET_INFO; 2699665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = sc->ctlr_info_phys_addr; 2700665484d8SDoug Ambrisko dcmd->sgl.sge32[0].length = sizeof(struct mrsas_ctrl_info); 2701665484d8SDoug Ambrisko 2702665484d8SDoug Ambrisko if (!mrsas_issue_polled(sc, cmd)) 2703665484d8SDoug Ambrisko memcpy(ctrl_info, sc->ctlr_info_mem, sizeof(struct mrsas_ctrl_info)); 2704665484d8SDoug Ambrisko else 2705665484d8SDoug Ambrisko retcode = 1; 2706665484d8SDoug Ambrisko 2707665484d8SDoug Ambrisko mrsas_free_ctlr_info_cmd(sc); 2708665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 2709665484d8SDoug Ambrisko return(retcode); 2710665484d8SDoug Ambrisko } 2711665484d8SDoug Ambrisko 2712665484d8SDoug Ambrisko /** 2713665484d8SDoug Ambrisko * mrsas_alloc_ctlr_info_cmd: Allocates memory for controller info command 2714665484d8SDoug Ambrisko * input: Adapter soft state 2715665484d8SDoug Ambrisko * 2716665484d8SDoug Ambrisko * Allocates DMAable memory for the controller info internal command. 2717665484d8SDoug Ambrisko */ 2718665484d8SDoug Ambrisko int mrsas_alloc_ctlr_info_cmd(struct mrsas_softc *sc) 2719665484d8SDoug Ambrisko { 2720665484d8SDoug Ambrisko int ctlr_info_size; 2721665484d8SDoug Ambrisko 2722665484d8SDoug Ambrisko /* Allocate get controller info command */ 2723665484d8SDoug Ambrisko ctlr_info_size = sizeof(struct mrsas_ctrl_info); 2724665484d8SDoug Ambrisko if (bus_dma_tag_create( sc->mrsas_parent_tag, // parent 2725665484d8SDoug Ambrisko 1, 0, // algnmnt, boundary 2726665484d8SDoug Ambrisko BUS_SPACE_MAXADDR_32BIT,// lowaddr 2727665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, // highaddr 2728665484d8SDoug Ambrisko NULL, NULL, // filter, filterarg 2729665484d8SDoug Ambrisko ctlr_info_size, // maxsize 2730665484d8SDoug Ambrisko 1, // msegments 2731665484d8SDoug Ambrisko ctlr_info_size, // maxsegsize 2732665484d8SDoug Ambrisko BUS_DMA_ALLOCNOW, // flags 2733665484d8SDoug Ambrisko NULL, NULL, // lockfunc, lockarg 2734665484d8SDoug Ambrisko &sc->ctlr_info_tag)) { 2735665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate ctlr info tag\n"); 2736665484d8SDoug Ambrisko return (ENOMEM); 2737665484d8SDoug Ambrisko } 2738665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->ctlr_info_tag, (void **)&sc->ctlr_info_mem, 2739665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->ctlr_info_dmamap)) { 2740665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate ctlr info cmd mem\n"); 2741665484d8SDoug Ambrisko return (ENOMEM); 2742665484d8SDoug Ambrisko } 2743665484d8SDoug Ambrisko if (bus_dmamap_load(sc->ctlr_info_tag, sc->ctlr_info_dmamap, 2744665484d8SDoug Ambrisko sc->ctlr_info_mem, ctlr_info_size, mrsas_addr_cb, 2745665484d8SDoug Ambrisko &sc->ctlr_info_phys_addr, BUS_DMA_NOWAIT)) { 2746665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load ctlr info cmd mem\n"); 2747665484d8SDoug Ambrisko return (ENOMEM); 2748665484d8SDoug Ambrisko } 2749665484d8SDoug Ambrisko 2750665484d8SDoug Ambrisko memset(sc->ctlr_info_mem, 0, ctlr_info_size); 2751665484d8SDoug Ambrisko return (0); 2752665484d8SDoug Ambrisko } 2753665484d8SDoug Ambrisko 2754665484d8SDoug Ambrisko /** 2755665484d8SDoug Ambrisko * mrsas_free_ctlr_info_cmd: Free memory for controller info command 2756665484d8SDoug Ambrisko * input: Adapter soft state 2757665484d8SDoug Ambrisko * 2758665484d8SDoug Ambrisko * Deallocates memory of the get controller info cmd. 2759665484d8SDoug Ambrisko */ 2760665484d8SDoug Ambrisko void mrsas_free_ctlr_info_cmd(struct mrsas_softc *sc) 2761665484d8SDoug Ambrisko { 2762665484d8SDoug Ambrisko if (sc->ctlr_info_phys_addr) 2763665484d8SDoug Ambrisko bus_dmamap_unload(sc->ctlr_info_tag, sc->ctlr_info_dmamap); 2764665484d8SDoug Ambrisko if (sc->ctlr_info_mem != NULL) 2765665484d8SDoug Ambrisko bus_dmamem_free(sc->ctlr_info_tag, sc->ctlr_info_mem, sc->ctlr_info_dmamap); 2766665484d8SDoug Ambrisko if (sc->ctlr_info_tag != NULL) 2767665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->ctlr_info_tag); 2768665484d8SDoug Ambrisko } 2769665484d8SDoug Ambrisko 2770665484d8SDoug Ambrisko /** 2771665484d8SDoug Ambrisko * mrsas_issue_polled: Issues a polling command 2772665484d8SDoug Ambrisko * inputs: Adapter soft state 2773665484d8SDoug Ambrisko * Command packet to be issued 2774665484d8SDoug Ambrisko * 2775665484d8SDoug Ambrisko * This function is for posting of internal commands to Firmware. MFI 2776665484d8SDoug Ambrisko * requires the cmd_status to be set to 0xFF before posting. The maximun 2777665484d8SDoug Ambrisko * wait time of the poll response timer is 180 seconds. 2778665484d8SDoug Ambrisko */ 2779665484d8SDoug Ambrisko int mrsas_issue_polled(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 2780665484d8SDoug Ambrisko { 2781665484d8SDoug Ambrisko struct mrsas_header *frame_hdr = &cmd->frame->hdr; 2782665484d8SDoug Ambrisko u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME; 2783665484d8SDoug Ambrisko int i, retcode = 0; 2784665484d8SDoug Ambrisko 2785665484d8SDoug Ambrisko frame_hdr->cmd_status = 0xFF; 2786665484d8SDoug Ambrisko frame_hdr->flags |= MFI_FRAME_DONT_POST_IN_REPLY_QUEUE; 2787665484d8SDoug Ambrisko 2788665484d8SDoug Ambrisko /* Issue the frame using inbound queue port */ 2789665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)) { 2790665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot issue DCMD internal command.\n"); 2791665484d8SDoug Ambrisko return(1); 2792665484d8SDoug Ambrisko } 2793665484d8SDoug Ambrisko 2794665484d8SDoug Ambrisko /* 2795665484d8SDoug Ambrisko * Poll response timer to wait for Firmware response. While this 2796665484d8SDoug Ambrisko * timer with the DELAY call could block CPU, the time interval for 2797665484d8SDoug Ambrisko * this is only 1 millisecond. 2798665484d8SDoug Ambrisko */ 2799665484d8SDoug Ambrisko if (frame_hdr->cmd_status == 0xFF) { 2800665484d8SDoug Ambrisko for (i=0; i < (max_wait * 1000); i++){ 2801665484d8SDoug Ambrisko if (frame_hdr->cmd_status == 0xFF) 2802665484d8SDoug Ambrisko DELAY(1000); 2803665484d8SDoug Ambrisko else 2804665484d8SDoug Ambrisko break; 2805665484d8SDoug Ambrisko } 2806665484d8SDoug Ambrisko } 2807665484d8SDoug Ambrisko if (frame_hdr->cmd_status != 0) 2808665484d8SDoug Ambrisko { 2809665484d8SDoug Ambrisko if (frame_hdr->cmd_status == 0xFF) 2810665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "DCMD timed out after %d seconds.\n", max_wait); 2811665484d8SDoug Ambrisko else 2812665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "DCMD failed, status = 0x%x\n", frame_hdr->cmd_status); 2813665484d8SDoug Ambrisko retcode = 1; 2814665484d8SDoug Ambrisko } 2815665484d8SDoug Ambrisko return(retcode); 2816665484d8SDoug Ambrisko } 2817665484d8SDoug Ambrisko 2818665484d8SDoug Ambrisko /** 2819665484d8SDoug Ambrisko * mrsas_issue_dcmd - Issues a MFI Pass thru cmd 2820665484d8SDoug Ambrisko * input: Adapter soft state 2821665484d8SDoug Ambrisko * mfi cmd pointer 2822665484d8SDoug Ambrisko * 2823665484d8SDoug Ambrisko * This function is called by mrsas_issued_blocked_cmd() and 2824665484d8SDoug Ambrisko * mrsas_issued_polled(), to build the MPT command and then fire the 2825665484d8SDoug Ambrisko * command to Firmware. 2826665484d8SDoug Ambrisko */ 2827665484d8SDoug Ambrisko int 2828665484d8SDoug Ambrisko mrsas_issue_dcmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 2829665484d8SDoug Ambrisko { 2830665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION *req_desc; 2831665484d8SDoug Ambrisko 2832665484d8SDoug Ambrisko req_desc = mrsas_build_mpt_cmd(sc, cmd); 2833665484d8SDoug Ambrisko if (!req_desc) { 2834665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot build MPT cmd.\n"); 2835665484d8SDoug Ambrisko return(1); 2836665484d8SDoug Ambrisko } 2837665484d8SDoug Ambrisko 2838665484d8SDoug Ambrisko mrsas_fire_cmd(sc, req_desc->addr.u.low, req_desc->addr.u.high); 2839665484d8SDoug Ambrisko 2840665484d8SDoug Ambrisko return(0); 2841665484d8SDoug Ambrisko } 2842665484d8SDoug Ambrisko 2843665484d8SDoug Ambrisko /** 2844665484d8SDoug Ambrisko * mrsas_build_mpt_cmd - Calls helper function to build Passthru cmd 2845665484d8SDoug Ambrisko * input: Adapter soft state 2846665484d8SDoug Ambrisko * mfi cmd to build 2847665484d8SDoug Ambrisko * 2848665484d8SDoug Ambrisko * This function is called by mrsas_issue_cmd() to build the MPT-MFI 2849665484d8SDoug Ambrisko * passthru command and prepares the MPT command to send to Firmware. 2850665484d8SDoug Ambrisko */ 2851665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION * 2852665484d8SDoug Ambrisko mrsas_build_mpt_cmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 2853665484d8SDoug Ambrisko { 2854665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION *req_desc; 2855665484d8SDoug Ambrisko u_int16_t index; 2856665484d8SDoug Ambrisko 2857665484d8SDoug Ambrisko if (mrsas_build_mptmfi_passthru(sc, cmd)) { 2858665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot build MPT-MFI passthru cmd.\n"); 2859665484d8SDoug Ambrisko return NULL; 2860665484d8SDoug Ambrisko } 2861665484d8SDoug Ambrisko 2862665484d8SDoug Ambrisko index = cmd->cmd_id.context.smid; 2863665484d8SDoug Ambrisko 2864665484d8SDoug Ambrisko req_desc = mrsas_get_request_desc(sc, index-1); 2865665484d8SDoug Ambrisko if(!req_desc) 2866665484d8SDoug Ambrisko return NULL; 2867665484d8SDoug Ambrisko 2868665484d8SDoug Ambrisko req_desc->addr.Words = 0; 2869665484d8SDoug Ambrisko req_desc->SCSIIO.RequestFlags = (MPI2_REQ_DESCRIPT_FLAGS_SCSI_IO << MRSAS_REQ_DESCRIPT_FLAGS_TYPE_SHIFT); 2870665484d8SDoug Ambrisko 2871665484d8SDoug Ambrisko req_desc->SCSIIO.SMID = index; 2872665484d8SDoug Ambrisko 2873665484d8SDoug Ambrisko return(req_desc); 2874665484d8SDoug Ambrisko } 2875665484d8SDoug Ambrisko 2876665484d8SDoug Ambrisko /** 2877665484d8SDoug Ambrisko * mrsas_build_mptmfi_passthru - Builds a MPT MFI Passthru command 2878665484d8SDoug Ambrisko * input: Adapter soft state 2879665484d8SDoug Ambrisko * mfi cmd pointer 2880665484d8SDoug Ambrisko * 2881665484d8SDoug Ambrisko * The MPT command and the io_request are setup as a passthru command. 2882665484d8SDoug Ambrisko * The SGE chain address is set to frame_phys_addr of the MFI command. 2883665484d8SDoug Ambrisko */ 2884665484d8SDoug Ambrisko u_int8_t 2885665484d8SDoug Ambrisko mrsas_build_mptmfi_passthru(struct mrsas_softc *sc, struct mrsas_mfi_cmd *mfi_cmd) 2886665484d8SDoug Ambrisko { 2887665484d8SDoug Ambrisko MPI25_IEEE_SGE_CHAIN64 *mpi25_ieee_chain; 2888665484d8SDoug Ambrisko PTR_MRSAS_RAID_SCSI_IO_REQUEST io_req; 2889665484d8SDoug Ambrisko struct mrsas_mpt_cmd *mpt_cmd; 2890665484d8SDoug Ambrisko struct mrsas_header *frame_hdr = &mfi_cmd->frame->hdr; 2891665484d8SDoug Ambrisko 2892665484d8SDoug Ambrisko mpt_cmd = mrsas_get_mpt_cmd(sc); 2893665484d8SDoug Ambrisko if (!mpt_cmd) 2894665484d8SDoug Ambrisko return(1); 2895665484d8SDoug Ambrisko 2896665484d8SDoug Ambrisko /* Save the smid. To be used for returning the cmd */ 2897665484d8SDoug Ambrisko mfi_cmd->cmd_id.context.smid = mpt_cmd->index; 2898665484d8SDoug Ambrisko 2899665484d8SDoug Ambrisko mpt_cmd->sync_cmd_idx = mfi_cmd->index; 2900665484d8SDoug Ambrisko 2901665484d8SDoug Ambrisko /* 2902665484d8SDoug Ambrisko * For cmds where the flag is set, store the flag and check 2903665484d8SDoug Ambrisko * on completion. For cmds with this flag, don't call 2904665484d8SDoug Ambrisko * mrsas_complete_cmd. 2905665484d8SDoug Ambrisko */ 2906665484d8SDoug Ambrisko 2907665484d8SDoug Ambrisko if (frame_hdr->flags & MFI_FRAME_DONT_POST_IN_REPLY_QUEUE) 2908665484d8SDoug Ambrisko mpt_cmd->flags = MFI_FRAME_DONT_POST_IN_REPLY_QUEUE; 2909665484d8SDoug Ambrisko 2910665484d8SDoug Ambrisko io_req = mpt_cmd->io_request; 2911665484d8SDoug Ambrisko 2912665484d8SDoug Ambrisko if ((sc->device_id == MRSAS_INVADER) || (sc->device_id == MRSAS_FURY)) { 2913665484d8SDoug Ambrisko pMpi25IeeeSgeChain64_t sgl_ptr_end = (pMpi25IeeeSgeChain64_t) &io_req->SGL; 2914665484d8SDoug Ambrisko sgl_ptr_end += sc->max_sge_in_main_msg - 1; 2915665484d8SDoug Ambrisko sgl_ptr_end->Flags = 0; 2916665484d8SDoug Ambrisko } 2917665484d8SDoug Ambrisko 2918665484d8SDoug Ambrisko mpi25_ieee_chain = (MPI25_IEEE_SGE_CHAIN64 *)&io_req->SGL.IeeeChain; 2919665484d8SDoug Ambrisko 2920665484d8SDoug Ambrisko io_req->Function = MRSAS_MPI2_FUNCTION_PASSTHRU_IO_REQUEST; 2921665484d8SDoug Ambrisko io_req->SGLOffset0 = offsetof(MRSAS_RAID_SCSI_IO_REQUEST, SGL) / 4; 2922665484d8SDoug Ambrisko io_req->ChainOffset = sc->chain_offset_mfi_pthru; 2923665484d8SDoug Ambrisko 2924665484d8SDoug Ambrisko mpi25_ieee_chain->Address = mfi_cmd->frame_phys_addr; 2925665484d8SDoug Ambrisko 2926665484d8SDoug Ambrisko mpi25_ieee_chain->Flags= IEEE_SGE_FLAGS_CHAIN_ELEMENT | 2927665484d8SDoug Ambrisko MPI2_IEEE_SGE_FLAGS_IOCPLBNTA_ADDR; 2928665484d8SDoug Ambrisko 2929665484d8SDoug Ambrisko mpi25_ieee_chain->Length = MRSAS_MAX_SZ_CHAIN_FRAME; 2930665484d8SDoug Ambrisko 2931665484d8SDoug Ambrisko return(0); 2932665484d8SDoug Ambrisko } 2933665484d8SDoug Ambrisko 2934665484d8SDoug Ambrisko /** 2935665484d8SDoug Ambrisko * mrsas_issue_blocked_cmd - Synchronous wrapper around regular FW cmds 2936665484d8SDoug Ambrisko * input: Adapter soft state 2937665484d8SDoug Ambrisko * Command to be issued 2938665484d8SDoug Ambrisko * 2939665484d8SDoug Ambrisko * This function waits on an event for the command to be returned 2940665484d8SDoug Ambrisko * from the ISR. Max wait time is MRSAS_INTERNAL_CMD_WAIT_TIME secs. 2941665484d8SDoug Ambrisko * Used for issuing internal and ioctl commands. 2942665484d8SDoug Ambrisko */ 2943665484d8SDoug Ambrisko int mrsas_issue_blocked_cmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 2944665484d8SDoug Ambrisko { 2945665484d8SDoug Ambrisko u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME; 2946665484d8SDoug Ambrisko unsigned long total_time = 0; 2947665484d8SDoug Ambrisko int retcode = 0; 2948665484d8SDoug Ambrisko 2949665484d8SDoug Ambrisko /* Initialize cmd_status */ 2950665484d8SDoug Ambrisko cmd->cmd_status = ECONNREFUSED; 2951665484d8SDoug Ambrisko 2952665484d8SDoug Ambrisko /* Build MPT-MFI command for issue to FW */ 2953665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)){ 2954665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot issue DCMD internal command.\n"); 2955665484d8SDoug Ambrisko return(1); 2956665484d8SDoug Ambrisko } 2957665484d8SDoug Ambrisko 2958665484d8SDoug Ambrisko sc->chan = (void*)&cmd; 2959665484d8SDoug Ambrisko 2960665484d8SDoug Ambrisko /* The following is for debug only... */ 2961665484d8SDoug Ambrisko //device_printf(sc->mrsas_dev,"DCMD issued to FW, about to sleep-wait...\n"); 2962665484d8SDoug Ambrisko //device_printf(sc->mrsas_dev,"sc->chan = %p\n", sc->chan); 2963665484d8SDoug Ambrisko 2964665484d8SDoug Ambrisko while (1) { 2965665484d8SDoug Ambrisko if (cmd->cmd_status == ECONNREFUSED){ 2966665484d8SDoug Ambrisko tsleep((void *)&sc->chan, 0, "mrsas_sleep", hz); 2967665484d8SDoug Ambrisko } 2968665484d8SDoug Ambrisko else 2969665484d8SDoug Ambrisko break; 2970665484d8SDoug Ambrisko total_time++; 2971665484d8SDoug Ambrisko if (total_time >= max_wait) { 2972665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Internal command timed out after %d seconds.\n", max_wait); 2973665484d8SDoug Ambrisko retcode = 1; 2974665484d8SDoug Ambrisko break; 2975665484d8SDoug Ambrisko } 2976665484d8SDoug Ambrisko } 2977665484d8SDoug Ambrisko return(retcode); 2978665484d8SDoug Ambrisko } 2979665484d8SDoug Ambrisko 2980665484d8SDoug Ambrisko /** 2981665484d8SDoug Ambrisko * mrsas_complete_mptmfi_passthru - Completes a command 2982665484d8SDoug Ambrisko * input: sc: Adapter soft state 2983665484d8SDoug Ambrisko * cmd: Command to be completed 2984665484d8SDoug Ambrisko * status: cmd completion status 2985665484d8SDoug Ambrisko * 2986665484d8SDoug Ambrisko * This function is called from mrsas_complete_cmd() after an interrupt 2987665484d8SDoug Ambrisko * is received from Firmware, and io_request->Function is 2988665484d8SDoug Ambrisko * MRSAS_MPI2_FUNCTION_PASSTHRU_IO_REQUEST. 2989665484d8SDoug Ambrisko */ 2990665484d8SDoug Ambrisko void 2991665484d8SDoug Ambrisko mrsas_complete_mptmfi_passthru(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd, 2992665484d8SDoug Ambrisko u_int8_t status) 2993665484d8SDoug Ambrisko { 2994665484d8SDoug Ambrisko struct mrsas_header *hdr = &cmd->frame->hdr; 2995665484d8SDoug Ambrisko u_int8_t cmd_status = cmd->frame->hdr.cmd_status; 2996665484d8SDoug Ambrisko 2997665484d8SDoug Ambrisko /* Reset the retry counter for future re-tries */ 2998665484d8SDoug Ambrisko cmd->retry_for_fw_reset = 0; 2999665484d8SDoug Ambrisko 3000665484d8SDoug Ambrisko if (cmd->ccb_ptr) 3001665484d8SDoug Ambrisko cmd->ccb_ptr = NULL; 3002665484d8SDoug Ambrisko 3003665484d8SDoug Ambrisko switch (hdr->cmd) { 3004665484d8SDoug Ambrisko case MFI_CMD_INVALID: 3005665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "MFI_CMD_INVALID command.\n"); 3006665484d8SDoug Ambrisko break; 3007665484d8SDoug Ambrisko case MFI_CMD_PD_SCSI_IO: 3008665484d8SDoug Ambrisko case MFI_CMD_LD_SCSI_IO: 3009665484d8SDoug Ambrisko /* 3010665484d8SDoug Ambrisko * MFI_CMD_PD_SCSI_IO and MFI_CMD_LD_SCSI_IO could have been 3011665484d8SDoug Ambrisko * issued either through an IO path or an IOCTL path. If it 3012665484d8SDoug Ambrisko * was via IOCTL, we will send it to internal completion. 3013665484d8SDoug Ambrisko */ 3014665484d8SDoug Ambrisko if (cmd->sync_cmd) { 3015665484d8SDoug Ambrisko cmd->sync_cmd = 0; 3016665484d8SDoug Ambrisko mrsas_wakeup(sc, cmd); 3017665484d8SDoug Ambrisko break; 3018665484d8SDoug Ambrisko } 3019665484d8SDoug Ambrisko case MFI_CMD_SMP: 3020665484d8SDoug Ambrisko case MFI_CMD_STP: 3021665484d8SDoug Ambrisko case MFI_CMD_DCMD: 3022665484d8SDoug Ambrisko /* Check for LD map update */ 3023665484d8SDoug Ambrisko if ((cmd->frame->dcmd.opcode == MR_DCMD_LD_MAP_GET_INFO) && 3024665484d8SDoug Ambrisko (cmd->frame->dcmd.mbox.b[1] == 1)) { 3025665484d8SDoug Ambrisko sc->fast_path_io = 0; 3026665484d8SDoug Ambrisko mtx_lock(&sc->raidmap_lock); 3027665484d8SDoug Ambrisko if (cmd_status != 0) { 3028665484d8SDoug Ambrisko if (cmd_status != MFI_STAT_NOT_FOUND) 3029665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "map sync failed, status=%x\n",cmd_status); 3030665484d8SDoug Ambrisko else { 3031665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3032665484d8SDoug Ambrisko mtx_unlock(&sc->raidmap_lock); 3033665484d8SDoug Ambrisko break; 3034665484d8SDoug Ambrisko } 3035665484d8SDoug Ambrisko } 3036665484d8SDoug Ambrisko else 3037665484d8SDoug Ambrisko sc->map_id++; 3038665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3039665484d8SDoug Ambrisko if (MR_ValidateMapInfo(sc)) 3040665484d8SDoug Ambrisko sc->fast_path_io = 0; 3041665484d8SDoug Ambrisko else 3042665484d8SDoug Ambrisko sc->fast_path_io = 1; 3043665484d8SDoug Ambrisko mrsas_sync_map_info(sc); 3044665484d8SDoug Ambrisko mtx_unlock(&sc->raidmap_lock); 3045665484d8SDoug Ambrisko break; 3046665484d8SDoug Ambrisko } 3047665484d8SDoug Ambrisko #if 0 //currently not supporting event handling, so commenting out 3048665484d8SDoug Ambrisko if (cmd->frame->dcmd.opcode == MR_DCMD_CTRL_EVENT_GET_INFO || 3049665484d8SDoug Ambrisko cmd->frame->dcmd.opcode == MR_DCMD_CTRL_EVENT_GET) { 3050665484d8SDoug Ambrisko mrsas_poll_wait_aen = 0; 3051665484d8SDoug Ambrisko } 3052665484d8SDoug Ambrisko #endif 3053665484d8SDoug Ambrisko /* See if got an event notification */ 3054665484d8SDoug Ambrisko if (cmd->frame->dcmd.opcode == MR_DCMD_CTRL_EVENT_WAIT) 3055665484d8SDoug Ambrisko mrsas_complete_aen(sc, cmd); 3056665484d8SDoug Ambrisko else 3057665484d8SDoug Ambrisko mrsas_wakeup(sc, cmd); 3058665484d8SDoug Ambrisko break; 3059665484d8SDoug Ambrisko case MFI_CMD_ABORT: 3060665484d8SDoug Ambrisko /* Command issued to abort another cmd return */ 3061665484d8SDoug Ambrisko mrsas_complete_abort(sc, cmd); 3062665484d8SDoug Ambrisko break; 3063665484d8SDoug Ambrisko default: 3064665484d8SDoug Ambrisko device_printf(sc->mrsas_dev,"Unknown command completed! [0x%X]\n", hdr->cmd); 3065665484d8SDoug Ambrisko break; 3066665484d8SDoug Ambrisko } 3067665484d8SDoug Ambrisko } 3068665484d8SDoug Ambrisko 3069665484d8SDoug Ambrisko /** 3070665484d8SDoug Ambrisko * mrsas_wakeup - Completes an internal command 3071665484d8SDoug Ambrisko * input: Adapter soft state 3072665484d8SDoug Ambrisko * Command to be completed 3073665484d8SDoug Ambrisko * 3074665484d8SDoug Ambrisko * In mrsas_issue_blocked_cmd(), after a command is issued to Firmware, 3075665484d8SDoug Ambrisko * a wait timer is started. This function is called from 3076665484d8SDoug Ambrisko * mrsas_complete_mptmfi_passthru() as it completes the command, 3077665484d8SDoug Ambrisko * to wake up from the command wait. 3078665484d8SDoug Ambrisko */ 3079665484d8SDoug Ambrisko void mrsas_wakeup(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 3080665484d8SDoug Ambrisko { 3081665484d8SDoug Ambrisko cmd->cmd_status = cmd->frame->io.cmd_status; 3082665484d8SDoug Ambrisko 3083665484d8SDoug Ambrisko if (cmd->cmd_status == ECONNREFUSED) 3084665484d8SDoug Ambrisko cmd->cmd_status = 0; 3085665484d8SDoug Ambrisko 3086665484d8SDoug Ambrisko /* For debug only ... */ 3087665484d8SDoug Ambrisko //device_printf(sc->mrsas_dev,"DCMD rec'd for wakeup, sc->chan=%p\n", sc->chan); 3088665484d8SDoug Ambrisko 3089665484d8SDoug Ambrisko sc->chan = (void*)&cmd; 3090665484d8SDoug Ambrisko wakeup_one((void *)&sc->chan); 3091665484d8SDoug Ambrisko return; 3092665484d8SDoug Ambrisko } 3093665484d8SDoug Ambrisko 3094665484d8SDoug Ambrisko /** 3095665484d8SDoug Ambrisko * mrsas_shutdown_ctlr: Instructs FW to shutdown the controller 3096665484d8SDoug Ambrisko * input: Adapter soft state 3097665484d8SDoug Ambrisko * Shutdown/Hibernate 3098665484d8SDoug Ambrisko * 3099665484d8SDoug Ambrisko * This function issues a DCMD internal command to Firmware to initiate 3100665484d8SDoug Ambrisko * shutdown of the controller. 3101665484d8SDoug Ambrisko */ 3102665484d8SDoug Ambrisko static void mrsas_shutdown_ctlr(struct mrsas_softc *sc, u_int32_t opcode) 3103665484d8SDoug Ambrisko { 3104665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3105665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 3106665484d8SDoug Ambrisko 3107665484d8SDoug Ambrisko if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) 3108665484d8SDoug Ambrisko return; 3109665484d8SDoug Ambrisko 3110665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3111665484d8SDoug Ambrisko if (!cmd) { 3112665484d8SDoug Ambrisko device_printf(sc->mrsas_dev,"Cannot allocate for shutdown cmd.\n"); 3113665484d8SDoug Ambrisko return; 3114665484d8SDoug Ambrisko } 3115665484d8SDoug Ambrisko 3116665484d8SDoug Ambrisko if (sc->aen_cmd) 3117665484d8SDoug Ambrisko mrsas_issue_blocked_abort_cmd(sc, sc->aen_cmd); 3118665484d8SDoug Ambrisko 3119665484d8SDoug Ambrisko if (sc->map_update_cmd) 3120665484d8SDoug Ambrisko mrsas_issue_blocked_abort_cmd(sc, sc->map_update_cmd); 3121665484d8SDoug Ambrisko 3122665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 3123665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 3124665484d8SDoug Ambrisko 3125665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 3126665484d8SDoug Ambrisko dcmd->cmd_status = 0x0; 3127665484d8SDoug Ambrisko dcmd->sge_count = 0; 3128665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_NONE; 3129665484d8SDoug Ambrisko dcmd->timeout = 0; 3130665484d8SDoug Ambrisko dcmd->pad_0 = 0; 3131665484d8SDoug Ambrisko dcmd->data_xfer_len = 0; 3132665484d8SDoug Ambrisko dcmd->opcode = opcode; 3133665484d8SDoug Ambrisko 3134665484d8SDoug Ambrisko device_printf(sc->mrsas_dev,"Preparing to shut down controller.\n"); 3135665484d8SDoug Ambrisko 3136665484d8SDoug Ambrisko mrsas_issue_blocked_cmd(sc, cmd); 3137665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3138665484d8SDoug Ambrisko 3139665484d8SDoug Ambrisko return; 3140665484d8SDoug Ambrisko } 3141665484d8SDoug Ambrisko 3142665484d8SDoug Ambrisko /** 3143665484d8SDoug Ambrisko * mrsas_flush_cache: Requests FW to flush all its caches 3144665484d8SDoug Ambrisko * input: Adapter soft state 3145665484d8SDoug Ambrisko * 3146665484d8SDoug Ambrisko * This function is issues a DCMD internal command to Firmware to initiate 3147665484d8SDoug Ambrisko * flushing of all caches. 3148665484d8SDoug Ambrisko */ 3149665484d8SDoug Ambrisko static void mrsas_flush_cache(struct mrsas_softc *sc) 3150665484d8SDoug Ambrisko { 3151665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3152665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 3153665484d8SDoug Ambrisko 3154665484d8SDoug Ambrisko if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) 3155665484d8SDoug Ambrisko return; 3156665484d8SDoug Ambrisko 3157665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3158665484d8SDoug Ambrisko if (!cmd) { 3159665484d8SDoug Ambrisko device_printf(sc->mrsas_dev,"Cannot allocate for flush cache cmd.\n"); 3160665484d8SDoug Ambrisko return; 3161665484d8SDoug Ambrisko } 3162665484d8SDoug Ambrisko 3163665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 3164665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 3165665484d8SDoug Ambrisko 3166665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 3167665484d8SDoug Ambrisko dcmd->cmd_status = 0x0; 3168665484d8SDoug Ambrisko dcmd->sge_count = 0; 3169665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_NONE; 3170665484d8SDoug Ambrisko dcmd->timeout = 0; 3171665484d8SDoug Ambrisko dcmd->pad_0 = 0; 3172665484d8SDoug Ambrisko dcmd->data_xfer_len = 0; 3173665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_CTRL_CACHE_FLUSH; 3174665484d8SDoug Ambrisko dcmd->mbox.b[0] = MR_FLUSH_CTRL_CACHE | MR_FLUSH_DISK_CACHE; 3175665484d8SDoug Ambrisko 3176665484d8SDoug Ambrisko mrsas_issue_blocked_cmd(sc, cmd); 3177665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3178665484d8SDoug Ambrisko 3179665484d8SDoug Ambrisko return; 3180665484d8SDoug Ambrisko } 3181665484d8SDoug Ambrisko 3182665484d8SDoug Ambrisko /** 3183665484d8SDoug Ambrisko * mrsas_get_map_info: Load and validate RAID map 3184665484d8SDoug Ambrisko * input: Adapter instance soft state 3185665484d8SDoug Ambrisko * 3186665484d8SDoug Ambrisko * This function calls mrsas_get_ld_map_info() and MR_ValidateMapInfo() 3187665484d8SDoug Ambrisko * to load and validate RAID map. It returns 0 if successful, 1 other- 3188665484d8SDoug Ambrisko * wise. 3189665484d8SDoug Ambrisko */ 3190665484d8SDoug Ambrisko static int mrsas_get_map_info(struct mrsas_softc *sc) 3191665484d8SDoug Ambrisko { 3192665484d8SDoug Ambrisko uint8_t retcode = 0; 3193665484d8SDoug Ambrisko 3194665484d8SDoug Ambrisko sc->fast_path_io = 0; 3195665484d8SDoug Ambrisko if (!mrsas_get_ld_map_info(sc)) { 3196665484d8SDoug Ambrisko retcode = MR_ValidateMapInfo(sc); 3197665484d8SDoug Ambrisko if (retcode == 0) { 3198665484d8SDoug Ambrisko sc->fast_path_io = 1; 3199665484d8SDoug Ambrisko return 0; 3200665484d8SDoug Ambrisko } 3201665484d8SDoug Ambrisko } 3202665484d8SDoug Ambrisko return 1; 3203665484d8SDoug Ambrisko } 3204665484d8SDoug Ambrisko 3205665484d8SDoug Ambrisko /** 3206665484d8SDoug Ambrisko * mrsas_get_ld_map_info: Get FW's ld_map structure 3207665484d8SDoug Ambrisko * input: Adapter instance soft state 3208665484d8SDoug Ambrisko * 3209665484d8SDoug Ambrisko * Issues an internal command (DCMD) to get the FW's controller PD 3210665484d8SDoug Ambrisko * list structure. 3211665484d8SDoug Ambrisko */ 3212665484d8SDoug Ambrisko static int mrsas_get_ld_map_info(struct mrsas_softc *sc) 3213665484d8SDoug Ambrisko { 3214665484d8SDoug Ambrisko int retcode = 0; 3215665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3216665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 3217*4799d485SKashyap D Desai void *map; 3218665484d8SDoug Ambrisko bus_addr_t map_phys_addr = 0; 3219665484d8SDoug Ambrisko 3220665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3221665484d8SDoug Ambrisko if (!cmd) { 3222*4799d485SKashyap D Desai device_printf(sc->mrsas_dev, 3223*4799d485SKashyap D Desai "Cannot alloc for ld map info cmd.\n"); 3224665484d8SDoug Ambrisko return 1; 3225665484d8SDoug Ambrisko } 3226665484d8SDoug Ambrisko 3227665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 3228665484d8SDoug Ambrisko 3229*4799d485SKashyap D Desai map = (void *)sc->raidmap_mem[(sc->map_id & 1)]; 3230665484d8SDoug Ambrisko map_phys_addr = sc->raidmap_phys_addr[(sc->map_id & 1)]; 3231665484d8SDoug Ambrisko if (!map) { 3232*4799d485SKashyap D Desai device_printf(sc->mrsas_dev, 3233*4799d485SKashyap D Desai "Failed to alloc mem for ld map info.\n"); 3234665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3235665484d8SDoug Ambrisko return (ENOMEM); 3236665484d8SDoug Ambrisko } 3237*4799d485SKashyap D Desai memset(map, 0, sizeof(sc->max_map_sz)); 3238665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 3239665484d8SDoug Ambrisko 3240665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 3241665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 3242665484d8SDoug Ambrisko dcmd->sge_count = 1; 3243665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 3244665484d8SDoug Ambrisko dcmd->timeout = 0; 3245665484d8SDoug Ambrisko dcmd->pad_0 = 0; 3246*4799d485SKashyap D Desai dcmd->data_xfer_len = sc->current_map_sz; 3247665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_LD_MAP_GET_INFO; 3248665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = map_phys_addr; 3249*4799d485SKashyap D Desai dcmd->sgl.sge32[0].length = sc->current_map_sz; 3250*4799d485SKashyap D Desai 3251665484d8SDoug Ambrisko if (!mrsas_issue_polled(sc, cmd)) 3252665484d8SDoug Ambrisko retcode = 0; 3253665484d8SDoug Ambrisko else 3254665484d8SDoug Ambrisko { 3255*4799d485SKashyap D Desai device_printf(sc->mrsas_dev, 3256*4799d485SKashyap D Desai "Fail to send get LD map info cmd.\n"); 3257665484d8SDoug Ambrisko retcode = 1; 3258665484d8SDoug Ambrisko } 3259665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3260*4799d485SKashyap D Desai 3261665484d8SDoug Ambrisko return(retcode); 3262665484d8SDoug Ambrisko } 3263665484d8SDoug Ambrisko 3264665484d8SDoug Ambrisko /** 3265665484d8SDoug Ambrisko * mrsas_sync_map_info: Get FW's ld_map structure 3266665484d8SDoug Ambrisko * input: Adapter instance soft state 3267665484d8SDoug Ambrisko * 3268665484d8SDoug Ambrisko * Issues an internal command (DCMD) to get the FW's controller PD 3269665484d8SDoug Ambrisko * list structure. 3270665484d8SDoug Ambrisko */ 3271665484d8SDoug Ambrisko static int mrsas_sync_map_info(struct mrsas_softc *sc) 3272665484d8SDoug Ambrisko { 3273665484d8SDoug Ambrisko int retcode = 0, i; 3274665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3275665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 3276665484d8SDoug Ambrisko uint32_t size_sync_info, num_lds; 3277665484d8SDoug Ambrisko MR_LD_TARGET_SYNC *target_map = NULL; 3278*4799d485SKashyap D Desai MR_DRV_RAID_MAP_ALL *map; 3279665484d8SDoug Ambrisko MR_LD_RAID *raid; 3280665484d8SDoug Ambrisko MR_LD_TARGET_SYNC *ld_sync; 3281665484d8SDoug Ambrisko bus_addr_t map_phys_addr = 0; 3282665484d8SDoug Ambrisko 3283665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3284665484d8SDoug Ambrisko if (!cmd) { 3285*4799d485SKashyap D Desai device_printf(sc->mrsas_dev, 3286*4799d485SKashyap D Desai "Cannot alloc for sync map info cmd\n"); 3287665484d8SDoug Ambrisko return 1; 3288665484d8SDoug Ambrisko } 3289665484d8SDoug Ambrisko 3290*4799d485SKashyap D Desai map = sc->ld_drv_map[sc->map_id & 1]; 3291665484d8SDoug Ambrisko num_lds = map->raidMap.ldCount; 3292665484d8SDoug Ambrisko 3293665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 3294665484d8SDoug Ambrisko size_sync_info = sizeof(MR_LD_TARGET_SYNC) * num_lds; 3295665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 3296665484d8SDoug Ambrisko 3297*4799d485SKashyap D Desai target_map = 3298*4799d485SKashyap D Desai (MR_LD_TARGET_SYNC *)sc->raidmap_mem[(sc->map_id - 1) & 1]; 3299*4799d485SKashyap D Desai memset(target_map, 0, sc->max_map_sz); 3300665484d8SDoug Ambrisko 3301665484d8SDoug Ambrisko map_phys_addr = sc->raidmap_phys_addr[(sc->map_id - 1) & 1]; 3302665484d8SDoug Ambrisko 3303665484d8SDoug Ambrisko ld_sync = (MR_LD_TARGET_SYNC *)target_map; 3304665484d8SDoug Ambrisko 3305665484d8SDoug Ambrisko for (i = 0; i < num_lds; i++, ld_sync++) { 3306665484d8SDoug Ambrisko raid = MR_LdRaidGet(i, map); 3307665484d8SDoug Ambrisko ld_sync->targetId = MR_GetLDTgtId(i, map); 3308665484d8SDoug Ambrisko ld_sync->seqNum = raid->seqNum; 3309665484d8SDoug Ambrisko } 3310665484d8SDoug Ambrisko 3311665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 3312665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 3313665484d8SDoug Ambrisko dcmd->sge_count = 1; 3314665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_WRITE; 3315665484d8SDoug Ambrisko dcmd->timeout = 0; 3316665484d8SDoug Ambrisko dcmd->pad_0 = 0; 3317*4799d485SKashyap D Desai dcmd->data_xfer_len = sc->current_map_sz; 3318665484d8SDoug Ambrisko dcmd->mbox.b[0] = num_lds; 3319665484d8SDoug Ambrisko dcmd->mbox.b[1] = MRSAS_DCMD_MBOX_PEND_FLAG; 3320665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_LD_MAP_GET_INFO; 3321665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = map_phys_addr; 3322*4799d485SKashyap D Desai dcmd->sgl.sge32[0].length = sc->current_map_sz; 3323665484d8SDoug Ambrisko 3324665484d8SDoug Ambrisko sc->map_update_cmd = cmd; 3325665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)) { 3326*4799d485SKashyap D Desai device_printf(sc->mrsas_dev, 3327*4799d485SKashyap D Desai "Fail to send sync map info command.\n"); 3328665484d8SDoug Ambrisko return(1); 3329665484d8SDoug Ambrisko } 3330665484d8SDoug Ambrisko return(retcode); 3331665484d8SDoug Ambrisko } 3332665484d8SDoug Ambrisko 3333665484d8SDoug Ambrisko /** 3334665484d8SDoug Ambrisko * mrsas_get_pd_list: Returns FW's PD list structure 3335665484d8SDoug Ambrisko * input: Adapter soft state 3336665484d8SDoug Ambrisko * 3337665484d8SDoug Ambrisko * Issues an internal command (DCMD) to get the FW's controller PD 3338665484d8SDoug Ambrisko * list structure. This information is mainly used to find out about 3339665484d8SDoug Ambrisko * system supported by Firmware. 3340665484d8SDoug Ambrisko */ 3341665484d8SDoug Ambrisko static int mrsas_get_pd_list(struct mrsas_softc *sc) 3342665484d8SDoug Ambrisko { 3343665484d8SDoug Ambrisko int retcode = 0, pd_index = 0, pd_count=0, pd_list_size; 3344665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3345665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 3346665484d8SDoug Ambrisko struct MR_PD_LIST *pd_list_mem; 3347665484d8SDoug Ambrisko struct MR_PD_ADDRESS *pd_addr; 3348665484d8SDoug Ambrisko bus_addr_t pd_list_phys_addr = 0; 3349665484d8SDoug Ambrisko struct mrsas_tmp_dcmd *tcmd; 3350665484d8SDoug Ambrisko 3351665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3352665484d8SDoug Ambrisko if (!cmd) { 3353*4799d485SKashyap D Desai device_printf(sc->mrsas_dev, 3354*4799d485SKashyap D Desai "Cannot alloc for get PD list cmd\n"); 3355665484d8SDoug Ambrisko return 1; 3356665484d8SDoug Ambrisko } 3357665484d8SDoug Ambrisko 3358665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 3359665484d8SDoug Ambrisko 3360665484d8SDoug Ambrisko tcmd = malloc(sizeof(struct mrsas_tmp_dcmd), M_MRSAS, M_NOWAIT); 3361665484d8SDoug Ambrisko pd_list_size = MRSAS_MAX_PD * sizeof(struct MR_PD_LIST); 3362665484d8SDoug Ambrisko if (mrsas_alloc_tmp_dcmd(sc, tcmd, pd_list_size) != SUCCESS) { 3363*4799d485SKashyap D Desai device_printf(sc->mrsas_dev, 3364*4799d485SKashyap D Desai "Cannot alloc dmamap for get PD list cmd\n"); 3365665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3366665484d8SDoug Ambrisko return(ENOMEM); 3367665484d8SDoug Ambrisko } 3368665484d8SDoug Ambrisko else { 3369665484d8SDoug Ambrisko pd_list_mem = tcmd->tmp_dcmd_mem; 3370665484d8SDoug Ambrisko pd_list_phys_addr = tcmd->tmp_dcmd_phys_addr; 3371665484d8SDoug Ambrisko } 3372665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 3373665484d8SDoug Ambrisko 3374665484d8SDoug Ambrisko dcmd->mbox.b[0] = MR_PD_QUERY_TYPE_EXPOSED_TO_HOST; 3375665484d8SDoug Ambrisko dcmd->mbox.b[1] = 0; 3376665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 3377665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 3378665484d8SDoug Ambrisko dcmd->sge_count = 1; 3379665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 3380665484d8SDoug Ambrisko dcmd->timeout = 0; 3381665484d8SDoug Ambrisko dcmd->pad_0 = 0; 3382665484d8SDoug Ambrisko dcmd->data_xfer_len = MRSAS_MAX_PD * sizeof(struct MR_PD_LIST); 3383665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_PD_LIST_QUERY; 3384665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = pd_list_phys_addr; 3385665484d8SDoug Ambrisko dcmd->sgl.sge32[0].length = MRSAS_MAX_PD * sizeof(struct MR_PD_LIST); 3386665484d8SDoug Ambrisko 3387665484d8SDoug Ambrisko if (!mrsas_issue_polled(sc, cmd)) 3388665484d8SDoug Ambrisko retcode = 0; 3389665484d8SDoug Ambrisko else 3390665484d8SDoug Ambrisko retcode = 1; 3391665484d8SDoug Ambrisko 3392665484d8SDoug Ambrisko /* Get the instance PD list */ 3393665484d8SDoug Ambrisko pd_count = MRSAS_MAX_PD; 3394665484d8SDoug Ambrisko pd_addr = pd_list_mem->addr; 3395665484d8SDoug Ambrisko if (retcode == 0 && pd_list_mem->count < pd_count) { 3396*4799d485SKashyap D Desai memset(sc->local_pd_list, 0, 3397*4799d485SKashyap D Desai MRSAS_MAX_PD * sizeof(struct mrsas_pd_list)); 3398665484d8SDoug Ambrisko for (pd_index = 0; pd_index < pd_list_mem->count; pd_index++) { 3399665484d8SDoug Ambrisko sc->local_pd_list[pd_addr->deviceId].tid = pd_addr->deviceId; 3400*4799d485SKashyap D Desai sc->local_pd_list[pd_addr->deviceId].driveType = 3401*4799d485SKashyap D Desai pd_addr->scsiDevType; 3402*4799d485SKashyap D Desai sc->local_pd_list[pd_addr->deviceId].driveState = 3403*4799d485SKashyap D Desai MR_PD_STATE_SYSTEM; 3404665484d8SDoug Ambrisko pd_addr++; 3405665484d8SDoug Ambrisko } 3406665484d8SDoug Ambrisko } 3407665484d8SDoug Ambrisko 3408665484d8SDoug Ambrisko /* Use mutext/spinlock if pd_list component size increase more than 32 bit. */ 3409665484d8SDoug Ambrisko memcpy(sc->pd_list, sc->local_pd_list, sizeof(sc->local_pd_list)); 3410665484d8SDoug Ambrisko mrsas_free_tmp_dcmd(tcmd); 3411665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3412665484d8SDoug Ambrisko free(tcmd, M_MRSAS); 3413665484d8SDoug Ambrisko return(retcode); 3414665484d8SDoug Ambrisko } 3415665484d8SDoug Ambrisko 3416665484d8SDoug Ambrisko /** 3417665484d8SDoug Ambrisko * mrsas_get_ld_list: Returns FW's LD list structure 3418665484d8SDoug Ambrisko * input: Adapter soft state 3419665484d8SDoug Ambrisko * 3420665484d8SDoug Ambrisko * Issues an internal command (DCMD) to get the FW's controller PD 3421665484d8SDoug Ambrisko * list structure. This information is mainly used to find out about 3422665484d8SDoug Ambrisko * supported by the FW. 3423665484d8SDoug Ambrisko */ 3424665484d8SDoug Ambrisko static int mrsas_get_ld_list(struct mrsas_softc *sc) 3425665484d8SDoug Ambrisko { 3426665484d8SDoug Ambrisko int ld_list_size, retcode = 0, ld_index = 0, ids = 0; 3427665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3428665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 3429665484d8SDoug Ambrisko struct MR_LD_LIST *ld_list_mem; 3430665484d8SDoug Ambrisko bus_addr_t ld_list_phys_addr = 0; 3431665484d8SDoug Ambrisko struct mrsas_tmp_dcmd *tcmd; 3432665484d8SDoug Ambrisko 3433665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3434665484d8SDoug Ambrisko if (!cmd) { 3435*4799d485SKashyap D Desai device_printf(sc->mrsas_dev, 3436*4799d485SKashyap D Desai "Cannot alloc for get LD list cmd\n"); 3437665484d8SDoug Ambrisko return 1; 3438665484d8SDoug Ambrisko } 3439665484d8SDoug Ambrisko 3440665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 3441665484d8SDoug Ambrisko 3442665484d8SDoug Ambrisko tcmd = malloc(sizeof(struct mrsas_tmp_dcmd), M_MRSAS, M_NOWAIT); 3443665484d8SDoug Ambrisko ld_list_size = sizeof(struct MR_LD_LIST); 3444665484d8SDoug Ambrisko if (mrsas_alloc_tmp_dcmd(sc, tcmd, ld_list_size) != SUCCESS) { 3445*4799d485SKashyap D Desai device_printf(sc->mrsas_dev, 3446*4799d485SKashyap D Desai "Cannot alloc dmamap for get LD list cmd\n"); 3447665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3448665484d8SDoug Ambrisko return(ENOMEM); 3449665484d8SDoug Ambrisko } 3450665484d8SDoug Ambrisko else { 3451665484d8SDoug Ambrisko ld_list_mem = tcmd->tmp_dcmd_mem; 3452665484d8SDoug Ambrisko ld_list_phys_addr = tcmd->tmp_dcmd_phys_addr; 3453665484d8SDoug Ambrisko } 3454665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 3455665484d8SDoug Ambrisko 3456*4799d485SKashyap D Desai if (sc->max256vdSupport) 3457*4799d485SKashyap D Desai dcmd->mbox.b[0]=1; 3458*4799d485SKashyap D Desai 3459665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 3460665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 3461665484d8SDoug Ambrisko dcmd->sge_count = 1; 3462665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 3463665484d8SDoug Ambrisko dcmd->timeout = 0; 3464665484d8SDoug Ambrisko dcmd->data_xfer_len = sizeof(struct MR_LD_LIST); 3465665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_LD_GET_LIST; 3466665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = ld_list_phys_addr; 3467665484d8SDoug Ambrisko dcmd->sgl.sge32[0].length = sizeof(struct MR_LD_LIST); 3468665484d8SDoug Ambrisko dcmd->pad_0 = 0; 3469665484d8SDoug Ambrisko 3470665484d8SDoug Ambrisko if (!mrsas_issue_polled(sc, cmd)) 3471665484d8SDoug Ambrisko retcode = 0; 3472665484d8SDoug Ambrisko else 3473665484d8SDoug Ambrisko retcode = 1; 3474665484d8SDoug Ambrisko 3475*4799d485SKashyap D Desai #if VD_EXT_DEBUG 3476*4799d485SKashyap D Desai printf ("Number of LDs %d\n", ld_list_mem->ldCount); 3477*4799d485SKashyap D Desai #endif 3478*4799d485SKashyap D Desai 3479665484d8SDoug Ambrisko /* Get the instance LD list */ 3480*4799d485SKashyap D Desai if ((retcode == 0) && 3481*4799d485SKashyap D Desai (ld_list_mem->ldCount <= sc->fw_supported_vd_count)){ 3482665484d8SDoug Ambrisko sc->CurLdCount = ld_list_mem->ldCount; 3483*4799d485SKashyap D Desai memset(sc->ld_ids, 0xff, MAX_LOGICAL_DRIVES_EXT); 3484665484d8SDoug Ambrisko for (ld_index = 0; ld_index < ld_list_mem->ldCount; ld_index++) { 3485665484d8SDoug Ambrisko if (ld_list_mem->ldList[ld_index].state != 0) { 3486665484d8SDoug Ambrisko ids = ld_list_mem->ldList[ld_index].ref.ld_context.targetId; 3487665484d8SDoug Ambrisko sc->ld_ids[ids] = ld_list_mem->ldList[ld_index].ref.ld_context.targetId; 3488665484d8SDoug Ambrisko } 3489665484d8SDoug Ambrisko } 3490665484d8SDoug Ambrisko } 3491665484d8SDoug Ambrisko 3492665484d8SDoug Ambrisko mrsas_free_tmp_dcmd(tcmd); 3493665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3494665484d8SDoug Ambrisko free(tcmd, M_MRSAS); 3495665484d8SDoug Ambrisko return(retcode); 3496665484d8SDoug Ambrisko } 3497665484d8SDoug Ambrisko 3498665484d8SDoug Ambrisko /** 3499665484d8SDoug Ambrisko * mrsas_alloc_tmp_dcmd: Allocates memory for temporary command 3500665484d8SDoug Ambrisko * input: Adapter soft state 3501665484d8SDoug Ambrisko * Temp command 3502665484d8SDoug Ambrisko * Size of alloction 3503665484d8SDoug Ambrisko * 3504665484d8SDoug Ambrisko * Allocates DMAable memory for a temporary internal command. The allocated 3505665484d8SDoug Ambrisko * memory is initialized to all zeros upon successful loading of the dma 3506665484d8SDoug Ambrisko * mapped memory. 3507665484d8SDoug Ambrisko */ 3508665484d8SDoug Ambrisko int mrsas_alloc_tmp_dcmd(struct mrsas_softc *sc, struct mrsas_tmp_dcmd *tcmd, 3509665484d8SDoug Ambrisko int size) 3510665484d8SDoug Ambrisko { 3511665484d8SDoug Ambrisko if (bus_dma_tag_create( sc->mrsas_parent_tag, // parent 3512665484d8SDoug Ambrisko 1, 0, // algnmnt, boundary 3513665484d8SDoug Ambrisko BUS_SPACE_MAXADDR_32BIT,// lowaddr 3514665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, // highaddr 3515665484d8SDoug Ambrisko NULL, NULL, // filter, filterarg 3516665484d8SDoug Ambrisko size, // maxsize 3517665484d8SDoug Ambrisko 1, // msegments 3518665484d8SDoug Ambrisko size, // maxsegsize 3519665484d8SDoug Ambrisko BUS_DMA_ALLOCNOW, // flags 3520665484d8SDoug Ambrisko NULL, NULL, // lockfunc, lockarg 3521665484d8SDoug Ambrisko &tcmd->tmp_dcmd_tag)) { 3522665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate tmp dcmd tag\n"); 3523665484d8SDoug Ambrisko return (ENOMEM); 3524665484d8SDoug Ambrisko } 3525665484d8SDoug Ambrisko if (bus_dmamem_alloc(tcmd->tmp_dcmd_tag, (void **)&tcmd->tmp_dcmd_mem, 3526665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &tcmd->tmp_dcmd_dmamap)) { 3527665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate tmp dcmd mem\n"); 3528665484d8SDoug Ambrisko return (ENOMEM); 3529665484d8SDoug Ambrisko } 3530665484d8SDoug Ambrisko if (bus_dmamap_load(tcmd->tmp_dcmd_tag, tcmd->tmp_dcmd_dmamap, 3531665484d8SDoug Ambrisko tcmd->tmp_dcmd_mem, size, mrsas_addr_cb, 3532665484d8SDoug Ambrisko &tcmd->tmp_dcmd_phys_addr, BUS_DMA_NOWAIT)) { 3533665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load tmp dcmd mem\n"); 3534665484d8SDoug Ambrisko return (ENOMEM); 3535665484d8SDoug Ambrisko } 3536665484d8SDoug Ambrisko 3537665484d8SDoug Ambrisko memset(tcmd->tmp_dcmd_mem, 0, size); 3538665484d8SDoug Ambrisko return (0); 3539665484d8SDoug Ambrisko } 3540665484d8SDoug Ambrisko 3541665484d8SDoug Ambrisko /** 3542665484d8SDoug Ambrisko * mrsas_free_tmp_dcmd: Free memory for temporary command 3543665484d8SDoug Ambrisko * input: temporary dcmd pointer 3544665484d8SDoug Ambrisko * 3545665484d8SDoug Ambrisko * Deallocates memory of the temporary command for use in the construction 3546665484d8SDoug Ambrisko * of the internal DCMD. 3547665484d8SDoug Ambrisko */ 3548665484d8SDoug Ambrisko void mrsas_free_tmp_dcmd(struct mrsas_tmp_dcmd *tmp) 3549665484d8SDoug Ambrisko { 3550665484d8SDoug Ambrisko if (tmp->tmp_dcmd_phys_addr) 3551665484d8SDoug Ambrisko bus_dmamap_unload(tmp->tmp_dcmd_tag, tmp->tmp_dcmd_dmamap); 3552665484d8SDoug Ambrisko if (tmp->tmp_dcmd_mem != NULL) 3553665484d8SDoug Ambrisko bus_dmamem_free(tmp->tmp_dcmd_tag, tmp->tmp_dcmd_mem, tmp->tmp_dcmd_dmamap); 3554665484d8SDoug Ambrisko if (tmp->tmp_dcmd_tag != NULL) 3555665484d8SDoug Ambrisko bus_dma_tag_destroy(tmp->tmp_dcmd_tag); 3556665484d8SDoug Ambrisko } 3557665484d8SDoug Ambrisko 3558665484d8SDoug Ambrisko /** 3559665484d8SDoug Ambrisko * mrsas_issue_blocked_abort_cmd: Aborts previously issued cmd 3560665484d8SDoug Ambrisko * input: Adapter soft state 3561665484d8SDoug Ambrisko * Previously issued cmd to be aborted 3562665484d8SDoug Ambrisko * 3563665484d8SDoug Ambrisko * This function is used to abort previously issued commands, such as AEN and 3564665484d8SDoug Ambrisko * RAID map sync map commands. The abort command is sent as a DCMD internal 3565665484d8SDoug Ambrisko * command and subsequently the driver will wait for a return status. The 3566665484d8SDoug Ambrisko * max wait time is MRSAS_INTERNAL_CMD_WAIT_TIME seconds. 3567665484d8SDoug Ambrisko */ 3568665484d8SDoug Ambrisko static int mrsas_issue_blocked_abort_cmd(struct mrsas_softc *sc, 3569665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd_to_abort) 3570665484d8SDoug Ambrisko { 3571665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3572665484d8SDoug Ambrisko struct mrsas_abort_frame *abort_fr; 3573665484d8SDoug Ambrisko u_int8_t retcode = 0; 3574665484d8SDoug Ambrisko unsigned long total_time = 0; 3575665484d8SDoug Ambrisko u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME; 3576665484d8SDoug Ambrisko 3577665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3578665484d8SDoug Ambrisko if (!cmd) { 3579665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc for abort cmd\n"); 3580665484d8SDoug Ambrisko return(1); 3581665484d8SDoug Ambrisko } 3582665484d8SDoug Ambrisko 3583665484d8SDoug Ambrisko abort_fr = &cmd->frame->abort; 3584665484d8SDoug Ambrisko 3585665484d8SDoug Ambrisko /* Prepare and issue the abort frame */ 3586665484d8SDoug Ambrisko abort_fr->cmd = MFI_CMD_ABORT; 3587665484d8SDoug Ambrisko abort_fr->cmd_status = 0xFF; 3588665484d8SDoug Ambrisko abort_fr->flags = 0; 3589665484d8SDoug Ambrisko abort_fr->abort_context = cmd_to_abort->index; 3590665484d8SDoug Ambrisko abort_fr->abort_mfi_phys_addr_lo = cmd_to_abort->frame_phys_addr; 3591665484d8SDoug Ambrisko abort_fr->abort_mfi_phys_addr_hi = 0; 3592665484d8SDoug Ambrisko 3593665484d8SDoug Ambrisko cmd->sync_cmd = 1; 3594665484d8SDoug Ambrisko cmd->cmd_status = 0xFF; 3595665484d8SDoug Ambrisko 3596665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)) { 3597665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Fail to send abort command.\n"); 3598665484d8SDoug Ambrisko return(1); 3599665484d8SDoug Ambrisko } 3600665484d8SDoug Ambrisko 3601665484d8SDoug Ambrisko /* Wait for this cmd to complete */ 3602665484d8SDoug Ambrisko sc->chan = (void*)&cmd; 3603665484d8SDoug Ambrisko while (1) { 3604665484d8SDoug Ambrisko if (cmd->cmd_status == 0xFF){ 3605665484d8SDoug Ambrisko tsleep((void *)&sc->chan, 0, "mrsas_sleep", hz); 3606665484d8SDoug Ambrisko } 3607665484d8SDoug Ambrisko else 3608665484d8SDoug Ambrisko break; 3609665484d8SDoug Ambrisko total_time++; 3610665484d8SDoug Ambrisko if (total_time >= max_wait) { 3611665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Abort cmd timed out after %d sec.\n", max_wait); 3612665484d8SDoug Ambrisko retcode = 1; 3613665484d8SDoug Ambrisko break; 3614665484d8SDoug Ambrisko } 3615665484d8SDoug Ambrisko } 3616665484d8SDoug Ambrisko 3617665484d8SDoug Ambrisko cmd->sync_cmd = 0; 3618665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3619665484d8SDoug Ambrisko return(retcode); 3620665484d8SDoug Ambrisko } 3621665484d8SDoug Ambrisko 3622665484d8SDoug Ambrisko /** 3623665484d8SDoug Ambrisko * mrsas_complete_abort: Completes aborting a command 3624665484d8SDoug Ambrisko * input: Adapter soft state 3625665484d8SDoug Ambrisko * Cmd that was issued to abort another cmd 3626665484d8SDoug Ambrisko * 3627665484d8SDoug Ambrisko * The mrsas_issue_blocked_abort_cmd() function waits for the command status 3628665484d8SDoug Ambrisko * to change after sending the command. This function is called from 3629665484d8SDoug Ambrisko * mrsas_complete_mptmfi_passthru() to wake up the sleep thread associated. 3630665484d8SDoug Ambrisko */ 3631665484d8SDoug Ambrisko void mrsas_complete_abort(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 3632665484d8SDoug Ambrisko { 3633665484d8SDoug Ambrisko if (cmd->sync_cmd) { 3634665484d8SDoug Ambrisko cmd->sync_cmd = 0; 3635665484d8SDoug Ambrisko cmd->cmd_status = 0; 3636665484d8SDoug Ambrisko sc->chan = (void*)&cmd; 3637665484d8SDoug Ambrisko wakeup_one((void *)&sc->chan); 3638665484d8SDoug Ambrisko } 3639665484d8SDoug Ambrisko return; 3640665484d8SDoug Ambrisko } 3641665484d8SDoug Ambrisko 3642665484d8SDoug Ambrisko /** 3643665484d8SDoug Ambrisko * mrsas_aen_handler: Callback function for AEN processing from thread context. 3644665484d8SDoug Ambrisko * input: Adapter soft state 3645665484d8SDoug Ambrisko * 3646665484d8SDoug Ambrisko */ 3647665484d8SDoug Ambrisko void mrsas_aen_handler(struct mrsas_softc *sc) 3648665484d8SDoug Ambrisko { 3649665484d8SDoug Ambrisko union mrsas_evt_class_locale class_locale; 3650665484d8SDoug Ambrisko int doscan = 0; 3651665484d8SDoug Ambrisko u_int32_t seq_num; 3652665484d8SDoug Ambrisko int error; 3653665484d8SDoug Ambrisko 3654665484d8SDoug Ambrisko if (!sc) { 3655665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "invalid instance!\n"); 3656665484d8SDoug Ambrisko return; 3657665484d8SDoug Ambrisko } 3658665484d8SDoug Ambrisko 3659665484d8SDoug Ambrisko if (sc->evt_detail_mem) { 3660665484d8SDoug Ambrisko switch (sc->evt_detail_mem->code) { 3661665484d8SDoug Ambrisko case MR_EVT_PD_INSERTED: 3662665484d8SDoug Ambrisko mrsas_get_pd_list(sc); 3663665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_1); 3664665484d8SDoug Ambrisko doscan = 0; 3665665484d8SDoug Ambrisko break; 3666665484d8SDoug Ambrisko case MR_EVT_PD_REMOVED: 3667665484d8SDoug Ambrisko mrsas_get_pd_list(sc); 3668665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_1); 3669665484d8SDoug Ambrisko doscan = 0; 3670665484d8SDoug Ambrisko break; 3671665484d8SDoug Ambrisko case MR_EVT_LD_OFFLINE: 3672665484d8SDoug Ambrisko case MR_EVT_CFG_CLEARED: 3673665484d8SDoug Ambrisko case MR_EVT_LD_DELETED: 3674665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_0); 3675665484d8SDoug Ambrisko doscan = 0; 3676665484d8SDoug Ambrisko break; 3677665484d8SDoug Ambrisko case MR_EVT_LD_CREATED: 3678665484d8SDoug Ambrisko mrsas_get_ld_list(sc); 3679665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_0); 3680665484d8SDoug Ambrisko doscan = 0; 3681665484d8SDoug Ambrisko break; 3682665484d8SDoug Ambrisko case MR_EVT_CTRL_HOST_BUS_SCAN_REQUESTED: 3683665484d8SDoug Ambrisko case MR_EVT_FOREIGN_CFG_IMPORTED: 3684665484d8SDoug Ambrisko case MR_EVT_LD_STATE_CHANGE: 3685665484d8SDoug Ambrisko doscan = 1; 3686665484d8SDoug Ambrisko break; 3687665484d8SDoug Ambrisko default: 3688665484d8SDoug Ambrisko doscan = 0; 3689665484d8SDoug Ambrisko break; 3690665484d8SDoug Ambrisko } 3691665484d8SDoug Ambrisko } else { 3692665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "invalid evt_detail\n"); 3693665484d8SDoug Ambrisko return; 3694665484d8SDoug Ambrisko } 3695665484d8SDoug Ambrisko if (doscan) { 3696665484d8SDoug Ambrisko mrsas_get_pd_list(sc); 3697665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_AEN, "scanning ...sim 1\n"); 3698665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_1); 3699665484d8SDoug Ambrisko mrsas_get_ld_list(sc); 3700665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_AEN, "scanning ...sim 0\n"); 3701665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_0); 3702665484d8SDoug Ambrisko } 3703665484d8SDoug Ambrisko 3704665484d8SDoug Ambrisko seq_num = sc->evt_detail_mem->seq_num + 1; 3705665484d8SDoug Ambrisko 3706665484d8SDoug Ambrisko // Register AEN with FW for latest sequence number plus 1 3707665484d8SDoug Ambrisko class_locale.members.reserved = 0; 3708665484d8SDoug Ambrisko class_locale.members.locale = MR_EVT_LOCALE_ALL; 3709665484d8SDoug Ambrisko class_locale.members.class = MR_EVT_CLASS_DEBUG; 3710665484d8SDoug Ambrisko 3711665484d8SDoug Ambrisko if (sc->aen_cmd != NULL ) 3712665484d8SDoug Ambrisko return ; 3713665484d8SDoug Ambrisko 3714665484d8SDoug Ambrisko mtx_lock(&sc->aen_lock); 3715665484d8SDoug Ambrisko error = mrsas_register_aen(sc, seq_num, 3716665484d8SDoug Ambrisko class_locale.word); 3717665484d8SDoug Ambrisko mtx_unlock(&sc->aen_lock); 3718665484d8SDoug Ambrisko 3719665484d8SDoug Ambrisko if (error) 3720665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "register aen failed error %x\n", error); 3721665484d8SDoug Ambrisko 3722665484d8SDoug Ambrisko } 3723665484d8SDoug Ambrisko 3724665484d8SDoug Ambrisko 3725665484d8SDoug Ambrisko /** 3726665484d8SDoug Ambrisko * mrsas_complete_aen: Completes AEN command 3727665484d8SDoug Ambrisko * input: Adapter soft state 3728665484d8SDoug Ambrisko * Cmd that was issued to abort another cmd 3729665484d8SDoug Ambrisko * 3730665484d8SDoug Ambrisko * This function will be called from ISR and will continue 3731665484d8SDoug Ambrisko * event processing from thread context by enqueuing task 3732665484d8SDoug Ambrisko * in ev_tq (callback function "mrsas_aen_handler"). 3733665484d8SDoug Ambrisko */ 3734665484d8SDoug Ambrisko void mrsas_complete_aen(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 3735665484d8SDoug Ambrisko { 3736665484d8SDoug Ambrisko /* 3737665484d8SDoug Ambrisko * Don't signal app if it is just an aborted previously registered aen 3738665484d8SDoug Ambrisko */ 3739665484d8SDoug Ambrisko if ((!cmd->abort_aen) && (sc->remove_in_progress == 0)) { 3740665484d8SDoug Ambrisko /* TO DO (?) */ 3741665484d8SDoug Ambrisko } 3742665484d8SDoug Ambrisko else 3743665484d8SDoug Ambrisko cmd->abort_aen = 0; 3744665484d8SDoug Ambrisko 3745665484d8SDoug Ambrisko sc->aen_cmd = NULL; 3746665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3747665484d8SDoug Ambrisko 3748665484d8SDoug Ambrisko if (!sc->remove_in_progress) 3749665484d8SDoug Ambrisko taskqueue_enqueue(sc->ev_tq, &sc->ev_task); 3750665484d8SDoug Ambrisko 3751665484d8SDoug Ambrisko return; 3752665484d8SDoug Ambrisko } 3753665484d8SDoug Ambrisko 3754665484d8SDoug Ambrisko static device_method_t mrsas_methods[] = { 3755665484d8SDoug Ambrisko DEVMETHOD(device_probe, mrsas_probe), 3756665484d8SDoug Ambrisko DEVMETHOD(device_attach, mrsas_attach), 3757665484d8SDoug Ambrisko DEVMETHOD(device_detach, mrsas_detach), 3758665484d8SDoug Ambrisko DEVMETHOD(device_suspend, mrsas_suspend), 3759665484d8SDoug Ambrisko DEVMETHOD(device_resume, mrsas_resume), 3760665484d8SDoug Ambrisko DEVMETHOD(bus_print_child, bus_generic_print_child), 3761665484d8SDoug Ambrisko DEVMETHOD(bus_driver_added, bus_generic_driver_added), 3762665484d8SDoug Ambrisko { 0, 0 } 3763665484d8SDoug Ambrisko }; 3764665484d8SDoug Ambrisko 3765665484d8SDoug Ambrisko static driver_t mrsas_driver = { 3766665484d8SDoug Ambrisko "mrsas", 3767665484d8SDoug Ambrisko mrsas_methods, 3768665484d8SDoug Ambrisko sizeof(struct mrsas_softc) 3769665484d8SDoug Ambrisko }; 3770665484d8SDoug Ambrisko 3771665484d8SDoug Ambrisko static devclass_t mrsas_devclass; 3772665484d8SDoug Ambrisko DRIVER_MODULE(mrsas, pci, mrsas_driver, mrsas_devclass, 0, 0); 3773665484d8SDoug Ambrisko MODULE_DEPEND(mrsas, cam, 1,1,1); 3774665484d8SDoug Ambrisko 3775