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> 518071588dSKashyap D Desai #include <sys/sysent.h> 52665484d8SDoug Ambrisko #include <sys/kthread.h> 53665484d8SDoug Ambrisko #include <sys/taskqueue.h> 54d18d1b47SKashyap D Desai #include <sys/smp.h> 55665484d8SDoug Ambrisko 56665484d8SDoug Ambrisko 57665484d8SDoug Ambrisko /* 58665484d8SDoug Ambrisko * Function prototypes 59665484d8SDoug Ambrisko */ 60665484d8SDoug Ambrisko static d_open_t mrsas_open; 61665484d8SDoug Ambrisko static d_close_t mrsas_close; 62665484d8SDoug Ambrisko static d_read_t mrsas_read; 63665484d8SDoug Ambrisko static d_write_t mrsas_write; 64665484d8SDoug Ambrisko static d_ioctl_t mrsas_ioctl; 65da011113SKashyap D Desai static d_poll_t mrsas_poll; 66665484d8SDoug Ambrisko 678071588dSKashyap D Desai static void mrsas_ich_startup(void *arg); 68536094dcSKashyap D Desai static struct mrsas_mgmt_info mrsas_mgmt_info; 69665484d8SDoug Ambrisko static struct mrsas_ident *mrsas_find_ident(device_t); 70d18d1b47SKashyap D Desai static int mrsas_setup_msix(struct mrsas_softc *sc); 71d18d1b47SKashyap D Desai static int mrsas_allocate_msix(struct mrsas_softc *sc); 72665484d8SDoug Ambrisko static void mrsas_shutdown_ctlr(struct mrsas_softc *sc, u_int32_t opcode); 73665484d8SDoug Ambrisko static void mrsas_flush_cache(struct mrsas_softc *sc); 74665484d8SDoug Ambrisko static void mrsas_reset_reply_desc(struct mrsas_softc *sc); 75665484d8SDoug Ambrisko static void mrsas_ocr_thread(void *arg); 76665484d8SDoug Ambrisko static int mrsas_get_map_info(struct mrsas_softc *sc); 77665484d8SDoug Ambrisko static int mrsas_get_ld_map_info(struct mrsas_softc *sc); 78665484d8SDoug Ambrisko static int mrsas_sync_map_info(struct mrsas_softc *sc); 79665484d8SDoug Ambrisko static int mrsas_get_pd_list(struct mrsas_softc *sc); 80665484d8SDoug Ambrisko static int mrsas_get_ld_list(struct mrsas_softc *sc); 81665484d8SDoug Ambrisko static int mrsas_setup_irq(struct mrsas_softc *sc); 82665484d8SDoug Ambrisko static int mrsas_alloc_mem(struct mrsas_softc *sc); 83665484d8SDoug Ambrisko static int mrsas_init_fw(struct mrsas_softc *sc); 84665484d8SDoug Ambrisko static int mrsas_setup_raidmap(struct mrsas_softc *sc); 85a688fcd0SKashyap D Desai static void megasas_setup_jbod_map(struct mrsas_softc *sc); 86a688fcd0SKashyap D Desai static int megasas_sync_pd_seq_num(struct mrsas_softc *sc, boolean_t pend); 87665484d8SDoug Ambrisko static int mrsas_clear_intr(struct mrsas_softc *sc); 88af51c29fSKashyap D Desai static int mrsas_get_ctrl_info(struct mrsas_softc *sc); 89af51c29fSKashyap D Desai static void mrsas_update_ext_vd_details(struct mrsas_softc *sc); 908e727371SKashyap D Desai static int 918e727371SKashyap D Desai mrsas_issue_blocked_abort_cmd(struct mrsas_softc *sc, 92665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd_to_abort); 93dbcc81dfSKashyap D Desai static struct mrsas_softc * 94dbcc81dfSKashyap D Desai mrsas_get_softc_instance(struct cdev *dev, 955844115eSKashyap D Desai u_long cmd, caddr_t arg); 96665484d8SDoug Ambrisko u_int32_t mrsas_read_reg(struct mrsas_softc *sc, int offset); 978e727371SKashyap D Desai u_int8_t 988e727371SKashyap D Desai mrsas_build_mptmfi_passthru(struct mrsas_softc *sc, 99665484d8SDoug Ambrisko struct mrsas_mfi_cmd *mfi_cmd); 100daeed973SKashyap D Desai void mrsas_complete_outstanding_ioctls(struct mrsas_softc *sc); 101665484d8SDoug Ambrisko int mrsas_transition_to_ready(struct mrsas_softc *sc, int ocr); 102665484d8SDoug Ambrisko int mrsas_init_adapter(struct mrsas_softc *sc); 103665484d8SDoug Ambrisko int mrsas_alloc_mpt_cmds(struct mrsas_softc *sc); 104665484d8SDoug Ambrisko int mrsas_alloc_ioc_cmd(struct mrsas_softc *sc); 105665484d8SDoug Ambrisko int mrsas_alloc_ctlr_info_cmd(struct mrsas_softc *sc); 106665484d8SDoug Ambrisko int mrsas_ioc_init(struct mrsas_softc *sc); 107665484d8SDoug Ambrisko int mrsas_bus_scan(struct mrsas_softc *sc); 108665484d8SDoug Ambrisko int mrsas_issue_dcmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 109665484d8SDoug Ambrisko int mrsas_issue_polled(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 110f0c7594bSKashyap D Desai int mrsas_reset_ctrl(struct mrsas_softc *sc, u_int8_t reset_reason); 111f0c7594bSKashyap D Desai int mrsas_wait_for_outstanding(struct mrsas_softc *sc, u_int8_t check_reason); 1124bb0a4f0SKashyap D Desai int mrsas_complete_cmd(struct mrsas_softc *sc, u_int32_t MSIxIndex); 1138bb601acSKashyap D Desai int mrsas_reset_targets(struct mrsas_softc *sc); 1148e727371SKashyap D Desai int 1158e727371SKashyap D Desai mrsas_issue_blocked_cmd(struct mrsas_softc *sc, 116665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd); 1178e727371SKashyap D Desai int 1188e727371SKashyap D Desai mrsas_alloc_tmp_dcmd(struct mrsas_softc *sc, struct mrsas_tmp_dcmd *tcmd, 119665484d8SDoug Ambrisko int size); 120665484d8SDoug Ambrisko void mrsas_release_mfi_cmd(struct mrsas_mfi_cmd *cmd); 121665484d8SDoug Ambrisko void mrsas_wakeup(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 122665484d8SDoug Ambrisko void mrsas_complete_aen(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 123665484d8SDoug Ambrisko void mrsas_complete_abort(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 124665484d8SDoug Ambrisko void mrsas_disable_intr(struct mrsas_softc *sc); 125665484d8SDoug Ambrisko void mrsas_enable_intr(struct mrsas_softc *sc); 126665484d8SDoug Ambrisko void mrsas_free_ioc_cmd(struct mrsas_softc *sc); 127665484d8SDoug Ambrisko void mrsas_free_mem(struct mrsas_softc *sc); 128665484d8SDoug Ambrisko void mrsas_free_tmp_dcmd(struct mrsas_tmp_dcmd *tmp); 129665484d8SDoug Ambrisko void mrsas_isr(void *arg); 130665484d8SDoug Ambrisko void mrsas_teardown_intr(struct mrsas_softc *sc); 131665484d8SDoug Ambrisko void mrsas_addr_cb(void *arg, bus_dma_segment_t *segs, int nsegs, int error); 132665484d8SDoug Ambrisko void mrsas_kill_hba(struct mrsas_softc *sc); 133665484d8SDoug Ambrisko void mrsas_aen_handler(struct mrsas_softc *sc); 1348e727371SKashyap D Desai void 1358e727371SKashyap D Desai mrsas_write_reg(struct mrsas_softc *sc, int offset, 136665484d8SDoug Ambrisko u_int32_t value); 1378e727371SKashyap D Desai void 1388e727371SKashyap D Desai mrsas_fire_cmd(struct mrsas_softc *sc, u_int32_t req_desc_lo, 139665484d8SDoug Ambrisko u_int32_t req_desc_hi); 140665484d8SDoug Ambrisko void mrsas_free_ctlr_info_cmd(struct mrsas_softc *sc); 1418e727371SKashyap D Desai void 1428e727371SKashyap D Desai mrsas_complete_mptmfi_passthru(struct mrsas_softc *sc, 143665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd, u_int8_t status); 1448e727371SKashyap D Desai void 1458e727371SKashyap D Desai mrsas_map_mpt_cmd_status(struct mrsas_mpt_cmd *cmd, u_int8_t status, 146665484d8SDoug Ambrisko u_int8_t extStatus); 147665484d8SDoug Ambrisko struct mrsas_mfi_cmd *mrsas_get_mfi_cmd(struct mrsas_softc *sc); 1488e727371SKashyap D Desai 1498e727371SKashyap D Desai MRSAS_REQUEST_DESCRIPTOR_UNION *mrsas_build_mpt_cmd 1508e727371SKashyap D Desai (struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 151665484d8SDoug Ambrisko 152665484d8SDoug Ambrisko extern int mrsas_cam_attach(struct mrsas_softc *sc); 153665484d8SDoug Ambrisko extern void mrsas_cam_detach(struct mrsas_softc *sc); 154665484d8SDoug Ambrisko extern void mrsas_cmd_done(struct mrsas_softc *sc, struct mrsas_mpt_cmd *cmd); 155665484d8SDoug Ambrisko extern void mrsas_free_frame(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 156665484d8SDoug Ambrisko extern int mrsas_alloc_mfi_cmds(struct mrsas_softc *sc); 157665484d8SDoug Ambrisko extern struct mrsas_mpt_cmd *mrsas_get_mpt_cmd(struct mrsas_softc *sc); 158536094dcSKashyap D Desai extern int mrsas_passthru(struct mrsas_softc *sc, void *arg, u_long ioctlCmd); 159665484d8SDoug Ambrisko extern uint8_t MR_ValidateMapInfo(struct mrsas_softc *sc); 1604799d485SKashyap D Desai extern u_int16_t MR_GetLDTgtId(u_int32_t ld, MR_DRV_RAID_MAP_ALL * map); 1614799d485SKashyap D Desai extern MR_LD_RAID *MR_LdRaidGet(u_int32_t ld, MR_DRV_RAID_MAP_ALL * map); 162665484d8SDoug Ambrisko extern void mrsas_xpt_freeze(struct mrsas_softc *sc); 163665484d8SDoug Ambrisko extern void mrsas_xpt_release(struct mrsas_softc *sc); 1648e727371SKashyap D Desai extern MRSAS_REQUEST_DESCRIPTOR_UNION * 1658e727371SKashyap D Desai mrsas_get_request_desc(struct mrsas_softc *sc, 166665484d8SDoug Ambrisko u_int16_t index); 167665484d8SDoug Ambrisko extern int mrsas_bus_scan_sim(struct mrsas_softc *sc, struct cam_sim *sim); 168665484d8SDoug Ambrisko static int mrsas_alloc_evt_log_info_cmd(struct mrsas_softc *sc); 169665484d8SDoug Ambrisko static void mrsas_free_evt_log_info_cmd(struct mrsas_softc *sc); 1708e727371SKashyap D Desai 171665484d8SDoug Ambrisko SYSCTL_NODE(_hw, OID_AUTO, mrsas, CTLFLAG_RD, 0, "MRSAS Driver Parameters"); 172665484d8SDoug Ambrisko 1738e727371SKashyap D Desai /* 174665484d8SDoug Ambrisko * PCI device struct and table 175665484d8SDoug Ambrisko * 176665484d8SDoug Ambrisko */ 177665484d8SDoug Ambrisko typedef struct mrsas_ident { 178665484d8SDoug Ambrisko uint16_t vendor; 179665484d8SDoug Ambrisko uint16_t device; 180665484d8SDoug Ambrisko uint16_t subvendor; 181665484d8SDoug Ambrisko uint16_t subdevice; 182665484d8SDoug Ambrisko const char *desc; 183665484d8SDoug Ambrisko } MRSAS_CTLR_ID; 184665484d8SDoug Ambrisko 185665484d8SDoug Ambrisko MRSAS_CTLR_ID device_table[] = { 186ecea5be4SKashyap D Desai {0x1000, MRSAS_TBOLT, 0xffff, 0xffff, "AVAGO Thunderbolt SAS Controller"}, 187ecea5be4SKashyap D Desai {0x1000, MRSAS_INVADER, 0xffff, 0xffff, "AVAGO Invader SAS Controller"}, 188ecea5be4SKashyap D Desai {0x1000, MRSAS_FURY, 0xffff, 0xffff, "AVAGO Fury SAS Controller"}, 189c620f351SKashyap D Desai {0x1000, MRSAS_INTRUDER, 0xffff, 0xffff, "AVAGO Intruder SAS Controller"}, 190c620f351SKashyap D Desai {0x1000, MRSAS_INTRUDER_24, 0xffff, 0xffff, "AVAGO Intruder_24 SAS Controller"}, 1918cd174a4SKashyap D Desai {0x1000, MRSAS_CUTLASS_52, 0xffff, 0xffff, "AVAGO Cutlass_52 SAS Controller"}, 1928cd174a4SKashyap D Desai {0x1000, MRSAS_CUTLASS_53, 0xffff, 0xffff, "AVAGO Cutlass_53 SAS Controller"}, 1937aade8bfSKashyap D Desai {0x1000, MRSAS_VENTURA, 0xffff, 0xffff, "AVAGO Ventura SAS Controller"}, 1947aade8bfSKashyap D Desai {0x1000, MRSAS_CRUSADER, 0xffff, 0xffff, "AVAGO Crusader SAS Controller"}, 1957aade8bfSKashyap D Desai {0x1000, MRSAS_HARPOON, 0xffff, 0xffff, "AVAGO Harpoon SAS Controller"}, 1967aade8bfSKashyap D Desai {0x1000, MRSAS_TOMCAT, 0xffff, 0xffff, "AVAGO Tomcat SAS Controller"}, 1977aade8bfSKashyap D Desai {0x1000, MRSAS_VENTURA_4PORT, 0xffff, 0xffff, "AVAGO Ventura_4Port SAS Controller"}, 1987aade8bfSKashyap D Desai {0x1000, MRSAS_CRUSADER_4PORT, 0xffff, 0xffff, "AVAGO Crusader_4Port SAS Controller"}, 199665484d8SDoug Ambrisko {0, 0, 0, 0, NULL} 200665484d8SDoug Ambrisko }; 201665484d8SDoug Ambrisko 2028e727371SKashyap D Desai /* 203665484d8SDoug Ambrisko * Character device entry points 204665484d8SDoug Ambrisko * 205665484d8SDoug Ambrisko */ 206665484d8SDoug Ambrisko static struct cdevsw mrsas_cdevsw = { 207665484d8SDoug Ambrisko .d_version = D_VERSION, 208665484d8SDoug Ambrisko .d_open = mrsas_open, 209665484d8SDoug Ambrisko .d_close = mrsas_close, 210665484d8SDoug Ambrisko .d_read = mrsas_read, 211665484d8SDoug Ambrisko .d_write = mrsas_write, 212665484d8SDoug Ambrisko .d_ioctl = mrsas_ioctl, 213da011113SKashyap D Desai .d_poll = mrsas_poll, 214665484d8SDoug Ambrisko .d_name = "mrsas", 215665484d8SDoug Ambrisko }; 216665484d8SDoug Ambrisko 217665484d8SDoug Ambrisko MALLOC_DEFINE(M_MRSAS, "mrsasbuf", "Buffers for the MRSAS driver"); 218665484d8SDoug Ambrisko 2198e727371SKashyap D Desai /* 2208e727371SKashyap D Desai * In the cdevsw routines, we find our softc by using the si_drv1 member of 2218e727371SKashyap D Desai * struct cdev. We set this variable to point to our softc in our attach 2228e727371SKashyap D Desai * routine when we create the /dev entry. 223665484d8SDoug Ambrisko */ 224665484d8SDoug Ambrisko int 2257fc5f329SJohn Baldwin mrsas_open(struct cdev *dev, int oflags, int devtype, struct thread *td) 226665484d8SDoug Ambrisko { 227665484d8SDoug Ambrisko struct mrsas_softc *sc; 228665484d8SDoug Ambrisko 229665484d8SDoug Ambrisko sc = dev->si_drv1; 230665484d8SDoug Ambrisko return (0); 231665484d8SDoug Ambrisko } 232665484d8SDoug Ambrisko 233665484d8SDoug Ambrisko int 2347fc5f329SJohn Baldwin mrsas_close(struct cdev *dev, int fflag, int devtype, struct thread *td) 235665484d8SDoug Ambrisko { 236665484d8SDoug Ambrisko struct mrsas_softc *sc; 237665484d8SDoug Ambrisko 238665484d8SDoug Ambrisko sc = dev->si_drv1; 239665484d8SDoug Ambrisko return (0); 240665484d8SDoug Ambrisko } 241665484d8SDoug Ambrisko 242665484d8SDoug Ambrisko int 243665484d8SDoug Ambrisko mrsas_read(struct cdev *dev, struct uio *uio, int ioflag) 244665484d8SDoug Ambrisko { 245665484d8SDoug Ambrisko struct mrsas_softc *sc; 246665484d8SDoug Ambrisko 247665484d8SDoug Ambrisko sc = dev->si_drv1; 248665484d8SDoug Ambrisko return (0); 249665484d8SDoug Ambrisko } 250665484d8SDoug Ambrisko int 251665484d8SDoug Ambrisko mrsas_write(struct cdev *dev, struct uio *uio, int ioflag) 252665484d8SDoug Ambrisko { 253665484d8SDoug Ambrisko struct mrsas_softc *sc; 254665484d8SDoug Ambrisko 255665484d8SDoug Ambrisko sc = dev->si_drv1; 256665484d8SDoug Ambrisko return (0); 257665484d8SDoug Ambrisko } 258665484d8SDoug Ambrisko 2598e727371SKashyap D Desai /* 260665484d8SDoug Ambrisko * Register Read/Write Functions 261665484d8SDoug Ambrisko * 262665484d8SDoug Ambrisko */ 263665484d8SDoug Ambrisko void 264665484d8SDoug Ambrisko mrsas_write_reg(struct mrsas_softc *sc, int offset, 265665484d8SDoug Ambrisko u_int32_t value) 266665484d8SDoug Ambrisko { 267665484d8SDoug Ambrisko bus_space_tag_t bus_tag = sc->bus_tag; 268665484d8SDoug Ambrisko bus_space_handle_t bus_handle = sc->bus_handle; 269665484d8SDoug Ambrisko 270665484d8SDoug Ambrisko bus_space_write_4(bus_tag, bus_handle, offset, value); 271665484d8SDoug Ambrisko } 272665484d8SDoug Ambrisko 273665484d8SDoug Ambrisko u_int32_t 274665484d8SDoug Ambrisko mrsas_read_reg(struct mrsas_softc *sc, int offset) 275665484d8SDoug Ambrisko { 276665484d8SDoug Ambrisko bus_space_tag_t bus_tag = sc->bus_tag; 277665484d8SDoug Ambrisko bus_space_handle_t bus_handle = sc->bus_handle; 278665484d8SDoug Ambrisko 279665484d8SDoug Ambrisko return ((u_int32_t)bus_space_read_4(bus_tag, bus_handle, offset)); 280665484d8SDoug Ambrisko } 281665484d8SDoug Ambrisko 282665484d8SDoug Ambrisko 2838e727371SKashyap D Desai /* 284665484d8SDoug Ambrisko * Interrupt Disable/Enable/Clear Functions 285665484d8SDoug Ambrisko * 286665484d8SDoug Ambrisko */ 2878e727371SKashyap D Desai void 2888e727371SKashyap D Desai mrsas_disable_intr(struct mrsas_softc *sc) 289665484d8SDoug Ambrisko { 290665484d8SDoug Ambrisko u_int32_t mask = 0xFFFFFFFF; 291665484d8SDoug Ambrisko u_int32_t status; 292665484d8SDoug Ambrisko 2932f863eb8SKashyap D Desai sc->mask_interrupts = 1; 294665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask), mask); 295665484d8SDoug Ambrisko /* Dummy read to force pci flush */ 296665484d8SDoug Ambrisko status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask)); 297665484d8SDoug Ambrisko } 298665484d8SDoug Ambrisko 2998e727371SKashyap D Desai void 3008e727371SKashyap D Desai mrsas_enable_intr(struct mrsas_softc *sc) 301665484d8SDoug Ambrisko { 302665484d8SDoug Ambrisko u_int32_t mask = MFI_FUSION_ENABLE_INTERRUPT_MASK; 303665484d8SDoug Ambrisko u_int32_t status; 304665484d8SDoug Ambrisko 3052f863eb8SKashyap D Desai sc->mask_interrupts = 0; 306665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status), ~0); 307665484d8SDoug Ambrisko status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status)); 308665484d8SDoug Ambrisko 309665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask), ~mask); 310665484d8SDoug Ambrisko status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask)); 311665484d8SDoug Ambrisko } 312665484d8SDoug Ambrisko 3138e727371SKashyap D Desai static int 3148e727371SKashyap D Desai mrsas_clear_intr(struct mrsas_softc *sc) 315665484d8SDoug Ambrisko { 3168bb601acSKashyap D Desai u_int32_t status; 317665484d8SDoug Ambrisko 318665484d8SDoug Ambrisko /* Read received interrupt */ 319665484d8SDoug Ambrisko status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status)); 320665484d8SDoug Ambrisko 321665484d8SDoug Ambrisko /* Not our interrupt, so just return */ 322665484d8SDoug Ambrisko if (!(status & MFI_FUSION_ENABLE_INTERRUPT_MASK)) 323665484d8SDoug Ambrisko return (0); 324665484d8SDoug Ambrisko 325665484d8SDoug Ambrisko /* We got a reply interrupt */ 326665484d8SDoug Ambrisko return (1); 327665484d8SDoug Ambrisko } 328665484d8SDoug Ambrisko 3298e727371SKashyap D Desai /* 330665484d8SDoug Ambrisko * PCI Support Functions 331665484d8SDoug Ambrisko * 332665484d8SDoug Ambrisko */ 3338e727371SKashyap D Desai static struct mrsas_ident * 3348e727371SKashyap D Desai mrsas_find_ident(device_t dev) 335665484d8SDoug Ambrisko { 336665484d8SDoug Ambrisko struct mrsas_ident *pci_device; 337665484d8SDoug Ambrisko 3388e727371SKashyap D Desai for (pci_device = device_table; pci_device->vendor != 0; pci_device++) { 339665484d8SDoug Ambrisko if ((pci_device->vendor == pci_get_vendor(dev)) && 340665484d8SDoug Ambrisko (pci_device->device == pci_get_device(dev)) && 341665484d8SDoug Ambrisko ((pci_device->subvendor == pci_get_subvendor(dev)) || 342665484d8SDoug Ambrisko (pci_device->subvendor == 0xffff)) && 343665484d8SDoug Ambrisko ((pci_device->subdevice == pci_get_subdevice(dev)) || 344665484d8SDoug Ambrisko (pci_device->subdevice == 0xffff))) 345665484d8SDoug Ambrisko return (pci_device); 346665484d8SDoug Ambrisko } 347665484d8SDoug Ambrisko return (NULL); 348665484d8SDoug Ambrisko } 349665484d8SDoug Ambrisko 3508e727371SKashyap D Desai static int 3518e727371SKashyap D Desai mrsas_probe(device_t dev) 352665484d8SDoug Ambrisko { 353665484d8SDoug Ambrisko static u_int8_t first_ctrl = 1; 354665484d8SDoug Ambrisko struct mrsas_ident *id; 355665484d8SDoug Ambrisko 356665484d8SDoug Ambrisko if ((id = mrsas_find_ident(dev)) != NULL) { 357665484d8SDoug Ambrisko if (first_ctrl) { 358ecea5be4SKashyap D Desai printf("AVAGO MegaRAID SAS FreeBSD mrsas driver version: %s\n", 3598e727371SKashyap D Desai MRSAS_VERSION); 360665484d8SDoug Ambrisko first_ctrl = 0; 361665484d8SDoug Ambrisko } 362665484d8SDoug Ambrisko device_set_desc(dev, id->desc); 363665484d8SDoug Ambrisko /* between BUS_PROBE_DEFAULT and BUS_PROBE_LOW_PRIORITY */ 364665484d8SDoug Ambrisko return (-30); 365665484d8SDoug Ambrisko } 366665484d8SDoug Ambrisko return (ENXIO); 367665484d8SDoug Ambrisko } 368665484d8SDoug Ambrisko 3698e727371SKashyap D Desai /* 370665484d8SDoug Ambrisko * mrsas_setup_sysctl: setup sysctl values for mrsas 371665484d8SDoug Ambrisko * input: Adapter instance soft state 372665484d8SDoug Ambrisko * 373665484d8SDoug Ambrisko * Setup sysctl entries for mrsas driver. 374665484d8SDoug Ambrisko */ 375665484d8SDoug Ambrisko static void 376665484d8SDoug Ambrisko mrsas_setup_sysctl(struct mrsas_softc *sc) 377665484d8SDoug Ambrisko { 378665484d8SDoug Ambrisko struct sysctl_ctx_list *sysctl_ctx = NULL; 379665484d8SDoug Ambrisko struct sysctl_oid *sysctl_tree = NULL; 380665484d8SDoug Ambrisko char tmpstr[80], tmpstr2[80]; 381665484d8SDoug Ambrisko 382665484d8SDoug Ambrisko /* 383665484d8SDoug Ambrisko * Setup the sysctl variable so the user can change the debug level 384665484d8SDoug Ambrisko * on the fly. 385665484d8SDoug Ambrisko */ 386665484d8SDoug Ambrisko snprintf(tmpstr, sizeof(tmpstr), "MRSAS controller %d", 387665484d8SDoug Ambrisko device_get_unit(sc->mrsas_dev)); 388665484d8SDoug Ambrisko snprintf(tmpstr2, sizeof(tmpstr2), "%d", device_get_unit(sc->mrsas_dev)); 389665484d8SDoug Ambrisko 390665484d8SDoug Ambrisko sysctl_ctx = device_get_sysctl_ctx(sc->mrsas_dev); 391665484d8SDoug Ambrisko if (sysctl_ctx != NULL) 392665484d8SDoug Ambrisko sysctl_tree = device_get_sysctl_tree(sc->mrsas_dev); 393665484d8SDoug Ambrisko 394665484d8SDoug Ambrisko if (sysctl_tree == NULL) { 395665484d8SDoug Ambrisko sysctl_ctx_init(&sc->sysctl_ctx); 396665484d8SDoug Ambrisko sc->sysctl_tree = SYSCTL_ADD_NODE(&sc->sysctl_ctx, 397665484d8SDoug Ambrisko SYSCTL_STATIC_CHILDREN(_hw_mrsas), OID_AUTO, tmpstr2, 398665484d8SDoug Ambrisko CTLFLAG_RD, 0, tmpstr); 399665484d8SDoug Ambrisko if (sc->sysctl_tree == NULL) 400665484d8SDoug Ambrisko return; 401665484d8SDoug Ambrisko sysctl_ctx = &sc->sysctl_ctx; 402665484d8SDoug Ambrisko sysctl_tree = sc->sysctl_tree; 403665484d8SDoug Ambrisko } 404665484d8SDoug Ambrisko SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 405665484d8SDoug Ambrisko OID_AUTO, "disable_ocr", CTLFLAG_RW, &sc->disableOnlineCtrlReset, 0, 406665484d8SDoug Ambrisko "Disable the use of OCR"); 407665484d8SDoug Ambrisko 408665484d8SDoug Ambrisko SYSCTL_ADD_STRING(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 409665484d8SDoug Ambrisko OID_AUTO, "driver_version", CTLFLAG_RD, MRSAS_VERSION, 410665484d8SDoug Ambrisko strlen(MRSAS_VERSION), "driver version"); 411665484d8SDoug Ambrisko 412665484d8SDoug Ambrisko SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 413665484d8SDoug Ambrisko OID_AUTO, "reset_count", CTLFLAG_RD, 414665484d8SDoug Ambrisko &sc->reset_count, 0, "number of ocr from start of the day"); 415665484d8SDoug Ambrisko 416665484d8SDoug Ambrisko SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 417665484d8SDoug Ambrisko OID_AUTO, "fw_outstanding", CTLFLAG_RD, 418f0188618SHans Petter Selasky &sc->fw_outstanding.val_rdonly, 0, "FW outstanding commands"); 419665484d8SDoug Ambrisko 420665484d8SDoug Ambrisko SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 421665484d8SDoug Ambrisko OID_AUTO, "io_cmds_highwater", CTLFLAG_RD, 422665484d8SDoug Ambrisko &sc->io_cmds_highwater, 0, "Max FW outstanding commands"); 423665484d8SDoug Ambrisko 424665484d8SDoug Ambrisko SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 425665484d8SDoug Ambrisko OID_AUTO, "mrsas_debug", CTLFLAG_RW, &sc->mrsas_debug, 0, 426665484d8SDoug Ambrisko "Driver debug level"); 427665484d8SDoug Ambrisko 428665484d8SDoug Ambrisko SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 429665484d8SDoug Ambrisko OID_AUTO, "mrsas_io_timeout", CTLFLAG_RW, &sc->mrsas_io_timeout, 430665484d8SDoug Ambrisko 0, "Driver IO timeout value in mili-second."); 431665484d8SDoug Ambrisko 432665484d8SDoug Ambrisko SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 433665484d8SDoug Ambrisko OID_AUTO, "mrsas_fw_fault_check_delay", CTLFLAG_RW, 434665484d8SDoug Ambrisko &sc->mrsas_fw_fault_check_delay, 435665484d8SDoug Ambrisko 0, "FW fault check thread delay in seconds. <default is 1 sec>"); 436665484d8SDoug Ambrisko 437665484d8SDoug Ambrisko SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 438665484d8SDoug Ambrisko OID_AUTO, "reset_in_progress", CTLFLAG_RD, 439665484d8SDoug Ambrisko &sc->reset_in_progress, 0, "ocr in progress status"); 440665484d8SDoug Ambrisko 441d993dd83SKashyap D Desai SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 442d993dd83SKashyap D Desai OID_AUTO, "block_sync_cache", CTLFLAG_RW, 443d993dd83SKashyap D Desai &sc->block_sync_cache, 0, 444d993dd83SKashyap D Desai "Block SYNC CACHE at driver. <default: 0, send it to FW>"); 445d993dd83SKashyap D Desai 446665484d8SDoug Ambrisko } 447665484d8SDoug Ambrisko 4488e727371SKashyap D Desai /* 449665484d8SDoug Ambrisko * mrsas_get_tunables: get tunable parameters. 450665484d8SDoug Ambrisko * input: Adapter instance soft state 451665484d8SDoug Ambrisko * 452665484d8SDoug Ambrisko * Get tunable parameters. This will help to debug driver at boot time. 453665484d8SDoug Ambrisko */ 454665484d8SDoug Ambrisko static void 455665484d8SDoug Ambrisko mrsas_get_tunables(struct mrsas_softc *sc) 456665484d8SDoug Ambrisko { 457665484d8SDoug Ambrisko char tmpstr[80]; 458665484d8SDoug Ambrisko 459665484d8SDoug Ambrisko /* XXX default to some debugging for now */ 460665484d8SDoug Ambrisko sc->mrsas_debug = MRSAS_FAULT; 461665484d8SDoug Ambrisko sc->mrsas_io_timeout = MRSAS_IO_TIMEOUT; 462665484d8SDoug Ambrisko sc->mrsas_fw_fault_check_delay = 1; 463665484d8SDoug Ambrisko sc->reset_count = 0; 464665484d8SDoug Ambrisko sc->reset_in_progress = 0; 465d993dd83SKashyap D Desai sc->block_sync_cache = 0; 466665484d8SDoug Ambrisko 467665484d8SDoug Ambrisko /* 468665484d8SDoug Ambrisko * Grab the global variables. 469665484d8SDoug Ambrisko */ 470665484d8SDoug Ambrisko TUNABLE_INT_FETCH("hw.mrsas.debug_level", &sc->mrsas_debug); 471665484d8SDoug Ambrisko 47216dc2814SKashyap D Desai /* 47316dc2814SKashyap D Desai * Grab the global variables. 47416dc2814SKashyap D Desai */ 47516dc2814SKashyap D Desai TUNABLE_INT_FETCH("hw.mrsas.lb_pending_cmds", &sc->lb_pending_cmds); 47616dc2814SKashyap D Desai 477665484d8SDoug Ambrisko /* Grab the unit-instance variables */ 478665484d8SDoug Ambrisko snprintf(tmpstr, sizeof(tmpstr), "dev.mrsas.%d.debug_level", 479665484d8SDoug Ambrisko device_get_unit(sc->mrsas_dev)); 480665484d8SDoug Ambrisko TUNABLE_INT_FETCH(tmpstr, &sc->mrsas_debug); 481665484d8SDoug Ambrisko } 482665484d8SDoug Ambrisko 4838e727371SKashyap D Desai /* 484665484d8SDoug Ambrisko * mrsas_alloc_evt_log_info cmd: Allocates memory to get event log information. 485665484d8SDoug Ambrisko * Used to get sequence number at driver load time. 486665484d8SDoug Ambrisko * input: Adapter soft state 487665484d8SDoug Ambrisko * 488665484d8SDoug Ambrisko * Allocates DMAable memory for the event log info internal command. 489665484d8SDoug Ambrisko */ 4908e727371SKashyap D Desai int 4918e727371SKashyap D Desai mrsas_alloc_evt_log_info_cmd(struct mrsas_softc *sc) 492665484d8SDoug Ambrisko { 493665484d8SDoug Ambrisko int el_info_size; 494665484d8SDoug Ambrisko 495665484d8SDoug Ambrisko /* Allocate get event log info command */ 496665484d8SDoug Ambrisko el_info_size = sizeof(struct mrsas_evt_log_info); 4978e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 4988e727371SKashyap D Desai 1, 0, 4998e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 5008e727371SKashyap D Desai BUS_SPACE_MAXADDR, 5018e727371SKashyap D Desai NULL, NULL, 5028e727371SKashyap D Desai el_info_size, 5038e727371SKashyap D Desai 1, 5048e727371SKashyap D Desai el_info_size, 5058e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 5068e727371SKashyap D Desai NULL, NULL, 507665484d8SDoug Ambrisko &sc->el_info_tag)) { 508665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate event log info tag\n"); 509665484d8SDoug Ambrisko return (ENOMEM); 510665484d8SDoug Ambrisko } 511665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->el_info_tag, (void **)&sc->el_info_mem, 512665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->el_info_dmamap)) { 513665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate event log info cmd mem\n"); 514665484d8SDoug Ambrisko return (ENOMEM); 515665484d8SDoug Ambrisko } 516665484d8SDoug Ambrisko if (bus_dmamap_load(sc->el_info_tag, sc->el_info_dmamap, 517665484d8SDoug Ambrisko sc->el_info_mem, el_info_size, mrsas_addr_cb, 518665484d8SDoug Ambrisko &sc->el_info_phys_addr, BUS_DMA_NOWAIT)) { 519665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load event log info cmd mem\n"); 520665484d8SDoug Ambrisko return (ENOMEM); 521665484d8SDoug Ambrisko } 522665484d8SDoug Ambrisko memset(sc->el_info_mem, 0, el_info_size); 523665484d8SDoug Ambrisko return (0); 524665484d8SDoug Ambrisko } 525665484d8SDoug Ambrisko 5268e727371SKashyap D Desai /* 527665484d8SDoug Ambrisko * mrsas_free_evt_info_cmd: Free memory for Event log info command 528665484d8SDoug Ambrisko * input: Adapter soft state 529665484d8SDoug Ambrisko * 530665484d8SDoug Ambrisko * Deallocates memory for the event log info internal command. 531665484d8SDoug Ambrisko */ 5328e727371SKashyap D Desai void 5338e727371SKashyap D Desai mrsas_free_evt_log_info_cmd(struct mrsas_softc *sc) 534665484d8SDoug Ambrisko { 535665484d8SDoug Ambrisko if (sc->el_info_phys_addr) 536665484d8SDoug Ambrisko bus_dmamap_unload(sc->el_info_tag, sc->el_info_dmamap); 537665484d8SDoug Ambrisko if (sc->el_info_mem != NULL) 538665484d8SDoug Ambrisko bus_dmamem_free(sc->el_info_tag, sc->el_info_mem, sc->el_info_dmamap); 539665484d8SDoug Ambrisko if (sc->el_info_tag != NULL) 540665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->el_info_tag); 541665484d8SDoug Ambrisko } 542665484d8SDoug Ambrisko 5438e727371SKashyap D Desai /* 544665484d8SDoug Ambrisko * mrsas_get_seq_num: Get latest event sequence number 545665484d8SDoug Ambrisko * @sc: Adapter soft state 546665484d8SDoug Ambrisko * @eli: Firmware event log sequence number information. 5478e727371SKashyap D Desai * 548665484d8SDoug Ambrisko * Firmware maintains a log of all events in a non-volatile area. 549665484d8SDoug Ambrisko * Driver get the sequence number using DCMD 550665484d8SDoug Ambrisko * "MR_DCMD_CTRL_EVENT_GET_INFO" at driver load time. 551665484d8SDoug Ambrisko */ 552665484d8SDoug Ambrisko 553665484d8SDoug Ambrisko static int 554665484d8SDoug Ambrisko mrsas_get_seq_num(struct mrsas_softc *sc, 555665484d8SDoug Ambrisko struct mrsas_evt_log_info *eli) 556665484d8SDoug Ambrisko { 557665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 558665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 559f0c7594bSKashyap D Desai u_int8_t do_ocr = 1, retcode = 0; 560665484d8SDoug Ambrisko 561665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 562665484d8SDoug Ambrisko 563665484d8SDoug Ambrisko if (!cmd) { 564665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Failed to get a free cmd\n"); 565665484d8SDoug Ambrisko return -ENOMEM; 566665484d8SDoug Ambrisko } 567665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 568665484d8SDoug Ambrisko 569665484d8SDoug Ambrisko if (mrsas_alloc_evt_log_info_cmd(sc) != SUCCESS) { 570665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate evt log info cmd\n"); 571665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 572665484d8SDoug Ambrisko return -ENOMEM; 573665484d8SDoug Ambrisko } 574665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 575665484d8SDoug Ambrisko 576665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 577665484d8SDoug Ambrisko dcmd->cmd_status = 0x0; 578665484d8SDoug Ambrisko dcmd->sge_count = 1; 579665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 580665484d8SDoug Ambrisko dcmd->timeout = 0; 581665484d8SDoug Ambrisko dcmd->pad_0 = 0; 582665484d8SDoug Ambrisko dcmd->data_xfer_len = sizeof(struct mrsas_evt_log_info); 583665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_CTRL_EVENT_GET_INFO; 584665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = sc->el_info_phys_addr; 585665484d8SDoug Ambrisko dcmd->sgl.sge32[0].length = sizeof(struct mrsas_evt_log_info); 586665484d8SDoug Ambrisko 587f0c7594bSKashyap D Desai retcode = mrsas_issue_blocked_cmd(sc, cmd); 588f0c7594bSKashyap D Desai if (retcode == ETIMEDOUT) 589f0c7594bSKashyap D Desai goto dcmd_timeout; 590665484d8SDoug Ambrisko 591f0c7594bSKashyap D Desai do_ocr = 0; 592665484d8SDoug Ambrisko /* 593665484d8SDoug Ambrisko * Copy the data back into callers buffer 594665484d8SDoug Ambrisko */ 595665484d8SDoug Ambrisko memcpy(eli, sc->el_info_mem, sizeof(struct mrsas_evt_log_info)); 596665484d8SDoug Ambrisko mrsas_free_evt_log_info_cmd(sc); 597f0c7594bSKashyap D Desai 598f0c7594bSKashyap D Desai dcmd_timeout: 599f0c7594bSKashyap D Desai if (do_ocr) 600f0c7594bSKashyap D Desai sc->do_timedout_reset = MFI_DCMD_TIMEOUT_OCR; 601f0c7594bSKashyap D Desai else 602665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 603665484d8SDoug Ambrisko 604f0c7594bSKashyap D Desai return retcode; 605665484d8SDoug Ambrisko } 606665484d8SDoug Ambrisko 607665484d8SDoug Ambrisko 6088e727371SKashyap D Desai /* 609665484d8SDoug Ambrisko * mrsas_register_aen: Register for asynchronous event notification 610665484d8SDoug Ambrisko * @sc: Adapter soft state 611665484d8SDoug Ambrisko * @seq_num: Starting sequence number 612665484d8SDoug Ambrisko * @class_locale: Class of the event 6138e727371SKashyap D Desai * 614665484d8SDoug Ambrisko * This function subscribes for events beyond the @seq_num 615665484d8SDoug Ambrisko * and type @class_locale. 616665484d8SDoug Ambrisko * 6178e727371SKashyap D Desai */ 618665484d8SDoug Ambrisko static int 619665484d8SDoug Ambrisko mrsas_register_aen(struct mrsas_softc *sc, u_int32_t seq_num, 620665484d8SDoug Ambrisko u_int32_t class_locale_word) 621665484d8SDoug Ambrisko { 622665484d8SDoug Ambrisko int ret_val; 623665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 624665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 625665484d8SDoug Ambrisko union mrsas_evt_class_locale curr_aen; 626665484d8SDoug Ambrisko union mrsas_evt_class_locale prev_aen; 627665484d8SDoug Ambrisko 628665484d8SDoug Ambrisko /* 629665484d8SDoug Ambrisko * If there an AEN pending already (aen_cmd), check if the 6308e727371SKashyap D Desai * class_locale of that pending AEN is inclusive of the new AEN 6318e727371SKashyap D Desai * request we currently have. If it is, then we don't have to do 6328e727371SKashyap D Desai * anything. In other words, whichever events the current AEN request 6338e727371SKashyap D Desai * is subscribing to, have already been subscribed to. If the old_cmd 6348e727371SKashyap D Desai * is _not_ inclusive, then we have to abort that command, form a 6358e727371SKashyap D Desai * class_locale that is superset of both old and current and re-issue 6368e727371SKashyap D Desai * to the FW 6378e727371SKashyap D Desai */ 638665484d8SDoug Ambrisko 639665484d8SDoug Ambrisko curr_aen.word = class_locale_word; 640665484d8SDoug Ambrisko 641665484d8SDoug Ambrisko if (sc->aen_cmd) { 642665484d8SDoug Ambrisko 643665484d8SDoug Ambrisko prev_aen.word = sc->aen_cmd->frame->dcmd.mbox.w[1]; 644665484d8SDoug Ambrisko 645665484d8SDoug Ambrisko /* 646665484d8SDoug Ambrisko * A class whose enum value is smaller is inclusive of all 647665484d8SDoug Ambrisko * higher values. If a PROGRESS (= -1) was previously 648665484d8SDoug Ambrisko * registered, then a new registration requests for higher 649665484d8SDoug Ambrisko * classes need not be sent to FW. They are automatically 6508e727371SKashyap D Desai * included. Locale numbers don't have such hierarchy. They 6518e727371SKashyap D Desai * are bitmap values 652665484d8SDoug Ambrisko */ 653665484d8SDoug Ambrisko if ((prev_aen.members.class <= curr_aen.members.class) && 654665484d8SDoug Ambrisko !((prev_aen.members.locale & curr_aen.members.locale) ^ 655665484d8SDoug Ambrisko curr_aen.members.locale)) { 656665484d8SDoug Ambrisko /* 657665484d8SDoug Ambrisko * Previously issued event registration includes 658665484d8SDoug Ambrisko * current request. Nothing to do. 659665484d8SDoug Ambrisko */ 660665484d8SDoug Ambrisko return 0; 661665484d8SDoug Ambrisko } else { 662665484d8SDoug Ambrisko curr_aen.members.locale |= prev_aen.members.locale; 663665484d8SDoug Ambrisko 664665484d8SDoug Ambrisko if (prev_aen.members.class < curr_aen.members.class) 665665484d8SDoug Ambrisko curr_aen.members.class = prev_aen.members.class; 666665484d8SDoug Ambrisko 667665484d8SDoug Ambrisko sc->aen_cmd->abort_aen = 1; 668665484d8SDoug Ambrisko ret_val = mrsas_issue_blocked_abort_cmd(sc, 669665484d8SDoug Ambrisko sc->aen_cmd); 670665484d8SDoug Ambrisko 671665484d8SDoug Ambrisko if (ret_val) { 672731b7561SKashyap D Desai printf("mrsas: Failed to abort previous AEN command\n"); 673665484d8SDoug Ambrisko return ret_val; 674c2a20ff9SKashyap D Desai } else 675c2a20ff9SKashyap D Desai sc->aen_cmd = NULL; 676665484d8SDoug Ambrisko } 677665484d8SDoug Ambrisko } 678665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 679665484d8SDoug Ambrisko if (!cmd) 680731b7561SKashyap D Desai return ENOMEM; 681665484d8SDoug Ambrisko 682665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 683665484d8SDoug Ambrisko 684665484d8SDoug Ambrisko memset(sc->evt_detail_mem, 0, sizeof(struct mrsas_evt_detail)); 685665484d8SDoug Ambrisko 686665484d8SDoug Ambrisko /* 687665484d8SDoug Ambrisko * Prepare DCMD for aen registration 688665484d8SDoug Ambrisko */ 689665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 690665484d8SDoug Ambrisko 691665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 692665484d8SDoug Ambrisko dcmd->cmd_status = 0x0; 693665484d8SDoug Ambrisko dcmd->sge_count = 1; 694665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 695665484d8SDoug Ambrisko dcmd->timeout = 0; 696665484d8SDoug Ambrisko dcmd->pad_0 = 0; 697665484d8SDoug Ambrisko dcmd->data_xfer_len = sizeof(struct mrsas_evt_detail); 698665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_CTRL_EVENT_WAIT; 699665484d8SDoug Ambrisko dcmd->mbox.w[0] = seq_num; 700665484d8SDoug Ambrisko sc->last_seq_num = seq_num; 701665484d8SDoug Ambrisko dcmd->mbox.w[1] = curr_aen.word; 702665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = (u_int32_t)sc->evt_detail_phys_addr; 703665484d8SDoug Ambrisko dcmd->sgl.sge32[0].length = sizeof(struct mrsas_evt_detail); 704665484d8SDoug Ambrisko 705665484d8SDoug Ambrisko if (sc->aen_cmd != NULL) { 706665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 707665484d8SDoug Ambrisko return 0; 708665484d8SDoug Ambrisko } 709665484d8SDoug Ambrisko /* 710665484d8SDoug Ambrisko * Store reference to the cmd used to register for AEN. When an 711665484d8SDoug Ambrisko * application wants us to register for AEN, we have to abort this 712665484d8SDoug Ambrisko * cmd and re-register with a new EVENT LOCALE supplied by that app 713665484d8SDoug Ambrisko */ 714665484d8SDoug Ambrisko sc->aen_cmd = cmd; 715665484d8SDoug Ambrisko 716665484d8SDoug Ambrisko /* 7178e727371SKashyap D Desai * Issue the aen registration frame 718665484d8SDoug Ambrisko */ 719665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)) { 720665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot issue AEN DCMD command.\n"); 721665484d8SDoug Ambrisko return (1); 722665484d8SDoug Ambrisko } 723665484d8SDoug Ambrisko return 0; 724665484d8SDoug Ambrisko } 7258e727371SKashyap D Desai 7268e727371SKashyap D Desai /* 7278e727371SKashyap D Desai * mrsas_start_aen: Subscribes to AEN during driver load time 728665484d8SDoug Ambrisko * @instance: Adapter soft state 729665484d8SDoug Ambrisko */ 7308e727371SKashyap D Desai static int 7318e727371SKashyap D Desai mrsas_start_aen(struct mrsas_softc *sc) 732665484d8SDoug Ambrisko { 733665484d8SDoug Ambrisko struct mrsas_evt_log_info eli; 734665484d8SDoug Ambrisko union mrsas_evt_class_locale class_locale; 735665484d8SDoug Ambrisko 736665484d8SDoug Ambrisko 737665484d8SDoug Ambrisko /* Get the latest sequence number from FW */ 738665484d8SDoug Ambrisko 739665484d8SDoug Ambrisko memset(&eli, 0, sizeof(eli)); 740665484d8SDoug Ambrisko 741665484d8SDoug Ambrisko if (mrsas_get_seq_num(sc, &eli)) 742665484d8SDoug Ambrisko return -1; 743665484d8SDoug Ambrisko 744665484d8SDoug Ambrisko /* Register AEN with FW for latest sequence number plus 1 */ 745665484d8SDoug Ambrisko class_locale.members.reserved = 0; 746665484d8SDoug Ambrisko class_locale.members.locale = MR_EVT_LOCALE_ALL; 747665484d8SDoug Ambrisko class_locale.members.class = MR_EVT_CLASS_DEBUG; 748665484d8SDoug Ambrisko 749665484d8SDoug Ambrisko return mrsas_register_aen(sc, eli.newest_seq_num + 1, 750665484d8SDoug Ambrisko class_locale.word); 751d18d1b47SKashyap D Desai 752665484d8SDoug Ambrisko } 753665484d8SDoug Ambrisko 7548e727371SKashyap D Desai /* 755d18d1b47SKashyap D Desai * mrsas_setup_msix: Allocate MSI-x vectors 7568e727371SKashyap D Desai * @sc: adapter soft state 757d18d1b47SKashyap D Desai */ 7588e727371SKashyap D Desai static int 7598e727371SKashyap D Desai mrsas_setup_msix(struct mrsas_softc *sc) 760d18d1b47SKashyap D Desai { 761d18d1b47SKashyap D Desai int i; 7628e727371SKashyap D Desai 763d18d1b47SKashyap D Desai for (i = 0; i < sc->msix_vectors; i++) { 764d18d1b47SKashyap D Desai sc->irq_context[i].sc = sc; 765d18d1b47SKashyap D Desai sc->irq_context[i].MSIxIndex = i; 766d18d1b47SKashyap D Desai sc->irq_id[i] = i + 1; 767d18d1b47SKashyap D Desai sc->mrsas_irq[i] = bus_alloc_resource_any 768d18d1b47SKashyap D Desai (sc->mrsas_dev, SYS_RES_IRQ, &sc->irq_id[i] 769d18d1b47SKashyap D Desai ,RF_ACTIVE); 770d18d1b47SKashyap D Desai if (sc->mrsas_irq[i] == NULL) { 771d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "Can't allocate MSI-x\n"); 772d18d1b47SKashyap D Desai goto irq_alloc_failed; 773d18d1b47SKashyap D Desai } 774d18d1b47SKashyap D Desai if (bus_setup_intr(sc->mrsas_dev, 775d18d1b47SKashyap D Desai sc->mrsas_irq[i], 776d18d1b47SKashyap D Desai INTR_MPSAFE | INTR_TYPE_CAM, 777d18d1b47SKashyap D Desai NULL, mrsas_isr, &sc->irq_context[i], 778d18d1b47SKashyap D Desai &sc->intr_handle[i])) { 779d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, 780d18d1b47SKashyap D Desai "Cannot set up MSI-x interrupt handler\n"); 781d18d1b47SKashyap D Desai goto irq_alloc_failed; 782d18d1b47SKashyap D Desai } 783d18d1b47SKashyap D Desai } 784d18d1b47SKashyap D Desai return SUCCESS; 785d18d1b47SKashyap D Desai 786d18d1b47SKashyap D Desai irq_alloc_failed: 787d18d1b47SKashyap D Desai mrsas_teardown_intr(sc); 788d18d1b47SKashyap D Desai return (FAIL); 789d18d1b47SKashyap D Desai } 790d18d1b47SKashyap D Desai 7918e727371SKashyap D Desai /* 792d18d1b47SKashyap D Desai * mrsas_allocate_msix: Setup MSI-x vectors 7938e727371SKashyap D Desai * @sc: adapter soft state 794d18d1b47SKashyap D Desai */ 7958e727371SKashyap D Desai static int 7968e727371SKashyap D Desai mrsas_allocate_msix(struct mrsas_softc *sc) 797d18d1b47SKashyap D Desai { 798d18d1b47SKashyap D Desai if (pci_alloc_msix(sc->mrsas_dev, &sc->msix_vectors) == 0) { 799d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "Using MSI-X with %d number" 800d18d1b47SKashyap D Desai " of vectors\n", sc->msix_vectors); 801d18d1b47SKashyap D Desai } else { 802d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "MSI-x setup failed\n"); 803d18d1b47SKashyap D Desai goto irq_alloc_failed; 804d18d1b47SKashyap D Desai } 805d18d1b47SKashyap D Desai return SUCCESS; 806d18d1b47SKashyap D Desai 807d18d1b47SKashyap D Desai irq_alloc_failed: 808d18d1b47SKashyap D Desai mrsas_teardown_intr(sc); 809d18d1b47SKashyap D Desai return (FAIL); 810d18d1b47SKashyap D Desai } 8118e727371SKashyap D Desai 8128e727371SKashyap D Desai /* 813665484d8SDoug Ambrisko * mrsas_attach: PCI entry point 8148e727371SKashyap D Desai * input: pointer to device struct 815665484d8SDoug Ambrisko * 8168e727371SKashyap D Desai * Performs setup of PCI and registers, initializes mutexes and linked lists, 8178e727371SKashyap D Desai * registers interrupts and CAM, and initializes the adapter/controller to 8188e727371SKashyap D Desai * its proper state. 819665484d8SDoug Ambrisko */ 8208e727371SKashyap D Desai static int 8218e727371SKashyap D Desai mrsas_attach(device_t dev) 822665484d8SDoug Ambrisko { 823665484d8SDoug Ambrisko struct mrsas_softc *sc = device_get_softc(dev); 8247aade8bfSKashyap D Desai uint32_t cmd, error; 825665484d8SDoug Ambrisko 8264bb0a4f0SKashyap D Desai memset(sc, 0, sizeof(struct mrsas_softc)); 8274bb0a4f0SKashyap D Desai 828665484d8SDoug Ambrisko /* Look up our softc and initialize its fields. */ 829665484d8SDoug Ambrisko sc->mrsas_dev = dev; 830665484d8SDoug Ambrisko sc->device_id = pci_get_device(dev); 831665484d8SDoug Ambrisko 832f9c63081SKashyap D Desai if ((sc->device_id == MRSAS_INVADER) || 833f9c63081SKashyap D Desai (sc->device_id == MRSAS_FURY) || 834f9c63081SKashyap D Desai (sc->device_id == MRSAS_INTRUDER) || 835f9c63081SKashyap D Desai (sc->device_id == MRSAS_INTRUDER_24) || 836f9c63081SKashyap D Desai (sc->device_id == MRSAS_CUTLASS_52) || 837f9c63081SKashyap D Desai (sc->device_id == MRSAS_CUTLASS_53)) { 838f9c63081SKashyap D Desai sc->mrsas_gen3_ctrl = 1; 8397aade8bfSKashyap D Desai } else if ((sc->device_id == MRSAS_VENTURA) || 8407aade8bfSKashyap D Desai (sc->device_id == MRSAS_CRUSADER) || 8417aade8bfSKashyap D Desai (sc->device_id == MRSAS_HARPOON) || 8427aade8bfSKashyap D Desai (sc->device_id == MRSAS_TOMCAT) || 8437aade8bfSKashyap D Desai (sc->device_id == MRSAS_VENTURA_4PORT) || 8447aade8bfSKashyap D Desai (sc->device_id == MRSAS_CRUSADER_4PORT)) { 8457aade8bfSKashyap D Desai sc->is_ventura = true; 846f9c63081SKashyap D Desai } 847f9c63081SKashyap D Desai 848665484d8SDoug Ambrisko mrsas_get_tunables(sc); 849665484d8SDoug Ambrisko 850665484d8SDoug Ambrisko /* 851665484d8SDoug Ambrisko * Set up PCI and registers 852665484d8SDoug Ambrisko */ 853665484d8SDoug Ambrisko cmd = pci_read_config(dev, PCIR_COMMAND, 2); 854665484d8SDoug Ambrisko if ((cmd & PCIM_CMD_PORTEN) == 0) { 855665484d8SDoug Ambrisko return (ENXIO); 856665484d8SDoug Ambrisko } 857665484d8SDoug Ambrisko /* Force the busmaster enable bit on. */ 858665484d8SDoug Ambrisko cmd |= PCIM_CMD_BUSMASTEREN; 859665484d8SDoug Ambrisko pci_write_config(dev, PCIR_COMMAND, cmd, 2); 860665484d8SDoug Ambrisko 8617aade8bfSKashyap D Desai /* For Ventura system registers are mapped to BAR0 */ 8627aade8bfSKashyap D Desai if (sc->is_ventura) 8637aade8bfSKashyap D Desai sc->reg_res_id = PCIR_BAR(0); /* BAR0 offset */ 8647aade8bfSKashyap D Desai else 8657aade8bfSKashyap D Desai sc->reg_res_id = PCIR_BAR(1); /* BAR1 offset */ 866665484d8SDoug Ambrisko 86743cd6160SJustin Hibbits if ((sc->reg_res = bus_alloc_resource_any(dev, SYS_RES_MEMORY, 86843cd6160SJustin Hibbits &(sc->reg_res_id), RF_ACTIVE)) 869665484d8SDoug Ambrisko == NULL) { 870665484d8SDoug Ambrisko device_printf(dev, "Cannot allocate PCI registers\n"); 871665484d8SDoug Ambrisko goto attach_fail; 872665484d8SDoug Ambrisko } 873665484d8SDoug Ambrisko sc->bus_tag = rman_get_bustag(sc->reg_res); 874665484d8SDoug Ambrisko sc->bus_handle = rman_get_bushandle(sc->reg_res); 875665484d8SDoug Ambrisko 876665484d8SDoug Ambrisko /* Intialize mutexes */ 877665484d8SDoug Ambrisko mtx_init(&sc->sim_lock, "mrsas_sim_lock", NULL, MTX_DEF); 878665484d8SDoug Ambrisko mtx_init(&sc->pci_lock, "mrsas_pci_lock", NULL, MTX_DEF); 879665484d8SDoug Ambrisko mtx_init(&sc->io_lock, "mrsas_io_lock", NULL, MTX_DEF); 880665484d8SDoug Ambrisko mtx_init(&sc->aen_lock, "mrsas_aen_lock", NULL, MTX_DEF); 881665484d8SDoug Ambrisko mtx_init(&sc->ioctl_lock, "mrsas_ioctl_lock", NULL, MTX_SPIN); 882665484d8SDoug Ambrisko mtx_init(&sc->mpt_cmd_pool_lock, "mrsas_mpt_cmd_pool_lock", NULL, MTX_DEF); 883665484d8SDoug Ambrisko mtx_init(&sc->mfi_cmd_pool_lock, "mrsas_mfi_cmd_pool_lock", NULL, MTX_DEF); 884665484d8SDoug Ambrisko mtx_init(&sc->raidmap_lock, "mrsas_raidmap_lock", NULL, MTX_DEF); 885665484d8SDoug Ambrisko 886665484d8SDoug Ambrisko /* Intialize linked list */ 887665484d8SDoug Ambrisko TAILQ_INIT(&sc->mrsas_mpt_cmd_list_head); 888665484d8SDoug Ambrisko TAILQ_INIT(&sc->mrsas_mfi_cmd_list_head); 889665484d8SDoug Ambrisko 890f5fb2237SKashyap D Desai mrsas_atomic_set(&sc->fw_outstanding, 0); 8918bb601acSKashyap D Desai mrsas_atomic_set(&sc->target_reset_outstanding, 0); 892665484d8SDoug Ambrisko 893665484d8SDoug Ambrisko sc->io_cmds_highwater = 0; 894665484d8SDoug Ambrisko 895665484d8SDoug Ambrisko sc->adprecovery = MRSAS_HBA_OPERATIONAL; 896665484d8SDoug Ambrisko sc->UnevenSpanSupport = 0; 897665484d8SDoug Ambrisko 898d18d1b47SKashyap D Desai sc->msix_enable = 0; 899d18d1b47SKashyap D Desai 900665484d8SDoug Ambrisko /* Initialize Firmware */ 901665484d8SDoug Ambrisko if (mrsas_init_fw(sc) != SUCCESS) { 902665484d8SDoug Ambrisko goto attach_fail_fw; 903665484d8SDoug Ambrisko } 9048071588dSKashyap D Desai /* Register mrsas to CAM layer */ 905665484d8SDoug Ambrisko if ((mrsas_cam_attach(sc) != SUCCESS)) { 906665484d8SDoug Ambrisko goto attach_fail_cam; 907665484d8SDoug Ambrisko } 908665484d8SDoug Ambrisko /* Register IRQs */ 909665484d8SDoug Ambrisko if (mrsas_setup_irq(sc) != SUCCESS) { 910665484d8SDoug Ambrisko goto attach_fail_irq; 911665484d8SDoug Ambrisko } 912665484d8SDoug Ambrisko error = mrsas_kproc_create(mrsas_ocr_thread, sc, 913665484d8SDoug Ambrisko &sc->ocr_thread, 0, 0, "mrsas_ocr%d", 914665484d8SDoug Ambrisko device_get_unit(sc->mrsas_dev)); 915665484d8SDoug Ambrisko if (error) { 9168071588dSKashyap D Desai device_printf(sc->mrsas_dev, "Error %d starting OCR thread\n", error); 9178071588dSKashyap D Desai goto attach_fail_ocr_thread; 918665484d8SDoug Ambrisko } 919536094dcSKashyap D Desai /* 9208071588dSKashyap D Desai * After FW initialization and OCR thread creation 9218071588dSKashyap D Desai * we will defer the cdev creation, AEN setup on ICH callback 922536094dcSKashyap D Desai */ 9238071588dSKashyap D Desai sc->mrsas_ich.ich_func = mrsas_ich_startup; 9248071588dSKashyap D Desai sc->mrsas_ich.ich_arg = sc; 9258071588dSKashyap D Desai if (config_intrhook_establish(&sc->mrsas_ich) != 0) { 9268071588dSKashyap D Desai device_printf(sc->mrsas_dev, "Config hook is already established\n"); 9278071588dSKashyap D Desai } 9288071588dSKashyap D Desai mrsas_setup_sysctl(sc); 9298071588dSKashyap D Desai return SUCCESS; 930536094dcSKashyap D Desai 9318071588dSKashyap D Desai attach_fail_ocr_thread: 9328071588dSKashyap D Desai if (sc->ocr_thread_active) 9338071588dSKashyap D Desai wakeup(&sc->ocr_chan); 934665484d8SDoug Ambrisko attach_fail_irq: 935665484d8SDoug Ambrisko mrsas_teardown_intr(sc); 936665484d8SDoug Ambrisko attach_fail_cam: 937665484d8SDoug Ambrisko mrsas_cam_detach(sc); 938665484d8SDoug Ambrisko attach_fail_fw: 939d18d1b47SKashyap D Desai /* if MSIX vector is allocated and FW Init FAILED then release MSIX */ 940d18d1b47SKashyap D Desai if (sc->msix_enable == 1) 941d18d1b47SKashyap D Desai pci_release_msi(sc->mrsas_dev); 942665484d8SDoug Ambrisko mrsas_free_mem(sc); 943665484d8SDoug Ambrisko mtx_destroy(&sc->sim_lock); 944665484d8SDoug Ambrisko mtx_destroy(&sc->aen_lock); 945665484d8SDoug Ambrisko mtx_destroy(&sc->pci_lock); 946665484d8SDoug Ambrisko mtx_destroy(&sc->io_lock); 947665484d8SDoug Ambrisko mtx_destroy(&sc->ioctl_lock); 948665484d8SDoug Ambrisko mtx_destroy(&sc->mpt_cmd_pool_lock); 949665484d8SDoug Ambrisko mtx_destroy(&sc->mfi_cmd_pool_lock); 950665484d8SDoug Ambrisko mtx_destroy(&sc->raidmap_lock); 951665484d8SDoug Ambrisko attach_fail: 952665484d8SDoug Ambrisko if (sc->reg_res) { 953665484d8SDoug Ambrisko bus_release_resource(sc->mrsas_dev, SYS_RES_MEMORY, 954665484d8SDoug Ambrisko sc->reg_res_id, sc->reg_res); 955665484d8SDoug Ambrisko } 956665484d8SDoug Ambrisko return (ENXIO); 957665484d8SDoug Ambrisko } 958665484d8SDoug Ambrisko 9598e727371SKashyap D Desai /* 9608071588dSKashyap D Desai * Interrupt config hook 9618071588dSKashyap D Desai */ 9628071588dSKashyap D Desai static void 9638071588dSKashyap D Desai mrsas_ich_startup(void *arg) 9648071588dSKashyap D Desai { 9658071588dSKashyap D Desai struct mrsas_softc *sc = (struct mrsas_softc *)arg; 9668071588dSKashyap D Desai 9678071588dSKashyap D Desai /* 9688071588dSKashyap D Desai * Intialize a counting Semaphore to take care no. of concurrent IOCTLs 9698071588dSKashyap D Desai */ 970731b7561SKashyap D Desai sema_init(&sc->ioctl_count_sema, MRSAS_MAX_IOCTL_CMDS, 9718071588dSKashyap D Desai IOCTL_SEMA_DESCRIPTION); 9728071588dSKashyap D Desai 9738071588dSKashyap D Desai /* Create a /dev entry for mrsas controller. */ 9748071588dSKashyap D Desai sc->mrsas_cdev = make_dev(&mrsas_cdevsw, device_get_unit(sc->mrsas_dev), UID_ROOT, 9758071588dSKashyap D Desai GID_OPERATOR, (S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP), "mrsas%u", 9768071588dSKashyap D Desai device_get_unit(sc->mrsas_dev)); 9778071588dSKashyap D Desai 9788071588dSKashyap D Desai if (device_get_unit(sc->mrsas_dev) == 0) { 9798071588dSKashyap D Desai make_dev_alias_p(MAKEDEV_CHECKNAME, 9808071588dSKashyap D Desai &sc->mrsas_linux_emulator_cdev, sc->mrsas_cdev, 9818071588dSKashyap D Desai "megaraid_sas_ioctl_node"); 9828071588dSKashyap D Desai } 9838071588dSKashyap D Desai if (sc->mrsas_cdev) 9848071588dSKashyap D Desai sc->mrsas_cdev->si_drv1 = sc; 9858071588dSKashyap D Desai 9868071588dSKashyap D Desai /* 9878071588dSKashyap D Desai * Add this controller to mrsas_mgmt_info structure so that it can be 9888071588dSKashyap D Desai * exported to management applications 9898071588dSKashyap D Desai */ 9908071588dSKashyap D Desai if (device_get_unit(sc->mrsas_dev) == 0) 9918071588dSKashyap D Desai memset(&mrsas_mgmt_info, 0, sizeof(mrsas_mgmt_info)); 9928071588dSKashyap D Desai 9938071588dSKashyap D Desai mrsas_mgmt_info.count++; 9948071588dSKashyap D Desai mrsas_mgmt_info.sc_ptr[mrsas_mgmt_info.max_index] = sc; 9958071588dSKashyap D Desai mrsas_mgmt_info.max_index++; 9968071588dSKashyap D Desai 9978071588dSKashyap D Desai /* Enable Interrupts */ 9988071588dSKashyap D Desai mrsas_enable_intr(sc); 9998071588dSKashyap D Desai 10008071588dSKashyap D Desai /* Initiate AEN (Asynchronous Event Notification) */ 10018071588dSKashyap D Desai if (mrsas_start_aen(sc)) { 10028071588dSKashyap D Desai device_printf(sc->mrsas_dev, "Error: AEN registration FAILED !!! " 10038071588dSKashyap D Desai "Further events from the controller will not be communicated.\n" 10048071588dSKashyap D Desai "Either there is some problem in the controller" 10058071588dSKashyap D Desai "or the controller does not support AEN.\n" 10068071588dSKashyap D Desai "Please contact to the SUPPORT TEAM if the problem persists\n"); 10078071588dSKashyap D Desai } 10088071588dSKashyap D Desai if (sc->mrsas_ich.ich_arg != NULL) { 10098071588dSKashyap D Desai device_printf(sc->mrsas_dev, "Disestablish mrsas intr hook\n"); 10108071588dSKashyap D Desai config_intrhook_disestablish(&sc->mrsas_ich); 10118071588dSKashyap D Desai sc->mrsas_ich.ich_arg = NULL; 10128071588dSKashyap D Desai } 10138071588dSKashyap D Desai } 10148071588dSKashyap D Desai 10158071588dSKashyap D Desai /* 1016665484d8SDoug Ambrisko * mrsas_detach: De-allocates and teardown resources 10178e727371SKashyap D Desai * input: pointer to device struct 1018665484d8SDoug Ambrisko * 10198e727371SKashyap D Desai * This function is the entry point for device disconnect and detach. 10208e727371SKashyap D Desai * It performs memory de-allocations, shutdown of the controller and various 1021665484d8SDoug Ambrisko * teardown and destroy resource functions. 1022665484d8SDoug Ambrisko */ 10238e727371SKashyap D Desai static int 10248e727371SKashyap D Desai mrsas_detach(device_t dev) 1025665484d8SDoug Ambrisko { 1026665484d8SDoug Ambrisko struct mrsas_softc *sc; 1027665484d8SDoug Ambrisko int i = 0; 1028665484d8SDoug Ambrisko 1029665484d8SDoug Ambrisko sc = device_get_softc(dev); 1030665484d8SDoug Ambrisko sc->remove_in_progress = 1; 1031536094dcSKashyap D Desai 1032839ee025SKashyap D Desai /* Destroy the character device so no other IOCTL will be handled */ 10338071588dSKashyap D Desai if ((device_get_unit(dev) == 0) && sc->mrsas_linux_emulator_cdev) 10348071588dSKashyap D Desai destroy_dev(sc->mrsas_linux_emulator_cdev); 1035839ee025SKashyap D Desai destroy_dev(sc->mrsas_cdev); 1036839ee025SKashyap D Desai 1037536094dcSKashyap D Desai /* 1038536094dcSKashyap D Desai * Take the instance off the instance array. Note that we will not 1039536094dcSKashyap D Desai * decrement the max_index. We let this array be sparse array 1040536094dcSKashyap D Desai */ 1041536094dcSKashyap D Desai for (i = 0; i < mrsas_mgmt_info.max_index; i++) { 1042536094dcSKashyap D Desai if (mrsas_mgmt_info.sc_ptr[i] == sc) { 1043536094dcSKashyap D Desai mrsas_mgmt_info.count--; 1044536094dcSKashyap D Desai mrsas_mgmt_info.sc_ptr[i] = NULL; 1045536094dcSKashyap D Desai break; 1046536094dcSKashyap D Desai } 1047536094dcSKashyap D Desai } 1048536094dcSKashyap D Desai 1049665484d8SDoug Ambrisko if (sc->ocr_thread_active) 1050665484d8SDoug Ambrisko wakeup(&sc->ocr_chan); 1051665484d8SDoug Ambrisko while (sc->reset_in_progress) { 1052665484d8SDoug Ambrisko i++; 1053665484d8SDoug Ambrisko if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) { 1054665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_INFO, 1055f0c7594bSKashyap D Desai "[%2d]waiting for OCR to be finished from %s\n", i, __func__); 1056665484d8SDoug Ambrisko } 1057665484d8SDoug Ambrisko pause("mr_shutdown", hz); 1058665484d8SDoug Ambrisko } 1059665484d8SDoug Ambrisko i = 0; 1060665484d8SDoug Ambrisko while (sc->ocr_thread_active) { 1061665484d8SDoug Ambrisko i++; 1062665484d8SDoug Ambrisko if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) { 1063665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_INFO, 1064665484d8SDoug Ambrisko "[%2d]waiting for " 1065665484d8SDoug Ambrisko "mrsas_ocr thread to quit ocr %d\n", i, 1066665484d8SDoug Ambrisko sc->ocr_thread_active); 1067665484d8SDoug Ambrisko } 1068665484d8SDoug Ambrisko pause("mr_shutdown", hz); 1069665484d8SDoug Ambrisko } 1070665484d8SDoug Ambrisko mrsas_flush_cache(sc); 1071665484d8SDoug Ambrisko mrsas_shutdown_ctlr(sc, MR_DCMD_CTRL_SHUTDOWN); 1072665484d8SDoug Ambrisko mrsas_disable_intr(sc); 1073665484d8SDoug Ambrisko mrsas_cam_detach(sc); 1074665484d8SDoug Ambrisko mrsas_teardown_intr(sc); 1075665484d8SDoug Ambrisko mrsas_free_mem(sc); 1076665484d8SDoug Ambrisko mtx_destroy(&sc->sim_lock); 1077665484d8SDoug Ambrisko mtx_destroy(&sc->aen_lock); 1078665484d8SDoug Ambrisko mtx_destroy(&sc->pci_lock); 1079665484d8SDoug Ambrisko mtx_destroy(&sc->io_lock); 1080665484d8SDoug Ambrisko mtx_destroy(&sc->ioctl_lock); 1081665484d8SDoug Ambrisko mtx_destroy(&sc->mpt_cmd_pool_lock); 1082665484d8SDoug Ambrisko mtx_destroy(&sc->mfi_cmd_pool_lock); 1083665484d8SDoug Ambrisko mtx_destroy(&sc->raidmap_lock); 1084839ee025SKashyap D Desai 1085839ee025SKashyap D Desai /* Wait for all the semaphores to be released */ 1086731b7561SKashyap D Desai while (sema_value(&sc->ioctl_count_sema) != MRSAS_MAX_IOCTL_CMDS) 1087839ee025SKashyap D Desai pause("mr_shutdown", hz); 1088839ee025SKashyap D Desai 1089839ee025SKashyap D Desai /* Destroy the counting semaphore created for Ioctl */ 1090839ee025SKashyap D Desai sema_destroy(&sc->ioctl_count_sema); 1091839ee025SKashyap D Desai 1092665484d8SDoug Ambrisko if (sc->reg_res) { 1093665484d8SDoug Ambrisko bus_release_resource(sc->mrsas_dev, 1094665484d8SDoug Ambrisko SYS_RES_MEMORY, sc->reg_res_id, sc->reg_res); 1095665484d8SDoug Ambrisko } 1096665484d8SDoug Ambrisko if (sc->sysctl_tree != NULL) 1097665484d8SDoug Ambrisko sysctl_ctx_free(&sc->sysctl_ctx); 1098839ee025SKashyap D Desai 1099665484d8SDoug Ambrisko return (0); 1100665484d8SDoug Ambrisko } 1101665484d8SDoug Ambrisko 11028e727371SKashyap D Desai /* 1103665484d8SDoug Ambrisko * mrsas_free_mem: Frees allocated memory 1104665484d8SDoug Ambrisko * input: Adapter instance soft state 1105665484d8SDoug Ambrisko * 1106665484d8SDoug Ambrisko * This function is called from mrsas_detach() to free previously allocated 1107665484d8SDoug Ambrisko * memory. 1108665484d8SDoug Ambrisko */ 11098e727371SKashyap D Desai void 11108e727371SKashyap D Desai mrsas_free_mem(struct mrsas_softc *sc) 1111665484d8SDoug Ambrisko { 1112665484d8SDoug Ambrisko int i; 1113665484d8SDoug Ambrisko u_int32_t max_cmd; 1114665484d8SDoug Ambrisko struct mrsas_mfi_cmd *mfi_cmd; 1115665484d8SDoug Ambrisko struct mrsas_mpt_cmd *mpt_cmd; 1116665484d8SDoug Ambrisko 1117665484d8SDoug Ambrisko /* 1118665484d8SDoug Ambrisko * Free RAID map memory 1119665484d8SDoug Ambrisko */ 11208e727371SKashyap D Desai for (i = 0; i < 2; i++) { 1121665484d8SDoug Ambrisko if (sc->raidmap_phys_addr[i]) 1122665484d8SDoug Ambrisko bus_dmamap_unload(sc->raidmap_tag[i], sc->raidmap_dmamap[i]); 1123665484d8SDoug Ambrisko if (sc->raidmap_mem[i] != NULL) 1124665484d8SDoug Ambrisko bus_dmamem_free(sc->raidmap_tag[i], sc->raidmap_mem[i], sc->raidmap_dmamap[i]); 1125665484d8SDoug Ambrisko if (sc->raidmap_tag[i] != NULL) 1126665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->raidmap_tag[i]); 11274799d485SKashyap D Desai 11284799d485SKashyap D Desai if (sc->ld_drv_map[i] != NULL) 11294799d485SKashyap D Desai free(sc->ld_drv_map[i], M_MRSAS); 1130665484d8SDoug Ambrisko } 1131a688fcd0SKashyap D Desai for (i = 0; i < 2; i++) { 1132a688fcd0SKashyap D Desai if (sc->jbodmap_phys_addr[i]) 1133a688fcd0SKashyap D Desai bus_dmamap_unload(sc->jbodmap_tag[i], sc->jbodmap_dmamap[i]); 1134a688fcd0SKashyap D Desai if (sc->jbodmap_mem[i] != NULL) 1135a688fcd0SKashyap D Desai bus_dmamem_free(sc->jbodmap_tag[i], sc->jbodmap_mem[i], sc->jbodmap_dmamap[i]); 1136a688fcd0SKashyap D Desai if (sc->jbodmap_tag[i] != NULL) 1137a688fcd0SKashyap D Desai bus_dma_tag_destroy(sc->jbodmap_tag[i]); 1138a688fcd0SKashyap D Desai } 1139665484d8SDoug Ambrisko /* 1140453130d9SPedro F. Giffuni * Free version buffer memory 1141665484d8SDoug Ambrisko */ 1142665484d8SDoug Ambrisko if (sc->verbuf_phys_addr) 1143665484d8SDoug Ambrisko bus_dmamap_unload(sc->verbuf_tag, sc->verbuf_dmamap); 1144665484d8SDoug Ambrisko if (sc->verbuf_mem != NULL) 1145665484d8SDoug Ambrisko bus_dmamem_free(sc->verbuf_tag, sc->verbuf_mem, sc->verbuf_dmamap); 1146665484d8SDoug Ambrisko if (sc->verbuf_tag != NULL) 1147665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->verbuf_tag); 1148665484d8SDoug Ambrisko 1149665484d8SDoug Ambrisko 1150665484d8SDoug Ambrisko /* 1151665484d8SDoug Ambrisko * Free sense buffer memory 1152665484d8SDoug Ambrisko */ 1153665484d8SDoug Ambrisko if (sc->sense_phys_addr) 1154665484d8SDoug Ambrisko bus_dmamap_unload(sc->sense_tag, sc->sense_dmamap); 1155665484d8SDoug Ambrisko if (sc->sense_mem != NULL) 1156665484d8SDoug Ambrisko bus_dmamem_free(sc->sense_tag, sc->sense_mem, sc->sense_dmamap); 1157665484d8SDoug Ambrisko if (sc->sense_tag != NULL) 1158665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->sense_tag); 1159665484d8SDoug Ambrisko 1160665484d8SDoug Ambrisko /* 1161665484d8SDoug Ambrisko * Free chain frame memory 1162665484d8SDoug Ambrisko */ 1163665484d8SDoug Ambrisko if (sc->chain_frame_phys_addr) 1164665484d8SDoug Ambrisko bus_dmamap_unload(sc->chain_frame_tag, sc->chain_frame_dmamap); 1165665484d8SDoug Ambrisko if (sc->chain_frame_mem != NULL) 1166665484d8SDoug Ambrisko bus_dmamem_free(sc->chain_frame_tag, sc->chain_frame_mem, sc->chain_frame_dmamap); 1167665484d8SDoug Ambrisko if (sc->chain_frame_tag != NULL) 1168665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->chain_frame_tag); 1169665484d8SDoug Ambrisko 1170665484d8SDoug Ambrisko /* 1171665484d8SDoug Ambrisko * Free IO Request memory 1172665484d8SDoug Ambrisko */ 1173665484d8SDoug Ambrisko if (sc->io_request_phys_addr) 1174665484d8SDoug Ambrisko bus_dmamap_unload(sc->io_request_tag, sc->io_request_dmamap); 1175665484d8SDoug Ambrisko if (sc->io_request_mem != NULL) 1176665484d8SDoug Ambrisko bus_dmamem_free(sc->io_request_tag, sc->io_request_mem, sc->io_request_dmamap); 1177665484d8SDoug Ambrisko if (sc->io_request_tag != NULL) 1178665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->io_request_tag); 1179665484d8SDoug Ambrisko 1180665484d8SDoug Ambrisko /* 1181665484d8SDoug Ambrisko * Free Reply Descriptor memory 1182665484d8SDoug Ambrisko */ 1183665484d8SDoug Ambrisko if (sc->reply_desc_phys_addr) 1184665484d8SDoug Ambrisko bus_dmamap_unload(sc->reply_desc_tag, sc->reply_desc_dmamap); 1185665484d8SDoug Ambrisko if (sc->reply_desc_mem != NULL) 1186665484d8SDoug Ambrisko bus_dmamem_free(sc->reply_desc_tag, sc->reply_desc_mem, sc->reply_desc_dmamap); 1187665484d8SDoug Ambrisko if (sc->reply_desc_tag != NULL) 1188665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->reply_desc_tag); 1189665484d8SDoug Ambrisko 1190665484d8SDoug Ambrisko /* 1191665484d8SDoug Ambrisko * Free event detail memory 1192665484d8SDoug Ambrisko */ 1193665484d8SDoug Ambrisko if (sc->evt_detail_phys_addr) 1194665484d8SDoug Ambrisko bus_dmamap_unload(sc->evt_detail_tag, sc->evt_detail_dmamap); 1195665484d8SDoug Ambrisko if (sc->evt_detail_mem != NULL) 1196665484d8SDoug Ambrisko bus_dmamem_free(sc->evt_detail_tag, sc->evt_detail_mem, sc->evt_detail_dmamap); 1197665484d8SDoug Ambrisko if (sc->evt_detail_tag != NULL) 1198665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->evt_detail_tag); 1199665484d8SDoug Ambrisko 1200665484d8SDoug Ambrisko /* 1201665484d8SDoug Ambrisko * Free MFI frames 1202665484d8SDoug Ambrisko */ 1203665484d8SDoug Ambrisko if (sc->mfi_cmd_list) { 1204665484d8SDoug Ambrisko for (i = 0; i < MRSAS_MAX_MFI_CMDS; i++) { 1205665484d8SDoug Ambrisko mfi_cmd = sc->mfi_cmd_list[i]; 1206665484d8SDoug Ambrisko mrsas_free_frame(sc, mfi_cmd); 1207665484d8SDoug Ambrisko } 1208665484d8SDoug Ambrisko } 1209665484d8SDoug Ambrisko if (sc->mficmd_frame_tag != NULL) 1210665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->mficmd_frame_tag); 1211665484d8SDoug Ambrisko 1212665484d8SDoug Ambrisko /* 1213665484d8SDoug Ambrisko * Free MPT internal command list 1214665484d8SDoug Ambrisko */ 1215665484d8SDoug Ambrisko max_cmd = sc->max_fw_cmds; 1216665484d8SDoug Ambrisko if (sc->mpt_cmd_list) { 1217665484d8SDoug Ambrisko for (i = 0; i < max_cmd; i++) { 1218665484d8SDoug Ambrisko mpt_cmd = sc->mpt_cmd_list[i]; 1219665484d8SDoug Ambrisko bus_dmamap_destroy(sc->data_tag, mpt_cmd->data_dmamap); 1220665484d8SDoug Ambrisko free(sc->mpt_cmd_list[i], M_MRSAS); 1221665484d8SDoug Ambrisko } 1222665484d8SDoug Ambrisko free(sc->mpt_cmd_list, M_MRSAS); 1223665484d8SDoug Ambrisko sc->mpt_cmd_list = NULL; 1224665484d8SDoug Ambrisko } 1225665484d8SDoug Ambrisko /* 1226665484d8SDoug Ambrisko * Free MFI internal command list 1227665484d8SDoug Ambrisko */ 1228665484d8SDoug Ambrisko 1229665484d8SDoug Ambrisko if (sc->mfi_cmd_list) { 1230665484d8SDoug Ambrisko for (i = 0; i < MRSAS_MAX_MFI_CMDS; i++) { 1231665484d8SDoug Ambrisko free(sc->mfi_cmd_list[i], M_MRSAS); 1232665484d8SDoug Ambrisko } 1233665484d8SDoug Ambrisko free(sc->mfi_cmd_list, M_MRSAS); 1234665484d8SDoug Ambrisko sc->mfi_cmd_list = NULL; 1235665484d8SDoug Ambrisko } 1236665484d8SDoug Ambrisko /* 1237665484d8SDoug Ambrisko * Free request descriptor memory 1238665484d8SDoug Ambrisko */ 1239665484d8SDoug Ambrisko free(sc->req_desc, M_MRSAS); 1240665484d8SDoug Ambrisko sc->req_desc = NULL; 1241665484d8SDoug Ambrisko 1242665484d8SDoug Ambrisko /* 1243665484d8SDoug Ambrisko * Destroy parent tag 1244665484d8SDoug Ambrisko */ 1245665484d8SDoug Ambrisko if (sc->mrsas_parent_tag != NULL) 1246665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->mrsas_parent_tag); 1247af51c29fSKashyap D Desai 1248af51c29fSKashyap D Desai /* 1249af51c29fSKashyap D Desai * Free ctrl_info memory 1250af51c29fSKashyap D Desai */ 1251af51c29fSKashyap D Desai if (sc->ctrl_info != NULL) 1252af51c29fSKashyap D Desai free(sc->ctrl_info, M_MRSAS); 1253665484d8SDoug Ambrisko } 1254665484d8SDoug Ambrisko 12558e727371SKashyap D Desai /* 1256665484d8SDoug Ambrisko * mrsas_teardown_intr: Teardown interrupt 1257665484d8SDoug Ambrisko * input: Adapter instance soft state 1258665484d8SDoug Ambrisko * 12598e727371SKashyap D Desai * This function is called from mrsas_detach() to teardown and release bus 12608e727371SKashyap D Desai * interrupt resourse. 1261665484d8SDoug Ambrisko */ 12628e727371SKashyap D Desai void 12638e727371SKashyap D Desai mrsas_teardown_intr(struct mrsas_softc *sc) 1264665484d8SDoug Ambrisko { 1265d18d1b47SKashyap D Desai int i; 12668e727371SKashyap D Desai 1267d18d1b47SKashyap D Desai if (!sc->msix_enable) { 1268d18d1b47SKashyap D Desai if (sc->intr_handle[0]) 1269d18d1b47SKashyap D Desai bus_teardown_intr(sc->mrsas_dev, sc->mrsas_irq[0], sc->intr_handle[0]); 1270d18d1b47SKashyap D Desai if (sc->mrsas_irq[0] != NULL) 12718e727371SKashyap D Desai bus_release_resource(sc->mrsas_dev, SYS_RES_IRQ, 12728e727371SKashyap D Desai sc->irq_id[0], sc->mrsas_irq[0]); 1273d18d1b47SKashyap D Desai sc->intr_handle[0] = NULL; 1274d18d1b47SKashyap D Desai } else { 1275d18d1b47SKashyap D Desai for (i = 0; i < sc->msix_vectors; i++) { 1276d18d1b47SKashyap D Desai if (sc->intr_handle[i]) 1277d18d1b47SKashyap D Desai bus_teardown_intr(sc->mrsas_dev, sc->mrsas_irq[i], 1278d18d1b47SKashyap D Desai sc->intr_handle[i]); 1279d18d1b47SKashyap D Desai 1280d18d1b47SKashyap D Desai if (sc->mrsas_irq[i] != NULL) 1281d18d1b47SKashyap D Desai bus_release_resource(sc->mrsas_dev, SYS_RES_IRQ, 1282d18d1b47SKashyap D Desai sc->irq_id[i], sc->mrsas_irq[i]); 1283d18d1b47SKashyap D Desai 1284d18d1b47SKashyap D Desai sc->intr_handle[i] = NULL; 1285d18d1b47SKashyap D Desai } 1286d18d1b47SKashyap D Desai pci_release_msi(sc->mrsas_dev); 1287d18d1b47SKashyap D Desai } 1288d18d1b47SKashyap D Desai 1289665484d8SDoug Ambrisko } 1290665484d8SDoug Ambrisko 12918e727371SKashyap D Desai /* 1292665484d8SDoug Ambrisko * mrsas_suspend: Suspend entry point 1293665484d8SDoug Ambrisko * input: Device struct pointer 1294665484d8SDoug Ambrisko * 1295665484d8SDoug Ambrisko * This function is the entry point for system suspend from the OS. 1296665484d8SDoug Ambrisko */ 12978e727371SKashyap D Desai static int 12988e727371SKashyap D Desai mrsas_suspend(device_t dev) 1299665484d8SDoug Ambrisko { 13004bb0a4f0SKashyap D Desai /* This will be filled when the driver will have hibernation support */ 1301665484d8SDoug Ambrisko return (0); 1302665484d8SDoug Ambrisko } 1303665484d8SDoug Ambrisko 13048e727371SKashyap D Desai /* 1305665484d8SDoug Ambrisko * mrsas_resume: Resume entry point 1306665484d8SDoug Ambrisko * input: Device struct pointer 1307665484d8SDoug Ambrisko * 1308665484d8SDoug Ambrisko * This function is the entry point for system resume from the OS. 1309665484d8SDoug Ambrisko */ 13108e727371SKashyap D Desai static int 13118e727371SKashyap D Desai mrsas_resume(device_t dev) 1312665484d8SDoug Ambrisko { 13134bb0a4f0SKashyap D Desai /* This will be filled when the driver will have hibernation support */ 1314665484d8SDoug Ambrisko return (0); 1315665484d8SDoug Ambrisko } 1316665484d8SDoug Ambrisko 13175844115eSKashyap D Desai /** 13185844115eSKashyap D Desai * mrsas_get_softc_instance: Find softc instance based on cmd type 13195844115eSKashyap D Desai * 13205844115eSKashyap D Desai * This function will return softc instance based on cmd type. 13215844115eSKashyap D Desai * In some case, application fire ioctl on required management instance and 13225844115eSKashyap D Desai * do not provide host_no. Use cdev->si_drv1 to get softc instance for those 13235844115eSKashyap D Desai * case, else get the softc instance from host_no provided by application in 13245844115eSKashyap D Desai * user data. 13255844115eSKashyap D Desai */ 13265844115eSKashyap D Desai 13275844115eSKashyap D Desai static struct mrsas_softc * 13285844115eSKashyap D Desai mrsas_get_softc_instance(struct cdev *dev, u_long cmd, caddr_t arg) 13295844115eSKashyap D Desai { 13305844115eSKashyap D Desai struct mrsas_softc *sc = NULL; 13315844115eSKashyap D Desai struct mrsas_iocpacket *user_ioc = (struct mrsas_iocpacket *)arg; 1332dbcc81dfSKashyap D Desai 13335844115eSKashyap D Desai if (cmd == MRSAS_IOC_GET_PCI_INFO) { 13345844115eSKashyap D Desai sc = dev->si_drv1; 13355844115eSKashyap D Desai } else { 1336dbcc81dfSKashyap D Desai /* 1337dbcc81dfSKashyap D Desai * get the Host number & the softc from data sent by the 1338dbcc81dfSKashyap D Desai * Application 1339dbcc81dfSKashyap D Desai */ 13405844115eSKashyap D Desai sc = mrsas_mgmt_info.sc_ptr[user_ioc->host_no]; 13415844115eSKashyap D Desai if (sc == NULL) 13425bae00d6SSteven Hartland printf("There is no Controller number %d\n", 13435bae00d6SSteven Hartland user_ioc->host_no); 13445bae00d6SSteven Hartland else if (user_ioc->host_no >= mrsas_mgmt_info.max_index) 13455844115eSKashyap D Desai mrsas_dprint(sc, MRSAS_FAULT, 13465bae00d6SSteven Hartland "Invalid Controller number %d\n", user_ioc->host_no); 13475844115eSKashyap D Desai } 13485844115eSKashyap D Desai 13495844115eSKashyap D Desai return sc; 13505844115eSKashyap D Desai } 13515844115eSKashyap D Desai 13528e727371SKashyap D Desai /* 1353665484d8SDoug Ambrisko * mrsas_ioctl: IOCtl commands entry point. 1354665484d8SDoug Ambrisko * 1355665484d8SDoug Ambrisko * This function is the entry point for IOCtls from the OS. It calls the 1356665484d8SDoug Ambrisko * appropriate function for processing depending on the command received. 1357665484d8SDoug Ambrisko */ 1358665484d8SDoug Ambrisko static int 13597fc5f329SJohn Baldwin mrsas_ioctl(struct cdev *dev, u_long cmd, caddr_t arg, int flag, 13607fc5f329SJohn Baldwin struct thread *td) 1361665484d8SDoug Ambrisko { 1362665484d8SDoug Ambrisko struct mrsas_softc *sc; 1363665484d8SDoug Ambrisko int ret = 0, i = 0; 13645844115eSKashyap D Desai MRSAS_DRV_PCI_INFORMATION *pciDrvInfo; 1365665484d8SDoug Ambrisko 13665844115eSKashyap D Desai sc = mrsas_get_softc_instance(dev, cmd, arg); 13675844115eSKashyap D Desai if (!sc) 1368536094dcSKashyap D Desai return ENOENT; 13695844115eSKashyap D Desai 1370808517a4SKashyap D Desai if (sc->remove_in_progress || 1371808517a4SKashyap D Desai (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR)) { 1372665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_INFO, 1373808517a4SKashyap D Desai "Either driver remove or shutdown called or " 1374808517a4SKashyap D Desai "HW is in unrecoverable critical error state.\n"); 1375665484d8SDoug Ambrisko return ENOENT; 1376665484d8SDoug Ambrisko } 1377665484d8SDoug Ambrisko mtx_lock_spin(&sc->ioctl_lock); 1378665484d8SDoug Ambrisko if (!sc->reset_in_progress) { 1379665484d8SDoug Ambrisko mtx_unlock_spin(&sc->ioctl_lock); 1380665484d8SDoug Ambrisko goto do_ioctl; 1381665484d8SDoug Ambrisko } 1382665484d8SDoug Ambrisko mtx_unlock_spin(&sc->ioctl_lock); 1383665484d8SDoug Ambrisko while (sc->reset_in_progress) { 1384665484d8SDoug Ambrisko i++; 1385665484d8SDoug Ambrisko if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) { 1386665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_INFO, 1387f0c7594bSKashyap D Desai "[%2d]waiting for OCR to be finished from %s\n", i, __func__); 1388665484d8SDoug Ambrisko } 1389665484d8SDoug Ambrisko pause("mr_ioctl", hz); 1390665484d8SDoug Ambrisko } 1391665484d8SDoug Ambrisko 1392665484d8SDoug Ambrisko do_ioctl: 1393665484d8SDoug Ambrisko switch (cmd) { 1394536094dcSKashyap D Desai case MRSAS_IOC_FIRMWARE_PASS_THROUGH64: 1395536094dcSKashyap D Desai #ifdef COMPAT_FREEBSD32 1396536094dcSKashyap D Desai case MRSAS_IOC_FIRMWARE_PASS_THROUGH32: 1397536094dcSKashyap D Desai #endif 13988e727371SKashyap D Desai /* 13998e727371SKashyap D Desai * Decrement the Ioctl counting Semaphore before getting an 14008e727371SKashyap D Desai * mfi command 14018e727371SKashyap D Desai */ 1402839ee025SKashyap D Desai sema_wait(&sc->ioctl_count_sema); 1403839ee025SKashyap D Desai 1404536094dcSKashyap D Desai ret = mrsas_passthru(sc, (void *)arg, cmd); 1405839ee025SKashyap D Desai 1406839ee025SKashyap D Desai /* Increment the Ioctl counting semaphore value */ 1407839ee025SKashyap D Desai sema_post(&sc->ioctl_count_sema); 1408839ee025SKashyap D Desai 1409665484d8SDoug Ambrisko break; 1410665484d8SDoug Ambrisko case MRSAS_IOC_SCAN_BUS: 1411665484d8SDoug Ambrisko ret = mrsas_bus_scan(sc); 1412665484d8SDoug Ambrisko break; 14135844115eSKashyap D Desai 14145844115eSKashyap D Desai case MRSAS_IOC_GET_PCI_INFO: 14155844115eSKashyap D Desai pciDrvInfo = (MRSAS_DRV_PCI_INFORMATION *) arg; 14165844115eSKashyap D Desai memset(pciDrvInfo, 0, sizeof(MRSAS_DRV_PCI_INFORMATION)); 14175844115eSKashyap D Desai pciDrvInfo->busNumber = pci_get_bus(sc->mrsas_dev); 14185844115eSKashyap D Desai pciDrvInfo->deviceNumber = pci_get_slot(sc->mrsas_dev); 14195844115eSKashyap D Desai pciDrvInfo->functionNumber = pci_get_function(sc->mrsas_dev); 14205844115eSKashyap D Desai pciDrvInfo->domainID = pci_get_domain(sc->mrsas_dev); 14215844115eSKashyap D Desai mrsas_dprint(sc, MRSAS_INFO, "pci bus no: %d," 14225844115eSKashyap D Desai "pci device no: %d, pci function no: %d," 14235844115eSKashyap D Desai "pci domain ID: %d\n", 14245844115eSKashyap D Desai pciDrvInfo->busNumber, pciDrvInfo->deviceNumber, 14255844115eSKashyap D Desai pciDrvInfo->functionNumber, pciDrvInfo->domainID); 14265844115eSKashyap D Desai ret = 0; 14275844115eSKashyap D Desai break; 14285844115eSKashyap D Desai 1429536094dcSKashyap D Desai default: 1430536094dcSKashyap D Desai mrsas_dprint(sc, MRSAS_TRACE, "IOCTL command 0x%lx is not handled\n", cmd); 1431839ee025SKashyap D Desai ret = ENOENT; 1432665484d8SDoug Ambrisko } 1433665484d8SDoug Ambrisko 1434665484d8SDoug Ambrisko return (ret); 1435665484d8SDoug Ambrisko } 1436665484d8SDoug Ambrisko 14378e727371SKashyap D Desai /* 1438da011113SKashyap D Desai * mrsas_poll: poll entry point for mrsas driver fd 1439da011113SKashyap D Desai * 14408e727371SKashyap D Desai * This function is the entry point for poll from the OS. It waits for some AEN 14418e727371SKashyap D Desai * events to be triggered from the controller and notifies back. 1442da011113SKashyap D Desai */ 1443da011113SKashyap D Desai static int 1444da011113SKashyap D Desai mrsas_poll(struct cdev *dev, int poll_events, struct thread *td) 1445da011113SKashyap D Desai { 1446da011113SKashyap D Desai struct mrsas_softc *sc; 1447da011113SKashyap D Desai int revents = 0; 1448da011113SKashyap D Desai 1449da011113SKashyap D Desai sc = dev->si_drv1; 1450da011113SKashyap D Desai 1451da011113SKashyap D Desai if (poll_events & (POLLIN | POLLRDNORM)) { 1452da011113SKashyap D Desai if (sc->mrsas_aen_triggered) { 1453da011113SKashyap D Desai revents |= poll_events & (POLLIN | POLLRDNORM); 1454da011113SKashyap D Desai } 1455da011113SKashyap D Desai } 1456da011113SKashyap D Desai if (revents == 0) { 1457da011113SKashyap D Desai if (poll_events & (POLLIN | POLLRDNORM)) { 1458ecea5be4SKashyap D Desai mtx_lock(&sc->aen_lock); 1459da011113SKashyap D Desai sc->mrsas_poll_waiting = 1; 1460da011113SKashyap D Desai selrecord(td, &sc->mrsas_select); 1461ecea5be4SKashyap D Desai mtx_unlock(&sc->aen_lock); 1462da011113SKashyap D Desai } 1463da011113SKashyap D Desai } 1464da011113SKashyap D Desai return revents; 1465da011113SKashyap D Desai } 1466da011113SKashyap D Desai 14678e727371SKashyap D Desai /* 14688e727371SKashyap D Desai * mrsas_setup_irq: Set up interrupt 1469665484d8SDoug Ambrisko * input: Adapter instance soft state 1470665484d8SDoug Ambrisko * 1471665484d8SDoug Ambrisko * This function sets up interrupts as a bus resource, with flags indicating 1472665484d8SDoug Ambrisko * resource permitting contemporaneous sharing and for resource to activate 1473665484d8SDoug Ambrisko * atomically. 1474665484d8SDoug Ambrisko */ 14758e727371SKashyap D Desai static int 14768e727371SKashyap D Desai mrsas_setup_irq(struct mrsas_softc *sc) 1477665484d8SDoug Ambrisko { 1478d18d1b47SKashyap D Desai if (sc->msix_enable && (mrsas_setup_msix(sc) == SUCCESS)) 1479d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "MSI-x interrupts setup success\n"); 1480665484d8SDoug Ambrisko 1481d18d1b47SKashyap D Desai else { 1482d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "Fall back to legacy interrupt\n"); 1483d18d1b47SKashyap D Desai sc->irq_context[0].sc = sc; 1484d18d1b47SKashyap D Desai sc->irq_context[0].MSIxIndex = 0; 1485d18d1b47SKashyap D Desai sc->irq_id[0] = 0; 1486d18d1b47SKashyap D Desai sc->mrsas_irq[0] = bus_alloc_resource_any(sc->mrsas_dev, 1487d18d1b47SKashyap D Desai SYS_RES_IRQ, &sc->irq_id[0], RF_SHAREABLE | RF_ACTIVE); 1488d18d1b47SKashyap D Desai if (sc->mrsas_irq[0] == NULL) { 1489d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "Cannot allocate legcay" 1490d18d1b47SKashyap D Desai "interrupt\n"); 1491d18d1b47SKashyap D Desai return (FAIL); 1492d18d1b47SKashyap D Desai } 1493d18d1b47SKashyap D Desai if (bus_setup_intr(sc->mrsas_dev, sc->mrsas_irq[0], 1494d18d1b47SKashyap D Desai INTR_MPSAFE | INTR_TYPE_CAM, NULL, mrsas_isr, 1495d18d1b47SKashyap D Desai &sc->irq_context[0], &sc->intr_handle[0])) { 1496d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "Cannot set up legacy" 1497d18d1b47SKashyap D Desai "interrupt\n"); 1498d18d1b47SKashyap D Desai return (FAIL); 1499d18d1b47SKashyap D Desai } 1500d18d1b47SKashyap D Desai } 1501665484d8SDoug Ambrisko return (0); 1502665484d8SDoug Ambrisko } 1503665484d8SDoug Ambrisko 1504665484d8SDoug Ambrisko /* 1505665484d8SDoug Ambrisko * mrsas_isr: ISR entry point 1506665484d8SDoug Ambrisko * input: argument pointer 1507665484d8SDoug Ambrisko * 15088e727371SKashyap D Desai * This function is the interrupt service routine entry point. There are two 15098e727371SKashyap D Desai * types of interrupts, state change interrupt and response interrupt. If an 15108e727371SKashyap D Desai * interrupt is not ours, we just return. 1511665484d8SDoug Ambrisko */ 15128e727371SKashyap D Desai void 15138e727371SKashyap D Desai mrsas_isr(void *arg) 1514665484d8SDoug Ambrisko { 1515d18d1b47SKashyap D Desai struct mrsas_irq_context *irq_context = (struct mrsas_irq_context *)arg; 1516d18d1b47SKashyap D Desai struct mrsas_softc *sc = irq_context->sc; 1517d18d1b47SKashyap D Desai int status = 0; 1518665484d8SDoug Ambrisko 15192f863eb8SKashyap D Desai if (sc->mask_interrupts) 15202f863eb8SKashyap D Desai return; 15212f863eb8SKashyap D Desai 1522d18d1b47SKashyap D Desai if (!sc->msix_vectors) { 1523665484d8SDoug Ambrisko status = mrsas_clear_intr(sc); 1524665484d8SDoug Ambrisko if (!status) 1525665484d8SDoug Ambrisko return; 1526d18d1b47SKashyap D Desai } 1527665484d8SDoug Ambrisko /* If we are resetting, bail */ 1528f5fb2237SKashyap D Desai if (mrsas_test_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags)) { 1529665484d8SDoug Ambrisko printf(" Entered into ISR when OCR is going active. \n"); 1530665484d8SDoug Ambrisko mrsas_clear_intr(sc); 1531665484d8SDoug Ambrisko return; 1532665484d8SDoug Ambrisko } 1533665484d8SDoug Ambrisko /* Process for reply request and clear response interrupt */ 1534d18d1b47SKashyap D Desai if (mrsas_complete_cmd(sc, irq_context->MSIxIndex) != SUCCESS) 1535665484d8SDoug Ambrisko mrsas_clear_intr(sc); 1536665484d8SDoug Ambrisko 1537665484d8SDoug Ambrisko return; 1538665484d8SDoug Ambrisko } 1539665484d8SDoug Ambrisko 1540665484d8SDoug Ambrisko /* 1541665484d8SDoug Ambrisko * mrsas_complete_cmd: Process reply request 1542665484d8SDoug Ambrisko * input: Adapter instance soft state 1543665484d8SDoug Ambrisko * 15448e727371SKashyap D Desai * This function is called from mrsas_isr() to process reply request and clear 15458e727371SKashyap D Desai * response interrupt. Processing of the reply request entails walking 15468e727371SKashyap D Desai * through the reply descriptor array for the command request pended from 15478e727371SKashyap D Desai * Firmware. We look at the Function field to determine the command type and 15488e727371SKashyap D Desai * perform the appropriate action. Before we return, we clear the response 15498e727371SKashyap D Desai * interrupt. 1550665484d8SDoug Ambrisko */ 15514bb0a4f0SKashyap D Desai int 15528e727371SKashyap D Desai mrsas_complete_cmd(struct mrsas_softc *sc, u_int32_t MSIxIndex) 1553665484d8SDoug Ambrisko { 1554665484d8SDoug Ambrisko Mpi2ReplyDescriptorsUnion_t *desc; 1555665484d8SDoug Ambrisko MPI2_SCSI_IO_SUCCESS_REPLY_DESCRIPTOR *reply_desc; 1556665484d8SDoug Ambrisko MRSAS_RAID_SCSI_IO_REQUEST *scsi_io_req; 1557665484d8SDoug Ambrisko struct mrsas_mpt_cmd *cmd_mpt; 1558665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd_mfi; 155916dc2814SKashyap D Desai u_int8_t reply_descript_type; 1560665484d8SDoug Ambrisko u_int16_t smid, num_completed; 1561665484d8SDoug Ambrisko u_int8_t status, extStatus; 1562665484d8SDoug Ambrisko union desc_value desc_val; 1563665484d8SDoug Ambrisko PLD_LOAD_BALANCE_INFO lbinfo; 1564665484d8SDoug Ambrisko u_int32_t device_id; 1565665484d8SDoug Ambrisko int threshold_reply_count = 0; 15668bb601acSKashyap D Desai #if TM_DEBUG 15678bb601acSKashyap D Desai MR_TASK_MANAGE_REQUEST *mr_tm_req; 15688bb601acSKashyap D Desai MPI2_SCSI_TASK_MANAGE_REQUEST *mpi_tm_req; 15698bb601acSKashyap D Desai #endif 1570665484d8SDoug Ambrisko 1571665484d8SDoug Ambrisko /* If we have a hardware error, not need to continue */ 1572665484d8SDoug Ambrisko if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) 1573665484d8SDoug Ambrisko return (DONE); 1574665484d8SDoug Ambrisko 1575665484d8SDoug Ambrisko desc = sc->reply_desc_mem; 1576d18d1b47SKashyap D Desai desc += ((MSIxIndex * sc->reply_alloc_sz) / sizeof(MPI2_REPLY_DESCRIPTORS_UNION)) 1577d18d1b47SKashyap D Desai + sc->last_reply_idx[MSIxIndex]; 1578665484d8SDoug Ambrisko 1579665484d8SDoug Ambrisko reply_desc = (MPI2_SCSI_IO_SUCCESS_REPLY_DESCRIPTOR *) desc; 1580665484d8SDoug Ambrisko 1581665484d8SDoug Ambrisko desc_val.word = desc->Words; 1582665484d8SDoug Ambrisko num_completed = 0; 1583665484d8SDoug Ambrisko 1584665484d8SDoug Ambrisko reply_descript_type = reply_desc->ReplyFlags & MPI2_RPY_DESCRIPT_FLAGS_TYPE_MASK; 1585665484d8SDoug Ambrisko 1586665484d8SDoug Ambrisko /* Find our reply descriptor for the command and process */ 15878e727371SKashyap D Desai while ((desc_val.u.low != 0xFFFFFFFF) && (desc_val.u.high != 0xFFFFFFFF)) { 1588665484d8SDoug Ambrisko smid = reply_desc->SMID; 1589665484d8SDoug Ambrisko cmd_mpt = sc->mpt_cmd_list[smid - 1]; 1590665484d8SDoug Ambrisko scsi_io_req = (MRSAS_RAID_SCSI_IO_REQUEST *) cmd_mpt->io_request; 1591665484d8SDoug Ambrisko 1592*503c4f8dSKashyap D Desai status = scsi_io_req->RaidContext.raid_context.status; 1593*503c4f8dSKashyap D Desai extStatus = scsi_io_req->RaidContext.raid_context.exStatus; 1594665484d8SDoug Ambrisko 15958e727371SKashyap D Desai switch (scsi_io_req->Function) { 15968bb601acSKashyap D Desai case MPI2_FUNCTION_SCSI_TASK_MGMT: 15978bb601acSKashyap D Desai #if TM_DEBUG 15988bb601acSKashyap D Desai mr_tm_req = (MR_TASK_MANAGE_REQUEST *) cmd_mpt->io_request; 15998bb601acSKashyap D Desai mpi_tm_req = (MPI2_SCSI_TASK_MANAGE_REQUEST *) 16008bb601acSKashyap D Desai &mr_tm_req->TmRequest; 16018bb601acSKashyap D Desai device_printf(sc->mrsas_dev, "TM completion type 0x%X, " 16028bb601acSKashyap D Desai "TaskMID: 0x%X", mpi_tm_req->TaskType, mpi_tm_req->TaskMID); 16038bb601acSKashyap D Desai #endif 16048bb601acSKashyap D Desai wakeup_one((void *)&sc->ocr_chan); 16058bb601acSKashyap D Desai break; 1606665484d8SDoug Ambrisko case MPI2_FUNCTION_SCSI_IO_REQUEST: /* Fast Path IO. */ 1607665484d8SDoug Ambrisko device_id = cmd_mpt->ccb_ptr->ccb_h.target_id; 1608665484d8SDoug Ambrisko lbinfo = &sc->load_balance_info[device_id]; 1609665484d8SDoug Ambrisko if (cmd_mpt->load_balance == MRSAS_LOAD_BALANCE_FLAG) { 161016dc2814SKashyap D Desai mrsas_atomic_dec(&lbinfo->scsi_pending_cmds[cmd_mpt->pd_r1_lb]); 1611665484d8SDoug Ambrisko cmd_mpt->load_balance &= ~MRSAS_LOAD_BALANCE_FLAG; 1612665484d8SDoug Ambrisko } 16138e727371SKashyap D Desai /* Fall thru and complete IO */ 1614665484d8SDoug Ambrisko case MRSAS_MPI2_FUNCTION_LD_IO_REQUEST: 1615665484d8SDoug Ambrisko mrsas_map_mpt_cmd_status(cmd_mpt, status, extStatus); 1616665484d8SDoug Ambrisko mrsas_cmd_done(sc, cmd_mpt); 1617*503c4f8dSKashyap D Desai scsi_io_req->RaidContext.raid_context.status = 0; 1618*503c4f8dSKashyap D Desai scsi_io_req->RaidContext.raid_context.exStatus = 0; 1619f5fb2237SKashyap D Desai mrsas_atomic_dec(&sc->fw_outstanding); 1620665484d8SDoug Ambrisko break; 1621665484d8SDoug Ambrisko case MRSAS_MPI2_FUNCTION_PASSTHRU_IO_REQUEST: /* MFI command */ 1622665484d8SDoug Ambrisko cmd_mfi = sc->mfi_cmd_list[cmd_mpt->sync_cmd_idx]; 1623731b7561SKashyap D Desai /* 1624731b7561SKashyap D Desai * Make sure NOT TO release the mfi command from the called 1625731b7561SKashyap D Desai * function's context if it is fired with issue_polled call. 1626731b7561SKashyap D Desai * And also make sure that the issue_polled call should only be 1627731b7561SKashyap D Desai * used if INTERRUPT IS DISABLED. 1628731b7561SKashyap D Desai */ 1629731b7561SKashyap D Desai if (cmd_mfi->frame->hdr.flags & MFI_FRAME_DONT_POST_IN_REPLY_QUEUE) 1630731b7561SKashyap D Desai mrsas_release_mfi_cmd(cmd_mfi); 1631731b7561SKashyap D Desai else 1632665484d8SDoug Ambrisko mrsas_complete_mptmfi_passthru(sc, cmd_mfi, status); 1633665484d8SDoug Ambrisko break; 1634665484d8SDoug Ambrisko } 1635665484d8SDoug Ambrisko 1636d18d1b47SKashyap D Desai sc->last_reply_idx[MSIxIndex]++; 1637d18d1b47SKashyap D Desai if (sc->last_reply_idx[MSIxIndex] >= sc->reply_q_depth) 1638d18d1b47SKashyap D Desai sc->last_reply_idx[MSIxIndex] = 0; 1639665484d8SDoug Ambrisko 16408e727371SKashyap D Desai desc->Words = ~((uint64_t)0x00); /* set it back to all 16418e727371SKashyap D Desai * 0xFFFFFFFFs */ 1642665484d8SDoug Ambrisko num_completed++; 1643665484d8SDoug Ambrisko threshold_reply_count++; 1644665484d8SDoug Ambrisko 1645665484d8SDoug Ambrisko /* Get the next reply descriptor */ 1646d18d1b47SKashyap D Desai if (!sc->last_reply_idx[MSIxIndex]) { 1647665484d8SDoug Ambrisko desc = sc->reply_desc_mem; 1648d18d1b47SKashyap D Desai desc += ((MSIxIndex * sc->reply_alloc_sz) / sizeof(MPI2_REPLY_DESCRIPTORS_UNION)); 1649d18d1b47SKashyap D Desai } else 1650665484d8SDoug Ambrisko desc++; 1651665484d8SDoug Ambrisko 1652665484d8SDoug Ambrisko reply_desc = (MPI2_SCSI_IO_SUCCESS_REPLY_DESCRIPTOR *) desc; 1653665484d8SDoug Ambrisko desc_val.word = desc->Words; 1654665484d8SDoug Ambrisko 1655665484d8SDoug Ambrisko reply_descript_type = reply_desc->ReplyFlags & MPI2_RPY_DESCRIPT_FLAGS_TYPE_MASK; 1656665484d8SDoug Ambrisko 1657665484d8SDoug Ambrisko if (reply_descript_type == MPI2_RPY_DESCRIPT_FLAGS_UNUSED) 1658665484d8SDoug Ambrisko break; 1659665484d8SDoug Ambrisko 1660665484d8SDoug Ambrisko /* 16618e727371SKashyap D Desai * Write to reply post index after completing threshold reply 16628e727371SKashyap D Desai * count and still there are more replies in reply queue 16638e727371SKashyap D Desai * pending to be completed. 1664665484d8SDoug Ambrisko */ 1665665484d8SDoug Ambrisko if (threshold_reply_count >= THRESHOLD_REPLY_COUNT) { 1666d18d1b47SKashyap D Desai if (sc->msix_enable) { 16677aade8bfSKashyap D Desai if (sc->msix_combined) 1668d18d1b47SKashyap D Desai mrsas_write_reg(sc, sc->msix_reg_offset[MSIxIndex / 8], 1669d18d1b47SKashyap D Desai ((MSIxIndex & 0x7) << 24) | 1670d18d1b47SKashyap D Desai sc->last_reply_idx[MSIxIndex]); 1671d18d1b47SKashyap D Desai else 1672d18d1b47SKashyap D Desai mrsas_write_reg(sc, sc->msix_reg_offset[0], (MSIxIndex << 24) | 1673d18d1b47SKashyap D Desai sc->last_reply_idx[MSIxIndex]); 1674d18d1b47SKashyap D Desai } else 1675d18d1b47SKashyap D Desai mrsas_write_reg(sc, offsetof(mrsas_reg_set, 1676d18d1b47SKashyap D Desai reply_post_host_index), sc->last_reply_idx[0]); 1677d18d1b47SKashyap D Desai 1678665484d8SDoug Ambrisko threshold_reply_count = 0; 1679665484d8SDoug Ambrisko } 1680665484d8SDoug Ambrisko } 1681665484d8SDoug Ambrisko 1682665484d8SDoug Ambrisko /* No match, just return */ 1683665484d8SDoug Ambrisko if (num_completed == 0) 1684665484d8SDoug Ambrisko return (DONE); 1685665484d8SDoug Ambrisko 1686665484d8SDoug Ambrisko /* Clear response interrupt */ 1687d18d1b47SKashyap D Desai if (sc->msix_enable) { 16887aade8bfSKashyap D Desai if (sc->msix_combined) { 1689d18d1b47SKashyap D Desai mrsas_write_reg(sc, sc->msix_reg_offset[MSIxIndex / 8], 1690d18d1b47SKashyap D Desai ((MSIxIndex & 0x7) << 24) | 1691d18d1b47SKashyap D Desai sc->last_reply_idx[MSIxIndex]); 1692d18d1b47SKashyap D Desai } else 1693d18d1b47SKashyap D Desai mrsas_write_reg(sc, sc->msix_reg_offset[0], (MSIxIndex << 24) | 1694d18d1b47SKashyap D Desai sc->last_reply_idx[MSIxIndex]); 1695d18d1b47SKashyap D Desai } else 1696d18d1b47SKashyap D Desai mrsas_write_reg(sc, offsetof(mrsas_reg_set, 1697d18d1b47SKashyap D Desai reply_post_host_index), sc->last_reply_idx[0]); 1698665484d8SDoug Ambrisko 1699665484d8SDoug Ambrisko return (0); 1700665484d8SDoug Ambrisko } 1701665484d8SDoug Ambrisko 1702665484d8SDoug Ambrisko /* 1703665484d8SDoug Ambrisko * mrsas_map_mpt_cmd_status: Allocate DMAable memory. 1704665484d8SDoug Ambrisko * input: Adapter instance soft state 1705665484d8SDoug Ambrisko * 1706665484d8SDoug Ambrisko * This function is called from mrsas_complete_cmd(), for LD IO and FastPath IO. 17078e727371SKashyap D Desai * It checks the command status and maps the appropriate CAM status for the 17088e727371SKashyap D Desai * CCB. 1709665484d8SDoug Ambrisko */ 17108e727371SKashyap D Desai void 17118e727371SKashyap D Desai mrsas_map_mpt_cmd_status(struct mrsas_mpt_cmd *cmd, u_int8_t status, u_int8_t extStatus) 1712665484d8SDoug Ambrisko { 1713665484d8SDoug Ambrisko struct mrsas_softc *sc = cmd->sc; 1714665484d8SDoug Ambrisko u_int8_t *sense_data; 1715665484d8SDoug Ambrisko 1716665484d8SDoug Ambrisko switch (status) { 1717665484d8SDoug Ambrisko case MFI_STAT_OK: 1718665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status = CAM_REQ_CMP; 1719665484d8SDoug Ambrisko break; 1720665484d8SDoug Ambrisko case MFI_STAT_SCSI_IO_FAILED: 1721665484d8SDoug Ambrisko case MFI_STAT_SCSI_DONE_WITH_ERROR: 1722665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status = CAM_SCSI_STATUS_ERROR; 1723665484d8SDoug Ambrisko sense_data = (u_int8_t *)&cmd->ccb_ptr->csio.sense_data; 1724665484d8SDoug Ambrisko if (sense_data) { 1725665484d8SDoug Ambrisko /* For now just copy 18 bytes back */ 1726665484d8SDoug Ambrisko memcpy(sense_data, cmd->sense, 18); 1727665484d8SDoug Ambrisko cmd->ccb_ptr->csio.sense_len = 18; 1728665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status |= CAM_AUTOSNS_VALID; 1729665484d8SDoug Ambrisko } 1730665484d8SDoug Ambrisko break; 1731665484d8SDoug Ambrisko case MFI_STAT_LD_OFFLINE: 1732665484d8SDoug Ambrisko case MFI_STAT_DEVICE_NOT_FOUND: 1733665484d8SDoug Ambrisko if (cmd->ccb_ptr->ccb_h.target_lun) 1734665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status |= CAM_LUN_INVALID; 1735665484d8SDoug Ambrisko else 1736665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status |= CAM_DEV_NOT_THERE; 1737665484d8SDoug Ambrisko break; 1738665484d8SDoug Ambrisko case MFI_STAT_CONFIG_SEQ_MISMATCH: 1739665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status |= CAM_REQUEUE_REQ; 1740665484d8SDoug Ambrisko break; 1741665484d8SDoug Ambrisko default: 1742665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "FW cmd complete status %x\n", status); 1743665484d8SDoug Ambrisko cmd->ccb_ptr->ccb_h.status = CAM_REQ_CMP_ERR; 1744665484d8SDoug Ambrisko cmd->ccb_ptr->csio.scsi_status = status; 1745665484d8SDoug Ambrisko } 1746665484d8SDoug Ambrisko return; 1747665484d8SDoug Ambrisko } 1748665484d8SDoug Ambrisko 1749665484d8SDoug Ambrisko /* 17508e727371SKashyap D Desai * mrsas_alloc_mem: Allocate DMAable memory 1751665484d8SDoug Ambrisko * input: Adapter instance soft state 1752665484d8SDoug Ambrisko * 17538e727371SKashyap D Desai * This function creates the parent DMA tag and allocates DMAable memory. DMA 17548e727371SKashyap D Desai * tag describes constraints of DMA mapping. Memory allocated is mapped into 17558e727371SKashyap D Desai * Kernel virtual address. Callback argument is physical memory address. 1756665484d8SDoug Ambrisko */ 17578e727371SKashyap D Desai static int 17588e727371SKashyap D Desai mrsas_alloc_mem(struct mrsas_softc *sc) 1759665484d8SDoug Ambrisko { 17604ad83576SKashyap D Desai u_int32_t verbuf_size, io_req_size, reply_desc_size, sense_size, chain_frame_size, 17614ad83576SKashyap D Desai evt_detail_size, count; 1762665484d8SDoug Ambrisko 1763665484d8SDoug Ambrisko /* 1764665484d8SDoug Ambrisko * Allocate parent DMA tag 1765665484d8SDoug Ambrisko */ 1766665484d8SDoug Ambrisko if (bus_dma_tag_create(NULL, /* parent */ 1767665484d8SDoug Ambrisko 1, /* alignment */ 1768665484d8SDoug Ambrisko 0, /* boundary */ 1769665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, /* lowaddr */ 1770665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, /* highaddr */ 1771665484d8SDoug Ambrisko NULL, NULL, /* filter, filterarg */ 17723a3fc6cbSKashyap D Desai MAXPHYS, /* maxsize */ 17733a3fc6cbSKashyap D Desai sc->max_num_sge, /* nsegments */ 17743a3fc6cbSKashyap D Desai MAXPHYS, /* maxsegsize */ 1775665484d8SDoug Ambrisko 0, /* flags */ 1776665484d8SDoug Ambrisko NULL, NULL, /* lockfunc, lockarg */ 1777665484d8SDoug Ambrisko &sc->mrsas_parent_tag /* tag */ 1778665484d8SDoug Ambrisko )) { 1779665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate parent DMA tag\n"); 1780665484d8SDoug Ambrisko return (ENOMEM); 1781665484d8SDoug Ambrisko } 1782665484d8SDoug Ambrisko /* 1783665484d8SDoug Ambrisko * Allocate for version buffer 1784665484d8SDoug Ambrisko */ 1785665484d8SDoug Ambrisko verbuf_size = MRSAS_MAX_NAME_LENGTH * (sizeof(bus_addr_t)); 17868e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 17878e727371SKashyap D Desai 1, 0, 17888e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 17898e727371SKashyap D Desai BUS_SPACE_MAXADDR, 17908e727371SKashyap D Desai NULL, NULL, 17918e727371SKashyap D Desai verbuf_size, 17928e727371SKashyap D Desai 1, 17938e727371SKashyap D Desai verbuf_size, 17948e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 17958e727371SKashyap D Desai NULL, NULL, 1796665484d8SDoug Ambrisko &sc->verbuf_tag)) { 1797665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate verbuf DMA tag\n"); 1798665484d8SDoug Ambrisko return (ENOMEM); 1799665484d8SDoug Ambrisko } 1800665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->verbuf_tag, (void **)&sc->verbuf_mem, 1801665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->verbuf_dmamap)) { 1802665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate verbuf memory\n"); 1803665484d8SDoug Ambrisko return (ENOMEM); 1804665484d8SDoug Ambrisko } 1805665484d8SDoug Ambrisko bzero(sc->verbuf_mem, verbuf_size); 1806665484d8SDoug Ambrisko if (bus_dmamap_load(sc->verbuf_tag, sc->verbuf_dmamap, sc->verbuf_mem, 18078e727371SKashyap D Desai verbuf_size, mrsas_addr_cb, &sc->verbuf_phys_addr, 18088e727371SKashyap D Desai BUS_DMA_NOWAIT)) { 1809665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load verbuf DMA map\n"); 1810665484d8SDoug Ambrisko return (ENOMEM); 1811665484d8SDoug Ambrisko } 1812665484d8SDoug Ambrisko /* 1813665484d8SDoug Ambrisko * Allocate IO Request Frames 1814665484d8SDoug Ambrisko */ 1815665484d8SDoug Ambrisko io_req_size = sc->io_frames_alloc_sz; 18168e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 18178e727371SKashyap D Desai 16, 0, 18188e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 18198e727371SKashyap D Desai BUS_SPACE_MAXADDR, 18208e727371SKashyap D Desai NULL, NULL, 18218e727371SKashyap D Desai io_req_size, 18228e727371SKashyap D Desai 1, 18238e727371SKashyap D Desai io_req_size, 18248e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 18258e727371SKashyap D Desai NULL, NULL, 1826665484d8SDoug Ambrisko &sc->io_request_tag)) { 1827665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create IO request tag\n"); 1828665484d8SDoug Ambrisko return (ENOMEM); 1829665484d8SDoug Ambrisko } 1830665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->io_request_tag, (void **)&sc->io_request_mem, 1831665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->io_request_dmamap)) { 1832665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc IO request memory\n"); 1833665484d8SDoug Ambrisko return (ENOMEM); 1834665484d8SDoug Ambrisko } 1835665484d8SDoug Ambrisko bzero(sc->io_request_mem, io_req_size); 1836665484d8SDoug Ambrisko if (bus_dmamap_load(sc->io_request_tag, sc->io_request_dmamap, 1837665484d8SDoug Ambrisko sc->io_request_mem, io_req_size, mrsas_addr_cb, 1838665484d8SDoug Ambrisko &sc->io_request_phys_addr, BUS_DMA_NOWAIT)) { 1839665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load IO request memory\n"); 1840665484d8SDoug Ambrisko return (ENOMEM); 1841665484d8SDoug Ambrisko } 1842665484d8SDoug Ambrisko /* 1843665484d8SDoug Ambrisko * Allocate Chain Frames 1844665484d8SDoug Ambrisko */ 1845665484d8SDoug Ambrisko chain_frame_size = sc->chain_frames_alloc_sz; 18468e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 18478e727371SKashyap D Desai 4, 0, 18488e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 18498e727371SKashyap D Desai BUS_SPACE_MAXADDR, 18508e727371SKashyap D Desai NULL, NULL, 18518e727371SKashyap D Desai chain_frame_size, 18528e727371SKashyap D Desai 1, 18538e727371SKashyap D Desai chain_frame_size, 18548e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 18558e727371SKashyap D Desai NULL, NULL, 1856665484d8SDoug Ambrisko &sc->chain_frame_tag)) { 1857665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create chain frame tag\n"); 1858665484d8SDoug Ambrisko return (ENOMEM); 1859665484d8SDoug Ambrisko } 1860665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->chain_frame_tag, (void **)&sc->chain_frame_mem, 1861665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->chain_frame_dmamap)) { 1862665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc chain frame memory\n"); 1863665484d8SDoug Ambrisko return (ENOMEM); 1864665484d8SDoug Ambrisko } 1865665484d8SDoug Ambrisko bzero(sc->chain_frame_mem, chain_frame_size); 1866665484d8SDoug Ambrisko if (bus_dmamap_load(sc->chain_frame_tag, sc->chain_frame_dmamap, 1867665484d8SDoug Ambrisko sc->chain_frame_mem, chain_frame_size, mrsas_addr_cb, 1868665484d8SDoug Ambrisko &sc->chain_frame_phys_addr, BUS_DMA_NOWAIT)) { 1869665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load chain frame memory\n"); 1870665484d8SDoug Ambrisko return (ENOMEM); 1871665484d8SDoug Ambrisko } 1872d18d1b47SKashyap D Desai count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 1873665484d8SDoug Ambrisko /* 1874665484d8SDoug Ambrisko * Allocate Reply Descriptor Array 1875665484d8SDoug Ambrisko */ 1876d18d1b47SKashyap D Desai reply_desc_size = sc->reply_alloc_sz * count; 18778e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 18788e727371SKashyap D Desai 16, 0, 18798e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 18808e727371SKashyap D Desai BUS_SPACE_MAXADDR, 18818e727371SKashyap D Desai NULL, NULL, 18828e727371SKashyap D Desai reply_desc_size, 18838e727371SKashyap D Desai 1, 18848e727371SKashyap D Desai reply_desc_size, 18858e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 18868e727371SKashyap D Desai NULL, NULL, 1887665484d8SDoug Ambrisko &sc->reply_desc_tag)) { 1888665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create reply descriptor tag\n"); 1889665484d8SDoug Ambrisko return (ENOMEM); 1890665484d8SDoug Ambrisko } 1891665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->reply_desc_tag, (void **)&sc->reply_desc_mem, 1892665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->reply_desc_dmamap)) { 1893665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc reply descriptor memory\n"); 1894665484d8SDoug Ambrisko return (ENOMEM); 1895665484d8SDoug Ambrisko } 1896665484d8SDoug Ambrisko if (bus_dmamap_load(sc->reply_desc_tag, sc->reply_desc_dmamap, 1897665484d8SDoug Ambrisko sc->reply_desc_mem, reply_desc_size, mrsas_addr_cb, 1898665484d8SDoug Ambrisko &sc->reply_desc_phys_addr, BUS_DMA_NOWAIT)) { 1899665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load reply descriptor memory\n"); 1900665484d8SDoug Ambrisko return (ENOMEM); 1901665484d8SDoug Ambrisko } 1902665484d8SDoug Ambrisko /* 1903665484d8SDoug Ambrisko * Allocate Sense Buffer Array. Keep in lower 4GB 1904665484d8SDoug Ambrisko */ 1905665484d8SDoug Ambrisko sense_size = sc->max_fw_cmds * MRSAS_SENSE_LEN; 19068e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 19078e727371SKashyap D Desai 64, 0, 19088e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 19098e727371SKashyap D Desai BUS_SPACE_MAXADDR, 19108e727371SKashyap D Desai NULL, NULL, 19118e727371SKashyap D Desai sense_size, 19128e727371SKashyap D Desai 1, 19138e727371SKashyap D Desai sense_size, 19148e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 19158e727371SKashyap D Desai NULL, NULL, 1916665484d8SDoug Ambrisko &sc->sense_tag)) { 1917665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate sense buf tag\n"); 1918665484d8SDoug Ambrisko return (ENOMEM); 1919665484d8SDoug Ambrisko } 1920665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->sense_tag, (void **)&sc->sense_mem, 1921665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->sense_dmamap)) { 1922665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate sense buf memory\n"); 1923665484d8SDoug Ambrisko return (ENOMEM); 1924665484d8SDoug Ambrisko } 1925665484d8SDoug Ambrisko if (bus_dmamap_load(sc->sense_tag, sc->sense_dmamap, 1926665484d8SDoug Ambrisko sc->sense_mem, sense_size, mrsas_addr_cb, &sc->sense_phys_addr, 1927665484d8SDoug Ambrisko BUS_DMA_NOWAIT)) { 1928665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load sense buf memory\n"); 1929665484d8SDoug Ambrisko return (ENOMEM); 1930665484d8SDoug Ambrisko } 1931665484d8SDoug Ambrisko /* 1932665484d8SDoug Ambrisko * Allocate for Event detail structure 1933665484d8SDoug Ambrisko */ 1934665484d8SDoug Ambrisko evt_detail_size = sizeof(struct mrsas_evt_detail); 19358e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 19368e727371SKashyap D Desai 1, 0, 19378e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 19388e727371SKashyap D Desai BUS_SPACE_MAXADDR, 19398e727371SKashyap D Desai NULL, NULL, 19408e727371SKashyap D Desai evt_detail_size, 19418e727371SKashyap D Desai 1, 19428e727371SKashyap D Desai evt_detail_size, 19438e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 19448e727371SKashyap D Desai NULL, NULL, 1945665484d8SDoug Ambrisko &sc->evt_detail_tag)) { 1946665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create Event detail tag\n"); 1947665484d8SDoug Ambrisko return (ENOMEM); 1948665484d8SDoug Ambrisko } 1949665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->evt_detail_tag, (void **)&sc->evt_detail_mem, 1950665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->evt_detail_dmamap)) { 1951665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc Event detail buffer memory\n"); 1952665484d8SDoug Ambrisko return (ENOMEM); 1953665484d8SDoug Ambrisko } 1954665484d8SDoug Ambrisko bzero(sc->evt_detail_mem, evt_detail_size); 1955665484d8SDoug Ambrisko if (bus_dmamap_load(sc->evt_detail_tag, sc->evt_detail_dmamap, 1956665484d8SDoug Ambrisko sc->evt_detail_mem, evt_detail_size, mrsas_addr_cb, 1957665484d8SDoug Ambrisko &sc->evt_detail_phys_addr, BUS_DMA_NOWAIT)) { 1958665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load Event detail buffer memory\n"); 1959665484d8SDoug Ambrisko return (ENOMEM); 1960665484d8SDoug Ambrisko } 1961665484d8SDoug Ambrisko /* 1962665484d8SDoug Ambrisko * Create a dma tag for data buffers; size will be the maximum 1963665484d8SDoug Ambrisko * possible I/O size (280kB). 1964665484d8SDoug Ambrisko */ 19658e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 19668e727371SKashyap D Desai 1, 19678e727371SKashyap D Desai 0, 19688e727371SKashyap D Desai BUS_SPACE_MAXADDR, 19698e727371SKashyap D Desai BUS_SPACE_MAXADDR, 19708e727371SKashyap D Desai NULL, NULL, 19713a3fc6cbSKashyap D Desai MAXPHYS, 19723a3fc6cbSKashyap D Desai sc->max_num_sge, /* nsegments */ 19733a3fc6cbSKashyap D Desai MAXPHYS, 19748e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 19758e727371SKashyap D Desai busdma_lock_mutex, 19768e727371SKashyap D Desai &sc->io_lock, 1977665484d8SDoug Ambrisko &sc->data_tag)) { 1978665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create data dma tag\n"); 1979665484d8SDoug Ambrisko return (ENOMEM); 1980665484d8SDoug Ambrisko } 1981665484d8SDoug Ambrisko return (0); 1982665484d8SDoug Ambrisko } 1983665484d8SDoug Ambrisko 1984665484d8SDoug Ambrisko /* 1985665484d8SDoug Ambrisko * mrsas_addr_cb: Callback function of bus_dmamap_load() 19868e727371SKashyap D Desai * input: callback argument, machine dependent type 19878e727371SKashyap D Desai * that describes DMA segments, number of segments, error code 1988665484d8SDoug Ambrisko * 19898e727371SKashyap D Desai * This function is for the driver to receive mapping information resultant of 19908e727371SKashyap D Desai * the bus_dmamap_load(). The information is actually not being used, but the 19918e727371SKashyap D Desai * address is saved anyway. 1992665484d8SDoug Ambrisko */ 1993665484d8SDoug Ambrisko void 1994665484d8SDoug Ambrisko mrsas_addr_cb(void *arg, bus_dma_segment_t *segs, int nsegs, int error) 1995665484d8SDoug Ambrisko { 1996665484d8SDoug Ambrisko bus_addr_t *addr; 1997665484d8SDoug Ambrisko 1998665484d8SDoug Ambrisko addr = arg; 1999665484d8SDoug Ambrisko *addr = segs[0].ds_addr; 2000665484d8SDoug Ambrisko } 2001665484d8SDoug Ambrisko 2002665484d8SDoug Ambrisko /* 2003665484d8SDoug Ambrisko * mrsas_setup_raidmap: Set up RAID map. 2004665484d8SDoug Ambrisko * input: Adapter instance soft state 2005665484d8SDoug Ambrisko * 2006665484d8SDoug Ambrisko * Allocate DMA memory for the RAID maps and perform setup. 2007665484d8SDoug Ambrisko */ 20088e727371SKashyap D Desai static int 20098e727371SKashyap D Desai mrsas_setup_raidmap(struct mrsas_softc *sc) 2010665484d8SDoug Ambrisko { 20114799d485SKashyap D Desai int i; 20124799d485SKashyap D Desai 20134799d485SKashyap D Desai for (i = 0; i < 2; i++) { 20144799d485SKashyap D Desai sc->ld_drv_map[i] = 20154799d485SKashyap D Desai (void *)malloc(sc->drv_map_sz, M_MRSAS, M_NOWAIT); 20164799d485SKashyap D Desai /* Do Error handling */ 20174799d485SKashyap D Desai if (!sc->ld_drv_map[i]) { 20184799d485SKashyap D Desai device_printf(sc->mrsas_dev, "Could not allocate memory for local map"); 20194799d485SKashyap D Desai 20204799d485SKashyap D Desai if (i == 1) 20214799d485SKashyap D Desai free(sc->ld_drv_map[0], M_MRSAS); 20228e727371SKashyap D Desai /* ABORT driver initialization */ 20234799d485SKashyap D Desai goto ABORT; 20244799d485SKashyap D Desai } 20254799d485SKashyap D Desai } 20264799d485SKashyap D Desai 20278e727371SKashyap D Desai for (int i = 0; i < 2; i++) { 20288e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 20298e727371SKashyap D Desai 4, 0, 20308e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 20318e727371SKashyap D Desai BUS_SPACE_MAXADDR, 20328e727371SKashyap D Desai NULL, NULL, 20338e727371SKashyap D Desai sc->max_map_sz, 20348e727371SKashyap D Desai 1, 20358e727371SKashyap D Desai sc->max_map_sz, 20368e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 20378e727371SKashyap D Desai NULL, NULL, 2038665484d8SDoug Ambrisko &sc->raidmap_tag[i])) { 20394799d485SKashyap D Desai device_printf(sc->mrsas_dev, 20404799d485SKashyap D Desai "Cannot allocate raid map tag.\n"); 2041665484d8SDoug Ambrisko return (ENOMEM); 2042665484d8SDoug Ambrisko } 20434799d485SKashyap D Desai if (bus_dmamem_alloc(sc->raidmap_tag[i], 20444799d485SKashyap D Desai (void **)&sc->raidmap_mem[i], 2045665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->raidmap_dmamap[i])) { 20464799d485SKashyap D Desai device_printf(sc->mrsas_dev, 20474799d485SKashyap D Desai "Cannot allocate raidmap memory.\n"); 2048665484d8SDoug Ambrisko return (ENOMEM); 2049665484d8SDoug Ambrisko } 20504799d485SKashyap D Desai bzero(sc->raidmap_mem[i], sc->max_map_sz); 20514799d485SKashyap D Desai 2052665484d8SDoug Ambrisko if (bus_dmamap_load(sc->raidmap_tag[i], sc->raidmap_dmamap[i], 20534799d485SKashyap D Desai sc->raidmap_mem[i], sc->max_map_sz, 20544799d485SKashyap D Desai mrsas_addr_cb, &sc->raidmap_phys_addr[i], 2055665484d8SDoug Ambrisko BUS_DMA_NOWAIT)) { 2056665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load raidmap memory.\n"); 2057665484d8SDoug Ambrisko return (ENOMEM); 2058665484d8SDoug Ambrisko } 2059665484d8SDoug Ambrisko if (!sc->raidmap_mem[i]) { 20604799d485SKashyap D Desai device_printf(sc->mrsas_dev, 20614799d485SKashyap D Desai "Cannot allocate memory for raid map.\n"); 2062665484d8SDoug Ambrisko return (ENOMEM); 2063665484d8SDoug Ambrisko } 2064665484d8SDoug Ambrisko } 2065665484d8SDoug Ambrisko 2066665484d8SDoug Ambrisko if (!mrsas_get_map_info(sc)) 2067665484d8SDoug Ambrisko mrsas_sync_map_info(sc); 2068665484d8SDoug Ambrisko 2069665484d8SDoug Ambrisko return (0); 20704799d485SKashyap D Desai 20714799d485SKashyap D Desai ABORT: 20724799d485SKashyap D Desai return (1); 2073665484d8SDoug Ambrisko } 2074665484d8SDoug Ambrisko 2075a688fcd0SKashyap D Desai /** 2076a688fcd0SKashyap D Desai * megasas_setup_jbod_map - setup jbod map for FP seq_number. 2077a688fcd0SKashyap D Desai * @sc: Adapter soft state 2078a688fcd0SKashyap D Desai * 2079a688fcd0SKashyap D Desai * Return 0 on success. 2080a688fcd0SKashyap D Desai */ 2081a688fcd0SKashyap D Desai void 2082a688fcd0SKashyap D Desai megasas_setup_jbod_map(struct mrsas_softc *sc) 2083a688fcd0SKashyap D Desai { 2084a688fcd0SKashyap D Desai int i; 2085a688fcd0SKashyap D Desai uint32_t pd_seq_map_sz; 2086a688fcd0SKashyap D Desai 2087a688fcd0SKashyap D Desai pd_seq_map_sz = sizeof(struct MR_PD_CFG_SEQ_NUM_SYNC) + 2088a688fcd0SKashyap D Desai (sizeof(struct MR_PD_CFG_SEQ) * (MAX_PHYSICAL_DEVICES - 1)); 2089a688fcd0SKashyap D Desai 2090a688fcd0SKashyap D Desai if (!sc->ctrl_info->adapterOperations3.useSeqNumJbodFP) { 2091a688fcd0SKashyap D Desai sc->use_seqnum_jbod_fp = 0; 2092a688fcd0SKashyap D Desai return; 2093a688fcd0SKashyap D Desai } 2094a688fcd0SKashyap D Desai if (sc->jbodmap_mem[0]) 2095a688fcd0SKashyap D Desai goto skip_alloc; 2096a688fcd0SKashyap D Desai 2097a688fcd0SKashyap D Desai for (i = 0; i < 2; i++) { 2098a688fcd0SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 2099a688fcd0SKashyap D Desai 4, 0, 2100a688fcd0SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 2101a688fcd0SKashyap D Desai BUS_SPACE_MAXADDR, 2102a688fcd0SKashyap D Desai NULL, NULL, 2103a688fcd0SKashyap D Desai pd_seq_map_sz, 2104a688fcd0SKashyap D Desai 1, 2105a688fcd0SKashyap D Desai pd_seq_map_sz, 2106a688fcd0SKashyap D Desai BUS_DMA_ALLOCNOW, 2107a688fcd0SKashyap D Desai NULL, NULL, 2108a688fcd0SKashyap D Desai &sc->jbodmap_tag[i])) { 2109a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, 2110a688fcd0SKashyap D Desai "Cannot allocate jbod map tag.\n"); 2111a688fcd0SKashyap D Desai return; 2112a688fcd0SKashyap D Desai } 2113a688fcd0SKashyap D Desai if (bus_dmamem_alloc(sc->jbodmap_tag[i], 2114a688fcd0SKashyap D Desai (void **)&sc->jbodmap_mem[i], 2115a688fcd0SKashyap D Desai BUS_DMA_NOWAIT, &sc->jbodmap_dmamap[i])) { 2116a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, 2117a688fcd0SKashyap D Desai "Cannot allocate jbod map memory.\n"); 2118a688fcd0SKashyap D Desai return; 2119a688fcd0SKashyap D Desai } 2120a688fcd0SKashyap D Desai bzero(sc->jbodmap_mem[i], pd_seq_map_sz); 2121a688fcd0SKashyap D Desai 2122a688fcd0SKashyap D Desai if (bus_dmamap_load(sc->jbodmap_tag[i], sc->jbodmap_dmamap[i], 2123a688fcd0SKashyap D Desai sc->jbodmap_mem[i], pd_seq_map_sz, 2124a688fcd0SKashyap D Desai mrsas_addr_cb, &sc->jbodmap_phys_addr[i], 2125a688fcd0SKashyap D Desai BUS_DMA_NOWAIT)) { 2126a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, "Cannot load jbod map memory.\n"); 2127a688fcd0SKashyap D Desai return; 2128a688fcd0SKashyap D Desai } 2129a688fcd0SKashyap D Desai if (!sc->jbodmap_mem[i]) { 2130a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, 2131a688fcd0SKashyap D Desai "Cannot allocate memory for jbod map.\n"); 2132a688fcd0SKashyap D Desai sc->use_seqnum_jbod_fp = 0; 2133a688fcd0SKashyap D Desai return; 2134a688fcd0SKashyap D Desai } 2135a688fcd0SKashyap D Desai } 2136a688fcd0SKashyap D Desai 2137a688fcd0SKashyap D Desai skip_alloc: 2138a688fcd0SKashyap D Desai if (!megasas_sync_pd_seq_num(sc, false) && 2139a688fcd0SKashyap D Desai !megasas_sync_pd_seq_num(sc, true)) 2140a688fcd0SKashyap D Desai sc->use_seqnum_jbod_fp = 1; 2141a688fcd0SKashyap D Desai else 2142a688fcd0SKashyap D Desai sc->use_seqnum_jbod_fp = 0; 2143a688fcd0SKashyap D Desai 2144a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, "Jbod map is supported\n"); 2145a688fcd0SKashyap D Desai } 2146a688fcd0SKashyap D Desai 21478e727371SKashyap D Desai /* 2148665484d8SDoug Ambrisko * mrsas_init_fw: Initialize Firmware 2149665484d8SDoug Ambrisko * input: Adapter soft state 2150665484d8SDoug Ambrisko * 21518e727371SKashyap D Desai * Calls transition_to_ready() to make sure Firmware is in operational state and 21528e727371SKashyap D Desai * calls mrsas_init_adapter() to send IOC_INIT command to Firmware. It 21538e727371SKashyap D Desai * issues internal commands to get the controller info after the IOC_INIT 21548e727371SKashyap D Desai * command response is received by Firmware. Note: code relating to 21558e727371SKashyap D Desai * get_pdlist, get_ld_list and max_sectors are currently not being used, it 21568e727371SKashyap D Desai * is left here as placeholder. 2157665484d8SDoug Ambrisko */ 21588e727371SKashyap D Desai static int 21598e727371SKashyap D Desai mrsas_init_fw(struct mrsas_softc *sc) 2160665484d8SDoug Ambrisko { 2161d18d1b47SKashyap D Desai 2162d18d1b47SKashyap D Desai int ret, loop, ocr = 0; 2163665484d8SDoug Ambrisko u_int32_t max_sectors_1; 2164665484d8SDoug Ambrisko u_int32_t max_sectors_2; 2165665484d8SDoug Ambrisko u_int32_t tmp_sectors; 21664ad83576SKashyap D Desai u_int32_t scratch_pad_2, scratch_pad_3; 2167d18d1b47SKashyap D Desai int msix_enable = 0; 2168d18d1b47SKashyap D Desai int fw_msix_count = 0; 2169665484d8SDoug Ambrisko 2170665484d8SDoug Ambrisko /* Make sure Firmware is ready */ 2171665484d8SDoug Ambrisko ret = mrsas_transition_to_ready(sc, ocr); 2172665484d8SDoug Ambrisko if (ret != SUCCESS) { 2173665484d8SDoug Ambrisko return (ret); 2174665484d8SDoug Ambrisko } 21754ad83576SKashyap D Desai if (sc->is_ventura) { 21764ad83576SKashyap D Desai scratch_pad_3 = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad_3)); 21774ad83576SKashyap D Desai #if VD_EXT_DEBUG 21784ad83576SKashyap D Desai device_printf(sc->mrsas_dev, "scratch_pad_3 0x%x\n", scratch_pad_3); 21794ad83576SKashyap D Desai #endif 21804ad83576SKashyap D Desai sc->maxRaidMapSize = ((scratch_pad_3 >> 21814ad83576SKashyap D Desai MR_MAX_RAID_MAP_SIZE_OFFSET_SHIFT) & 21824ad83576SKashyap D Desai MR_MAX_RAID_MAP_SIZE_MASK); 21834ad83576SKashyap D Desai } 2184d18d1b47SKashyap D Desai /* MSI-x index 0- reply post host index register */ 2185d18d1b47SKashyap D Desai sc->msix_reg_offset[0] = MPI2_REPLY_POST_HOST_INDEX_OFFSET; 2186d18d1b47SKashyap D Desai /* Check if MSI-X is supported while in ready state */ 2187d18d1b47SKashyap D Desai msix_enable = (mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad)) & 0x4000000) >> 0x1a; 2188d18d1b47SKashyap D Desai 2189d18d1b47SKashyap D Desai if (msix_enable) { 2190d18d1b47SKashyap D Desai scratch_pad_2 = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2191d18d1b47SKashyap D Desai outbound_scratch_pad_2)); 2192d18d1b47SKashyap D Desai 2193d18d1b47SKashyap D Desai /* Check max MSI-X vectors */ 2194d18d1b47SKashyap D Desai if (sc->device_id == MRSAS_TBOLT) { 2195d18d1b47SKashyap D Desai sc->msix_vectors = (scratch_pad_2 2196d18d1b47SKashyap D Desai & MR_MAX_REPLY_QUEUES_OFFSET) + 1; 2197d18d1b47SKashyap D Desai fw_msix_count = sc->msix_vectors; 2198d18d1b47SKashyap D Desai } else { 2199d18d1b47SKashyap D Desai /* Invader/Fury supports 96 MSI-X vectors */ 2200d18d1b47SKashyap D Desai sc->msix_vectors = ((scratch_pad_2 2201d18d1b47SKashyap D Desai & MR_MAX_REPLY_QUEUES_EXT_OFFSET) 2202d18d1b47SKashyap D Desai >> MR_MAX_REPLY_QUEUES_EXT_OFFSET_SHIFT) + 1; 2203d18d1b47SKashyap D Desai fw_msix_count = sc->msix_vectors; 2204d18d1b47SKashyap D Desai 22057aade8bfSKashyap D Desai if ((sc->mrsas_gen3_ctrl && (sc->msix_vectors > 8)) || 22067aade8bfSKashyap D Desai (sc->is_ventura && (sc->msix_vectors > 16))) 22077aade8bfSKashyap D Desai sc->msix_combined = true; 22087aade8bfSKashyap D Desai /* 22097aade8bfSKashyap D Desai * Save 1-15 reply post index 22107aade8bfSKashyap D Desai * address to local memory Index 0 22117aade8bfSKashyap D Desai * is already saved from reg offset 22127aade8bfSKashyap D Desai * MPI2_REPLY_POST_HOST_INDEX_OFFSET 22137aade8bfSKashyap D Desai */ 2214d18d1b47SKashyap D Desai for (loop = 1; loop < MR_MAX_MSIX_REG_ARRAY; 2215d18d1b47SKashyap D Desai loop++) { 2216d18d1b47SKashyap D Desai sc->msix_reg_offset[loop] = 2217d18d1b47SKashyap D Desai MPI2_SUP_REPLY_POST_HOST_INDEX_OFFSET + 2218d18d1b47SKashyap D Desai (loop * 0x10); 2219d18d1b47SKashyap D Desai } 2220d18d1b47SKashyap D Desai } 2221d18d1b47SKashyap D Desai 2222d18d1b47SKashyap D Desai /* Don't bother allocating more MSI-X vectors than cpus */ 2223d18d1b47SKashyap D Desai sc->msix_vectors = min(sc->msix_vectors, 2224d18d1b47SKashyap D Desai mp_ncpus); 2225d18d1b47SKashyap D Desai 2226d18d1b47SKashyap D Desai /* Allocate MSI-x vectors */ 2227d18d1b47SKashyap D Desai if (mrsas_allocate_msix(sc) == SUCCESS) 2228d18d1b47SKashyap D Desai sc->msix_enable = 1; 2229d18d1b47SKashyap D Desai else 2230d18d1b47SKashyap D Desai sc->msix_enable = 0; 2231d18d1b47SKashyap D Desai 2232d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "FW supports <%d> MSIX vector," 2233d18d1b47SKashyap D Desai "Online CPU %d Current MSIX <%d>\n", 2234d18d1b47SKashyap D Desai fw_msix_count, mp_ncpus, sc->msix_vectors); 2235d18d1b47SKashyap D Desai } 22367aade8bfSKashyap D Desai /* 22377aade8bfSKashyap D Desai * MSI-X host index 0 is common for all adapter. 22387aade8bfSKashyap D Desai * It is used for all MPT based Adapters. 22397aade8bfSKashyap D Desai */ 22407aade8bfSKashyap D Desai if (sc->msix_combined) { 22417aade8bfSKashyap D Desai sc->msix_reg_offset[0] = 22427aade8bfSKashyap D Desai MPI2_SUP_REPLY_POST_HOST_INDEX_OFFSET; 22437aade8bfSKashyap D Desai } 2244665484d8SDoug Ambrisko if (mrsas_init_adapter(sc) != SUCCESS) { 2245665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Adapter initialize Fail.\n"); 2246665484d8SDoug Ambrisko return (1); 2247665484d8SDoug Ambrisko } 2248665484d8SDoug Ambrisko /* Allocate internal commands for pass-thru */ 2249665484d8SDoug Ambrisko if (mrsas_alloc_mfi_cmds(sc) != SUCCESS) { 2250665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Allocate MFI cmd failed.\n"); 2251665484d8SDoug Ambrisko return (1); 2252665484d8SDoug Ambrisko } 2253af51c29fSKashyap D Desai sc->ctrl_info = malloc(sizeof(struct mrsas_ctrl_info), M_MRSAS, M_NOWAIT); 2254af51c29fSKashyap D Desai if (!sc->ctrl_info) { 2255af51c29fSKashyap D Desai device_printf(sc->mrsas_dev, "Malloc for ctrl_info failed.\n"); 2256af51c29fSKashyap D Desai return (1); 2257af51c29fSKashyap D Desai } 22584799d485SKashyap D Desai /* 22598e727371SKashyap D Desai * Get the controller info from FW, so that the MAX VD support 22608e727371SKashyap D Desai * availability can be decided. 22614799d485SKashyap D Desai */ 2262af51c29fSKashyap D Desai if (mrsas_get_ctrl_info(sc)) { 22634799d485SKashyap D Desai device_printf(sc->mrsas_dev, "Unable to get FW ctrl_info.\n"); 2264af51c29fSKashyap D Desai return (1); 22654799d485SKashyap D Desai } 226677cf7df8SKashyap D Desai sc->secure_jbod_support = 2267af51c29fSKashyap D Desai (u_int8_t)sc->ctrl_info->adapterOperations3.supportSecurityonJBOD; 226877cf7df8SKashyap D Desai 226977cf7df8SKashyap D Desai if (sc->secure_jbod_support) 227077cf7df8SKashyap D Desai device_printf(sc->mrsas_dev, "FW supports SED \n"); 227177cf7df8SKashyap D Desai 2272a688fcd0SKashyap D Desai if (sc->use_seqnum_jbod_fp) 2273a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, "FW supports JBOD Map \n"); 2274a688fcd0SKashyap D Desai 2275665484d8SDoug Ambrisko if (mrsas_setup_raidmap(sc) != SUCCESS) { 2276a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, "Error: RAID map setup FAILED !!! " 2277a688fcd0SKashyap D Desai "There seems to be some problem in the controller\n" 2278a688fcd0SKashyap D Desai "Please contact to the SUPPORT TEAM if the problem persists\n"); 2279665484d8SDoug Ambrisko } 2280a688fcd0SKashyap D Desai megasas_setup_jbod_map(sc); 2281a688fcd0SKashyap D Desai 2282665484d8SDoug Ambrisko /* For pass-thru, get PD/LD list and controller info */ 22834799d485SKashyap D Desai memset(sc->pd_list, 0, 22844799d485SKashyap D Desai MRSAS_MAX_PD * sizeof(struct mrsas_pd_list)); 2285a688fcd0SKashyap D Desai if (mrsas_get_pd_list(sc) != SUCCESS) { 2286a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, "Get PD list failed.\n"); 2287a688fcd0SKashyap D Desai return (1); 2288a688fcd0SKashyap D Desai } 22894799d485SKashyap D Desai memset(sc->ld_ids, 0xff, MRSAS_MAX_LD_IDS); 2290a688fcd0SKashyap D Desai if (mrsas_get_ld_list(sc) != SUCCESS) { 2291a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, "Get LD lsit failed.\n"); 2292a688fcd0SKashyap D Desai return (1); 2293a688fcd0SKashyap D Desai } 2294665484d8SDoug Ambrisko /* 22958e727371SKashyap D Desai * Compute the max allowed sectors per IO: The controller info has 22968e727371SKashyap D Desai * two limits on max sectors. Driver should use the minimum of these 22978e727371SKashyap D Desai * two. 2298665484d8SDoug Ambrisko * 2299665484d8SDoug Ambrisko * 1 << stripe_sz_ops.min = max sectors per strip 2300665484d8SDoug Ambrisko * 23018e727371SKashyap D Desai * Note that older firmwares ( < FW ver 30) didn't report information to 23028e727371SKashyap D Desai * calculate max_sectors_1. So the number ended up as zero always. 2303665484d8SDoug Ambrisko */ 2304665484d8SDoug Ambrisko tmp_sectors = 0; 2305af51c29fSKashyap D Desai max_sectors_1 = (1 << sc->ctrl_info->stripe_sz_ops.min) * 2306af51c29fSKashyap D Desai sc->ctrl_info->max_strips_per_io; 2307af51c29fSKashyap D Desai max_sectors_2 = sc->ctrl_info->max_request_size; 2308665484d8SDoug Ambrisko tmp_sectors = min(max_sectors_1, max_sectors_2); 23094799d485SKashyap D Desai sc->max_sectors_per_req = sc->max_num_sge * MRSAS_PAGE_SIZE / 512; 23104799d485SKashyap D Desai 23114799d485SKashyap D Desai if (tmp_sectors && (sc->max_sectors_per_req > tmp_sectors)) 23124799d485SKashyap D Desai sc->max_sectors_per_req = tmp_sectors; 23134799d485SKashyap D Desai 2314665484d8SDoug Ambrisko sc->disableOnlineCtrlReset = 2315af51c29fSKashyap D Desai sc->ctrl_info->properties.OnOffProperties.disableOnlineCtrlReset; 2316665484d8SDoug Ambrisko sc->UnevenSpanSupport = 2317af51c29fSKashyap D Desai sc->ctrl_info->adapterOperations2.supportUnevenSpans; 2318665484d8SDoug Ambrisko if (sc->UnevenSpanSupport) { 23198e727371SKashyap D Desai device_printf(sc->mrsas_dev, "FW supports: UnevenSpanSupport=%x\n\n", 2320665484d8SDoug Ambrisko sc->UnevenSpanSupport); 23214799d485SKashyap D Desai 2322665484d8SDoug Ambrisko if (MR_ValidateMapInfo(sc)) 2323665484d8SDoug Ambrisko sc->fast_path_io = 1; 2324665484d8SDoug Ambrisko else 2325665484d8SDoug Ambrisko sc->fast_path_io = 0; 2326665484d8SDoug Ambrisko } 2327665484d8SDoug Ambrisko return (0); 2328665484d8SDoug Ambrisko } 2329665484d8SDoug Ambrisko 23308e727371SKashyap D Desai /* 2331665484d8SDoug Ambrisko * mrsas_init_adapter: Initializes the adapter/controller 2332665484d8SDoug Ambrisko * input: Adapter soft state 2333665484d8SDoug Ambrisko * 2334665484d8SDoug Ambrisko * Prepares for the issuing of the IOC Init cmd to FW for initializing the 2335665484d8SDoug Ambrisko * ROC/controller. The FW register is read to determined the number of 2336665484d8SDoug Ambrisko * commands that is supported. All memory allocations for IO is based on 2337665484d8SDoug Ambrisko * max_cmd. Appropriate calculations are performed in this function. 2338665484d8SDoug Ambrisko */ 23398e727371SKashyap D Desai int 23408e727371SKashyap D Desai mrsas_init_adapter(struct mrsas_softc *sc) 2341665484d8SDoug Ambrisko { 2342665484d8SDoug Ambrisko uint32_t status; 23433a3fc6cbSKashyap D Desai u_int32_t max_cmd, scratch_pad_2; 2344665484d8SDoug Ambrisko int ret; 2345d18d1b47SKashyap D Desai int i = 0; 2346665484d8SDoug Ambrisko 2347665484d8SDoug Ambrisko /* Read FW status register */ 2348665484d8SDoug Ambrisko status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad)); 2349665484d8SDoug Ambrisko 2350665484d8SDoug Ambrisko /* Get operational params from status register */ 2351665484d8SDoug Ambrisko sc->max_fw_cmds = status & MRSAS_FWSTATE_MAXCMD_MASK; 2352665484d8SDoug Ambrisko 2353665484d8SDoug Ambrisko /* Decrement the max supported by 1, to correlate with FW */ 2354665484d8SDoug Ambrisko sc->max_fw_cmds = sc->max_fw_cmds - 1; 2355665484d8SDoug Ambrisko max_cmd = sc->max_fw_cmds; 2356665484d8SDoug Ambrisko 2357665484d8SDoug Ambrisko /* Determine allocation size of command frames */ 23582f863eb8SKashyap D Desai sc->reply_q_depth = ((max_cmd + 1 + 15) / 16 * 16) * 2; 2359665484d8SDoug Ambrisko sc->request_alloc_sz = sizeof(MRSAS_REQUEST_DESCRIPTOR_UNION) * max_cmd; 2360665484d8SDoug Ambrisko sc->reply_alloc_sz = sizeof(MPI2_REPLY_DESCRIPTORS_UNION) * (sc->reply_q_depth); 2361665484d8SDoug Ambrisko sc->io_frames_alloc_sz = MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE + (MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE * (max_cmd + 1)); 23623a3fc6cbSKashyap D Desai scratch_pad_2 = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 23633a3fc6cbSKashyap D Desai outbound_scratch_pad_2)); 23643a3fc6cbSKashyap D Desai /* 23653a3fc6cbSKashyap D Desai * If scratch_pad_2 & MEGASAS_MAX_CHAIN_SIZE_UNITS_MASK is set, 23663a3fc6cbSKashyap D Desai * Firmware support extended IO chain frame which is 4 time more 23673a3fc6cbSKashyap D Desai * than legacy Firmware. Legacy Firmware - Frame size is (8 * 128) = 23683a3fc6cbSKashyap D Desai * 1K 1M IO Firmware - Frame size is (8 * 128 * 4) = 4K 23693a3fc6cbSKashyap D Desai */ 23703a3fc6cbSKashyap D Desai if (scratch_pad_2 & MEGASAS_MAX_CHAIN_SIZE_UNITS_MASK) 23713a3fc6cbSKashyap D Desai sc->max_chain_frame_sz = 23723a3fc6cbSKashyap D Desai ((scratch_pad_2 & MEGASAS_MAX_CHAIN_SIZE_MASK) >> 5) 23733a3fc6cbSKashyap D Desai * MEGASAS_1MB_IO; 23743a3fc6cbSKashyap D Desai else 23753a3fc6cbSKashyap D Desai sc->max_chain_frame_sz = 23763a3fc6cbSKashyap D Desai ((scratch_pad_2 & MEGASAS_MAX_CHAIN_SIZE_MASK) >> 5) 23773a3fc6cbSKashyap D Desai * MEGASAS_256K_IO; 23783a3fc6cbSKashyap D Desai 23793a3fc6cbSKashyap D Desai sc->chain_frames_alloc_sz = sc->max_chain_frame_sz * max_cmd; 2380665484d8SDoug Ambrisko sc->max_sge_in_main_msg = (MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE - 2381665484d8SDoug Ambrisko offsetof(MRSAS_RAID_SCSI_IO_REQUEST, SGL)) / 16; 2382665484d8SDoug Ambrisko 23833a3fc6cbSKashyap D Desai sc->max_sge_in_chain = sc->max_chain_frame_sz / sizeof(MPI2_SGE_IO_UNION); 2384665484d8SDoug Ambrisko sc->max_num_sge = sc->max_sge_in_main_msg + sc->max_sge_in_chain - 2; 2385665484d8SDoug Ambrisko 23863a3fc6cbSKashyap D Desai mrsas_dprint(sc, MRSAS_INFO, "Avago Debug: MAX sge 0x%X MAX chain frame size 0x%X \n", 23873a3fc6cbSKashyap D Desai sc->max_num_sge, sc->max_chain_frame_sz); 23883a3fc6cbSKashyap D Desai 2389665484d8SDoug Ambrisko /* Used for pass thru MFI frame (DCMD) */ 2390665484d8SDoug Ambrisko sc->chain_offset_mfi_pthru = offsetof(MRSAS_RAID_SCSI_IO_REQUEST, SGL) / 16; 2391665484d8SDoug Ambrisko 2392665484d8SDoug Ambrisko sc->chain_offset_io_request = (MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE - 2393665484d8SDoug Ambrisko sizeof(MPI2_SGE_IO_UNION)) / 16; 2394665484d8SDoug Ambrisko 2395d18d1b47SKashyap D Desai int count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 23968e727371SKashyap D Desai 2397d18d1b47SKashyap D Desai for (i = 0; i < count; i++) 2398d18d1b47SKashyap D Desai sc->last_reply_idx[i] = 0; 2399665484d8SDoug Ambrisko 2400665484d8SDoug Ambrisko ret = mrsas_alloc_mem(sc); 2401665484d8SDoug Ambrisko if (ret != SUCCESS) 2402665484d8SDoug Ambrisko return (ret); 2403665484d8SDoug Ambrisko 2404665484d8SDoug Ambrisko ret = mrsas_alloc_mpt_cmds(sc); 2405665484d8SDoug Ambrisko if (ret != SUCCESS) 2406665484d8SDoug Ambrisko return (ret); 2407665484d8SDoug Ambrisko 2408665484d8SDoug Ambrisko ret = mrsas_ioc_init(sc); 2409665484d8SDoug Ambrisko if (ret != SUCCESS) 2410665484d8SDoug Ambrisko return (ret); 2411665484d8SDoug Ambrisko 2412665484d8SDoug Ambrisko return (0); 2413665484d8SDoug Ambrisko } 2414665484d8SDoug Ambrisko 24158e727371SKashyap D Desai /* 2416665484d8SDoug Ambrisko * mrsas_alloc_ioc_cmd: Allocates memory for IOC Init command 2417665484d8SDoug Ambrisko * input: Adapter soft state 2418665484d8SDoug Ambrisko * 2419665484d8SDoug Ambrisko * Allocates for the IOC Init cmd to FW to initialize the ROC/controller. 2420665484d8SDoug Ambrisko */ 24218e727371SKashyap D Desai int 24228e727371SKashyap D Desai mrsas_alloc_ioc_cmd(struct mrsas_softc *sc) 2423665484d8SDoug Ambrisko { 2424665484d8SDoug Ambrisko int ioc_init_size; 2425665484d8SDoug Ambrisko 2426665484d8SDoug Ambrisko /* Allocate IOC INIT command */ 2427665484d8SDoug Ambrisko ioc_init_size = 1024 + sizeof(MPI2_IOC_INIT_REQUEST); 24288e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 24298e727371SKashyap D Desai 1, 0, 24308e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 24318e727371SKashyap D Desai BUS_SPACE_MAXADDR, 24328e727371SKashyap D Desai NULL, NULL, 24338e727371SKashyap D Desai ioc_init_size, 24348e727371SKashyap D Desai 1, 24358e727371SKashyap D Desai ioc_init_size, 24368e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 24378e727371SKashyap D Desai NULL, NULL, 2438665484d8SDoug Ambrisko &sc->ioc_init_tag)) { 2439665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate ioc init tag\n"); 2440665484d8SDoug Ambrisko return (ENOMEM); 2441665484d8SDoug Ambrisko } 2442665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->ioc_init_tag, (void **)&sc->ioc_init_mem, 2443665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->ioc_init_dmamap)) { 2444665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate ioc init cmd mem\n"); 2445665484d8SDoug Ambrisko return (ENOMEM); 2446665484d8SDoug Ambrisko } 2447665484d8SDoug Ambrisko bzero(sc->ioc_init_mem, ioc_init_size); 2448665484d8SDoug Ambrisko if (bus_dmamap_load(sc->ioc_init_tag, sc->ioc_init_dmamap, 2449665484d8SDoug Ambrisko sc->ioc_init_mem, ioc_init_size, mrsas_addr_cb, 2450665484d8SDoug Ambrisko &sc->ioc_init_phys_mem, BUS_DMA_NOWAIT)) { 2451665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load ioc init cmd mem\n"); 2452665484d8SDoug Ambrisko return (ENOMEM); 2453665484d8SDoug Ambrisko } 2454665484d8SDoug Ambrisko return (0); 2455665484d8SDoug Ambrisko } 2456665484d8SDoug Ambrisko 24578e727371SKashyap D Desai /* 2458665484d8SDoug Ambrisko * mrsas_free_ioc_cmd: Allocates memory for IOC Init command 2459665484d8SDoug Ambrisko * input: Adapter soft state 2460665484d8SDoug Ambrisko * 2461665484d8SDoug Ambrisko * Deallocates memory of the IOC Init cmd. 2462665484d8SDoug Ambrisko */ 24638e727371SKashyap D Desai void 24648e727371SKashyap D Desai mrsas_free_ioc_cmd(struct mrsas_softc *sc) 2465665484d8SDoug Ambrisko { 2466665484d8SDoug Ambrisko if (sc->ioc_init_phys_mem) 2467665484d8SDoug Ambrisko bus_dmamap_unload(sc->ioc_init_tag, sc->ioc_init_dmamap); 2468665484d8SDoug Ambrisko if (sc->ioc_init_mem != NULL) 2469665484d8SDoug Ambrisko bus_dmamem_free(sc->ioc_init_tag, sc->ioc_init_mem, sc->ioc_init_dmamap); 2470665484d8SDoug Ambrisko if (sc->ioc_init_tag != NULL) 2471665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->ioc_init_tag); 2472665484d8SDoug Ambrisko } 2473665484d8SDoug Ambrisko 24748e727371SKashyap D Desai /* 2475665484d8SDoug Ambrisko * mrsas_ioc_init: Sends IOC Init command to FW 2476665484d8SDoug Ambrisko * input: Adapter soft state 2477665484d8SDoug Ambrisko * 2478665484d8SDoug Ambrisko * Issues the IOC Init cmd to FW to initialize the ROC/controller. 2479665484d8SDoug Ambrisko */ 24808e727371SKashyap D Desai int 24818e727371SKashyap D Desai mrsas_ioc_init(struct mrsas_softc *sc) 2482665484d8SDoug Ambrisko { 2483665484d8SDoug Ambrisko struct mrsas_init_frame *init_frame; 2484665484d8SDoug Ambrisko pMpi2IOCInitRequest_t IOCInitMsg; 2485665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION req_desc; 2486665484d8SDoug Ambrisko u_int8_t max_wait = MRSAS_IOC_INIT_WAIT_TIME; 2487665484d8SDoug Ambrisko bus_addr_t phys_addr; 2488665484d8SDoug Ambrisko int i, retcode = 0; 2489d993dd83SKashyap D Desai u_int32_t scratch_pad_2; 2490665484d8SDoug Ambrisko 2491665484d8SDoug Ambrisko /* Allocate memory for the IOC INIT command */ 2492665484d8SDoug Ambrisko if (mrsas_alloc_ioc_cmd(sc)) { 2493665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate IOC command.\n"); 2494665484d8SDoug Ambrisko return (1); 2495665484d8SDoug Ambrisko } 2496d993dd83SKashyap D Desai 2497d993dd83SKashyap D Desai if (!sc->block_sync_cache) { 2498d993dd83SKashyap D Desai scratch_pad_2 = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2499d993dd83SKashyap D Desai outbound_scratch_pad_2)); 2500d993dd83SKashyap D Desai sc->fw_sync_cache_support = (scratch_pad_2 & 2501d993dd83SKashyap D Desai MR_CAN_HANDLE_SYNC_CACHE_OFFSET) ? 1 : 0; 2502d993dd83SKashyap D Desai } 2503d993dd83SKashyap D Desai 2504665484d8SDoug Ambrisko IOCInitMsg = (pMpi2IOCInitRequest_t)(((char *)sc->ioc_init_mem) + 1024); 2505665484d8SDoug Ambrisko IOCInitMsg->Function = MPI2_FUNCTION_IOC_INIT; 2506665484d8SDoug Ambrisko IOCInitMsg->WhoInit = MPI2_WHOINIT_HOST_DRIVER; 2507665484d8SDoug Ambrisko IOCInitMsg->MsgVersion = MPI2_VERSION; 2508665484d8SDoug Ambrisko IOCInitMsg->HeaderVersion = MPI2_HEADER_VERSION; 2509665484d8SDoug Ambrisko IOCInitMsg->SystemRequestFrameSize = MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE / 4; 2510665484d8SDoug Ambrisko IOCInitMsg->ReplyDescriptorPostQueueDepth = sc->reply_q_depth; 2511665484d8SDoug Ambrisko IOCInitMsg->ReplyDescriptorPostQueueAddress = sc->reply_desc_phys_addr; 2512665484d8SDoug Ambrisko IOCInitMsg->SystemRequestFrameBaseAddress = sc->io_request_phys_addr; 2513d18d1b47SKashyap D Desai IOCInitMsg->HostMSIxVectors = (sc->msix_vectors > 0 ? sc->msix_vectors : 0); 2514665484d8SDoug Ambrisko 2515665484d8SDoug Ambrisko init_frame = (struct mrsas_init_frame *)sc->ioc_init_mem; 2516665484d8SDoug Ambrisko init_frame->cmd = MFI_CMD_INIT; 2517665484d8SDoug Ambrisko init_frame->cmd_status = 0xFF; 2518665484d8SDoug Ambrisko init_frame->flags |= MFI_FRAME_DONT_POST_IN_REPLY_QUEUE; 2519665484d8SDoug Ambrisko 2520d18d1b47SKashyap D Desai /* driver support Extended MSIX */ 25217aade8bfSKashyap D Desai if (sc->mrsas_gen3_ctrl || sc->is_ventura) { 2522d18d1b47SKashyap D Desai init_frame->driver_operations. 2523d18d1b47SKashyap D Desai mfi_capabilities.support_additional_msix = 1; 2524d18d1b47SKashyap D Desai } 2525665484d8SDoug Ambrisko if (sc->verbuf_mem) { 2526665484d8SDoug Ambrisko snprintf((char *)sc->verbuf_mem, strlen(MRSAS_VERSION) + 2, "%s\n", 2527665484d8SDoug Ambrisko MRSAS_VERSION); 2528665484d8SDoug Ambrisko init_frame->driver_ver_lo = (bus_addr_t)sc->verbuf_phys_addr; 2529665484d8SDoug Ambrisko init_frame->driver_ver_hi = 0; 2530665484d8SDoug Ambrisko } 253116dc2814SKashyap D Desai init_frame->driver_operations.mfi_capabilities.support_ndrive_r1_lb = 1; 25324799d485SKashyap D Desai init_frame->driver_operations.mfi_capabilities.support_max_255lds = 1; 253377cf7df8SKashyap D Desai init_frame->driver_operations.mfi_capabilities.security_protocol_cmds_fw = 1; 25343a3fc6cbSKashyap D Desai if (sc->max_chain_frame_sz > MEGASAS_CHAIN_FRAME_SZ_MIN) 25353a3fc6cbSKashyap D Desai init_frame->driver_operations.mfi_capabilities.support_ext_io_size = 1; 2536665484d8SDoug Ambrisko phys_addr = (bus_addr_t)sc->ioc_init_phys_mem + 1024; 2537665484d8SDoug Ambrisko init_frame->queue_info_new_phys_addr_lo = phys_addr; 2538665484d8SDoug Ambrisko init_frame->data_xfer_len = sizeof(Mpi2IOCInitRequest_t); 2539665484d8SDoug Ambrisko 2540665484d8SDoug Ambrisko req_desc.addr.Words = (bus_addr_t)sc->ioc_init_phys_mem; 2541665484d8SDoug Ambrisko req_desc.MFAIo.RequestFlags = 2542665484d8SDoug Ambrisko (MRSAS_REQ_DESCRIPT_FLAGS_MFA << MRSAS_REQ_DESCRIPT_FLAGS_TYPE_SHIFT); 2543665484d8SDoug Ambrisko 2544665484d8SDoug Ambrisko mrsas_disable_intr(sc); 2545665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "Issuing IOC INIT command to FW.\n"); 2546665484d8SDoug Ambrisko mrsas_fire_cmd(sc, req_desc.addr.u.low, req_desc.addr.u.high); 2547665484d8SDoug Ambrisko 2548665484d8SDoug Ambrisko /* 2549665484d8SDoug Ambrisko * Poll response timer to wait for Firmware response. While this 2550665484d8SDoug Ambrisko * timer with the DELAY call could block CPU, the time interval for 2551665484d8SDoug Ambrisko * this is only 1 millisecond. 2552665484d8SDoug Ambrisko */ 2553665484d8SDoug Ambrisko if (init_frame->cmd_status == 0xFF) { 2554665484d8SDoug Ambrisko for (i = 0; i < (max_wait * 1000); i++) { 2555665484d8SDoug Ambrisko if (init_frame->cmd_status == 0xFF) 2556665484d8SDoug Ambrisko DELAY(1000); 2557665484d8SDoug Ambrisko else 2558665484d8SDoug Ambrisko break; 2559665484d8SDoug Ambrisko } 2560665484d8SDoug Ambrisko } 2561665484d8SDoug Ambrisko if (init_frame->cmd_status == 0) 2562665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2563665484d8SDoug Ambrisko "IOC INIT response received from FW.\n"); 25648e727371SKashyap D Desai else { 2565665484d8SDoug Ambrisko if (init_frame->cmd_status == 0xFF) 2566665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "IOC Init timed out after %d seconds.\n", max_wait); 2567665484d8SDoug Ambrisko else 2568665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "IOC Init failed, status = 0x%x\n", init_frame->cmd_status); 2569665484d8SDoug Ambrisko retcode = 1; 2570665484d8SDoug Ambrisko } 2571665484d8SDoug Ambrisko 2572665484d8SDoug Ambrisko mrsas_free_ioc_cmd(sc); 2573665484d8SDoug Ambrisko return (retcode); 2574665484d8SDoug Ambrisko } 2575665484d8SDoug Ambrisko 25768e727371SKashyap D Desai /* 2577665484d8SDoug Ambrisko * mrsas_alloc_mpt_cmds: Allocates the command packets 2578665484d8SDoug Ambrisko * input: Adapter instance soft state 2579665484d8SDoug Ambrisko * 2580665484d8SDoug Ambrisko * This function allocates the internal commands for IOs. Each command that is 25818e727371SKashyap D Desai * issued to FW is wrapped in a local data structure called mrsas_mpt_cmd. An 25828e727371SKashyap D Desai * array is allocated with mrsas_mpt_cmd context. The free commands are 2583665484d8SDoug Ambrisko * maintained in a linked list (cmd pool). SMID value range is from 1 to 2584665484d8SDoug Ambrisko * max_fw_cmds. 2585665484d8SDoug Ambrisko */ 25868e727371SKashyap D Desai int 25878e727371SKashyap D Desai mrsas_alloc_mpt_cmds(struct mrsas_softc *sc) 2588665484d8SDoug Ambrisko { 2589665484d8SDoug Ambrisko int i, j; 2590d18d1b47SKashyap D Desai u_int32_t max_cmd, count; 2591665484d8SDoug Ambrisko struct mrsas_mpt_cmd *cmd; 2592665484d8SDoug Ambrisko pMpi2ReplyDescriptorsUnion_t reply_desc; 2593665484d8SDoug Ambrisko u_int32_t offset, chain_offset, sense_offset; 2594665484d8SDoug Ambrisko bus_addr_t io_req_base_phys, chain_frame_base_phys, sense_base_phys; 2595665484d8SDoug Ambrisko u_int8_t *io_req_base, *chain_frame_base, *sense_base; 2596665484d8SDoug Ambrisko 2597665484d8SDoug Ambrisko max_cmd = sc->max_fw_cmds; 2598665484d8SDoug Ambrisko 2599665484d8SDoug Ambrisko sc->req_desc = malloc(sc->request_alloc_sz, M_MRSAS, M_NOWAIT); 2600665484d8SDoug Ambrisko if (!sc->req_desc) { 2601665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Out of memory, cannot alloc req desc\n"); 2602665484d8SDoug Ambrisko return (ENOMEM); 2603665484d8SDoug Ambrisko } 2604665484d8SDoug Ambrisko memset(sc->req_desc, 0, sc->request_alloc_sz); 2605665484d8SDoug Ambrisko 2606665484d8SDoug Ambrisko /* 26078e727371SKashyap D Desai * sc->mpt_cmd_list is an array of struct mrsas_mpt_cmd pointers. 26088e727371SKashyap D Desai * Allocate the dynamic array first and then allocate individual 26098e727371SKashyap D Desai * commands. 2610665484d8SDoug Ambrisko */ 2611ac2fffa4SPedro F. Giffuni sc->mpt_cmd_list = malloc(sizeof(struct mrsas_mpt_cmd *) * max_cmd, M_MRSAS, M_NOWAIT); 2612665484d8SDoug Ambrisko if (!sc->mpt_cmd_list) { 2613665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc memory for mpt_cmd_list.\n"); 2614665484d8SDoug Ambrisko return (ENOMEM); 2615665484d8SDoug Ambrisko } 2616665484d8SDoug Ambrisko memset(sc->mpt_cmd_list, 0, sizeof(struct mrsas_mpt_cmd *) * max_cmd); 2617665484d8SDoug Ambrisko for (i = 0; i < max_cmd; i++) { 2618665484d8SDoug Ambrisko sc->mpt_cmd_list[i] = malloc(sizeof(struct mrsas_mpt_cmd), 2619665484d8SDoug Ambrisko M_MRSAS, M_NOWAIT); 2620665484d8SDoug Ambrisko if (!sc->mpt_cmd_list[i]) { 2621665484d8SDoug Ambrisko for (j = 0; j < i; j++) 2622665484d8SDoug Ambrisko free(sc->mpt_cmd_list[j], M_MRSAS); 2623665484d8SDoug Ambrisko free(sc->mpt_cmd_list, M_MRSAS); 2624665484d8SDoug Ambrisko sc->mpt_cmd_list = NULL; 2625665484d8SDoug Ambrisko return (ENOMEM); 2626665484d8SDoug Ambrisko } 2627665484d8SDoug Ambrisko } 2628665484d8SDoug Ambrisko 2629665484d8SDoug Ambrisko io_req_base = (u_int8_t *)sc->io_request_mem + MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE; 2630665484d8SDoug Ambrisko io_req_base_phys = (bus_addr_t)sc->io_request_phys_addr + MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE; 2631665484d8SDoug Ambrisko chain_frame_base = (u_int8_t *)sc->chain_frame_mem; 2632665484d8SDoug Ambrisko chain_frame_base_phys = (bus_addr_t)sc->chain_frame_phys_addr; 2633665484d8SDoug Ambrisko sense_base = (u_int8_t *)sc->sense_mem; 2634665484d8SDoug Ambrisko sense_base_phys = (bus_addr_t)sc->sense_phys_addr; 2635665484d8SDoug Ambrisko for (i = 0; i < max_cmd; i++) { 2636665484d8SDoug Ambrisko cmd = sc->mpt_cmd_list[i]; 2637665484d8SDoug Ambrisko offset = MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE * i; 26383a3fc6cbSKashyap D Desai chain_offset = sc->max_chain_frame_sz * i; 2639665484d8SDoug Ambrisko sense_offset = MRSAS_SENSE_LEN * i; 2640665484d8SDoug Ambrisko memset(cmd, 0, sizeof(struct mrsas_mpt_cmd)); 2641665484d8SDoug Ambrisko cmd->index = i + 1; 2642665484d8SDoug Ambrisko cmd->ccb_ptr = NULL; 26438bb601acSKashyap D Desai callout_init_mtx(&cmd->cm_callout, &sc->sim_lock, 0); 2644665484d8SDoug Ambrisko cmd->sync_cmd_idx = (u_int32_t)MRSAS_ULONG_MAX; 2645665484d8SDoug Ambrisko cmd->sc = sc; 2646665484d8SDoug Ambrisko cmd->io_request = (MRSAS_RAID_SCSI_IO_REQUEST *) (io_req_base + offset); 2647665484d8SDoug Ambrisko memset(cmd->io_request, 0, sizeof(MRSAS_RAID_SCSI_IO_REQUEST)); 2648665484d8SDoug Ambrisko cmd->io_request_phys_addr = io_req_base_phys + offset; 2649665484d8SDoug Ambrisko cmd->chain_frame = (MPI2_SGE_IO_UNION *) (chain_frame_base + chain_offset); 2650665484d8SDoug Ambrisko cmd->chain_frame_phys_addr = chain_frame_base_phys + chain_offset; 2651665484d8SDoug Ambrisko cmd->sense = sense_base + sense_offset; 2652665484d8SDoug Ambrisko cmd->sense_phys_addr = sense_base_phys + sense_offset; 2653665484d8SDoug Ambrisko if (bus_dmamap_create(sc->data_tag, 0, &cmd->data_dmamap)) { 2654665484d8SDoug Ambrisko return (FAIL); 2655665484d8SDoug Ambrisko } 2656665484d8SDoug Ambrisko TAILQ_INSERT_TAIL(&(sc->mrsas_mpt_cmd_list_head), cmd, next); 2657665484d8SDoug Ambrisko } 2658665484d8SDoug Ambrisko 2659665484d8SDoug Ambrisko /* Initialize reply descriptor array to 0xFFFFFFFF */ 2660665484d8SDoug Ambrisko reply_desc = sc->reply_desc_mem; 2661d18d1b47SKashyap D Desai count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 2662d18d1b47SKashyap D Desai for (i = 0; i < sc->reply_q_depth * count; i++, reply_desc++) { 2663665484d8SDoug Ambrisko reply_desc->Words = MRSAS_ULONG_MAX; 2664665484d8SDoug Ambrisko } 2665665484d8SDoug Ambrisko return (0); 2666665484d8SDoug Ambrisko } 2667665484d8SDoug Ambrisko 26688e727371SKashyap D Desai /* 2669665484d8SDoug Ambrisko * mrsas_fire_cmd: Sends command to FW 2670665484d8SDoug Ambrisko * input: Adapter softstate 2671665484d8SDoug Ambrisko * request descriptor address low 2672665484d8SDoug Ambrisko * request descriptor address high 2673665484d8SDoug Ambrisko * 2674665484d8SDoug Ambrisko * This functions fires the command to Firmware by writing to the 2675665484d8SDoug Ambrisko * inbound_low_queue_port and inbound_high_queue_port. 2676665484d8SDoug Ambrisko */ 26778e727371SKashyap D Desai void 26788e727371SKashyap D Desai mrsas_fire_cmd(struct mrsas_softc *sc, u_int32_t req_desc_lo, 2679665484d8SDoug Ambrisko u_int32_t req_desc_hi) 2680665484d8SDoug Ambrisko { 2681665484d8SDoug Ambrisko mtx_lock(&sc->pci_lock); 2682665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, inbound_low_queue_port), 2683665484d8SDoug Ambrisko req_desc_lo); 2684665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, inbound_high_queue_port), 2685665484d8SDoug Ambrisko req_desc_hi); 2686665484d8SDoug Ambrisko mtx_unlock(&sc->pci_lock); 2687665484d8SDoug Ambrisko } 2688665484d8SDoug Ambrisko 26898e727371SKashyap D Desai /* 26908e727371SKashyap D Desai * mrsas_transition_to_ready: Move FW to Ready state input: 26918e727371SKashyap D Desai * Adapter instance soft state 2692665484d8SDoug Ambrisko * 26938e727371SKashyap D Desai * During the initialization, FW passes can potentially be in any one of several 26948e727371SKashyap D Desai * possible states. If the FW in operational, waiting-for-handshake states, 26958e727371SKashyap D Desai * driver must take steps to bring it to ready state. Otherwise, it has to 26968e727371SKashyap D Desai * wait for the ready state. 2697665484d8SDoug Ambrisko */ 26988e727371SKashyap D Desai int 26998e727371SKashyap D Desai mrsas_transition_to_ready(struct mrsas_softc *sc, int ocr) 2700665484d8SDoug Ambrisko { 2701665484d8SDoug Ambrisko int i; 2702665484d8SDoug Ambrisko u_int8_t max_wait; 2703665484d8SDoug Ambrisko u_int32_t val, fw_state; 2704665484d8SDoug Ambrisko u_int32_t cur_state; 2705665484d8SDoug Ambrisko u_int32_t abs_state, curr_abs_state; 2706665484d8SDoug Ambrisko 2707665484d8SDoug Ambrisko val = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad)); 2708665484d8SDoug Ambrisko fw_state = val & MFI_STATE_MASK; 2709665484d8SDoug Ambrisko max_wait = MRSAS_RESET_WAIT_TIME; 2710665484d8SDoug Ambrisko 2711665484d8SDoug Ambrisko if (fw_state != MFI_STATE_READY) 2712665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Waiting for FW to come to ready state\n"); 2713665484d8SDoug Ambrisko 2714665484d8SDoug Ambrisko while (fw_state != MFI_STATE_READY) { 2715665484d8SDoug Ambrisko abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad)); 2716665484d8SDoug Ambrisko switch (fw_state) { 2717665484d8SDoug Ambrisko case MFI_STATE_FAULT: 2718665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "FW is in FAULT state!!\n"); 2719665484d8SDoug Ambrisko if (ocr) { 2720665484d8SDoug Ambrisko cur_state = MFI_STATE_FAULT; 2721665484d8SDoug Ambrisko break; 27228e727371SKashyap D Desai } else 2723665484d8SDoug Ambrisko return -ENODEV; 2724665484d8SDoug Ambrisko case MFI_STATE_WAIT_HANDSHAKE: 2725665484d8SDoug Ambrisko /* Set the CLR bit in inbound doorbell */ 2726665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell), 2727665484d8SDoug Ambrisko MFI_INIT_CLEAR_HANDSHAKE | MFI_INIT_HOTPLUG); 2728665484d8SDoug Ambrisko cur_state = MFI_STATE_WAIT_HANDSHAKE; 2729665484d8SDoug Ambrisko break; 2730665484d8SDoug Ambrisko case MFI_STATE_BOOT_MESSAGE_PENDING: 2731665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell), 2732665484d8SDoug Ambrisko MFI_INIT_HOTPLUG); 2733665484d8SDoug Ambrisko cur_state = MFI_STATE_BOOT_MESSAGE_PENDING; 2734665484d8SDoug Ambrisko break; 2735665484d8SDoug Ambrisko case MFI_STATE_OPERATIONAL: 27368e727371SKashyap D Desai /* 27378e727371SKashyap D Desai * Bring it to READY state; assuming max wait 10 27388e727371SKashyap D Desai * secs 27398e727371SKashyap D Desai */ 2740665484d8SDoug Ambrisko mrsas_disable_intr(sc); 2741665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell), MFI_RESET_FLAGS); 2742665484d8SDoug Ambrisko for (i = 0; i < max_wait * 1000; i++) { 2743665484d8SDoug Ambrisko if (mrsas_read_reg(sc, offsetof(mrsas_reg_set, doorbell)) & 1) 2744665484d8SDoug Ambrisko DELAY(1000); 2745665484d8SDoug Ambrisko else 2746665484d8SDoug Ambrisko break; 2747665484d8SDoug Ambrisko } 2748665484d8SDoug Ambrisko cur_state = MFI_STATE_OPERATIONAL; 2749665484d8SDoug Ambrisko break; 2750665484d8SDoug Ambrisko case MFI_STATE_UNDEFINED: 27518e727371SKashyap D Desai /* 27528e727371SKashyap D Desai * This state should not last for more than 2 27538e727371SKashyap D Desai * seconds 27548e727371SKashyap D Desai */ 2755665484d8SDoug Ambrisko cur_state = MFI_STATE_UNDEFINED; 2756665484d8SDoug Ambrisko break; 2757665484d8SDoug Ambrisko case MFI_STATE_BB_INIT: 2758665484d8SDoug Ambrisko cur_state = MFI_STATE_BB_INIT; 2759665484d8SDoug Ambrisko break; 2760665484d8SDoug Ambrisko case MFI_STATE_FW_INIT: 2761665484d8SDoug Ambrisko cur_state = MFI_STATE_FW_INIT; 2762665484d8SDoug Ambrisko break; 2763665484d8SDoug Ambrisko case MFI_STATE_FW_INIT_2: 2764665484d8SDoug Ambrisko cur_state = MFI_STATE_FW_INIT_2; 2765665484d8SDoug Ambrisko break; 2766665484d8SDoug Ambrisko case MFI_STATE_DEVICE_SCAN: 2767665484d8SDoug Ambrisko cur_state = MFI_STATE_DEVICE_SCAN; 2768665484d8SDoug Ambrisko break; 2769665484d8SDoug Ambrisko case MFI_STATE_FLUSH_CACHE: 2770665484d8SDoug Ambrisko cur_state = MFI_STATE_FLUSH_CACHE; 2771665484d8SDoug Ambrisko break; 2772665484d8SDoug Ambrisko default: 2773665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Unknown state 0x%x\n", fw_state); 2774665484d8SDoug Ambrisko return -ENODEV; 2775665484d8SDoug Ambrisko } 2776665484d8SDoug Ambrisko 2777665484d8SDoug Ambrisko /* 2778665484d8SDoug Ambrisko * The cur_state should not last for more than max_wait secs 2779665484d8SDoug Ambrisko */ 2780665484d8SDoug Ambrisko for (i = 0; i < (max_wait * 1000); i++) { 2781665484d8SDoug Ambrisko fw_state = (mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2782665484d8SDoug Ambrisko outbound_scratch_pad)) & MFI_STATE_MASK); 2783665484d8SDoug Ambrisko curr_abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 2784665484d8SDoug Ambrisko outbound_scratch_pad)); 2785665484d8SDoug Ambrisko if (abs_state == curr_abs_state) 2786665484d8SDoug Ambrisko DELAY(1000); 2787665484d8SDoug Ambrisko else 2788665484d8SDoug Ambrisko break; 2789665484d8SDoug Ambrisko } 2790665484d8SDoug Ambrisko 2791665484d8SDoug Ambrisko /* 2792665484d8SDoug Ambrisko * Return error if fw_state hasn't changed after max_wait 2793665484d8SDoug Ambrisko */ 2794665484d8SDoug Ambrisko if (curr_abs_state == abs_state) { 2795665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "FW state [%d] hasn't changed " 2796665484d8SDoug Ambrisko "in %d secs\n", fw_state, max_wait); 2797665484d8SDoug Ambrisko return -ENODEV; 2798665484d8SDoug Ambrisko } 2799665484d8SDoug Ambrisko } 2800665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "FW now in Ready state\n"); 2801665484d8SDoug Ambrisko return 0; 2802665484d8SDoug Ambrisko } 2803665484d8SDoug Ambrisko 28048e727371SKashyap D Desai /* 2805665484d8SDoug Ambrisko * mrsas_get_mfi_cmd: Get a cmd from free command pool 2806665484d8SDoug Ambrisko * input: Adapter soft state 2807665484d8SDoug Ambrisko * 2808665484d8SDoug Ambrisko * This function removes an MFI command from the command list. 2809665484d8SDoug Ambrisko */ 28108e727371SKashyap D Desai struct mrsas_mfi_cmd * 28118e727371SKashyap D Desai mrsas_get_mfi_cmd(struct mrsas_softc *sc) 2812665484d8SDoug Ambrisko { 2813665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd = NULL; 2814665484d8SDoug Ambrisko 2815665484d8SDoug Ambrisko mtx_lock(&sc->mfi_cmd_pool_lock); 2816665484d8SDoug Ambrisko if (!TAILQ_EMPTY(&sc->mrsas_mfi_cmd_list_head)) { 2817665484d8SDoug Ambrisko cmd = TAILQ_FIRST(&sc->mrsas_mfi_cmd_list_head); 2818665484d8SDoug Ambrisko TAILQ_REMOVE(&sc->mrsas_mfi_cmd_list_head, cmd, next); 2819665484d8SDoug Ambrisko } 2820665484d8SDoug Ambrisko mtx_unlock(&sc->mfi_cmd_pool_lock); 2821665484d8SDoug Ambrisko 2822665484d8SDoug Ambrisko return cmd; 2823665484d8SDoug Ambrisko } 2824665484d8SDoug Ambrisko 28258e727371SKashyap D Desai /* 28268e727371SKashyap D Desai * mrsas_ocr_thread: Thread to handle OCR/Kill Adapter. 2827665484d8SDoug Ambrisko * input: Adapter Context. 2828665484d8SDoug Ambrisko * 28298e727371SKashyap D Desai * This function will check FW status register and flag do_timeout_reset flag. 28308e727371SKashyap D Desai * It will do OCR/Kill adapter if FW is in fault state or IO timed out has 28318e727371SKashyap D Desai * trigger reset. 2832665484d8SDoug Ambrisko */ 2833665484d8SDoug Ambrisko static void 2834665484d8SDoug Ambrisko mrsas_ocr_thread(void *arg) 2835665484d8SDoug Ambrisko { 2836665484d8SDoug Ambrisko struct mrsas_softc *sc; 2837665484d8SDoug Ambrisko u_int32_t fw_status, fw_state; 28388bb601acSKashyap D Desai u_int8_t tm_target_reset_failed = 0; 2839665484d8SDoug Ambrisko 2840665484d8SDoug Ambrisko sc = (struct mrsas_softc *)arg; 2841665484d8SDoug Ambrisko 2842665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_TRACE, "%s\n", __func__); 2843665484d8SDoug Ambrisko 2844665484d8SDoug Ambrisko sc->ocr_thread_active = 1; 2845665484d8SDoug Ambrisko mtx_lock(&sc->sim_lock); 2846665484d8SDoug Ambrisko for (;;) { 2847665484d8SDoug Ambrisko /* Sleep for 1 second and check the queue status */ 2848665484d8SDoug Ambrisko msleep(&sc->ocr_chan, &sc->sim_lock, PRIBIO, 2849665484d8SDoug Ambrisko "mrsas_ocr", sc->mrsas_fw_fault_check_delay * hz); 2850f0c7594bSKashyap D Desai if (sc->remove_in_progress || 2851f0c7594bSKashyap D Desai sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) { 2852665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2853f0c7594bSKashyap D Desai "Exit due to %s from %s\n", 2854f0c7594bSKashyap D Desai sc->remove_in_progress ? "Shutdown" : 2855f0c7594bSKashyap D Desai "Hardware critical error", __func__); 2856665484d8SDoug Ambrisko break; 2857665484d8SDoug Ambrisko } 2858665484d8SDoug Ambrisko fw_status = mrsas_read_reg(sc, 2859665484d8SDoug Ambrisko offsetof(mrsas_reg_set, outbound_scratch_pad)); 2860665484d8SDoug Ambrisko fw_state = fw_status & MFI_STATE_MASK; 28618bb601acSKashyap D Desai if (fw_state == MFI_STATE_FAULT || sc->do_timedout_reset || 28628bb601acSKashyap D Desai mrsas_atomic_read(&sc->target_reset_outstanding)) { 28638bb601acSKashyap D Desai 28648bb601acSKashyap D Desai /* First, freeze further IOs to come to the SIM */ 28658bb601acSKashyap D Desai mrsas_xpt_freeze(sc); 28668bb601acSKashyap D Desai 28678bb601acSKashyap D Desai /* If this is an IO timeout then go for target reset */ 28688bb601acSKashyap D Desai if (mrsas_atomic_read(&sc->target_reset_outstanding)) { 28698bb601acSKashyap D Desai device_printf(sc->mrsas_dev, "Initiating Target RESET " 28708bb601acSKashyap D Desai "because of SCSI IO timeout!\n"); 28718bb601acSKashyap D Desai 28728bb601acSKashyap D Desai /* Let the remaining IOs to complete */ 28738bb601acSKashyap D Desai msleep(&sc->ocr_chan, &sc->sim_lock, PRIBIO, 28748bb601acSKashyap D Desai "mrsas_reset_targets", 5 * hz); 28758bb601acSKashyap D Desai 28768bb601acSKashyap D Desai /* Try to reset the target device */ 28778bb601acSKashyap D Desai if (mrsas_reset_targets(sc) == FAIL) 28788bb601acSKashyap D Desai tm_target_reset_failed = 1; 28798bb601acSKashyap D Desai } 28808bb601acSKashyap D Desai 28818bb601acSKashyap D Desai /* If this is a DCMD timeout or FW fault, 28828bb601acSKashyap D Desai * then go for controller reset 28838bb601acSKashyap D Desai */ 28848bb601acSKashyap D Desai if (fw_state == MFI_STATE_FAULT || tm_target_reset_failed || 28858bb601acSKashyap D Desai (sc->do_timedout_reset == MFI_DCMD_TIMEOUT_OCR)) { 28868bb601acSKashyap D Desai if (tm_target_reset_failed) 28878bb601acSKashyap D Desai device_printf(sc->mrsas_dev, "Initiaiting OCR because of " 28888bb601acSKashyap D Desai "TM FAILURE!\n"); 28898bb601acSKashyap D Desai else 28908bb601acSKashyap D Desai device_printf(sc->mrsas_dev, "Initiaiting OCR " 28918bb601acSKashyap D Desai "because of %s!\n", sc->do_timedout_reset ? 28928bb601acSKashyap D Desai "DCMD IO Timeout" : "FW fault"); 28938bb601acSKashyap D Desai 2894665484d8SDoug Ambrisko mtx_lock_spin(&sc->ioctl_lock); 2895665484d8SDoug Ambrisko sc->reset_in_progress = 1; 2896665484d8SDoug Ambrisko mtx_unlock_spin(&sc->ioctl_lock); 28978bb601acSKashyap D Desai sc->reset_count++; 28988bb601acSKashyap D Desai 289985c0a961SKashyap D Desai /* 290085c0a961SKashyap D Desai * Wait for the AEN task to be completed if it is running. 290185c0a961SKashyap D Desai */ 290285c0a961SKashyap D Desai mtx_unlock(&sc->sim_lock); 290385c0a961SKashyap D Desai taskqueue_drain(sc->ev_tq, &sc->ev_task); 290485c0a961SKashyap D Desai mtx_lock(&sc->sim_lock); 290585c0a961SKashyap D Desai 290685c0a961SKashyap D Desai taskqueue_block(sc->ev_tq); 29078bb601acSKashyap D Desai /* Try to reset the controller */ 2908f0c7594bSKashyap D Desai mrsas_reset_ctrl(sc, sc->do_timedout_reset); 29098bb601acSKashyap D Desai 2910665484d8SDoug Ambrisko sc->do_timedout_reset = 0; 29118bb601acSKashyap D Desai sc->reset_in_progress = 0; 29128bb601acSKashyap D Desai tm_target_reset_failed = 0; 29138bb601acSKashyap D Desai mrsas_atomic_set(&sc->target_reset_outstanding, 0); 29148bb601acSKashyap D Desai memset(sc->target_reset_pool, 0, 29158bb601acSKashyap D Desai sizeof(sc->target_reset_pool)); 291685c0a961SKashyap D Desai taskqueue_unblock(sc->ev_tq); 29178bb601acSKashyap D Desai } 29188bb601acSKashyap D Desai 29198bb601acSKashyap D Desai /* Now allow IOs to come to the SIM */ 29208bb601acSKashyap D Desai mrsas_xpt_release(sc); 2921665484d8SDoug Ambrisko } 2922665484d8SDoug Ambrisko } 2923665484d8SDoug Ambrisko mtx_unlock(&sc->sim_lock); 2924665484d8SDoug Ambrisko sc->ocr_thread_active = 0; 2925665484d8SDoug Ambrisko mrsas_kproc_exit(0); 2926665484d8SDoug Ambrisko } 2927665484d8SDoug Ambrisko 29288e727371SKashyap D Desai /* 29298e727371SKashyap D Desai * mrsas_reset_reply_desc: Reset Reply descriptor as part of OCR. 2930665484d8SDoug Ambrisko * input: Adapter Context. 2931665484d8SDoug Ambrisko * 29328e727371SKashyap D Desai * This function will clear reply descriptor so that post OCR driver and FW will 29338e727371SKashyap D Desai * lost old history. 2934665484d8SDoug Ambrisko */ 29358e727371SKashyap D Desai void 29368e727371SKashyap D Desai mrsas_reset_reply_desc(struct mrsas_softc *sc) 2937665484d8SDoug Ambrisko { 2938d18d1b47SKashyap D Desai int i, count; 2939665484d8SDoug Ambrisko pMpi2ReplyDescriptorsUnion_t reply_desc; 2940665484d8SDoug Ambrisko 2941d18d1b47SKashyap D Desai count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 2942d18d1b47SKashyap D Desai for (i = 0; i < count; i++) 2943d18d1b47SKashyap D Desai sc->last_reply_idx[i] = 0; 2944d18d1b47SKashyap D Desai 2945665484d8SDoug Ambrisko reply_desc = sc->reply_desc_mem; 2946665484d8SDoug Ambrisko for (i = 0; i < sc->reply_q_depth; i++, reply_desc++) { 2947665484d8SDoug Ambrisko reply_desc->Words = MRSAS_ULONG_MAX; 2948665484d8SDoug Ambrisko } 2949665484d8SDoug Ambrisko } 2950665484d8SDoug Ambrisko 29518e727371SKashyap D Desai /* 29528e727371SKashyap D Desai * mrsas_reset_ctrl: Core function to OCR/Kill adapter. 2953665484d8SDoug Ambrisko * input: Adapter Context. 2954665484d8SDoug Ambrisko * 29558e727371SKashyap D Desai * This function will run from thread context so that it can sleep. 1. Do not 29568e727371SKashyap D Desai * handle OCR if FW is in HW critical error. 2. Wait for outstanding command 29578e727371SKashyap D Desai * to complete for 180 seconds. 3. If #2 does not find any outstanding 29588e727371SKashyap D Desai * command Controller is in working state, so skip OCR. Otherwise, do 29598e727371SKashyap D Desai * OCR/kill Adapter based on flag disableOnlineCtrlReset. 4. Start of the 29608e727371SKashyap D Desai * OCR, return all SCSI command back to CAM layer which has ccb_ptr. 5. Post 2961453130d9SPedro F. Giffuni * OCR, Re-fire Management command and move Controller to Operation state. 2962665484d8SDoug Ambrisko */ 29638e727371SKashyap D Desai int 2964f0c7594bSKashyap D Desai mrsas_reset_ctrl(struct mrsas_softc *sc, u_int8_t reset_reason) 2965665484d8SDoug Ambrisko { 2966665484d8SDoug Ambrisko int retval = SUCCESS, i, j, retry = 0; 2967665484d8SDoug Ambrisko u_int32_t host_diag, abs_state, status_reg, reset_adapter; 2968665484d8SDoug Ambrisko union ccb *ccb; 2969665484d8SDoug Ambrisko struct mrsas_mfi_cmd *mfi_cmd; 2970665484d8SDoug Ambrisko struct mrsas_mpt_cmd *mpt_cmd; 2971f0c7594bSKashyap D Desai union mrsas_evt_class_locale class_locale; 29722d53b485SKashyap D Desai MRSAS_REQUEST_DESCRIPTOR_UNION *req_desc; 2973665484d8SDoug Ambrisko 2974665484d8SDoug Ambrisko if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) { 2975665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, 2976665484d8SDoug Ambrisko "mrsas: Hardware critical error, returning FAIL.\n"); 2977665484d8SDoug Ambrisko return FAIL; 2978665484d8SDoug Ambrisko } 2979f5fb2237SKashyap D Desai mrsas_set_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags); 2980665484d8SDoug Ambrisko sc->adprecovery = MRSAS_ADPRESET_SM_INFAULT; 2981665484d8SDoug Ambrisko mrsas_disable_intr(sc); 2982f0c7594bSKashyap D Desai msleep(&sc->ocr_chan, &sc->sim_lock, PRIBIO, "mrsas_ocr", 2983f0c7594bSKashyap D Desai sc->mrsas_fw_fault_check_delay * hz); 2984665484d8SDoug Ambrisko 2985665484d8SDoug Ambrisko /* First try waiting for commands to complete */ 2986f0c7594bSKashyap D Desai if (mrsas_wait_for_outstanding(sc, reset_reason)) { 2987665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2988665484d8SDoug Ambrisko "resetting adapter from %s.\n", 2989665484d8SDoug Ambrisko __func__); 2990665484d8SDoug Ambrisko /* Now return commands back to the CAM layer */ 29915b2490f8SKashyap D Desai mtx_unlock(&sc->sim_lock); 2992665484d8SDoug Ambrisko for (i = 0; i < sc->max_fw_cmds; i++) { 2993665484d8SDoug Ambrisko mpt_cmd = sc->mpt_cmd_list[i]; 2994665484d8SDoug Ambrisko if (mpt_cmd->ccb_ptr) { 2995665484d8SDoug Ambrisko ccb = (union ccb *)(mpt_cmd->ccb_ptr); 2996665484d8SDoug Ambrisko ccb->ccb_h.status = CAM_SCSI_BUS_RESET; 2997665484d8SDoug Ambrisko mrsas_cmd_done(sc, mpt_cmd); 2998f5fb2237SKashyap D Desai mrsas_atomic_dec(&sc->fw_outstanding); 2999665484d8SDoug Ambrisko } 3000665484d8SDoug Ambrisko } 30015b2490f8SKashyap D Desai mtx_lock(&sc->sim_lock); 3002665484d8SDoug Ambrisko 3003665484d8SDoug Ambrisko status_reg = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 3004665484d8SDoug Ambrisko outbound_scratch_pad)); 3005665484d8SDoug Ambrisko abs_state = status_reg & MFI_STATE_MASK; 3006665484d8SDoug Ambrisko reset_adapter = status_reg & MFI_RESET_ADAPTER; 3007665484d8SDoug Ambrisko if (sc->disableOnlineCtrlReset || 3008665484d8SDoug Ambrisko (abs_state == MFI_STATE_FAULT && !reset_adapter)) { 3009665484d8SDoug Ambrisko /* Reset not supported, kill adapter */ 3010665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "Reset not supported, killing adapter.\n"); 3011665484d8SDoug Ambrisko mrsas_kill_hba(sc); 3012665484d8SDoug Ambrisko retval = FAIL; 3013665484d8SDoug Ambrisko goto out; 3014665484d8SDoug Ambrisko } 3015665484d8SDoug Ambrisko /* Now try to reset the chip */ 3016665484d8SDoug Ambrisko for (i = 0; i < MRSAS_FUSION_MAX_RESET_TRIES; i++) { 3017665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 3018665484d8SDoug Ambrisko MPI2_WRSEQ_FLUSH_KEY_VALUE); 3019665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 3020665484d8SDoug Ambrisko MPI2_WRSEQ_1ST_KEY_VALUE); 3021665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 3022665484d8SDoug Ambrisko MPI2_WRSEQ_2ND_KEY_VALUE); 3023665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 3024665484d8SDoug Ambrisko MPI2_WRSEQ_3RD_KEY_VALUE); 3025665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 3026665484d8SDoug Ambrisko MPI2_WRSEQ_4TH_KEY_VALUE); 3027665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 3028665484d8SDoug Ambrisko MPI2_WRSEQ_5TH_KEY_VALUE); 3029665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 3030665484d8SDoug Ambrisko MPI2_WRSEQ_6TH_KEY_VALUE); 3031665484d8SDoug Ambrisko 3032665484d8SDoug Ambrisko /* Check that the diag write enable (DRWE) bit is on */ 3033665484d8SDoug Ambrisko host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 3034665484d8SDoug Ambrisko fusion_host_diag)); 3035665484d8SDoug Ambrisko retry = 0; 3036665484d8SDoug Ambrisko while (!(host_diag & HOST_DIAG_WRITE_ENABLE)) { 3037665484d8SDoug Ambrisko DELAY(100 * 1000); 3038665484d8SDoug Ambrisko host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 3039665484d8SDoug Ambrisko fusion_host_diag)); 3040665484d8SDoug Ambrisko if (retry++ == 100) { 3041665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 3042665484d8SDoug Ambrisko "Host diag unlock failed!\n"); 3043665484d8SDoug Ambrisko break; 3044665484d8SDoug Ambrisko } 3045665484d8SDoug Ambrisko } 3046665484d8SDoug Ambrisko if (!(host_diag & HOST_DIAG_WRITE_ENABLE)) 3047665484d8SDoug Ambrisko continue; 3048665484d8SDoug Ambrisko 3049665484d8SDoug Ambrisko /* Send chip reset command */ 3050665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_host_diag), 3051665484d8SDoug Ambrisko host_diag | HOST_DIAG_RESET_ADAPTER); 3052665484d8SDoug Ambrisko DELAY(3000 * 1000); 3053665484d8SDoug Ambrisko 3054665484d8SDoug Ambrisko /* Make sure reset adapter bit is cleared */ 3055665484d8SDoug Ambrisko host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 3056665484d8SDoug Ambrisko fusion_host_diag)); 3057665484d8SDoug Ambrisko retry = 0; 3058665484d8SDoug Ambrisko while (host_diag & HOST_DIAG_RESET_ADAPTER) { 3059665484d8SDoug Ambrisko DELAY(100 * 1000); 3060665484d8SDoug Ambrisko host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 3061665484d8SDoug Ambrisko fusion_host_diag)); 3062665484d8SDoug Ambrisko if (retry++ == 1000) { 3063665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 3064665484d8SDoug Ambrisko "Diag reset adapter never cleared!\n"); 3065665484d8SDoug Ambrisko break; 3066665484d8SDoug Ambrisko } 3067665484d8SDoug Ambrisko } 3068665484d8SDoug Ambrisko if (host_diag & HOST_DIAG_RESET_ADAPTER) 3069665484d8SDoug Ambrisko continue; 3070665484d8SDoug Ambrisko 3071665484d8SDoug Ambrisko abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 3072665484d8SDoug Ambrisko outbound_scratch_pad)) & MFI_STATE_MASK; 3073665484d8SDoug Ambrisko retry = 0; 3074665484d8SDoug Ambrisko 3075665484d8SDoug Ambrisko while ((abs_state <= MFI_STATE_FW_INIT) && (retry++ < 1000)) { 3076665484d8SDoug Ambrisko DELAY(100 * 1000); 3077665484d8SDoug Ambrisko abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 3078665484d8SDoug Ambrisko outbound_scratch_pad)) & MFI_STATE_MASK; 3079665484d8SDoug Ambrisko } 3080665484d8SDoug Ambrisko if (abs_state <= MFI_STATE_FW_INIT) { 3081665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "firmware state < MFI_STATE_FW_INIT," 3082665484d8SDoug Ambrisko " state = 0x%x\n", abs_state); 3083665484d8SDoug Ambrisko continue; 3084665484d8SDoug Ambrisko } 3085665484d8SDoug Ambrisko /* Wait for FW to become ready */ 3086665484d8SDoug Ambrisko if (mrsas_transition_to_ready(sc, 1)) { 3087665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 3088665484d8SDoug Ambrisko "mrsas: Failed to transition controller to ready.\n"); 3089665484d8SDoug Ambrisko continue; 3090665484d8SDoug Ambrisko } 3091665484d8SDoug Ambrisko mrsas_reset_reply_desc(sc); 3092665484d8SDoug Ambrisko if (mrsas_ioc_init(sc)) { 3093665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "mrsas_ioc_init() failed!\n"); 3094665484d8SDoug Ambrisko continue; 3095665484d8SDoug Ambrisko } 3096665484d8SDoug Ambrisko for (j = 0; j < sc->max_fw_cmds; j++) { 3097665484d8SDoug Ambrisko mpt_cmd = sc->mpt_cmd_list[j]; 3098665484d8SDoug Ambrisko if (mpt_cmd->sync_cmd_idx != (u_int32_t)MRSAS_ULONG_MAX) { 3099665484d8SDoug Ambrisko mfi_cmd = sc->mfi_cmd_list[mpt_cmd->sync_cmd_idx]; 31002d53b485SKashyap D Desai /* If not an IOCTL then release the command else re-fire */ 31012d53b485SKashyap D Desai if (!mfi_cmd->sync_cmd) { 3102665484d8SDoug Ambrisko mrsas_release_mfi_cmd(mfi_cmd); 31032d53b485SKashyap D Desai } else { 31042d53b485SKashyap D Desai req_desc = mrsas_get_request_desc(sc, 31052d53b485SKashyap D Desai mfi_cmd->cmd_id.context.smid - 1); 31062d53b485SKashyap D Desai mrsas_dprint(sc, MRSAS_OCR, 31072d53b485SKashyap D Desai "Re-fire command DCMD opcode 0x%x index %d\n ", 31082d53b485SKashyap D Desai mfi_cmd->frame->dcmd.opcode, j); 31092d53b485SKashyap D Desai if (!req_desc) 31102d53b485SKashyap D Desai device_printf(sc->mrsas_dev, 31112d53b485SKashyap D Desai "Cannot build MPT cmd.\n"); 31122d53b485SKashyap D Desai else 31132d53b485SKashyap D Desai mrsas_fire_cmd(sc, req_desc->addr.u.low, 31142d53b485SKashyap D Desai req_desc->addr.u.high); 31152d53b485SKashyap D Desai } 3116665484d8SDoug Ambrisko } 3117665484d8SDoug Ambrisko } 3118f0c7594bSKashyap D Desai 3119665484d8SDoug Ambrisko /* Reset load balance info */ 3120665484d8SDoug Ambrisko memset(sc->load_balance_info, 0, 31214799d485SKashyap D Desai sizeof(LD_LOAD_BALANCE_INFO) * MAX_LOGICAL_DRIVES_EXT); 3122665484d8SDoug Ambrisko 3123af51c29fSKashyap D Desai if (mrsas_get_ctrl_info(sc)) { 3124af51c29fSKashyap D Desai mrsas_kill_hba(sc); 31252f863eb8SKashyap D Desai retval = FAIL; 31262f863eb8SKashyap D Desai goto out; 3127af51c29fSKashyap D Desai } 3128665484d8SDoug Ambrisko if (!mrsas_get_map_info(sc)) 3129665484d8SDoug Ambrisko mrsas_sync_map_info(sc); 3130665484d8SDoug Ambrisko 3131a688fcd0SKashyap D Desai megasas_setup_jbod_map(sc); 3132a688fcd0SKashyap D Desai 31332f863eb8SKashyap D Desai mrsas_clear_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags); 31342f863eb8SKashyap D Desai mrsas_enable_intr(sc); 31352f863eb8SKashyap D Desai sc->adprecovery = MRSAS_HBA_OPERATIONAL; 31362f863eb8SKashyap D Desai 3137f0c7594bSKashyap D Desai /* Register AEN with FW for last sequence number */ 3138f0c7594bSKashyap D Desai class_locale.members.reserved = 0; 3139f0c7594bSKashyap D Desai class_locale.members.locale = MR_EVT_LOCALE_ALL; 3140f0c7594bSKashyap D Desai class_locale.members.class = MR_EVT_CLASS_DEBUG; 3141f0c7594bSKashyap D Desai 31422d53b485SKashyap D Desai mtx_unlock(&sc->sim_lock); 3143f0c7594bSKashyap D Desai if (mrsas_register_aen(sc, sc->last_seq_num, 3144f0c7594bSKashyap D Desai class_locale.word)) { 3145f0c7594bSKashyap D Desai device_printf(sc->mrsas_dev, 3146f0c7594bSKashyap D Desai "ERROR: AEN registration FAILED from OCR !!! " 3147f0c7594bSKashyap D Desai "Further events from the controller cannot be notified." 3148f0c7594bSKashyap D Desai "Either there is some problem in the controller" 3149f0c7594bSKashyap D Desai "or the controller does not support AEN.\n" 3150f0c7594bSKashyap D Desai "Please contact to the SUPPORT TEAM if the problem persists\n"); 3151f0c7594bSKashyap D Desai } 31522d53b485SKashyap D Desai mtx_lock(&sc->sim_lock); 31532d53b485SKashyap D Desai 3154665484d8SDoug Ambrisko /* Adapter reset completed successfully */ 3155665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Reset successful\n"); 3156665484d8SDoug Ambrisko retval = SUCCESS; 3157665484d8SDoug Ambrisko goto out; 3158665484d8SDoug Ambrisko } 3159665484d8SDoug Ambrisko /* Reset failed, kill the adapter */ 3160665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Reset failed, killing adapter.\n"); 3161665484d8SDoug Ambrisko mrsas_kill_hba(sc); 3162665484d8SDoug Ambrisko retval = FAIL; 3163665484d8SDoug Ambrisko } else { 3164f5fb2237SKashyap D Desai mrsas_clear_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags); 3165665484d8SDoug Ambrisko mrsas_enable_intr(sc); 3166665484d8SDoug Ambrisko sc->adprecovery = MRSAS_HBA_OPERATIONAL; 3167665484d8SDoug Ambrisko } 3168665484d8SDoug Ambrisko out: 3169f5fb2237SKashyap D Desai mrsas_clear_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags); 3170665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 3171665484d8SDoug Ambrisko "Reset Exit with %d.\n", retval); 3172665484d8SDoug Ambrisko return retval; 3173665484d8SDoug Ambrisko } 3174665484d8SDoug Ambrisko 31758e727371SKashyap D Desai /* 31768e727371SKashyap D Desai * mrsas_kill_hba: Kill HBA when OCR is not supported 3177665484d8SDoug Ambrisko * input: Adapter Context. 3178665484d8SDoug Ambrisko * 3179665484d8SDoug Ambrisko * This function will kill HBA when OCR is not supported. 3180665484d8SDoug Ambrisko */ 31818e727371SKashyap D Desai void 31828e727371SKashyap D Desai mrsas_kill_hba(struct mrsas_softc *sc) 3183665484d8SDoug Ambrisko { 3184daeed973SKashyap D Desai sc->adprecovery = MRSAS_HW_CRITICAL_ERROR; 3185f0c7594bSKashyap D Desai DELAY(1000 * 1000); 3186665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "%s\n", __func__); 3187665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell), 3188665484d8SDoug Ambrisko MFI_STOP_ADP); 3189665484d8SDoug Ambrisko /* Flush */ 3190665484d8SDoug Ambrisko mrsas_read_reg(sc, offsetof(mrsas_reg_set, doorbell)); 3191daeed973SKashyap D Desai mrsas_complete_outstanding_ioctls(sc); 3192daeed973SKashyap D Desai } 3193daeed973SKashyap D Desai 3194daeed973SKashyap D Desai /** 3195daeed973SKashyap D Desai * mrsas_complete_outstanding_ioctls Complete pending IOCTLS after kill_hba 3196daeed973SKashyap D Desai * input: Controller softc 3197daeed973SKashyap D Desai * 3198daeed973SKashyap D Desai * Returns void 3199daeed973SKashyap D Desai */ 3200dbcc81dfSKashyap D Desai void 3201dbcc81dfSKashyap D Desai mrsas_complete_outstanding_ioctls(struct mrsas_softc *sc) 3202dbcc81dfSKashyap D Desai { 3203daeed973SKashyap D Desai int i; 3204daeed973SKashyap D Desai struct mrsas_mpt_cmd *cmd_mpt; 3205daeed973SKashyap D Desai struct mrsas_mfi_cmd *cmd_mfi; 3206daeed973SKashyap D Desai u_int32_t count, MSIxIndex; 3207daeed973SKashyap D Desai 3208daeed973SKashyap D Desai count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 3209daeed973SKashyap D Desai for (i = 0; i < sc->max_fw_cmds; i++) { 3210daeed973SKashyap D Desai cmd_mpt = sc->mpt_cmd_list[i]; 3211daeed973SKashyap D Desai 3212daeed973SKashyap D Desai if (cmd_mpt->sync_cmd_idx != (u_int32_t)MRSAS_ULONG_MAX) { 3213daeed973SKashyap D Desai cmd_mfi = sc->mfi_cmd_list[cmd_mpt->sync_cmd_idx]; 3214daeed973SKashyap D Desai if (cmd_mfi->sync_cmd && cmd_mfi->frame->hdr.cmd != MFI_CMD_ABORT) { 3215daeed973SKashyap D Desai for (MSIxIndex = 0; MSIxIndex < count; MSIxIndex++) 3216daeed973SKashyap D Desai mrsas_complete_mptmfi_passthru(sc, cmd_mfi, 3217*503c4f8dSKashyap D Desai cmd_mpt->io_request->RaidContext.raid_context.status); 3218daeed973SKashyap D Desai } 3219daeed973SKashyap D Desai } 3220daeed973SKashyap D Desai } 3221665484d8SDoug Ambrisko } 3222665484d8SDoug Ambrisko 32238e727371SKashyap D Desai /* 32248e727371SKashyap D Desai * mrsas_wait_for_outstanding: Wait for outstanding commands 3225665484d8SDoug Ambrisko * input: Adapter Context. 3226665484d8SDoug Ambrisko * 32278e727371SKashyap D Desai * This function will wait for 180 seconds for outstanding commands to be 32288e727371SKashyap D Desai * completed. 3229665484d8SDoug Ambrisko */ 32308e727371SKashyap D Desai int 3231f0c7594bSKashyap D Desai mrsas_wait_for_outstanding(struct mrsas_softc *sc, u_int8_t check_reason) 3232665484d8SDoug Ambrisko { 3233665484d8SDoug Ambrisko int i, outstanding, retval = 0; 3234d18d1b47SKashyap D Desai u_int32_t fw_state, count, MSIxIndex; 3235d18d1b47SKashyap D Desai 3236665484d8SDoug Ambrisko 3237665484d8SDoug Ambrisko for (i = 0; i < MRSAS_RESET_WAIT_TIME; i++) { 3238665484d8SDoug Ambrisko if (sc->remove_in_progress) { 3239665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 3240665484d8SDoug Ambrisko "Driver remove or shutdown called.\n"); 3241665484d8SDoug Ambrisko retval = 1; 3242665484d8SDoug Ambrisko goto out; 3243665484d8SDoug Ambrisko } 3244665484d8SDoug Ambrisko /* Check if firmware is in fault state */ 3245665484d8SDoug Ambrisko fw_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set, 3246665484d8SDoug Ambrisko outbound_scratch_pad)) & MFI_STATE_MASK; 3247665484d8SDoug Ambrisko if (fw_state == MFI_STATE_FAULT) { 3248665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 3249665484d8SDoug Ambrisko "Found FW in FAULT state, will reset adapter.\n"); 3250e2e8afb1SKashyap D Desai count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 3251e2e8afb1SKashyap D Desai mtx_unlock(&sc->sim_lock); 3252e2e8afb1SKashyap D Desai for (MSIxIndex = 0; MSIxIndex < count; MSIxIndex++) 3253e2e8afb1SKashyap D Desai mrsas_complete_cmd(sc, MSIxIndex); 3254e2e8afb1SKashyap D Desai mtx_lock(&sc->sim_lock); 3255665484d8SDoug Ambrisko retval = 1; 3256665484d8SDoug Ambrisko goto out; 3257665484d8SDoug Ambrisko } 3258f0c7594bSKashyap D Desai if (check_reason == MFI_DCMD_TIMEOUT_OCR) { 3259f0c7594bSKashyap D Desai mrsas_dprint(sc, MRSAS_OCR, 3260f0c7594bSKashyap D Desai "DCMD IO TIMEOUT detected, will reset adapter.\n"); 3261f0c7594bSKashyap D Desai retval = 1; 3262f0c7594bSKashyap D Desai goto out; 3263f0c7594bSKashyap D Desai } 3264f5fb2237SKashyap D Desai outstanding = mrsas_atomic_read(&sc->fw_outstanding); 3265665484d8SDoug Ambrisko if (!outstanding) 3266665484d8SDoug Ambrisko goto out; 3267665484d8SDoug Ambrisko 3268665484d8SDoug Ambrisko if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) { 3269665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "[%2d]waiting for %d " 3270665484d8SDoug Ambrisko "commands to complete\n", i, outstanding); 3271d18d1b47SKashyap D Desai count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 32722d53b485SKashyap D Desai mtx_unlock(&sc->sim_lock); 3273d18d1b47SKashyap D Desai for (MSIxIndex = 0; MSIxIndex < count; MSIxIndex++) 3274d18d1b47SKashyap D Desai mrsas_complete_cmd(sc, MSIxIndex); 32752d53b485SKashyap D Desai mtx_lock(&sc->sim_lock); 3276665484d8SDoug Ambrisko } 3277665484d8SDoug Ambrisko DELAY(1000 * 1000); 3278665484d8SDoug Ambrisko } 3279665484d8SDoug Ambrisko 3280f5fb2237SKashyap D Desai if (mrsas_atomic_read(&sc->fw_outstanding)) { 3281665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 3282665484d8SDoug Ambrisko " pending commands remain after waiting," 3283665484d8SDoug Ambrisko " will reset adapter.\n"); 3284665484d8SDoug Ambrisko retval = 1; 3285665484d8SDoug Ambrisko } 3286665484d8SDoug Ambrisko out: 3287665484d8SDoug Ambrisko return retval; 3288665484d8SDoug Ambrisko } 3289665484d8SDoug Ambrisko 32908e727371SKashyap D Desai /* 3291665484d8SDoug Ambrisko * mrsas_release_mfi_cmd: Return a cmd to free command pool 3292665484d8SDoug Ambrisko * input: Command packet for return to free cmd pool 3293665484d8SDoug Ambrisko * 3294731b7561SKashyap D Desai * This function returns the MFI & MPT command to the command list. 3295665484d8SDoug Ambrisko */ 32968e727371SKashyap D Desai void 3297731b7561SKashyap D Desai mrsas_release_mfi_cmd(struct mrsas_mfi_cmd *cmd_mfi) 3298665484d8SDoug Ambrisko { 3299731b7561SKashyap D Desai struct mrsas_softc *sc = cmd_mfi->sc; 3300731b7561SKashyap D Desai struct mrsas_mpt_cmd *cmd_mpt; 3301731b7561SKashyap D Desai 3302665484d8SDoug Ambrisko 3303665484d8SDoug Ambrisko mtx_lock(&sc->mfi_cmd_pool_lock); 3304731b7561SKashyap D Desai /* 3305731b7561SKashyap D Desai * Release the mpt command (if at all it is allocated 3306731b7561SKashyap D Desai * associated with the mfi command 3307731b7561SKashyap D Desai */ 3308731b7561SKashyap D Desai if (cmd_mfi->cmd_id.context.smid) { 3309731b7561SKashyap D Desai mtx_lock(&sc->mpt_cmd_pool_lock); 3310731b7561SKashyap D Desai /* Get the mpt cmd from mfi cmd frame's smid value */ 3311731b7561SKashyap D Desai cmd_mpt = sc->mpt_cmd_list[cmd_mfi->cmd_id.context.smid-1]; 3312731b7561SKashyap D Desai cmd_mpt->flags = 0; 3313731b7561SKashyap D Desai cmd_mpt->sync_cmd_idx = (u_int32_t)MRSAS_ULONG_MAX; 3314731b7561SKashyap D Desai TAILQ_INSERT_HEAD(&(sc->mrsas_mpt_cmd_list_head), cmd_mpt, next); 3315731b7561SKashyap D Desai mtx_unlock(&sc->mpt_cmd_pool_lock); 3316731b7561SKashyap D Desai } 3317731b7561SKashyap D Desai /* Release the mfi command */ 3318731b7561SKashyap D Desai cmd_mfi->ccb_ptr = NULL; 3319731b7561SKashyap D Desai cmd_mfi->cmd_id.frame_count = 0; 3320731b7561SKashyap D Desai TAILQ_INSERT_HEAD(&(sc->mrsas_mfi_cmd_list_head), cmd_mfi, next); 3321665484d8SDoug Ambrisko mtx_unlock(&sc->mfi_cmd_pool_lock); 3322665484d8SDoug Ambrisko 3323665484d8SDoug Ambrisko return; 3324665484d8SDoug Ambrisko } 3325665484d8SDoug Ambrisko 33268e727371SKashyap D Desai /* 33278e727371SKashyap D Desai * mrsas_get_controller_info: Returns FW's controller structure 3328665484d8SDoug Ambrisko * input: Adapter soft state 3329665484d8SDoug Ambrisko * Controller information structure 3330665484d8SDoug Ambrisko * 33318e727371SKashyap D Desai * Issues an internal command (DCMD) to get the FW's controller structure. This 33328e727371SKashyap D Desai * information is mainly used to find out the maximum IO transfer per command 33338e727371SKashyap D Desai * supported by the FW. 3334665484d8SDoug Ambrisko */ 33358e727371SKashyap D Desai static int 3336af51c29fSKashyap D Desai mrsas_get_ctrl_info(struct mrsas_softc *sc) 3337665484d8SDoug Ambrisko { 3338665484d8SDoug Ambrisko int retcode = 0; 3339f0c7594bSKashyap D Desai u_int8_t do_ocr = 1; 3340665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3341665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 3342665484d8SDoug Ambrisko 3343665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3344665484d8SDoug Ambrisko 3345665484d8SDoug Ambrisko if (!cmd) { 3346665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Failed to get a free cmd\n"); 3347665484d8SDoug Ambrisko return -ENOMEM; 3348665484d8SDoug Ambrisko } 3349665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 3350665484d8SDoug Ambrisko 3351665484d8SDoug Ambrisko if (mrsas_alloc_ctlr_info_cmd(sc) != SUCCESS) { 3352665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate get ctlr info cmd\n"); 3353665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3354665484d8SDoug Ambrisko return -ENOMEM; 3355665484d8SDoug Ambrisko } 3356665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 3357665484d8SDoug Ambrisko 3358665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 3359665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 3360665484d8SDoug Ambrisko dcmd->sge_count = 1; 3361665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 3362665484d8SDoug Ambrisko dcmd->timeout = 0; 3363665484d8SDoug Ambrisko dcmd->pad_0 = 0; 3364665484d8SDoug Ambrisko dcmd->data_xfer_len = sizeof(struct mrsas_ctrl_info); 3365665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_CTRL_GET_INFO; 3366665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = sc->ctlr_info_phys_addr; 3367665484d8SDoug Ambrisko dcmd->sgl.sge32[0].length = sizeof(struct mrsas_ctrl_info); 3368665484d8SDoug Ambrisko 33698bc320adSKashyap D Desai if (!sc->mask_interrupts) 33708bc320adSKashyap D Desai retcode = mrsas_issue_blocked_cmd(sc, cmd); 33718bc320adSKashyap D Desai else 3372f0c7594bSKashyap D Desai retcode = mrsas_issue_polled(sc, cmd); 33738bc320adSKashyap D Desai 3374f0c7594bSKashyap D Desai if (retcode == ETIMEDOUT) 3375f0c7594bSKashyap D Desai goto dcmd_timeout; 3376665484d8SDoug Ambrisko else 3377f0c7594bSKashyap D Desai memcpy(sc->ctrl_info, sc->ctlr_info_mem, sizeof(struct mrsas_ctrl_info)); 3378665484d8SDoug Ambrisko 3379f0c7594bSKashyap D Desai do_ocr = 0; 3380af51c29fSKashyap D Desai mrsas_update_ext_vd_details(sc); 3381af51c29fSKashyap D Desai 3382a688fcd0SKashyap D Desai sc->use_seqnum_jbod_fp = 3383a688fcd0SKashyap D Desai sc->ctrl_info->adapterOperations3.useSeqNumJbodFP; 33848bc320adSKashyap D Desai sc->disableOnlineCtrlReset = 33858bc320adSKashyap D Desai sc->ctrl_info->properties.OnOffProperties.disableOnlineCtrlReset; 3386a688fcd0SKashyap D Desai 3387f0c7594bSKashyap D Desai dcmd_timeout: 3388665484d8SDoug Ambrisko mrsas_free_ctlr_info_cmd(sc); 3389f0c7594bSKashyap D Desai 3390f0c7594bSKashyap D Desai if (do_ocr) 3391f0c7594bSKashyap D Desai sc->do_timedout_reset = MFI_DCMD_TIMEOUT_OCR; 3392f0c7594bSKashyap D Desai 33938bc320adSKashyap D Desai if (!sc->mask_interrupts) 33948bc320adSKashyap D Desai mrsas_release_mfi_cmd(cmd); 33958bc320adSKashyap D Desai 3396665484d8SDoug Ambrisko return (retcode); 3397665484d8SDoug Ambrisko } 3398665484d8SDoug Ambrisko 33998e727371SKashyap D Desai /* 3400af51c29fSKashyap D Desai * mrsas_update_ext_vd_details : Update details w.r.t Extended VD 3401af51c29fSKashyap D Desai * input: 3402af51c29fSKashyap D Desai * sc - Controller's softc 3403af51c29fSKashyap D Desai */ 3404dbcc81dfSKashyap D Desai static void 3405dbcc81dfSKashyap D Desai mrsas_update_ext_vd_details(struct mrsas_softc *sc) 3406af51c29fSKashyap D Desai { 34074ad83576SKashyap D Desai u_int32_t ventura_map_sz = 0; 3408af51c29fSKashyap D Desai sc->max256vdSupport = 3409af51c29fSKashyap D Desai sc->ctrl_info->adapterOperations3.supportMaxExtLDs; 34104ad83576SKashyap D Desai 3411af51c29fSKashyap D Desai /* Below is additional check to address future FW enhancement */ 3412af51c29fSKashyap D Desai if (sc->ctrl_info->max_lds > 64) 3413af51c29fSKashyap D Desai sc->max256vdSupport = 1; 3414af51c29fSKashyap D Desai 3415af51c29fSKashyap D Desai sc->drv_supported_vd_count = MRSAS_MAX_LD_CHANNELS 3416af51c29fSKashyap D Desai * MRSAS_MAX_DEV_PER_CHANNEL; 3417af51c29fSKashyap D Desai sc->drv_supported_pd_count = MRSAS_MAX_PD_CHANNELS 3418af51c29fSKashyap D Desai * MRSAS_MAX_DEV_PER_CHANNEL; 3419af51c29fSKashyap D Desai if (sc->max256vdSupport) { 3420af51c29fSKashyap D Desai sc->fw_supported_vd_count = MAX_LOGICAL_DRIVES_EXT; 3421af51c29fSKashyap D Desai sc->fw_supported_pd_count = MAX_PHYSICAL_DEVICES; 3422af51c29fSKashyap D Desai } else { 3423af51c29fSKashyap D Desai sc->fw_supported_vd_count = MAX_LOGICAL_DRIVES; 3424af51c29fSKashyap D Desai sc->fw_supported_pd_count = MAX_PHYSICAL_DEVICES; 3425af51c29fSKashyap D Desai } 3426af51c29fSKashyap D Desai 34274ad83576SKashyap D Desai if (sc->maxRaidMapSize) { 34284ad83576SKashyap D Desai ventura_map_sz = sc->maxRaidMapSize * 34294ad83576SKashyap D Desai MR_MIN_MAP_SIZE; 34304ad83576SKashyap D Desai sc->current_map_sz = ventura_map_sz; 34314ad83576SKashyap D Desai sc->max_map_sz = ventura_map_sz; 34324ad83576SKashyap D Desai } else { 3433af51c29fSKashyap D Desai sc->old_map_sz = sizeof(MR_FW_RAID_MAP) + 34344ad83576SKashyap D Desai (sizeof(MR_LD_SPAN_MAP) * (sc->fw_supported_vd_count - 1)); 3435af51c29fSKashyap D Desai sc->new_map_sz = sizeof(MR_FW_RAID_MAP_EXT); 3436af51c29fSKashyap D Desai sc->max_map_sz = max(sc->old_map_sz, sc->new_map_sz); 3437af51c29fSKashyap D Desai if (sc->max256vdSupport) 3438af51c29fSKashyap D Desai sc->current_map_sz = sc->new_map_sz; 3439af51c29fSKashyap D Desai else 3440af51c29fSKashyap D Desai sc->current_map_sz = sc->old_map_sz; 3441af51c29fSKashyap D Desai } 3442af51c29fSKashyap D Desai 34434ad83576SKashyap D Desai sc->drv_map_sz = sizeof(MR_DRV_RAID_MAP_ALL); 34444ad83576SKashyap D Desai #if VD_EXT_DEBUG 34454ad83576SKashyap D Desai device_printf(sc->mrsas_dev, "sc->maxRaidMapSize 0x%x \n", 34464ad83576SKashyap D Desai sc->maxRaidMapSize); 34474ad83576SKashyap D Desai device_printf(sc->mrsas_dev, 34484ad83576SKashyap D Desai "new_map_sz = 0x%x, old_map_sz = 0x%x, " 34494ad83576SKashyap D Desai "ventura_map_sz = 0x%x, current_map_sz = 0x%x " 34504ad83576SKashyap D Desai "fusion->drv_map_sz =0x%x, size of driver raid map 0x%lx \n", 34514ad83576SKashyap D Desai sc->new_map_sz, sc->old_map_sz, ventura_map_sz, 34524ad83576SKashyap D Desai sc->current_map_sz, sc->drv_map_sz, sizeof(MR_DRV_RAID_MAP_ALL)); 34534ad83576SKashyap D Desai #endif 34544ad83576SKashyap D Desai } 34554ad83576SKashyap D Desai 3456af51c29fSKashyap D Desai /* 3457665484d8SDoug Ambrisko * mrsas_alloc_ctlr_info_cmd: Allocates memory for controller info command 3458665484d8SDoug Ambrisko * input: Adapter soft state 3459665484d8SDoug Ambrisko * 3460665484d8SDoug Ambrisko * Allocates DMAable memory for the controller info internal command. 3461665484d8SDoug Ambrisko */ 34628e727371SKashyap D Desai int 34638e727371SKashyap D Desai mrsas_alloc_ctlr_info_cmd(struct mrsas_softc *sc) 3464665484d8SDoug Ambrisko { 3465665484d8SDoug Ambrisko int ctlr_info_size; 3466665484d8SDoug Ambrisko 3467665484d8SDoug Ambrisko /* Allocate get controller info command */ 3468665484d8SDoug Ambrisko ctlr_info_size = sizeof(struct mrsas_ctrl_info); 34698e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 34708e727371SKashyap D Desai 1, 0, 34718e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 34728e727371SKashyap D Desai BUS_SPACE_MAXADDR, 34738e727371SKashyap D Desai NULL, NULL, 34748e727371SKashyap D Desai ctlr_info_size, 34758e727371SKashyap D Desai 1, 34768e727371SKashyap D Desai ctlr_info_size, 34778e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 34788e727371SKashyap D Desai NULL, NULL, 3479665484d8SDoug Ambrisko &sc->ctlr_info_tag)) { 3480665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate ctlr info tag\n"); 3481665484d8SDoug Ambrisko return (ENOMEM); 3482665484d8SDoug Ambrisko } 3483665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->ctlr_info_tag, (void **)&sc->ctlr_info_mem, 3484665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->ctlr_info_dmamap)) { 3485665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate ctlr info cmd mem\n"); 3486665484d8SDoug Ambrisko return (ENOMEM); 3487665484d8SDoug Ambrisko } 3488665484d8SDoug Ambrisko if (bus_dmamap_load(sc->ctlr_info_tag, sc->ctlr_info_dmamap, 3489665484d8SDoug Ambrisko sc->ctlr_info_mem, ctlr_info_size, mrsas_addr_cb, 3490665484d8SDoug Ambrisko &sc->ctlr_info_phys_addr, BUS_DMA_NOWAIT)) { 3491665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load ctlr info cmd mem\n"); 3492665484d8SDoug Ambrisko return (ENOMEM); 3493665484d8SDoug Ambrisko } 3494665484d8SDoug Ambrisko memset(sc->ctlr_info_mem, 0, ctlr_info_size); 3495665484d8SDoug Ambrisko return (0); 3496665484d8SDoug Ambrisko } 3497665484d8SDoug Ambrisko 34988e727371SKashyap D Desai /* 3499665484d8SDoug Ambrisko * mrsas_free_ctlr_info_cmd: Free memory for controller info command 3500665484d8SDoug Ambrisko * input: Adapter soft state 3501665484d8SDoug Ambrisko * 3502665484d8SDoug Ambrisko * Deallocates memory of the get controller info cmd. 3503665484d8SDoug Ambrisko */ 35048e727371SKashyap D Desai void 35058e727371SKashyap D Desai mrsas_free_ctlr_info_cmd(struct mrsas_softc *sc) 3506665484d8SDoug Ambrisko { 3507665484d8SDoug Ambrisko if (sc->ctlr_info_phys_addr) 3508665484d8SDoug Ambrisko bus_dmamap_unload(sc->ctlr_info_tag, sc->ctlr_info_dmamap); 3509665484d8SDoug Ambrisko if (sc->ctlr_info_mem != NULL) 3510665484d8SDoug Ambrisko bus_dmamem_free(sc->ctlr_info_tag, sc->ctlr_info_mem, sc->ctlr_info_dmamap); 3511665484d8SDoug Ambrisko if (sc->ctlr_info_tag != NULL) 3512665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->ctlr_info_tag); 3513665484d8SDoug Ambrisko } 3514665484d8SDoug Ambrisko 35158e727371SKashyap D Desai /* 3516665484d8SDoug Ambrisko * mrsas_issue_polled: Issues a polling command 3517665484d8SDoug Ambrisko * inputs: Adapter soft state 3518665484d8SDoug Ambrisko * Command packet to be issued 3519665484d8SDoug Ambrisko * 35208e727371SKashyap D Desai * This function is for posting of internal commands to Firmware. MFI requires 35218e727371SKashyap D Desai * the cmd_status to be set to 0xFF before posting. The maximun wait time of 35228e727371SKashyap D Desai * the poll response timer is 180 seconds. 3523665484d8SDoug Ambrisko */ 35248e727371SKashyap D Desai int 35258e727371SKashyap D Desai mrsas_issue_polled(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 3526665484d8SDoug Ambrisko { 3527665484d8SDoug Ambrisko struct mrsas_header *frame_hdr = &cmd->frame->hdr; 3528665484d8SDoug Ambrisko u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME; 3529f0c7594bSKashyap D Desai int i, retcode = SUCCESS; 3530665484d8SDoug Ambrisko 3531665484d8SDoug Ambrisko frame_hdr->cmd_status = 0xFF; 3532665484d8SDoug Ambrisko frame_hdr->flags |= MFI_FRAME_DONT_POST_IN_REPLY_QUEUE; 3533665484d8SDoug Ambrisko 3534665484d8SDoug Ambrisko /* Issue the frame using inbound queue port */ 3535665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)) { 3536665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot issue DCMD internal command.\n"); 3537665484d8SDoug Ambrisko return (1); 3538665484d8SDoug Ambrisko } 3539665484d8SDoug Ambrisko /* 3540665484d8SDoug Ambrisko * Poll response timer to wait for Firmware response. While this 3541665484d8SDoug Ambrisko * timer with the DELAY call could block CPU, the time interval for 3542665484d8SDoug Ambrisko * this is only 1 millisecond. 3543665484d8SDoug Ambrisko */ 3544665484d8SDoug Ambrisko if (frame_hdr->cmd_status == 0xFF) { 3545665484d8SDoug Ambrisko for (i = 0; i < (max_wait * 1000); i++) { 3546665484d8SDoug Ambrisko if (frame_hdr->cmd_status == 0xFF) 3547665484d8SDoug Ambrisko DELAY(1000); 3548665484d8SDoug Ambrisko else 3549665484d8SDoug Ambrisko break; 3550665484d8SDoug Ambrisko } 3551665484d8SDoug Ambrisko } 3552f0c7594bSKashyap D Desai if (frame_hdr->cmd_status == 0xFF) { 3553f0c7594bSKashyap D Desai device_printf(sc->mrsas_dev, "DCMD timed out after %d " 3554f0c7594bSKashyap D Desai "seconds from %s\n", max_wait, __func__); 3555f0c7594bSKashyap D Desai device_printf(sc->mrsas_dev, "DCMD opcode 0x%X\n", 3556f0c7594bSKashyap D Desai cmd->frame->dcmd.opcode); 3557f0c7594bSKashyap D Desai retcode = ETIMEDOUT; 3558665484d8SDoug Ambrisko } 3559665484d8SDoug Ambrisko return (retcode); 3560665484d8SDoug Ambrisko } 3561665484d8SDoug Ambrisko 35628e727371SKashyap D Desai /* 35638e727371SKashyap D Desai * mrsas_issue_dcmd: Issues a MFI Pass thru cmd 35648e727371SKashyap D Desai * input: Adapter soft state mfi cmd pointer 3565665484d8SDoug Ambrisko * 3566665484d8SDoug Ambrisko * This function is called by mrsas_issued_blocked_cmd() and 35678e727371SKashyap D Desai * mrsas_issued_polled(), to build the MPT command and then fire the command 35688e727371SKashyap D Desai * to Firmware. 3569665484d8SDoug Ambrisko */ 3570665484d8SDoug Ambrisko int 3571665484d8SDoug Ambrisko mrsas_issue_dcmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 3572665484d8SDoug Ambrisko { 3573665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION *req_desc; 3574665484d8SDoug Ambrisko 3575665484d8SDoug Ambrisko req_desc = mrsas_build_mpt_cmd(sc, cmd); 3576665484d8SDoug Ambrisko if (!req_desc) { 3577665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot build MPT cmd.\n"); 3578665484d8SDoug Ambrisko return (1); 3579665484d8SDoug Ambrisko } 3580665484d8SDoug Ambrisko mrsas_fire_cmd(sc, req_desc->addr.u.low, req_desc->addr.u.high); 3581665484d8SDoug Ambrisko 3582665484d8SDoug Ambrisko return (0); 3583665484d8SDoug Ambrisko } 3584665484d8SDoug Ambrisko 35858e727371SKashyap D Desai /* 35868e727371SKashyap D Desai * mrsas_build_mpt_cmd: Calls helper function to build Passthru cmd 35878e727371SKashyap D Desai * input: Adapter soft state mfi cmd to build 3588665484d8SDoug Ambrisko * 35898e727371SKashyap D Desai * This function is called by mrsas_issue_cmd() to build the MPT-MFI passthru 35908e727371SKashyap D Desai * command and prepares the MPT command to send to Firmware. 3591665484d8SDoug Ambrisko */ 3592665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION * 3593665484d8SDoug Ambrisko mrsas_build_mpt_cmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 3594665484d8SDoug Ambrisko { 3595665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION *req_desc; 3596665484d8SDoug Ambrisko u_int16_t index; 3597665484d8SDoug Ambrisko 3598665484d8SDoug Ambrisko if (mrsas_build_mptmfi_passthru(sc, cmd)) { 3599665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot build MPT-MFI passthru cmd.\n"); 3600665484d8SDoug Ambrisko return NULL; 3601665484d8SDoug Ambrisko } 3602665484d8SDoug Ambrisko index = cmd->cmd_id.context.smid; 3603665484d8SDoug Ambrisko 3604665484d8SDoug Ambrisko req_desc = mrsas_get_request_desc(sc, index - 1); 3605665484d8SDoug Ambrisko if (!req_desc) 3606665484d8SDoug Ambrisko return NULL; 3607665484d8SDoug Ambrisko 3608665484d8SDoug Ambrisko req_desc->addr.Words = 0; 3609665484d8SDoug Ambrisko req_desc->SCSIIO.RequestFlags = (MPI2_REQ_DESCRIPT_FLAGS_SCSI_IO << MRSAS_REQ_DESCRIPT_FLAGS_TYPE_SHIFT); 3610665484d8SDoug Ambrisko 3611665484d8SDoug Ambrisko req_desc->SCSIIO.SMID = index; 3612665484d8SDoug Ambrisko 3613665484d8SDoug Ambrisko return (req_desc); 3614665484d8SDoug Ambrisko } 3615665484d8SDoug Ambrisko 36168e727371SKashyap D Desai /* 36178e727371SKashyap D Desai * mrsas_build_mptmfi_passthru: Builds a MPT MFI Passthru command 36188e727371SKashyap D Desai * input: Adapter soft state mfi cmd pointer 3619665484d8SDoug Ambrisko * 36208e727371SKashyap D Desai * The MPT command and the io_request are setup as a passthru command. The SGE 36218e727371SKashyap D Desai * chain address is set to frame_phys_addr of the MFI command. 3622665484d8SDoug Ambrisko */ 3623665484d8SDoug Ambrisko u_int8_t 3624665484d8SDoug Ambrisko mrsas_build_mptmfi_passthru(struct mrsas_softc *sc, struct mrsas_mfi_cmd *mfi_cmd) 3625665484d8SDoug Ambrisko { 3626665484d8SDoug Ambrisko MPI25_IEEE_SGE_CHAIN64 *mpi25_ieee_chain; 3627665484d8SDoug Ambrisko PTR_MRSAS_RAID_SCSI_IO_REQUEST io_req; 3628665484d8SDoug Ambrisko struct mrsas_mpt_cmd *mpt_cmd; 3629665484d8SDoug Ambrisko struct mrsas_header *frame_hdr = &mfi_cmd->frame->hdr; 3630665484d8SDoug Ambrisko 3631665484d8SDoug Ambrisko mpt_cmd = mrsas_get_mpt_cmd(sc); 3632665484d8SDoug Ambrisko if (!mpt_cmd) 3633665484d8SDoug Ambrisko return (1); 3634665484d8SDoug Ambrisko 3635665484d8SDoug Ambrisko /* Save the smid. To be used for returning the cmd */ 3636665484d8SDoug Ambrisko mfi_cmd->cmd_id.context.smid = mpt_cmd->index; 3637665484d8SDoug Ambrisko 3638665484d8SDoug Ambrisko mpt_cmd->sync_cmd_idx = mfi_cmd->index; 3639665484d8SDoug Ambrisko 3640665484d8SDoug Ambrisko /* 36418e727371SKashyap D Desai * For cmds where the flag is set, store the flag and check on 36428e727371SKashyap D Desai * completion. For cmds with this flag, don't call 3643665484d8SDoug Ambrisko * mrsas_complete_cmd. 3644665484d8SDoug Ambrisko */ 3645665484d8SDoug Ambrisko 3646665484d8SDoug Ambrisko if (frame_hdr->flags & MFI_FRAME_DONT_POST_IN_REPLY_QUEUE) 3647665484d8SDoug Ambrisko mpt_cmd->flags = MFI_FRAME_DONT_POST_IN_REPLY_QUEUE; 3648665484d8SDoug Ambrisko 3649665484d8SDoug Ambrisko io_req = mpt_cmd->io_request; 3650665484d8SDoug Ambrisko 36517aade8bfSKashyap D Desai if (sc->mrsas_gen3_ctrl || sc->is_ventura) { 3652665484d8SDoug Ambrisko pMpi25IeeeSgeChain64_t sgl_ptr_end = (pMpi25IeeeSgeChain64_t)&io_req->SGL; 36538e727371SKashyap D Desai 3654665484d8SDoug Ambrisko sgl_ptr_end += sc->max_sge_in_main_msg - 1; 3655665484d8SDoug Ambrisko sgl_ptr_end->Flags = 0; 3656665484d8SDoug Ambrisko } 3657665484d8SDoug Ambrisko mpi25_ieee_chain = (MPI25_IEEE_SGE_CHAIN64 *) & io_req->SGL.IeeeChain; 3658665484d8SDoug Ambrisko 3659665484d8SDoug Ambrisko io_req->Function = MRSAS_MPI2_FUNCTION_PASSTHRU_IO_REQUEST; 3660665484d8SDoug Ambrisko io_req->SGLOffset0 = offsetof(MRSAS_RAID_SCSI_IO_REQUEST, SGL) / 4; 3661665484d8SDoug Ambrisko io_req->ChainOffset = sc->chain_offset_mfi_pthru; 3662665484d8SDoug Ambrisko 3663665484d8SDoug Ambrisko mpi25_ieee_chain->Address = mfi_cmd->frame_phys_addr; 3664665484d8SDoug Ambrisko 3665665484d8SDoug Ambrisko mpi25_ieee_chain->Flags = IEEE_SGE_FLAGS_CHAIN_ELEMENT | 3666665484d8SDoug Ambrisko MPI2_IEEE_SGE_FLAGS_IOCPLBNTA_ADDR; 3667665484d8SDoug Ambrisko 36683a3fc6cbSKashyap D Desai mpi25_ieee_chain->Length = sc->max_chain_frame_sz; 3669665484d8SDoug Ambrisko 3670665484d8SDoug Ambrisko return (0); 3671665484d8SDoug Ambrisko } 3672665484d8SDoug Ambrisko 36738e727371SKashyap D Desai /* 36748e727371SKashyap D Desai * mrsas_issue_blocked_cmd: Synchronous wrapper around regular FW cmds 36758e727371SKashyap D Desai * input: Adapter soft state Command to be issued 3676665484d8SDoug Ambrisko * 36778e727371SKashyap D Desai * This function waits on an event for the command to be returned from the ISR. 36788e727371SKashyap D Desai * Max wait time is MRSAS_INTERNAL_CMD_WAIT_TIME secs. Used for issuing 36798e727371SKashyap D Desai * internal and ioctl commands. 3680665484d8SDoug Ambrisko */ 36818e727371SKashyap D Desai int 36828e727371SKashyap D Desai mrsas_issue_blocked_cmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 3683665484d8SDoug Ambrisko { 3684665484d8SDoug Ambrisko u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME; 3685665484d8SDoug Ambrisko unsigned long total_time = 0; 3686f0c7594bSKashyap D Desai int retcode = SUCCESS; 3687665484d8SDoug Ambrisko 3688665484d8SDoug Ambrisko /* Initialize cmd_status */ 3689f0c7594bSKashyap D Desai cmd->cmd_status = 0xFF; 3690665484d8SDoug Ambrisko 3691665484d8SDoug Ambrisko /* Build MPT-MFI command for issue to FW */ 3692665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)) { 3693665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot issue DCMD internal command.\n"); 3694665484d8SDoug Ambrisko return (1); 3695665484d8SDoug Ambrisko } 3696665484d8SDoug Ambrisko sc->chan = (void *)&cmd; 3697665484d8SDoug Ambrisko 3698665484d8SDoug Ambrisko while (1) { 3699f0c7594bSKashyap D Desai if (cmd->cmd_status == 0xFF) { 3700665484d8SDoug Ambrisko tsleep((void *)&sc->chan, 0, "mrsas_sleep", hz); 37018e727371SKashyap D Desai } else 3702665484d8SDoug Ambrisko break; 3703f0c7594bSKashyap D Desai 3704f0c7594bSKashyap D Desai if (!cmd->sync_cmd) { /* cmd->sync will be set for an IOCTL 3705f0c7594bSKashyap D Desai * command */ 3706665484d8SDoug Ambrisko total_time++; 3707665484d8SDoug Ambrisko if (total_time >= max_wait) { 37088e727371SKashyap D Desai device_printf(sc->mrsas_dev, 37098e727371SKashyap D Desai "Internal command timed out after %d seconds.\n", max_wait); 3710665484d8SDoug Ambrisko retcode = 1; 3711665484d8SDoug Ambrisko break; 3712665484d8SDoug Ambrisko } 3713665484d8SDoug Ambrisko } 3714f0c7594bSKashyap D Desai } 3715f0c7594bSKashyap D Desai 3716f0c7594bSKashyap D Desai if (cmd->cmd_status == 0xFF) { 3717f0c7594bSKashyap D Desai device_printf(sc->mrsas_dev, "DCMD timed out after %d " 3718f0c7594bSKashyap D Desai "seconds from %s\n", max_wait, __func__); 3719f0c7594bSKashyap D Desai device_printf(sc->mrsas_dev, "DCMD opcode 0x%X\n", 3720f0c7594bSKashyap D Desai cmd->frame->dcmd.opcode); 3721f0c7594bSKashyap D Desai retcode = ETIMEDOUT; 3722f0c7594bSKashyap D Desai } 3723665484d8SDoug Ambrisko return (retcode); 3724665484d8SDoug Ambrisko } 3725665484d8SDoug Ambrisko 37268e727371SKashyap D Desai /* 37278e727371SKashyap D Desai * mrsas_complete_mptmfi_passthru: Completes a command 37288e727371SKashyap D Desai * input: @sc: Adapter soft state 37298e727371SKashyap D Desai * @cmd: Command to be completed 37308e727371SKashyap D Desai * @status: cmd completion status 3731665484d8SDoug Ambrisko * 37328e727371SKashyap D Desai * This function is called from mrsas_complete_cmd() after an interrupt is 37338e727371SKashyap D Desai * received from Firmware, and io_request->Function is 3734665484d8SDoug Ambrisko * MRSAS_MPI2_FUNCTION_PASSTHRU_IO_REQUEST. 3735665484d8SDoug Ambrisko */ 3736665484d8SDoug Ambrisko void 3737665484d8SDoug Ambrisko mrsas_complete_mptmfi_passthru(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd, 3738665484d8SDoug Ambrisko u_int8_t status) 3739665484d8SDoug Ambrisko { 3740665484d8SDoug Ambrisko struct mrsas_header *hdr = &cmd->frame->hdr; 3741665484d8SDoug Ambrisko u_int8_t cmd_status = cmd->frame->hdr.cmd_status; 3742665484d8SDoug Ambrisko 3743665484d8SDoug Ambrisko /* Reset the retry counter for future re-tries */ 3744665484d8SDoug Ambrisko cmd->retry_for_fw_reset = 0; 3745665484d8SDoug Ambrisko 3746665484d8SDoug Ambrisko if (cmd->ccb_ptr) 3747665484d8SDoug Ambrisko cmd->ccb_ptr = NULL; 3748665484d8SDoug Ambrisko 3749665484d8SDoug Ambrisko switch (hdr->cmd) { 3750665484d8SDoug Ambrisko case MFI_CMD_INVALID: 3751665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "MFI_CMD_INVALID command.\n"); 3752665484d8SDoug Ambrisko break; 3753665484d8SDoug Ambrisko case MFI_CMD_PD_SCSI_IO: 3754665484d8SDoug Ambrisko case MFI_CMD_LD_SCSI_IO: 3755665484d8SDoug Ambrisko /* 3756665484d8SDoug Ambrisko * MFI_CMD_PD_SCSI_IO and MFI_CMD_LD_SCSI_IO could have been 3757665484d8SDoug Ambrisko * issued either through an IO path or an IOCTL path. If it 3758665484d8SDoug Ambrisko * was via IOCTL, we will send it to internal completion. 3759665484d8SDoug Ambrisko */ 3760665484d8SDoug Ambrisko if (cmd->sync_cmd) { 3761665484d8SDoug Ambrisko cmd->sync_cmd = 0; 3762665484d8SDoug Ambrisko mrsas_wakeup(sc, cmd); 3763665484d8SDoug Ambrisko break; 3764665484d8SDoug Ambrisko } 3765665484d8SDoug Ambrisko case MFI_CMD_SMP: 3766665484d8SDoug Ambrisko case MFI_CMD_STP: 3767665484d8SDoug Ambrisko case MFI_CMD_DCMD: 3768665484d8SDoug Ambrisko /* Check for LD map update */ 3769665484d8SDoug Ambrisko if ((cmd->frame->dcmd.opcode == MR_DCMD_LD_MAP_GET_INFO) && 3770665484d8SDoug Ambrisko (cmd->frame->dcmd.mbox.b[1] == 1)) { 3771665484d8SDoug Ambrisko sc->fast_path_io = 0; 3772665484d8SDoug Ambrisko mtx_lock(&sc->raidmap_lock); 3773f0c7594bSKashyap D Desai sc->map_update_cmd = NULL; 3774665484d8SDoug Ambrisko if (cmd_status != 0) { 3775665484d8SDoug Ambrisko if (cmd_status != MFI_STAT_NOT_FOUND) 3776665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "map sync failed, status=%x\n", cmd_status); 3777665484d8SDoug Ambrisko else { 3778665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3779665484d8SDoug Ambrisko mtx_unlock(&sc->raidmap_lock); 3780665484d8SDoug Ambrisko break; 3781665484d8SDoug Ambrisko } 37828e727371SKashyap D Desai } else 3783665484d8SDoug Ambrisko sc->map_id++; 3784665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3785665484d8SDoug Ambrisko if (MR_ValidateMapInfo(sc)) 3786665484d8SDoug Ambrisko sc->fast_path_io = 0; 3787665484d8SDoug Ambrisko else 3788665484d8SDoug Ambrisko sc->fast_path_io = 1; 3789665484d8SDoug Ambrisko mrsas_sync_map_info(sc); 3790665484d8SDoug Ambrisko mtx_unlock(&sc->raidmap_lock); 3791665484d8SDoug Ambrisko break; 3792665484d8SDoug Ambrisko } 3793665484d8SDoug Ambrisko if (cmd->frame->dcmd.opcode == MR_DCMD_CTRL_EVENT_GET_INFO || 3794665484d8SDoug Ambrisko cmd->frame->dcmd.opcode == MR_DCMD_CTRL_EVENT_GET) { 3795da011113SKashyap D Desai sc->mrsas_aen_triggered = 0; 3796665484d8SDoug Ambrisko } 3797a688fcd0SKashyap D Desai /* FW has an updated PD sequence */ 3798a688fcd0SKashyap D Desai if ((cmd->frame->dcmd.opcode == 3799a688fcd0SKashyap D Desai MR_DCMD_SYSTEM_PD_MAP_GET_INFO) && 3800a688fcd0SKashyap D Desai (cmd->frame->dcmd.mbox.b[0] == 1)) { 3801a688fcd0SKashyap D Desai 3802a688fcd0SKashyap D Desai mtx_lock(&sc->raidmap_lock); 3803a688fcd0SKashyap D Desai sc->jbod_seq_cmd = NULL; 3804a688fcd0SKashyap D Desai mrsas_release_mfi_cmd(cmd); 3805a688fcd0SKashyap D Desai 3806a688fcd0SKashyap D Desai if (cmd_status == MFI_STAT_OK) { 3807a688fcd0SKashyap D Desai sc->pd_seq_map_id++; 3808a688fcd0SKashyap D Desai /* Re-register a pd sync seq num cmd */ 3809a688fcd0SKashyap D Desai if (megasas_sync_pd_seq_num(sc, true)) 3810a688fcd0SKashyap D Desai sc->use_seqnum_jbod_fp = 0; 3811a688fcd0SKashyap D Desai } else { 3812a688fcd0SKashyap D Desai sc->use_seqnum_jbod_fp = 0; 3813a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, 3814a688fcd0SKashyap D Desai "Jbod map sync failed, status=%x\n", cmd_status); 3815a688fcd0SKashyap D Desai } 3816a688fcd0SKashyap D Desai mtx_unlock(&sc->raidmap_lock); 3817a688fcd0SKashyap D Desai break; 3818a688fcd0SKashyap D Desai } 3819665484d8SDoug Ambrisko /* See if got an event notification */ 3820665484d8SDoug Ambrisko if (cmd->frame->dcmd.opcode == MR_DCMD_CTRL_EVENT_WAIT) 3821665484d8SDoug Ambrisko mrsas_complete_aen(sc, cmd); 3822665484d8SDoug Ambrisko else 3823665484d8SDoug Ambrisko mrsas_wakeup(sc, cmd); 3824665484d8SDoug Ambrisko break; 3825665484d8SDoug Ambrisko case MFI_CMD_ABORT: 3826665484d8SDoug Ambrisko /* Command issued to abort another cmd return */ 3827665484d8SDoug Ambrisko mrsas_complete_abort(sc, cmd); 3828665484d8SDoug Ambrisko break; 3829665484d8SDoug Ambrisko default: 3830665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Unknown command completed! [0x%X]\n", hdr->cmd); 3831665484d8SDoug Ambrisko break; 3832665484d8SDoug Ambrisko } 3833665484d8SDoug Ambrisko } 3834665484d8SDoug Ambrisko 38358e727371SKashyap D Desai /* 38368e727371SKashyap D Desai * mrsas_wakeup: Completes an internal command 3837665484d8SDoug Ambrisko * input: Adapter soft state 3838665484d8SDoug Ambrisko * Command to be completed 3839665484d8SDoug Ambrisko * 38408e727371SKashyap D Desai * In mrsas_issue_blocked_cmd(), after a command is issued to Firmware, a wait 38418e727371SKashyap D Desai * timer is started. This function is called from 38428e727371SKashyap D Desai * mrsas_complete_mptmfi_passthru() as it completes the command, to wake up 38438e727371SKashyap D Desai * from the command wait. 3844665484d8SDoug Ambrisko */ 38458e727371SKashyap D Desai void 38468e727371SKashyap D Desai mrsas_wakeup(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 3847665484d8SDoug Ambrisko { 3848665484d8SDoug Ambrisko cmd->cmd_status = cmd->frame->io.cmd_status; 3849665484d8SDoug Ambrisko 3850f0c7594bSKashyap D Desai if (cmd->cmd_status == 0xFF) 3851665484d8SDoug Ambrisko cmd->cmd_status = 0; 3852665484d8SDoug Ambrisko 3853665484d8SDoug Ambrisko sc->chan = (void *)&cmd; 3854665484d8SDoug Ambrisko wakeup_one((void *)&sc->chan); 3855665484d8SDoug Ambrisko return; 3856665484d8SDoug Ambrisko } 3857665484d8SDoug Ambrisko 38588e727371SKashyap D Desai /* 38598e727371SKashyap D Desai * mrsas_shutdown_ctlr: Instructs FW to shutdown the controller input: 38608e727371SKashyap D Desai * Adapter soft state Shutdown/Hibernate 3861665484d8SDoug Ambrisko * 38628e727371SKashyap D Desai * This function issues a DCMD internal command to Firmware to initiate shutdown 38638e727371SKashyap D Desai * of the controller. 3864665484d8SDoug Ambrisko */ 38658e727371SKashyap D Desai static void 38668e727371SKashyap D Desai mrsas_shutdown_ctlr(struct mrsas_softc *sc, u_int32_t opcode) 3867665484d8SDoug Ambrisko { 3868665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3869665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 3870665484d8SDoug Ambrisko 3871665484d8SDoug Ambrisko if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) 3872665484d8SDoug Ambrisko return; 3873665484d8SDoug Ambrisko 3874665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3875665484d8SDoug Ambrisko if (!cmd) { 3876665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate for shutdown cmd.\n"); 3877665484d8SDoug Ambrisko return; 3878665484d8SDoug Ambrisko } 3879665484d8SDoug Ambrisko if (sc->aen_cmd) 3880665484d8SDoug Ambrisko mrsas_issue_blocked_abort_cmd(sc, sc->aen_cmd); 3881665484d8SDoug Ambrisko if (sc->map_update_cmd) 3882665484d8SDoug Ambrisko mrsas_issue_blocked_abort_cmd(sc, sc->map_update_cmd); 3883a688fcd0SKashyap D Desai if (sc->jbod_seq_cmd) 3884a688fcd0SKashyap D Desai mrsas_issue_blocked_abort_cmd(sc, sc->jbod_seq_cmd); 3885665484d8SDoug Ambrisko 3886665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 3887665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 3888665484d8SDoug Ambrisko 3889665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 3890665484d8SDoug Ambrisko dcmd->cmd_status = 0x0; 3891665484d8SDoug Ambrisko dcmd->sge_count = 0; 3892665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_NONE; 3893665484d8SDoug Ambrisko dcmd->timeout = 0; 3894665484d8SDoug Ambrisko dcmd->pad_0 = 0; 3895665484d8SDoug Ambrisko dcmd->data_xfer_len = 0; 3896665484d8SDoug Ambrisko dcmd->opcode = opcode; 3897665484d8SDoug Ambrisko 3898665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Preparing to shut down controller.\n"); 3899665484d8SDoug Ambrisko 3900665484d8SDoug Ambrisko mrsas_issue_blocked_cmd(sc, cmd); 3901665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3902665484d8SDoug Ambrisko 3903665484d8SDoug Ambrisko return; 3904665484d8SDoug Ambrisko } 3905665484d8SDoug Ambrisko 39068e727371SKashyap D Desai /* 39078e727371SKashyap D Desai * mrsas_flush_cache: Requests FW to flush all its caches input: 39088e727371SKashyap D Desai * Adapter soft state 3909665484d8SDoug Ambrisko * 3910665484d8SDoug Ambrisko * This function is issues a DCMD internal command to Firmware to initiate 3911665484d8SDoug Ambrisko * flushing of all caches. 3912665484d8SDoug Ambrisko */ 39138e727371SKashyap D Desai static void 39148e727371SKashyap D Desai mrsas_flush_cache(struct mrsas_softc *sc) 3915665484d8SDoug Ambrisko { 3916665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3917665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 3918665484d8SDoug Ambrisko 3919665484d8SDoug Ambrisko if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) 3920665484d8SDoug Ambrisko return; 3921665484d8SDoug Ambrisko 3922665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3923665484d8SDoug Ambrisko if (!cmd) { 3924665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate for flush cache cmd.\n"); 3925665484d8SDoug Ambrisko return; 3926665484d8SDoug Ambrisko } 3927665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 3928665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 3929665484d8SDoug Ambrisko 3930665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 3931665484d8SDoug Ambrisko dcmd->cmd_status = 0x0; 3932665484d8SDoug Ambrisko dcmd->sge_count = 0; 3933665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_NONE; 3934665484d8SDoug Ambrisko dcmd->timeout = 0; 3935665484d8SDoug Ambrisko dcmd->pad_0 = 0; 3936665484d8SDoug Ambrisko dcmd->data_xfer_len = 0; 3937665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_CTRL_CACHE_FLUSH; 3938665484d8SDoug Ambrisko dcmd->mbox.b[0] = MR_FLUSH_CTRL_CACHE | MR_FLUSH_DISK_CACHE; 3939665484d8SDoug Ambrisko 3940665484d8SDoug Ambrisko mrsas_issue_blocked_cmd(sc, cmd); 3941665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3942665484d8SDoug Ambrisko 3943665484d8SDoug Ambrisko return; 3944665484d8SDoug Ambrisko } 3945665484d8SDoug Ambrisko 3946a688fcd0SKashyap D Desai int 3947a688fcd0SKashyap D Desai megasas_sync_pd_seq_num(struct mrsas_softc *sc, boolean_t pend) 3948a688fcd0SKashyap D Desai { 3949a688fcd0SKashyap D Desai int retcode = 0; 3950a688fcd0SKashyap D Desai u_int8_t do_ocr = 1; 3951a688fcd0SKashyap D Desai struct mrsas_mfi_cmd *cmd; 3952a688fcd0SKashyap D Desai struct mrsas_dcmd_frame *dcmd; 3953a688fcd0SKashyap D Desai uint32_t pd_seq_map_sz; 3954a688fcd0SKashyap D Desai struct MR_PD_CFG_SEQ_NUM_SYNC *pd_sync; 3955a688fcd0SKashyap D Desai bus_addr_t pd_seq_h; 3956a688fcd0SKashyap D Desai 3957a688fcd0SKashyap D Desai pd_seq_map_sz = sizeof(struct MR_PD_CFG_SEQ_NUM_SYNC) + 3958a688fcd0SKashyap D Desai (sizeof(struct MR_PD_CFG_SEQ) * 3959a688fcd0SKashyap D Desai (MAX_PHYSICAL_DEVICES - 1)); 3960a688fcd0SKashyap D Desai 3961a688fcd0SKashyap D Desai cmd = mrsas_get_mfi_cmd(sc); 3962a688fcd0SKashyap D Desai if (!cmd) { 3963a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, 3964a688fcd0SKashyap D Desai "Cannot alloc for ld map info cmd.\n"); 3965a688fcd0SKashyap D Desai return 1; 3966a688fcd0SKashyap D Desai } 3967a688fcd0SKashyap D Desai dcmd = &cmd->frame->dcmd; 3968a688fcd0SKashyap D Desai 3969a688fcd0SKashyap D Desai pd_sync = (void *)sc->jbodmap_mem[(sc->pd_seq_map_id & 1)]; 3970a688fcd0SKashyap D Desai pd_seq_h = sc->jbodmap_phys_addr[(sc->pd_seq_map_id & 1)]; 3971a688fcd0SKashyap D Desai if (!pd_sync) { 3972a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, 3973a688fcd0SKashyap D Desai "Failed to alloc mem for jbod map info.\n"); 3974a688fcd0SKashyap D Desai mrsas_release_mfi_cmd(cmd); 3975a688fcd0SKashyap D Desai return (ENOMEM); 3976a688fcd0SKashyap D Desai } 3977a688fcd0SKashyap D Desai memset(pd_sync, 0, pd_seq_map_sz); 3978a688fcd0SKashyap D Desai memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 3979a688fcd0SKashyap D Desai dcmd->cmd = MFI_CMD_DCMD; 3980a688fcd0SKashyap D Desai dcmd->cmd_status = 0xFF; 3981a688fcd0SKashyap D Desai dcmd->sge_count = 1; 3982a688fcd0SKashyap D Desai dcmd->timeout = 0; 3983a688fcd0SKashyap D Desai dcmd->pad_0 = 0; 3984a688fcd0SKashyap D Desai dcmd->data_xfer_len = (pd_seq_map_sz); 3985a688fcd0SKashyap D Desai dcmd->opcode = (MR_DCMD_SYSTEM_PD_MAP_GET_INFO); 3986a688fcd0SKashyap D Desai dcmd->sgl.sge32[0].phys_addr = (pd_seq_h); 3987a688fcd0SKashyap D Desai dcmd->sgl.sge32[0].length = (pd_seq_map_sz); 3988a688fcd0SKashyap D Desai 3989a688fcd0SKashyap D Desai if (pend) { 3990a688fcd0SKashyap D Desai dcmd->mbox.b[0] = MRSAS_DCMD_MBOX_PEND_FLAG; 3991a688fcd0SKashyap D Desai dcmd->flags = (MFI_FRAME_DIR_WRITE); 3992a688fcd0SKashyap D Desai sc->jbod_seq_cmd = cmd; 3993a688fcd0SKashyap D Desai if (mrsas_issue_dcmd(sc, cmd)) { 3994a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, 3995a688fcd0SKashyap D Desai "Fail to send sync map info command.\n"); 3996a688fcd0SKashyap D Desai return 1; 3997a688fcd0SKashyap D Desai } else 3998a688fcd0SKashyap D Desai return 0; 3999a688fcd0SKashyap D Desai } else 4000a688fcd0SKashyap D Desai dcmd->flags = MFI_FRAME_DIR_READ; 4001a688fcd0SKashyap D Desai 4002a688fcd0SKashyap D Desai retcode = mrsas_issue_polled(sc, cmd); 4003a688fcd0SKashyap D Desai if (retcode == ETIMEDOUT) 4004a688fcd0SKashyap D Desai goto dcmd_timeout; 4005a688fcd0SKashyap D Desai 4006a688fcd0SKashyap D Desai if (pd_sync->count > MAX_PHYSICAL_DEVICES) { 4007a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, 4008a688fcd0SKashyap D Desai "driver supports max %d JBOD, but FW reports %d\n", 4009a688fcd0SKashyap D Desai MAX_PHYSICAL_DEVICES, pd_sync->count); 4010a688fcd0SKashyap D Desai retcode = -EINVAL; 4011a688fcd0SKashyap D Desai } 4012a688fcd0SKashyap D Desai if (!retcode) 4013a688fcd0SKashyap D Desai sc->pd_seq_map_id++; 4014a688fcd0SKashyap D Desai do_ocr = 0; 4015a688fcd0SKashyap D Desai 4016a688fcd0SKashyap D Desai dcmd_timeout: 4017a688fcd0SKashyap D Desai if (do_ocr) 4018a688fcd0SKashyap D Desai sc->do_timedout_reset = MFI_DCMD_TIMEOUT_OCR; 4019a688fcd0SKashyap D Desai 4020a688fcd0SKashyap D Desai return (retcode); 4021a688fcd0SKashyap D Desai } 4022a688fcd0SKashyap D Desai 40238e727371SKashyap D Desai /* 40248e727371SKashyap D Desai * mrsas_get_map_info: Load and validate RAID map input: 40258e727371SKashyap D Desai * Adapter instance soft state 4026665484d8SDoug Ambrisko * 40278e727371SKashyap D Desai * This function calls mrsas_get_ld_map_info() and MR_ValidateMapInfo() to load 40288e727371SKashyap D Desai * and validate RAID map. It returns 0 if successful, 1 other- wise. 4029665484d8SDoug Ambrisko */ 40308e727371SKashyap D Desai static int 40318e727371SKashyap D Desai mrsas_get_map_info(struct mrsas_softc *sc) 4032665484d8SDoug Ambrisko { 4033665484d8SDoug Ambrisko uint8_t retcode = 0; 4034665484d8SDoug Ambrisko 4035665484d8SDoug Ambrisko sc->fast_path_io = 0; 4036665484d8SDoug Ambrisko if (!mrsas_get_ld_map_info(sc)) { 4037665484d8SDoug Ambrisko retcode = MR_ValidateMapInfo(sc); 4038665484d8SDoug Ambrisko if (retcode == 0) { 4039665484d8SDoug Ambrisko sc->fast_path_io = 1; 4040665484d8SDoug Ambrisko return 0; 4041665484d8SDoug Ambrisko } 4042665484d8SDoug Ambrisko } 4043665484d8SDoug Ambrisko return 1; 4044665484d8SDoug Ambrisko } 4045665484d8SDoug Ambrisko 40468e727371SKashyap D Desai /* 40478e727371SKashyap D Desai * mrsas_get_ld_map_info: Get FW's ld_map structure input: 40488e727371SKashyap D Desai * Adapter instance soft state 4049665484d8SDoug Ambrisko * 40508e727371SKashyap D Desai * Issues an internal command (DCMD) to get the FW's controller PD list 40518e727371SKashyap D Desai * structure. 4052665484d8SDoug Ambrisko */ 40538e727371SKashyap D Desai static int 40548e727371SKashyap D Desai mrsas_get_ld_map_info(struct mrsas_softc *sc) 4055665484d8SDoug Ambrisko { 4056665484d8SDoug Ambrisko int retcode = 0; 4057665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 4058665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 40594799d485SKashyap D Desai void *map; 4060665484d8SDoug Ambrisko bus_addr_t map_phys_addr = 0; 4061665484d8SDoug Ambrisko 4062665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 4063665484d8SDoug Ambrisko if (!cmd) { 40644799d485SKashyap D Desai device_printf(sc->mrsas_dev, 40654799d485SKashyap D Desai "Cannot alloc for ld map info cmd.\n"); 4066665484d8SDoug Ambrisko return 1; 4067665484d8SDoug Ambrisko } 4068665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 4069665484d8SDoug Ambrisko 40704799d485SKashyap D Desai map = (void *)sc->raidmap_mem[(sc->map_id & 1)]; 4071665484d8SDoug Ambrisko map_phys_addr = sc->raidmap_phys_addr[(sc->map_id & 1)]; 4072665484d8SDoug Ambrisko if (!map) { 40734799d485SKashyap D Desai device_printf(sc->mrsas_dev, 40744799d485SKashyap D Desai "Failed to alloc mem for ld map info.\n"); 4075665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 4076665484d8SDoug Ambrisko return (ENOMEM); 4077665484d8SDoug Ambrisko } 40784799d485SKashyap D Desai memset(map, 0, sizeof(sc->max_map_sz)); 4079665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 4080665484d8SDoug Ambrisko 4081665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 4082665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 4083665484d8SDoug Ambrisko dcmd->sge_count = 1; 4084665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 4085665484d8SDoug Ambrisko dcmd->timeout = 0; 4086665484d8SDoug Ambrisko dcmd->pad_0 = 0; 40874799d485SKashyap D Desai dcmd->data_xfer_len = sc->current_map_sz; 4088665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_LD_MAP_GET_INFO; 4089665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = map_phys_addr; 40904799d485SKashyap D Desai dcmd->sgl.sge32[0].length = sc->current_map_sz; 40914799d485SKashyap D Desai 4092f0c7594bSKashyap D Desai retcode = mrsas_issue_polled(sc, cmd); 4093f0c7594bSKashyap D Desai if (retcode == ETIMEDOUT) 4094f0c7594bSKashyap D Desai sc->do_timedout_reset = MFI_DCMD_TIMEOUT_OCR; 40954799d485SKashyap D Desai 4096665484d8SDoug Ambrisko return (retcode); 4097665484d8SDoug Ambrisko } 4098665484d8SDoug Ambrisko 40998e727371SKashyap D Desai /* 41008e727371SKashyap D Desai * mrsas_sync_map_info: Get FW's ld_map structure input: 41018e727371SKashyap D Desai * Adapter instance soft state 4102665484d8SDoug Ambrisko * 41038e727371SKashyap D Desai * Issues an internal command (DCMD) to get the FW's controller PD list 41048e727371SKashyap D Desai * structure. 4105665484d8SDoug Ambrisko */ 41068e727371SKashyap D Desai static int 41078e727371SKashyap D Desai mrsas_sync_map_info(struct mrsas_softc *sc) 4108665484d8SDoug Ambrisko { 4109665484d8SDoug Ambrisko int retcode = 0, i; 4110665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 4111665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 4112665484d8SDoug Ambrisko uint32_t size_sync_info, num_lds; 4113665484d8SDoug Ambrisko MR_LD_TARGET_SYNC *target_map = NULL; 41144799d485SKashyap D Desai MR_DRV_RAID_MAP_ALL *map; 4115665484d8SDoug Ambrisko MR_LD_RAID *raid; 4116665484d8SDoug Ambrisko MR_LD_TARGET_SYNC *ld_sync; 4117665484d8SDoug Ambrisko bus_addr_t map_phys_addr = 0; 4118665484d8SDoug Ambrisko 4119665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 4120665484d8SDoug Ambrisko if (!cmd) { 4121731b7561SKashyap D Desai device_printf(sc->mrsas_dev, "Cannot alloc for sync map info cmd\n"); 4122731b7561SKashyap D Desai return ENOMEM; 4123665484d8SDoug Ambrisko } 41244799d485SKashyap D Desai map = sc->ld_drv_map[sc->map_id & 1]; 4125665484d8SDoug Ambrisko num_lds = map->raidMap.ldCount; 4126665484d8SDoug Ambrisko 4127665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 4128665484d8SDoug Ambrisko size_sync_info = sizeof(MR_LD_TARGET_SYNC) * num_lds; 4129665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 4130665484d8SDoug Ambrisko 41318e727371SKashyap D Desai target_map = (MR_LD_TARGET_SYNC *) sc->raidmap_mem[(sc->map_id - 1) & 1]; 41324799d485SKashyap D Desai memset(target_map, 0, sc->max_map_sz); 4133665484d8SDoug Ambrisko 4134665484d8SDoug Ambrisko map_phys_addr = sc->raidmap_phys_addr[(sc->map_id - 1) & 1]; 4135665484d8SDoug Ambrisko 4136665484d8SDoug Ambrisko ld_sync = (MR_LD_TARGET_SYNC *) target_map; 4137665484d8SDoug Ambrisko 4138665484d8SDoug Ambrisko for (i = 0; i < num_lds; i++, ld_sync++) { 4139665484d8SDoug Ambrisko raid = MR_LdRaidGet(i, map); 4140665484d8SDoug Ambrisko ld_sync->targetId = MR_GetLDTgtId(i, map); 4141665484d8SDoug Ambrisko ld_sync->seqNum = raid->seqNum; 4142665484d8SDoug Ambrisko } 4143665484d8SDoug Ambrisko 4144665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 4145665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 4146665484d8SDoug Ambrisko dcmd->sge_count = 1; 4147665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_WRITE; 4148665484d8SDoug Ambrisko dcmd->timeout = 0; 4149665484d8SDoug Ambrisko dcmd->pad_0 = 0; 41504799d485SKashyap D Desai dcmd->data_xfer_len = sc->current_map_sz; 4151665484d8SDoug Ambrisko dcmd->mbox.b[0] = num_lds; 4152665484d8SDoug Ambrisko dcmd->mbox.b[1] = MRSAS_DCMD_MBOX_PEND_FLAG; 4153665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_LD_MAP_GET_INFO; 4154665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = map_phys_addr; 41554799d485SKashyap D Desai dcmd->sgl.sge32[0].length = sc->current_map_sz; 4156665484d8SDoug Ambrisko 4157665484d8SDoug Ambrisko sc->map_update_cmd = cmd; 4158665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)) { 41594799d485SKashyap D Desai device_printf(sc->mrsas_dev, 41604799d485SKashyap D Desai "Fail to send sync map info command.\n"); 4161665484d8SDoug Ambrisko return (1); 4162665484d8SDoug Ambrisko } 4163665484d8SDoug Ambrisko return (retcode); 4164665484d8SDoug Ambrisko } 4165665484d8SDoug Ambrisko 41668e727371SKashyap D Desai /* 41678e727371SKashyap D Desai * mrsas_get_pd_list: Returns FW's PD list structure input: 41688e727371SKashyap D Desai * Adapter soft state 4169665484d8SDoug Ambrisko * 41708e727371SKashyap D Desai * Issues an internal command (DCMD) to get the FW's controller PD list 41718e727371SKashyap D Desai * structure. This information is mainly used to find out about system 41728e727371SKashyap D Desai * supported by Firmware. 4173665484d8SDoug Ambrisko */ 41748e727371SKashyap D Desai static int 41758e727371SKashyap D Desai mrsas_get_pd_list(struct mrsas_softc *sc) 4176665484d8SDoug Ambrisko { 4177665484d8SDoug Ambrisko int retcode = 0, pd_index = 0, pd_count = 0, pd_list_size; 4178f0c7594bSKashyap D Desai u_int8_t do_ocr = 1; 4179665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 4180665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 4181665484d8SDoug Ambrisko struct MR_PD_LIST *pd_list_mem; 4182665484d8SDoug Ambrisko struct MR_PD_ADDRESS *pd_addr; 4183665484d8SDoug Ambrisko bus_addr_t pd_list_phys_addr = 0; 4184665484d8SDoug Ambrisko struct mrsas_tmp_dcmd *tcmd; 4185665484d8SDoug Ambrisko 4186665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 4187665484d8SDoug Ambrisko if (!cmd) { 41884799d485SKashyap D Desai device_printf(sc->mrsas_dev, 41894799d485SKashyap D Desai "Cannot alloc for get PD list cmd\n"); 4190665484d8SDoug Ambrisko return 1; 4191665484d8SDoug Ambrisko } 4192665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 4193665484d8SDoug Ambrisko 4194665484d8SDoug Ambrisko tcmd = malloc(sizeof(struct mrsas_tmp_dcmd), M_MRSAS, M_NOWAIT); 4195665484d8SDoug Ambrisko pd_list_size = MRSAS_MAX_PD * sizeof(struct MR_PD_LIST); 4196665484d8SDoug Ambrisko if (mrsas_alloc_tmp_dcmd(sc, tcmd, pd_list_size) != SUCCESS) { 41974799d485SKashyap D Desai device_printf(sc->mrsas_dev, 41984799d485SKashyap D Desai "Cannot alloc dmamap for get PD list cmd\n"); 4199665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 4200f0c7594bSKashyap D Desai mrsas_free_tmp_dcmd(tcmd); 4201f0c7594bSKashyap D Desai free(tcmd, M_MRSAS); 4202665484d8SDoug Ambrisko return (ENOMEM); 42038e727371SKashyap D Desai } else { 4204665484d8SDoug Ambrisko pd_list_mem = tcmd->tmp_dcmd_mem; 4205665484d8SDoug Ambrisko pd_list_phys_addr = tcmd->tmp_dcmd_phys_addr; 4206665484d8SDoug Ambrisko } 4207665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 4208665484d8SDoug Ambrisko 4209665484d8SDoug Ambrisko dcmd->mbox.b[0] = MR_PD_QUERY_TYPE_EXPOSED_TO_HOST; 4210665484d8SDoug Ambrisko dcmd->mbox.b[1] = 0; 4211665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 4212665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 4213665484d8SDoug Ambrisko dcmd->sge_count = 1; 4214665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 4215665484d8SDoug Ambrisko dcmd->timeout = 0; 4216665484d8SDoug Ambrisko dcmd->pad_0 = 0; 4217665484d8SDoug Ambrisko dcmd->data_xfer_len = MRSAS_MAX_PD * sizeof(struct MR_PD_LIST); 4218665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_PD_LIST_QUERY; 4219665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = pd_list_phys_addr; 4220665484d8SDoug Ambrisko dcmd->sgl.sge32[0].length = MRSAS_MAX_PD * sizeof(struct MR_PD_LIST); 4221665484d8SDoug Ambrisko 4222731b7561SKashyap D Desai if (!sc->mask_interrupts) 4223731b7561SKashyap D Desai retcode = mrsas_issue_blocked_cmd(sc, cmd); 4224731b7561SKashyap D Desai else 4225f0c7594bSKashyap D Desai retcode = mrsas_issue_polled(sc, cmd); 4226731b7561SKashyap D Desai 4227f0c7594bSKashyap D Desai if (retcode == ETIMEDOUT) 4228f0c7594bSKashyap D Desai goto dcmd_timeout; 4229665484d8SDoug Ambrisko 4230665484d8SDoug Ambrisko /* Get the instance PD list */ 4231665484d8SDoug Ambrisko pd_count = MRSAS_MAX_PD; 4232665484d8SDoug Ambrisko pd_addr = pd_list_mem->addr; 4233f0c7594bSKashyap D Desai if (pd_list_mem->count < pd_count) { 42344799d485SKashyap D Desai memset(sc->local_pd_list, 0, 42354799d485SKashyap D Desai MRSAS_MAX_PD * sizeof(struct mrsas_pd_list)); 4236665484d8SDoug Ambrisko for (pd_index = 0; pd_index < pd_list_mem->count; pd_index++) { 4237665484d8SDoug Ambrisko sc->local_pd_list[pd_addr->deviceId].tid = pd_addr->deviceId; 42384799d485SKashyap D Desai sc->local_pd_list[pd_addr->deviceId].driveType = 42394799d485SKashyap D Desai pd_addr->scsiDevType; 42404799d485SKashyap D Desai sc->local_pd_list[pd_addr->deviceId].driveState = 42414799d485SKashyap D Desai MR_PD_STATE_SYSTEM; 4242665484d8SDoug Ambrisko pd_addr++; 4243665484d8SDoug Ambrisko } 42448e727371SKashyap D Desai /* 42458e727371SKashyap D Desai * Use mutext/spinlock if pd_list component size increase more than 42468e727371SKashyap D Desai * 32 bit. 42478e727371SKashyap D Desai */ 4248665484d8SDoug Ambrisko memcpy(sc->pd_list, sc->local_pd_list, sizeof(sc->local_pd_list)); 4249f0c7594bSKashyap D Desai do_ocr = 0; 4250f0c7594bSKashyap D Desai } 4251f0c7594bSKashyap D Desai dcmd_timeout: 4252665484d8SDoug Ambrisko mrsas_free_tmp_dcmd(tcmd); 4253665484d8SDoug Ambrisko free(tcmd, M_MRSAS); 4254f0c7594bSKashyap D Desai 4255f0c7594bSKashyap D Desai if (do_ocr) 4256f0c7594bSKashyap D Desai sc->do_timedout_reset = MFI_DCMD_TIMEOUT_OCR; 4257731b7561SKashyap D Desai 4258731b7561SKashyap D Desai if (!sc->mask_interrupts) 4259f0c7594bSKashyap D Desai mrsas_release_mfi_cmd(cmd); 4260f0c7594bSKashyap D Desai 4261665484d8SDoug Ambrisko return (retcode); 4262665484d8SDoug Ambrisko } 4263665484d8SDoug Ambrisko 42648e727371SKashyap D Desai /* 42658e727371SKashyap D Desai * mrsas_get_ld_list: Returns FW's LD list structure input: 42668e727371SKashyap D Desai * Adapter soft state 4267665484d8SDoug Ambrisko * 42688e727371SKashyap D Desai * Issues an internal command (DCMD) to get the FW's controller PD list 42698e727371SKashyap D Desai * structure. This information is mainly used to find out about supported by 42708e727371SKashyap D Desai * the FW. 4271665484d8SDoug Ambrisko */ 42728e727371SKashyap D Desai static int 42738e727371SKashyap D Desai mrsas_get_ld_list(struct mrsas_softc *sc) 4274665484d8SDoug Ambrisko { 4275665484d8SDoug Ambrisko int ld_list_size, retcode = 0, ld_index = 0, ids = 0; 4276f0c7594bSKashyap D Desai u_int8_t do_ocr = 1; 4277665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 4278665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 4279665484d8SDoug Ambrisko struct MR_LD_LIST *ld_list_mem; 4280665484d8SDoug Ambrisko bus_addr_t ld_list_phys_addr = 0; 4281665484d8SDoug Ambrisko struct mrsas_tmp_dcmd *tcmd; 4282665484d8SDoug Ambrisko 4283665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 4284665484d8SDoug Ambrisko if (!cmd) { 42854799d485SKashyap D Desai device_printf(sc->mrsas_dev, 42864799d485SKashyap D Desai "Cannot alloc for get LD list cmd\n"); 4287665484d8SDoug Ambrisko return 1; 4288665484d8SDoug Ambrisko } 4289665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 4290665484d8SDoug Ambrisko 4291665484d8SDoug Ambrisko tcmd = malloc(sizeof(struct mrsas_tmp_dcmd), M_MRSAS, M_NOWAIT); 4292665484d8SDoug Ambrisko ld_list_size = sizeof(struct MR_LD_LIST); 4293665484d8SDoug Ambrisko if (mrsas_alloc_tmp_dcmd(sc, tcmd, ld_list_size) != SUCCESS) { 42944799d485SKashyap D Desai device_printf(sc->mrsas_dev, 42954799d485SKashyap D Desai "Cannot alloc dmamap for get LD list cmd\n"); 4296665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 4297f0c7594bSKashyap D Desai mrsas_free_tmp_dcmd(tcmd); 4298f0c7594bSKashyap D Desai free(tcmd, M_MRSAS); 4299665484d8SDoug Ambrisko return (ENOMEM); 43008e727371SKashyap D Desai } else { 4301665484d8SDoug Ambrisko ld_list_mem = tcmd->tmp_dcmd_mem; 4302665484d8SDoug Ambrisko ld_list_phys_addr = tcmd->tmp_dcmd_phys_addr; 4303665484d8SDoug Ambrisko } 4304665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 4305665484d8SDoug Ambrisko 43064799d485SKashyap D Desai if (sc->max256vdSupport) 43074799d485SKashyap D Desai dcmd->mbox.b[0] = 1; 43084799d485SKashyap D Desai 4309665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 4310665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 4311665484d8SDoug Ambrisko dcmd->sge_count = 1; 4312665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 4313665484d8SDoug Ambrisko dcmd->timeout = 0; 4314665484d8SDoug Ambrisko dcmd->data_xfer_len = sizeof(struct MR_LD_LIST); 4315665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_LD_GET_LIST; 4316665484d8SDoug Ambrisko dcmd->sgl.sge32[0].phys_addr = ld_list_phys_addr; 4317665484d8SDoug Ambrisko dcmd->sgl.sge32[0].length = sizeof(struct MR_LD_LIST); 4318665484d8SDoug Ambrisko dcmd->pad_0 = 0; 4319665484d8SDoug Ambrisko 4320731b7561SKashyap D Desai if (!sc->mask_interrupts) 4321731b7561SKashyap D Desai retcode = mrsas_issue_blocked_cmd(sc, cmd); 4322731b7561SKashyap D Desai else 4323f0c7594bSKashyap D Desai retcode = mrsas_issue_polled(sc, cmd); 4324731b7561SKashyap D Desai 4325f0c7594bSKashyap D Desai if (retcode == ETIMEDOUT) 4326f0c7594bSKashyap D Desai goto dcmd_timeout; 4327665484d8SDoug Ambrisko 43284799d485SKashyap D Desai #if VD_EXT_DEBUG 43294799d485SKashyap D Desai printf("Number of LDs %d\n", ld_list_mem->ldCount); 43304799d485SKashyap D Desai #endif 43314799d485SKashyap D Desai 4332665484d8SDoug Ambrisko /* Get the instance LD list */ 4333f0c7594bSKashyap D Desai if (ld_list_mem->ldCount <= sc->fw_supported_vd_count) { 4334665484d8SDoug Ambrisko sc->CurLdCount = ld_list_mem->ldCount; 43354799d485SKashyap D Desai memset(sc->ld_ids, 0xff, MAX_LOGICAL_DRIVES_EXT); 4336665484d8SDoug Ambrisko for (ld_index = 0; ld_index < ld_list_mem->ldCount; ld_index++) { 4337665484d8SDoug Ambrisko if (ld_list_mem->ldList[ld_index].state != 0) { 4338665484d8SDoug Ambrisko ids = ld_list_mem->ldList[ld_index].ref.ld_context.targetId; 4339665484d8SDoug Ambrisko sc->ld_ids[ids] = ld_list_mem->ldList[ld_index].ref.ld_context.targetId; 4340665484d8SDoug Ambrisko } 4341665484d8SDoug Ambrisko } 4342f0c7594bSKashyap D Desai do_ocr = 0; 4343665484d8SDoug Ambrisko } 4344f0c7594bSKashyap D Desai dcmd_timeout: 4345665484d8SDoug Ambrisko mrsas_free_tmp_dcmd(tcmd); 4346665484d8SDoug Ambrisko free(tcmd, M_MRSAS); 4347f0c7594bSKashyap D Desai 4348f0c7594bSKashyap D Desai if (do_ocr) 4349f0c7594bSKashyap D Desai sc->do_timedout_reset = MFI_DCMD_TIMEOUT_OCR; 4350731b7561SKashyap D Desai if (!sc->mask_interrupts) 4351f0c7594bSKashyap D Desai mrsas_release_mfi_cmd(cmd); 4352f0c7594bSKashyap D Desai 4353665484d8SDoug Ambrisko return (retcode); 4354665484d8SDoug Ambrisko } 4355665484d8SDoug Ambrisko 43568e727371SKashyap D Desai /* 43578e727371SKashyap D Desai * mrsas_alloc_tmp_dcmd: Allocates memory for temporary command input: 43588e727371SKashyap D Desai * Adapter soft state Temp command Size of alloction 4359665484d8SDoug Ambrisko * 4360665484d8SDoug Ambrisko * Allocates DMAable memory for a temporary internal command. The allocated 4361665484d8SDoug Ambrisko * memory is initialized to all zeros upon successful loading of the dma 4362665484d8SDoug Ambrisko * mapped memory. 4363665484d8SDoug Ambrisko */ 43648e727371SKashyap D Desai int 43658e727371SKashyap D Desai mrsas_alloc_tmp_dcmd(struct mrsas_softc *sc, 43668e727371SKashyap D Desai struct mrsas_tmp_dcmd *tcmd, int size) 4367665484d8SDoug Ambrisko { 43688e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 43698e727371SKashyap D Desai 1, 0, 43708e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 43718e727371SKashyap D Desai BUS_SPACE_MAXADDR, 43728e727371SKashyap D Desai NULL, NULL, 43738e727371SKashyap D Desai size, 43748e727371SKashyap D Desai 1, 43758e727371SKashyap D Desai size, 43768e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 43778e727371SKashyap D Desai NULL, NULL, 4378665484d8SDoug Ambrisko &tcmd->tmp_dcmd_tag)) { 4379665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate tmp dcmd tag\n"); 4380665484d8SDoug Ambrisko return (ENOMEM); 4381665484d8SDoug Ambrisko } 4382665484d8SDoug Ambrisko if (bus_dmamem_alloc(tcmd->tmp_dcmd_tag, (void **)&tcmd->tmp_dcmd_mem, 4383665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &tcmd->tmp_dcmd_dmamap)) { 4384665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate tmp dcmd mem\n"); 4385665484d8SDoug Ambrisko return (ENOMEM); 4386665484d8SDoug Ambrisko } 4387665484d8SDoug Ambrisko if (bus_dmamap_load(tcmd->tmp_dcmd_tag, tcmd->tmp_dcmd_dmamap, 4388665484d8SDoug Ambrisko tcmd->tmp_dcmd_mem, size, mrsas_addr_cb, 4389665484d8SDoug Ambrisko &tcmd->tmp_dcmd_phys_addr, BUS_DMA_NOWAIT)) { 4390665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load tmp dcmd mem\n"); 4391665484d8SDoug Ambrisko return (ENOMEM); 4392665484d8SDoug Ambrisko } 4393665484d8SDoug Ambrisko memset(tcmd->tmp_dcmd_mem, 0, size); 4394665484d8SDoug Ambrisko return (0); 4395665484d8SDoug Ambrisko } 4396665484d8SDoug Ambrisko 43978e727371SKashyap D Desai /* 43988e727371SKashyap D Desai * mrsas_free_tmp_dcmd: Free memory for temporary command input: 43998e727371SKashyap D Desai * temporary dcmd pointer 4400665484d8SDoug Ambrisko * 44018e727371SKashyap D Desai * Deallocates memory of the temporary command for use in the construction of 44028e727371SKashyap D Desai * the internal DCMD. 4403665484d8SDoug Ambrisko */ 44048e727371SKashyap D Desai void 44058e727371SKashyap D Desai mrsas_free_tmp_dcmd(struct mrsas_tmp_dcmd *tmp) 4406665484d8SDoug Ambrisko { 4407665484d8SDoug Ambrisko if (tmp->tmp_dcmd_phys_addr) 4408665484d8SDoug Ambrisko bus_dmamap_unload(tmp->tmp_dcmd_tag, tmp->tmp_dcmd_dmamap); 4409665484d8SDoug Ambrisko if (tmp->tmp_dcmd_mem != NULL) 4410665484d8SDoug Ambrisko bus_dmamem_free(tmp->tmp_dcmd_tag, tmp->tmp_dcmd_mem, tmp->tmp_dcmd_dmamap); 4411665484d8SDoug Ambrisko if (tmp->tmp_dcmd_tag != NULL) 4412665484d8SDoug Ambrisko bus_dma_tag_destroy(tmp->tmp_dcmd_tag); 4413665484d8SDoug Ambrisko } 4414665484d8SDoug Ambrisko 44158e727371SKashyap D Desai /* 44168e727371SKashyap D Desai * mrsas_issue_blocked_abort_cmd: Aborts previously issued cmd input: 44178e727371SKashyap D Desai * Adapter soft state Previously issued cmd to be aborted 4418665484d8SDoug Ambrisko * 4419665484d8SDoug Ambrisko * This function is used to abort previously issued commands, such as AEN and 4420665484d8SDoug Ambrisko * RAID map sync map commands. The abort command is sent as a DCMD internal 4421665484d8SDoug Ambrisko * command and subsequently the driver will wait for a return status. The 4422665484d8SDoug Ambrisko * max wait time is MRSAS_INTERNAL_CMD_WAIT_TIME seconds. 4423665484d8SDoug Ambrisko */ 44248e727371SKashyap D Desai static int 44258e727371SKashyap D Desai mrsas_issue_blocked_abort_cmd(struct mrsas_softc *sc, 4426665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd_to_abort) 4427665484d8SDoug Ambrisko { 4428665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 4429665484d8SDoug Ambrisko struct mrsas_abort_frame *abort_fr; 4430665484d8SDoug Ambrisko u_int8_t retcode = 0; 4431665484d8SDoug Ambrisko unsigned long total_time = 0; 4432665484d8SDoug Ambrisko u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME; 4433665484d8SDoug Ambrisko 4434665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 4435665484d8SDoug Ambrisko if (!cmd) { 4436665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc for abort cmd\n"); 4437665484d8SDoug Ambrisko return (1); 4438665484d8SDoug Ambrisko } 4439665484d8SDoug Ambrisko abort_fr = &cmd->frame->abort; 4440665484d8SDoug Ambrisko 4441665484d8SDoug Ambrisko /* Prepare and issue the abort frame */ 4442665484d8SDoug Ambrisko abort_fr->cmd = MFI_CMD_ABORT; 4443665484d8SDoug Ambrisko abort_fr->cmd_status = 0xFF; 4444665484d8SDoug Ambrisko abort_fr->flags = 0; 4445665484d8SDoug Ambrisko abort_fr->abort_context = cmd_to_abort->index; 4446665484d8SDoug Ambrisko abort_fr->abort_mfi_phys_addr_lo = cmd_to_abort->frame_phys_addr; 4447665484d8SDoug Ambrisko abort_fr->abort_mfi_phys_addr_hi = 0; 4448665484d8SDoug Ambrisko 4449665484d8SDoug Ambrisko cmd->sync_cmd = 1; 4450665484d8SDoug Ambrisko cmd->cmd_status = 0xFF; 4451665484d8SDoug Ambrisko 4452665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)) { 4453665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Fail to send abort command.\n"); 4454665484d8SDoug Ambrisko return (1); 4455665484d8SDoug Ambrisko } 4456665484d8SDoug Ambrisko /* Wait for this cmd to complete */ 4457665484d8SDoug Ambrisko sc->chan = (void *)&cmd; 4458665484d8SDoug Ambrisko while (1) { 4459665484d8SDoug Ambrisko if (cmd->cmd_status == 0xFF) { 4460665484d8SDoug Ambrisko tsleep((void *)&sc->chan, 0, "mrsas_sleep", hz); 44618e727371SKashyap D Desai } else 4462665484d8SDoug Ambrisko break; 4463665484d8SDoug Ambrisko total_time++; 4464665484d8SDoug Ambrisko if (total_time >= max_wait) { 4465665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Abort cmd timed out after %d sec.\n", max_wait); 4466665484d8SDoug Ambrisko retcode = 1; 4467665484d8SDoug Ambrisko break; 4468665484d8SDoug Ambrisko } 4469665484d8SDoug Ambrisko } 4470665484d8SDoug Ambrisko 4471665484d8SDoug Ambrisko cmd->sync_cmd = 0; 4472665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 4473665484d8SDoug Ambrisko return (retcode); 4474665484d8SDoug Ambrisko } 4475665484d8SDoug Ambrisko 44768e727371SKashyap D Desai /* 44778e727371SKashyap D Desai * mrsas_complete_abort: Completes aborting a command input: 44788e727371SKashyap D Desai * Adapter soft state Cmd that was issued to abort another cmd 4479665484d8SDoug Ambrisko * 44808e727371SKashyap D Desai * The mrsas_issue_blocked_abort_cmd() function waits for the command status to 44818e727371SKashyap D Desai * change after sending the command. This function is called from 4482665484d8SDoug Ambrisko * mrsas_complete_mptmfi_passthru() to wake up the sleep thread associated. 4483665484d8SDoug Ambrisko */ 44848e727371SKashyap D Desai void 44858e727371SKashyap D Desai mrsas_complete_abort(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 4486665484d8SDoug Ambrisko { 4487665484d8SDoug Ambrisko if (cmd->sync_cmd) { 4488665484d8SDoug Ambrisko cmd->sync_cmd = 0; 4489665484d8SDoug Ambrisko cmd->cmd_status = 0; 4490665484d8SDoug Ambrisko sc->chan = (void *)&cmd; 4491665484d8SDoug Ambrisko wakeup_one((void *)&sc->chan); 4492665484d8SDoug Ambrisko } 4493665484d8SDoug Ambrisko return; 4494665484d8SDoug Ambrisko } 4495665484d8SDoug Ambrisko 44968e727371SKashyap D Desai /* 44978e727371SKashyap D Desai * mrsas_aen_handler: AEN processing callback function from thread context 4498665484d8SDoug Ambrisko * input: Adapter soft state 4499665484d8SDoug Ambrisko * 45008e727371SKashyap D Desai * Asynchronous event handler 4501665484d8SDoug Ambrisko */ 45028e727371SKashyap D Desai void 45038e727371SKashyap D Desai mrsas_aen_handler(struct mrsas_softc *sc) 4504665484d8SDoug Ambrisko { 4505665484d8SDoug Ambrisko union mrsas_evt_class_locale class_locale; 4506665484d8SDoug Ambrisko int doscan = 0; 4507665484d8SDoug Ambrisko u_int32_t seq_num; 4508f0c7594bSKashyap D Desai int error, fail_aen = 0; 4509665484d8SDoug Ambrisko 45105bae00d6SSteven Hartland if (sc == NULL) { 45115bae00d6SSteven Hartland printf("invalid instance!\n"); 4512665484d8SDoug Ambrisko return; 4513665484d8SDoug Ambrisko } 451485c0a961SKashyap D Desai if (sc->remove_in_progress || sc->reset_in_progress) { 451585c0a961SKashyap D Desai device_printf(sc->mrsas_dev, "Returning from %s, line no %d\n", 451685c0a961SKashyap D Desai __func__, __LINE__); 451785c0a961SKashyap D Desai return; 451885c0a961SKashyap D Desai } 4519665484d8SDoug Ambrisko if (sc->evt_detail_mem) { 4520665484d8SDoug Ambrisko switch (sc->evt_detail_mem->code) { 4521665484d8SDoug Ambrisko case MR_EVT_PD_INSERTED: 4522f0c7594bSKashyap D Desai fail_aen = mrsas_get_pd_list(sc); 4523f0c7594bSKashyap D Desai if (!fail_aen) 4524665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_1); 4525f0c7594bSKashyap D Desai else 4526f0c7594bSKashyap D Desai goto skip_register_aen; 4527665484d8SDoug Ambrisko break; 4528665484d8SDoug Ambrisko case MR_EVT_PD_REMOVED: 4529f0c7594bSKashyap D Desai fail_aen = mrsas_get_pd_list(sc); 4530f0c7594bSKashyap D Desai if (!fail_aen) 4531665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_1); 4532f0c7594bSKashyap D Desai else 4533f0c7594bSKashyap D Desai goto skip_register_aen; 4534665484d8SDoug Ambrisko break; 4535665484d8SDoug Ambrisko case MR_EVT_LD_OFFLINE: 4536665484d8SDoug Ambrisko case MR_EVT_CFG_CLEARED: 4537665484d8SDoug Ambrisko case MR_EVT_LD_DELETED: 4538665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_0); 4539665484d8SDoug Ambrisko break; 4540665484d8SDoug Ambrisko case MR_EVT_LD_CREATED: 4541f0c7594bSKashyap D Desai fail_aen = mrsas_get_ld_list(sc); 4542f0c7594bSKashyap D Desai if (!fail_aen) 4543665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_0); 4544f0c7594bSKashyap D Desai else 4545f0c7594bSKashyap D Desai goto skip_register_aen; 4546665484d8SDoug Ambrisko break; 4547665484d8SDoug Ambrisko case MR_EVT_CTRL_HOST_BUS_SCAN_REQUESTED: 4548665484d8SDoug Ambrisko case MR_EVT_FOREIGN_CFG_IMPORTED: 4549665484d8SDoug Ambrisko case MR_EVT_LD_STATE_CHANGE: 4550665484d8SDoug Ambrisko doscan = 1; 4551665484d8SDoug Ambrisko break; 45528bc320adSKashyap D Desai case MR_EVT_CTRL_PROP_CHANGED: 45538bc320adSKashyap D Desai fail_aen = mrsas_get_ctrl_info(sc); 45548bc320adSKashyap D Desai if (fail_aen) 45558bc320adSKashyap D Desai goto skip_register_aen; 45568bc320adSKashyap D Desai break; 4557665484d8SDoug Ambrisko default: 4558665484d8SDoug Ambrisko break; 4559665484d8SDoug Ambrisko } 4560665484d8SDoug Ambrisko } else { 4561665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "invalid evt_detail\n"); 4562665484d8SDoug Ambrisko return; 4563665484d8SDoug Ambrisko } 4564665484d8SDoug Ambrisko if (doscan) { 4565f0c7594bSKashyap D Desai fail_aen = mrsas_get_pd_list(sc); 4566f0c7594bSKashyap D Desai if (!fail_aen) { 4567665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_AEN, "scanning ...sim 1\n"); 4568665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_1); 4569f0c7594bSKashyap D Desai } else 4570f0c7594bSKashyap D Desai goto skip_register_aen; 4571f0c7594bSKashyap D Desai 4572f0c7594bSKashyap D Desai fail_aen = mrsas_get_ld_list(sc); 4573f0c7594bSKashyap D Desai if (!fail_aen) { 4574665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_AEN, "scanning ...sim 0\n"); 4575665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_0); 4576f0c7594bSKashyap D Desai } else 4577f0c7594bSKashyap D Desai goto skip_register_aen; 4578665484d8SDoug Ambrisko } 4579665484d8SDoug Ambrisko seq_num = sc->evt_detail_mem->seq_num + 1; 4580665484d8SDoug Ambrisko 45818e727371SKashyap D Desai /* Register AEN with FW for latest sequence number plus 1 */ 4582665484d8SDoug Ambrisko class_locale.members.reserved = 0; 4583665484d8SDoug Ambrisko class_locale.members.locale = MR_EVT_LOCALE_ALL; 4584665484d8SDoug Ambrisko class_locale.members.class = MR_EVT_CLASS_DEBUG; 4585665484d8SDoug Ambrisko 4586665484d8SDoug Ambrisko if (sc->aen_cmd != NULL) 4587665484d8SDoug Ambrisko return; 4588665484d8SDoug Ambrisko 4589665484d8SDoug Ambrisko mtx_lock(&sc->aen_lock); 4590665484d8SDoug Ambrisko error = mrsas_register_aen(sc, seq_num, 4591665484d8SDoug Ambrisko class_locale.word); 4592665484d8SDoug Ambrisko mtx_unlock(&sc->aen_lock); 4593665484d8SDoug Ambrisko 4594665484d8SDoug Ambrisko if (error) 4595665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "register aen failed error %x\n", error); 4596665484d8SDoug Ambrisko 4597f0c7594bSKashyap D Desai skip_register_aen: 4598f0c7594bSKashyap D Desai return; 4599f0c7594bSKashyap D Desai 4600665484d8SDoug Ambrisko } 4601665484d8SDoug Ambrisko 4602665484d8SDoug Ambrisko 46038e727371SKashyap D Desai /* 4604665484d8SDoug Ambrisko * mrsas_complete_aen: Completes AEN command 4605665484d8SDoug Ambrisko * input: Adapter soft state 4606665484d8SDoug Ambrisko * Cmd that was issued to abort another cmd 4607665484d8SDoug Ambrisko * 46088e727371SKashyap D Desai * This function will be called from ISR and will continue event processing from 46098e727371SKashyap D Desai * thread context by enqueuing task in ev_tq (callback function 46108e727371SKashyap D Desai * "mrsas_aen_handler"). 4611665484d8SDoug Ambrisko */ 46128e727371SKashyap D Desai void 46138e727371SKashyap D Desai mrsas_complete_aen(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 4614665484d8SDoug Ambrisko { 4615665484d8SDoug Ambrisko /* 46168e727371SKashyap D Desai * Don't signal app if it is just an aborted previously registered 46178e727371SKashyap D Desai * aen 4618665484d8SDoug Ambrisko */ 4619665484d8SDoug Ambrisko if ((!cmd->abort_aen) && (sc->remove_in_progress == 0)) { 4620da011113SKashyap D Desai sc->mrsas_aen_triggered = 1; 4621ecea5be4SKashyap D Desai mtx_lock(&sc->aen_lock); 4622da011113SKashyap D Desai if (sc->mrsas_poll_waiting) { 4623da011113SKashyap D Desai sc->mrsas_poll_waiting = 0; 4624da011113SKashyap D Desai selwakeup(&sc->mrsas_select); 4625da011113SKashyap D Desai } 4626ecea5be4SKashyap D Desai mtx_unlock(&sc->aen_lock); 46278e727371SKashyap D Desai } else 4628665484d8SDoug Ambrisko cmd->abort_aen = 0; 4629665484d8SDoug Ambrisko 4630665484d8SDoug Ambrisko sc->aen_cmd = NULL; 4631665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 4632665484d8SDoug Ambrisko 4633665484d8SDoug Ambrisko taskqueue_enqueue(sc->ev_tq, &sc->ev_task); 4634665484d8SDoug Ambrisko 4635665484d8SDoug Ambrisko return; 4636665484d8SDoug Ambrisko } 4637665484d8SDoug Ambrisko 4638665484d8SDoug Ambrisko static device_method_t mrsas_methods[] = { 4639665484d8SDoug Ambrisko DEVMETHOD(device_probe, mrsas_probe), 4640665484d8SDoug Ambrisko DEVMETHOD(device_attach, mrsas_attach), 4641665484d8SDoug Ambrisko DEVMETHOD(device_detach, mrsas_detach), 4642665484d8SDoug Ambrisko DEVMETHOD(device_suspend, mrsas_suspend), 4643665484d8SDoug Ambrisko DEVMETHOD(device_resume, mrsas_resume), 4644665484d8SDoug Ambrisko DEVMETHOD(bus_print_child, bus_generic_print_child), 4645665484d8SDoug Ambrisko DEVMETHOD(bus_driver_added, bus_generic_driver_added), 4646665484d8SDoug Ambrisko {0, 0} 4647665484d8SDoug Ambrisko }; 4648665484d8SDoug Ambrisko 4649665484d8SDoug Ambrisko static driver_t mrsas_driver = { 4650665484d8SDoug Ambrisko "mrsas", 4651665484d8SDoug Ambrisko mrsas_methods, 4652665484d8SDoug Ambrisko sizeof(struct mrsas_softc) 4653665484d8SDoug Ambrisko }; 4654665484d8SDoug Ambrisko 4655665484d8SDoug Ambrisko static devclass_t mrsas_devclass; 46568e727371SKashyap D Desai 4657665484d8SDoug Ambrisko DRIVER_MODULE(mrsas, pci, mrsas_driver, mrsas_devclass, 0, 0); 4658665484d8SDoug Ambrisko MODULE_DEPEND(mrsas, cam, 1, 1, 1); 4659