1665484d8SDoug Ambrisko /* 2ecea5be4SKashyap D Desai * Copyright (c) 2015, AVAGO Tech. All rights reserved. Author: Marian Choy 38e727371SKashyap D Desai * Copyright (c) 2014, LSI Corp. All rights reserved. Author: Marian Choy 4ecea5be4SKashyap D Desai * Support: freebsdraid@avagotech.com 5665484d8SDoug Ambrisko * 6665484d8SDoug Ambrisko * Redistribution and use in source and binary forms, with or without 78e727371SKashyap D Desai * modification, are permitted provided that the following conditions are 88e727371SKashyap D Desai * met: 9665484d8SDoug Ambrisko * 108e727371SKashyap D Desai * 1. Redistributions of source code must retain the above copyright notice, 118e727371SKashyap D Desai * this list of conditions and the following disclaimer. 2. Redistributions 128e727371SKashyap D Desai * in binary form must reproduce the above copyright notice, this list of 138e727371SKashyap D Desai * conditions and the following disclaimer in the documentation and/or other 148e727371SKashyap D Desai * materials provided with the distribution. 3. Neither the name of the 158e727371SKashyap D Desai * <ORGANIZATION> nor the names of its contributors may be used to endorse or 168e727371SKashyap D Desai * promote products derived from this software without specific prior written 178e727371SKashyap D Desai * permission. 18665484d8SDoug Ambrisko * 198e727371SKashyap D Desai * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 208e727371SKashyap D Desai * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 218e727371SKashyap D Desai * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 228e727371SKashyap D Desai * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE 238e727371SKashyap D Desai * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 248e727371SKashyap D Desai * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 258e727371SKashyap D Desai * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 268e727371SKashyap D Desai * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 278e727371SKashyap D Desai * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 288e727371SKashyap D Desai * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 29665484d8SDoug Ambrisko * POSSIBILITY OF SUCH DAMAGE. 30665484d8SDoug Ambrisko * 318e727371SKashyap D Desai * The views and conclusions contained in the software and documentation are 328e727371SKashyap D Desai * those of the authors and should not be interpreted as representing 33665484d8SDoug Ambrisko * official policies,either expressed or implied, of the FreeBSD Project. 34665484d8SDoug Ambrisko * 35ecea5be4SKashyap D Desai * Send feedback to: <megaraidfbsd@avagotech.com> Mail to: AVAGO TECHNOLOGIES 1621 368e727371SKashyap D Desai * Barber Lane, Milpitas, CA 95035 ATTN: MegaRaid FreeBSD 37665484d8SDoug Ambrisko * 38665484d8SDoug Ambrisko */ 39665484d8SDoug Ambrisko 40665484d8SDoug Ambrisko #include <sys/cdefs.h> 41665484d8SDoug Ambrisko __FBSDID("$FreeBSD$"); 42665484d8SDoug Ambrisko 43665484d8SDoug Ambrisko #include <dev/mrsas/mrsas.h> 44665484d8SDoug Ambrisko #include <dev/mrsas/mrsas_ioctl.h> 45665484d8SDoug Ambrisko 46665484d8SDoug Ambrisko #include <cam/cam.h> 47665484d8SDoug Ambrisko #include <cam/cam_ccb.h> 48665484d8SDoug Ambrisko 49665484d8SDoug Ambrisko #include <sys/sysctl.h> 50665484d8SDoug Ambrisko #include <sys/types.h> 51665484d8SDoug Ambrisko #include <sys/kthread.h> 52665484d8SDoug Ambrisko #include <sys/taskqueue.h> 53d18d1b47SKashyap D Desai #include <sys/smp.h> 54665484d8SDoug Ambrisko 55665484d8SDoug Ambrisko 56665484d8SDoug Ambrisko /* 57665484d8SDoug Ambrisko * Function prototypes 58665484d8SDoug Ambrisko */ 59665484d8SDoug Ambrisko static d_open_t mrsas_open; 60665484d8SDoug Ambrisko static d_close_t mrsas_close; 61665484d8SDoug Ambrisko static d_read_t mrsas_read; 62665484d8SDoug Ambrisko static d_write_t mrsas_write; 63665484d8SDoug Ambrisko static d_ioctl_t mrsas_ioctl; 64da011113SKashyap D Desai static d_poll_t mrsas_poll; 65665484d8SDoug Ambrisko 66536094dcSKashyap D Desai static struct mrsas_mgmt_info mrsas_mgmt_info; 67665484d8SDoug Ambrisko static struct mrsas_ident *mrsas_find_ident(device_t); 68d18d1b47SKashyap D Desai static int mrsas_setup_msix(struct mrsas_softc *sc); 69d18d1b47SKashyap D Desai static int mrsas_allocate_msix(struct mrsas_softc *sc); 70665484d8SDoug Ambrisko static void mrsas_shutdown_ctlr(struct mrsas_softc *sc, u_int32_t opcode); 71665484d8SDoug Ambrisko static void mrsas_flush_cache(struct mrsas_softc *sc); 72665484d8SDoug Ambrisko static void mrsas_reset_reply_desc(struct mrsas_softc *sc); 73665484d8SDoug Ambrisko static void mrsas_ocr_thread(void *arg); 74665484d8SDoug Ambrisko static int mrsas_get_map_info(struct mrsas_softc *sc); 75665484d8SDoug Ambrisko static int mrsas_get_ld_map_info(struct mrsas_softc *sc); 76665484d8SDoug Ambrisko static int mrsas_sync_map_info(struct mrsas_softc *sc); 77665484d8SDoug Ambrisko static int mrsas_get_pd_list(struct mrsas_softc *sc); 78665484d8SDoug Ambrisko static int mrsas_get_ld_list(struct mrsas_softc *sc); 79665484d8SDoug Ambrisko static int mrsas_setup_irq(struct mrsas_softc *sc); 80665484d8SDoug Ambrisko static int mrsas_alloc_mem(struct mrsas_softc *sc); 81665484d8SDoug Ambrisko static int mrsas_init_fw(struct mrsas_softc *sc); 82665484d8SDoug Ambrisko static int mrsas_setup_raidmap(struct mrsas_softc *sc); 83d18d1b47SKashyap D Desai static int mrsas_complete_cmd(struct mrsas_softc *sc, u_int32_t MSIxIndex); 84665484d8SDoug Ambrisko static int mrsas_clear_intr(struct mrsas_softc *sc); 85af51c29fSKashyap D Desai static int mrsas_get_ctrl_info(struct mrsas_softc *sc); 86af51c29fSKashyap D Desai static void mrsas_update_ext_vd_details(struct mrsas_softc *sc); 878e727371SKashyap D Desai static int 888e727371SKashyap D Desai mrsas_issue_blocked_abort_cmd(struct mrsas_softc *sc, 89665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd_to_abort); 90*dbcc81dfSKashyap D Desai static struct mrsas_softc * 91*dbcc81dfSKashyap D Desai mrsas_get_softc_instance(struct cdev *dev, 925844115eSKashyap D Desai u_long cmd, caddr_t arg); 93665484d8SDoug Ambrisko u_int32_t mrsas_read_reg(struct mrsas_softc *sc, int offset); 948e727371SKashyap D Desai u_int8_t 958e727371SKashyap D Desai mrsas_build_mptmfi_passthru(struct mrsas_softc *sc, 96665484d8SDoug Ambrisko struct mrsas_mfi_cmd *mfi_cmd); 97daeed973SKashyap D Desai void mrsas_complete_outstanding_ioctls(struct mrsas_softc *sc); 98665484d8SDoug Ambrisko int mrsas_transition_to_ready(struct mrsas_softc *sc, int ocr); 99665484d8SDoug Ambrisko int mrsas_init_adapter(struct mrsas_softc *sc); 100665484d8SDoug Ambrisko int mrsas_alloc_mpt_cmds(struct mrsas_softc *sc); 101665484d8SDoug Ambrisko int mrsas_alloc_ioc_cmd(struct mrsas_softc *sc); 102665484d8SDoug Ambrisko int mrsas_alloc_ctlr_info_cmd(struct mrsas_softc *sc); 103665484d8SDoug Ambrisko int mrsas_ioc_init(struct mrsas_softc *sc); 104665484d8SDoug Ambrisko int mrsas_bus_scan(struct mrsas_softc *sc); 105665484d8SDoug Ambrisko int mrsas_issue_dcmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 106665484d8SDoug Ambrisko int mrsas_issue_polled(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 107665484d8SDoug Ambrisko int mrsas_reset_ctrl(struct mrsas_softc *sc); 108665484d8SDoug Ambrisko int mrsas_wait_for_outstanding(struct mrsas_softc *sc); 1098e727371SKashyap D Desai int 1108e727371SKashyap D Desai mrsas_issue_blocked_cmd(struct mrsas_softc *sc, 111665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd); 1128e727371SKashyap D Desai int 1138e727371SKashyap D Desai mrsas_alloc_tmp_dcmd(struct mrsas_softc *sc, struct mrsas_tmp_dcmd *tcmd, 114665484d8SDoug Ambrisko int size); 115665484d8SDoug Ambrisko void mrsas_release_mfi_cmd(struct mrsas_mfi_cmd *cmd); 116665484d8SDoug Ambrisko void mrsas_wakeup(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 117665484d8SDoug Ambrisko void mrsas_complete_aen(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 118665484d8SDoug Ambrisko void mrsas_complete_abort(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 119665484d8SDoug Ambrisko void mrsas_disable_intr(struct mrsas_softc *sc); 120665484d8SDoug Ambrisko void mrsas_enable_intr(struct mrsas_softc *sc); 121665484d8SDoug Ambrisko void mrsas_free_ioc_cmd(struct mrsas_softc *sc); 122665484d8SDoug Ambrisko void mrsas_free_mem(struct mrsas_softc *sc); 123665484d8SDoug Ambrisko void mrsas_free_tmp_dcmd(struct mrsas_tmp_dcmd *tmp); 124665484d8SDoug Ambrisko void mrsas_isr(void *arg); 125665484d8SDoug Ambrisko void mrsas_teardown_intr(struct mrsas_softc *sc); 126665484d8SDoug Ambrisko void mrsas_addr_cb(void *arg, bus_dma_segment_t *segs, int nsegs, int error); 127665484d8SDoug Ambrisko void mrsas_kill_hba(struct mrsas_softc *sc); 128665484d8SDoug Ambrisko void mrsas_aen_handler(struct mrsas_softc *sc); 1298e727371SKashyap D Desai void 1308e727371SKashyap D Desai mrsas_write_reg(struct mrsas_softc *sc, int offset, 131665484d8SDoug Ambrisko u_int32_t value); 1328e727371SKashyap D Desai void 1338e727371SKashyap D Desai mrsas_fire_cmd(struct mrsas_softc *sc, u_int32_t req_desc_lo, 134665484d8SDoug Ambrisko u_int32_t req_desc_hi); 135665484d8SDoug Ambrisko void mrsas_free_ctlr_info_cmd(struct mrsas_softc *sc); 1368e727371SKashyap D Desai void 1378e727371SKashyap D Desai mrsas_complete_mptmfi_passthru(struct mrsas_softc *sc, 138665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd, u_int8_t status); 1398e727371SKashyap D Desai void 1408e727371SKashyap D Desai mrsas_map_mpt_cmd_status(struct mrsas_mpt_cmd *cmd, u_int8_t status, 141665484d8SDoug Ambrisko u_int8_t extStatus); 142665484d8SDoug Ambrisko struct mrsas_mfi_cmd *mrsas_get_mfi_cmd(struct mrsas_softc *sc); 1438e727371SKashyap D Desai 1448e727371SKashyap D Desai MRSAS_REQUEST_DESCRIPTOR_UNION *mrsas_build_mpt_cmd 1458e727371SKashyap D Desai (struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 146665484d8SDoug Ambrisko 147665484d8SDoug Ambrisko extern int mrsas_cam_attach(struct mrsas_softc *sc); 148665484d8SDoug Ambrisko extern void mrsas_cam_detach(struct mrsas_softc *sc); 149665484d8SDoug Ambrisko extern void mrsas_cmd_done(struct mrsas_softc *sc, struct mrsas_mpt_cmd *cmd); 150665484d8SDoug Ambrisko extern void mrsas_free_frame(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 151665484d8SDoug Ambrisko extern int mrsas_alloc_mfi_cmds(struct mrsas_softc *sc); 152665484d8SDoug Ambrisko extern void mrsas_release_mpt_cmd(struct mrsas_mpt_cmd *cmd); 153665484d8SDoug Ambrisko extern struct mrsas_mpt_cmd *mrsas_get_mpt_cmd(struct mrsas_softc *sc); 154536094dcSKashyap D Desai extern int mrsas_passthru(struct mrsas_softc *sc, void *arg, u_long ioctlCmd); 155665484d8SDoug Ambrisko extern uint8_t MR_ValidateMapInfo(struct mrsas_softc *sc); 1564799d485SKashyap D Desai extern u_int16_t MR_GetLDTgtId(u_int32_t ld, MR_DRV_RAID_MAP_ALL * map); 1574799d485SKashyap D Desai extern MR_LD_RAID *MR_LdRaidGet(u_int32_t ld, MR_DRV_RAID_MAP_ALL * map); 158665484d8SDoug Ambrisko extern void mrsas_xpt_freeze(struct mrsas_softc *sc); 159665484d8SDoug Ambrisko extern void mrsas_xpt_release(struct mrsas_softc *sc); 1608e727371SKashyap D Desai extern MRSAS_REQUEST_DESCRIPTOR_UNION * 1618e727371SKashyap D Desai mrsas_get_request_desc(struct mrsas_softc *sc, 162665484d8SDoug Ambrisko u_int16_t index); 163665484d8SDoug Ambrisko extern int mrsas_bus_scan_sim(struct mrsas_softc *sc, struct cam_sim *sim); 164665484d8SDoug Ambrisko static int mrsas_alloc_evt_log_info_cmd(struct mrsas_softc *sc); 165665484d8SDoug Ambrisko static void mrsas_free_evt_log_info_cmd(struct mrsas_softc *sc); 1668e727371SKashyap D Desai 167665484d8SDoug Ambrisko SYSCTL_NODE(_hw, OID_AUTO, mrsas, CTLFLAG_RD, 0, "MRSAS Driver Parameters"); 168665484d8SDoug Ambrisko 1698e727371SKashyap D Desai /* 170665484d8SDoug Ambrisko * PCI device struct and table 171665484d8SDoug Ambrisko * 172665484d8SDoug Ambrisko */ 173665484d8SDoug Ambrisko typedef struct mrsas_ident { 174665484d8SDoug Ambrisko uint16_t vendor; 175665484d8SDoug Ambrisko uint16_t device; 176665484d8SDoug Ambrisko uint16_t subvendor; 177665484d8SDoug Ambrisko uint16_t subdevice; 178665484d8SDoug Ambrisko const char *desc; 179665484d8SDoug Ambrisko } MRSAS_CTLR_ID; 180665484d8SDoug Ambrisko 181665484d8SDoug Ambrisko MRSAS_CTLR_ID device_table[] = { 182ecea5be4SKashyap D Desai {0x1000, MRSAS_TBOLT, 0xffff, 0xffff, "AVAGO Thunderbolt SAS Controller"}, 183ecea5be4SKashyap D Desai {0x1000, MRSAS_INVADER, 0xffff, 0xffff, "AVAGO Invader SAS Controller"}, 184ecea5be4SKashyap D Desai {0x1000, MRSAS_FURY, 0xffff, 0xffff, "AVAGO Fury SAS Controller"}, 185665484d8SDoug Ambrisko {0, 0, 0, 0, NULL} 186665484d8SDoug Ambrisko }; 187665484d8SDoug Ambrisko 1888e727371SKashyap D Desai /* 189665484d8SDoug Ambrisko * Character device entry points 190665484d8SDoug Ambrisko * 191665484d8SDoug Ambrisko */ 192665484d8SDoug Ambrisko static struct cdevsw mrsas_cdevsw = { 193665484d8SDoug Ambrisko .d_version = D_VERSION, 194665484d8SDoug Ambrisko .d_open = mrsas_open, 195665484d8SDoug Ambrisko .d_close = mrsas_close, 196665484d8SDoug Ambrisko .d_read = mrsas_read, 197665484d8SDoug Ambrisko .d_write = mrsas_write, 198665484d8SDoug Ambrisko .d_ioctl = mrsas_ioctl, 199da011113SKashyap D Desai .d_poll = mrsas_poll, 200665484d8SDoug Ambrisko .d_name = "mrsas", 201665484d8SDoug Ambrisko }; 202665484d8SDoug Ambrisko 203665484d8SDoug Ambrisko MALLOC_DEFINE(M_MRSAS, "mrsasbuf", "Buffers for the MRSAS driver"); 204665484d8SDoug Ambrisko 2058e727371SKashyap D Desai /* 2068e727371SKashyap D Desai * In the cdevsw routines, we find our softc by using the si_drv1 member of 2078e727371SKashyap D Desai * struct cdev. We set this variable to point to our softc in our attach 2088e727371SKashyap D Desai * routine when we create the /dev entry. 209665484d8SDoug Ambrisko */ 210665484d8SDoug Ambrisko int 2117fc5f329SJohn Baldwin mrsas_open(struct cdev *dev, int oflags, int devtype, struct thread *td) 212665484d8SDoug Ambrisko { 213665484d8SDoug Ambrisko struct mrsas_softc *sc; 214665484d8SDoug Ambrisko 215665484d8SDoug Ambrisko sc = dev->si_drv1; 216665484d8SDoug Ambrisko return (0); 217665484d8SDoug Ambrisko } 218665484d8SDoug Ambrisko 219665484d8SDoug Ambrisko int 2207fc5f329SJohn Baldwin mrsas_close(struct cdev *dev, int fflag, int devtype, struct thread *td) 221665484d8SDoug Ambrisko { 222665484d8SDoug Ambrisko struct mrsas_softc *sc; 223665484d8SDoug Ambrisko 224665484d8SDoug Ambrisko sc = dev->si_drv1; 225665484d8SDoug Ambrisko return (0); 226665484d8SDoug Ambrisko } 227665484d8SDoug Ambrisko 228665484d8SDoug Ambrisko int 229665484d8SDoug Ambrisko mrsas_read(struct cdev *dev, struct uio *uio, int ioflag) 230665484d8SDoug Ambrisko { 231665484d8SDoug Ambrisko struct mrsas_softc *sc; 232665484d8SDoug Ambrisko 233665484d8SDoug Ambrisko sc = dev->si_drv1; 234665484d8SDoug Ambrisko return (0); 235665484d8SDoug Ambrisko } 236665484d8SDoug Ambrisko int 237665484d8SDoug Ambrisko mrsas_write(struct cdev *dev, struct uio *uio, int ioflag) 238665484d8SDoug Ambrisko { 239665484d8SDoug Ambrisko struct mrsas_softc *sc; 240665484d8SDoug Ambrisko 241665484d8SDoug Ambrisko sc = dev->si_drv1; 242665484d8SDoug Ambrisko return (0); 243665484d8SDoug Ambrisko } 244665484d8SDoug Ambrisko 2458e727371SKashyap D Desai /* 246665484d8SDoug Ambrisko * Register Read/Write Functions 247665484d8SDoug Ambrisko * 248665484d8SDoug Ambrisko */ 249665484d8SDoug Ambrisko void 250665484d8SDoug Ambrisko mrsas_write_reg(struct mrsas_softc *sc, int offset, 251665484d8SDoug Ambrisko u_int32_t value) 252665484d8SDoug Ambrisko { 253665484d8SDoug Ambrisko bus_space_tag_t bus_tag = sc->bus_tag; 254665484d8SDoug Ambrisko bus_space_handle_t bus_handle = sc->bus_handle; 255665484d8SDoug Ambrisko 256665484d8SDoug Ambrisko bus_space_write_4(bus_tag, bus_handle, offset, value); 257665484d8SDoug Ambrisko } 258665484d8SDoug Ambrisko 259665484d8SDoug Ambrisko u_int32_t 260665484d8SDoug Ambrisko mrsas_read_reg(struct mrsas_softc *sc, int offset) 261665484d8SDoug Ambrisko { 262665484d8SDoug Ambrisko bus_space_tag_t bus_tag = sc->bus_tag; 263665484d8SDoug Ambrisko bus_space_handle_t bus_handle = sc->bus_handle; 264665484d8SDoug Ambrisko 265665484d8SDoug Ambrisko return ((u_int32_t)bus_space_read_4(bus_tag, bus_handle, offset)); 266665484d8SDoug Ambrisko } 267665484d8SDoug Ambrisko 268665484d8SDoug Ambrisko 2698e727371SKashyap D Desai /* 270665484d8SDoug Ambrisko * Interrupt Disable/Enable/Clear Functions 271665484d8SDoug Ambrisko * 272665484d8SDoug Ambrisko */ 2738e727371SKashyap D Desai void 2748e727371SKashyap D Desai mrsas_disable_intr(struct mrsas_softc *sc) 275665484d8SDoug Ambrisko { 276665484d8SDoug Ambrisko u_int32_t mask = 0xFFFFFFFF; 277665484d8SDoug Ambrisko u_int32_t status; 278665484d8SDoug Ambrisko 2792f863eb8SKashyap D Desai sc->mask_interrupts = 1; 280665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask), mask); 281665484d8SDoug Ambrisko /* Dummy read to force pci flush */ 282665484d8SDoug Ambrisko status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask)); 283665484d8SDoug Ambrisko } 284665484d8SDoug Ambrisko 2858e727371SKashyap D Desai void 2868e727371SKashyap D Desai mrsas_enable_intr(struct mrsas_softc *sc) 287665484d8SDoug Ambrisko { 288665484d8SDoug Ambrisko u_int32_t mask = MFI_FUSION_ENABLE_INTERRUPT_MASK; 289665484d8SDoug Ambrisko u_int32_t status; 290665484d8SDoug Ambrisko 2912f863eb8SKashyap D Desai sc->mask_interrupts = 0; 292665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status), ~0); 293665484d8SDoug Ambrisko status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status)); 294665484d8SDoug Ambrisko 295665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask), ~mask); 296665484d8SDoug Ambrisko status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask)); 297665484d8SDoug Ambrisko } 298665484d8SDoug Ambrisko 2998e727371SKashyap D Desai static int 3008e727371SKashyap D Desai mrsas_clear_intr(struct mrsas_softc *sc) 301665484d8SDoug Ambrisko { 302665484d8SDoug Ambrisko u_int32_t status, fw_status, fw_state; 303665484d8SDoug Ambrisko 304665484d8SDoug Ambrisko /* Read received interrupt */ 305665484d8SDoug Ambrisko status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status)); 306665484d8SDoug Ambrisko 3078e727371SKashyap D Desai /* 3088e727371SKashyap D Desai * If FW state change interrupt is received, write to it again to 3098e727371SKashyap D Desai * clear 3108e727371SKashyap D Desai */ 311665484d8SDoug Ambrisko if (status & MRSAS_FW_STATE_CHNG_INTERRUPT) { 312665484d8SDoug Ambrisko fw_status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 313665484d8SDoug Ambrisko outbound_scratch_pad)); 314665484d8SDoug Ambrisko fw_state = fw_status & MFI_STATE_MASK; 315665484d8SDoug Ambrisko if (fw_state == MFI_STATE_FAULT) { 316665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "FW is in FAULT state!\n"); 317665484d8SDoug Ambrisko if (sc->ocr_thread_active) 318665484d8SDoug Ambrisko wakeup(&sc->ocr_chan); 319665484d8SDoug Ambrisko } 320665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status), status); 321665484d8SDoug Ambrisko mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status)); 322665484d8SDoug Ambrisko return (1); 323665484d8SDoug Ambrisko } 324665484d8SDoug Ambrisko /* Not our interrupt, so just return */ 325665484d8SDoug Ambrisko if (!(status & MFI_FUSION_ENABLE_INTERRUPT_MASK)) 326665484d8SDoug Ambrisko return (0); 327665484d8SDoug Ambrisko 328665484d8SDoug Ambrisko /* We got a reply interrupt */ 329665484d8SDoug Ambrisko return (1); 330665484d8SDoug Ambrisko } 331665484d8SDoug Ambrisko 3328e727371SKashyap D Desai /* 333665484d8SDoug Ambrisko * PCI Support Functions 334665484d8SDoug Ambrisko * 335665484d8SDoug Ambrisko */ 3368e727371SKashyap D Desai static struct mrsas_ident * 3378e727371SKashyap D Desai mrsas_find_ident(device_t dev) 338665484d8SDoug Ambrisko { 339665484d8SDoug Ambrisko struct mrsas_ident *pci_device; 340665484d8SDoug Ambrisko 3418e727371SKashyap D Desai for (pci_device = device_table; pci_device->vendor != 0; pci_device++) { 342665484d8SDoug Ambrisko if ((pci_device->vendor == pci_get_vendor(dev)) && 343665484d8SDoug Ambrisko (pci_device->device == pci_get_device(dev)) && 344665484d8SDoug Ambrisko ((pci_device->subvendor == pci_get_subvendor(dev)) || 345665484d8SDoug Ambrisko (pci_device->subvendor == 0xffff)) && 346665484d8SDoug Ambrisko ((pci_device->subdevice == pci_get_subdevice(dev)) || 347665484d8SDoug Ambrisko (pci_device->subdevice == 0xffff))) 348665484d8SDoug Ambrisko return (pci_device); 349665484d8SDoug Ambrisko } 350665484d8SDoug Ambrisko return (NULL); 351665484d8SDoug Ambrisko } 352665484d8SDoug Ambrisko 3538e727371SKashyap D Desai static int 3548e727371SKashyap D Desai mrsas_probe(device_t dev) 355665484d8SDoug Ambrisko { 356665484d8SDoug Ambrisko static u_int8_t first_ctrl = 1; 357665484d8SDoug Ambrisko struct mrsas_ident *id; 358665484d8SDoug Ambrisko 359665484d8SDoug Ambrisko if ((id = mrsas_find_ident(dev)) != NULL) { 360665484d8SDoug Ambrisko if (first_ctrl) { 361ecea5be4SKashyap D Desai printf("AVAGO MegaRAID SAS FreeBSD mrsas driver version: %s\n", 3628e727371SKashyap D Desai MRSAS_VERSION); 363665484d8SDoug Ambrisko first_ctrl = 0; 364665484d8SDoug Ambrisko } 365665484d8SDoug Ambrisko device_set_desc(dev, id->desc); 366665484d8SDoug Ambrisko /* between BUS_PROBE_DEFAULT and BUS_PROBE_LOW_PRIORITY */ 367665484d8SDoug Ambrisko return (-30); 368665484d8SDoug Ambrisko } 369665484d8SDoug Ambrisko return (ENXIO); 370665484d8SDoug Ambrisko } 371665484d8SDoug Ambrisko 3728e727371SKashyap D Desai /* 373665484d8SDoug Ambrisko * mrsas_setup_sysctl: setup sysctl values for mrsas 374665484d8SDoug Ambrisko * input: Adapter instance soft state 375665484d8SDoug Ambrisko * 376665484d8SDoug Ambrisko * Setup sysctl entries for mrsas driver. 377665484d8SDoug Ambrisko */ 378665484d8SDoug Ambrisko static void 379665484d8SDoug Ambrisko mrsas_setup_sysctl(struct mrsas_softc *sc) 380665484d8SDoug Ambrisko { 381665484d8SDoug Ambrisko struct sysctl_ctx_list *sysctl_ctx = NULL; 382665484d8SDoug Ambrisko struct sysctl_oid *sysctl_tree = NULL; 383665484d8SDoug Ambrisko char tmpstr[80], tmpstr2[80]; 384665484d8SDoug Ambrisko 385665484d8SDoug Ambrisko /* 386665484d8SDoug Ambrisko * Setup the sysctl variable so the user can change the debug level 387665484d8SDoug Ambrisko * on the fly. 388665484d8SDoug Ambrisko */ 389665484d8SDoug Ambrisko snprintf(tmpstr, sizeof(tmpstr), "MRSAS controller %d", 390665484d8SDoug Ambrisko device_get_unit(sc->mrsas_dev)); 391665484d8SDoug Ambrisko snprintf(tmpstr2, sizeof(tmpstr2), "%d", device_get_unit(sc->mrsas_dev)); 392665484d8SDoug Ambrisko 393665484d8SDoug Ambrisko sysctl_ctx = device_get_sysctl_ctx(sc->mrsas_dev); 394665484d8SDoug Ambrisko if (sysctl_ctx != NULL) 395665484d8SDoug Ambrisko sysctl_tree = device_get_sysctl_tree(sc->mrsas_dev); 396665484d8SDoug Ambrisko 397665484d8SDoug Ambrisko if (sysctl_tree == NULL) { 398665484d8SDoug Ambrisko sysctl_ctx_init(&sc->sysctl_ctx); 399665484d8SDoug Ambrisko sc->sysctl_tree = SYSCTL_ADD_NODE(&sc->sysctl_ctx, 400665484d8SDoug Ambrisko SYSCTL_STATIC_CHILDREN(_hw_mrsas), OID_AUTO, tmpstr2, 401665484d8SDoug Ambrisko CTLFLAG_RD, 0, tmpstr); 402665484d8SDoug Ambrisko if (sc->sysctl_tree == NULL) 403665484d8SDoug Ambrisko return; 404665484d8SDoug Ambrisko sysctl_ctx = &sc->sysctl_ctx; 405665484d8SDoug Ambrisko sysctl_tree = sc->sysctl_tree; 406665484d8SDoug Ambrisko } 407665484d8SDoug Ambrisko SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 408665484d8SDoug Ambrisko OID_AUTO, "disable_ocr", CTLFLAG_RW, &sc->disableOnlineCtrlReset, 0, 409665484d8SDoug Ambrisko "Disable the use of OCR"); 410665484d8SDoug Ambrisko 411665484d8SDoug Ambrisko SYSCTL_ADD_STRING(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 412665484d8SDoug Ambrisko OID_AUTO, "driver_version", CTLFLAG_RD, MRSAS_VERSION, 413665484d8SDoug Ambrisko strlen(MRSAS_VERSION), "driver version"); 414665484d8SDoug Ambrisko 415665484d8SDoug Ambrisko SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 416665484d8SDoug Ambrisko OID_AUTO, "reset_count", CTLFLAG_RD, 417665484d8SDoug Ambrisko &sc->reset_count, 0, "number of ocr from start of the day"); 418665484d8SDoug Ambrisko 419665484d8SDoug Ambrisko SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 420665484d8SDoug Ambrisko OID_AUTO, "fw_outstanding", CTLFLAG_RD, 421f0188618SHans Petter Selasky &sc->fw_outstanding.val_rdonly, 0, "FW outstanding commands"); 422665484d8SDoug Ambrisko 423665484d8SDoug Ambrisko SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 424665484d8SDoug Ambrisko OID_AUTO, "io_cmds_highwater", CTLFLAG_RD, 425665484d8SDoug Ambrisko &sc->io_cmds_highwater, 0, "Max FW outstanding commands"); 426665484d8SDoug Ambrisko 427665484d8SDoug Ambrisko SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 428665484d8SDoug Ambrisko OID_AUTO, "mrsas_debug", CTLFLAG_RW, &sc->mrsas_debug, 0, 429665484d8SDoug Ambrisko "Driver debug level"); 430665484d8SDoug Ambrisko 431665484d8SDoug Ambrisko SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 432665484d8SDoug Ambrisko OID_AUTO, "mrsas_io_timeout", CTLFLAG_RW, &sc->mrsas_io_timeout, 433665484d8SDoug Ambrisko 0, "Driver IO timeout value in mili-second."); 434665484d8SDoug Ambrisko 435665484d8SDoug Ambrisko SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 436665484d8SDoug Ambrisko OID_AUTO, "mrsas_fw_fault_check_delay", CTLFLAG_RW, 437665484d8SDoug Ambrisko &sc->mrsas_fw_fault_check_delay, 438665484d8SDoug Ambrisko 0, "FW fault check thread delay in seconds. <default is 1 sec>"); 439665484d8SDoug Ambrisko 440665484d8SDoug Ambrisko SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 441665484d8SDoug Ambrisko OID_AUTO, "reset_in_progress", CTLFLAG_RD, 442665484d8SDoug Ambrisko &sc->reset_in_progress, 0, "ocr in progress status"); 443665484d8SDoug Ambrisko 444665484d8SDoug Ambrisko } 445665484d8SDoug Ambrisko 4468e727371SKashyap D Desai /* 447665484d8SDoug Ambrisko * mrsas_get_tunables: get tunable parameters. 448665484d8SDoug Ambrisko * input: Adapter instance soft state 449665484d8SDoug Ambrisko * 450665484d8SDoug Ambrisko * Get tunable parameters. This will help to debug driver at boot time. 451665484d8SDoug Ambrisko */ 452665484d8SDoug Ambrisko static void 453665484d8SDoug Ambrisko mrsas_get_tunables(struct mrsas_softc *sc) 454665484d8SDoug Ambrisko { 455665484d8SDoug Ambrisko char tmpstr[80]; 456665484d8SDoug Ambrisko 457665484d8SDoug Ambrisko /* XXX default to some debugging for now */ 458665484d8SDoug Ambrisko sc->mrsas_debug = MRSAS_FAULT; 459665484d8SDoug Ambrisko sc->mrsas_io_timeout = MRSAS_IO_TIMEOUT; 460665484d8SDoug Ambrisko sc->mrsas_fw_fault_check_delay = 1; 461665484d8SDoug Ambrisko sc->reset_count = 0; 462665484d8SDoug Ambrisko sc->reset_in_progress = 0; 463665484d8SDoug Ambrisko 464665484d8SDoug Ambrisko /* 465665484d8SDoug Ambrisko * Grab the global variables. 466665484d8SDoug Ambrisko */ 467665484d8SDoug Ambrisko TUNABLE_INT_FETCH("hw.mrsas.debug_level", &sc->mrsas_debug); 468665484d8SDoug Ambrisko 46916dc2814SKashyap D Desai /* 47016dc2814SKashyap D Desai * Grab the global variables. 47116dc2814SKashyap D Desai */ 47216dc2814SKashyap D Desai TUNABLE_INT_FETCH("hw.mrsas.lb_pending_cmds", &sc->lb_pending_cmds); 47316dc2814SKashyap D Desai 474665484d8SDoug Ambrisko /* Grab the unit-instance variables */ 475665484d8SDoug Ambrisko snprintf(tmpstr, sizeof(tmpstr), "dev.mrsas.%d.debug_level", 476665484d8SDoug Ambrisko device_get_unit(sc->mrsas_dev)); 477665484d8SDoug Ambrisko TUNABLE_INT_FETCH(tmpstr, &sc->mrsas_debug); 478665484d8SDoug Ambrisko } 479665484d8SDoug Ambrisko 4808e727371SKashyap D Desai /* 481665484d8SDoug Ambrisko * mrsas_alloc_evt_log_info cmd: Allocates memory to get event log information. 482665484d8SDoug Ambrisko * Used to get sequence number at driver load time. 483665484d8SDoug Ambrisko * input: Adapter soft state 484665484d8SDoug Ambrisko * 485665484d8SDoug Ambrisko * Allocates DMAable memory for the event log info internal command. 486665484d8SDoug Ambrisko */ 4878e727371SKashyap D Desai int 4888e727371SKashyap D Desai mrsas_alloc_evt_log_info_cmd(struct mrsas_softc *sc) 489665484d8SDoug Ambrisko { 490665484d8SDoug Ambrisko int el_info_size; 491665484d8SDoug Ambrisko 492665484d8SDoug Ambrisko /* Allocate get event log info command */ 493665484d8SDoug Ambrisko el_info_size = sizeof(struct mrsas_evt_log_info); 4948e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 4958e727371SKashyap D Desai 1, 0, 4968e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 4978e727371SKashyap D Desai BUS_SPACE_MAXADDR, 4988e727371SKashyap D Desai NULL, NULL, 4998e727371SKashyap D Desai el_info_size, 5008e727371SKashyap D Desai 1, 5018e727371SKashyap D Desai el_info_size, 5028e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 5038e727371SKashyap D Desai NULL, NULL, 504665484d8SDoug Ambrisko &sc->el_info_tag)) { 505665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate event log info tag\n"); 506665484d8SDoug Ambrisko return (ENOMEM); 507665484d8SDoug Ambrisko } 508665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->el_info_tag, (void **)&sc->el_info_mem, 509665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->el_info_dmamap)) { 510665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate event log info cmd mem\n"); 511665484d8SDoug Ambrisko return (ENOMEM); 512665484d8SDoug Ambrisko } 513665484d8SDoug Ambrisko if (bus_dmamap_load(sc->el_info_tag, sc->el_info_dmamap, 514665484d8SDoug Ambrisko sc->el_info_mem, el_info_size, mrsas_addr_cb, 515665484d8SDoug Ambrisko &sc->el_info_phys_addr, BUS_DMA_NOWAIT)) { 516665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load event log info cmd mem\n"); 517665484d8SDoug Ambrisko return (ENOMEM); 518665484d8SDoug Ambrisko } 519665484d8SDoug Ambrisko memset(sc->el_info_mem, 0, el_info_size); 520665484d8SDoug Ambrisko return (0); 521665484d8SDoug Ambrisko } 522665484d8SDoug Ambrisko 5238e727371SKashyap D Desai /* 524665484d8SDoug Ambrisko * mrsas_free_evt_info_cmd: Free memory for Event log info command 525665484d8SDoug Ambrisko * input: Adapter soft state 526665484d8SDoug Ambrisko * 527665484d8SDoug Ambrisko * Deallocates memory for the event log info internal command. 528665484d8SDoug Ambrisko */ 5298e727371SKashyap D Desai void 5308e727371SKashyap D Desai mrsas_free_evt_log_info_cmd(struct mrsas_softc *sc) 531665484d8SDoug Ambrisko { 532665484d8SDoug Ambrisko if (sc->el_info_phys_addr) 533665484d8SDoug Ambrisko bus_dmamap_unload(sc->el_info_tag, sc->el_info_dmamap); 534665484d8SDoug Ambrisko if (sc->el_info_mem != NULL) 535665484d8SDoug Ambrisko bus_dmamem_free(sc->el_info_tag, sc->el_info_mem, sc->el_info_dmamap); 536665484d8SDoug Ambrisko if (sc->el_info_tag != NULL) 537665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->el_info_tag); 538665484d8SDoug Ambrisko } 539665484d8SDoug Ambrisko 5408e727371SKashyap D Desai /* 541665484d8SDoug Ambrisko * mrsas_get_seq_num: Get latest event sequence number 542665484d8SDoug Ambrisko * @sc: Adapter soft state 543665484d8SDoug Ambrisko * @eli: Firmware event log sequence number information. 5448e727371SKashyap D Desai * 545665484d8SDoug Ambrisko * Firmware maintains a log of all events in a non-volatile area. 546665484d8SDoug Ambrisko * Driver get the sequence number using DCMD 547665484d8SDoug Ambrisko * "MR_DCMD_CTRL_EVENT_GET_INFO" at driver load time. 548665484d8SDoug Ambrisko */ 549665484d8SDoug Ambrisko 550665484d8SDoug Ambrisko static int 551665484d8SDoug Ambrisko mrsas_get_seq_num(struct mrsas_softc *sc, 552665484d8SDoug Ambrisko struct mrsas_evt_log_info *eli) 553665484d8SDoug Ambrisko { 554665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 555665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 556665484d8SDoug Ambrisko 557665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 558665484d8SDoug Ambrisko 559665484d8SDoug Ambrisko if (!cmd) { 560665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Failed to get a free cmd\n"); 561665484d8SDoug Ambrisko return -ENOMEM; 562665484d8SDoug Ambrisko } 563665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 564665484d8SDoug Ambrisko 565665484d8SDoug Ambrisko if (mrsas_alloc_evt_log_info_cmd(sc) != SUCCESS) { 566665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate evt log info cmd\n"); 567665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 568665484d8SDoug Ambrisko return -ENOMEM; 569665484d8SDoug Ambrisko } 570665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 571665484d8SDoug Ambrisko 572665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 573665484d8SDoug Ambrisko dcmd->cmd_status = 0x0; 574665484d8SDoug Ambrisko dcmd->sge_count = 1; 575665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 576665484d8SDoug Ambrisko dcmd->timeout = 0; 577665484d8SDoug Ambrisko dcmd->pad_0 = 0; 578665484d8SDoug Ambrisko dcmd->data_xfer_len = sizeof(struct mrsas_evt_log_info); 579665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_CTRL_EVENT_GET_INFO; 580665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = sc->el_info_phys_addr; 581665484d8SDoug Ambrisko dcmd->sgl.sge32[0].length = sizeof(struct mrsas_evt_log_info); 582665484d8SDoug Ambrisko 583665484d8SDoug Ambrisko mrsas_issue_blocked_cmd(sc, cmd); 584665484d8SDoug Ambrisko 585665484d8SDoug Ambrisko /* 586665484d8SDoug Ambrisko * Copy the data back into callers buffer 587665484d8SDoug Ambrisko */ 588665484d8SDoug Ambrisko memcpy(eli, sc->el_info_mem, sizeof(struct mrsas_evt_log_info)); 589665484d8SDoug Ambrisko mrsas_free_evt_log_info_cmd(sc); 590665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 591665484d8SDoug Ambrisko 592665484d8SDoug Ambrisko return 0; 593665484d8SDoug Ambrisko } 594665484d8SDoug Ambrisko 595665484d8SDoug Ambrisko 5968e727371SKashyap D Desai /* 597665484d8SDoug Ambrisko * mrsas_register_aen: Register for asynchronous event notification 598665484d8SDoug Ambrisko * @sc: Adapter soft state 599665484d8SDoug Ambrisko * @seq_num: Starting sequence number 600665484d8SDoug Ambrisko * @class_locale: Class of the event 6018e727371SKashyap D Desai * 602665484d8SDoug Ambrisko * This function subscribes for events beyond the @seq_num 603665484d8SDoug Ambrisko * and type @class_locale. 604665484d8SDoug Ambrisko * 6058e727371SKashyap D Desai */ 606665484d8SDoug Ambrisko static int 607665484d8SDoug Ambrisko mrsas_register_aen(struct mrsas_softc *sc, u_int32_t seq_num, 608665484d8SDoug Ambrisko u_int32_t class_locale_word) 609665484d8SDoug Ambrisko { 610665484d8SDoug Ambrisko int ret_val; 611665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 612665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 613665484d8SDoug Ambrisko union mrsas_evt_class_locale curr_aen; 614665484d8SDoug Ambrisko union mrsas_evt_class_locale prev_aen; 615665484d8SDoug Ambrisko 616665484d8SDoug Ambrisko /* 617665484d8SDoug Ambrisko * If there an AEN pending already (aen_cmd), check if the 6188e727371SKashyap D Desai * class_locale of that pending AEN is inclusive of the new AEN 6198e727371SKashyap D Desai * request we currently have. If it is, then we don't have to do 6208e727371SKashyap D Desai * anything. In other words, whichever events the current AEN request 6218e727371SKashyap D Desai * is subscribing to, have already been subscribed to. If the old_cmd 6228e727371SKashyap D Desai * is _not_ inclusive, then we have to abort that command, form a 6238e727371SKashyap D Desai * class_locale that is superset of both old and current and re-issue 6248e727371SKashyap D Desai * to the FW 6258e727371SKashyap D Desai */ 626665484d8SDoug Ambrisko 627665484d8SDoug Ambrisko curr_aen.word = class_locale_word; 628665484d8SDoug Ambrisko 629665484d8SDoug Ambrisko if (sc->aen_cmd) { 630665484d8SDoug Ambrisko 631665484d8SDoug Ambrisko prev_aen.word = sc->aen_cmd->frame->dcmd.mbox.w[1]; 632665484d8SDoug Ambrisko 633665484d8SDoug Ambrisko /* 634665484d8SDoug Ambrisko * A class whose enum value is smaller is inclusive of all 635665484d8SDoug Ambrisko * higher values. If a PROGRESS (= -1) was previously 636665484d8SDoug Ambrisko * registered, then a new registration requests for higher 637665484d8SDoug Ambrisko * classes need not be sent to FW. They are automatically 6388e727371SKashyap D Desai * included. Locale numbers don't have such hierarchy. They 6398e727371SKashyap D Desai * are bitmap values 640665484d8SDoug Ambrisko */ 641665484d8SDoug Ambrisko if ((prev_aen.members.class <= curr_aen.members.class) && 642665484d8SDoug Ambrisko !((prev_aen.members.locale & curr_aen.members.locale) ^ 643665484d8SDoug Ambrisko curr_aen.members.locale)) { 644665484d8SDoug Ambrisko /* 645665484d8SDoug Ambrisko * Previously issued event registration includes 646665484d8SDoug Ambrisko * current request. Nothing to do. 647665484d8SDoug Ambrisko */ 648665484d8SDoug Ambrisko return 0; 649665484d8SDoug Ambrisko } else { 650665484d8SDoug Ambrisko curr_aen.members.locale |= prev_aen.members.locale; 651665484d8SDoug Ambrisko 652665484d8SDoug Ambrisko if (prev_aen.members.class < curr_aen.members.class) 653665484d8SDoug Ambrisko curr_aen.members.class = prev_aen.members.class; 654665484d8SDoug Ambrisko 655665484d8SDoug Ambrisko sc->aen_cmd->abort_aen = 1; 656665484d8SDoug Ambrisko ret_val = mrsas_issue_blocked_abort_cmd(sc, 657665484d8SDoug Ambrisko sc->aen_cmd); 658665484d8SDoug Ambrisko 659665484d8SDoug Ambrisko if (ret_val) { 660665484d8SDoug Ambrisko printf("mrsas: Failed to abort " 661665484d8SDoug Ambrisko "previous AEN command\n"); 662665484d8SDoug Ambrisko return ret_val; 663665484d8SDoug Ambrisko } 664665484d8SDoug Ambrisko } 665665484d8SDoug Ambrisko } 666665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 667665484d8SDoug Ambrisko 668665484d8SDoug Ambrisko if (!cmd) 669665484d8SDoug Ambrisko return -ENOMEM; 670665484d8SDoug Ambrisko 671665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 672665484d8SDoug Ambrisko 673665484d8SDoug Ambrisko memset(sc->evt_detail_mem, 0, sizeof(struct mrsas_evt_detail)); 674665484d8SDoug Ambrisko 675665484d8SDoug Ambrisko /* 676665484d8SDoug Ambrisko * Prepare DCMD for aen registration 677665484d8SDoug Ambrisko */ 678665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 679665484d8SDoug Ambrisko 680665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 681665484d8SDoug Ambrisko dcmd->cmd_status = 0x0; 682665484d8SDoug Ambrisko dcmd->sge_count = 1; 683665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 684665484d8SDoug Ambrisko dcmd->timeout = 0; 685665484d8SDoug Ambrisko dcmd->pad_0 = 0; 686665484d8SDoug Ambrisko dcmd->data_xfer_len = sizeof(struct mrsas_evt_detail); 687665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_CTRL_EVENT_WAIT; 688665484d8SDoug Ambrisko dcmd->mbox.w[0] = seq_num; 689665484d8SDoug Ambrisko sc->last_seq_num = seq_num; 690665484d8SDoug Ambrisko dcmd->mbox.w[1] = curr_aen.word; 691665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = (u_int32_t)sc->evt_detail_phys_addr; 692665484d8SDoug Ambrisko dcmd->sgl.sge32[0].length = sizeof(struct mrsas_evt_detail); 693665484d8SDoug Ambrisko 694665484d8SDoug Ambrisko if (sc->aen_cmd != NULL) { 695665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 696665484d8SDoug Ambrisko return 0; 697665484d8SDoug Ambrisko } 698665484d8SDoug Ambrisko /* 699665484d8SDoug Ambrisko * Store reference to the cmd used to register for AEN. When an 700665484d8SDoug Ambrisko * application wants us to register for AEN, we have to abort this 701665484d8SDoug Ambrisko * cmd and re-register with a new EVENT LOCALE supplied by that app 702665484d8SDoug Ambrisko */ 703665484d8SDoug Ambrisko sc->aen_cmd = cmd; 704665484d8SDoug Ambrisko 705665484d8SDoug Ambrisko /* 7068e727371SKashyap D Desai * Issue the aen registration frame 707665484d8SDoug Ambrisko */ 708665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)) { 709665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot issue AEN DCMD command.\n"); 710665484d8SDoug Ambrisko return (1); 711665484d8SDoug Ambrisko } 712665484d8SDoug Ambrisko return 0; 713665484d8SDoug Ambrisko } 7148e727371SKashyap D Desai 7158e727371SKashyap D Desai /* 7168e727371SKashyap D Desai * mrsas_start_aen: Subscribes to AEN during driver load time 717665484d8SDoug Ambrisko * @instance: Adapter soft state 718665484d8SDoug Ambrisko */ 7198e727371SKashyap D Desai static int 7208e727371SKashyap D Desai mrsas_start_aen(struct mrsas_softc *sc) 721665484d8SDoug Ambrisko { 722665484d8SDoug Ambrisko struct mrsas_evt_log_info eli; 723665484d8SDoug Ambrisko union mrsas_evt_class_locale class_locale; 724665484d8SDoug Ambrisko 725665484d8SDoug Ambrisko 726665484d8SDoug Ambrisko /* Get the latest sequence number from FW */ 727665484d8SDoug Ambrisko 728665484d8SDoug Ambrisko memset(&eli, 0, sizeof(eli)); 729665484d8SDoug Ambrisko 730665484d8SDoug Ambrisko if (mrsas_get_seq_num(sc, &eli)) 731665484d8SDoug Ambrisko return -1; 732665484d8SDoug Ambrisko 733665484d8SDoug Ambrisko /* Register AEN with FW for latest sequence number plus 1 */ 734665484d8SDoug Ambrisko class_locale.members.reserved = 0; 735665484d8SDoug Ambrisko class_locale.members.locale = MR_EVT_LOCALE_ALL; 736665484d8SDoug Ambrisko class_locale.members.class = MR_EVT_CLASS_DEBUG; 737665484d8SDoug Ambrisko 738665484d8SDoug Ambrisko return mrsas_register_aen(sc, eli.newest_seq_num + 1, 739665484d8SDoug Ambrisko class_locale.word); 740d18d1b47SKashyap D Desai 741665484d8SDoug Ambrisko } 742665484d8SDoug Ambrisko 7438e727371SKashyap D Desai /* 744d18d1b47SKashyap D Desai * mrsas_setup_msix: Allocate MSI-x vectors 7458e727371SKashyap D Desai * @sc: adapter soft state 746d18d1b47SKashyap D Desai */ 7478e727371SKashyap D Desai static int 7488e727371SKashyap D Desai mrsas_setup_msix(struct mrsas_softc *sc) 749d18d1b47SKashyap D Desai { 750d18d1b47SKashyap D Desai int i; 7518e727371SKashyap D Desai 752d18d1b47SKashyap D Desai for (i = 0; i < sc->msix_vectors; i++) { 753d18d1b47SKashyap D Desai sc->irq_context[i].sc = sc; 754d18d1b47SKashyap D Desai sc->irq_context[i].MSIxIndex = i; 755d18d1b47SKashyap D Desai sc->irq_id[i] = i + 1; 756d18d1b47SKashyap D Desai sc->mrsas_irq[i] = bus_alloc_resource_any 757d18d1b47SKashyap D Desai (sc->mrsas_dev, SYS_RES_IRQ, &sc->irq_id[i] 758d18d1b47SKashyap D Desai ,RF_ACTIVE); 759d18d1b47SKashyap D Desai if (sc->mrsas_irq[i] == NULL) { 760d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "Can't allocate MSI-x\n"); 761d18d1b47SKashyap D Desai goto irq_alloc_failed; 762d18d1b47SKashyap D Desai } 763d18d1b47SKashyap D Desai if (bus_setup_intr(sc->mrsas_dev, 764d18d1b47SKashyap D Desai sc->mrsas_irq[i], 765d18d1b47SKashyap D Desai INTR_MPSAFE | INTR_TYPE_CAM, 766d18d1b47SKashyap D Desai NULL, mrsas_isr, &sc->irq_context[i], 767d18d1b47SKashyap D Desai &sc->intr_handle[i])) { 768d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, 769d18d1b47SKashyap D Desai "Cannot set up MSI-x interrupt handler\n"); 770d18d1b47SKashyap D Desai goto irq_alloc_failed; 771d18d1b47SKashyap D Desai } 772d18d1b47SKashyap D Desai } 773d18d1b47SKashyap D Desai return SUCCESS; 774d18d1b47SKashyap D Desai 775d18d1b47SKashyap D Desai irq_alloc_failed: 776d18d1b47SKashyap D Desai mrsas_teardown_intr(sc); 777d18d1b47SKashyap D Desai return (FAIL); 778d18d1b47SKashyap D Desai } 779d18d1b47SKashyap D Desai 7808e727371SKashyap D Desai /* 781d18d1b47SKashyap D Desai * mrsas_allocate_msix: Setup MSI-x vectors 7828e727371SKashyap D Desai * @sc: adapter soft state 783d18d1b47SKashyap D Desai */ 7848e727371SKashyap D Desai static int 7858e727371SKashyap D Desai mrsas_allocate_msix(struct mrsas_softc *sc) 786d18d1b47SKashyap D Desai { 787d18d1b47SKashyap D Desai if (pci_alloc_msix(sc->mrsas_dev, &sc->msix_vectors) == 0) { 788d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "Using MSI-X with %d number" 789d18d1b47SKashyap D Desai " of vectors\n", sc->msix_vectors); 790d18d1b47SKashyap D Desai } else { 791d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "MSI-x setup failed\n"); 792d18d1b47SKashyap D Desai goto irq_alloc_failed; 793d18d1b47SKashyap D Desai } 794d18d1b47SKashyap D Desai return SUCCESS; 795d18d1b47SKashyap D Desai 796d18d1b47SKashyap D Desai irq_alloc_failed: 797d18d1b47SKashyap D Desai mrsas_teardown_intr(sc); 798d18d1b47SKashyap D Desai return (FAIL); 799d18d1b47SKashyap D Desai } 8008e727371SKashyap D Desai 8018e727371SKashyap D Desai /* 802665484d8SDoug Ambrisko * mrsas_attach: PCI entry point 8038e727371SKashyap D Desai * input: pointer to device struct 804665484d8SDoug Ambrisko * 8058e727371SKashyap D Desai * Performs setup of PCI and registers, initializes mutexes and linked lists, 8068e727371SKashyap D Desai * registers interrupts and CAM, and initializes the adapter/controller to 8078e727371SKashyap D Desai * its proper state. 808665484d8SDoug Ambrisko */ 8098e727371SKashyap D Desai static int 8108e727371SKashyap D Desai mrsas_attach(device_t dev) 811665484d8SDoug Ambrisko { 812665484d8SDoug Ambrisko struct mrsas_softc *sc = device_get_softc(dev); 813665484d8SDoug Ambrisko uint32_t cmd, bar, error; 814665484d8SDoug Ambrisko 815665484d8SDoug Ambrisko /* Look up our softc and initialize its fields. */ 816665484d8SDoug Ambrisko sc->mrsas_dev = dev; 817665484d8SDoug Ambrisko sc->device_id = pci_get_device(dev); 818665484d8SDoug Ambrisko 819665484d8SDoug Ambrisko mrsas_get_tunables(sc); 820665484d8SDoug Ambrisko 821665484d8SDoug Ambrisko /* 822665484d8SDoug Ambrisko * Set up PCI and registers 823665484d8SDoug Ambrisko */ 824665484d8SDoug Ambrisko cmd = pci_read_config(dev, PCIR_COMMAND, 2); 825665484d8SDoug Ambrisko if ((cmd & PCIM_CMD_PORTEN) == 0) { 826665484d8SDoug Ambrisko return (ENXIO); 827665484d8SDoug Ambrisko } 828665484d8SDoug Ambrisko /* Force the busmaster enable bit on. */ 829665484d8SDoug Ambrisko cmd |= PCIM_CMD_BUSMASTEREN; 830665484d8SDoug Ambrisko pci_write_config(dev, PCIR_COMMAND, cmd, 2); 831665484d8SDoug Ambrisko 832665484d8SDoug Ambrisko bar = pci_read_config(dev, MRSAS_PCI_BAR1, 4); 833665484d8SDoug Ambrisko 834665484d8SDoug Ambrisko sc->reg_res_id = MRSAS_PCI_BAR1;/* BAR1 offset */ 835665484d8SDoug Ambrisko if ((sc->reg_res = bus_alloc_resource(dev, SYS_RES_MEMORY, 836665484d8SDoug Ambrisko &(sc->reg_res_id), 0, ~0, 1, RF_ACTIVE)) 837665484d8SDoug Ambrisko == NULL) { 838665484d8SDoug Ambrisko device_printf(dev, "Cannot allocate PCI registers\n"); 839665484d8SDoug Ambrisko goto attach_fail; 840665484d8SDoug Ambrisko } 841665484d8SDoug Ambrisko sc->bus_tag = rman_get_bustag(sc->reg_res); 842665484d8SDoug Ambrisko sc->bus_handle = rman_get_bushandle(sc->reg_res); 843665484d8SDoug Ambrisko 844665484d8SDoug Ambrisko /* Intialize mutexes */ 845665484d8SDoug Ambrisko mtx_init(&sc->sim_lock, "mrsas_sim_lock", NULL, MTX_DEF); 846665484d8SDoug Ambrisko mtx_init(&sc->pci_lock, "mrsas_pci_lock", NULL, MTX_DEF); 847665484d8SDoug Ambrisko mtx_init(&sc->io_lock, "mrsas_io_lock", NULL, MTX_DEF); 848665484d8SDoug Ambrisko mtx_init(&sc->aen_lock, "mrsas_aen_lock", NULL, MTX_DEF); 849665484d8SDoug Ambrisko mtx_init(&sc->ioctl_lock, "mrsas_ioctl_lock", NULL, MTX_SPIN); 850665484d8SDoug Ambrisko mtx_init(&sc->mpt_cmd_pool_lock, "mrsas_mpt_cmd_pool_lock", NULL, MTX_DEF); 851665484d8SDoug Ambrisko mtx_init(&sc->mfi_cmd_pool_lock, "mrsas_mfi_cmd_pool_lock", NULL, MTX_DEF); 852665484d8SDoug Ambrisko mtx_init(&sc->raidmap_lock, "mrsas_raidmap_lock", NULL, MTX_DEF); 853665484d8SDoug Ambrisko 8548e727371SKashyap D Desai /* 8558e727371SKashyap D Desai * Intialize a counting Semaphore to take care no. of concurrent 8568e727371SKashyap D Desai * IOCTLs 8578e727371SKashyap D Desai */ 858839ee025SKashyap D Desai sema_init(&sc->ioctl_count_sema, MRSAS_MAX_MFI_CMDS - 5, IOCTL_SEMA_DESCRIPTION); 859839ee025SKashyap D Desai 860665484d8SDoug Ambrisko /* Intialize linked list */ 861665484d8SDoug Ambrisko TAILQ_INIT(&sc->mrsas_mpt_cmd_list_head); 862665484d8SDoug Ambrisko TAILQ_INIT(&sc->mrsas_mfi_cmd_list_head); 863665484d8SDoug Ambrisko 864f5fb2237SKashyap D Desai mrsas_atomic_set(&sc->fw_outstanding, 0); 865665484d8SDoug Ambrisko 866665484d8SDoug Ambrisko sc->io_cmds_highwater = 0; 867665484d8SDoug Ambrisko 868665484d8SDoug Ambrisko /* Create a /dev entry for this device. */ 869665484d8SDoug Ambrisko sc->mrsas_cdev = make_dev(&mrsas_cdevsw, device_get_unit(dev), UID_ROOT, 870665484d8SDoug Ambrisko GID_OPERATOR, (S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP), "mrsas%u", 871665484d8SDoug Ambrisko device_get_unit(dev)); 872536094dcSKashyap D Desai if (device_get_unit(dev) == 0) 873536094dcSKashyap D Desai make_dev_alias(sc->mrsas_cdev, "megaraid_sas_ioctl_node"); 874665484d8SDoug Ambrisko if (sc->mrsas_cdev) 875665484d8SDoug Ambrisko sc->mrsas_cdev->si_drv1 = sc; 876665484d8SDoug Ambrisko 877665484d8SDoug Ambrisko sc->adprecovery = MRSAS_HBA_OPERATIONAL; 878665484d8SDoug Ambrisko sc->UnevenSpanSupport = 0; 879665484d8SDoug Ambrisko 880d18d1b47SKashyap D Desai sc->msix_enable = 0; 881d18d1b47SKashyap D Desai 882665484d8SDoug Ambrisko /* Initialize Firmware */ 883665484d8SDoug Ambrisko if (mrsas_init_fw(sc) != SUCCESS) { 884665484d8SDoug Ambrisko goto attach_fail_fw; 885665484d8SDoug Ambrisko } 886665484d8SDoug Ambrisko /* Register SCSI mid-layer */ 887665484d8SDoug Ambrisko if ((mrsas_cam_attach(sc) != SUCCESS)) { 888665484d8SDoug Ambrisko goto attach_fail_cam; 889665484d8SDoug Ambrisko } 890665484d8SDoug Ambrisko /* Register IRQs */ 891665484d8SDoug Ambrisko if (mrsas_setup_irq(sc) != SUCCESS) { 892665484d8SDoug Ambrisko goto attach_fail_irq; 893665484d8SDoug Ambrisko } 894665484d8SDoug Ambrisko /* Enable Interrupts */ 895665484d8SDoug Ambrisko mrsas_enable_intr(sc); 896665484d8SDoug Ambrisko 897665484d8SDoug Ambrisko error = mrsas_kproc_create(mrsas_ocr_thread, sc, 898665484d8SDoug Ambrisko &sc->ocr_thread, 0, 0, "mrsas_ocr%d", 899665484d8SDoug Ambrisko device_get_unit(sc->mrsas_dev)); 900665484d8SDoug Ambrisko if (error) { 901665484d8SDoug Ambrisko printf("Error %d starting rescan thread\n", error); 902665484d8SDoug Ambrisko goto attach_fail_irq; 903665484d8SDoug Ambrisko } 904665484d8SDoug Ambrisko mrsas_setup_sysctl(sc); 905665484d8SDoug Ambrisko 906665484d8SDoug Ambrisko /* Initiate AEN (Asynchronous Event Notification) */ 907665484d8SDoug Ambrisko 908665484d8SDoug Ambrisko if (mrsas_start_aen(sc)) { 909665484d8SDoug Ambrisko printf("Error: start aen failed\n"); 910665484d8SDoug Ambrisko goto fail_start_aen; 911665484d8SDoug Ambrisko } 912536094dcSKashyap D Desai /* 9138e727371SKashyap D Desai * Add this controller to mrsas_mgmt_info structure so that it can be 9148e727371SKashyap D Desai * exported to management applications 915536094dcSKashyap D Desai */ 916536094dcSKashyap D Desai if (device_get_unit(dev) == 0) 917536094dcSKashyap D Desai memset(&mrsas_mgmt_info, 0, sizeof(mrsas_mgmt_info)); 918536094dcSKashyap D Desai 919536094dcSKashyap D Desai mrsas_mgmt_info.count++; 920536094dcSKashyap D Desai mrsas_mgmt_info.sc_ptr[mrsas_mgmt_info.max_index] = sc; 921536094dcSKashyap D Desai mrsas_mgmt_info.max_index++; 922536094dcSKashyap D Desai 923665484d8SDoug Ambrisko return (0); 924665484d8SDoug Ambrisko 925665484d8SDoug Ambrisko fail_start_aen: 926665484d8SDoug Ambrisko attach_fail_irq: 927665484d8SDoug Ambrisko mrsas_teardown_intr(sc); 928665484d8SDoug Ambrisko attach_fail_cam: 929665484d8SDoug Ambrisko mrsas_cam_detach(sc); 930665484d8SDoug Ambrisko attach_fail_fw: 931d18d1b47SKashyap D Desai /* if MSIX vector is allocated and FW Init FAILED then release MSIX */ 932d18d1b47SKashyap D Desai if (sc->msix_enable == 1) 933d18d1b47SKashyap D Desai pci_release_msi(sc->mrsas_dev); 934665484d8SDoug Ambrisko mrsas_free_mem(sc); 935665484d8SDoug Ambrisko mtx_destroy(&sc->sim_lock); 936665484d8SDoug Ambrisko mtx_destroy(&sc->aen_lock); 937665484d8SDoug Ambrisko mtx_destroy(&sc->pci_lock); 938665484d8SDoug Ambrisko mtx_destroy(&sc->io_lock); 939665484d8SDoug Ambrisko mtx_destroy(&sc->ioctl_lock); 940665484d8SDoug Ambrisko mtx_destroy(&sc->mpt_cmd_pool_lock); 941665484d8SDoug Ambrisko mtx_destroy(&sc->mfi_cmd_pool_lock); 942665484d8SDoug Ambrisko mtx_destroy(&sc->raidmap_lock); 943839ee025SKashyap D Desai /* Destroy the counting semaphore created for Ioctl */ 944839ee025SKashyap D Desai sema_destroy(&sc->ioctl_count_sema); 945665484d8SDoug Ambrisko attach_fail: 946665484d8SDoug Ambrisko destroy_dev(sc->mrsas_cdev); 947665484d8SDoug Ambrisko if (sc->reg_res) { 948665484d8SDoug Ambrisko bus_release_resource(sc->mrsas_dev, SYS_RES_MEMORY, 949665484d8SDoug Ambrisko sc->reg_res_id, sc->reg_res); 950665484d8SDoug Ambrisko } 951665484d8SDoug Ambrisko return (ENXIO); 952665484d8SDoug Ambrisko } 953665484d8SDoug Ambrisko 9548e727371SKashyap D Desai /* 955665484d8SDoug Ambrisko * mrsas_detach: De-allocates and teardown resources 9568e727371SKashyap D Desai * input: pointer to device struct 957665484d8SDoug Ambrisko * 9588e727371SKashyap D Desai * This function is the entry point for device disconnect and detach. 9598e727371SKashyap D Desai * It performs memory de-allocations, shutdown of the controller and various 960665484d8SDoug Ambrisko * teardown and destroy resource functions. 961665484d8SDoug Ambrisko */ 9628e727371SKashyap D Desai static int 9638e727371SKashyap D Desai mrsas_detach(device_t dev) 964665484d8SDoug Ambrisko { 965665484d8SDoug Ambrisko struct mrsas_softc *sc; 966665484d8SDoug Ambrisko int i = 0; 967665484d8SDoug Ambrisko 968665484d8SDoug Ambrisko sc = device_get_softc(dev); 969665484d8SDoug Ambrisko sc->remove_in_progress = 1; 970536094dcSKashyap D Desai 971839ee025SKashyap D Desai /* Destroy the character device so no other IOCTL will be handled */ 972839ee025SKashyap D Desai destroy_dev(sc->mrsas_cdev); 973839ee025SKashyap D Desai 974536094dcSKashyap D Desai /* 975536094dcSKashyap D Desai * Take the instance off the instance array. Note that we will not 976536094dcSKashyap D Desai * decrement the max_index. We let this array be sparse array 977536094dcSKashyap D Desai */ 978536094dcSKashyap D Desai for (i = 0; i < mrsas_mgmt_info.max_index; i++) { 979536094dcSKashyap D Desai if (mrsas_mgmt_info.sc_ptr[i] == sc) { 980536094dcSKashyap D Desai mrsas_mgmt_info.count--; 981536094dcSKashyap D Desai mrsas_mgmt_info.sc_ptr[i] = NULL; 982536094dcSKashyap D Desai break; 983536094dcSKashyap D Desai } 984536094dcSKashyap D Desai } 985536094dcSKashyap D Desai 986665484d8SDoug Ambrisko if (sc->ocr_thread_active) 987665484d8SDoug Ambrisko wakeup(&sc->ocr_chan); 988665484d8SDoug Ambrisko while (sc->reset_in_progress) { 989665484d8SDoug Ambrisko i++; 990665484d8SDoug Ambrisko if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) { 991665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_INFO, 992665484d8SDoug Ambrisko "[%2d]waiting for ocr to be finished\n", i); 993665484d8SDoug Ambrisko } 994665484d8SDoug Ambrisko pause("mr_shutdown", hz); 995665484d8SDoug Ambrisko } 996665484d8SDoug Ambrisko i = 0; 997665484d8SDoug Ambrisko while (sc->ocr_thread_active) { 998665484d8SDoug Ambrisko i++; 999665484d8SDoug Ambrisko if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) { 1000665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_INFO, 1001665484d8SDoug Ambrisko "[%2d]waiting for " 1002665484d8SDoug Ambrisko "mrsas_ocr thread to quit ocr %d\n", i, 1003665484d8SDoug Ambrisko sc->ocr_thread_active); 1004665484d8SDoug Ambrisko } 1005665484d8SDoug Ambrisko pause("mr_shutdown", hz); 1006665484d8SDoug Ambrisko } 1007665484d8SDoug Ambrisko mrsas_flush_cache(sc); 1008665484d8SDoug Ambrisko mrsas_shutdown_ctlr(sc, MR_DCMD_CTRL_SHUTDOWN); 1009665484d8SDoug Ambrisko mrsas_disable_intr(sc); 1010665484d8SDoug Ambrisko mrsas_cam_detach(sc); 1011665484d8SDoug Ambrisko mrsas_teardown_intr(sc); 1012665484d8SDoug Ambrisko mrsas_free_mem(sc); 1013665484d8SDoug Ambrisko mtx_destroy(&sc->sim_lock); 1014665484d8SDoug Ambrisko mtx_destroy(&sc->aen_lock); 1015665484d8SDoug Ambrisko mtx_destroy(&sc->pci_lock); 1016665484d8SDoug Ambrisko mtx_destroy(&sc->io_lock); 1017665484d8SDoug Ambrisko mtx_destroy(&sc->ioctl_lock); 1018665484d8SDoug Ambrisko mtx_destroy(&sc->mpt_cmd_pool_lock); 1019665484d8SDoug Ambrisko mtx_destroy(&sc->mfi_cmd_pool_lock); 1020665484d8SDoug Ambrisko mtx_destroy(&sc->raidmap_lock); 1021839ee025SKashyap D Desai 1022839ee025SKashyap D Desai /* Wait for all the semaphores to be released */ 1023839ee025SKashyap D Desai while (sema_value(&sc->ioctl_count_sema) != (MRSAS_MAX_MFI_CMDS - 5)) 1024839ee025SKashyap D Desai pause("mr_shutdown", hz); 1025839ee025SKashyap D Desai 1026839ee025SKashyap D Desai /* Destroy the counting semaphore created for Ioctl */ 1027839ee025SKashyap D Desai sema_destroy(&sc->ioctl_count_sema); 1028839ee025SKashyap D Desai 1029665484d8SDoug Ambrisko if (sc->reg_res) { 1030665484d8SDoug Ambrisko bus_release_resource(sc->mrsas_dev, 1031665484d8SDoug Ambrisko SYS_RES_MEMORY, sc->reg_res_id, sc->reg_res); 1032665484d8SDoug Ambrisko } 1033665484d8SDoug Ambrisko if (sc->sysctl_tree != NULL) 1034665484d8SDoug Ambrisko sysctl_ctx_free(&sc->sysctl_ctx); 1035839ee025SKashyap D Desai 1036665484d8SDoug Ambrisko return (0); 1037665484d8SDoug Ambrisko } 1038665484d8SDoug Ambrisko 10398e727371SKashyap D Desai /* 1040665484d8SDoug Ambrisko * mrsas_free_mem: Frees allocated memory 1041665484d8SDoug Ambrisko * input: Adapter instance soft state 1042665484d8SDoug Ambrisko * 1043665484d8SDoug Ambrisko * This function is called from mrsas_detach() to free previously allocated 1044665484d8SDoug Ambrisko * memory. 1045665484d8SDoug Ambrisko */ 10468e727371SKashyap D Desai void 10478e727371SKashyap D Desai mrsas_free_mem(struct mrsas_softc *sc) 1048665484d8SDoug Ambrisko { 1049665484d8SDoug Ambrisko int i; 1050665484d8SDoug Ambrisko u_int32_t max_cmd; 1051665484d8SDoug Ambrisko struct mrsas_mfi_cmd *mfi_cmd; 1052665484d8SDoug Ambrisko struct mrsas_mpt_cmd *mpt_cmd; 1053665484d8SDoug Ambrisko 1054665484d8SDoug Ambrisko /* 1055665484d8SDoug Ambrisko * Free RAID map memory 1056665484d8SDoug Ambrisko */ 10578e727371SKashyap D Desai for (i = 0; i < 2; i++) { 1058665484d8SDoug Ambrisko if (sc->raidmap_phys_addr[i]) 1059665484d8SDoug Ambrisko bus_dmamap_unload(sc->raidmap_tag[i], sc->raidmap_dmamap[i]); 1060665484d8SDoug Ambrisko if (sc->raidmap_mem[i] != NULL) 1061665484d8SDoug Ambrisko bus_dmamem_free(sc->raidmap_tag[i], sc->raidmap_mem[i], sc->raidmap_dmamap[i]); 1062665484d8SDoug Ambrisko if (sc->raidmap_tag[i] != NULL) 1063665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->raidmap_tag[i]); 10644799d485SKashyap D Desai 10654799d485SKashyap D Desai if (sc->ld_drv_map[i] != NULL) 10664799d485SKashyap D Desai free(sc->ld_drv_map[i], M_MRSAS); 1067665484d8SDoug Ambrisko } 1068665484d8SDoug Ambrisko 1069665484d8SDoug Ambrisko /* 1070665484d8SDoug Ambrisko * Free version buffer memroy 1071665484d8SDoug Ambrisko */ 1072665484d8SDoug Ambrisko if (sc->verbuf_phys_addr) 1073665484d8SDoug Ambrisko bus_dmamap_unload(sc->verbuf_tag, sc->verbuf_dmamap); 1074665484d8SDoug Ambrisko if (sc->verbuf_mem != NULL) 1075665484d8SDoug Ambrisko bus_dmamem_free(sc->verbuf_tag, sc->verbuf_mem, sc->verbuf_dmamap); 1076665484d8SDoug Ambrisko if (sc->verbuf_tag != NULL) 1077665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->verbuf_tag); 1078665484d8SDoug Ambrisko 1079665484d8SDoug Ambrisko 1080665484d8SDoug Ambrisko /* 1081665484d8SDoug Ambrisko * Free sense buffer memory 1082665484d8SDoug Ambrisko */ 1083665484d8SDoug Ambrisko if (sc->sense_phys_addr) 1084665484d8SDoug Ambrisko bus_dmamap_unload(sc->sense_tag, sc->sense_dmamap); 1085665484d8SDoug Ambrisko if (sc->sense_mem != NULL) 1086665484d8SDoug Ambrisko bus_dmamem_free(sc->sense_tag, sc->sense_mem, sc->sense_dmamap); 1087665484d8SDoug Ambrisko if (sc->sense_tag != NULL) 1088665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->sense_tag); 1089665484d8SDoug Ambrisko 1090665484d8SDoug Ambrisko /* 1091665484d8SDoug Ambrisko * Free chain frame memory 1092665484d8SDoug Ambrisko */ 1093665484d8SDoug Ambrisko if (sc->chain_frame_phys_addr) 1094665484d8SDoug Ambrisko bus_dmamap_unload(sc->chain_frame_tag, sc->chain_frame_dmamap); 1095665484d8SDoug Ambrisko if (sc->chain_frame_mem != NULL) 1096665484d8SDoug Ambrisko bus_dmamem_free(sc->chain_frame_tag, sc->chain_frame_mem, sc->chain_frame_dmamap); 1097665484d8SDoug Ambrisko if (sc->chain_frame_tag != NULL) 1098665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->chain_frame_tag); 1099665484d8SDoug Ambrisko 1100665484d8SDoug Ambrisko /* 1101665484d8SDoug Ambrisko * Free IO Request memory 1102665484d8SDoug Ambrisko */ 1103665484d8SDoug Ambrisko if (sc->io_request_phys_addr) 1104665484d8SDoug Ambrisko bus_dmamap_unload(sc->io_request_tag, sc->io_request_dmamap); 1105665484d8SDoug Ambrisko if (sc->io_request_mem != NULL) 1106665484d8SDoug Ambrisko bus_dmamem_free(sc->io_request_tag, sc->io_request_mem, sc->io_request_dmamap); 1107665484d8SDoug Ambrisko if (sc->io_request_tag != NULL) 1108665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->io_request_tag); 1109665484d8SDoug Ambrisko 1110665484d8SDoug Ambrisko /* 1111665484d8SDoug Ambrisko * Free Reply Descriptor memory 1112665484d8SDoug Ambrisko */ 1113665484d8SDoug Ambrisko if (sc->reply_desc_phys_addr) 1114665484d8SDoug Ambrisko bus_dmamap_unload(sc->reply_desc_tag, sc->reply_desc_dmamap); 1115665484d8SDoug Ambrisko if (sc->reply_desc_mem != NULL) 1116665484d8SDoug Ambrisko bus_dmamem_free(sc->reply_desc_tag, sc->reply_desc_mem, sc->reply_desc_dmamap); 1117665484d8SDoug Ambrisko if (sc->reply_desc_tag != NULL) 1118665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->reply_desc_tag); 1119665484d8SDoug Ambrisko 1120665484d8SDoug Ambrisko /* 1121665484d8SDoug Ambrisko * Free event detail memory 1122665484d8SDoug Ambrisko */ 1123665484d8SDoug Ambrisko if (sc->evt_detail_phys_addr) 1124665484d8SDoug Ambrisko bus_dmamap_unload(sc->evt_detail_tag, sc->evt_detail_dmamap); 1125665484d8SDoug Ambrisko if (sc->evt_detail_mem != NULL) 1126665484d8SDoug Ambrisko bus_dmamem_free(sc->evt_detail_tag, sc->evt_detail_mem, sc->evt_detail_dmamap); 1127665484d8SDoug Ambrisko if (sc->evt_detail_tag != NULL) 1128665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->evt_detail_tag); 1129665484d8SDoug Ambrisko 1130665484d8SDoug Ambrisko /* 1131665484d8SDoug Ambrisko * Free MFI frames 1132665484d8SDoug Ambrisko */ 1133665484d8SDoug Ambrisko if (sc->mfi_cmd_list) { 1134665484d8SDoug Ambrisko for (i = 0; i < MRSAS_MAX_MFI_CMDS; i++) { 1135665484d8SDoug Ambrisko mfi_cmd = sc->mfi_cmd_list[i]; 1136665484d8SDoug Ambrisko mrsas_free_frame(sc, mfi_cmd); 1137665484d8SDoug Ambrisko } 1138665484d8SDoug Ambrisko } 1139665484d8SDoug Ambrisko if (sc->mficmd_frame_tag != NULL) 1140665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->mficmd_frame_tag); 1141665484d8SDoug Ambrisko 1142665484d8SDoug Ambrisko /* 1143665484d8SDoug Ambrisko * Free MPT internal command list 1144665484d8SDoug Ambrisko */ 1145665484d8SDoug Ambrisko max_cmd = sc->max_fw_cmds; 1146665484d8SDoug Ambrisko if (sc->mpt_cmd_list) { 1147665484d8SDoug Ambrisko for (i = 0; i < max_cmd; i++) { 1148665484d8SDoug Ambrisko mpt_cmd = sc->mpt_cmd_list[i]; 1149665484d8SDoug Ambrisko bus_dmamap_destroy(sc->data_tag, mpt_cmd->data_dmamap); 1150665484d8SDoug Ambrisko free(sc->mpt_cmd_list[i], M_MRSAS); 1151665484d8SDoug Ambrisko } 1152665484d8SDoug Ambrisko free(sc->mpt_cmd_list, M_MRSAS); 1153665484d8SDoug Ambrisko sc->mpt_cmd_list = NULL; 1154665484d8SDoug Ambrisko } 1155665484d8SDoug Ambrisko /* 1156665484d8SDoug Ambrisko * Free MFI internal command list 1157665484d8SDoug Ambrisko */ 1158665484d8SDoug Ambrisko 1159665484d8SDoug Ambrisko if (sc->mfi_cmd_list) { 1160665484d8SDoug Ambrisko for (i = 0; i < MRSAS_MAX_MFI_CMDS; i++) { 1161665484d8SDoug Ambrisko free(sc->mfi_cmd_list[i], M_MRSAS); 1162665484d8SDoug Ambrisko } 1163665484d8SDoug Ambrisko free(sc->mfi_cmd_list, M_MRSAS); 1164665484d8SDoug Ambrisko sc->mfi_cmd_list = NULL; 1165665484d8SDoug Ambrisko } 1166665484d8SDoug Ambrisko /* 1167665484d8SDoug Ambrisko * Free request descriptor memory 1168665484d8SDoug Ambrisko */ 1169665484d8SDoug Ambrisko free(sc->req_desc, M_MRSAS); 1170665484d8SDoug Ambrisko sc->req_desc = NULL; 1171665484d8SDoug Ambrisko 1172665484d8SDoug Ambrisko /* 1173665484d8SDoug Ambrisko * Destroy parent tag 1174665484d8SDoug Ambrisko */ 1175665484d8SDoug Ambrisko if (sc->mrsas_parent_tag != NULL) 1176665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->mrsas_parent_tag); 1177af51c29fSKashyap D Desai 1178af51c29fSKashyap D Desai /* 1179af51c29fSKashyap D Desai * Free ctrl_info memory 1180af51c29fSKashyap D Desai */ 1181af51c29fSKashyap D Desai if (sc->ctrl_info != NULL) 1182af51c29fSKashyap D Desai free(sc->ctrl_info, M_MRSAS); 1183665484d8SDoug Ambrisko } 1184665484d8SDoug Ambrisko 11858e727371SKashyap D Desai /* 1186665484d8SDoug Ambrisko * mrsas_teardown_intr: Teardown interrupt 1187665484d8SDoug Ambrisko * input: Adapter instance soft state 1188665484d8SDoug Ambrisko * 11898e727371SKashyap D Desai * This function is called from mrsas_detach() to teardown and release bus 11908e727371SKashyap D Desai * interrupt resourse. 1191665484d8SDoug Ambrisko */ 11928e727371SKashyap D Desai void 11938e727371SKashyap D Desai mrsas_teardown_intr(struct mrsas_softc *sc) 1194665484d8SDoug Ambrisko { 1195d18d1b47SKashyap D Desai int i; 11968e727371SKashyap D Desai 1197d18d1b47SKashyap D Desai if (!sc->msix_enable) { 1198d18d1b47SKashyap D Desai if (sc->intr_handle[0]) 1199d18d1b47SKashyap D Desai bus_teardown_intr(sc->mrsas_dev, sc->mrsas_irq[0], sc->intr_handle[0]); 1200d18d1b47SKashyap D Desai if (sc->mrsas_irq[0] != NULL) 12018e727371SKashyap D Desai bus_release_resource(sc->mrsas_dev, SYS_RES_IRQ, 12028e727371SKashyap D Desai sc->irq_id[0], sc->mrsas_irq[0]); 1203d18d1b47SKashyap D Desai sc->intr_handle[0] = NULL; 1204d18d1b47SKashyap D Desai } else { 1205d18d1b47SKashyap D Desai for (i = 0; i < sc->msix_vectors; i++) { 1206d18d1b47SKashyap D Desai if (sc->intr_handle[i]) 1207d18d1b47SKashyap D Desai bus_teardown_intr(sc->mrsas_dev, sc->mrsas_irq[i], 1208d18d1b47SKashyap D Desai sc->intr_handle[i]); 1209d18d1b47SKashyap D Desai 1210d18d1b47SKashyap D Desai if (sc->mrsas_irq[i] != NULL) 1211d18d1b47SKashyap D Desai bus_release_resource(sc->mrsas_dev, SYS_RES_IRQ, 1212d18d1b47SKashyap D Desai sc->irq_id[i], sc->mrsas_irq[i]); 1213d18d1b47SKashyap D Desai 1214d18d1b47SKashyap D Desai sc->intr_handle[i] = NULL; 1215d18d1b47SKashyap D Desai } 1216d18d1b47SKashyap D Desai pci_release_msi(sc->mrsas_dev); 1217d18d1b47SKashyap D Desai } 1218d18d1b47SKashyap D Desai 1219665484d8SDoug Ambrisko } 1220665484d8SDoug Ambrisko 12218e727371SKashyap D Desai /* 1222665484d8SDoug Ambrisko * mrsas_suspend: Suspend entry point 1223665484d8SDoug Ambrisko * input: Device struct pointer 1224665484d8SDoug Ambrisko * 1225665484d8SDoug Ambrisko * This function is the entry point for system suspend from the OS. 1226665484d8SDoug Ambrisko */ 12278e727371SKashyap D Desai static int 12288e727371SKashyap D Desai mrsas_suspend(device_t dev) 1229665484d8SDoug Ambrisko { 1230665484d8SDoug Ambrisko struct mrsas_softc *sc; 1231665484d8SDoug Ambrisko 1232665484d8SDoug Ambrisko sc = device_get_softc(dev); 1233665484d8SDoug Ambrisko return (0); 1234665484d8SDoug Ambrisko } 1235665484d8SDoug Ambrisko 12368e727371SKashyap D Desai /* 1237665484d8SDoug Ambrisko * mrsas_resume: Resume entry point 1238665484d8SDoug Ambrisko * input: Device struct pointer 1239665484d8SDoug Ambrisko * 1240665484d8SDoug Ambrisko * This function is the entry point for system resume from the OS. 1241665484d8SDoug Ambrisko */ 12428e727371SKashyap D Desai static int 12438e727371SKashyap D Desai mrsas_resume(device_t dev) 1244665484d8SDoug Ambrisko { 1245665484d8SDoug Ambrisko struct mrsas_softc *sc; 1246665484d8SDoug Ambrisko 1247665484d8SDoug Ambrisko sc = device_get_softc(dev); 1248665484d8SDoug Ambrisko return (0); 1249665484d8SDoug Ambrisko } 1250665484d8SDoug Ambrisko 12515844115eSKashyap D Desai /** 12525844115eSKashyap D Desai * mrsas_get_softc_instance: Find softc instance based on cmd type 12535844115eSKashyap D Desai * 12545844115eSKashyap D Desai * This function will return softc instance based on cmd type. 12555844115eSKashyap D Desai * In some case, application fire ioctl on required management instance and 12565844115eSKashyap D Desai * do not provide host_no. Use cdev->si_drv1 to get softc instance for those 12575844115eSKashyap D Desai * case, else get the softc instance from host_no provided by application in 12585844115eSKashyap D Desai * user data. 12595844115eSKashyap D Desai */ 12605844115eSKashyap D Desai 12615844115eSKashyap D Desai static struct mrsas_softc * 12625844115eSKashyap D Desai mrsas_get_softc_instance(struct cdev *dev, u_long cmd, caddr_t arg) 12635844115eSKashyap D Desai { 12645844115eSKashyap D Desai struct mrsas_softc *sc = NULL; 12655844115eSKashyap D Desai struct mrsas_iocpacket *user_ioc = (struct mrsas_iocpacket *)arg; 1266*dbcc81dfSKashyap D Desai 12675844115eSKashyap D Desai if (cmd == MRSAS_IOC_GET_PCI_INFO) { 12685844115eSKashyap D Desai sc = dev->si_drv1; 12695844115eSKashyap D Desai } else { 1270*dbcc81dfSKashyap D Desai /* 1271*dbcc81dfSKashyap D Desai * get the Host number & the softc from data sent by the 1272*dbcc81dfSKashyap D Desai * Application 1273*dbcc81dfSKashyap D Desai */ 12745844115eSKashyap D Desai sc = mrsas_mgmt_info.sc_ptr[user_ioc->host_no]; 12755844115eSKashyap D Desai if ((user_ioc->host_no >= mrsas_mgmt_info.max_index) || (sc == NULL)) { 12765844115eSKashyap D Desai if (sc == NULL) 12775844115eSKashyap D Desai mrsas_dprint(sc, MRSAS_FAULT, 12785844115eSKashyap D Desai "There is no Controller number %d .\n", user_ioc->host_no); 12795844115eSKashyap D Desai else 12805844115eSKashyap D Desai mrsas_dprint(sc, MRSAS_FAULT, 12815844115eSKashyap D Desai "Invalid Controller number %d .\n", user_ioc->host_no); 12825844115eSKashyap D Desai } 12835844115eSKashyap D Desai } 12845844115eSKashyap D Desai 12855844115eSKashyap D Desai return sc; 12865844115eSKashyap D Desai } 12875844115eSKashyap D Desai 12888e727371SKashyap D Desai /* 1289665484d8SDoug Ambrisko * mrsas_ioctl: IOCtl commands entry point. 1290665484d8SDoug Ambrisko * 1291665484d8SDoug Ambrisko * This function is the entry point for IOCtls from the OS. It calls the 1292665484d8SDoug Ambrisko * appropriate function for processing depending on the command received. 1293665484d8SDoug Ambrisko */ 1294665484d8SDoug Ambrisko static int 12957fc5f329SJohn Baldwin mrsas_ioctl(struct cdev *dev, u_long cmd, caddr_t arg, int flag, 12967fc5f329SJohn Baldwin struct thread *td) 1297665484d8SDoug Ambrisko { 1298665484d8SDoug Ambrisko struct mrsas_softc *sc; 1299665484d8SDoug Ambrisko int ret = 0, i = 0; 13005844115eSKashyap D Desai MRSAS_DRV_PCI_INFORMATION *pciDrvInfo; 1301665484d8SDoug Ambrisko 13025844115eSKashyap D Desai sc = mrsas_get_softc_instance(dev, cmd, arg); 13035844115eSKashyap D Desai if (!sc) 1304536094dcSKashyap D Desai return ENOENT; 13055844115eSKashyap D Desai 1306665484d8SDoug Ambrisko if (sc->remove_in_progress) { 1307665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_INFO, 1308665484d8SDoug Ambrisko "Driver remove or shutdown called.\n"); 1309665484d8SDoug Ambrisko return ENOENT; 1310665484d8SDoug Ambrisko } 1311665484d8SDoug Ambrisko mtx_lock_spin(&sc->ioctl_lock); 1312665484d8SDoug Ambrisko if (!sc->reset_in_progress) { 1313665484d8SDoug Ambrisko mtx_unlock_spin(&sc->ioctl_lock); 1314665484d8SDoug Ambrisko goto do_ioctl; 1315665484d8SDoug Ambrisko } 1316665484d8SDoug Ambrisko mtx_unlock_spin(&sc->ioctl_lock); 1317665484d8SDoug Ambrisko while (sc->reset_in_progress) { 1318665484d8SDoug Ambrisko i++; 1319665484d8SDoug Ambrisko if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) { 1320665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_INFO, 1321665484d8SDoug Ambrisko "[%2d]waiting for " 1322665484d8SDoug Ambrisko "OCR to be finished %d\n", i, 1323665484d8SDoug Ambrisko sc->ocr_thread_active); 1324665484d8SDoug Ambrisko } 1325665484d8SDoug Ambrisko pause("mr_ioctl", hz); 1326665484d8SDoug Ambrisko } 1327665484d8SDoug Ambrisko 1328665484d8SDoug Ambrisko do_ioctl: 1329665484d8SDoug Ambrisko switch (cmd) { 1330536094dcSKashyap D Desai case MRSAS_IOC_FIRMWARE_PASS_THROUGH64: 1331536094dcSKashyap D Desai #ifdef COMPAT_FREEBSD32 1332536094dcSKashyap D Desai case MRSAS_IOC_FIRMWARE_PASS_THROUGH32: 1333536094dcSKashyap D Desai #endif 13348e727371SKashyap D Desai /* 13358e727371SKashyap D Desai * Decrement the Ioctl counting Semaphore before getting an 13368e727371SKashyap D Desai * mfi command 13378e727371SKashyap D Desai */ 1338839ee025SKashyap D Desai sema_wait(&sc->ioctl_count_sema); 1339839ee025SKashyap D Desai 1340536094dcSKashyap D Desai ret = mrsas_passthru(sc, (void *)arg, cmd); 1341839ee025SKashyap D Desai 1342839ee025SKashyap D Desai /* Increment the Ioctl counting semaphore value */ 1343839ee025SKashyap D Desai sema_post(&sc->ioctl_count_sema); 1344839ee025SKashyap D Desai 1345665484d8SDoug Ambrisko break; 1346665484d8SDoug Ambrisko case MRSAS_IOC_SCAN_BUS: 1347665484d8SDoug Ambrisko ret = mrsas_bus_scan(sc); 1348665484d8SDoug Ambrisko break; 13495844115eSKashyap D Desai 13505844115eSKashyap D Desai case MRSAS_IOC_GET_PCI_INFO: 13515844115eSKashyap D Desai pciDrvInfo = (MRSAS_DRV_PCI_INFORMATION *) arg; 13525844115eSKashyap D Desai memset(pciDrvInfo, 0, sizeof(MRSAS_DRV_PCI_INFORMATION)); 13535844115eSKashyap D Desai pciDrvInfo->busNumber = pci_get_bus(sc->mrsas_dev); 13545844115eSKashyap D Desai pciDrvInfo->deviceNumber = pci_get_slot(sc->mrsas_dev); 13555844115eSKashyap D Desai pciDrvInfo->functionNumber = pci_get_function(sc->mrsas_dev); 13565844115eSKashyap D Desai pciDrvInfo->domainID = pci_get_domain(sc->mrsas_dev); 13575844115eSKashyap D Desai mrsas_dprint(sc, MRSAS_INFO, "pci bus no: %d," 13585844115eSKashyap D Desai "pci device no: %d, pci function no: %d," 13595844115eSKashyap D Desai "pci domain ID: %d\n", 13605844115eSKashyap D Desai pciDrvInfo->busNumber, pciDrvInfo->deviceNumber, 13615844115eSKashyap D Desai pciDrvInfo->functionNumber, pciDrvInfo->domainID); 13625844115eSKashyap D Desai ret = 0; 13635844115eSKashyap D Desai break; 13645844115eSKashyap D Desai 1365536094dcSKashyap D Desai default: 1366536094dcSKashyap D Desai mrsas_dprint(sc, MRSAS_TRACE, "IOCTL command 0x%lx is not handled\n", cmd); 1367839ee025SKashyap D Desai ret = ENOENT; 1368665484d8SDoug Ambrisko } 1369665484d8SDoug Ambrisko 1370665484d8SDoug Ambrisko return (ret); 1371665484d8SDoug Ambrisko } 1372665484d8SDoug Ambrisko 13738e727371SKashyap D Desai /* 1374da011113SKashyap D Desai * mrsas_poll: poll entry point for mrsas driver fd 1375da011113SKashyap D Desai * 13768e727371SKashyap D Desai * This function is the entry point for poll from the OS. It waits for some AEN 13778e727371SKashyap D Desai * events to be triggered from the controller and notifies back. 1378da011113SKashyap D Desai */ 1379da011113SKashyap D Desai static int 1380da011113SKashyap D Desai mrsas_poll(struct cdev *dev, int poll_events, struct thread *td) 1381da011113SKashyap D Desai { 1382da011113SKashyap D Desai struct mrsas_softc *sc; 1383da011113SKashyap D Desai int revents = 0; 1384da011113SKashyap D Desai 1385da011113SKashyap D Desai sc = dev->si_drv1; 1386da011113SKashyap D Desai 1387da011113SKashyap D Desai if (poll_events & (POLLIN | POLLRDNORM)) { 1388da011113SKashyap D Desai if (sc->mrsas_aen_triggered) { 1389da011113SKashyap D Desai revents |= poll_events & (POLLIN | POLLRDNORM); 1390da011113SKashyap D Desai } 1391da011113SKashyap D Desai } 1392da011113SKashyap D Desai if (revents == 0) { 1393da011113SKashyap D Desai if (poll_events & (POLLIN | POLLRDNORM)) { 1394ecea5be4SKashyap D Desai mtx_lock(&sc->aen_lock); 1395da011113SKashyap D Desai sc->mrsas_poll_waiting = 1; 1396da011113SKashyap D Desai selrecord(td, &sc->mrsas_select); 1397ecea5be4SKashyap D Desai mtx_unlock(&sc->aen_lock); 1398da011113SKashyap D Desai } 1399da011113SKashyap D Desai } 1400da011113SKashyap D Desai return revents; 1401da011113SKashyap D Desai } 1402da011113SKashyap D Desai 14038e727371SKashyap D Desai /* 14048e727371SKashyap D Desai * mrsas_setup_irq: Set up interrupt 1405665484d8SDoug Ambrisko * input: Adapter instance soft state 1406665484d8SDoug Ambrisko * 1407665484d8SDoug Ambrisko * This function sets up interrupts as a bus resource, with flags indicating 1408665484d8SDoug Ambrisko * resource permitting contemporaneous sharing and for resource to activate 1409665484d8SDoug Ambrisko * atomically. 1410665484d8SDoug Ambrisko */ 14118e727371SKashyap D Desai static int 14128e727371SKashyap D Desai mrsas_setup_irq(struct mrsas_softc *sc) 1413665484d8SDoug Ambrisko { 1414d18d1b47SKashyap D Desai if (sc->msix_enable && (mrsas_setup_msix(sc) == SUCCESS)) 1415d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "MSI-x interrupts setup success\n"); 1416665484d8SDoug Ambrisko 1417d18d1b47SKashyap D Desai else { 1418d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "Fall back to legacy interrupt\n"); 1419d18d1b47SKashyap D Desai sc->irq_context[0].sc = sc; 1420d18d1b47SKashyap D Desai sc->irq_context[0].MSIxIndex = 0; 1421d18d1b47SKashyap D Desai sc->irq_id[0] = 0; 1422d18d1b47SKashyap D Desai sc->mrsas_irq[0] = bus_alloc_resource_any(sc->mrsas_dev, 1423d18d1b47SKashyap D Desai SYS_RES_IRQ, &sc->irq_id[0], RF_SHAREABLE | RF_ACTIVE); 1424d18d1b47SKashyap D Desai if (sc->mrsas_irq[0] == NULL) { 1425d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "Cannot allocate legcay" 1426d18d1b47SKashyap D Desai "interrupt\n"); 1427d18d1b47SKashyap D Desai return (FAIL); 1428d18d1b47SKashyap D Desai } 1429d18d1b47SKashyap D Desai if (bus_setup_intr(sc->mrsas_dev, sc->mrsas_irq[0], 1430d18d1b47SKashyap D Desai INTR_MPSAFE | INTR_TYPE_CAM, NULL, mrsas_isr, 1431d18d1b47SKashyap D Desai &sc->irq_context[0], &sc->intr_handle[0])) { 1432d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "Cannot set up legacy" 1433d18d1b47SKashyap D Desai "interrupt\n"); 1434d18d1b47SKashyap D Desai return (FAIL); 1435d18d1b47SKashyap D Desai } 1436d18d1b47SKashyap D Desai } 1437665484d8SDoug Ambrisko return (0); 1438665484d8SDoug Ambrisko } 1439665484d8SDoug Ambrisko 1440665484d8SDoug Ambrisko /* 1441665484d8SDoug Ambrisko * mrsas_isr: ISR entry point 1442665484d8SDoug Ambrisko * input: argument pointer 1443665484d8SDoug Ambrisko * 14448e727371SKashyap D Desai * This function is the interrupt service routine entry point. There are two 14458e727371SKashyap D Desai * types of interrupts, state change interrupt and response interrupt. If an 14468e727371SKashyap D Desai * interrupt is not ours, we just return. 1447665484d8SDoug Ambrisko */ 14488e727371SKashyap D Desai void 14498e727371SKashyap D Desai mrsas_isr(void *arg) 1450665484d8SDoug Ambrisko { 1451d18d1b47SKashyap D Desai struct mrsas_irq_context *irq_context = (struct mrsas_irq_context *)arg; 1452d18d1b47SKashyap D Desai struct mrsas_softc *sc = irq_context->sc; 1453d18d1b47SKashyap D Desai int status = 0; 1454665484d8SDoug Ambrisko 14552f863eb8SKashyap D Desai if (sc->mask_interrupts) 14562f863eb8SKashyap D Desai return; 14572f863eb8SKashyap D Desai 1458d18d1b47SKashyap D Desai if (!sc->msix_vectors) { 1459665484d8SDoug Ambrisko status = mrsas_clear_intr(sc); 1460665484d8SDoug Ambrisko if (!status) 1461665484d8SDoug Ambrisko return; 1462d18d1b47SKashyap D Desai } 1463665484d8SDoug Ambrisko /* If we are resetting, bail */ 1464f5fb2237SKashyap D Desai if (mrsas_test_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags)) { 1465665484d8SDoug Ambrisko printf(" Entered into ISR when OCR is going active. \n"); 1466665484d8SDoug Ambrisko mrsas_clear_intr(sc); 1467665484d8SDoug Ambrisko return; 1468665484d8SDoug Ambrisko } 1469665484d8SDoug Ambrisko /* Process for reply request and clear response interrupt */ 1470d18d1b47SKashyap D Desai if (mrsas_complete_cmd(sc, irq_context->MSIxIndex) != SUCCESS) 1471665484d8SDoug Ambrisko mrsas_clear_intr(sc); 1472665484d8SDoug Ambrisko 1473665484d8SDoug Ambrisko return; 1474665484d8SDoug Ambrisko } 1475665484d8SDoug Ambrisko 1476665484d8SDoug Ambrisko /* 1477665484d8SDoug Ambrisko * mrsas_complete_cmd: Process reply request 1478665484d8SDoug Ambrisko * input: Adapter instance soft state 1479665484d8SDoug Ambrisko * 14808e727371SKashyap D Desai * This function is called from mrsas_isr() to process reply request and clear 14818e727371SKashyap D Desai * response interrupt. Processing of the reply request entails walking 14828e727371SKashyap D Desai * through the reply descriptor array for the command request pended from 14838e727371SKashyap D Desai * Firmware. We look at the Function field to determine the command type and 14848e727371SKashyap D Desai * perform the appropriate action. Before we return, we clear the response 14858e727371SKashyap D Desai * interrupt. 1486665484d8SDoug Ambrisko */ 14878e727371SKashyap D Desai static int 14888e727371SKashyap D Desai mrsas_complete_cmd(struct mrsas_softc *sc, u_int32_t MSIxIndex) 1489665484d8SDoug Ambrisko { 1490665484d8SDoug Ambrisko Mpi2ReplyDescriptorsUnion_t *desc; 1491665484d8SDoug Ambrisko MPI2_SCSI_IO_SUCCESS_REPLY_DESCRIPTOR *reply_desc; 1492665484d8SDoug Ambrisko MRSAS_RAID_SCSI_IO_REQUEST *scsi_io_req; 1493665484d8SDoug Ambrisko struct mrsas_mpt_cmd *cmd_mpt; 1494665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd_mfi; 149516dc2814SKashyap D Desai u_int8_t reply_descript_type; 1496665484d8SDoug Ambrisko u_int16_t smid, num_completed; 1497665484d8SDoug Ambrisko u_int8_t status, extStatus; 1498665484d8SDoug Ambrisko union desc_value desc_val; 1499665484d8SDoug Ambrisko PLD_LOAD_BALANCE_INFO lbinfo; 1500665484d8SDoug Ambrisko u_int32_t device_id; 1501665484d8SDoug Ambrisko int threshold_reply_count = 0; 1502665484d8SDoug Ambrisko 1503665484d8SDoug Ambrisko 1504665484d8SDoug Ambrisko /* If we have a hardware error, not need to continue */ 1505665484d8SDoug Ambrisko if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) 1506665484d8SDoug Ambrisko return (DONE); 1507665484d8SDoug Ambrisko 1508665484d8SDoug Ambrisko desc = sc->reply_desc_mem; 1509d18d1b47SKashyap D Desai desc += ((MSIxIndex * sc->reply_alloc_sz) / sizeof(MPI2_REPLY_DESCRIPTORS_UNION)) 1510d18d1b47SKashyap D Desai + sc->last_reply_idx[MSIxIndex]; 1511665484d8SDoug Ambrisko 1512665484d8SDoug Ambrisko reply_desc = (MPI2_SCSI_IO_SUCCESS_REPLY_DESCRIPTOR *) desc; 1513665484d8SDoug Ambrisko 1514665484d8SDoug Ambrisko desc_val.word = desc->Words; 1515665484d8SDoug Ambrisko num_completed = 0; 1516665484d8SDoug Ambrisko 1517665484d8SDoug Ambrisko reply_descript_type = reply_desc->ReplyFlags & MPI2_RPY_DESCRIPT_FLAGS_TYPE_MASK; 1518665484d8SDoug Ambrisko 1519665484d8SDoug Ambrisko /* Find our reply descriptor for the command and process */ 15208e727371SKashyap D Desai while ((desc_val.u.low != 0xFFFFFFFF) && (desc_val.u.high != 0xFFFFFFFF)) { 1521665484d8SDoug Ambrisko smid = reply_desc->SMID; 1522665484d8SDoug Ambrisko cmd_mpt = sc->mpt_cmd_list[smid - 1]; 1523665484d8SDoug Ambrisko scsi_io_req = (MRSAS_RAID_SCSI_IO_REQUEST *) cmd_mpt->io_request; 1524665484d8SDoug Ambrisko 1525665484d8SDoug Ambrisko status = scsi_io_req->RaidContext.status; 1526665484d8SDoug Ambrisko extStatus = scsi_io_req->RaidContext.exStatus; 1527665484d8SDoug Ambrisko 15288e727371SKashyap D Desai switch (scsi_io_req->Function) { 1529665484d8SDoug Ambrisko case MPI2_FUNCTION_SCSI_IO_REQUEST: /* Fast Path IO. */ 1530665484d8SDoug Ambrisko device_id = cmd_mpt->ccb_ptr->ccb_h.target_id; 1531665484d8SDoug Ambrisko lbinfo = &sc->load_balance_info[device_id]; 1532665484d8SDoug Ambrisko if (cmd_mpt->load_balance == MRSAS_LOAD_BALANCE_FLAG) { 153316dc2814SKashyap D Desai mrsas_atomic_dec(&lbinfo->scsi_pending_cmds[cmd_mpt->pd_r1_lb]); 1534665484d8SDoug Ambrisko cmd_mpt->load_balance &= ~MRSAS_LOAD_BALANCE_FLAG; 1535665484d8SDoug Ambrisko } 15368e727371SKashyap D Desai /* Fall thru and complete IO */ 1537665484d8SDoug Ambrisko case MRSAS_MPI2_FUNCTION_LD_IO_REQUEST: 1538665484d8SDoug Ambrisko mrsas_map_mpt_cmd_status(cmd_mpt, status, extStatus); 1539665484d8SDoug Ambrisko mrsas_cmd_done(sc, cmd_mpt); 1540665484d8SDoug Ambrisko scsi_io_req->RaidContext.status = 0; 1541665484d8SDoug Ambrisko scsi_io_req->RaidContext.exStatus = 0; 1542f5fb2237SKashyap D Desai mrsas_atomic_dec(&sc->fw_outstanding); 1543665484d8SDoug Ambrisko break; 1544665484d8SDoug Ambrisko case MRSAS_MPI2_FUNCTION_PASSTHRU_IO_REQUEST: /* MFI command */ 1545665484d8SDoug Ambrisko cmd_mfi = sc->mfi_cmd_list[cmd_mpt->sync_cmd_idx]; 1546665484d8SDoug Ambrisko mrsas_complete_mptmfi_passthru(sc, cmd_mfi, status); 1547665484d8SDoug Ambrisko cmd_mpt->flags = 0; 1548665484d8SDoug Ambrisko mrsas_release_mpt_cmd(cmd_mpt); 1549665484d8SDoug Ambrisko break; 1550665484d8SDoug Ambrisko } 1551665484d8SDoug Ambrisko 1552d18d1b47SKashyap D Desai sc->last_reply_idx[MSIxIndex]++; 1553d18d1b47SKashyap D Desai if (sc->last_reply_idx[MSIxIndex] >= sc->reply_q_depth) 1554d18d1b47SKashyap D Desai sc->last_reply_idx[MSIxIndex] = 0; 1555665484d8SDoug Ambrisko 15568e727371SKashyap D Desai desc->Words = ~((uint64_t)0x00); /* set it back to all 15578e727371SKashyap D Desai * 0xFFFFFFFFs */ 1558665484d8SDoug Ambrisko num_completed++; 1559665484d8SDoug Ambrisko threshold_reply_count++; 1560665484d8SDoug Ambrisko 1561665484d8SDoug Ambrisko /* Get the next reply descriptor */ 1562d18d1b47SKashyap D Desai if (!sc->last_reply_idx[MSIxIndex]) { 1563665484d8SDoug Ambrisko desc = sc->reply_desc_mem; 1564d18d1b47SKashyap D Desai desc += ((MSIxIndex * sc->reply_alloc_sz) / sizeof(MPI2_REPLY_DESCRIPTORS_UNION)); 1565d18d1b47SKashyap D Desai } else 1566665484d8SDoug Ambrisko desc++; 1567665484d8SDoug Ambrisko 1568665484d8SDoug Ambrisko reply_desc = (MPI2_SCSI_IO_SUCCESS_REPLY_DESCRIPTOR *) desc; 1569665484d8SDoug Ambrisko desc_val.word = desc->Words; 1570665484d8SDoug Ambrisko 1571665484d8SDoug Ambrisko reply_descript_type = reply_desc->ReplyFlags & MPI2_RPY_DESCRIPT_FLAGS_TYPE_MASK; 1572665484d8SDoug Ambrisko 1573665484d8SDoug Ambrisko if (reply_descript_type == MPI2_RPY_DESCRIPT_FLAGS_UNUSED) 1574665484d8SDoug Ambrisko break; 1575665484d8SDoug Ambrisko 1576665484d8SDoug Ambrisko /* 15778e727371SKashyap D Desai * Write to reply post index after completing threshold reply 15788e727371SKashyap D Desai * count and still there are more replies in reply queue 15798e727371SKashyap D Desai * pending to be completed. 1580665484d8SDoug Ambrisko */ 1581665484d8SDoug Ambrisko if (threshold_reply_count >= THRESHOLD_REPLY_COUNT) { 1582d18d1b47SKashyap D Desai if (sc->msix_enable) { 1583d18d1b47SKashyap D Desai if ((sc->device_id == MRSAS_INVADER) || 1584d18d1b47SKashyap D Desai (sc->device_id == MRSAS_FURY)) 1585d18d1b47SKashyap D Desai mrsas_write_reg(sc, sc->msix_reg_offset[MSIxIndex / 8], 1586d18d1b47SKashyap D Desai ((MSIxIndex & 0x7) << 24) | 1587d18d1b47SKashyap D Desai sc->last_reply_idx[MSIxIndex]); 1588d18d1b47SKashyap D Desai else 1589d18d1b47SKashyap D Desai mrsas_write_reg(sc, sc->msix_reg_offset[0], (MSIxIndex << 24) | 1590d18d1b47SKashyap D Desai sc->last_reply_idx[MSIxIndex]); 1591d18d1b47SKashyap D Desai } else 1592d18d1b47SKashyap D Desai mrsas_write_reg(sc, offsetof(mrsas_reg_set, 1593d18d1b47SKashyap D Desai reply_post_host_index), sc->last_reply_idx[0]); 1594d18d1b47SKashyap D Desai 1595665484d8SDoug Ambrisko threshold_reply_count = 0; 1596665484d8SDoug Ambrisko } 1597665484d8SDoug Ambrisko } 1598665484d8SDoug Ambrisko 1599665484d8SDoug Ambrisko /* No match, just return */ 1600665484d8SDoug Ambrisko if (num_completed == 0) 1601665484d8SDoug Ambrisko return (DONE); 1602665484d8SDoug Ambrisko 1603665484d8SDoug Ambrisko /* Clear response interrupt */ 1604d18d1b47SKashyap D Desai if (sc->msix_enable) { 1605d18d1b47SKashyap D Desai if ((sc->device_id == MRSAS_INVADER) || 1606d18d1b47SKashyap D Desai (sc->device_id == MRSAS_FURY)) { 1607d18d1b47SKashyap D Desai mrsas_write_reg(sc, sc->msix_reg_offset[MSIxIndex / 8], 1608d18d1b47SKashyap D Desai ((MSIxIndex & 0x7) << 24) | 1609d18d1b47SKashyap D Desai sc->last_reply_idx[MSIxIndex]); 1610d18d1b47SKashyap D Desai } else 1611d18d1b47SKashyap D Desai mrsas_write_reg(sc, sc->msix_reg_offset[0], (MSIxIndex << 24) | 1612d18d1b47SKashyap D Desai sc->last_reply_idx[MSIxIndex]); 1613d18d1b47SKashyap D Desai } else 1614d18d1b47SKashyap D Desai mrsas_write_reg(sc, offsetof(mrsas_reg_set, 1615d18d1b47SKashyap D Desai reply_post_host_index), sc->last_reply_idx[0]); 1616665484d8SDoug Ambrisko 1617665484d8SDoug Ambrisko return (0); 1618665484d8SDoug Ambrisko } 1619665484d8SDoug Ambrisko 1620665484d8SDoug Ambrisko /* 1621665484d8SDoug Ambrisko * mrsas_map_mpt_cmd_status: Allocate DMAable memory. 1622665484d8SDoug Ambrisko * input: Adapter instance soft state 1623665484d8SDoug Ambrisko * 1624665484d8SDoug Ambrisko * This function is called from mrsas_complete_cmd(), for LD IO and FastPath IO. 16258e727371SKashyap D Desai * It checks the command status and maps the appropriate CAM status for the 16268e727371SKashyap D Desai * CCB. 1627665484d8SDoug Ambrisko */ 16288e727371SKashyap D Desai void 16298e727371SKashyap D Desai mrsas_map_mpt_cmd_status(struct mrsas_mpt_cmd *cmd, u_int8_t status, u_int8_t extStatus) 1630665484d8SDoug Ambrisko { 1631665484d8SDoug Ambrisko struct mrsas_softc *sc = cmd->sc; 1632665484d8SDoug Ambrisko u_int8_t *sense_data; 1633665484d8SDoug Ambrisko 1634665484d8SDoug Ambrisko switch (status) { 1635665484d8SDoug Ambrisko case MFI_STAT_OK: 1636665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status = CAM_REQ_CMP; 1637665484d8SDoug Ambrisko break; 1638665484d8SDoug Ambrisko case MFI_STAT_SCSI_IO_FAILED: 1639665484d8SDoug Ambrisko case MFI_STAT_SCSI_DONE_WITH_ERROR: 1640665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status = CAM_SCSI_STATUS_ERROR; 1641665484d8SDoug Ambrisko sense_data = (u_int8_t *)&cmd->ccb_ptr->csio.sense_data; 1642665484d8SDoug Ambrisko if (sense_data) { 1643665484d8SDoug Ambrisko /* For now just copy 18 bytes back */ 1644665484d8SDoug Ambrisko memcpy(sense_data, cmd->sense, 18); 1645665484d8SDoug Ambrisko cmd->ccb_ptr->csio.sense_len = 18; 1646665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status |= CAM_AUTOSNS_VALID; 1647665484d8SDoug Ambrisko } 1648665484d8SDoug Ambrisko break; 1649665484d8SDoug Ambrisko case MFI_STAT_LD_OFFLINE: 1650665484d8SDoug Ambrisko case MFI_STAT_DEVICE_NOT_FOUND: 1651665484d8SDoug Ambrisko if (cmd->ccb_ptr->ccb_h.target_lun) 1652665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status |= CAM_LUN_INVALID; 1653665484d8SDoug Ambrisko else 1654665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status |= CAM_DEV_NOT_THERE; 1655665484d8SDoug Ambrisko break; 1656665484d8SDoug Ambrisko case MFI_STAT_CONFIG_SEQ_MISMATCH: 1657665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status |= CAM_REQUEUE_REQ; 1658665484d8SDoug Ambrisko break; 1659665484d8SDoug Ambrisko default: 1660665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "FW cmd complete status %x\n", status); 1661665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status = CAM_REQ_CMP_ERR; 1662665484d8SDoug Ambrisko cmd->ccb_ptr->csio.scsi_status = status; 1663665484d8SDoug Ambrisko } 1664665484d8SDoug Ambrisko return; 1665665484d8SDoug Ambrisko } 1666665484d8SDoug Ambrisko 1667665484d8SDoug Ambrisko /* 16688e727371SKashyap D Desai * mrsas_alloc_mem: Allocate DMAable memory 1669665484d8SDoug Ambrisko * input: Adapter instance soft state 1670665484d8SDoug Ambrisko * 16718e727371SKashyap D Desai * This function creates the parent DMA tag and allocates DMAable memory. DMA 16728e727371SKashyap D Desai * tag describes constraints of DMA mapping. Memory allocated is mapped into 16738e727371SKashyap D Desai * Kernel virtual address. Callback argument is physical memory address. 1674665484d8SDoug Ambrisko */ 16758e727371SKashyap D Desai static int 16768e727371SKashyap D Desai mrsas_alloc_mem(struct mrsas_softc *sc) 1677665484d8SDoug Ambrisko { 1678*dbcc81dfSKashyap D Desai u_int32_t verbuf_size, io_req_size, reply_desc_size, sense_size, 1679*dbcc81dfSKashyap D Desai chain_frame_size, evt_detail_size, count; 1680665484d8SDoug Ambrisko 1681665484d8SDoug Ambrisko /* 1682665484d8SDoug Ambrisko * Allocate parent DMA tag 1683665484d8SDoug Ambrisko */ 1684665484d8SDoug Ambrisko if (bus_dma_tag_create(NULL, /* parent */ 1685665484d8SDoug Ambrisko 1, /* alignment */ 1686665484d8SDoug Ambrisko 0, /* boundary */ 1687665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, /* lowaddr */ 1688665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, /* highaddr */ 1689665484d8SDoug Ambrisko NULL, NULL, /* filter, filterarg */ 1690665484d8SDoug Ambrisko MRSAS_MAX_IO_SIZE, /* maxsize */ 1691665484d8SDoug Ambrisko MRSAS_MAX_SGL, /* nsegments */ 1692665484d8SDoug Ambrisko MRSAS_MAX_IO_SIZE, /* maxsegsize */ 1693665484d8SDoug Ambrisko 0, /* flags */ 1694665484d8SDoug Ambrisko NULL, NULL, /* lockfunc, lockarg */ 1695665484d8SDoug Ambrisko &sc->mrsas_parent_tag /* tag */ 1696665484d8SDoug Ambrisko )) { 1697665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate parent DMA tag\n"); 1698665484d8SDoug Ambrisko return (ENOMEM); 1699665484d8SDoug Ambrisko } 1700665484d8SDoug Ambrisko /* 1701665484d8SDoug Ambrisko * Allocate for version buffer 1702665484d8SDoug Ambrisko */ 1703665484d8SDoug Ambrisko verbuf_size = MRSAS_MAX_NAME_LENGTH * (sizeof(bus_addr_t)); 17048e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 17058e727371SKashyap D Desai 1, 0, 17068e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 17078e727371SKashyap D Desai BUS_SPACE_MAXADDR, 17088e727371SKashyap D Desai NULL, NULL, 17098e727371SKashyap D Desai verbuf_size, 17108e727371SKashyap D Desai 1, 17118e727371SKashyap D Desai verbuf_size, 17128e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 17138e727371SKashyap D Desai NULL, NULL, 1714665484d8SDoug Ambrisko &sc->verbuf_tag)) { 1715665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate verbuf DMA tag\n"); 1716665484d8SDoug Ambrisko return (ENOMEM); 1717665484d8SDoug Ambrisko } 1718665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->verbuf_tag, (void **)&sc->verbuf_mem, 1719665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->verbuf_dmamap)) { 1720665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate verbuf memory\n"); 1721665484d8SDoug Ambrisko return (ENOMEM); 1722665484d8SDoug Ambrisko } 1723665484d8SDoug Ambrisko bzero(sc->verbuf_mem, verbuf_size); 1724665484d8SDoug Ambrisko if (bus_dmamap_load(sc->verbuf_tag, sc->verbuf_dmamap, sc->verbuf_mem, 17258e727371SKashyap D Desai verbuf_size, mrsas_addr_cb, &sc->verbuf_phys_addr, 17268e727371SKashyap D Desai BUS_DMA_NOWAIT)) { 1727665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load verbuf DMA map\n"); 1728665484d8SDoug Ambrisko return (ENOMEM); 1729665484d8SDoug Ambrisko } 1730665484d8SDoug Ambrisko /* 1731665484d8SDoug Ambrisko * Allocate IO Request Frames 1732665484d8SDoug Ambrisko */ 1733665484d8SDoug Ambrisko io_req_size = sc->io_frames_alloc_sz; 17348e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 17358e727371SKashyap D Desai 16, 0, 17368e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 17378e727371SKashyap D Desai BUS_SPACE_MAXADDR, 17388e727371SKashyap D Desai NULL, NULL, 17398e727371SKashyap D Desai io_req_size, 17408e727371SKashyap D Desai 1, 17418e727371SKashyap D Desai io_req_size, 17428e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 17438e727371SKashyap D Desai NULL, NULL, 1744665484d8SDoug Ambrisko &sc->io_request_tag)) { 1745665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create IO request tag\n"); 1746665484d8SDoug Ambrisko return (ENOMEM); 1747665484d8SDoug Ambrisko } 1748665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->io_request_tag, (void **)&sc->io_request_mem, 1749665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->io_request_dmamap)) { 1750665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc IO request memory\n"); 1751665484d8SDoug Ambrisko return (ENOMEM); 1752665484d8SDoug Ambrisko } 1753665484d8SDoug Ambrisko bzero(sc->io_request_mem, io_req_size); 1754665484d8SDoug Ambrisko if (bus_dmamap_load(sc->io_request_tag, sc->io_request_dmamap, 1755665484d8SDoug Ambrisko sc->io_request_mem, io_req_size, mrsas_addr_cb, 1756665484d8SDoug Ambrisko &sc->io_request_phys_addr, BUS_DMA_NOWAIT)) { 1757665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load IO request memory\n"); 1758665484d8SDoug Ambrisko return (ENOMEM); 1759665484d8SDoug Ambrisko } 1760665484d8SDoug Ambrisko /* 1761665484d8SDoug Ambrisko * Allocate Chain Frames 1762665484d8SDoug Ambrisko */ 1763665484d8SDoug Ambrisko chain_frame_size = sc->chain_frames_alloc_sz; 17648e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 17658e727371SKashyap D Desai 4, 0, 17668e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 17678e727371SKashyap D Desai BUS_SPACE_MAXADDR, 17688e727371SKashyap D Desai NULL, NULL, 17698e727371SKashyap D Desai chain_frame_size, 17708e727371SKashyap D Desai 1, 17718e727371SKashyap D Desai chain_frame_size, 17728e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 17738e727371SKashyap D Desai NULL, NULL, 1774665484d8SDoug Ambrisko &sc->chain_frame_tag)) { 1775665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create chain frame tag\n"); 1776665484d8SDoug Ambrisko return (ENOMEM); 1777665484d8SDoug Ambrisko } 1778665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->chain_frame_tag, (void **)&sc->chain_frame_mem, 1779665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->chain_frame_dmamap)) { 1780665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc chain frame memory\n"); 1781665484d8SDoug Ambrisko return (ENOMEM); 1782665484d8SDoug Ambrisko } 1783665484d8SDoug Ambrisko bzero(sc->chain_frame_mem, chain_frame_size); 1784665484d8SDoug Ambrisko if (bus_dmamap_load(sc->chain_frame_tag, sc->chain_frame_dmamap, 1785665484d8SDoug Ambrisko sc->chain_frame_mem, chain_frame_size, mrsas_addr_cb, 1786665484d8SDoug Ambrisko &sc->chain_frame_phys_addr, BUS_DMA_NOWAIT)) { 1787665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load chain frame memory\n"); 1788665484d8SDoug Ambrisko return (ENOMEM); 1789665484d8SDoug Ambrisko } 1790d18d1b47SKashyap D Desai count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 1791665484d8SDoug Ambrisko /* 1792665484d8SDoug Ambrisko * Allocate Reply Descriptor Array 1793665484d8SDoug Ambrisko */ 1794d18d1b47SKashyap D Desai reply_desc_size = sc->reply_alloc_sz * count; 17958e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 17968e727371SKashyap D Desai 16, 0, 17978e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 17988e727371SKashyap D Desai BUS_SPACE_MAXADDR, 17998e727371SKashyap D Desai NULL, NULL, 18008e727371SKashyap D Desai reply_desc_size, 18018e727371SKashyap D Desai 1, 18028e727371SKashyap D Desai reply_desc_size, 18038e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 18048e727371SKashyap D Desai NULL, NULL, 1805665484d8SDoug Ambrisko &sc->reply_desc_tag)) { 1806665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create reply descriptor tag\n"); 1807665484d8SDoug Ambrisko return (ENOMEM); 1808665484d8SDoug Ambrisko } 1809665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->reply_desc_tag, (void **)&sc->reply_desc_mem, 1810665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->reply_desc_dmamap)) { 1811665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc reply descriptor memory\n"); 1812665484d8SDoug Ambrisko return (ENOMEM); 1813665484d8SDoug Ambrisko } 1814665484d8SDoug Ambrisko if (bus_dmamap_load(sc->reply_desc_tag, sc->reply_desc_dmamap, 1815665484d8SDoug Ambrisko sc->reply_desc_mem, reply_desc_size, mrsas_addr_cb, 1816665484d8SDoug Ambrisko &sc->reply_desc_phys_addr, BUS_DMA_NOWAIT)) { 1817665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load reply descriptor memory\n"); 1818665484d8SDoug Ambrisko return (ENOMEM); 1819665484d8SDoug Ambrisko } 1820665484d8SDoug Ambrisko /* 1821665484d8SDoug Ambrisko * Allocate Sense Buffer Array. Keep in lower 4GB 1822665484d8SDoug Ambrisko */ 1823665484d8SDoug Ambrisko sense_size = sc->max_fw_cmds * MRSAS_SENSE_LEN; 18248e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 18258e727371SKashyap D Desai 64, 0, 18268e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 18278e727371SKashyap D Desai BUS_SPACE_MAXADDR, 18288e727371SKashyap D Desai NULL, NULL, 18298e727371SKashyap D Desai sense_size, 18308e727371SKashyap D Desai 1, 18318e727371SKashyap D Desai sense_size, 18328e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 18338e727371SKashyap D Desai NULL, NULL, 1834665484d8SDoug Ambrisko &sc->sense_tag)) { 1835665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate sense buf tag\n"); 1836665484d8SDoug Ambrisko return (ENOMEM); 1837665484d8SDoug Ambrisko } 1838665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->sense_tag, (void **)&sc->sense_mem, 1839665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->sense_dmamap)) { 1840665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate sense buf memory\n"); 1841665484d8SDoug Ambrisko return (ENOMEM); 1842665484d8SDoug Ambrisko } 1843665484d8SDoug Ambrisko if (bus_dmamap_load(sc->sense_tag, sc->sense_dmamap, 1844665484d8SDoug Ambrisko sc->sense_mem, sense_size, mrsas_addr_cb, &sc->sense_phys_addr, 1845665484d8SDoug Ambrisko BUS_DMA_NOWAIT)) { 1846665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load sense buf memory\n"); 1847665484d8SDoug Ambrisko return (ENOMEM); 1848665484d8SDoug Ambrisko } 1849665484d8SDoug Ambrisko /* 1850665484d8SDoug Ambrisko * Allocate for Event detail structure 1851665484d8SDoug Ambrisko */ 1852665484d8SDoug Ambrisko evt_detail_size = sizeof(struct mrsas_evt_detail); 18538e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 18548e727371SKashyap D Desai 1, 0, 18558e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 18568e727371SKashyap D Desai BUS_SPACE_MAXADDR, 18578e727371SKashyap D Desai NULL, NULL, 18588e727371SKashyap D Desai evt_detail_size, 18598e727371SKashyap D Desai 1, 18608e727371SKashyap D Desai evt_detail_size, 18618e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 18628e727371SKashyap D Desai NULL, NULL, 1863665484d8SDoug Ambrisko &sc->evt_detail_tag)) { 1864665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create Event detail tag\n"); 1865665484d8SDoug Ambrisko return (ENOMEM); 1866665484d8SDoug Ambrisko } 1867665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->evt_detail_tag, (void **)&sc->evt_detail_mem, 1868665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->evt_detail_dmamap)) { 1869665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc Event detail buffer memory\n"); 1870665484d8SDoug Ambrisko return (ENOMEM); 1871665484d8SDoug Ambrisko } 1872665484d8SDoug Ambrisko bzero(sc->evt_detail_mem, evt_detail_size); 1873665484d8SDoug Ambrisko if (bus_dmamap_load(sc->evt_detail_tag, sc->evt_detail_dmamap, 1874665484d8SDoug Ambrisko sc->evt_detail_mem, evt_detail_size, mrsas_addr_cb, 1875665484d8SDoug Ambrisko &sc->evt_detail_phys_addr, BUS_DMA_NOWAIT)) { 1876665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load Event detail buffer memory\n"); 1877665484d8SDoug Ambrisko return (ENOMEM); 1878665484d8SDoug Ambrisko } 1879665484d8SDoug Ambrisko /* 1880665484d8SDoug Ambrisko * Create a dma tag for data buffers; size will be the maximum 1881665484d8SDoug Ambrisko * possible I/O size (280kB). 1882665484d8SDoug Ambrisko */ 18838e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 18848e727371SKashyap D Desai 1, 18858e727371SKashyap D Desai 0, 18868e727371SKashyap D Desai BUS_SPACE_MAXADDR, 18878e727371SKashyap D Desai BUS_SPACE_MAXADDR, 18888e727371SKashyap D Desai NULL, NULL, 18898e727371SKashyap D Desai MRSAS_MAX_IO_SIZE, 18908e727371SKashyap D Desai MRSAS_MAX_SGL, 18918e727371SKashyap D Desai MRSAS_MAX_IO_SIZE, 18928e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 18938e727371SKashyap D Desai busdma_lock_mutex, 18948e727371SKashyap D Desai &sc->io_lock, 1895665484d8SDoug Ambrisko &sc->data_tag)) { 1896665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create data dma tag\n"); 1897665484d8SDoug Ambrisko return (ENOMEM); 1898665484d8SDoug Ambrisko } 1899665484d8SDoug Ambrisko return (0); 1900665484d8SDoug Ambrisko } 1901665484d8SDoug Ambrisko 1902665484d8SDoug Ambrisko /* 1903665484d8SDoug Ambrisko * mrsas_addr_cb: Callback function of bus_dmamap_load() 19048e727371SKashyap D Desai * input: callback argument, machine dependent type 19058e727371SKashyap D Desai * that describes DMA segments, number of segments, error code 1906665484d8SDoug Ambrisko * 19078e727371SKashyap D Desai * This function is for the driver to receive mapping information resultant of 19088e727371SKashyap D Desai * the bus_dmamap_load(). The information is actually not being used, but the 19098e727371SKashyap D Desai * address is saved anyway. 1910665484d8SDoug Ambrisko */ 1911665484d8SDoug Ambrisko void 1912665484d8SDoug Ambrisko mrsas_addr_cb(void *arg, bus_dma_segment_t *segs, int nsegs, int error) 1913665484d8SDoug Ambrisko { 1914665484d8SDoug Ambrisko bus_addr_t *addr; 1915665484d8SDoug Ambrisko 1916665484d8SDoug Ambrisko addr = arg; 1917665484d8SDoug Ambrisko *addr = segs[0].ds_addr; 1918665484d8SDoug Ambrisko } 1919665484d8SDoug Ambrisko 1920665484d8SDoug Ambrisko /* 1921665484d8SDoug Ambrisko * mrsas_setup_raidmap: Set up RAID map. 1922665484d8SDoug Ambrisko * input: Adapter instance soft state 1923665484d8SDoug Ambrisko * 1924665484d8SDoug Ambrisko * Allocate DMA memory for the RAID maps and perform setup. 1925665484d8SDoug Ambrisko */ 19268e727371SKashyap D Desai static int 19278e727371SKashyap D Desai mrsas_setup_raidmap(struct mrsas_softc *sc) 1928665484d8SDoug Ambrisko { 19294799d485SKashyap D Desai int i; 19304799d485SKashyap D Desai 19314799d485SKashyap D Desai for (i = 0; i < 2; i++) { 19324799d485SKashyap D Desai sc->ld_drv_map[i] = 19334799d485SKashyap D Desai (void *)malloc(sc->drv_map_sz, M_MRSAS, M_NOWAIT); 19344799d485SKashyap D Desai /* Do Error handling */ 19354799d485SKashyap D Desai if (!sc->ld_drv_map[i]) { 19364799d485SKashyap D Desai device_printf(sc->mrsas_dev, "Could not allocate memory for local map"); 19374799d485SKashyap D Desai 19384799d485SKashyap D Desai if (i == 1) 19394799d485SKashyap D Desai free(sc->ld_drv_map[0], M_MRSAS); 19408e727371SKashyap D Desai /* ABORT driver initialization */ 19414799d485SKashyap D Desai goto ABORT; 19424799d485SKashyap D Desai } 19434799d485SKashyap D Desai } 19444799d485SKashyap D Desai 19458e727371SKashyap D Desai for (int i = 0; i < 2; i++) { 19468e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 19478e727371SKashyap D Desai 4, 0, 19488e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 19498e727371SKashyap D Desai BUS_SPACE_MAXADDR, 19508e727371SKashyap D Desai NULL, NULL, 19518e727371SKashyap D Desai sc->max_map_sz, 19528e727371SKashyap D Desai 1, 19538e727371SKashyap D Desai sc->max_map_sz, 19548e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 19558e727371SKashyap D Desai NULL, NULL, 1956665484d8SDoug Ambrisko &sc->raidmap_tag[i])) { 19574799d485SKashyap D Desai device_printf(sc->mrsas_dev, 19584799d485SKashyap D Desai "Cannot allocate raid map tag.\n"); 1959665484d8SDoug Ambrisko return (ENOMEM); 1960665484d8SDoug Ambrisko } 19614799d485SKashyap D Desai if (bus_dmamem_alloc(sc->raidmap_tag[i], 19624799d485SKashyap D Desai (void **)&sc->raidmap_mem[i], 1963665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->raidmap_dmamap[i])) { 19644799d485SKashyap D Desai device_printf(sc->mrsas_dev, 19654799d485SKashyap D Desai "Cannot allocate raidmap memory.\n"); 1966665484d8SDoug Ambrisko return (ENOMEM); 1967665484d8SDoug Ambrisko } 19684799d485SKashyap D Desai bzero(sc->raidmap_mem[i], sc->max_map_sz); 19694799d485SKashyap D Desai 1970665484d8SDoug Ambrisko if (bus_dmamap_load(sc->raidmap_tag[i], sc->raidmap_dmamap[i], 19714799d485SKashyap D Desai sc->raidmap_mem[i], sc->max_map_sz, 19724799d485SKashyap D Desai mrsas_addr_cb, &sc->raidmap_phys_addr[i], 1973665484d8SDoug Ambrisko BUS_DMA_NOWAIT)) { 1974665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load raidmap memory.\n"); 1975665484d8SDoug Ambrisko return (ENOMEM); 1976665484d8SDoug Ambrisko } 1977665484d8SDoug Ambrisko if (!sc->raidmap_mem[i]) { 19784799d485SKashyap D Desai device_printf(sc->mrsas_dev, 19794799d485SKashyap D Desai "Cannot allocate memory for raid map.\n"); 1980665484d8SDoug Ambrisko return (ENOMEM); 1981665484d8SDoug Ambrisko } 1982665484d8SDoug Ambrisko } 1983665484d8SDoug Ambrisko 1984665484d8SDoug Ambrisko if (!mrsas_get_map_info(sc)) 1985665484d8SDoug Ambrisko mrsas_sync_map_info(sc); 1986665484d8SDoug Ambrisko 1987665484d8SDoug Ambrisko return (0); 19884799d485SKashyap D Desai 19894799d485SKashyap D Desai ABORT: 19904799d485SKashyap D Desai return (1); 1991665484d8SDoug Ambrisko } 1992665484d8SDoug Ambrisko 19938e727371SKashyap D Desai /* 1994665484d8SDoug Ambrisko * mrsas_init_fw: Initialize Firmware 1995665484d8SDoug Ambrisko * input: Adapter soft state 1996665484d8SDoug Ambrisko * 19978e727371SKashyap D Desai * Calls transition_to_ready() to make sure Firmware is in operational state and 19988e727371SKashyap D Desai * calls mrsas_init_adapter() to send IOC_INIT command to Firmware. It 19998e727371SKashyap D Desai * issues internal commands to get the controller info after the IOC_INIT 20008e727371SKashyap D Desai * command response is received by Firmware. Note: code relating to 20018e727371SKashyap D Desai * get_pdlist, get_ld_list and max_sectors are currently not being used, it 20028e727371SKashyap D Desai * is left here as placeholder. 2003665484d8SDoug Ambrisko */ 20048e727371SKashyap D Desai static int 20058e727371SKashyap D Desai mrsas_init_fw(struct mrsas_softc *sc) 2006665484d8SDoug Ambrisko { 2007d18d1b47SKashyap D Desai 2008d18d1b47SKashyap D Desai int ret, loop, ocr = 0; 2009665484d8SDoug Ambrisko u_int32_t max_sectors_1; 2010665484d8SDoug Ambrisko u_int32_t max_sectors_2; 2011665484d8SDoug Ambrisko u_int32_t tmp_sectors; 2012d18d1b47SKashyap D Desai u_int32_t scratch_pad_2; 2013d18d1b47SKashyap D Desai int msix_enable = 0; 2014d18d1b47SKashyap D Desai int fw_msix_count = 0; 2015665484d8SDoug Ambrisko 2016665484d8SDoug Ambrisko /* Make sure Firmware is ready */ 2017665484d8SDoug Ambrisko ret = mrsas_transition_to_ready(sc, ocr); 2018665484d8SDoug Ambrisko if (ret != SUCCESS) { 2019665484d8SDoug Ambrisko return (ret); 2020665484d8SDoug Ambrisko } 2021d18d1b47SKashyap D Desai /* MSI-x index 0- reply post host index register */ 2022d18d1b47SKashyap D Desai sc->msix_reg_offset[0] = MPI2_REPLY_POST_HOST_INDEX_OFFSET; 2023d18d1b47SKashyap D Desai /* Check if MSI-X is supported while in ready state */ 2024d18d1b47SKashyap D Desai msix_enable = (mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad)) & 0x4000000) >> 0x1a; 2025d18d1b47SKashyap D Desai 2026d18d1b47SKashyap D Desai if (msix_enable) { 2027d18d1b47SKashyap D Desai scratch_pad_2 = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2028d18d1b47SKashyap D Desai outbound_scratch_pad_2)); 2029d18d1b47SKashyap D Desai 2030d18d1b47SKashyap D Desai /* Check max MSI-X vectors */ 2031d18d1b47SKashyap D Desai if (sc->device_id == MRSAS_TBOLT) { 2032d18d1b47SKashyap D Desai sc->msix_vectors = (scratch_pad_2 2033d18d1b47SKashyap D Desai & MR_MAX_REPLY_QUEUES_OFFSET) + 1; 2034d18d1b47SKashyap D Desai fw_msix_count = sc->msix_vectors; 2035d18d1b47SKashyap D Desai } else { 2036d18d1b47SKashyap D Desai /* Invader/Fury supports 96 MSI-X vectors */ 2037d18d1b47SKashyap D Desai sc->msix_vectors = ((scratch_pad_2 2038d18d1b47SKashyap D Desai & MR_MAX_REPLY_QUEUES_EXT_OFFSET) 2039d18d1b47SKashyap D Desai >> MR_MAX_REPLY_QUEUES_EXT_OFFSET_SHIFT) + 1; 2040d18d1b47SKashyap D Desai fw_msix_count = sc->msix_vectors; 2041d18d1b47SKashyap D Desai 2042d18d1b47SKashyap D Desai for (loop = 1; loop < MR_MAX_MSIX_REG_ARRAY; 2043d18d1b47SKashyap D Desai loop++) { 2044d18d1b47SKashyap D Desai sc->msix_reg_offset[loop] = 2045d18d1b47SKashyap D Desai MPI2_SUP_REPLY_POST_HOST_INDEX_OFFSET + 2046d18d1b47SKashyap D Desai (loop * 0x10); 2047d18d1b47SKashyap D Desai } 2048d18d1b47SKashyap D Desai } 2049d18d1b47SKashyap D Desai 2050d18d1b47SKashyap D Desai /* Don't bother allocating more MSI-X vectors than cpus */ 2051d18d1b47SKashyap D Desai sc->msix_vectors = min(sc->msix_vectors, 2052d18d1b47SKashyap D Desai mp_ncpus); 2053d18d1b47SKashyap D Desai 2054d18d1b47SKashyap D Desai /* Allocate MSI-x vectors */ 2055d18d1b47SKashyap D Desai if (mrsas_allocate_msix(sc) == SUCCESS) 2056d18d1b47SKashyap D Desai sc->msix_enable = 1; 2057d18d1b47SKashyap D Desai else 2058d18d1b47SKashyap D Desai sc->msix_enable = 0; 2059d18d1b47SKashyap D Desai 2060d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "FW supports <%d> MSIX vector," 2061d18d1b47SKashyap D Desai "Online CPU %d Current MSIX <%d>\n", 2062d18d1b47SKashyap D Desai fw_msix_count, mp_ncpus, sc->msix_vectors); 2063d18d1b47SKashyap D Desai } 2064665484d8SDoug Ambrisko if (mrsas_init_adapter(sc) != SUCCESS) { 2065665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Adapter initialize Fail.\n"); 2066665484d8SDoug Ambrisko return (1); 2067665484d8SDoug Ambrisko } 2068665484d8SDoug Ambrisko /* Allocate internal commands for pass-thru */ 2069665484d8SDoug Ambrisko if (mrsas_alloc_mfi_cmds(sc) != SUCCESS) { 2070665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Allocate MFI cmd failed.\n"); 2071665484d8SDoug Ambrisko return (1); 2072665484d8SDoug Ambrisko } 2073af51c29fSKashyap D Desai sc->ctrl_info = malloc(sizeof(struct mrsas_ctrl_info), M_MRSAS, M_NOWAIT); 2074af51c29fSKashyap D Desai if (!sc->ctrl_info) { 2075af51c29fSKashyap D Desai device_printf(sc->mrsas_dev, "Malloc for ctrl_info failed.\n"); 2076af51c29fSKashyap D Desai return (1); 2077af51c29fSKashyap D Desai } 20784799d485SKashyap D Desai /* 20798e727371SKashyap D Desai * Get the controller info from FW, so that the MAX VD support 20808e727371SKashyap D Desai * availability can be decided. 20814799d485SKashyap D Desai */ 2082af51c29fSKashyap D Desai if (mrsas_get_ctrl_info(sc)) { 20834799d485SKashyap D Desai device_printf(sc->mrsas_dev, "Unable to get FW ctrl_info.\n"); 2084af51c29fSKashyap D Desai return (1); 20854799d485SKashyap D Desai } 208677cf7df8SKashyap D Desai sc->secure_jbod_support = 2087af51c29fSKashyap D Desai (u_int8_t)sc->ctrl_info->adapterOperations3.supportSecurityonJBOD; 208877cf7df8SKashyap D Desai 208977cf7df8SKashyap D Desai if (sc->secure_jbod_support) 209077cf7df8SKashyap D Desai device_printf(sc->mrsas_dev, "FW supports SED \n"); 209177cf7df8SKashyap D Desai 2092665484d8SDoug Ambrisko if (mrsas_setup_raidmap(sc) != SUCCESS) { 2093665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Set up RAID map failed.\n"); 2094665484d8SDoug Ambrisko return (1); 2095665484d8SDoug Ambrisko } 2096665484d8SDoug Ambrisko /* For pass-thru, get PD/LD list and controller info */ 20974799d485SKashyap D Desai memset(sc->pd_list, 0, 20984799d485SKashyap D Desai MRSAS_MAX_PD * sizeof(struct mrsas_pd_list)); 2099665484d8SDoug Ambrisko mrsas_get_pd_list(sc); 2100665484d8SDoug Ambrisko 21014799d485SKashyap D Desai memset(sc->ld_ids, 0xff, MRSAS_MAX_LD_IDS); 2102665484d8SDoug Ambrisko mrsas_get_ld_list(sc); 2103665484d8SDoug Ambrisko 2104665484d8SDoug Ambrisko /* 21058e727371SKashyap D Desai * Compute the max allowed sectors per IO: The controller info has 21068e727371SKashyap D Desai * two limits on max sectors. Driver should use the minimum of these 21078e727371SKashyap D Desai * two. 2108665484d8SDoug Ambrisko * 2109665484d8SDoug Ambrisko * 1 << stripe_sz_ops.min = max sectors per strip 2110665484d8SDoug Ambrisko * 21118e727371SKashyap D Desai * Note that older firmwares ( < FW ver 30) didn't report information to 21128e727371SKashyap D Desai * calculate max_sectors_1. So the number ended up as zero always. 2113665484d8SDoug Ambrisko */ 2114665484d8SDoug Ambrisko tmp_sectors = 0; 2115af51c29fSKashyap D Desai max_sectors_1 = (1 << sc->ctrl_info->stripe_sz_ops.min) * 2116af51c29fSKashyap D Desai sc->ctrl_info->max_strips_per_io; 2117af51c29fSKashyap D Desai max_sectors_2 = sc->ctrl_info->max_request_size; 2118665484d8SDoug Ambrisko tmp_sectors = min(max_sectors_1, max_sectors_2); 21194799d485SKashyap D Desai sc->max_sectors_per_req = sc->max_num_sge * MRSAS_PAGE_SIZE / 512; 21204799d485SKashyap D Desai 21214799d485SKashyap D Desai if (tmp_sectors && (sc->max_sectors_per_req > tmp_sectors)) 21224799d485SKashyap D Desai sc->max_sectors_per_req = tmp_sectors; 21234799d485SKashyap D Desai 2124665484d8SDoug Ambrisko sc->disableOnlineCtrlReset = 2125af51c29fSKashyap D Desai sc->ctrl_info->properties.OnOffProperties.disableOnlineCtrlReset; 2126665484d8SDoug Ambrisko sc->UnevenSpanSupport = 2127af51c29fSKashyap D Desai sc->ctrl_info->adapterOperations2.supportUnevenSpans; 2128665484d8SDoug Ambrisko if (sc->UnevenSpanSupport) { 21298e727371SKashyap D Desai device_printf(sc->mrsas_dev, "FW supports: UnevenSpanSupport=%x\n\n", 2130665484d8SDoug Ambrisko sc->UnevenSpanSupport); 21314799d485SKashyap D Desai 2132665484d8SDoug Ambrisko if (MR_ValidateMapInfo(sc)) 2133665484d8SDoug Ambrisko sc->fast_path_io = 1; 2134665484d8SDoug Ambrisko else 2135665484d8SDoug Ambrisko sc->fast_path_io = 0; 2136665484d8SDoug Ambrisko } 2137665484d8SDoug Ambrisko return (0); 2138665484d8SDoug Ambrisko } 2139665484d8SDoug Ambrisko 21408e727371SKashyap D Desai /* 2141665484d8SDoug Ambrisko * mrsas_init_adapter: Initializes the adapter/controller 2142665484d8SDoug Ambrisko * input: Adapter soft state 2143665484d8SDoug Ambrisko * 2144665484d8SDoug Ambrisko * Prepares for the issuing of the IOC Init cmd to FW for initializing the 2145665484d8SDoug Ambrisko * ROC/controller. The FW register is read to determined the number of 2146665484d8SDoug Ambrisko * commands that is supported. All memory allocations for IO is based on 2147665484d8SDoug Ambrisko * max_cmd. Appropriate calculations are performed in this function. 2148665484d8SDoug Ambrisko */ 21498e727371SKashyap D Desai int 21508e727371SKashyap D Desai mrsas_init_adapter(struct mrsas_softc *sc) 2151665484d8SDoug Ambrisko { 2152665484d8SDoug Ambrisko uint32_t status; 2153665484d8SDoug Ambrisko u_int32_t max_cmd; 2154665484d8SDoug Ambrisko int ret; 2155d18d1b47SKashyap D Desai int i = 0; 2156665484d8SDoug Ambrisko 2157665484d8SDoug Ambrisko /* Read FW status register */ 2158665484d8SDoug Ambrisko status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad)); 2159665484d8SDoug Ambrisko 2160665484d8SDoug Ambrisko /* Get operational params from status register */ 2161665484d8SDoug Ambrisko sc->max_fw_cmds = status & MRSAS_FWSTATE_MAXCMD_MASK; 2162665484d8SDoug Ambrisko 2163665484d8SDoug Ambrisko /* Decrement the max supported by 1, to correlate with FW */ 2164665484d8SDoug Ambrisko sc->max_fw_cmds = sc->max_fw_cmds - 1; 2165665484d8SDoug Ambrisko max_cmd = sc->max_fw_cmds; 2166665484d8SDoug Ambrisko 2167665484d8SDoug Ambrisko /* Determine allocation size of command frames */ 21682f863eb8SKashyap D Desai sc->reply_q_depth = ((max_cmd + 1 + 15) / 16 * 16) * 2; 2169665484d8SDoug Ambrisko sc->request_alloc_sz = sizeof(MRSAS_REQUEST_DESCRIPTOR_UNION) * max_cmd; 2170665484d8SDoug Ambrisko sc->reply_alloc_sz = sizeof(MPI2_REPLY_DESCRIPTORS_UNION) * (sc->reply_q_depth); 2171665484d8SDoug Ambrisko sc->io_frames_alloc_sz = MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE + (MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE * (max_cmd + 1)); 2172665484d8SDoug Ambrisko sc->chain_frames_alloc_sz = 1024 * max_cmd; 2173665484d8SDoug Ambrisko sc->max_sge_in_main_msg = (MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE - 2174665484d8SDoug Ambrisko offsetof(MRSAS_RAID_SCSI_IO_REQUEST, SGL)) / 16; 2175665484d8SDoug Ambrisko 2176665484d8SDoug Ambrisko sc->max_sge_in_chain = MRSAS_MAX_SZ_CHAIN_FRAME / sizeof(MPI2_SGE_IO_UNION); 2177665484d8SDoug Ambrisko sc->max_num_sge = sc->max_sge_in_main_msg + sc->max_sge_in_chain - 2; 2178665484d8SDoug Ambrisko 2179665484d8SDoug Ambrisko /* Used for pass thru MFI frame (DCMD) */ 2180665484d8SDoug Ambrisko sc->chain_offset_mfi_pthru = offsetof(MRSAS_RAID_SCSI_IO_REQUEST, SGL) / 16; 2181665484d8SDoug Ambrisko 2182665484d8SDoug Ambrisko sc->chain_offset_io_request = (MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE - 2183665484d8SDoug Ambrisko sizeof(MPI2_SGE_IO_UNION)) / 16; 2184665484d8SDoug Ambrisko 2185d18d1b47SKashyap D Desai int count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 21868e727371SKashyap D Desai 2187d18d1b47SKashyap D Desai for (i = 0; i < count; i++) 2188d18d1b47SKashyap D Desai sc->last_reply_idx[i] = 0; 2189665484d8SDoug Ambrisko 2190665484d8SDoug Ambrisko ret = mrsas_alloc_mem(sc); 2191665484d8SDoug Ambrisko if (ret != SUCCESS) 2192665484d8SDoug Ambrisko return (ret); 2193665484d8SDoug Ambrisko 2194665484d8SDoug Ambrisko ret = mrsas_alloc_mpt_cmds(sc); 2195665484d8SDoug Ambrisko if (ret != SUCCESS) 2196665484d8SDoug Ambrisko return (ret); 2197665484d8SDoug Ambrisko 2198665484d8SDoug Ambrisko ret = mrsas_ioc_init(sc); 2199665484d8SDoug Ambrisko if (ret != SUCCESS) 2200665484d8SDoug Ambrisko return (ret); 2201665484d8SDoug Ambrisko 2202665484d8SDoug Ambrisko return (0); 2203665484d8SDoug Ambrisko } 2204665484d8SDoug Ambrisko 22058e727371SKashyap D Desai /* 2206665484d8SDoug Ambrisko * mrsas_alloc_ioc_cmd: Allocates memory for IOC Init command 2207665484d8SDoug Ambrisko * input: Adapter soft state 2208665484d8SDoug Ambrisko * 2209665484d8SDoug Ambrisko * Allocates for the IOC Init cmd to FW to initialize the ROC/controller. 2210665484d8SDoug Ambrisko */ 22118e727371SKashyap D Desai int 22128e727371SKashyap D Desai mrsas_alloc_ioc_cmd(struct mrsas_softc *sc) 2213665484d8SDoug Ambrisko { 2214665484d8SDoug Ambrisko int ioc_init_size; 2215665484d8SDoug Ambrisko 2216665484d8SDoug Ambrisko /* Allocate IOC INIT command */ 2217665484d8SDoug Ambrisko ioc_init_size = 1024 + sizeof(MPI2_IOC_INIT_REQUEST); 22188e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 22198e727371SKashyap D Desai 1, 0, 22208e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 22218e727371SKashyap D Desai BUS_SPACE_MAXADDR, 22228e727371SKashyap D Desai NULL, NULL, 22238e727371SKashyap D Desai ioc_init_size, 22248e727371SKashyap D Desai 1, 22258e727371SKashyap D Desai ioc_init_size, 22268e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 22278e727371SKashyap D Desai NULL, NULL, 2228665484d8SDoug Ambrisko &sc->ioc_init_tag)) { 2229665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate ioc init tag\n"); 2230665484d8SDoug Ambrisko return (ENOMEM); 2231665484d8SDoug Ambrisko } 2232665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->ioc_init_tag, (void **)&sc->ioc_init_mem, 2233665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->ioc_init_dmamap)) { 2234665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate ioc init cmd mem\n"); 2235665484d8SDoug Ambrisko return (ENOMEM); 2236665484d8SDoug Ambrisko } 2237665484d8SDoug Ambrisko bzero(sc->ioc_init_mem, ioc_init_size); 2238665484d8SDoug Ambrisko if (bus_dmamap_load(sc->ioc_init_tag, sc->ioc_init_dmamap, 2239665484d8SDoug Ambrisko sc->ioc_init_mem, ioc_init_size, mrsas_addr_cb, 2240665484d8SDoug Ambrisko &sc->ioc_init_phys_mem, BUS_DMA_NOWAIT)) { 2241665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load ioc init cmd mem\n"); 2242665484d8SDoug Ambrisko return (ENOMEM); 2243665484d8SDoug Ambrisko } 2244665484d8SDoug Ambrisko return (0); 2245665484d8SDoug Ambrisko } 2246665484d8SDoug Ambrisko 22478e727371SKashyap D Desai /* 2248665484d8SDoug Ambrisko * mrsas_free_ioc_cmd: Allocates memory for IOC Init command 2249665484d8SDoug Ambrisko * input: Adapter soft state 2250665484d8SDoug Ambrisko * 2251665484d8SDoug Ambrisko * Deallocates memory of the IOC Init cmd. 2252665484d8SDoug Ambrisko */ 22538e727371SKashyap D Desai void 22548e727371SKashyap D Desai mrsas_free_ioc_cmd(struct mrsas_softc *sc) 2255665484d8SDoug Ambrisko { 2256665484d8SDoug Ambrisko if (sc->ioc_init_phys_mem) 2257665484d8SDoug Ambrisko bus_dmamap_unload(sc->ioc_init_tag, sc->ioc_init_dmamap); 2258665484d8SDoug Ambrisko if (sc->ioc_init_mem != NULL) 2259665484d8SDoug Ambrisko bus_dmamem_free(sc->ioc_init_tag, sc->ioc_init_mem, sc->ioc_init_dmamap); 2260665484d8SDoug Ambrisko if (sc->ioc_init_tag != NULL) 2261665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->ioc_init_tag); 2262665484d8SDoug Ambrisko } 2263665484d8SDoug Ambrisko 22648e727371SKashyap D Desai /* 2265665484d8SDoug Ambrisko * mrsas_ioc_init: Sends IOC Init command to FW 2266665484d8SDoug Ambrisko * input: Adapter soft state 2267665484d8SDoug Ambrisko * 2268665484d8SDoug Ambrisko * Issues the IOC Init cmd to FW to initialize the ROC/controller. 2269665484d8SDoug Ambrisko */ 22708e727371SKashyap D Desai int 22718e727371SKashyap D Desai mrsas_ioc_init(struct mrsas_softc *sc) 2272665484d8SDoug Ambrisko { 2273665484d8SDoug Ambrisko struct mrsas_init_frame *init_frame; 2274665484d8SDoug Ambrisko pMpi2IOCInitRequest_t IOCInitMsg; 2275665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION req_desc; 2276665484d8SDoug Ambrisko u_int8_t max_wait = MRSAS_IOC_INIT_WAIT_TIME; 2277665484d8SDoug Ambrisko bus_addr_t phys_addr; 2278665484d8SDoug Ambrisko int i, retcode = 0; 2279665484d8SDoug Ambrisko 2280665484d8SDoug Ambrisko /* Allocate memory for the IOC INIT command */ 2281665484d8SDoug Ambrisko if (mrsas_alloc_ioc_cmd(sc)) { 2282665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate IOC command.\n"); 2283665484d8SDoug Ambrisko return (1); 2284665484d8SDoug Ambrisko } 2285665484d8SDoug Ambrisko IOCInitMsg = (pMpi2IOCInitRequest_t)(((char *)sc->ioc_init_mem) + 1024); 2286665484d8SDoug Ambrisko IOCInitMsg->Function = MPI2_FUNCTION_IOC_INIT; 2287665484d8SDoug Ambrisko IOCInitMsg->WhoInit = MPI2_WHOINIT_HOST_DRIVER; 2288665484d8SDoug Ambrisko IOCInitMsg->MsgVersion = MPI2_VERSION; 2289665484d8SDoug Ambrisko IOCInitMsg->HeaderVersion = MPI2_HEADER_VERSION; 2290665484d8SDoug Ambrisko IOCInitMsg->SystemRequestFrameSize = MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE / 4; 2291665484d8SDoug Ambrisko IOCInitMsg->ReplyDescriptorPostQueueDepth = sc->reply_q_depth; 2292665484d8SDoug Ambrisko IOCInitMsg->ReplyDescriptorPostQueueAddress = sc->reply_desc_phys_addr; 2293665484d8SDoug Ambrisko IOCInitMsg->SystemRequestFrameBaseAddress = sc->io_request_phys_addr; 2294d18d1b47SKashyap D Desai IOCInitMsg->HostMSIxVectors = (sc->msix_vectors > 0 ? sc->msix_vectors : 0); 2295665484d8SDoug Ambrisko 2296665484d8SDoug Ambrisko init_frame = (struct mrsas_init_frame *)sc->ioc_init_mem; 2297665484d8SDoug Ambrisko init_frame->cmd = MFI_CMD_INIT; 2298665484d8SDoug Ambrisko init_frame->cmd_status = 0xFF; 2299665484d8SDoug Ambrisko init_frame->flags |= MFI_FRAME_DONT_POST_IN_REPLY_QUEUE; 2300665484d8SDoug Ambrisko 2301d18d1b47SKashyap D Desai /* driver support Extended MSIX */ 2302d18d1b47SKashyap D Desai if ((sc->device_id == MRSAS_INVADER) || 2303d18d1b47SKashyap D Desai (sc->device_id == MRSAS_FURY)) { 2304d18d1b47SKashyap D Desai init_frame->driver_operations. 2305d18d1b47SKashyap D Desai mfi_capabilities.support_additional_msix = 1; 2306d18d1b47SKashyap D Desai } 2307665484d8SDoug Ambrisko if (sc->verbuf_mem) { 2308665484d8SDoug Ambrisko snprintf((char *)sc->verbuf_mem, strlen(MRSAS_VERSION) + 2, "%s\n", 2309665484d8SDoug Ambrisko MRSAS_VERSION); 2310665484d8SDoug Ambrisko init_frame->driver_ver_lo = (bus_addr_t)sc->verbuf_phys_addr; 2311665484d8SDoug Ambrisko init_frame->driver_ver_hi = 0; 2312665484d8SDoug Ambrisko } 231316dc2814SKashyap D Desai init_frame->driver_operations.mfi_capabilities.support_ndrive_r1_lb = 1; 23144799d485SKashyap D Desai init_frame->driver_operations.mfi_capabilities.support_max_255lds = 1; 231577cf7df8SKashyap D Desai init_frame->driver_operations.mfi_capabilities.security_protocol_cmds_fw = 1; 2316665484d8SDoug Ambrisko phys_addr = (bus_addr_t)sc->ioc_init_phys_mem + 1024; 2317665484d8SDoug Ambrisko init_frame->queue_info_new_phys_addr_lo = phys_addr; 2318665484d8SDoug Ambrisko init_frame->data_xfer_len = sizeof(Mpi2IOCInitRequest_t); 2319665484d8SDoug Ambrisko 2320665484d8SDoug Ambrisko req_desc.addr.Words = (bus_addr_t)sc->ioc_init_phys_mem; 2321665484d8SDoug Ambrisko req_desc.MFAIo.RequestFlags = 2322665484d8SDoug Ambrisko (MRSAS_REQ_DESCRIPT_FLAGS_MFA << MRSAS_REQ_DESCRIPT_FLAGS_TYPE_SHIFT); 2323665484d8SDoug Ambrisko 2324665484d8SDoug Ambrisko mrsas_disable_intr(sc); 2325665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "Issuing IOC INIT command to FW.\n"); 2326665484d8SDoug Ambrisko mrsas_fire_cmd(sc, req_desc.addr.u.low, req_desc.addr.u.high); 2327665484d8SDoug Ambrisko 2328665484d8SDoug Ambrisko /* 2329665484d8SDoug Ambrisko * Poll response timer to wait for Firmware response. While this 2330665484d8SDoug Ambrisko * timer with the DELAY call could block CPU, the time interval for 2331665484d8SDoug Ambrisko * this is only 1 millisecond. 2332665484d8SDoug Ambrisko */ 2333665484d8SDoug Ambrisko if (init_frame->cmd_status == 0xFF) { 2334665484d8SDoug Ambrisko for (i = 0; i < (max_wait * 1000); i++) { 2335665484d8SDoug Ambrisko if (init_frame->cmd_status == 0xFF) 2336665484d8SDoug Ambrisko DELAY(1000); 2337665484d8SDoug Ambrisko else 2338665484d8SDoug Ambrisko break; 2339665484d8SDoug Ambrisko } 2340665484d8SDoug Ambrisko } 2341665484d8SDoug Ambrisko if (init_frame->cmd_status == 0) 2342665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2343665484d8SDoug Ambrisko "IOC INIT response received from FW.\n"); 23448e727371SKashyap D Desai else { 2345665484d8SDoug Ambrisko if (init_frame->cmd_status == 0xFF) 2346665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "IOC Init timed out after %d seconds.\n", max_wait); 2347665484d8SDoug Ambrisko else 2348665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "IOC Init failed, status = 0x%x\n", init_frame->cmd_status); 2349665484d8SDoug Ambrisko retcode = 1; 2350665484d8SDoug Ambrisko } 2351665484d8SDoug Ambrisko 2352665484d8SDoug Ambrisko mrsas_free_ioc_cmd(sc); 2353665484d8SDoug Ambrisko return (retcode); 2354665484d8SDoug Ambrisko } 2355665484d8SDoug Ambrisko 23568e727371SKashyap D Desai /* 2357665484d8SDoug Ambrisko * mrsas_alloc_mpt_cmds: Allocates the command packets 2358665484d8SDoug Ambrisko * input: Adapter instance soft state 2359665484d8SDoug Ambrisko * 2360665484d8SDoug Ambrisko * This function allocates the internal commands for IOs. Each command that is 23618e727371SKashyap D Desai * issued to FW is wrapped in a local data structure called mrsas_mpt_cmd. An 23628e727371SKashyap D Desai * array is allocated with mrsas_mpt_cmd context. The free commands are 2363665484d8SDoug Ambrisko * maintained in a linked list (cmd pool). SMID value range is from 1 to 2364665484d8SDoug Ambrisko * max_fw_cmds. 2365665484d8SDoug Ambrisko */ 23668e727371SKashyap D Desai int 23678e727371SKashyap D Desai mrsas_alloc_mpt_cmds(struct mrsas_softc *sc) 2368665484d8SDoug Ambrisko { 2369665484d8SDoug Ambrisko int i, j; 2370d18d1b47SKashyap D Desai u_int32_t max_cmd, count; 2371665484d8SDoug Ambrisko struct mrsas_mpt_cmd *cmd; 2372665484d8SDoug Ambrisko pMpi2ReplyDescriptorsUnion_t reply_desc; 2373665484d8SDoug Ambrisko u_int32_t offset, chain_offset, sense_offset; 2374665484d8SDoug Ambrisko bus_addr_t io_req_base_phys, chain_frame_base_phys, sense_base_phys; 2375665484d8SDoug Ambrisko u_int8_t *io_req_base, *chain_frame_base, *sense_base; 2376665484d8SDoug Ambrisko 2377665484d8SDoug Ambrisko max_cmd = sc->max_fw_cmds; 2378665484d8SDoug Ambrisko 2379665484d8SDoug Ambrisko sc->req_desc = malloc(sc->request_alloc_sz, M_MRSAS, M_NOWAIT); 2380665484d8SDoug Ambrisko if (!sc->req_desc) { 2381665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Out of memory, cannot alloc req desc\n"); 2382665484d8SDoug Ambrisko return (ENOMEM); 2383665484d8SDoug Ambrisko } 2384665484d8SDoug Ambrisko memset(sc->req_desc, 0, sc->request_alloc_sz); 2385665484d8SDoug Ambrisko 2386665484d8SDoug Ambrisko /* 23878e727371SKashyap D Desai * sc->mpt_cmd_list is an array of struct mrsas_mpt_cmd pointers. 23888e727371SKashyap D Desai * Allocate the dynamic array first and then allocate individual 23898e727371SKashyap D Desai * commands. 2390665484d8SDoug Ambrisko */ 2391665484d8SDoug Ambrisko sc->mpt_cmd_list = malloc(sizeof(struct mrsas_mpt_cmd *) * max_cmd, M_MRSAS, M_NOWAIT); 2392665484d8SDoug Ambrisko if (!sc->mpt_cmd_list) { 2393665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc memory for mpt_cmd_list.\n"); 2394665484d8SDoug Ambrisko return (ENOMEM); 2395665484d8SDoug Ambrisko } 2396665484d8SDoug Ambrisko memset(sc->mpt_cmd_list, 0, sizeof(struct mrsas_mpt_cmd *) * max_cmd); 2397665484d8SDoug Ambrisko for (i = 0; i < max_cmd; i++) { 2398665484d8SDoug Ambrisko sc->mpt_cmd_list[i] = malloc(sizeof(struct mrsas_mpt_cmd), 2399665484d8SDoug Ambrisko M_MRSAS, M_NOWAIT); 2400665484d8SDoug Ambrisko if (!sc->mpt_cmd_list[i]) { 2401665484d8SDoug Ambrisko for (j = 0; j < i; j++) 2402665484d8SDoug Ambrisko free(sc->mpt_cmd_list[j], M_MRSAS); 2403665484d8SDoug Ambrisko free(sc->mpt_cmd_list, M_MRSAS); 2404665484d8SDoug Ambrisko sc->mpt_cmd_list = NULL; 2405665484d8SDoug Ambrisko return (ENOMEM); 2406665484d8SDoug Ambrisko } 2407665484d8SDoug Ambrisko } 2408665484d8SDoug Ambrisko 2409665484d8SDoug Ambrisko io_req_base = (u_int8_t *)sc->io_request_mem + MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE; 2410665484d8SDoug Ambrisko io_req_base_phys = (bus_addr_t)sc->io_request_phys_addr + MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE; 2411665484d8SDoug Ambrisko chain_frame_base = (u_int8_t *)sc->chain_frame_mem; 2412665484d8SDoug Ambrisko chain_frame_base_phys = (bus_addr_t)sc->chain_frame_phys_addr; 2413665484d8SDoug Ambrisko sense_base = (u_int8_t *)sc->sense_mem; 2414665484d8SDoug Ambrisko sense_base_phys = (bus_addr_t)sc->sense_phys_addr; 2415665484d8SDoug Ambrisko for (i = 0; i < max_cmd; i++) { 2416665484d8SDoug Ambrisko cmd = sc->mpt_cmd_list[i]; 2417665484d8SDoug Ambrisko offset = MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE * i; 2418665484d8SDoug Ambrisko chain_offset = 1024 * i; 2419665484d8SDoug Ambrisko sense_offset = MRSAS_SENSE_LEN * i; 2420665484d8SDoug Ambrisko memset(cmd, 0, sizeof(struct mrsas_mpt_cmd)); 2421665484d8SDoug Ambrisko cmd->index = i + 1; 2422665484d8SDoug Ambrisko cmd->ccb_ptr = NULL; 2423665484d8SDoug Ambrisko callout_init(&cmd->cm_callout, 0); 2424665484d8SDoug Ambrisko cmd->sync_cmd_idx = (u_int32_t)MRSAS_ULONG_MAX; 2425665484d8SDoug Ambrisko cmd->sc = sc; 2426665484d8SDoug Ambrisko cmd->io_request = (MRSAS_RAID_SCSI_IO_REQUEST *) (io_req_base + offset); 2427665484d8SDoug Ambrisko memset(cmd->io_request, 0, sizeof(MRSAS_RAID_SCSI_IO_REQUEST)); 2428665484d8SDoug Ambrisko cmd->io_request_phys_addr = io_req_base_phys + offset; 2429665484d8SDoug Ambrisko cmd->chain_frame = (MPI2_SGE_IO_UNION *) (chain_frame_base + chain_offset); 2430665484d8SDoug Ambrisko cmd->chain_frame_phys_addr = chain_frame_base_phys + chain_offset; 2431665484d8SDoug Ambrisko cmd->sense = sense_base + sense_offset; 2432665484d8SDoug Ambrisko cmd->sense_phys_addr = sense_base_phys + sense_offset; 2433665484d8SDoug Ambrisko if (bus_dmamap_create(sc->data_tag, 0, &cmd->data_dmamap)) { 2434665484d8SDoug Ambrisko return (FAIL); 2435665484d8SDoug Ambrisko } 2436665484d8SDoug Ambrisko TAILQ_INSERT_TAIL(&(sc->mrsas_mpt_cmd_list_head), cmd, next); 2437665484d8SDoug Ambrisko } 2438665484d8SDoug Ambrisko 2439665484d8SDoug Ambrisko /* Initialize reply descriptor array to 0xFFFFFFFF */ 2440665484d8SDoug Ambrisko reply_desc = sc->reply_desc_mem; 2441d18d1b47SKashyap D Desai count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 2442d18d1b47SKashyap D Desai for (i = 0; i < sc->reply_q_depth * count; i++, reply_desc++) { 2443665484d8SDoug Ambrisko reply_desc->Words = MRSAS_ULONG_MAX; 2444665484d8SDoug Ambrisko } 2445665484d8SDoug Ambrisko return (0); 2446665484d8SDoug Ambrisko } 2447665484d8SDoug Ambrisko 24488e727371SKashyap D Desai /* 2449665484d8SDoug Ambrisko * mrsas_fire_cmd: Sends command to FW 2450665484d8SDoug Ambrisko * input: Adapter softstate 2451665484d8SDoug Ambrisko * request descriptor address low 2452665484d8SDoug Ambrisko * request descriptor address high 2453665484d8SDoug Ambrisko * 2454665484d8SDoug Ambrisko * This functions fires the command to Firmware by writing to the 2455665484d8SDoug Ambrisko * inbound_low_queue_port and inbound_high_queue_port. 2456665484d8SDoug Ambrisko */ 24578e727371SKashyap D Desai void 24588e727371SKashyap D Desai mrsas_fire_cmd(struct mrsas_softc *sc, u_int32_t req_desc_lo, 2459665484d8SDoug Ambrisko u_int32_t req_desc_hi) 2460665484d8SDoug Ambrisko { 2461665484d8SDoug Ambrisko mtx_lock(&sc->pci_lock); 2462665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, inbound_low_queue_port), 2463665484d8SDoug Ambrisko req_desc_lo); 2464665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, inbound_high_queue_port), 2465665484d8SDoug Ambrisko req_desc_hi); 2466665484d8SDoug Ambrisko mtx_unlock(&sc->pci_lock); 2467665484d8SDoug Ambrisko } 2468665484d8SDoug Ambrisko 24698e727371SKashyap D Desai /* 24708e727371SKashyap D Desai * mrsas_transition_to_ready: Move FW to Ready state input: 24718e727371SKashyap D Desai * Adapter instance soft state 2472665484d8SDoug Ambrisko * 24738e727371SKashyap D Desai * During the initialization, FW passes can potentially be in any one of several 24748e727371SKashyap D Desai * possible states. If the FW in operational, waiting-for-handshake states, 24758e727371SKashyap D Desai * driver must take steps to bring it to ready state. Otherwise, it has to 24768e727371SKashyap D Desai * wait for the ready state. 2477665484d8SDoug Ambrisko */ 24788e727371SKashyap D Desai int 24798e727371SKashyap D Desai mrsas_transition_to_ready(struct mrsas_softc *sc, int ocr) 2480665484d8SDoug Ambrisko { 2481665484d8SDoug Ambrisko int i; 2482665484d8SDoug Ambrisko u_int8_t max_wait; 2483665484d8SDoug Ambrisko u_int32_t val, fw_state; 2484665484d8SDoug Ambrisko u_int32_t cur_state; 2485665484d8SDoug Ambrisko u_int32_t abs_state, curr_abs_state; 2486665484d8SDoug Ambrisko 2487665484d8SDoug Ambrisko val = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad)); 2488665484d8SDoug Ambrisko fw_state = val & MFI_STATE_MASK; 2489665484d8SDoug Ambrisko max_wait = MRSAS_RESET_WAIT_TIME; 2490665484d8SDoug Ambrisko 2491665484d8SDoug Ambrisko if (fw_state != MFI_STATE_READY) 2492665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Waiting for FW to come to ready state\n"); 2493665484d8SDoug Ambrisko 2494665484d8SDoug Ambrisko while (fw_state != MFI_STATE_READY) { 2495665484d8SDoug Ambrisko abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad)); 2496665484d8SDoug Ambrisko switch (fw_state) { 2497665484d8SDoug Ambrisko case MFI_STATE_FAULT: 2498665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "FW is in FAULT state!!\n"); 2499665484d8SDoug Ambrisko if (ocr) { 2500665484d8SDoug Ambrisko cur_state = MFI_STATE_FAULT; 2501665484d8SDoug Ambrisko break; 25028e727371SKashyap D Desai } else 2503665484d8SDoug Ambrisko return -ENODEV; 2504665484d8SDoug Ambrisko case MFI_STATE_WAIT_HANDSHAKE: 2505665484d8SDoug Ambrisko /* Set the CLR bit in inbound doorbell */ 2506665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell), 2507665484d8SDoug Ambrisko MFI_INIT_CLEAR_HANDSHAKE | MFI_INIT_HOTPLUG); 2508665484d8SDoug Ambrisko cur_state = MFI_STATE_WAIT_HANDSHAKE; 2509665484d8SDoug Ambrisko break; 2510665484d8SDoug Ambrisko case MFI_STATE_BOOT_MESSAGE_PENDING: 2511665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell), 2512665484d8SDoug Ambrisko MFI_INIT_HOTPLUG); 2513665484d8SDoug Ambrisko cur_state = MFI_STATE_BOOT_MESSAGE_PENDING; 2514665484d8SDoug Ambrisko break; 2515665484d8SDoug Ambrisko case MFI_STATE_OPERATIONAL: 25168e727371SKashyap D Desai /* 25178e727371SKashyap D Desai * Bring it to READY state; assuming max wait 10 25188e727371SKashyap D Desai * secs 25198e727371SKashyap D Desai */ 2520665484d8SDoug Ambrisko mrsas_disable_intr(sc); 2521665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell), MFI_RESET_FLAGS); 2522665484d8SDoug Ambrisko for (i = 0; i < max_wait * 1000; i++) { 2523665484d8SDoug Ambrisko if (mrsas_read_reg(sc, offsetof(mrsas_reg_set, doorbell)) & 1) 2524665484d8SDoug Ambrisko DELAY(1000); 2525665484d8SDoug Ambrisko else 2526665484d8SDoug Ambrisko break; 2527665484d8SDoug Ambrisko } 2528665484d8SDoug Ambrisko cur_state = MFI_STATE_OPERATIONAL; 2529665484d8SDoug Ambrisko break; 2530665484d8SDoug Ambrisko case MFI_STATE_UNDEFINED: 25318e727371SKashyap D Desai /* 25328e727371SKashyap D Desai * This state should not last for more than 2 25338e727371SKashyap D Desai * seconds 25348e727371SKashyap D Desai */ 2535665484d8SDoug Ambrisko cur_state = MFI_STATE_UNDEFINED; 2536665484d8SDoug Ambrisko break; 2537665484d8SDoug Ambrisko case MFI_STATE_BB_INIT: 2538665484d8SDoug Ambrisko cur_state = MFI_STATE_BB_INIT; 2539665484d8SDoug Ambrisko break; 2540665484d8SDoug Ambrisko case MFI_STATE_FW_INIT: 2541665484d8SDoug Ambrisko cur_state = MFI_STATE_FW_INIT; 2542665484d8SDoug Ambrisko break; 2543665484d8SDoug Ambrisko case MFI_STATE_FW_INIT_2: 2544665484d8SDoug Ambrisko cur_state = MFI_STATE_FW_INIT_2; 2545665484d8SDoug Ambrisko break; 2546665484d8SDoug Ambrisko case MFI_STATE_DEVICE_SCAN: 2547665484d8SDoug Ambrisko cur_state = MFI_STATE_DEVICE_SCAN; 2548665484d8SDoug Ambrisko break; 2549665484d8SDoug Ambrisko case MFI_STATE_FLUSH_CACHE: 2550665484d8SDoug Ambrisko cur_state = MFI_STATE_FLUSH_CACHE; 2551665484d8SDoug Ambrisko break; 2552665484d8SDoug Ambrisko default: 2553665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Unknown state 0x%x\n", fw_state); 2554665484d8SDoug Ambrisko return -ENODEV; 2555665484d8SDoug Ambrisko } 2556665484d8SDoug Ambrisko 2557665484d8SDoug Ambrisko /* 2558665484d8SDoug Ambrisko * The cur_state should not last for more than max_wait secs 2559665484d8SDoug Ambrisko */ 2560665484d8SDoug Ambrisko for (i = 0; i < (max_wait * 1000); i++) { 2561665484d8SDoug Ambrisko fw_state = (mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2562665484d8SDoug Ambrisko outbound_scratch_pad)) & MFI_STATE_MASK); 2563665484d8SDoug Ambrisko curr_abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2564665484d8SDoug Ambrisko outbound_scratch_pad)); 2565665484d8SDoug Ambrisko if (abs_state == curr_abs_state) 2566665484d8SDoug Ambrisko DELAY(1000); 2567665484d8SDoug Ambrisko else 2568665484d8SDoug Ambrisko break; 2569665484d8SDoug Ambrisko } 2570665484d8SDoug Ambrisko 2571665484d8SDoug Ambrisko /* 2572665484d8SDoug Ambrisko * Return error if fw_state hasn't changed after max_wait 2573665484d8SDoug Ambrisko */ 2574665484d8SDoug Ambrisko if (curr_abs_state == abs_state) { 2575665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "FW state [%d] hasn't changed " 2576665484d8SDoug Ambrisko "in %d secs\n", fw_state, max_wait); 2577665484d8SDoug Ambrisko return -ENODEV; 2578665484d8SDoug Ambrisko } 2579665484d8SDoug Ambrisko } 2580665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "FW now in Ready state\n"); 2581665484d8SDoug Ambrisko return 0; 2582665484d8SDoug Ambrisko } 2583665484d8SDoug Ambrisko 25848e727371SKashyap D Desai /* 2585665484d8SDoug Ambrisko * mrsas_get_mfi_cmd: Get a cmd from free command pool 2586665484d8SDoug Ambrisko * input: Adapter soft state 2587665484d8SDoug Ambrisko * 2588665484d8SDoug Ambrisko * This function removes an MFI command from the command list. 2589665484d8SDoug Ambrisko */ 25908e727371SKashyap D Desai struct mrsas_mfi_cmd * 25918e727371SKashyap D Desai mrsas_get_mfi_cmd(struct mrsas_softc *sc) 2592665484d8SDoug Ambrisko { 2593665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd = NULL; 2594665484d8SDoug Ambrisko 2595665484d8SDoug Ambrisko mtx_lock(&sc->mfi_cmd_pool_lock); 2596665484d8SDoug Ambrisko if (!TAILQ_EMPTY(&sc->mrsas_mfi_cmd_list_head)) { 2597665484d8SDoug Ambrisko cmd = TAILQ_FIRST(&sc->mrsas_mfi_cmd_list_head); 2598665484d8SDoug Ambrisko TAILQ_REMOVE(&sc->mrsas_mfi_cmd_list_head, cmd, next); 2599665484d8SDoug Ambrisko } 2600665484d8SDoug Ambrisko mtx_unlock(&sc->mfi_cmd_pool_lock); 2601665484d8SDoug Ambrisko 2602665484d8SDoug Ambrisko return cmd; 2603665484d8SDoug Ambrisko } 2604665484d8SDoug Ambrisko 26058e727371SKashyap D Desai /* 26068e727371SKashyap D Desai * mrsas_ocr_thread: Thread to handle OCR/Kill Adapter. 2607665484d8SDoug Ambrisko * input: Adapter Context. 2608665484d8SDoug Ambrisko * 26098e727371SKashyap D Desai * This function will check FW status register and flag do_timeout_reset flag. 26108e727371SKashyap D Desai * It will do OCR/Kill adapter if FW is in fault state or IO timed out has 26118e727371SKashyap D Desai * trigger reset. 2612665484d8SDoug Ambrisko */ 2613665484d8SDoug Ambrisko static void 2614665484d8SDoug Ambrisko mrsas_ocr_thread(void *arg) 2615665484d8SDoug Ambrisko { 2616665484d8SDoug Ambrisko struct mrsas_softc *sc; 2617665484d8SDoug Ambrisko u_int32_t fw_status, fw_state; 2618665484d8SDoug Ambrisko 2619665484d8SDoug Ambrisko sc = (struct mrsas_softc *)arg; 2620665484d8SDoug Ambrisko 2621665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_TRACE, "%s\n", __func__); 2622665484d8SDoug Ambrisko 2623665484d8SDoug Ambrisko sc->ocr_thread_active = 1; 2624665484d8SDoug Ambrisko mtx_lock(&sc->sim_lock); 2625665484d8SDoug Ambrisko for (;;) { 2626665484d8SDoug Ambrisko /* Sleep for 1 second and check the queue status */ 2627665484d8SDoug Ambrisko msleep(&sc->ocr_chan, &sc->sim_lock, PRIBIO, 2628665484d8SDoug Ambrisko "mrsas_ocr", sc->mrsas_fw_fault_check_delay * hz); 2629665484d8SDoug Ambrisko if (sc->remove_in_progress) { 2630665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2631665484d8SDoug Ambrisko "Exit due to shutdown from %s\n", __func__); 2632665484d8SDoug Ambrisko break; 2633665484d8SDoug Ambrisko } 2634665484d8SDoug Ambrisko fw_status = mrsas_read_reg(sc, 2635665484d8SDoug Ambrisko offsetof(mrsas_reg_set, outbound_scratch_pad)); 2636665484d8SDoug Ambrisko fw_state = fw_status & MFI_STATE_MASK; 2637665484d8SDoug Ambrisko if (fw_state == MFI_STATE_FAULT || sc->do_timedout_reset) { 2638665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "OCR started due to %s!\n", 2639665484d8SDoug Ambrisko sc->do_timedout_reset ? "IO Timeout" : 2640665484d8SDoug Ambrisko "FW fault detected"); 2641665484d8SDoug Ambrisko mtx_lock_spin(&sc->ioctl_lock); 2642665484d8SDoug Ambrisko sc->reset_in_progress = 1; 2643665484d8SDoug Ambrisko sc->reset_count++; 2644665484d8SDoug Ambrisko mtx_unlock_spin(&sc->ioctl_lock); 2645665484d8SDoug Ambrisko mrsas_xpt_freeze(sc); 2646665484d8SDoug Ambrisko mrsas_reset_ctrl(sc); 2647665484d8SDoug Ambrisko mrsas_xpt_release(sc); 2648665484d8SDoug Ambrisko sc->reset_in_progress = 0; 2649665484d8SDoug Ambrisko sc->do_timedout_reset = 0; 2650665484d8SDoug Ambrisko } 2651665484d8SDoug Ambrisko } 2652665484d8SDoug Ambrisko mtx_unlock(&sc->sim_lock); 2653665484d8SDoug Ambrisko sc->ocr_thread_active = 0; 2654665484d8SDoug Ambrisko mrsas_kproc_exit(0); 2655665484d8SDoug Ambrisko } 2656665484d8SDoug Ambrisko 26578e727371SKashyap D Desai /* 26588e727371SKashyap D Desai * mrsas_reset_reply_desc: Reset Reply descriptor as part of OCR. 2659665484d8SDoug Ambrisko * input: Adapter Context. 2660665484d8SDoug Ambrisko * 26618e727371SKashyap D Desai * This function will clear reply descriptor so that post OCR driver and FW will 26628e727371SKashyap D Desai * lost old history. 2663665484d8SDoug Ambrisko */ 26648e727371SKashyap D Desai void 26658e727371SKashyap D Desai mrsas_reset_reply_desc(struct mrsas_softc *sc) 2666665484d8SDoug Ambrisko { 2667d18d1b47SKashyap D Desai int i, count; 2668665484d8SDoug Ambrisko pMpi2ReplyDescriptorsUnion_t reply_desc; 2669665484d8SDoug Ambrisko 2670d18d1b47SKashyap D Desai count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 2671d18d1b47SKashyap D Desai for (i = 0; i < count; i++) 2672d18d1b47SKashyap D Desai sc->last_reply_idx[i] = 0; 2673d18d1b47SKashyap D Desai 2674665484d8SDoug Ambrisko reply_desc = sc->reply_desc_mem; 2675665484d8SDoug Ambrisko for (i = 0; i < sc->reply_q_depth; i++, reply_desc++) { 2676665484d8SDoug Ambrisko reply_desc->Words = MRSAS_ULONG_MAX; 2677665484d8SDoug Ambrisko } 2678665484d8SDoug Ambrisko } 2679665484d8SDoug Ambrisko 26808e727371SKashyap D Desai /* 26818e727371SKashyap D Desai * mrsas_reset_ctrl: Core function to OCR/Kill adapter. 2682665484d8SDoug Ambrisko * input: Adapter Context. 2683665484d8SDoug Ambrisko * 26848e727371SKashyap D Desai * This function will run from thread context so that it can sleep. 1. Do not 26858e727371SKashyap D Desai * handle OCR if FW is in HW critical error. 2. Wait for outstanding command 26868e727371SKashyap D Desai * to complete for 180 seconds. 3. If #2 does not find any outstanding 26878e727371SKashyap D Desai * command Controller is in working state, so skip OCR. Otherwise, do 26888e727371SKashyap D Desai * OCR/kill Adapter based on flag disableOnlineCtrlReset. 4. Start of the 26898e727371SKashyap D Desai * OCR, return all SCSI command back to CAM layer which has ccb_ptr. 5. Post 26908e727371SKashyap D Desai * OCR, Re-fire Managment command and move Controller to Operation state. 2691665484d8SDoug Ambrisko */ 26928e727371SKashyap D Desai int 26938e727371SKashyap D Desai mrsas_reset_ctrl(struct mrsas_softc *sc) 2694665484d8SDoug Ambrisko { 2695665484d8SDoug Ambrisko int retval = SUCCESS, i, j, retry = 0; 2696665484d8SDoug Ambrisko u_int32_t host_diag, abs_state, status_reg, reset_adapter; 2697665484d8SDoug Ambrisko union ccb *ccb; 2698665484d8SDoug Ambrisko struct mrsas_mfi_cmd *mfi_cmd; 2699665484d8SDoug Ambrisko struct mrsas_mpt_cmd *mpt_cmd; 2700665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION *req_desc; 2701665484d8SDoug Ambrisko 2702665484d8SDoug Ambrisko if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) { 2703665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, 2704665484d8SDoug Ambrisko "mrsas: Hardware critical error, returning FAIL.\n"); 2705665484d8SDoug Ambrisko return FAIL; 2706665484d8SDoug Ambrisko } 2707f5fb2237SKashyap D Desai mrsas_set_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags); 2708665484d8SDoug Ambrisko sc->adprecovery = MRSAS_ADPRESET_SM_INFAULT; 2709665484d8SDoug Ambrisko mrsas_disable_intr(sc); 2710665484d8SDoug Ambrisko DELAY(1000 * 1000); 2711665484d8SDoug Ambrisko 2712665484d8SDoug Ambrisko /* First try waiting for commands to complete */ 2713665484d8SDoug Ambrisko if (mrsas_wait_for_outstanding(sc)) { 2714665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2715665484d8SDoug Ambrisko "resetting adapter from %s.\n", 2716665484d8SDoug Ambrisko __func__); 2717665484d8SDoug Ambrisko /* Now return commands back to the CAM layer */ 2718665484d8SDoug Ambrisko for (i = 0; i < sc->max_fw_cmds; i++) { 2719665484d8SDoug Ambrisko mpt_cmd = sc->mpt_cmd_list[i]; 2720665484d8SDoug Ambrisko if (mpt_cmd->ccb_ptr) { 2721665484d8SDoug Ambrisko ccb = (union ccb *)(mpt_cmd->ccb_ptr); 2722665484d8SDoug Ambrisko ccb->ccb_h.status = CAM_SCSI_BUS_RESET; 2723665484d8SDoug Ambrisko mrsas_cmd_done(sc, mpt_cmd); 2724f5fb2237SKashyap D Desai mrsas_atomic_dec(&sc->fw_outstanding); 2725665484d8SDoug Ambrisko } 2726665484d8SDoug Ambrisko } 2727665484d8SDoug Ambrisko 2728665484d8SDoug Ambrisko status_reg = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2729665484d8SDoug Ambrisko outbound_scratch_pad)); 2730665484d8SDoug Ambrisko abs_state = status_reg & MFI_STATE_MASK; 2731665484d8SDoug Ambrisko reset_adapter = status_reg & MFI_RESET_ADAPTER; 2732665484d8SDoug Ambrisko if (sc->disableOnlineCtrlReset || 2733665484d8SDoug Ambrisko (abs_state == MFI_STATE_FAULT && !reset_adapter)) { 2734665484d8SDoug Ambrisko /* Reset not supported, kill adapter */ 2735665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "Reset not supported, killing adapter.\n"); 2736665484d8SDoug Ambrisko mrsas_kill_hba(sc); 2737665484d8SDoug Ambrisko retval = FAIL; 2738665484d8SDoug Ambrisko goto out; 2739665484d8SDoug Ambrisko } 2740665484d8SDoug Ambrisko /* Now try to reset the chip */ 2741665484d8SDoug Ambrisko for (i = 0; i < MRSAS_FUSION_MAX_RESET_TRIES; i++) { 2742665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 2743665484d8SDoug Ambrisko MPI2_WRSEQ_FLUSH_KEY_VALUE); 2744665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 2745665484d8SDoug Ambrisko MPI2_WRSEQ_1ST_KEY_VALUE); 2746665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 2747665484d8SDoug Ambrisko MPI2_WRSEQ_2ND_KEY_VALUE); 2748665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 2749665484d8SDoug Ambrisko MPI2_WRSEQ_3RD_KEY_VALUE); 2750665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 2751665484d8SDoug Ambrisko MPI2_WRSEQ_4TH_KEY_VALUE); 2752665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 2753665484d8SDoug Ambrisko MPI2_WRSEQ_5TH_KEY_VALUE); 2754665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 2755665484d8SDoug Ambrisko MPI2_WRSEQ_6TH_KEY_VALUE); 2756665484d8SDoug Ambrisko 2757665484d8SDoug Ambrisko /* Check that the diag write enable (DRWE) bit is on */ 2758665484d8SDoug Ambrisko host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2759665484d8SDoug Ambrisko fusion_host_diag)); 2760665484d8SDoug Ambrisko retry = 0; 2761665484d8SDoug Ambrisko while (!(host_diag & HOST_DIAG_WRITE_ENABLE)) { 2762665484d8SDoug Ambrisko DELAY(100 * 1000); 2763665484d8SDoug Ambrisko host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2764665484d8SDoug Ambrisko fusion_host_diag)); 2765665484d8SDoug Ambrisko if (retry++ == 100) { 2766665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2767665484d8SDoug Ambrisko "Host diag unlock failed!\n"); 2768665484d8SDoug Ambrisko break; 2769665484d8SDoug Ambrisko } 2770665484d8SDoug Ambrisko } 2771665484d8SDoug Ambrisko if (!(host_diag & HOST_DIAG_WRITE_ENABLE)) 2772665484d8SDoug Ambrisko continue; 2773665484d8SDoug Ambrisko 2774665484d8SDoug Ambrisko /* Send chip reset command */ 2775665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_host_diag), 2776665484d8SDoug Ambrisko host_diag | HOST_DIAG_RESET_ADAPTER); 2777665484d8SDoug Ambrisko DELAY(3000 * 1000); 2778665484d8SDoug Ambrisko 2779665484d8SDoug Ambrisko /* Make sure reset adapter bit is cleared */ 2780665484d8SDoug Ambrisko host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2781665484d8SDoug Ambrisko fusion_host_diag)); 2782665484d8SDoug Ambrisko retry = 0; 2783665484d8SDoug Ambrisko while (host_diag & HOST_DIAG_RESET_ADAPTER) { 2784665484d8SDoug Ambrisko DELAY(100 * 1000); 2785665484d8SDoug Ambrisko host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2786665484d8SDoug Ambrisko fusion_host_diag)); 2787665484d8SDoug Ambrisko if (retry++ == 1000) { 2788665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2789665484d8SDoug Ambrisko "Diag reset adapter never cleared!\n"); 2790665484d8SDoug Ambrisko break; 2791665484d8SDoug Ambrisko } 2792665484d8SDoug Ambrisko } 2793665484d8SDoug Ambrisko if (host_diag & HOST_DIAG_RESET_ADAPTER) 2794665484d8SDoug Ambrisko continue; 2795665484d8SDoug Ambrisko 2796665484d8SDoug Ambrisko abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2797665484d8SDoug Ambrisko outbound_scratch_pad)) & MFI_STATE_MASK; 2798665484d8SDoug Ambrisko retry = 0; 2799665484d8SDoug Ambrisko 2800665484d8SDoug Ambrisko while ((abs_state <= MFI_STATE_FW_INIT) && (retry++ < 1000)) { 2801665484d8SDoug Ambrisko DELAY(100 * 1000); 2802665484d8SDoug Ambrisko abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2803665484d8SDoug Ambrisko outbound_scratch_pad)) & MFI_STATE_MASK; 2804665484d8SDoug Ambrisko } 2805665484d8SDoug Ambrisko if (abs_state <= MFI_STATE_FW_INIT) { 2806665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "firmware state < MFI_STATE_FW_INIT," 2807665484d8SDoug Ambrisko " state = 0x%x\n", abs_state); 2808665484d8SDoug Ambrisko continue; 2809665484d8SDoug Ambrisko } 2810665484d8SDoug Ambrisko /* Wait for FW to become ready */ 2811665484d8SDoug Ambrisko if (mrsas_transition_to_ready(sc, 1)) { 2812665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2813665484d8SDoug Ambrisko "mrsas: Failed to transition controller to ready.\n"); 2814665484d8SDoug Ambrisko continue; 2815665484d8SDoug Ambrisko } 2816665484d8SDoug Ambrisko mrsas_reset_reply_desc(sc); 2817665484d8SDoug Ambrisko if (mrsas_ioc_init(sc)) { 2818665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "mrsas_ioc_init() failed!\n"); 2819665484d8SDoug Ambrisko continue; 2820665484d8SDoug Ambrisko } 2821665484d8SDoug Ambrisko /* Re-fire management commands */ 2822665484d8SDoug Ambrisko for (j = 0; j < sc->max_fw_cmds; j++) { 2823665484d8SDoug Ambrisko mpt_cmd = sc->mpt_cmd_list[j]; 2824665484d8SDoug Ambrisko if (mpt_cmd->sync_cmd_idx != (u_int32_t)MRSAS_ULONG_MAX) { 2825665484d8SDoug Ambrisko mfi_cmd = sc->mfi_cmd_list[mpt_cmd->sync_cmd_idx]; 2826665484d8SDoug Ambrisko if (mfi_cmd->frame->dcmd.opcode == 2827665484d8SDoug Ambrisko MR_DCMD_LD_MAP_GET_INFO) { 2828665484d8SDoug Ambrisko mrsas_release_mfi_cmd(mfi_cmd); 2829665484d8SDoug Ambrisko mrsas_release_mpt_cmd(mpt_cmd); 2830665484d8SDoug Ambrisko } else { 2831665484d8SDoug Ambrisko req_desc = mrsas_get_request_desc(sc, 2832665484d8SDoug Ambrisko mfi_cmd->cmd_id.context.smid - 1); 2833665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2834665484d8SDoug Ambrisko "Re-fire command DCMD opcode 0x%x index %d\n ", 2835665484d8SDoug Ambrisko mfi_cmd->frame->dcmd.opcode, j); 2836665484d8SDoug Ambrisko if (!req_desc) 2837665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, 2838665484d8SDoug Ambrisko "Cannot build MPT cmd.\n"); 2839665484d8SDoug Ambrisko else 2840665484d8SDoug Ambrisko mrsas_fire_cmd(sc, req_desc->addr.u.low, 2841665484d8SDoug Ambrisko req_desc->addr.u.high); 2842665484d8SDoug Ambrisko } 2843665484d8SDoug Ambrisko } 2844665484d8SDoug Ambrisko } 2845665484d8SDoug Ambrisko 2846665484d8SDoug Ambrisko /* Reset load balance info */ 2847665484d8SDoug Ambrisko memset(sc->load_balance_info, 0, 28484799d485SKashyap D Desai sizeof(LD_LOAD_BALANCE_INFO) * MAX_LOGICAL_DRIVES_EXT); 2849665484d8SDoug Ambrisko 2850af51c29fSKashyap D Desai if (mrsas_get_ctrl_info(sc)) { 2851af51c29fSKashyap D Desai mrsas_kill_hba(sc); 28522f863eb8SKashyap D Desai retval = FAIL; 28532f863eb8SKashyap D Desai goto out; 2854af51c29fSKashyap D Desai } 2855665484d8SDoug Ambrisko if (!mrsas_get_map_info(sc)) 2856665484d8SDoug Ambrisko mrsas_sync_map_info(sc); 2857665484d8SDoug Ambrisko 28582f863eb8SKashyap D Desai mrsas_clear_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags); 28592f863eb8SKashyap D Desai mrsas_enable_intr(sc); 28602f863eb8SKashyap D Desai sc->adprecovery = MRSAS_HBA_OPERATIONAL; 28612f863eb8SKashyap D Desai 2862665484d8SDoug Ambrisko /* Adapter reset completed successfully */ 2863665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Reset successful\n"); 2864665484d8SDoug Ambrisko retval = SUCCESS; 2865665484d8SDoug Ambrisko goto out; 2866665484d8SDoug Ambrisko } 2867665484d8SDoug Ambrisko /* Reset failed, kill the adapter */ 2868665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Reset failed, killing adapter.\n"); 2869665484d8SDoug Ambrisko mrsas_kill_hba(sc); 2870665484d8SDoug Ambrisko retval = FAIL; 2871665484d8SDoug Ambrisko } else { 2872f5fb2237SKashyap D Desai mrsas_clear_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags); 2873665484d8SDoug Ambrisko mrsas_enable_intr(sc); 2874665484d8SDoug Ambrisko sc->adprecovery = MRSAS_HBA_OPERATIONAL; 2875665484d8SDoug Ambrisko } 2876665484d8SDoug Ambrisko out: 2877f5fb2237SKashyap D Desai mrsas_clear_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags); 2878665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2879665484d8SDoug Ambrisko "Reset Exit with %d.\n", retval); 2880665484d8SDoug Ambrisko return retval; 2881665484d8SDoug Ambrisko } 2882665484d8SDoug Ambrisko 28838e727371SKashyap D Desai /* 28848e727371SKashyap D Desai * mrsas_kill_hba: Kill HBA when OCR is not supported 2885665484d8SDoug Ambrisko * input: Adapter Context. 2886665484d8SDoug Ambrisko * 2887665484d8SDoug Ambrisko * This function will kill HBA when OCR is not supported. 2888665484d8SDoug Ambrisko */ 28898e727371SKashyap D Desai void 28908e727371SKashyap D Desai mrsas_kill_hba(struct mrsas_softc *sc) 2891665484d8SDoug Ambrisko { 2892daeed973SKashyap D Desai sc->adprecovery = MRSAS_HW_CRITICAL_ERROR; 2893daeed973SKashyap D Desai pause("mrsas_kill_hba", 1000); 2894665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "%s\n", __func__); 2895665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell), 2896665484d8SDoug Ambrisko MFI_STOP_ADP); 2897665484d8SDoug Ambrisko /* Flush */ 2898665484d8SDoug Ambrisko mrsas_read_reg(sc, offsetof(mrsas_reg_set, doorbell)); 2899daeed973SKashyap D Desai mrsas_complete_outstanding_ioctls(sc); 2900daeed973SKashyap D Desai } 2901daeed973SKashyap D Desai 2902daeed973SKashyap D Desai /** 2903daeed973SKashyap D Desai * mrsas_complete_outstanding_ioctls Complete pending IOCTLS after kill_hba 2904daeed973SKashyap D Desai * input: Controller softc 2905daeed973SKashyap D Desai * 2906daeed973SKashyap D Desai * Returns void 2907daeed973SKashyap D Desai */ 2908*dbcc81dfSKashyap D Desai void 2909*dbcc81dfSKashyap D Desai mrsas_complete_outstanding_ioctls(struct mrsas_softc *sc) 2910*dbcc81dfSKashyap D Desai { 2911daeed973SKashyap D Desai int i; 2912daeed973SKashyap D Desai struct mrsas_mpt_cmd *cmd_mpt; 2913daeed973SKashyap D Desai struct mrsas_mfi_cmd *cmd_mfi; 2914daeed973SKashyap D Desai u_int32_t count, MSIxIndex; 2915daeed973SKashyap D Desai 2916daeed973SKashyap D Desai count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 2917daeed973SKashyap D Desai for (i = 0; i < sc->max_fw_cmds; i++) { 2918daeed973SKashyap D Desai cmd_mpt = sc->mpt_cmd_list[i]; 2919daeed973SKashyap D Desai 2920daeed973SKashyap D Desai if (cmd_mpt->sync_cmd_idx != (u_int32_t)MRSAS_ULONG_MAX) { 2921daeed973SKashyap D Desai cmd_mfi = sc->mfi_cmd_list[cmd_mpt->sync_cmd_idx]; 2922daeed973SKashyap D Desai if (cmd_mfi->sync_cmd && cmd_mfi->frame->hdr.cmd != MFI_CMD_ABORT) { 2923daeed973SKashyap D Desai for (MSIxIndex = 0; MSIxIndex < count; MSIxIndex++) 2924daeed973SKashyap D Desai mrsas_complete_mptmfi_passthru(sc, cmd_mfi, 2925daeed973SKashyap D Desai cmd_mpt->io_request->RaidContext.status); 2926daeed973SKashyap D Desai } 2927daeed973SKashyap D Desai } 2928daeed973SKashyap D Desai } 2929665484d8SDoug Ambrisko } 2930665484d8SDoug Ambrisko 29318e727371SKashyap D Desai /* 29328e727371SKashyap D Desai * mrsas_wait_for_outstanding: Wait for outstanding commands 2933665484d8SDoug Ambrisko * input: Adapter Context. 2934665484d8SDoug Ambrisko * 29358e727371SKashyap D Desai * This function will wait for 180 seconds for outstanding commands to be 29368e727371SKashyap D Desai * completed. 2937665484d8SDoug Ambrisko */ 29388e727371SKashyap D Desai int 29398e727371SKashyap D Desai mrsas_wait_for_outstanding(struct mrsas_softc *sc) 2940665484d8SDoug Ambrisko { 2941665484d8SDoug Ambrisko int i, outstanding, retval = 0; 2942d18d1b47SKashyap D Desai u_int32_t fw_state, count, MSIxIndex; 2943d18d1b47SKashyap D Desai 2944665484d8SDoug Ambrisko 2945665484d8SDoug Ambrisko for (i = 0; i < MRSAS_RESET_WAIT_TIME; i++) { 2946665484d8SDoug Ambrisko if (sc->remove_in_progress) { 2947665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2948665484d8SDoug Ambrisko "Driver remove or shutdown called.\n"); 2949665484d8SDoug Ambrisko retval = 1; 2950665484d8SDoug Ambrisko goto out; 2951665484d8SDoug Ambrisko } 2952665484d8SDoug Ambrisko /* Check if firmware is in fault state */ 2953665484d8SDoug Ambrisko fw_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2954665484d8SDoug Ambrisko outbound_scratch_pad)) & MFI_STATE_MASK; 2955665484d8SDoug Ambrisko if (fw_state == MFI_STATE_FAULT) { 2956665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2957665484d8SDoug Ambrisko "Found FW in FAULT state, will reset adapter.\n"); 2958665484d8SDoug Ambrisko retval = 1; 2959665484d8SDoug Ambrisko goto out; 2960665484d8SDoug Ambrisko } 2961f5fb2237SKashyap D Desai outstanding = mrsas_atomic_read(&sc->fw_outstanding); 2962665484d8SDoug Ambrisko if (!outstanding) 2963665484d8SDoug Ambrisko goto out; 2964665484d8SDoug Ambrisko 2965665484d8SDoug Ambrisko if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) { 2966665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "[%2d]waiting for %d " 2967665484d8SDoug Ambrisko "commands to complete\n", i, outstanding); 2968d18d1b47SKashyap D Desai count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 2969d18d1b47SKashyap D Desai for (MSIxIndex = 0; MSIxIndex < count; MSIxIndex++) 2970d18d1b47SKashyap D Desai mrsas_complete_cmd(sc, MSIxIndex); 2971665484d8SDoug Ambrisko } 2972665484d8SDoug Ambrisko DELAY(1000 * 1000); 2973665484d8SDoug Ambrisko } 2974665484d8SDoug Ambrisko 2975f5fb2237SKashyap D Desai if (mrsas_atomic_read(&sc->fw_outstanding)) { 2976665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2977665484d8SDoug Ambrisko " pending commands remain after waiting," 2978665484d8SDoug Ambrisko " will reset adapter.\n"); 2979665484d8SDoug Ambrisko retval = 1; 2980665484d8SDoug Ambrisko } 2981665484d8SDoug Ambrisko out: 2982665484d8SDoug Ambrisko return retval; 2983665484d8SDoug Ambrisko } 2984665484d8SDoug Ambrisko 29858e727371SKashyap D Desai /* 2986665484d8SDoug Ambrisko * mrsas_release_mfi_cmd: Return a cmd to free command pool 2987665484d8SDoug Ambrisko * input: Command packet for return to free cmd pool 2988665484d8SDoug Ambrisko * 2989665484d8SDoug Ambrisko * This function returns the MFI command to the command list. 2990665484d8SDoug Ambrisko */ 29918e727371SKashyap D Desai void 29928e727371SKashyap D Desai mrsas_release_mfi_cmd(struct mrsas_mfi_cmd *cmd) 2993665484d8SDoug Ambrisko { 2994665484d8SDoug Ambrisko struct mrsas_softc *sc = cmd->sc; 2995665484d8SDoug Ambrisko 2996665484d8SDoug Ambrisko mtx_lock(&sc->mfi_cmd_pool_lock); 2997665484d8SDoug Ambrisko cmd->ccb_ptr = NULL; 2998665484d8SDoug Ambrisko cmd->cmd_id.frame_count = 0; 2999665484d8SDoug Ambrisko TAILQ_INSERT_TAIL(&(sc->mrsas_mfi_cmd_list_head), cmd, next); 3000665484d8SDoug Ambrisko mtx_unlock(&sc->mfi_cmd_pool_lock); 3001665484d8SDoug Ambrisko 3002665484d8SDoug Ambrisko return; 3003665484d8SDoug Ambrisko } 3004665484d8SDoug Ambrisko 30058e727371SKashyap D Desai /* 30068e727371SKashyap D Desai * mrsas_get_controller_info: Returns FW's controller structure 3007665484d8SDoug Ambrisko * input: Adapter soft state 3008665484d8SDoug Ambrisko * Controller information structure 3009665484d8SDoug Ambrisko * 30108e727371SKashyap D Desai * Issues an internal command (DCMD) to get the FW's controller structure. This 30118e727371SKashyap D Desai * information is mainly used to find out the maximum IO transfer per command 30128e727371SKashyap D Desai * supported by the FW. 3013665484d8SDoug Ambrisko */ 30148e727371SKashyap D Desai static int 3015af51c29fSKashyap D Desai mrsas_get_ctrl_info(struct mrsas_softc *sc) 3016665484d8SDoug Ambrisko { 3017665484d8SDoug Ambrisko int retcode = 0; 3018665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3019665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 3020665484d8SDoug Ambrisko 3021665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3022665484d8SDoug Ambrisko 3023665484d8SDoug Ambrisko if (!cmd) { 3024665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Failed to get a free cmd\n"); 3025665484d8SDoug Ambrisko return -ENOMEM; 3026665484d8SDoug Ambrisko } 3027665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 3028665484d8SDoug Ambrisko 3029665484d8SDoug Ambrisko if (mrsas_alloc_ctlr_info_cmd(sc) != SUCCESS) { 3030665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate get ctlr info cmd\n"); 3031665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3032665484d8SDoug Ambrisko return -ENOMEM; 3033665484d8SDoug Ambrisko } 3034665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 3035665484d8SDoug Ambrisko 3036665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 3037665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 3038665484d8SDoug Ambrisko dcmd->sge_count = 1; 3039665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 3040665484d8SDoug Ambrisko dcmd->timeout = 0; 3041665484d8SDoug Ambrisko dcmd->pad_0 = 0; 3042665484d8SDoug Ambrisko dcmd->data_xfer_len = sizeof(struct mrsas_ctrl_info); 3043665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_CTRL_GET_INFO; 3044665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = sc->ctlr_info_phys_addr; 3045665484d8SDoug Ambrisko dcmd->sgl.sge32[0].length = sizeof(struct mrsas_ctrl_info); 3046665484d8SDoug Ambrisko 3047665484d8SDoug Ambrisko if (!mrsas_issue_polled(sc, cmd)) 3048af51c29fSKashyap D Desai memcpy(sc->ctrl_info, sc->ctlr_info_mem, sizeof(struct mrsas_ctrl_info)); 3049665484d8SDoug Ambrisko else 3050665484d8SDoug Ambrisko retcode = 1; 3051665484d8SDoug Ambrisko 3052af51c29fSKashyap D Desai mrsas_update_ext_vd_details(sc); 3053af51c29fSKashyap D Desai 3054665484d8SDoug Ambrisko mrsas_free_ctlr_info_cmd(sc); 3055665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3056665484d8SDoug Ambrisko return (retcode); 3057665484d8SDoug Ambrisko } 3058665484d8SDoug Ambrisko 30598e727371SKashyap D Desai /* 3060af51c29fSKashyap D Desai * mrsas_update_ext_vd_details : Update details w.r.t Extended VD 3061af51c29fSKashyap D Desai * input: 3062af51c29fSKashyap D Desai * sc - Controller's softc 3063af51c29fSKashyap D Desai */ 3064*dbcc81dfSKashyap D Desai static void 3065*dbcc81dfSKashyap D Desai mrsas_update_ext_vd_details(struct mrsas_softc *sc) 3066af51c29fSKashyap D Desai { 3067af51c29fSKashyap D Desai sc->max256vdSupport = 3068af51c29fSKashyap D Desai sc->ctrl_info->adapterOperations3.supportMaxExtLDs; 3069af51c29fSKashyap D Desai /* Below is additional check to address future FW enhancement */ 3070af51c29fSKashyap D Desai if (sc->ctrl_info->max_lds > 64) 3071af51c29fSKashyap D Desai sc->max256vdSupport = 1; 3072af51c29fSKashyap D Desai 3073af51c29fSKashyap D Desai sc->drv_supported_vd_count = MRSAS_MAX_LD_CHANNELS 3074af51c29fSKashyap D Desai * MRSAS_MAX_DEV_PER_CHANNEL; 3075af51c29fSKashyap D Desai sc->drv_supported_pd_count = MRSAS_MAX_PD_CHANNELS 3076af51c29fSKashyap D Desai * MRSAS_MAX_DEV_PER_CHANNEL; 3077af51c29fSKashyap D Desai if (sc->max256vdSupport) { 3078af51c29fSKashyap D Desai sc->fw_supported_vd_count = MAX_LOGICAL_DRIVES_EXT; 3079af51c29fSKashyap D Desai sc->fw_supported_pd_count = MAX_PHYSICAL_DEVICES; 3080af51c29fSKashyap D Desai } else { 3081af51c29fSKashyap D Desai sc->fw_supported_vd_count = MAX_LOGICAL_DRIVES; 3082af51c29fSKashyap D Desai sc->fw_supported_pd_count = MAX_PHYSICAL_DEVICES; 3083af51c29fSKashyap D Desai } 3084af51c29fSKashyap D Desai 3085af51c29fSKashyap D Desai sc->old_map_sz = sizeof(MR_FW_RAID_MAP) + 3086af51c29fSKashyap D Desai (sizeof(MR_LD_SPAN_MAP) * 3087af51c29fSKashyap D Desai (sc->fw_supported_vd_count - 1)); 3088af51c29fSKashyap D Desai sc->new_map_sz = sizeof(MR_FW_RAID_MAP_EXT); 3089af51c29fSKashyap D Desai sc->drv_map_sz = sizeof(MR_DRV_RAID_MAP) + 3090af51c29fSKashyap D Desai (sizeof(MR_LD_SPAN_MAP) * 3091af51c29fSKashyap D Desai (sc->drv_supported_vd_count - 1)); 3092af51c29fSKashyap D Desai 3093af51c29fSKashyap D Desai sc->max_map_sz = max(sc->old_map_sz, sc->new_map_sz); 3094af51c29fSKashyap D Desai 3095af51c29fSKashyap D Desai if (sc->max256vdSupport) 3096af51c29fSKashyap D Desai sc->current_map_sz = sc->new_map_sz; 3097af51c29fSKashyap D Desai else 3098af51c29fSKashyap D Desai sc->current_map_sz = sc->old_map_sz; 3099af51c29fSKashyap D Desai } 3100af51c29fSKashyap D Desai 3101af51c29fSKashyap D Desai /* 3102665484d8SDoug Ambrisko * mrsas_alloc_ctlr_info_cmd: Allocates memory for controller info command 3103665484d8SDoug Ambrisko * input: Adapter soft state 3104665484d8SDoug Ambrisko * 3105665484d8SDoug Ambrisko * Allocates DMAable memory for the controller info internal command. 3106665484d8SDoug Ambrisko */ 31078e727371SKashyap D Desai int 31088e727371SKashyap D Desai mrsas_alloc_ctlr_info_cmd(struct mrsas_softc *sc) 3109665484d8SDoug Ambrisko { 3110665484d8SDoug Ambrisko int ctlr_info_size; 3111665484d8SDoug Ambrisko 3112665484d8SDoug Ambrisko /* Allocate get controller info command */ 3113665484d8SDoug Ambrisko ctlr_info_size = sizeof(struct mrsas_ctrl_info); 31148e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 31158e727371SKashyap D Desai 1, 0, 31168e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 31178e727371SKashyap D Desai BUS_SPACE_MAXADDR, 31188e727371SKashyap D Desai NULL, NULL, 31198e727371SKashyap D Desai ctlr_info_size, 31208e727371SKashyap D Desai 1, 31218e727371SKashyap D Desai ctlr_info_size, 31228e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 31238e727371SKashyap D Desai NULL, NULL, 3124665484d8SDoug Ambrisko &sc->ctlr_info_tag)) { 3125665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate ctlr info tag\n"); 3126665484d8SDoug Ambrisko return (ENOMEM); 3127665484d8SDoug Ambrisko } 3128665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->ctlr_info_tag, (void **)&sc->ctlr_info_mem, 3129665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->ctlr_info_dmamap)) { 3130665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate ctlr info cmd mem\n"); 3131665484d8SDoug Ambrisko return (ENOMEM); 3132665484d8SDoug Ambrisko } 3133665484d8SDoug Ambrisko if (bus_dmamap_load(sc->ctlr_info_tag, sc->ctlr_info_dmamap, 3134665484d8SDoug Ambrisko sc->ctlr_info_mem, ctlr_info_size, mrsas_addr_cb, 3135665484d8SDoug Ambrisko &sc->ctlr_info_phys_addr, BUS_DMA_NOWAIT)) { 3136665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load ctlr info cmd mem\n"); 3137665484d8SDoug Ambrisko return (ENOMEM); 3138665484d8SDoug Ambrisko } 3139665484d8SDoug Ambrisko memset(sc->ctlr_info_mem, 0, ctlr_info_size); 3140665484d8SDoug Ambrisko return (0); 3141665484d8SDoug Ambrisko } 3142665484d8SDoug Ambrisko 31438e727371SKashyap D Desai /* 3144665484d8SDoug Ambrisko * mrsas_free_ctlr_info_cmd: Free memory for controller info command 3145665484d8SDoug Ambrisko * input: Adapter soft state 3146665484d8SDoug Ambrisko * 3147665484d8SDoug Ambrisko * Deallocates memory of the get controller info cmd. 3148665484d8SDoug Ambrisko */ 31498e727371SKashyap D Desai void 31508e727371SKashyap D Desai mrsas_free_ctlr_info_cmd(struct mrsas_softc *sc) 3151665484d8SDoug Ambrisko { 3152665484d8SDoug Ambrisko if (sc->ctlr_info_phys_addr) 3153665484d8SDoug Ambrisko bus_dmamap_unload(sc->ctlr_info_tag, sc->ctlr_info_dmamap); 3154665484d8SDoug Ambrisko if (sc->ctlr_info_mem != NULL) 3155665484d8SDoug Ambrisko bus_dmamem_free(sc->ctlr_info_tag, sc->ctlr_info_mem, sc->ctlr_info_dmamap); 3156665484d8SDoug Ambrisko if (sc->ctlr_info_tag != NULL) 3157665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->ctlr_info_tag); 3158665484d8SDoug Ambrisko } 3159665484d8SDoug Ambrisko 31608e727371SKashyap D Desai /* 3161665484d8SDoug Ambrisko * mrsas_issue_polled: Issues a polling command 3162665484d8SDoug Ambrisko * inputs: Adapter soft state 3163665484d8SDoug Ambrisko * Command packet to be issued 3164665484d8SDoug Ambrisko * 31658e727371SKashyap D Desai * This function is for posting of internal commands to Firmware. MFI requires 31668e727371SKashyap D Desai * the cmd_status to be set to 0xFF before posting. The maximun wait time of 31678e727371SKashyap D Desai * the poll response timer is 180 seconds. 3168665484d8SDoug Ambrisko */ 31698e727371SKashyap D Desai int 31708e727371SKashyap D Desai mrsas_issue_polled(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 3171665484d8SDoug Ambrisko { 3172665484d8SDoug Ambrisko struct mrsas_header *frame_hdr = &cmd->frame->hdr; 3173665484d8SDoug Ambrisko u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME; 3174665484d8SDoug Ambrisko int i, retcode = 0; 3175665484d8SDoug Ambrisko 3176665484d8SDoug Ambrisko frame_hdr->cmd_status = 0xFF; 3177665484d8SDoug Ambrisko frame_hdr->flags |= MFI_FRAME_DONT_POST_IN_REPLY_QUEUE; 3178665484d8SDoug Ambrisko 3179665484d8SDoug Ambrisko /* Issue the frame using inbound queue port */ 3180665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)) { 3181665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot issue DCMD internal command.\n"); 3182665484d8SDoug Ambrisko return (1); 3183665484d8SDoug Ambrisko } 3184665484d8SDoug Ambrisko /* 3185665484d8SDoug Ambrisko * Poll response timer to wait for Firmware response. While this 3186665484d8SDoug Ambrisko * timer with the DELAY call could block CPU, the time interval for 3187665484d8SDoug Ambrisko * this is only 1 millisecond. 3188665484d8SDoug Ambrisko */ 3189665484d8SDoug Ambrisko if (frame_hdr->cmd_status == 0xFF) { 3190665484d8SDoug Ambrisko for (i = 0; i < (max_wait * 1000); i++) { 3191665484d8SDoug Ambrisko if (frame_hdr->cmd_status == 0xFF) 3192665484d8SDoug Ambrisko DELAY(1000); 3193665484d8SDoug Ambrisko else 3194665484d8SDoug Ambrisko break; 3195665484d8SDoug Ambrisko } 3196665484d8SDoug Ambrisko } 31978e727371SKashyap D Desai if (frame_hdr->cmd_status != 0) { 3198665484d8SDoug Ambrisko if (frame_hdr->cmd_status == 0xFF) 3199665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "DCMD timed out after %d seconds.\n", max_wait); 3200665484d8SDoug Ambrisko else 3201665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "DCMD failed, status = 0x%x\n", frame_hdr->cmd_status); 3202665484d8SDoug Ambrisko retcode = 1; 3203665484d8SDoug Ambrisko } 3204665484d8SDoug Ambrisko return (retcode); 3205665484d8SDoug Ambrisko } 3206665484d8SDoug Ambrisko 32078e727371SKashyap D Desai /* 32088e727371SKashyap D Desai * mrsas_issue_dcmd: Issues a MFI Pass thru cmd 32098e727371SKashyap D Desai * input: Adapter soft state mfi cmd pointer 3210665484d8SDoug Ambrisko * 3211665484d8SDoug Ambrisko * This function is called by mrsas_issued_blocked_cmd() and 32128e727371SKashyap D Desai * mrsas_issued_polled(), to build the MPT command and then fire the command 32138e727371SKashyap D Desai * to Firmware. 3214665484d8SDoug Ambrisko */ 3215665484d8SDoug Ambrisko int 3216665484d8SDoug Ambrisko mrsas_issue_dcmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 3217665484d8SDoug Ambrisko { 3218665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION *req_desc; 3219665484d8SDoug Ambrisko 3220665484d8SDoug Ambrisko req_desc = mrsas_build_mpt_cmd(sc, cmd); 3221665484d8SDoug Ambrisko if (!req_desc) { 3222665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot build MPT cmd.\n"); 3223665484d8SDoug Ambrisko return (1); 3224665484d8SDoug Ambrisko } 3225665484d8SDoug Ambrisko mrsas_fire_cmd(sc, req_desc->addr.u.low, req_desc->addr.u.high); 3226665484d8SDoug Ambrisko 3227665484d8SDoug Ambrisko return (0); 3228665484d8SDoug Ambrisko } 3229665484d8SDoug Ambrisko 32308e727371SKashyap D Desai /* 32318e727371SKashyap D Desai * mrsas_build_mpt_cmd: Calls helper function to build Passthru cmd 32328e727371SKashyap D Desai * input: Adapter soft state mfi cmd to build 3233665484d8SDoug Ambrisko * 32348e727371SKashyap D Desai * This function is called by mrsas_issue_cmd() to build the MPT-MFI passthru 32358e727371SKashyap D Desai * command and prepares the MPT command to send to Firmware. 3236665484d8SDoug Ambrisko */ 3237665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION * 3238665484d8SDoug Ambrisko mrsas_build_mpt_cmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 3239665484d8SDoug Ambrisko { 3240665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION *req_desc; 3241665484d8SDoug Ambrisko u_int16_t index; 3242665484d8SDoug Ambrisko 3243665484d8SDoug Ambrisko if (mrsas_build_mptmfi_passthru(sc, cmd)) { 3244665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot build MPT-MFI passthru cmd.\n"); 3245665484d8SDoug Ambrisko return NULL; 3246665484d8SDoug Ambrisko } 3247665484d8SDoug Ambrisko index = cmd->cmd_id.context.smid; 3248665484d8SDoug Ambrisko 3249665484d8SDoug Ambrisko req_desc = mrsas_get_request_desc(sc, index - 1); 3250665484d8SDoug Ambrisko if (!req_desc) 3251665484d8SDoug Ambrisko return NULL; 3252665484d8SDoug Ambrisko 3253665484d8SDoug Ambrisko req_desc->addr.Words = 0; 3254665484d8SDoug Ambrisko req_desc->SCSIIO.RequestFlags = (MPI2_REQ_DESCRIPT_FLAGS_SCSI_IO << MRSAS_REQ_DESCRIPT_FLAGS_TYPE_SHIFT); 3255665484d8SDoug Ambrisko 3256665484d8SDoug Ambrisko req_desc->SCSIIO.SMID = index; 3257665484d8SDoug Ambrisko 3258665484d8SDoug Ambrisko return (req_desc); 3259665484d8SDoug Ambrisko } 3260665484d8SDoug Ambrisko 32618e727371SKashyap D Desai /* 32628e727371SKashyap D Desai * mrsas_build_mptmfi_passthru: Builds a MPT MFI Passthru command 32638e727371SKashyap D Desai * input: Adapter soft state mfi cmd pointer 3264665484d8SDoug Ambrisko * 32658e727371SKashyap D Desai * The MPT command and the io_request are setup as a passthru command. The SGE 32668e727371SKashyap D Desai * chain address is set to frame_phys_addr of the MFI command. 3267665484d8SDoug Ambrisko */ 3268665484d8SDoug Ambrisko u_int8_t 3269665484d8SDoug Ambrisko mrsas_build_mptmfi_passthru(struct mrsas_softc *sc, struct mrsas_mfi_cmd *mfi_cmd) 3270665484d8SDoug Ambrisko { 3271665484d8SDoug Ambrisko MPI25_IEEE_SGE_CHAIN64 *mpi25_ieee_chain; 3272665484d8SDoug Ambrisko PTR_MRSAS_RAID_SCSI_IO_REQUEST io_req; 3273665484d8SDoug Ambrisko struct mrsas_mpt_cmd *mpt_cmd; 3274665484d8SDoug Ambrisko struct mrsas_header *frame_hdr = &mfi_cmd->frame->hdr; 3275665484d8SDoug Ambrisko 3276665484d8SDoug Ambrisko mpt_cmd = mrsas_get_mpt_cmd(sc); 3277665484d8SDoug Ambrisko if (!mpt_cmd) 3278665484d8SDoug Ambrisko return (1); 3279665484d8SDoug Ambrisko 3280665484d8SDoug Ambrisko /* Save the smid. To be used for returning the cmd */ 3281665484d8SDoug Ambrisko mfi_cmd->cmd_id.context.smid = mpt_cmd->index; 3282665484d8SDoug Ambrisko 3283665484d8SDoug Ambrisko mpt_cmd->sync_cmd_idx = mfi_cmd->index; 3284665484d8SDoug Ambrisko 3285665484d8SDoug Ambrisko /* 32868e727371SKashyap D Desai * For cmds where the flag is set, store the flag and check on 32878e727371SKashyap D Desai * completion. For cmds with this flag, don't call 3288665484d8SDoug Ambrisko * mrsas_complete_cmd. 3289665484d8SDoug Ambrisko */ 3290665484d8SDoug Ambrisko 3291665484d8SDoug Ambrisko if (frame_hdr->flags & MFI_FRAME_DONT_POST_IN_REPLY_QUEUE) 3292665484d8SDoug Ambrisko mpt_cmd->flags = MFI_FRAME_DONT_POST_IN_REPLY_QUEUE; 3293665484d8SDoug Ambrisko 3294665484d8SDoug Ambrisko io_req = mpt_cmd->io_request; 3295665484d8SDoug Ambrisko 3296665484d8SDoug Ambrisko if ((sc->device_id == MRSAS_INVADER) || (sc->device_id == MRSAS_FURY)) { 3297665484d8SDoug Ambrisko pMpi25IeeeSgeChain64_t sgl_ptr_end = (pMpi25IeeeSgeChain64_t)&io_req->SGL; 32988e727371SKashyap D Desai 3299665484d8SDoug Ambrisko sgl_ptr_end += sc->max_sge_in_main_msg - 1; 3300665484d8SDoug Ambrisko sgl_ptr_end->Flags = 0; 3301665484d8SDoug Ambrisko } 3302665484d8SDoug Ambrisko mpi25_ieee_chain = (MPI25_IEEE_SGE_CHAIN64 *) & io_req->SGL.IeeeChain; 3303665484d8SDoug Ambrisko 3304665484d8SDoug Ambrisko io_req->Function = MRSAS_MPI2_FUNCTION_PASSTHRU_IO_REQUEST; 3305665484d8SDoug Ambrisko io_req->SGLOffset0 = offsetof(MRSAS_RAID_SCSI_IO_REQUEST, SGL) / 4; 3306665484d8SDoug Ambrisko io_req->ChainOffset = sc->chain_offset_mfi_pthru; 3307665484d8SDoug Ambrisko 3308665484d8SDoug Ambrisko mpi25_ieee_chain->Address = mfi_cmd->frame_phys_addr; 3309665484d8SDoug Ambrisko 3310665484d8SDoug Ambrisko mpi25_ieee_chain->Flags = IEEE_SGE_FLAGS_CHAIN_ELEMENT | 3311665484d8SDoug Ambrisko MPI2_IEEE_SGE_FLAGS_IOCPLBNTA_ADDR; 3312665484d8SDoug Ambrisko 3313665484d8SDoug Ambrisko mpi25_ieee_chain->Length = MRSAS_MAX_SZ_CHAIN_FRAME; 3314665484d8SDoug Ambrisko 3315665484d8SDoug Ambrisko return (0); 3316665484d8SDoug Ambrisko } 3317665484d8SDoug Ambrisko 33188e727371SKashyap D Desai /* 33198e727371SKashyap D Desai * mrsas_issue_blocked_cmd: Synchronous wrapper around regular FW cmds 33208e727371SKashyap D Desai * input: Adapter soft state Command to be issued 3321665484d8SDoug Ambrisko * 33228e727371SKashyap D Desai * This function waits on an event for the command to be returned from the ISR. 33238e727371SKashyap D Desai * Max wait time is MRSAS_INTERNAL_CMD_WAIT_TIME secs. Used for issuing 33248e727371SKashyap D Desai * internal and ioctl commands. 3325665484d8SDoug Ambrisko */ 33268e727371SKashyap D Desai int 33278e727371SKashyap D Desai mrsas_issue_blocked_cmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 3328665484d8SDoug Ambrisko { 3329665484d8SDoug Ambrisko u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME; 3330665484d8SDoug Ambrisko unsigned long total_time = 0; 3331665484d8SDoug Ambrisko int retcode = 0; 3332665484d8SDoug Ambrisko 3333665484d8SDoug Ambrisko /* Initialize cmd_status */ 3334665484d8SDoug Ambrisko cmd->cmd_status = ECONNREFUSED; 3335665484d8SDoug Ambrisko 3336665484d8SDoug Ambrisko /* Build MPT-MFI command for issue to FW */ 3337665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)) { 3338665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot issue DCMD internal command.\n"); 3339665484d8SDoug Ambrisko return (1); 3340665484d8SDoug Ambrisko } 3341665484d8SDoug Ambrisko sc->chan = (void *)&cmd; 3342665484d8SDoug Ambrisko 3343665484d8SDoug Ambrisko while (1) { 3344665484d8SDoug Ambrisko if (cmd->cmd_status == ECONNREFUSED) { 3345665484d8SDoug Ambrisko tsleep((void *)&sc->chan, 0, "mrsas_sleep", hz); 33468e727371SKashyap D Desai } else 3347665484d8SDoug Ambrisko break; 3348665484d8SDoug Ambrisko total_time++; 3349665484d8SDoug Ambrisko if (total_time >= max_wait) { 33508e727371SKashyap D Desai device_printf(sc->mrsas_dev, 33518e727371SKashyap D Desai "Internal command timed out after %d seconds.\n", max_wait); 3352665484d8SDoug Ambrisko retcode = 1; 3353665484d8SDoug Ambrisko break; 3354665484d8SDoug Ambrisko } 3355665484d8SDoug Ambrisko } 3356665484d8SDoug Ambrisko return (retcode); 3357665484d8SDoug Ambrisko } 3358665484d8SDoug Ambrisko 33598e727371SKashyap D Desai /* 33608e727371SKashyap D Desai * mrsas_complete_mptmfi_passthru: Completes a command 33618e727371SKashyap D Desai * input: @sc: Adapter soft state 33628e727371SKashyap D Desai * @cmd: Command to be completed 33638e727371SKashyap D Desai * @status: cmd completion status 3364665484d8SDoug Ambrisko * 33658e727371SKashyap D Desai * This function is called from mrsas_complete_cmd() after an interrupt is 33668e727371SKashyap D Desai * received from Firmware, and io_request->Function is 3367665484d8SDoug Ambrisko * MRSAS_MPI2_FUNCTION_PASSTHRU_IO_REQUEST. 3368665484d8SDoug Ambrisko */ 3369665484d8SDoug Ambrisko void 3370665484d8SDoug Ambrisko mrsas_complete_mptmfi_passthru(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd, 3371665484d8SDoug Ambrisko u_int8_t status) 3372665484d8SDoug Ambrisko { 3373665484d8SDoug Ambrisko struct mrsas_header *hdr = &cmd->frame->hdr; 3374665484d8SDoug Ambrisko u_int8_t cmd_status = cmd->frame->hdr.cmd_status; 3375665484d8SDoug Ambrisko 3376665484d8SDoug Ambrisko /* Reset the retry counter for future re-tries */ 3377665484d8SDoug Ambrisko cmd->retry_for_fw_reset = 0; 3378665484d8SDoug Ambrisko 3379665484d8SDoug Ambrisko if (cmd->ccb_ptr) 3380665484d8SDoug Ambrisko cmd->ccb_ptr = NULL; 3381665484d8SDoug Ambrisko 3382665484d8SDoug Ambrisko switch (hdr->cmd) { 3383665484d8SDoug Ambrisko case MFI_CMD_INVALID: 3384665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "MFI_CMD_INVALID command.\n"); 3385665484d8SDoug Ambrisko break; 3386665484d8SDoug Ambrisko case MFI_CMD_PD_SCSI_IO: 3387665484d8SDoug Ambrisko case MFI_CMD_LD_SCSI_IO: 3388665484d8SDoug Ambrisko /* 3389665484d8SDoug Ambrisko * MFI_CMD_PD_SCSI_IO and MFI_CMD_LD_SCSI_IO could have been 3390665484d8SDoug Ambrisko * issued either through an IO path or an IOCTL path. If it 3391665484d8SDoug Ambrisko * was via IOCTL, we will send it to internal completion. 3392665484d8SDoug Ambrisko */ 3393665484d8SDoug Ambrisko if (cmd->sync_cmd) { 3394665484d8SDoug Ambrisko cmd->sync_cmd = 0; 3395665484d8SDoug Ambrisko mrsas_wakeup(sc, cmd); 3396665484d8SDoug Ambrisko break; 3397665484d8SDoug Ambrisko } 3398665484d8SDoug Ambrisko case MFI_CMD_SMP: 3399665484d8SDoug Ambrisko case MFI_CMD_STP: 3400665484d8SDoug Ambrisko case MFI_CMD_DCMD: 3401665484d8SDoug Ambrisko /* Check for LD map update */ 3402665484d8SDoug Ambrisko if ((cmd->frame->dcmd.opcode == MR_DCMD_LD_MAP_GET_INFO) && 3403665484d8SDoug Ambrisko (cmd->frame->dcmd.mbox.b[1] == 1)) { 3404665484d8SDoug Ambrisko sc->fast_path_io = 0; 3405665484d8SDoug Ambrisko mtx_lock(&sc->raidmap_lock); 3406665484d8SDoug Ambrisko if (cmd_status != 0) { 3407665484d8SDoug Ambrisko if (cmd_status != MFI_STAT_NOT_FOUND) 3408665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "map sync failed, status=%x\n", cmd_status); 3409665484d8SDoug Ambrisko else { 3410665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3411665484d8SDoug Ambrisko mtx_unlock(&sc->raidmap_lock); 3412665484d8SDoug Ambrisko break; 3413665484d8SDoug Ambrisko } 34148e727371SKashyap D Desai } else 3415665484d8SDoug Ambrisko sc->map_id++; 3416665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3417665484d8SDoug Ambrisko if (MR_ValidateMapInfo(sc)) 3418665484d8SDoug Ambrisko sc->fast_path_io = 0; 3419665484d8SDoug Ambrisko else 3420665484d8SDoug Ambrisko sc->fast_path_io = 1; 3421665484d8SDoug Ambrisko mrsas_sync_map_info(sc); 3422665484d8SDoug Ambrisko mtx_unlock(&sc->raidmap_lock); 3423665484d8SDoug Ambrisko break; 3424665484d8SDoug Ambrisko } 3425665484d8SDoug Ambrisko if (cmd->frame->dcmd.opcode == MR_DCMD_CTRL_EVENT_GET_INFO || 3426665484d8SDoug Ambrisko cmd->frame->dcmd.opcode == MR_DCMD_CTRL_EVENT_GET) { 3427da011113SKashyap D Desai sc->mrsas_aen_triggered = 0; 3428665484d8SDoug Ambrisko } 3429665484d8SDoug Ambrisko /* See if got an event notification */ 3430665484d8SDoug Ambrisko if (cmd->frame->dcmd.opcode == MR_DCMD_CTRL_EVENT_WAIT) 3431665484d8SDoug Ambrisko mrsas_complete_aen(sc, cmd); 3432665484d8SDoug Ambrisko else 3433665484d8SDoug Ambrisko mrsas_wakeup(sc, cmd); 3434665484d8SDoug Ambrisko break; 3435665484d8SDoug Ambrisko case MFI_CMD_ABORT: 3436665484d8SDoug Ambrisko /* Command issued to abort another cmd return */ 3437665484d8SDoug Ambrisko mrsas_complete_abort(sc, cmd); 3438665484d8SDoug Ambrisko break; 3439665484d8SDoug Ambrisko default: 3440665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Unknown command completed! [0x%X]\n", hdr->cmd); 3441665484d8SDoug Ambrisko break; 3442665484d8SDoug Ambrisko } 3443665484d8SDoug Ambrisko } 3444665484d8SDoug Ambrisko 34458e727371SKashyap D Desai /* 34468e727371SKashyap D Desai * mrsas_wakeup: Completes an internal command 3447665484d8SDoug Ambrisko * input: Adapter soft state 3448665484d8SDoug Ambrisko * Command to be completed 3449665484d8SDoug Ambrisko * 34508e727371SKashyap D Desai * In mrsas_issue_blocked_cmd(), after a command is issued to Firmware, a wait 34518e727371SKashyap D Desai * timer is started. This function is called from 34528e727371SKashyap D Desai * mrsas_complete_mptmfi_passthru() as it completes the command, to wake up 34538e727371SKashyap D Desai * from the command wait. 3454665484d8SDoug Ambrisko */ 34558e727371SKashyap D Desai void 34568e727371SKashyap D Desai mrsas_wakeup(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 3457665484d8SDoug Ambrisko { 3458665484d8SDoug Ambrisko cmd->cmd_status = cmd->frame->io.cmd_status; 3459665484d8SDoug Ambrisko 3460665484d8SDoug Ambrisko if (cmd->cmd_status == ECONNREFUSED) 3461665484d8SDoug Ambrisko cmd->cmd_status = 0; 3462665484d8SDoug Ambrisko 3463665484d8SDoug Ambrisko sc->chan = (void *)&cmd; 3464665484d8SDoug Ambrisko wakeup_one((void *)&sc->chan); 3465665484d8SDoug Ambrisko return; 3466665484d8SDoug Ambrisko } 3467665484d8SDoug Ambrisko 34688e727371SKashyap D Desai /* 34698e727371SKashyap D Desai * mrsas_shutdown_ctlr: Instructs FW to shutdown the controller input: 34708e727371SKashyap D Desai * Adapter soft state Shutdown/Hibernate 3471665484d8SDoug Ambrisko * 34728e727371SKashyap D Desai * This function issues a DCMD internal command to Firmware to initiate shutdown 34738e727371SKashyap D Desai * of the controller. 3474665484d8SDoug Ambrisko */ 34758e727371SKashyap D Desai static void 34768e727371SKashyap D Desai mrsas_shutdown_ctlr(struct mrsas_softc *sc, u_int32_t opcode) 3477665484d8SDoug Ambrisko { 3478665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3479665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 3480665484d8SDoug Ambrisko 3481665484d8SDoug Ambrisko if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) 3482665484d8SDoug Ambrisko return; 3483665484d8SDoug Ambrisko 3484665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3485665484d8SDoug Ambrisko if (!cmd) { 3486665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate for shutdown cmd.\n"); 3487665484d8SDoug Ambrisko return; 3488665484d8SDoug Ambrisko } 3489665484d8SDoug Ambrisko if (sc->aen_cmd) 3490665484d8SDoug Ambrisko mrsas_issue_blocked_abort_cmd(sc, sc->aen_cmd); 3491665484d8SDoug Ambrisko 3492665484d8SDoug Ambrisko if (sc->map_update_cmd) 3493665484d8SDoug Ambrisko mrsas_issue_blocked_abort_cmd(sc, sc->map_update_cmd); 3494665484d8SDoug Ambrisko 3495665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 3496665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 3497665484d8SDoug Ambrisko 3498665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 3499665484d8SDoug Ambrisko dcmd->cmd_status = 0x0; 3500665484d8SDoug Ambrisko dcmd->sge_count = 0; 3501665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_NONE; 3502665484d8SDoug Ambrisko dcmd->timeout = 0; 3503665484d8SDoug Ambrisko dcmd->pad_0 = 0; 3504665484d8SDoug Ambrisko dcmd->data_xfer_len = 0; 3505665484d8SDoug Ambrisko dcmd->opcode = opcode; 3506665484d8SDoug Ambrisko 3507665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Preparing to shut down controller.\n"); 3508665484d8SDoug Ambrisko 3509665484d8SDoug Ambrisko mrsas_issue_blocked_cmd(sc, cmd); 3510665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3511665484d8SDoug Ambrisko 3512665484d8SDoug Ambrisko return; 3513665484d8SDoug Ambrisko } 3514665484d8SDoug Ambrisko 35158e727371SKashyap D Desai /* 35168e727371SKashyap D Desai * mrsas_flush_cache: Requests FW to flush all its caches input: 35178e727371SKashyap D Desai * Adapter soft state 3518665484d8SDoug Ambrisko * 3519665484d8SDoug Ambrisko * This function is issues a DCMD internal command to Firmware to initiate 3520665484d8SDoug Ambrisko * flushing of all caches. 3521665484d8SDoug Ambrisko */ 35228e727371SKashyap D Desai static void 35238e727371SKashyap D Desai mrsas_flush_cache(struct mrsas_softc *sc) 3524665484d8SDoug Ambrisko { 3525665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3526665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 3527665484d8SDoug Ambrisko 3528665484d8SDoug Ambrisko if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) 3529665484d8SDoug Ambrisko return; 3530665484d8SDoug Ambrisko 3531665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3532665484d8SDoug Ambrisko if (!cmd) { 3533665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate for flush cache cmd.\n"); 3534665484d8SDoug Ambrisko return; 3535665484d8SDoug Ambrisko } 3536665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 3537665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 3538665484d8SDoug Ambrisko 3539665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 3540665484d8SDoug Ambrisko dcmd->cmd_status = 0x0; 3541665484d8SDoug Ambrisko dcmd->sge_count = 0; 3542665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_NONE; 3543665484d8SDoug Ambrisko dcmd->timeout = 0; 3544665484d8SDoug Ambrisko dcmd->pad_0 = 0; 3545665484d8SDoug Ambrisko dcmd->data_xfer_len = 0; 3546665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_CTRL_CACHE_FLUSH; 3547665484d8SDoug Ambrisko dcmd->mbox.b[0] = MR_FLUSH_CTRL_CACHE | MR_FLUSH_DISK_CACHE; 3548665484d8SDoug Ambrisko 3549665484d8SDoug Ambrisko mrsas_issue_blocked_cmd(sc, cmd); 3550665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3551665484d8SDoug Ambrisko 3552665484d8SDoug Ambrisko return; 3553665484d8SDoug Ambrisko } 3554665484d8SDoug Ambrisko 35558e727371SKashyap D Desai /* 35568e727371SKashyap D Desai * mrsas_get_map_info: Load and validate RAID map input: 35578e727371SKashyap D Desai * Adapter instance soft state 3558665484d8SDoug Ambrisko * 35598e727371SKashyap D Desai * This function calls mrsas_get_ld_map_info() and MR_ValidateMapInfo() to load 35608e727371SKashyap D Desai * and validate RAID map. It returns 0 if successful, 1 other- wise. 3561665484d8SDoug Ambrisko */ 35628e727371SKashyap D Desai static int 35638e727371SKashyap D Desai mrsas_get_map_info(struct mrsas_softc *sc) 3564665484d8SDoug Ambrisko { 3565665484d8SDoug Ambrisko uint8_t retcode = 0; 3566665484d8SDoug Ambrisko 3567665484d8SDoug Ambrisko sc->fast_path_io = 0; 3568665484d8SDoug Ambrisko if (!mrsas_get_ld_map_info(sc)) { 3569665484d8SDoug Ambrisko retcode = MR_ValidateMapInfo(sc); 3570665484d8SDoug Ambrisko if (retcode == 0) { 3571665484d8SDoug Ambrisko sc->fast_path_io = 1; 3572665484d8SDoug Ambrisko return 0; 3573665484d8SDoug Ambrisko } 3574665484d8SDoug Ambrisko } 3575665484d8SDoug Ambrisko return 1; 3576665484d8SDoug Ambrisko } 3577665484d8SDoug Ambrisko 35788e727371SKashyap D Desai /* 35798e727371SKashyap D Desai * mrsas_get_ld_map_info: Get FW's ld_map structure input: 35808e727371SKashyap D Desai * Adapter instance soft state 3581665484d8SDoug Ambrisko * 35828e727371SKashyap D Desai * Issues an internal command (DCMD) to get the FW's controller PD list 35838e727371SKashyap D Desai * structure. 3584665484d8SDoug Ambrisko */ 35858e727371SKashyap D Desai static int 35868e727371SKashyap D Desai mrsas_get_ld_map_info(struct mrsas_softc *sc) 3587665484d8SDoug Ambrisko { 3588665484d8SDoug Ambrisko int retcode = 0; 3589665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3590665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 35914799d485SKashyap D Desai void *map; 3592665484d8SDoug Ambrisko bus_addr_t map_phys_addr = 0; 3593665484d8SDoug Ambrisko 3594665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3595665484d8SDoug Ambrisko if (!cmd) { 35964799d485SKashyap D Desai device_printf(sc->mrsas_dev, 35974799d485SKashyap D Desai "Cannot alloc for ld map info cmd.\n"); 3598665484d8SDoug Ambrisko return 1; 3599665484d8SDoug Ambrisko } 3600665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 3601665484d8SDoug Ambrisko 36024799d485SKashyap D Desai map = (void *)sc->raidmap_mem[(sc->map_id & 1)]; 3603665484d8SDoug Ambrisko map_phys_addr = sc->raidmap_phys_addr[(sc->map_id & 1)]; 3604665484d8SDoug Ambrisko if (!map) { 36054799d485SKashyap D Desai device_printf(sc->mrsas_dev, 36064799d485SKashyap D Desai "Failed to alloc mem for ld map info.\n"); 3607665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3608665484d8SDoug Ambrisko return (ENOMEM); 3609665484d8SDoug Ambrisko } 36104799d485SKashyap D Desai memset(map, 0, sizeof(sc->max_map_sz)); 3611665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 3612665484d8SDoug Ambrisko 3613665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 3614665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 3615665484d8SDoug Ambrisko dcmd->sge_count = 1; 3616665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 3617665484d8SDoug Ambrisko dcmd->timeout = 0; 3618665484d8SDoug Ambrisko dcmd->pad_0 = 0; 36194799d485SKashyap D Desai dcmd->data_xfer_len = sc->current_map_sz; 3620665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_LD_MAP_GET_INFO; 3621665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = map_phys_addr; 36224799d485SKashyap D Desai dcmd->sgl.sge32[0].length = sc->current_map_sz; 36234799d485SKashyap D Desai 3624665484d8SDoug Ambrisko if (!mrsas_issue_polled(sc, cmd)) 3625665484d8SDoug Ambrisko retcode = 0; 36268e727371SKashyap D Desai else { 36274799d485SKashyap D Desai device_printf(sc->mrsas_dev, 36284799d485SKashyap D Desai "Fail to send get LD map info cmd.\n"); 3629665484d8SDoug Ambrisko retcode = 1; 3630665484d8SDoug Ambrisko } 3631665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 36324799d485SKashyap D Desai 3633665484d8SDoug Ambrisko return (retcode); 3634665484d8SDoug Ambrisko } 3635665484d8SDoug Ambrisko 36368e727371SKashyap D Desai /* 36378e727371SKashyap D Desai * mrsas_sync_map_info: Get FW's ld_map structure input: 36388e727371SKashyap D Desai * Adapter instance soft state 3639665484d8SDoug Ambrisko * 36408e727371SKashyap D Desai * Issues an internal command (DCMD) to get the FW's controller PD list 36418e727371SKashyap D Desai * structure. 3642665484d8SDoug Ambrisko */ 36438e727371SKashyap D Desai static int 36448e727371SKashyap D Desai mrsas_sync_map_info(struct mrsas_softc *sc) 3645665484d8SDoug Ambrisko { 3646665484d8SDoug Ambrisko int retcode = 0, i; 3647665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3648665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 3649665484d8SDoug Ambrisko uint32_t size_sync_info, num_lds; 3650665484d8SDoug Ambrisko MR_LD_TARGET_SYNC *target_map = NULL; 36514799d485SKashyap D Desai MR_DRV_RAID_MAP_ALL *map; 3652665484d8SDoug Ambrisko MR_LD_RAID *raid; 3653665484d8SDoug Ambrisko MR_LD_TARGET_SYNC *ld_sync; 3654665484d8SDoug Ambrisko bus_addr_t map_phys_addr = 0; 3655665484d8SDoug Ambrisko 3656665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3657665484d8SDoug Ambrisko if (!cmd) { 36584799d485SKashyap D Desai device_printf(sc->mrsas_dev, 36594799d485SKashyap D Desai "Cannot alloc for sync map info cmd\n"); 3660665484d8SDoug Ambrisko return 1; 3661665484d8SDoug Ambrisko } 36624799d485SKashyap D Desai map = sc->ld_drv_map[sc->map_id & 1]; 3663665484d8SDoug Ambrisko num_lds = map->raidMap.ldCount; 3664665484d8SDoug Ambrisko 3665665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 3666665484d8SDoug Ambrisko size_sync_info = sizeof(MR_LD_TARGET_SYNC) * num_lds; 3667665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 3668665484d8SDoug Ambrisko 36698e727371SKashyap D Desai target_map = (MR_LD_TARGET_SYNC *) sc->raidmap_mem[(sc->map_id - 1) & 1]; 36704799d485SKashyap D Desai memset(target_map, 0, sc->max_map_sz); 3671665484d8SDoug Ambrisko 3672665484d8SDoug Ambrisko map_phys_addr = sc->raidmap_phys_addr[(sc->map_id - 1) & 1]; 3673665484d8SDoug Ambrisko 3674665484d8SDoug Ambrisko ld_sync = (MR_LD_TARGET_SYNC *) target_map; 3675665484d8SDoug Ambrisko 3676665484d8SDoug Ambrisko for (i = 0; i < num_lds; i++, ld_sync++) { 3677665484d8SDoug Ambrisko raid = MR_LdRaidGet(i, map); 3678665484d8SDoug Ambrisko ld_sync->targetId = MR_GetLDTgtId(i, map); 3679665484d8SDoug Ambrisko ld_sync->seqNum = raid->seqNum; 3680665484d8SDoug Ambrisko } 3681665484d8SDoug Ambrisko 3682665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 3683665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 3684665484d8SDoug Ambrisko dcmd->sge_count = 1; 3685665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_WRITE; 3686665484d8SDoug Ambrisko dcmd->timeout = 0; 3687665484d8SDoug Ambrisko dcmd->pad_0 = 0; 36884799d485SKashyap D Desai dcmd->data_xfer_len = sc->current_map_sz; 3689665484d8SDoug Ambrisko dcmd->mbox.b[0] = num_lds; 3690665484d8SDoug Ambrisko dcmd->mbox.b[1] = MRSAS_DCMD_MBOX_PEND_FLAG; 3691665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_LD_MAP_GET_INFO; 3692665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = map_phys_addr; 36934799d485SKashyap D Desai dcmd->sgl.sge32[0].length = sc->current_map_sz; 3694665484d8SDoug Ambrisko 3695665484d8SDoug Ambrisko sc->map_update_cmd = cmd; 3696665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)) { 36974799d485SKashyap D Desai device_printf(sc->mrsas_dev, 36984799d485SKashyap D Desai "Fail to send sync map info command.\n"); 3699665484d8SDoug Ambrisko return (1); 3700665484d8SDoug Ambrisko } 3701665484d8SDoug Ambrisko return (retcode); 3702665484d8SDoug Ambrisko } 3703665484d8SDoug Ambrisko 37048e727371SKashyap D Desai /* 37058e727371SKashyap D Desai * mrsas_get_pd_list: Returns FW's PD list structure input: 37068e727371SKashyap D Desai * Adapter soft state 3707665484d8SDoug Ambrisko * 37088e727371SKashyap D Desai * Issues an internal command (DCMD) to get the FW's controller PD list 37098e727371SKashyap D Desai * structure. This information is mainly used to find out about system 37108e727371SKashyap D Desai * supported by Firmware. 3711665484d8SDoug Ambrisko */ 37128e727371SKashyap D Desai static int 37138e727371SKashyap D Desai mrsas_get_pd_list(struct mrsas_softc *sc) 3714665484d8SDoug Ambrisko { 3715665484d8SDoug Ambrisko int retcode = 0, pd_index = 0, pd_count = 0, pd_list_size; 3716665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3717665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 3718665484d8SDoug Ambrisko struct MR_PD_LIST *pd_list_mem; 3719665484d8SDoug Ambrisko struct MR_PD_ADDRESS *pd_addr; 3720665484d8SDoug Ambrisko bus_addr_t pd_list_phys_addr = 0; 3721665484d8SDoug Ambrisko struct mrsas_tmp_dcmd *tcmd; 3722665484d8SDoug Ambrisko 3723665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3724665484d8SDoug Ambrisko if (!cmd) { 37254799d485SKashyap D Desai device_printf(sc->mrsas_dev, 37264799d485SKashyap D Desai "Cannot alloc for get PD list cmd\n"); 3727665484d8SDoug Ambrisko return 1; 3728665484d8SDoug Ambrisko } 3729665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 3730665484d8SDoug Ambrisko 3731665484d8SDoug Ambrisko tcmd = malloc(sizeof(struct mrsas_tmp_dcmd), M_MRSAS, M_NOWAIT); 3732665484d8SDoug Ambrisko pd_list_size = MRSAS_MAX_PD * sizeof(struct MR_PD_LIST); 3733665484d8SDoug Ambrisko if (mrsas_alloc_tmp_dcmd(sc, tcmd, pd_list_size) != SUCCESS) { 37344799d485SKashyap D Desai device_printf(sc->mrsas_dev, 37354799d485SKashyap D Desai "Cannot alloc dmamap for get PD list cmd\n"); 3736665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3737665484d8SDoug Ambrisko return (ENOMEM); 37388e727371SKashyap D Desai } else { 3739665484d8SDoug Ambrisko pd_list_mem = tcmd->tmp_dcmd_mem; 3740665484d8SDoug Ambrisko pd_list_phys_addr = tcmd->tmp_dcmd_phys_addr; 3741665484d8SDoug Ambrisko } 3742665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 3743665484d8SDoug Ambrisko 3744665484d8SDoug Ambrisko dcmd->mbox.b[0] = MR_PD_QUERY_TYPE_EXPOSED_TO_HOST; 3745665484d8SDoug Ambrisko dcmd->mbox.b[1] = 0; 3746665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 3747665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 3748665484d8SDoug Ambrisko dcmd->sge_count = 1; 3749665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 3750665484d8SDoug Ambrisko dcmd->timeout = 0; 3751665484d8SDoug Ambrisko dcmd->pad_0 = 0; 3752665484d8SDoug Ambrisko dcmd->data_xfer_len = MRSAS_MAX_PD * sizeof(struct MR_PD_LIST); 3753665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_PD_LIST_QUERY; 3754665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = pd_list_phys_addr; 3755665484d8SDoug Ambrisko dcmd->sgl.sge32[0].length = MRSAS_MAX_PD * sizeof(struct MR_PD_LIST); 3756665484d8SDoug Ambrisko 3757665484d8SDoug Ambrisko if (!mrsas_issue_polled(sc, cmd)) 3758665484d8SDoug Ambrisko retcode = 0; 3759665484d8SDoug Ambrisko else 3760665484d8SDoug Ambrisko retcode = 1; 3761665484d8SDoug Ambrisko 3762665484d8SDoug Ambrisko /* Get the instance PD list */ 3763665484d8SDoug Ambrisko pd_count = MRSAS_MAX_PD; 3764665484d8SDoug Ambrisko pd_addr = pd_list_mem->addr; 3765665484d8SDoug Ambrisko if (retcode == 0 && pd_list_mem->count < pd_count) { 37664799d485SKashyap D Desai memset(sc->local_pd_list, 0, 37674799d485SKashyap D Desai MRSAS_MAX_PD * sizeof(struct mrsas_pd_list)); 3768665484d8SDoug Ambrisko for (pd_index = 0; pd_index < pd_list_mem->count; pd_index++) { 3769665484d8SDoug Ambrisko sc->local_pd_list[pd_addr->deviceId].tid = pd_addr->deviceId; 37704799d485SKashyap D Desai sc->local_pd_list[pd_addr->deviceId].driveType = 37714799d485SKashyap D Desai pd_addr->scsiDevType; 37724799d485SKashyap D Desai sc->local_pd_list[pd_addr->deviceId].driveState = 37734799d485SKashyap D Desai MR_PD_STATE_SYSTEM; 3774665484d8SDoug Ambrisko pd_addr++; 3775665484d8SDoug Ambrisko } 3776665484d8SDoug Ambrisko } 37778e727371SKashyap D Desai /* 37788e727371SKashyap D Desai * Use mutext/spinlock if pd_list component size increase more than 37798e727371SKashyap D Desai * 32 bit. 37808e727371SKashyap D Desai */ 3781665484d8SDoug Ambrisko memcpy(sc->pd_list, sc->local_pd_list, sizeof(sc->local_pd_list)); 3782665484d8SDoug Ambrisko mrsas_free_tmp_dcmd(tcmd); 3783665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3784665484d8SDoug Ambrisko free(tcmd, M_MRSAS); 3785665484d8SDoug Ambrisko return (retcode); 3786665484d8SDoug Ambrisko } 3787665484d8SDoug Ambrisko 37888e727371SKashyap D Desai /* 37898e727371SKashyap D Desai * mrsas_get_ld_list: Returns FW's LD list structure input: 37908e727371SKashyap D Desai * Adapter soft state 3791665484d8SDoug Ambrisko * 37928e727371SKashyap D Desai * Issues an internal command (DCMD) to get the FW's controller PD list 37938e727371SKashyap D Desai * structure. This information is mainly used to find out about supported by 37948e727371SKashyap D Desai * the FW. 3795665484d8SDoug Ambrisko */ 37968e727371SKashyap D Desai static int 37978e727371SKashyap D Desai mrsas_get_ld_list(struct mrsas_softc *sc) 3798665484d8SDoug Ambrisko { 3799665484d8SDoug Ambrisko int ld_list_size, retcode = 0, ld_index = 0, ids = 0; 3800665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3801665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 3802665484d8SDoug Ambrisko struct MR_LD_LIST *ld_list_mem; 3803665484d8SDoug Ambrisko bus_addr_t ld_list_phys_addr = 0; 3804665484d8SDoug Ambrisko struct mrsas_tmp_dcmd *tcmd; 3805665484d8SDoug Ambrisko 3806665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3807665484d8SDoug Ambrisko if (!cmd) { 38084799d485SKashyap D Desai device_printf(sc->mrsas_dev, 38094799d485SKashyap D Desai "Cannot alloc for get LD list cmd\n"); 3810665484d8SDoug Ambrisko return 1; 3811665484d8SDoug Ambrisko } 3812665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 3813665484d8SDoug Ambrisko 3814665484d8SDoug Ambrisko tcmd = malloc(sizeof(struct mrsas_tmp_dcmd), M_MRSAS, M_NOWAIT); 3815665484d8SDoug Ambrisko ld_list_size = sizeof(struct MR_LD_LIST); 3816665484d8SDoug Ambrisko if (mrsas_alloc_tmp_dcmd(sc, tcmd, ld_list_size) != SUCCESS) { 38174799d485SKashyap D Desai device_printf(sc->mrsas_dev, 38184799d485SKashyap D Desai "Cannot alloc dmamap for get LD list cmd\n"); 3819665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3820665484d8SDoug Ambrisko return (ENOMEM); 38218e727371SKashyap D Desai } else { 3822665484d8SDoug Ambrisko ld_list_mem = tcmd->tmp_dcmd_mem; 3823665484d8SDoug Ambrisko ld_list_phys_addr = tcmd->tmp_dcmd_phys_addr; 3824665484d8SDoug Ambrisko } 3825665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 3826665484d8SDoug Ambrisko 38274799d485SKashyap D Desai if (sc->max256vdSupport) 38284799d485SKashyap D Desai dcmd->mbox.b[0] = 1; 38294799d485SKashyap D Desai 3830665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 3831665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 3832665484d8SDoug Ambrisko dcmd->sge_count = 1; 3833665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 3834665484d8SDoug Ambrisko dcmd->timeout = 0; 3835665484d8SDoug Ambrisko dcmd->data_xfer_len = sizeof(struct MR_LD_LIST); 3836665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_LD_GET_LIST; 3837665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = ld_list_phys_addr; 3838665484d8SDoug Ambrisko dcmd->sgl.sge32[0].length = sizeof(struct MR_LD_LIST); 3839665484d8SDoug Ambrisko dcmd->pad_0 = 0; 3840665484d8SDoug Ambrisko 3841665484d8SDoug Ambrisko if (!mrsas_issue_polled(sc, cmd)) 3842665484d8SDoug Ambrisko retcode = 0; 3843665484d8SDoug Ambrisko else 3844665484d8SDoug Ambrisko retcode = 1; 3845665484d8SDoug Ambrisko 38464799d485SKashyap D Desai #if VD_EXT_DEBUG 38474799d485SKashyap D Desai printf("Number of LDs %d\n", ld_list_mem->ldCount); 38484799d485SKashyap D Desai #endif 38494799d485SKashyap D Desai 3850665484d8SDoug Ambrisko /* Get the instance LD list */ 38514799d485SKashyap D Desai if ((retcode == 0) && 38524799d485SKashyap D Desai (ld_list_mem->ldCount <= sc->fw_supported_vd_count)) { 3853665484d8SDoug Ambrisko sc->CurLdCount = ld_list_mem->ldCount; 38544799d485SKashyap D Desai memset(sc->ld_ids, 0xff, MAX_LOGICAL_DRIVES_EXT); 3855665484d8SDoug Ambrisko for (ld_index = 0; ld_index < ld_list_mem->ldCount; ld_index++) { 3856665484d8SDoug Ambrisko if (ld_list_mem->ldList[ld_index].state != 0) { 3857665484d8SDoug Ambrisko ids = ld_list_mem->ldList[ld_index].ref.ld_context.targetId; 3858665484d8SDoug Ambrisko sc->ld_ids[ids] = ld_list_mem->ldList[ld_index].ref.ld_context.targetId; 3859665484d8SDoug Ambrisko } 3860665484d8SDoug Ambrisko } 3861665484d8SDoug Ambrisko } 3862665484d8SDoug Ambrisko mrsas_free_tmp_dcmd(tcmd); 3863665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3864665484d8SDoug Ambrisko free(tcmd, M_MRSAS); 3865665484d8SDoug Ambrisko return (retcode); 3866665484d8SDoug Ambrisko } 3867665484d8SDoug Ambrisko 38688e727371SKashyap D Desai /* 38698e727371SKashyap D Desai * mrsas_alloc_tmp_dcmd: Allocates memory for temporary command input: 38708e727371SKashyap D Desai * Adapter soft state Temp command Size of alloction 3871665484d8SDoug Ambrisko * 3872665484d8SDoug Ambrisko * Allocates DMAable memory for a temporary internal command. The allocated 3873665484d8SDoug Ambrisko * memory is initialized to all zeros upon successful loading of the dma 3874665484d8SDoug Ambrisko * mapped memory. 3875665484d8SDoug Ambrisko */ 38768e727371SKashyap D Desai int 38778e727371SKashyap D Desai mrsas_alloc_tmp_dcmd(struct mrsas_softc *sc, 38788e727371SKashyap D Desai struct mrsas_tmp_dcmd *tcmd, int size) 3879665484d8SDoug Ambrisko { 38808e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 38818e727371SKashyap D Desai 1, 0, 38828e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 38838e727371SKashyap D Desai BUS_SPACE_MAXADDR, 38848e727371SKashyap D Desai NULL, NULL, 38858e727371SKashyap D Desai size, 38868e727371SKashyap D Desai 1, 38878e727371SKashyap D Desai size, 38888e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 38898e727371SKashyap D Desai NULL, NULL, 3890665484d8SDoug Ambrisko &tcmd->tmp_dcmd_tag)) { 3891665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate tmp dcmd tag\n"); 3892665484d8SDoug Ambrisko return (ENOMEM); 3893665484d8SDoug Ambrisko } 3894665484d8SDoug Ambrisko if (bus_dmamem_alloc(tcmd->tmp_dcmd_tag, (void **)&tcmd->tmp_dcmd_mem, 3895665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &tcmd->tmp_dcmd_dmamap)) { 3896665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate tmp dcmd mem\n"); 3897665484d8SDoug Ambrisko return (ENOMEM); 3898665484d8SDoug Ambrisko } 3899665484d8SDoug Ambrisko if (bus_dmamap_load(tcmd->tmp_dcmd_tag, tcmd->tmp_dcmd_dmamap, 3900665484d8SDoug Ambrisko tcmd->tmp_dcmd_mem, size, mrsas_addr_cb, 3901665484d8SDoug Ambrisko &tcmd->tmp_dcmd_phys_addr, BUS_DMA_NOWAIT)) { 3902665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load tmp dcmd mem\n"); 3903665484d8SDoug Ambrisko return (ENOMEM); 3904665484d8SDoug Ambrisko } 3905665484d8SDoug Ambrisko memset(tcmd->tmp_dcmd_mem, 0, size); 3906665484d8SDoug Ambrisko return (0); 3907665484d8SDoug Ambrisko } 3908665484d8SDoug Ambrisko 39098e727371SKashyap D Desai /* 39108e727371SKashyap D Desai * mrsas_free_tmp_dcmd: Free memory for temporary command input: 39118e727371SKashyap D Desai * temporary dcmd pointer 3912665484d8SDoug Ambrisko * 39138e727371SKashyap D Desai * Deallocates memory of the temporary command for use in the construction of 39148e727371SKashyap D Desai * the internal DCMD. 3915665484d8SDoug Ambrisko */ 39168e727371SKashyap D Desai void 39178e727371SKashyap D Desai mrsas_free_tmp_dcmd(struct mrsas_tmp_dcmd *tmp) 3918665484d8SDoug Ambrisko { 3919665484d8SDoug Ambrisko if (tmp->tmp_dcmd_phys_addr) 3920665484d8SDoug Ambrisko bus_dmamap_unload(tmp->tmp_dcmd_tag, tmp->tmp_dcmd_dmamap); 3921665484d8SDoug Ambrisko if (tmp->tmp_dcmd_mem != NULL) 3922665484d8SDoug Ambrisko bus_dmamem_free(tmp->tmp_dcmd_tag, tmp->tmp_dcmd_mem, tmp->tmp_dcmd_dmamap); 3923665484d8SDoug Ambrisko if (tmp->tmp_dcmd_tag != NULL) 3924665484d8SDoug Ambrisko bus_dma_tag_destroy(tmp->tmp_dcmd_tag); 3925665484d8SDoug Ambrisko } 3926665484d8SDoug Ambrisko 39278e727371SKashyap D Desai /* 39288e727371SKashyap D Desai * mrsas_issue_blocked_abort_cmd: Aborts previously issued cmd input: 39298e727371SKashyap D Desai * Adapter soft state Previously issued cmd to be aborted 3930665484d8SDoug Ambrisko * 3931665484d8SDoug Ambrisko * This function is used to abort previously issued commands, such as AEN and 3932665484d8SDoug Ambrisko * RAID map sync map commands. The abort command is sent as a DCMD internal 3933665484d8SDoug Ambrisko * command and subsequently the driver will wait for a return status. The 3934665484d8SDoug Ambrisko * max wait time is MRSAS_INTERNAL_CMD_WAIT_TIME seconds. 3935665484d8SDoug Ambrisko */ 39368e727371SKashyap D Desai static int 39378e727371SKashyap D Desai mrsas_issue_blocked_abort_cmd(struct mrsas_softc *sc, 3938665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd_to_abort) 3939665484d8SDoug Ambrisko { 3940665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3941665484d8SDoug Ambrisko struct mrsas_abort_frame *abort_fr; 3942665484d8SDoug Ambrisko u_int8_t retcode = 0; 3943665484d8SDoug Ambrisko unsigned long total_time = 0; 3944665484d8SDoug Ambrisko u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME; 3945665484d8SDoug Ambrisko 3946665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3947665484d8SDoug Ambrisko if (!cmd) { 3948665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc for abort cmd\n"); 3949665484d8SDoug Ambrisko return (1); 3950665484d8SDoug Ambrisko } 3951665484d8SDoug Ambrisko abort_fr = &cmd->frame->abort; 3952665484d8SDoug Ambrisko 3953665484d8SDoug Ambrisko /* Prepare and issue the abort frame */ 3954665484d8SDoug Ambrisko abort_fr->cmd = MFI_CMD_ABORT; 3955665484d8SDoug Ambrisko abort_fr->cmd_status = 0xFF; 3956665484d8SDoug Ambrisko abort_fr->flags = 0; 3957665484d8SDoug Ambrisko abort_fr->abort_context = cmd_to_abort->index; 3958665484d8SDoug Ambrisko abort_fr->abort_mfi_phys_addr_lo = cmd_to_abort->frame_phys_addr; 3959665484d8SDoug Ambrisko abort_fr->abort_mfi_phys_addr_hi = 0; 3960665484d8SDoug Ambrisko 3961665484d8SDoug Ambrisko cmd->sync_cmd = 1; 3962665484d8SDoug Ambrisko cmd->cmd_status = 0xFF; 3963665484d8SDoug Ambrisko 3964665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)) { 3965665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Fail to send abort command.\n"); 3966665484d8SDoug Ambrisko return (1); 3967665484d8SDoug Ambrisko } 3968665484d8SDoug Ambrisko /* Wait for this cmd to complete */ 3969665484d8SDoug Ambrisko sc->chan = (void *)&cmd; 3970665484d8SDoug Ambrisko while (1) { 3971665484d8SDoug Ambrisko if (cmd->cmd_status == 0xFF) { 3972665484d8SDoug Ambrisko tsleep((void *)&sc->chan, 0, "mrsas_sleep", hz); 39738e727371SKashyap D Desai } else 3974665484d8SDoug Ambrisko break; 3975665484d8SDoug Ambrisko total_time++; 3976665484d8SDoug Ambrisko if (total_time >= max_wait) { 3977665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Abort cmd timed out after %d sec.\n", max_wait); 3978665484d8SDoug Ambrisko retcode = 1; 3979665484d8SDoug Ambrisko break; 3980665484d8SDoug Ambrisko } 3981665484d8SDoug Ambrisko } 3982665484d8SDoug Ambrisko 3983665484d8SDoug Ambrisko cmd->sync_cmd = 0; 3984665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3985665484d8SDoug Ambrisko return (retcode); 3986665484d8SDoug Ambrisko } 3987665484d8SDoug Ambrisko 39888e727371SKashyap D Desai /* 39898e727371SKashyap D Desai * mrsas_complete_abort: Completes aborting a command input: 39908e727371SKashyap D Desai * Adapter soft state Cmd that was issued to abort another cmd 3991665484d8SDoug Ambrisko * 39928e727371SKashyap D Desai * The mrsas_issue_blocked_abort_cmd() function waits for the command status to 39938e727371SKashyap D Desai * change after sending the command. This function is called from 3994665484d8SDoug Ambrisko * mrsas_complete_mptmfi_passthru() to wake up the sleep thread associated. 3995665484d8SDoug Ambrisko */ 39968e727371SKashyap D Desai void 39978e727371SKashyap D Desai mrsas_complete_abort(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 3998665484d8SDoug Ambrisko { 3999665484d8SDoug Ambrisko if (cmd->sync_cmd) { 4000665484d8SDoug Ambrisko cmd->sync_cmd = 0; 4001665484d8SDoug Ambrisko cmd->cmd_status = 0; 4002665484d8SDoug Ambrisko sc->chan = (void *)&cmd; 4003665484d8SDoug Ambrisko wakeup_one((void *)&sc->chan); 4004665484d8SDoug Ambrisko } 4005665484d8SDoug Ambrisko return; 4006665484d8SDoug Ambrisko } 4007665484d8SDoug Ambrisko 40088e727371SKashyap D Desai /* 40098e727371SKashyap D Desai * mrsas_aen_handler: AEN processing callback function from thread context 4010665484d8SDoug Ambrisko * input: Adapter soft state 4011665484d8SDoug Ambrisko * 40128e727371SKashyap D Desai * Asynchronous event handler 4013665484d8SDoug Ambrisko */ 40148e727371SKashyap D Desai void 40158e727371SKashyap D Desai mrsas_aen_handler(struct mrsas_softc *sc) 4016665484d8SDoug Ambrisko { 4017665484d8SDoug Ambrisko union mrsas_evt_class_locale class_locale; 4018665484d8SDoug Ambrisko int doscan = 0; 4019665484d8SDoug Ambrisko u_int32_t seq_num; 4020665484d8SDoug Ambrisko int error; 4021665484d8SDoug Ambrisko 4022665484d8SDoug Ambrisko if (!sc) { 4023665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "invalid instance!\n"); 4024665484d8SDoug Ambrisko return; 4025665484d8SDoug Ambrisko } 4026665484d8SDoug Ambrisko if (sc->evt_detail_mem) { 4027665484d8SDoug Ambrisko switch (sc->evt_detail_mem->code) { 4028665484d8SDoug Ambrisko case MR_EVT_PD_INSERTED: 4029665484d8SDoug Ambrisko mrsas_get_pd_list(sc); 4030665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_1); 4031665484d8SDoug Ambrisko doscan = 0; 4032665484d8SDoug Ambrisko break; 4033665484d8SDoug Ambrisko case MR_EVT_PD_REMOVED: 4034665484d8SDoug Ambrisko mrsas_get_pd_list(sc); 4035665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_1); 4036665484d8SDoug Ambrisko doscan = 0; 4037665484d8SDoug Ambrisko break; 4038665484d8SDoug Ambrisko case MR_EVT_LD_OFFLINE: 4039665484d8SDoug Ambrisko case MR_EVT_CFG_CLEARED: 4040665484d8SDoug Ambrisko case MR_EVT_LD_DELETED: 4041665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_0); 4042665484d8SDoug Ambrisko doscan = 0; 4043665484d8SDoug Ambrisko break; 4044665484d8SDoug Ambrisko case MR_EVT_LD_CREATED: 4045665484d8SDoug Ambrisko mrsas_get_ld_list(sc); 4046665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_0); 4047665484d8SDoug Ambrisko doscan = 0; 4048665484d8SDoug Ambrisko break; 4049665484d8SDoug Ambrisko case MR_EVT_CTRL_HOST_BUS_SCAN_REQUESTED: 4050665484d8SDoug Ambrisko case MR_EVT_FOREIGN_CFG_IMPORTED: 4051665484d8SDoug Ambrisko case MR_EVT_LD_STATE_CHANGE: 4052665484d8SDoug Ambrisko doscan = 1; 4053665484d8SDoug Ambrisko break; 4054665484d8SDoug Ambrisko default: 4055665484d8SDoug Ambrisko doscan = 0; 4056665484d8SDoug Ambrisko break; 4057665484d8SDoug Ambrisko } 4058665484d8SDoug Ambrisko } else { 4059665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "invalid evt_detail\n"); 4060665484d8SDoug Ambrisko return; 4061665484d8SDoug Ambrisko } 4062665484d8SDoug Ambrisko if (doscan) { 4063665484d8SDoug Ambrisko mrsas_get_pd_list(sc); 4064665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_AEN, "scanning ...sim 1\n"); 4065665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_1); 4066665484d8SDoug Ambrisko mrsas_get_ld_list(sc); 4067665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_AEN, "scanning ...sim 0\n"); 4068665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_0); 4069665484d8SDoug Ambrisko } 4070665484d8SDoug Ambrisko seq_num = sc->evt_detail_mem->seq_num + 1; 4071665484d8SDoug Ambrisko 40728e727371SKashyap D Desai /* Register AEN with FW for latest sequence number plus 1 */ 4073665484d8SDoug Ambrisko class_locale.members.reserved = 0; 4074665484d8SDoug Ambrisko class_locale.members.locale = MR_EVT_LOCALE_ALL; 4075665484d8SDoug Ambrisko class_locale.members.class = MR_EVT_CLASS_DEBUG; 4076665484d8SDoug Ambrisko 4077665484d8SDoug Ambrisko if (sc->aen_cmd != NULL) 4078665484d8SDoug Ambrisko return; 4079665484d8SDoug Ambrisko 4080665484d8SDoug Ambrisko mtx_lock(&sc->aen_lock); 4081665484d8SDoug Ambrisko error = mrsas_register_aen(sc, seq_num, 4082665484d8SDoug Ambrisko class_locale.word); 4083665484d8SDoug Ambrisko mtx_unlock(&sc->aen_lock); 4084665484d8SDoug Ambrisko 4085665484d8SDoug Ambrisko if (error) 4086665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "register aen failed error %x\n", error); 4087665484d8SDoug Ambrisko 4088665484d8SDoug Ambrisko } 4089665484d8SDoug Ambrisko 4090665484d8SDoug Ambrisko 40918e727371SKashyap D Desai /* 4092665484d8SDoug Ambrisko * mrsas_complete_aen: Completes AEN command 4093665484d8SDoug Ambrisko * input: Adapter soft state 4094665484d8SDoug Ambrisko * Cmd that was issued to abort another cmd 4095665484d8SDoug Ambrisko * 40968e727371SKashyap D Desai * This function will be called from ISR and will continue event processing from 40978e727371SKashyap D Desai * thread context by enqueuing task in ev_tq (callback function 40988e727371SKashyap D Desai * "mrsas_aen_handler"). 4099665484d8SDoug Ambrisko */ 41008e727371SKashyap D Desai void 41018e727371SKashyap D Desai mrsas_complete_aen(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 4102665484d8SDoug Ambrisko { 4103665484d8SDoug Ambrisko /* 41048e727371SKashyap D Desai * Don't signal app if it is just an aborted previously registered 41058e727371SKashyap D Desai * aen 4106665484d8SDoug Ambrisko */ 4107665484d8SDoug Ambrisko if ((!cmd->abort_aen) && (sc->remove_in_progress == 0)) { 4108da011113SKashyap D Desai sc->mrsas_aen_triggered = 1; 4109ecea5be4SKashyap D Desai mtx_lock(&sc->aen_lock); 4110da011113SKashyap D Desai if (sc->mrsas_poll_waiting) { 4111da011113SKashyap D Desai sc->mrsas_poll_waiting = 0; 4112da011113SKashyap D Desai selwakeup(&sc->mrsas_select); 4113da011113SKashyap D Desai } 4114ecea5be4SKashyap D Desai mtx_unlock(&sc->aen_lock); 41158e727371SKashyap D Desai } else 4116665484d8SDoug Ambrisko cmd->abort_aen = 0; 4117665484d8SDoug Ambrisko 4118665484d8SDoug Ambrisko sc->aen_cmd = NULL; 4119665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 4120665484d8SDoug Ambrisko 4121665484d8SDoug Ambrisko if (!sc->remove_in_progress) 4122665484d8SDoug Ambrisko taskqueue_enqueue(sc->ev_tq, &sc->ev_task); 4123665484d8SDoug Ambrisko 4124665484d8SDoug Ambrisko return; 4125665484d8SDoug Ambrisko } 4126665484d8SDoug Ambrisko 4127665484d8SDoug Ambrisko static device_method_t mrsas_methods[] = { 4128665484d8SDoug Ambrisko DEVMETHOD(device_probe, mrsas_probe), 4129665484d8SDoug Ambrisko DEVMETHOD(device_attach, mrsas_attach), 4130665484d8SDoug Ambrisko DEVMETHOD(device_detach, mrsas_detach), 4131665484d8SDoug Ambrisko DEVMETHOD(device_suspend, mrsas_suspend), 4132665484d8SDoug Ambrisko DEVMETHOD(device_resume, mrsas_resume), 4133665484d8SDoug Ambrisko DEVMETHOD(bus_print_child, bus_generic_print_child), 4134665484d8SDoug Ambrisko DEVMETHOD(bus_driver_added, bus_generic_driver_added), 4135665484d8SDoug Ambrisko {0, 0} 4136665484d8SDoug Ambrisko }; 4137665484d8SDoug Ambrisko 4138665484d8SDoug Ambrisko static driver_t mrsas_driver = { 4139665484d8SDoug Ambrisko "mrsas", 4140665484d8SDoug Ambrisko mrsas_methods, 4141665484d8SDoug Ambrisko sizeof(struct mrsas_softc) 4142665484d8SDoug Ambrisko }; 4143665484d8SDoug Ambrisko 4144665484d8SDoug Ambrisko static devclass_t mrsas_devclass; 41458e727371SKashyap D Desai 4146665484d8SDoug Ambrisko DRIVER_MODULE(mrsas, pci, mrsas_driver, mrsas_devclass, 0, 0); 4147665484d8SDoug Ambrisko MODULE_DEPEND(mrsas, cam, 1, 1, 1); 4148