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> 55*e34a057cSAlfredo Dal'Ava Junior #include <sys/endian.h> 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); 9379b4460bSKashyap D Desai static void 9479b4460bSKashyap D Desai mrsas_get_pd_info(struct mrsas_softc *sc, u_int16_t device_id); 95dbcc81dfSKashyap D Desai static struct mrsas_softc * 96dbcc81dfSKashyap D Desai mrsas_get_softc_instance(struct cdev *dev, 975844115eSKashyap D Desai u_long cmd, caddr_t arg); 98e315cf4dSKashyap D Desai u_int32_t 99e315cf4dSKashyap D Desai mrsas_read_reg_with_retries(struct mrsas_softc *sc, int offset); 100665484d8SDoug Ambrisko u_int32_t mrsas_read_reg(struct mrsas_softc *sc, int offset); 1018e727371SKashyap D Desai u_int8_t 1028e727371SKashyap D Desai mrsas_build_mptmfi_passthru(struct mrsas_softc *sc, 103665484d8SDoug Ambrisko struct mrsas_mfi_cmd *mfi_cmd); 104daeed973SKashyap D Desai void mrsas_complete_outstanding_ioctls(struct mrsas_softc *sc); 105665484d8SDoug Ambrisko int mrsas_transition_to_ready(struct mrsas_softc *sc, int ocr); 106665484d8SDoug Ambrisko int mrsas_init_adapter(struct mrsas_softc *sc); 107665484d8SDoug Ambrisko int mrsas_alloc_mpt_cmds(struct mrsas_softc *sc); 108665484d8SDoug Ambrisko int mrsas_alloc_ioc_cmd(struct mrsas_softc *sc); 109665484d8SDoug Ambrisko int mrsas_alloc_ctlr_info_cmd(struct mrsas_softc *sc); 110665484d8SDoug Ambrisko int mrsas_ioc_init(struct mrsas_softc *sc); 111665484d8SDoug Ambrisko int mrsas_bus_scan(struct mrsas_softc *sc); 112665484d8SDoug Ambrisko int mrsas_issue_dcmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 113665484d8SDoug Ambrisko int mrsas_issue_polled(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 114f0c7594bSKashyap D Desai int mrsas_reset_ctrl(struct mrsas_softc *sc, u_int8_t reset_reason); 115f0c7594bSKashyap D Desai int mrsas_wait_for_outstanding(struct mrsas_softc *sc, u_int8_t check_reason); 1164bb0a4f0SKashyap D Desai int mrsas_complete_cmd(struct mrsas_softc *sc, u_int32_t MSIxIndex); 1178bb601acSKashyap D Desai int mrsas_reset_targets(struct mrsas_softc *sc); 1188e727371SKashyap D Desai int 1198e727371SKashyap D Desai mrsas_issue_blocked_cmd(struct mrsas_softc *sc, 120665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd); 1218e727371SKashyap D Desai int 1228e727371SKashyap D Desai mrsas_alloc_tmp_dcmd(struct mrsas_softc *sc, struct mrsas_tmp_dcmd *tcmd, 123665484d8SDoug Ambrisko int size); 124665484d8SDoug Ambrisko void mrsas_release_mfi_cmd(struct mrsas_mfi_cmd *cmd); 125665484d8SDoug Ambrisko void mrsas_wakeup(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 126665484d8SDoug Ambrisko void mrsas_complete_aen(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 127665484d8SDoug Ambrisko void mrsas_complete_abort(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 128665484d8SDoug Ambrisko void mrsas_disable_intr(struct mrsas_softc *sc); 129665484d8SDoug Ambrisko void mrsas_enable_intr(struct mrsas_softc *sc); 130665484d8SDoug Ambrisko void mrsas_free_ioc_cmd(struct mrsas_softc *sc); 131665484d8SDoug Ambrisko void mrsas_free_mem(struct mrsas_softc *sc); 132665484d8SDoug Ambrisko void mrsas_free_tmp_dcmd(struct mrsas_tmp_dcmd *tmp); 133665484d8SDoug Ambrisko void mrsas_isr(void *arg); 134665484d8SDoug Ambrisko void mrsas_teardown_intr(struct mrsas_softc *sc); 135665484d8SDoug Ambrisko void mrsas_addr_cb(void *arg, bus_dma_segment_t *segs, int nsegs, int error); 136665484d8SDoug Ambrisko void mrsas_kill_hba(struct mrsas_softc *sc); 137665484d8SDoug Ambrisko void mrsas_aen_handler(struct mrsas_softc *sc); 1388e727371SKashyap D Desai void 1398e727371SKashyap D Desai mrsas_write_reg(struct mrsas_softc *sc, int offset, 140665484d8SDoug Ambrisko u_int32_t value); 1418e727371SKashyap D Desai void 1428e727371SKashyap D Desai mrsas_fire_cmd(struct mrsas_softc *sc, u_int32_t req_desc_lo, 143665484d8SDoug Ambrisko u_int32_t req_desc_hi); 144665484d8SDoug Ambrisko void mrsas_free_ctlr_info_cmd(struct mrsas_softc *sc); 1458e727371SKashyap D Desai void 1468e727371SKashyap D Desai mrsas_complete_mptmfi_passthru(struct mrsas_softc *sc, 147665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd, u_int8_t status); 148665484d8SDoug Ambrisko struct mrsas_mfi_cmd *mrsas_get_mfi_cmd(struct mrsas_softc *sc); 1498e727371SKashyap D Desai 1508e727371SKashyap D Desai MRSAS_REQUEST_DESCRIPTOR_UNION *mrsas_build_mpt_cmd 1518e727371SKashyap D Desai (struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 152665484d8SDoug Ambrisko 153665484d8SDoug Ambrisko extern int mrsas_cam_attach(struct mrsas_softc *sc); 154665484d8SDoug Ambrisko extern void mrsas_cam_detach(struct mrsas_softc *sc); 155665484d8SDoug Ambrisko extern void mrsas_cmd_done(struct mrsas_softc *sc, struct mrsas_mpt_cmd *cmd); 156665484d8SDoug Ambrisko extern void mrsas_free_frame(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd); 157665484d8SDoug Ambrisko extern int mrsas_alloc_mfi_cmds(struct mrsas_softc *sc); 158665484d8SDoug Ambrisko extern struct mrsas_mpt_cmd *mrsas_get_mpt_cmd(struct mrsas_softc *sc); 159536094dcSKashyap D Desai extern int mrsas_passthru(struct mrsas_softc *sc, void *arg, u_long ioctlCmd); 160665484d8SDoug Ambrisko extern uint8_t MR_ValidateMapInfo(struct mrsas_softc *sc); 1614799d485SKashyap D Desai extern u_int16_t MR_GetLDTgtId(u_int32_t ld, MR_DRV_RAID_MAP_ALL * map); 1624799d485SKashyap D Desai extern MR_LD_RAID *MR_LdRaidGet(u_int32_t ld, MR_DRV_RAID_MAP_ALL * map); 163665484d8SDoug Ambrisko extern void mrsas_xpt_freeze(struct mrsas_softc *sc); 164665484d8SDoug Ambrisko extern void mrsas_xpt_release(struct mrsas_softc *sc); 1658e727371SKashyap D Desai extern MRSAS_REQUEST_DESCRIPTOR_UNION * 1668e727371SKashyap D Desai mrsas_get_request_desc(struct mrsas_softc *sc, 167665484d8SDoug Ambrisko u_int16_t index); 168665484d8SDoug Ambrisko extern int mrsas_bus_scan_sim(struct mrsas_softc *sc, struct cam_sim *sim); 169665484d8SDoug Ambrisko static int mrsas_alloc_evt_log_info_cmd(struct mrsas_softc *sc); 170665484d8SDoug Ambrisko static void mrsas_free_evt_log_info_cmd(struct mrsas_softc *sc); 1712a1d3bcdSKashyap D Desai void mrsas_release_mpt_cmd(struct mrsas_mpt_cmd *cmd); 1722a1d3bcdSKashyap D Desai 1732a1d3bcdSKashyap D Desai void mrsas_map_mpt_cmd_status(struct mrsas_mpt_cmd *cmd, 1742a1d3bcdSKashyap D Desai union ccb *ccb_ptr, u_int8_t status, u_int8_t extStatus, 1752a1d3bcdSKashyap D Desai u_int32_t data_length, u_int8_t *sense); 176b518670cSKashyap D Desai void 177b518670cSKashyap D Desai mrsas_write_64bit_req_desc(struct mrsas_softc *sc, u_int32_t req_desc_lo, 178b518670cSKashyap D Desai u_int32_t req_desc_hi); 1792a1d3bcdSKashyap D Desai 1807029da5cSPawel Biernacki SYSCTL_NODE(_hw, OID_AUTO, mrsas, CTLFLAG_RD | CTLFLAG_MPSAFE, 0, 1817029da5cSPawel Biernacki "MRSAS Driver Parameters"); 182665484d8SDoug Ambrisko 1838e727371SKashyap D Desai /* 184665484d8SDoug Ambrisko * PCI device struct and table 185665484d8SDoug Ambrisko * 186665484d8SDoug Ambrisko */ 187665484d8SDoug Ambrisko typedef struct mrsas_ident { 188665484d8SDoug Ambrisko uint16_t vendor; 189665484d8SDoug Ambrisko uint16_t device; 190665484d8SDoug Ambrisko uint16_t subvendor; 191665484d8SDoug Ambrisko uint16_t subdevice; 192665484d8SDoug Ambrisko const char *desc; 193665484d8SDoug Ambrisko } MRSAS_CTLR_ID; 194665484d8SDoug Ambrisko 195665484d8SDoug Ambrisko MRSAS_CTLR_ID device_table[] = { 196ecea5be4SKashyap D Desai {0x1000, MRSAS_TBOLT, 0xffff, 0xffff, "AVAGO Thunderbolt SAS Controller"}, 197ecea5be4SKashyap D Desai {0x1000, MRSAS_INVADER, 0xffff, 0xffff, "AVAGO Invader SAS Controller"}, 198ecea5be4SKashyap D Desai {0x1000, MRSAS_FURY, 0xffff, 0xffff, "AVAGO Fury SAS Controller"}, 199c620f351SKashyap D Desai {0x1000, MRSAS_INTRUDER, 0xffff, 0xffff, "AVAGO Intruder SAS Controller"}, 200c620f351SKashyap D Desai {0x1000, MRSAS_INTRUDER_24, 0xffff, 0xffff, "AVAGO Intruder_24 SAS Controller"}, 2018cd174a4SKashyap D Desai {0x1000, MRSAS_CUTLASS_52, 0xffff, 0xffff, "AVAGO Cutlass_52 SAS Controller"}, 2028cd174a4SKashyap D Desai {0x1000, MRSAS_CUTLASS_53, 0xffff, 0xffff, "AVAGO Cutlass_53 SAS Controller"}, 2037aade8bfSKashyap D Desai {0x1000, MRSAS_VENTURA, 0xffff, 0xffff, "AVAGO Ventura SAS Controller"}, 2047aade8bfSKashyap D Desai {0x1000, MRSAS_CRUSADER, 0xffff, 0xffff, "AVAGO Crusader SAS Controller"}, 2057aade8bfSKashyap D Desai {0x1000, MRSAS_HARPOON, 0xffff, 0xffff, "AVAGO Harpoon SAS Controller"}, 2067aade8bfSKashyap D Desai {0x1000, MRSAS_TOMCAT, 0xffff, 0xffff, "AVAGO Tomcat SAS Controller"}, 2077aade8bfSKashyap D Desai {0x1000, MRSAS_VENTURA_4PORT, 0xffff, 0xffff, "AVAGO Ventura_4Port SAS Controller"}, 2087aade8bfSKashyap D Desai {0x1000, MRSAS_CRUSADER_4PORT, 0xffff, 0xffff, "AVAGO Crusader_4Port SAS Controller"}, 2092909aab4SKashyap D Desai {0x1000, MRSAS_AERO_10E0, 0xffff, 0xffff, "BROADCOM AERO-10E0 SAS Controller"}, 2102909aab4SKashyap D Desai {0x1000, MRSAS_AERO_10E1, 0xffff, 0xffff, "BROADCOM AERO-10E1 SAS Controller"}, 2112909aab4SKashyap D Desai {0x1000, MRSAS_AERO_10E2, 0xffff, 0xffff, "BROADCOM AERO-10E2 SAS Controller"}, 2122909aab4SKashyap D Desai {0x1000, MRSAS_AERO_10E3, 0xffff, 0xffff, "BROADCOM AERO-10E3 SAS Controller"}, 2132909aab4SKashyap D Desai {0x1000, MRSAS_AERO_10E4, 0xffff, 0xffff, "BROADCOM AERO-10E4 SAS Controller"}, 2142909aab4SKashyap D Desai {0x1000, MRSAS_AERO_10E5, 0xffff, 0xffff, "BROADCOM AERO-10E5 SAS Controller"}, 2152909aab4SKashyap D Desai {0x1000, MRSAS_AERO_10E6, 0xffff, 0xffff, "BROADCOM AERO-10E6 SAS Controller"}, 2162909aab4SKashyap D Desai {0x1000, MRSAS_AERO_10E7, 0xffff, 0xffff, "BROADCOM AERO-10E7 SAS Controller"}, 217665484d8SDoug Ambrisko {0, 0, 0, 0, NULL} 218665484d8SDoug Ambrisko }; 219665484d8SDoug Ambrisko 2208e727371SKashyap D Desai /* 221665484d8SDoug Ambrisko * Character device entry points 222665484d8SDoug Ambrisko * 223665484d8SDoug Ambrisko */ 224665484d8SDoug Ambrisko static struct cdevsw mrsas_cdevsw = { 225665484d8SDoug Ambrisko .d_version = D_VERSION, 226665484d8SDoug Ambrisko .d_open = mrsas_open, 227665484d8SDoug Ambrisko .d_close = mrsas_close, 228665484d8SDoug Ambrisko .d_read = mrsas_read, 229665484d8SDoug Ambrisko .d_write = mrsas_write, 230665484d8SDoug Ambrisko .d_ioctl = mrsas_ioctl, 231da011113SKashyap D Desai .d_poll = mrsas_poll, 232665484d8SDoug Ambrisko .d_name = "mrsas", 233665484d8SDoug Ambrisko }; 234665484d8SDoug Ambrisko 235665484d8SDoug Ambrisko MALLOC_DEFINE(M_MRSAS, "mrsasbuf", "Buffers for the MRSAS driver"); 236665484d8SDoug Ambrisko 2378e727371SKashyap D Desai /* 2388e727371SKashyap D Desai * In the cdevsw routines, we find our softc by using the si_drv1 member of 2398e727371SKashyap D Desai * struct cdev. We set this variable to point to our softc in our attach 2408e727371SKashyap D Desai * routine when we create the /dev entry. 241665484d8SDoug Ambrisko */ 242665484d8SDoug Ambrisko int 2437fc5f329SJohn Baldwin mrsas_open(struct cdev *dev, int oflags, int devtype, struct thread *td) 244665484d8SDoug Ambrisko { 245665484d8SDoug Ambrisko struct mrsas_softc *sc; 246665484d8SDoug Ambrisko 247665484d8SDoug Ambrisko sc = dev->si_drv1; 248665484d8SDoug Ambrisko return (0); 249665484d8SDoug Ambrisko } 250665484d8SDoug Ambrisko 251665484d8SDoug Ambrisko int 2527fc5f329SJohn Baldwin mrsas_close(struct cdev *dev, int fflag, int devtype, struct thread *td) 253665484d8SDoug Ambrisko { 254665484d8SDoug Ambrisko struct mrsas_softc *sc; 255665484d8SDoug Ambrisko 256665484d8SDoug Ambrisko sc = dev->si_drv1; 257665484d8SDoug Ambrisko return (0); 258665484d8SDoug Ambrisko } 259665484d8SDoug Ambrisko 260665484d8SDoug Ambrisko int 261665484d8SDoug Ambrisko mrsas_read(struct cdev *dev, struct uio *uio, int ioflag) 262665484d8SDoug Ambrisko { 263665484d8SDoug Ambrisko struct mrsas_softc *sc; 264665484d8SDoug Ambrisko 265665484d8SDoug Ambrisko sc = dev->si_drv1; 266665484d8SDoug Ambrisko return (0); 267665484d8SDoug Ambrisko } 268665484d8SDoug Ambrisko int 269665484d8SDoug Ambrisko mrsas_write(struct cdev *dev, struct uio *uio, int ioflag) 270665484d8SDoug Ambrisko { 271665484d8SDoug Ambrisko struct mrsas_softc *sc; 272665484d8SDoug Ambrisko 273665484d8SDoug Ambrisko sc = dev->si_drv1; 274665484d8SDoug Ambrisko return (0); 275665484d8SDoug Ambrisko } 276665484d8SDoug Ambrisko 277e315cf4dSKashyap D Desai u_int32_t 278e315cf4dSKashyap D Desai mrsas_read_reg_with_retries(struct mrsas_softc *sc, int offset) 279e315cf4dSKashyap D Desai { 280e315cf4dSKashyap D Desai u_int32_t i = 0, ret_val; 281e315cf4dSKashyap D Desai 282e315cf4dSKashyap D Desai if (sc->is_aero) { 283e315cf4dSKashyap D Desai do { 284e315cf4dSKashyap D Desai ret_val = mrsas_read_reg(sc, offset); 285e315cf4dSKashyap D Desai i++; 286e315cf4dSKashyap D Desai } while(ret_val == 0 && i < 3); 287e315cf4dSKashyap D Desai } else 288e315cf4dSKashyap D Desai ret_val = mrsas_read_reg(sc, offset); 289e315cf4dSKashyap D Desai 290e315cf4dSKashyap D Desai return ret_val; 291e315cf4dSKashyap D Desai } 292e315cf4dSKashyap D Desai 2938e727371SKashyap D Desai /* 294665484d8SDoug Ambrisko * Register Read/Write Functions 295665484d8SDoug Ambrisko * 296665484d8SDoug Ambrisko */ 297665484d8SDoug Ambrisko void 298665484d8SDoug Ambrisko mrsas_write_reg(struct mrsas_softc *sc, int offset, 299665484d8SDoug Ambrisko u_int32_t value) 300665484d8SDoug Ambrisko { 301665484d8SDoug Ambrisko bus_space_tag_t bus_tag = sc->bus_tag; 302665484d8SDoug Ambrisko bus_space_handle_t bus_handle = sc->bus_handle; 303665484d8SDoug Ambrisko 304665484d8SDoug Ambrisko bus_space_write_4(bus_tag, bus_handle, offset, value); 305665484d8SDoug Ambrisko } 306665484d8SDoug Ambrisko 307665484d8SDoug Ambrisko u_int32_t 308665484d8SDoug Ambrisko mrsas_read_reg(struct mrsas_softc *sc, int offset) 309665484d8SDoug Ambrisko { 310665484d8SDoug Ambrisko bus_space_tag_t bus_tag = sc->bus_tag; 311665484d8SDoug Ambrisko bus_space_handle_t bus_handle = sc->bus_handle; 312665484d8SDoug Ambrisko 313665484d8SDoug Ambrisko return ((u_int32_t)bus_space_read_4(bus_tag, bus_handle, offset)); 314665484d8SDoug Ambrisko } 315665484d8SDoug Ambrisko 3168e727371SKashyap D Desai /* 317665484d8SDoug Ambrisko * Interrupt Disable/Enable/Clear Functions 318665484d8SDoug Ambrisko * 319665484d8SDoug Ambrisko */ 3208e727371SKashyap D Desai void 3218e727371SKashyap D Desai mrsas_disable_intr(struct mrsas_softc *sc) 322665484d8SDoug Ambrisko { 323665484d8SDoug Ambrisko u_int32_t mask = 0xFFFFFFFF; 324665484d8SDoug Ambrisko u_int32_t status; 325665484d8SDoug Ambrisko 3262f863eb8SKashyap D Desai sc->mask_interrupts = 1; 327665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask), mask); 328665484d8SDoug Ambrisko /* Dummy read to force pci flush */ 329665484d8SDoug Ambrisko status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask)); 330665484d8SDoug Ambrisko } 331665484d8SDoug Ambrisko 3328e727371SKashyap D Desai void 3338e727371SKashyap D Desai mrsas_enable_intr(struct mrsas_softc *sc) 334665484d8SDoug Ambrisko { 335665484d8SDoug Ambrisko u_int32_t mask = MFI_FUSION_ENABLE_INTERRUPT_MASK; 336665484d8SDoug Ambrisko u_int32_t status; 337665484d8SDoug Ambrisko 3382f863eb8SKashyap D Desai sc->mask_interrupts = 0; 339665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status), ~0); 340665484d8SDoug Ambrisko status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status)); 341665484d8SDoug Ambrisko 342665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask), ~mask); 343665484d8SDoug Ambrisko status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask)); 344665484d8SDoug Ambrisko } 345665484d8SDoug Ambrisko 3468e727371SKashyap D Desai static int 3478e727371SKashyap D Desai mrsas_clear_intr(struct mrsas_softc *sc) 348665484d8SDoug Ambrisko { 3498bb601acSKashyap D Desai u_int32_t status; 350665484d8SDoug Ambrisko 351665484d8SDoug Ambrisko /* Read received interrupt */ 352e315cf4dSKashyap D Desai status = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, outbound_intr_status)); 353665484d8SDoug Ambrisko 354665484d8SDoug Ambrisko /* Not our interrupt, so just return */ 355665484d8SDoug Ambrisko if (!(status & MFI_FUSION_ENABLE_INTERRUPT_MASK)) 356665484d8SDoug Ambrisko return (0); 357665484d8SDoug Ambrisko 358665484d8SDoug Ambrisko /* We got a reply interrupt */ 359665484d8SDoug Ambrisko return (1); 360665484d8SDoug Ambrisko } 361665484d8SDoug Ambrisko 3628e727371SKashyap D Desai /* 363665484d8SDoug Ambrisko * PCI Support Functions 364665484d8SDoug Ambrisko * 365665484d8SDoug Ambrisko */ 3668e727371SKashyap D Desai static struct mrsas_ident * 3678e727371SKashyap D Desai mrsas_find_ident(device_t dev) 368665484d8SDoug Ambrisko { 369665484d8SDoug Ambrisko struct mrsas_ident *pci_device; 370665484d8SDoug Ambrisko 3718e727371SKashyap D Desai for (pci_device = device_table; pci_device->vendor != 0; pci_device++) { 372665484d8SDoug Ambrisko if ((pci_device->vendor == pci_get_vendor(dev)) && 373665484d8SDoug Ambrisko (pci_device->device == pci_get_device(dev)) && 374665484d8SDoug Ambrisko ((pci_device->subvendor == pci_get_subvendor(dev)) || 375665484d8SDoug Ambrisko (pci_device->subvendor == 0xffff)) && 376665484d8SDoug Ambrisko ((pci_device->subdevice == pci_get_subdevice(dev)) || 377665484d8SDoug Ambrisko (pci_device->subdevice == 0xffff))) 378665484d8SDoug Ambrisko return (pci_device); 379665484d8SDoug Ambrisko } 380665484d8SDoug Ambrisko return (NULL); 381665484d8SDoug Ambrisko } 382665484d8SDoug Ambrisko 3838e727371SKashyap D Desai static int 3848e727371SKashyap D Desai mrsas_probe(device_t dev) 385665484d8SDoug Ambrisko { 386665484d8SDoug Ambrisko static u_int8_t first_ctrl = 1; 387665484d8SDoug Ambrisko struct mrsas_ident *id; 388665484d8SDoug Ambrisko 389665484d8SDoug Ambrisko if ((id = mrsas_find_ident(dev)) != NULL) { 390665484d8SDoug Ambrisko if (first_ctrl) { 391ecea5be4SKashyap D Desai printf("AVAGO MegaRAID SAS FreeBSD mrsas driver version: %s\n", 3928e727371SKashyap D Desai MRSAS_VERSION); 393665484d8SDoug Ambrisko first_ctrl = 0; 394665484d8SDoug Ambrisko } 395665484d8SDoug Ambrisko device_set_desc(dev, id->desc); 396665484d8SDoug Ambrisko /* between BUS_PROBE_DEFAULT and BUS_PROBE_LOW_PRIORITY */ 397665484d8SDoug Ambrisko return (-30); 398665484d8SDoug Ambrisko } 399665484d8SDoug Ambrisko return (ENXIO); 400665484d8SDoug Ambrisko } 401665484d8SDoug Ambrisko 4028e727371SKashyap D Desai /* 403665484d8SDoug Ambrisko * mrsas_setup_sysctl: setup sysctl values for mrsas 404665484d8SDoug Ambrisko * input: Adapter instance soft state 405665484d8SDoug Ambrisko * 406665484d8SDoug Ambrisko * Setup sysctl entries for mrsas driver. 407665484d8SDoug Ambrisko */ 408665484d8SDoug Ambrisko static void 409665484d8SDoug Ambrisko mrsas_setup_sysctl(struct mrsas_softc *sc) 410665484d8SDoug Ambrisko { 411665484d8SDoug Ambrisko struct sysctl_ctx_list *sysctl_ctx = NULL; 412665484d8SDoug Ambrisko struct sysctl_oid *sysctl_tree = NULL; 413665484d8SDoug Ambrisko char tmpstr[80], tmpstr2[80]; 414665484d8SDoug Ambrisko 415665484d8SDoug Ambrisko /* 416665484d8SDoug Ambrisko * Setup the sysctl variable so the user can change the debug level 417665484d8SDoug Ambrisko * on the fly. 418665484d8SDoug Ambrisko */ 419665484d8SDoug Ambrisko snprintf(tmpstr, sizeof(tmpstr), "MRSAS controller %d", 420665484d8SDoug Ambrisko device_get_unit(sc->mrsas_dev)); 421665484d8SDoug Ambrisko snprintf(tmpstr2, sizeof(tmpstr2), "%d", device_get_unit(sc->mrsas_dev)); 422665484d8SDoug Ambrisko 423665484d8SDoug Ambrisko sysctl_ctx = device_get_sysctl_ctx(sc->mrsas_dev); 424665484d8SDoug Ambrisko if (sysctl_ctx != NULL) 425665484d8SDoug Ambrisko sysctl_tree = device_get_sysctl_tree(sc->mrsas_dev); 426665484d8SDoug Ambrisko 427665484d8SDoug Ambrisko if (sysctl_tree == NULL) { 428665484d8SDoug Ambrisko sysctl_ctx_init(&sc->sysctl_ctx); 429665484d8SDoug Ambrisko sc->sysctl_tree = SYSCTL_ADD_NODE(&sc->sysctl_ctx, 430665484d8SDoug Ambrisko SYSCTL_STATIC_CHILDREN(_hw_mrsas), OID_AUTO, tmpstr2, 4317029da5cSPawel Biernacki CTLFLAG_RD | CTLFLAG_MPSAFE, 0, tmpstr); 432665484d8SDoug Ambrisko if (sc->sysctl_tree == NULL) 433665484d8SDoug Ambrisko return; 434665484d8SDoug Ambrisko sysctl_ctx = &sc->sysctl_ctx; 435665484d8SDoug Ambrisko sysctl_tree = sc->sysctl_tree; 436665484d8SDoug Ambrisko } 437665484d8SDoug Ambrisko SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 438665484d8SDoug Ambrisko OID_AUTO, "disable_ocr", CTLFLAG_RW, &sc->disableOnlineCtrlReset, 0, 439665484d8SDoug Ambrisko "Disable the use of OCR"); 440665484d8SDoug Ambrisko 441665484d8SDoug Ambrisko SYSCTL_ADD_STRING(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 442665484d8SDoug Ambrisko OID_AUTO, "driver_version", CTLFLAG_RD, MRSAS_VERSION, 443665484d8SDoug Ambrisko strlen(MRSAS_VERSION), "driver version"); 444665484d8SDoug Ambrisko 445665484d8SDoug Ambrisko SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 446665484d8SDoug Ambrisko OID_AUTO, "reset_count", CTLFLAG_RD, 447665484d8SDoug Ambrisko &sc->reset_count, 0, "number of ocr from start of the day"); 448665484d8SDoug Ambrisko 449665484d8SDoug Ambrisko SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 450665484d8SDoug Ambrisko OID_AUTO, "fw_outstanding", CTLFLAG_RD, 451f0188618SHans Petter Selasky &sc->fw_outstanding.val_rdonly, 0, "FW outstanding commands"); 452665484d8SDoug Ambrisko 453665484d8SDoug Ambrisko SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 454665484d8SDoug Ambrisko OID_AUTO, "io_cmds_highwater", CTLFLAG_RD, 455665484d8SDoug Ambrisko &sc->io_cmds_highwater, 0, "Max FW outstanding commands"); 456665484d8SDoug Ambrisko 457665484d8SDoug Ambrisko SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 458665484d8SDoug Ambrisko OID_AUTO, "mrsas_debug", CTLFLAG_RW, &sc->mrsas_debug, 0, 459665484d8SDoug Ambrisko "Driver debug level"); 460665484d8SDoug Ambrisko 461665484d8SDoug Ambrisko SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 462665484d8SDoug Ambrisko OID_AUTO, "mrsas_io_timeout", CTLFLAG_RW, &sc->mrsas_io_timeout, 463665484d8SDoug Ambrisko 0, "Driver IO timeout value in mili-second."); 464665484d8SDoug Ambrisko 465665484d8SDoug Ambrisko SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 466665484d8SDoug Ambrisko OID_AUTO, "mrsas_fw_fault_check_delay", CTLFLAG_RW, 467665484d8SDoug Ambrisko &sc->mrsas_fw_fault_check_delay, 468665484d8SDoug Ambrisko 0, "FW fault check thread delay in seconds. <default is 1 sec>"); 469665484d8SDoug Ambrisko 470665484d8SDoug Ambrisko SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 471665484d8SDoug Ambrisko OID_AUTO, "reset_in_progress", CTLFLAG_RD, 472665484d8SDoug Ambrisko &sc->reset_in_progress, 0, "ocr in progress status"); 473665484d8SDoug Ambrisko 474d993dd83SKashyap D Desai SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 475d993dd83SKashyap D Desai OID_AUTO, "block_sync_cache", CTLFLAG_RW, 476d993dd83SKashyap D Desai &sc->block_sync_cache, 0, 477d993dd83SKashyap D Desai "Block SYNC CACHE at driver. <default: 0, send it to FW>"); 478821df4b9SKashyap D Desai SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 479821df4b9SKashyap D Desai OID_AUTO, "stream detection", CTLFLAG_RW, 480821df4b9SKashyap D Desai &sc->drv_stream_detection, 0, 481821df4b9SKashyap D Desai "Disable/Enable Stream detection. <default: 1, Enable Stream Detection>"); 4823d273176SKashyap D Desai SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 4833d273176SKashyap D Desai OID_AUTO, "prp_count", CTLFLAG_RD, 4843d273176SKashyap D Desai &sc->prp_count.val_rdonly, 0, "Number of IOs for which PRPs are built"); 4853d273176SKashyap D Desai SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree), 4863d273176SKashyap D Desai OID_AUTO, "SGE holes", CTLFLAG_RD, 4873d273176SKashyap D Desai &sc->sge_holes.val_rdonly, 0, "Number of IOs with holes in SGEs"); 488665484d8SDoug Ambrisko } 489665484d8SDoug Ambrisko 4908e727371SKashyap D Desai /* 491665484d8SDoug Ambrisko * mrsas_get_tunables: get tunable parameters. 492665484d8SDoug Ambrisko * input: Adapter instance soft state 493665484d8SDoug Ambrisko * 494665484d8SDoug Ambrisko * Get tunable parameters. This will help to debug driver at boot time. 495665484d8SDoug Ambrisko */ 496665484d8SDoug Ambrisko static void 497665484d8SDoug Ambrisko mrsas_get_tunables(struct mrsas_softc *sc) 498665484d8SDoug Ambrisko { 499665484d8SDoug Ambrisko char tmpstr[80]; 500665484d8SDoug Ambrisko 501665484d8SDoug Ambrisko /* XXX default to some debugging for now */ 50256d91e49SKashyap D Desai sc->mrsas_debug = 50356d91e49SKashyap D Desai (MRSAS_FAULT | MRSAS_OCR | MRSAS_INFO | MRSAS_TRACE | MRSAS_AEN); 504665484d8SDoug Ambrisko sc->mrsas_io_timeout = MRSAS_IO_TIMEOUT; 505665484d8SDoug Ambrisko sc->mrsas_fw_fault_check_delay = 1; 506665484d8SDoug Ambrisko sc->reset_count = 0; 507665484d8SDoug Ambrisko sc->reset_in_progress = 0; 508d993dd83SKashyap D Desai sc->block_sync_cache = 0; 509821df4b9SKashyap D Desai sc->drv_stream_detection = 1; 510665484d8SDoug Ambrisko 511665484d8SDoug Ambrisko /* 512665484d8SDoug Ambrisko * Grab the global variables. 513665484d8SDoug Ambrisko */ 514665484d8SDoug Ambrisko TUNABLE_INT_FETCH("hw.mrsas.debug_level", &sc->mrsas_debug); 515665484d8SDoug Ambrisko 51616dc2814SKashyap D Desai /* 51716dc2814SKashyap D Desai * Grab the global variables. 51816dc2814SKashyap D Desai */ 51916dc2814SKashyap D Desai TUNABLE_INT_FETCH("hw.mrsas.lb_pending_cmds", &sc->lb_pending_cmds); 52016dc2814SKashyap D Desai 521665484d8SDoug Ambrisko /* Grab the unit-instance variables */ 522665484d8SDoug Ambrisko snprintf(tmpstr, sizeof(tmpstr), "dev.mrsas.%d.debug_level", 523665484d8SDoug Ambrisko device_get_unit(sc->mrsas_dev)); 524665484d8SDoug Ambrisko TUNABLE_INT_FETCH(tmpstr, &sc->mrsas_debug); 525665484d8SDoug Ambrisko } 526665484d8SDoug Ambrisko 5278e727371SKashyap D Desai /* 528665484d8SDoug Ambrisko * mrsas_alloc_evt_log_info cmd: Allocates memory to get event log information. 529665484d8SDoug Ambrisko * Used to get sequence number at driver load time. 530665484d8SDoug Ambrisko * input: Adapter soft state 531665484d8SDoug Ambrisko * 532665484d8SDoug Ambrisko * Allocates DMAable memory for the event log info internal command. 533665484d8SDoug Ambrisko */ 5348e727371SKashyap D Desai int 5358e727371SKashyap D Desai mrsas_alloc_evt_log_info_cmd(struct mrsas_softc *sc) 536665484d8SDoug Ambrisko { 537665484d8SDoug Ambrisko int el_info_size; 538665484d8SDoug Ambrisko 539665484d8SDoug Ambrisko /* Allocate get event log info command */ 540665484d8SDoug Ambrisko el_info_size = sizeof(struct mrsas_evt_log_info); 5418e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 5428e727371SKashyap D Desai 1, 0, 5438e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 5448e727371SKashyap D Desai BUS_SPACE_MAXADDR, 5458e727371SKashyap D Desai NULL, NULL, 5468e727371SKashyap D Desai el_info_size, 5478e727371SKashyap D Desai 1, 5488e727371SKashyap D Desai el_info_size, 5498e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 5508e727371SKashyap D Desai NULL, NULL, 551665484d8SDoug Ambrisko &sc->el_info_tag)) { 552665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate event log info tag\n"); 553665484d8SDoug Ambrisko return (ENOMEM); 554665484d8SDoug Ambrisko } 555665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->el_info_tag, (void **)&sc->el_info_mem, 556665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->el_info_dmamap)) { 557665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate event log info cmd mem\n"); 558665484d8SDoug Ambrisko return (ENOMEM); 559665484d8SDoug Ambrisko } 560665484d8SDoug Ambrisko if (bus_dmamap_load(sc->el_info_tag, sc->el_info_dmamap, 561665484d8SDoug Ambrisko sc->el_info_mem, el_info_size, mrsas_addr_cb, 562665484d8SDoug Ambrisko &sc->el_info_phys_addr, BUS_DMA_NOWAIT)) { 563665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load event log info cmd mem\n"); 564665484d8SDoug Ambrisko return (ENOMEM); 565665484d8SDoug Ambrisko } 566665484d8SDoug Ambrisko memset(sc->el_info_mem, 0, el_info_size); 567665484d8SDoug Ambrisko return (0); 568665484d8SDoug Ambrisko } 569665484d8SDoug Ambrisko 5708e727371SKashyap D Desai /* 571665484d8SDoug Ambrisko * mrsas_free_evt_info_cmd: Free memory for Event log info command 572665484d8SDoug Ambrisko * input: Adapter soft state 573665484d8SDoug Ambrisko * 574665484d8SDoug Ambrisko * Deallocates memory for the event log info internal command. 575665484d8SDoug Ambrisko */ 5768e727371SKashyap D Desai void 5778e727371SKashyap D Desai mrsas_free_evt_log_info_cmd(struct mrsas_softc *sc) 578665484d8SDoug Ambrisko { 579665484d8SDoug Ambrisko if (sc->el_info_phys_addr) 580665484d8SDoug Ambrisko bus_dmamap_unload(sc->el_info_tag, sc->el_info_dmamap); 581665484d8SDoug Ambrisko if (sc->el_info_mem != NULL) 582665484d8SDoug Ambrisko bus_dmamem_free(sc->el_info_tag, sc->el_info_mem, sc->el_info_dmamap); 583665484d8SDoug Ambrisko if (sc->el_info_tag != NULL) 584665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->el_info_tag); 585665484d8SDoug Ambrisko } 586665484d8SDoug Ambrisko 5878e727371SKashyap D Desai /* 588665484d8SDoug Ambrisko * mrsas_get_seq_num: Get latest event sequence number 589665484d8SDoug Ambrisko * @sc: Adapter soft state 590665484d8SDoug Ambrisko * @eli: Firmware event log sequence number information. 5918e727371SKashyap D Desai * 592665484d8SDoug Ambrisko * Firmware maintains a log of all events in a non-volatile area. 593665484d8SDoug Ambrisko * Driver get the sequence number using DCMD 594665484d8SDoug Ambrisko * "MR_DCMD_CTRL_EVENT_GET_INFO" at driver load time. 595665484d8SDoug Ambrisko */ 596665484d8SDoug Ambrisko 597665484d8SDoug Ambrisko static int 598665484d8SDoug Ambrisko mrsas_get_seq_num(struct mrsas_softc *sc, 599665484d8SDoug Ambrisko struct mrsas_evt_log_info *eli) 600665484d8SDoug Ambrisko { 601665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 602665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 603f0c7594bSKashyap D Desai u_int8_t do_ocr = 1, retcode = 0; 604665484d8SDoug Ambrisko 605665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 606665484d8SDoug Ambrisko 607665484d8SDoug Ambrisko if (!cmd) { 608665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Failed to get a free cmd\n"); 609665484d8SDoug Ambrisko return -ENOMEM; 610665484d8SDoug Ambrisko } 611665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 612665484d8SDoug Ambrisko 613665484d8SDoug Ambrisko if (mrsas_alloc_evt_log_info_cmd(sc) != SUCCESS) { 614665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate evt log info cmd\n"); 615665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 616665484d8SDoug Ambrisko return -ENOMEM; 617665484d8SDoug Ambrisko } 618665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 619665484d8SDoug Ambrisko 620665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 621665484d8SDoug Ambrisko dcmd->cmd_status = 0x0; 622665484d8SDoug Ambrisko dcmd->sge_count = 1; 623*e34a057cSAlfredo Dal'Ava Junior dcmd->flags = htole16(MFI_FRAME_DIR_READ); 624665484d8SDoug Ambrisko dcmd->timeout = 0; 625665484d8SDoug Ambrisko dcmd->pad_0 = 0; 626*e34a057cSAlfredo Dal'Ava Junior dcmd->data_xfer_len = htole32(sizeof(struct mrsas_evt_log_info)); 627*e34a057cSAlfredo Dal'Ava Junior dcmd->opcode = htole32(MR_DCMD_CTRL_EVENT_GET_INFO); 628*e34a057cSAlfredo Dal'Ava Junior dcmd->sgl.sge32[0].phys_addr = htole32(sc->el_info_phys_addr & 0xFFFFFFFF); 629*e34a057cSAlfredo Dal'Ava Junior dcmd->sgl.sge32[0].length = htole32(sizeof(struct mrsas_evt_log_info)); 630665484d8SDoug Ambrisko 631f0c7594bSKashyap D Desai retcode = mrsas_issue_blocked_cmd(sc, cmd); 632f0c7594bSKashyap D Desai if (retcode == ETIMEDOUT) 633f0c7594bSKashyap D Desai goto dcmd_timeout; 634665484d8SDoug Ambrisko 635f0c7594bSKashyap D Desai do_ocr = 0; 636665484d8SDoug Ambrisko /* 637665484d8SDoug Ambrisko * Copy the data back into callers buffer 638665484d8SDoug Ambrisko */ 639665484d8SDoug Ambrisko memcpy(eli, sc->el_info_mem, sizeof(struct mrsas_evt_log_info)); 640665484d8SDoug Ambrisko mrsas_free_evt_log_info_cmd(sc); 641f0c7594bSKashyap D Desai 642f0c7594bSKashyap D Desai dcmd_timeout: 643f0c7594bSKashyap D Desai if (do_ocr) 644f0c7594bSKashyap D Desai sc->do_timedout_reset = MFI_DCMD_TIMEOUT_OCR; 645f0c7594bSKashyap D Desai else 646665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 647665484d8SDoug Ambrisko 648f0c7594bSKashyap D Desai return retcode; 649665484d8SDoug Ambrisko } 650665484d8SDoug Ambrisko 6518e727371SKashyap D Desai /* 652665484d8SDoug Ambrisko * mrsas_register_aen: Register for asynchronous event notification 653665484d8SDoug Ambrisko * @sc: Adapter soft state 654665484d8SDoug Ambrisko * @seq_num: Starting sequence number 655665484d8SDoug Ambrisko * @class_locale: Class of the event 6568e727371SKashyap D Desai * 657665484d8SDoug Ambrisko * This function subscribes for events beyond the @seq_num 658665484d8SDoug Ambrisko * and type @class_locale. 659665484d8SDoug Ambrisko * 6608e727371SKashyap D Desai */ 661665484d8SDoug Ambrisko static int 662665484d8SDoug Ambrisko mrsas_register_aen(struct mrsas_softc *sc, u_int32_t seq_num, 663665484d8SDoug Ambrisko u_int32_t class_locale_word) 664665484d8SDoug Ambrisko { 665665484d8SDoug Ambrisko int ret_val; 666665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 667665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 668665484d8SDoug Ambrisko union mrsas_evt_class_locale curr_aen; 669665484d8SDoug Ambrisko union mrsas_evt_class_locale prev_aen; 670665484d8SDoug Ambrisko 671665484d8SDoug Ambrisko /* 672665484d8SDoug Ambrisko * If there an AEN pending already (aen_cmd), check if the 6738e727371SKashyap D Desai * class_locale of that pending AEN is inclusive of the new AEN 6748e727371SKashyap D Desai * request we currently have. If it is, then we don't have to do 6758e727371SKashyap D Desai * anything. In other words, whichever events the current AEN request 6768e727371SKashyap D Desai * is subscribing to, have already been subscribed to. If the old_cmd 6778e727371SKashyap D Desai * is _not_ inclusive, then we have to abort that command, form a 6788e727371SKashyap D Desai * class_locale that is superset of both old and current and re-issue 6798e727371SKashyap D Desai * to the FW 6808e727371SKashyap D Desai */ 681665484d8SDoug Ambrisko 682665484d8SDoug Ambrisko curr_aen.word = class_locale_word; 683665484d8SDoug Ambrisko 684665484d8SDoug Ambrisko if (sc->aen_cmd) { 685*e34a057cSAlfredo Dal'Ava Junior prev_aen.word = le32toh(sc->aen_cmd->frame->dcmd.mbox.w[1]); 686665484d8SDoug Ambrisko 687665484d8SDoug Ambrisko /* 688665484d8SDoug Ambrisko * A class whose enum value is smaller is inclusive of all 689665484d8SDoug Ambrisko * higher values. If a PROGRESS (= -1) was previously 690665484d8SDoug Ambrisko * registered, then a new registration requests for higher 691665484d8SDoug Ambrisko * classes need not be sent to FW. They are automatically 6928e727371SKashyap D Desai * included. Locale numbers don't have such hierarchy. They 6938e727371SKashyap D Desai * are bitmap values 694665484d8SDoug Ambrisko */ 695665484d8SDoug Ambrisko if ((prev_aen.members.class <= curr_aen.members.class) && 696665484d8SDoug Ambrisko !((prev_aen.members.locale & curr_aen.members.locale) ^ 697665484d8SDoug Ambrisko curr_aen.members.locale)) { 698665484d8SDoug Ambrisko /* 699665484d8SDoug Ambrisko * Previously issued event registration includes 700665484d8SDoug Ambrisko * current request. Nothing to do. 701665484d8SDoug Ambrisko */ 702665484d8SDoug Ambrisko return 0; 703665484d8SDoug Ambrisko } else { 704665484d8SDoug Ambrisko curr_aen.members.locale |= prev_aen.members.locale; 705665484d8SDoug Ambrisko 706665484d8SDoug Ambrisko if (prev_aen.members.class < curr_aen.members.class) 707665484d8SDoug Ambrisko curr_aen.members.class = prev_aen.members.class; 708665484d8SDoug Ambrisko 709665484d8SDoug Ambrisko sc->aen_cmd->abort_aen = 1; 710665484d8SDoug Ambrisko ret_val = mrsas_issue_blocked_abort_cmd(sc, 711665484d8SDoug Ambrisko sc->aen_cmd); 712665484d8SDoug Ambrisko 713665484d8SDoug Ambrisko if (ret_val) { 714731b7561SKashyap D Desai printf("mrsas: Failed to abort previous AEN command\n"); 715665484d8SDoug Ambrisko return ret_val; 716c2a20ff9SKashyap D Desai } else 717c2a20ff9SKashyap D Desai sc->aen_cmd = NULL; 718665484d8SDoug Ambrisko } 719665484d8SDoug Ambrisko } 720665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 721665484d8SDoug Ambrisko if (!cmd) 722731b7561SKashyap D Desai return ENOMEM; 723665484d8SDoug Ambrisko 724665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 725665484d8SDoug Ambrisko 726665484d8SDoug Ambrisko memset(sc->evt_detail_mem, 0, sizeof(struct mrsas_evt_detail)); 727665484d8SDoug Ambrisko 728665484d8SDoug Ambrisko /* 729665484d8SDoug Ambrisko * Prepare DCMD for aen registration 730665484d8SDoug Ambrisko */ 731665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 732665484d8SDoug Ambrisko 733665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 734665484d8SDoug Ambrisko dcmd->cmd_status = 0x0; 735665484d8SDoug Ambrisko dcmd->sge_count = 1; 736*e34a057cSAlfredo Dal'Ava Junior dcmd->flags = htole16(MFI_FRAME_DIR_READ); 737665484d8SDoug Ambrisko dcmd->timeout = 0; 738665484d8SDoug Ambrisko dcmd->pad_0 = 0; 739*e34a057cSAlfredo Dal'Ava Junior dcmd->data_xfer_len = htole32(sizeof(struct mrsas_evt_detail)); 740*e34a057cSAlfredo Dal'Ava Junior dcmd->opcode = htole32(MR_DCMD_CTRL_EVENT_WAIT); 741*e34a057cSAlfredo Dal'Ava Junior dcmd->mbox.w[0] = htole32(seq_num); 742665484d8SDoug Ambrisko sc->last_seq_num = seq_num; 743*e34a057cSAlfredo Dal'Ava Junior dcmd->mbox.w[1] = htole32(curr_aen.word); 744*e34a057cSAlfredo Dal'Ava Junior dcmd->sgl.sge32[0].phys_addr = htole32((u_int32_t)sc->evt_detail_phys_addr & 0xFFFFFFFF); 745*e34a057cSAlfredo Dal'Ava Junior dcmd->sgl.sge32[0].length = htole32(sizeof(struct mrsas_evt_detail)); 746665484d8SDoug Ambrisko 747665484d8SDoug Ambrisko if (sc->aen_cmd != NULL) { 748665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 749665484d8SDoug Ambrisko return 0; 750665484d8SDoug Ambrisko } 751665484d8SDoug Ambrisko /* 752665484d8SDoug Ambrisko * Store reference to the cmd used to register for AEN. When an 753665484d8SDoug Ambrisko * application wants us to register for AEN, we have to abort this 754665484d8SDoug Ambrisko * cmd and re-register with a new EVENT LOCALE supplied by that app 755665484d8SDoug Ambrisko */ 756665484d8SDoug Ambrisko sc->aen_cmd = cmd; 757665484d8SDoug Ambrisko 758665484d8SDoug Ambrisko /* 7598e727371SKashyap D Desai * Issue the aen registration frame 760665484d8SDoug Ambrisko */ 761665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)) { 762665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot issue AEN DCMD command.\n"); 763665484d8SDoug Ambrisko return (1); 764665484d8SDoug Ambrisko } 765665484d8SDoug Ambrisko return 0; 766665484d8SDoug Ambrisko } 7678e727371SKashyap D Desai 7688e727371SKashyap D Desai /* 7698e727371SKashyap D Desai * mrsas_start_aen: Subscribes to AEN during driver load time 770665484d8SDoug Ambrisko * @instance: Adapter soft state 771665484d8SDoug Ambrisko */ 7728e727371SKashyap D Desai static int 7738e727371SKashyap D Desai mrsas_start_aen(struct mrsas_softc *sc) 774665484d8SDoug Ambrisko { 775665484d8SDoug Ambrisko struct mrsas_evt_log_info eli; 776665484d8SDoug Ambrisko union mrsas_evt_class_locale class_locale; 777665484d8SDoug Ambrisko 778665484d8SDoug Ambrisko /* Get the latest sequence number from FW */ 779665484d8SDoug Ambrisko 780665484d8SDoug Ambrisko memset(&eli, 0, sizeof(eli)); 781665484d8SDoug Ambrisko 782665484d8SDoug Ambrisko if (mrsas_get_seq_num(sc, &eli)) 783665484d8SDoug Ambrisko return -1; 784665484d8SDoug Ambrisko 785665484d8SDoug Ambrisko /* Register AEN with FW for latest sequence number plus 1 */ 786665484d8SDoug Ambrisko class_locale.members.reserved = 0; 787665484d8SDoug Ambrisko class_locale.members.locale = MR_EVT_LOCALE_ALL; 788665484d8SDoug Ambrisko class_locale.members.class = MR_EVT_CLASS_DEBUG; 789665484d8SDoug Ambrisko 790665484d8SDoug Ambrisko return mrsas_register_aen(sc, eli.newest_seq_num + 1, 791665484d8SDoug Ambrisko class_locale.word); 792d18d1b47SKashyap D Desai 793665484d8SDoug Ambrisko } 794665484d8SDoug Ambrisko 7958e727371SKashyap D Desai /* 796d18d1b47SKashyap D Desai * mrsas_setup_msix: Allocate MSI-x vectors 7978e727371SKashyap D Desai * @sc: adapter soft state 798d18d1b47SKashyap D Desai */ 7998e727371SKashyap D Desai static int 8008e727371SKashyap D Desai mrsas_setup_msix(struct mrsas_softc *sc) 801d18d1b47SKashyap D Desai { 802d18d1b47SKashyap D Desai int i; 8038e727371SKashyap D Desai 804d18d1b47SKashyap D Desai for (i = 0; i < sc->msix_vectors; i++) { 805d18d1b47SKashyap D Desai sc->irq_context[i].sc = sc; 806d18d1b47SKashyap D Desai sc->irq_context[i].MSIxIndex = i; 807d18d1b47SKashyap D Desai sc->irq_id[i] = i + 1; 808d18d1b47SKashyap D Desai sc->mrsas_irq[i] = bus_alloc_resource_any 809d18d1b47SKashyap D Desai (sc->mrsas_dev, SYS_RES_IRQ, &sc->irq_id[i] 810d18d1b47SKashyap D Desai ,RF_ACTIVE); 811d18d1b47SKashyap D Desai if (sc->mrsas_irq[i] == NULL) { 812d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "Can't allocate MSI-x\n"); 813d18d1b47SKashyap D Desai goto irq_alloc_failed; 814d18d1b47SKashyap D Desai } 815d18d1b47SKashyap D Desai if (bus_setup_intr(sc->mrsas_dev, 816d18d1b47SKashyap D Desai sc->mrsas_irq[i], 817d18d1b47SKashyap D Desai INTR_MPSAFE | INTR_TYPE_CAM, 818d18d1b47SKashyap D Desai NULL, mrsas_isr, &sc->irq_context[i], 819d18d1b47SKashyap D Desai &sc->intr_handle[i])) { 820d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, 821d18d1b47SKashyap D Desai "Cannot set up MSI-x interrupt handler\n"); 822d18d1b47SKashyap D Desai goto irq_alloc_failed; 823d18d1b47SKashyap D Desai } 824d18d1b47SKashyap D Desai } 825d18d1b47SKashyap D Desai return SUCCESS; 826d18d1b47SKashyap D Desai 827d18d1b47SKashyap D Desai irq_alloc_failed: 828d18d1b47SKashyap D Desai mrsas_teardown_intr(sc); 829d18d1b47SKashyap D Desai return (FAIL); 830d18d1b47SKashyap D Desai } 831d18d1b47SKashyap D Desai 8328e727371SKashyap D Desai /* 833d18d1b47SKashyap D Desai * mrsas_allocate_msix: Setup MSI-x vectors 8348e727371SKashyap D Desai * @sc: adapter soft state 835d18d1b47SKashyap D Desai */ 8368e727371SKashyap D Desai static int 8378e727371SKashyap D Desai mrsas_allocate_msix(struct mrsas_softc *sc) 838d18d1b47SKashyap D Desai { 839d18d1b47SKashyap D Desai if (pci_alloc_msix(sc->mrsas_dev, &sc->msix_vectors) == 0) { 840d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "Using MSI-X with %d number" 841d18d1b47SKashyap D Desai " of vectors\n", sc->msix_vectors); 842d18d1b47SKashyap D Desai } else { 843d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "MSI-x setup failed\n"); 844d18d1b47SKashyap D Desai goto irq_alloc_failed; 845d18d1b47SKashyap D Desai } 846d18d1b47SKashyap D Desai return SUCCESS; 847d18d1b47SKashyap D Desai 848d18d1b47SKashyap D Desai irq_alloc_failed: 849d18d1b47SKashyap D Desai mrsas_teardown_intr(sc); 850d18d1b47SKashyap D Desai return (FAIL); 851d18d1b47SKashyap D Desai } 8528e727371SKashyap D Desai 8538e727371SKashyap D Desai /* 854665484d8SDoug Ambrisko * mrsas_attach: PCI entry point 8558e727371SKashyap D Desai * input: pointer to device struct 856665484d8SDoug Ambrisko * 8578e727371SKashyap D Desai * Performs setup of PCI and registers, initializes mutexes and linked lists, 8588e727371SKashyap D Desai * registers interrupts and CAM, and initializes the adapter/controller to 8598e727371SKashyap D Desai * its proper state. 860665484d8SDoug Ambrisko */ 8618e727371SKashyap D Desai static int 8628e727371SKashyap D Desai mrsas_attach(device_t dev) 863665484d8SDoug Ambrisko { 864665484d8SDoug Ambrisko struct mrsas_softc *sc = device_get_softc(dev); 8657aade8bfSKashyap D Desai uint32_t cmd, error; 866665484d8SDoug Ambrisko 8674bb0a4f0SKashyap D Desai memset(sc, 0, sizeof(struct mrsas_softc)); 8684bb0a4f0SKashyap D Desai 869665484d8SDoug Ambrisko /* Look up our softc and initialize its fields. */ 870665484d8SDoug Ambrisko sc->mrsas_dev = dev; 871665484d8SDoug Ambrisko sc->device_id = pci_get_device(dev); 872665484d8SDoug Ambrisko 8732909aab4SKashyap D Desai switch (sc->device_id) { 8742909aab4SKashyap D Desai case MRSAS_INVADER: 8752909aab4SKashyap D Desai case MRSAS_FURY: 8762909aab4SKashyap D Desai case MRSAS_INTRUDER: 8772909aab4SKashyap D Desai case MRSAS_INTRUDER_24: 8782909aab4SKashyap D Desai case MRSAS_CUTLASS_52: 8792909aab4SKashyap D Desai case MRSAS_CUTLASS_53: 880f9c63081SKashyap D Desai sc->mrsas_gen3_ctrl = 1; 8812909aab4SKashyap D Desai break; 8822909aab4SKashyap D Desai case MRSAS_VENTURA: 8832909aab4SKashyap D Desai case MRSAS_CRUSADER: 8842909aab4SKashyap D Desai case MRSAS_HARPOON: 8852909aab4SKashyap D Desai case MRSAS_TOMCAT: 8862909aab4SKashyap D Desai case MRSAS_VENTURA_4PORT: 8872909aab4SKashyap D Desai case MRSAS_CRUSADER_4PORT: 8887aade8bfSKashyap D Desai sc->is_ventura = true; 8892909aab4SKashyap D Desai break; 8902909aab4SKashyap D Desai case MRSAS_AERO_10E1: 8912909aab4SKashyap D Desai case MRSAS_AERO_10E5: 8922909aab4SKashyap D Desai device_printf(dev, "Adapter is in configurable secure mode\n"); 8932909aab4SKashyap D Desai case MRSAS_AERO_10E2: 8942909aab4SKashyap D Desai case MRSAS_AERO_10E6: 8952909aab4SKashyap D Desai sc->is_aero = true; 8962909aab4SKashyap D Desai break; 8972909aab4SKashyap D Desai case MRSAS_AERO_10E0: 8982909aab4SKashyap D Desai case MRSAS_AERO_10E3: 8992909aab4SKashyap D Desai case MRSAS_AERO_10E4: 9002909aab4SKashyap D Desai case MRSAS_AERO_10E7: 9012909aab4SKashyap D Desai device_printf(dev, "Adapter is in non-secure mode\n"); 9022909aab4SKashyap D Desai return SUCCESS; 903f9c63081SKashyap D Desai } 904f9c63081SKashyap D Desai 905665484d8SDoug Ambrisko mrsas_get_tunables(sc); 906665484d8SDoug Ambrisko 907665484d8SDoug Ambrisko /* 908665484d8SDoug Ambrisko * Set up PCI and registers 909665484d8SDoug Ambrisko */ 910665484d8SDoug Ambrisko cmd = pci_read_config(dev, PCIR_COMMAND, 2); 911665484d8SDoug Ambrisko /* Force the busmaster enable bit on. */ 912665484d8SDoug Ambrisko cmd |= PCIM_CMD_BUSMASTEREN; 913665484d8SDoug Ambrisko pci_write_config(dev, PCIR_COMMAND, cmd, 2); 914665484d8SDoug Ambrisko 9152909aab4SKashyap D Desai /* For Ventura/Aero system registers are mapped to BAR0 */ 9162909aab4SKashyap D Desai if (sc->is_ventura || sc->is_aero) 9177aade8bfSKashyap D Desai sc->reg_res_id = PCIR_BAR(0); /* BAR0 offset */ 9187aade8bfSKashyap D Desai else 9197aade8bfSKashyap D Desai sc->reg_res_id = PCIR_BAR(1); /* BAR1 offset */ 920665484d8SDoug Ambrisko 92143cd6160SJustin Hibbits if ((sc->reg_res = bus_alloc_resource_any(dev, SYS_RES_MEMORY, 92243cd6160SJustin Hibbits &(sc->reg_res_id), RF_ACTIVE)) 923665484d8SDoug Ambrisko == NULL) { 924665484d8SDoug Ambrisko device_printf(dev, "Cannot allocate PCI registers\n"); 925665484d8SDoug Ambrisko goto attach_fail; 926665484d8SDoug Ambrisko } 927665484d8SDoug Ambrisko sc->bus_tag = rman_get_bustag(sc->reg_res); 928665484d8SDoug Ambrisko sc->bus_handle = rman_get_bushandle(sc->reg_res); 929665484d8SDoug Ambrisko 930665484d8SDoug Ambrisko /* Intialize mutexes */ 931665484d8SDoug Ambrisko mtx_init(&sc->sim_lock, "mrsas_sim_lock", NULL, MTX_DEF); 932665484d8SDoug Ambrisko mtx_init(&sc->pci_lock, "mrsas_pci_lock", NULL, MTX_DEF); 933665484d8SDoug Ambrisko mtx_init(&sc->io_lock, "mrsas_io_lock", NULL, MTX_DEF); 934665484d8SDoug Ambrisko mtx_init(&sc->aen_lock, "mrsas_aen_lock", NULL, MTX_DEF); 935665484d8SDoug Ambrisko mtx_init(&sc->ioctl_lock, "mrsas_ioctl_lock", NULL, MTX_SPIN); 936665484d8SDoug Ambrisko mtx_init(&sc->mpt_cmd_pool_lock, "mrsas_mpt_cmd_pool_lock", NULL, MTX_DEF); 937665484d8SDoug Ambrisko mtx_init(&sc->mfi_cmd_pool_lock, "mrsas_mfi_cmd_pool_lock", NULL, MTX_DEF); 938665484d8SDoug Ambrisko mtx_init(&sc->raidmap_lock, "mrsas_raidmap_lock", NULL, MTX_DEF); 939821df4b9SKashyap D Desai mtx_init(&sc->stream_lock, "mrsas_stream_lock", NULL, MTX_DEF); 940665484d8SDoug Ambrisko 941665484d8SDoug Ambrisko /* Intialize linked list */ 942665484d8SDoug Ambrisko TAILQ_INIT(&sc->mrsas_mpt_cmd_list_head); 943665484d8SDoug Ambrisko TAILQ_INIT(&sc->mrsas_mfi_cmd_list_head); 944665484d8SDoug Ambrisko 945f5fb2237SKashyap D Desai mrsas_atomic_set(&sc->fw_outstanding, 0); 9468bb601acSKashyap D Desai mrsas_atomic_set(&sc->target_reset_outstanding, 0); 9473d273176SKashyap D Desai mrsas_atomic_set(&sc->prp_count, 0); 9483d273176SKashyap D Desai mrsas_atomic_set(&sc->sge_holes, 0); 949665484d8SDoug Ambrisko 950665484d8SDoug Ambrisko sc->io_cmds_highwater = 0; 951665484d8SDoug Ambrisko 952665484d8SDoug Ambrisko sc->adprecovery = MRSAS_HBA_OPERATIONAL; 953665484d8SDoug Ambrisko sc->UnevenSpanSupport = 0; 954665484d8SDoug Ambrisko 955d18d1b47SKashyap D Desai sc->msix_enable = 0; 956d18d1b47SKashyap D Desai 957665484d8SDoug Ambrisko /* Initialize Firmware */ 958665484d8SDoug Ambrisko if (mrsas_init_fw(sc) != SUCCESS) { 959665484d8SDoug Ambrisko goto attach_fail_fw; 960665484d8SDoug Ambrisko } 9618071588dSKashyap D Desai /* Register mrsas to CAM layer */ 962665484d8SDoug Ambrisko if ((mrsas_cam_attach(sc) != SUCCESS)) { 963665484d8SDoug Ambrisko goto attach_fail_cam; 964665484d8SDoug Ambrisko } 965665484d8SDoug Ambrisko /* Register IRQs */ 966665484d8SDoug Ambrisko if (mrsas_setup_irq(sc) != SUCCESS) { 967665484d8SDoug Ambrisko goto attach_fail_irq; 968665484d8SDoug Ambrisko } 969665484d8SDoug Ambrisko error = mrsas_kproc_create(mrsas_ocr_thread, sc, 970665484d8SDoug Ambrisko &sc->ocr_thread, 0, 0, "mrsas_ocr%d", 971665484d8SDoug Ambrisko device_get_unit(sc->mrsas_dev)); 972665484d8SDoug Ambrisko if (error) { 9738071588dSKashyap D Desai device_printf(sc->mrsas_dev, "Error %d starting OCR thread\n", error); 9748071588dSKashyap D Desai goto attach_fail_ocr_thread; 975665484d8SDoug Ambrisko } 976536094dcSKashyap D Desai /* 9778071588dSKashyap D Desai * After FW initialization and OCR thread creation 9788071588dSKashyap D Desai * we will defer the cdev creation, AEN setup on ICH callback 979536094dcSKashyap D Desai */ 9808071588dSKashyap D Desai sc->mrsas_ich.ich_func = mrsas_ich_startup; 9818071588dSKashyap D Desai sc->mrsas_ich.ich_arg = sc; 9828071588dSKashyap D Desai if (config_intrhook_establish(&sc->mrsas_ich) != 0) { 9838071588dSKashyap D Desai device_printf(sc->mrsas_dev, "Config hook is already established\n"); 9848071588dSKashyap D Desai } 9858071588dSKashyap D Desai mrsas_setup_sysctl(sc); 9868071588dSKashyap D Desai return SUCCESS; 987536094dcSKashyap D Desai 9888071588dSKashyap D Desai attach_fail_ocr_thread: 9898071588dSKashyap D Desai if (sc->ocr_thread_active) 9908071588dSKashyap D Desai wakeup(&sc->ocr_chan); 991665484d8SDoug Ambrisko attach_fail_irq: 992665484d8SDoug Ambrisko mrsas_teardown_intr(sc); 993665484d8SDoug Ambrisko attach_fail_cam: 994665484d8SDoug Ambrisko mrsas_cam_detach(sc); 995665484d8SDoug Ambrisko attach_fail_fw: 996d18d1b47SKashyap D Desai /* if MSIX vector is allocated and FW Init FAILED then release MSIX */ 997d18d1b47SKashyap D Desai if (sc->msix_enable == 1) 998d18d1b47SKashyap D Desai pci_release_msi(sc->mrsas_dev); 999665484d8SDoug Ambrisko mrsas_free_mem(sc); 1000665484d8SDoug Ambrisko mtx_destroy(&sc->sim_lock); 1001665484d8SDoug Ambrisko mtx_destroy(&sc->aen_lock); 1002665484d8SDoug Ambrisko mtx_destroy(&sc->pci_lock); 1003665484d8SDoug Ambrisko mtx_destroy(&sc->io_lock); 1004665484d8SDoug Ambrisko mtx_destroy(&sc->ioctl_lock); 1005665484d8SDoug Ambrisko mtx_destroy(&sc->mpt_cmd_pool_lock); 1006665484d8SDoug Ambrisko mtx_destroy(&sc->mfi_cmd_pool_lock); 1007665484d8SDoug Ambrisko mtx_destroy(&sc->raidmap_lock); 1008821df4b9SKashyap D Desai mtx_destroy(&sc->stream_lock); 1009665484d8SDoug Ambrisko attach_fail: 1010665484d8SDoug Ambrisko if (sc->reg_res) { 1011665484d8SDoug Ambrisko bus_release_resource(sc->mrsas_dev, SYS_RES_MEMORY, 1012665484d8SDoug Ambrisko sc->reg_res_id, sc->reg_res); 1013665484d8SDoug Ambrisko } 1014665484d8SDoug Ambrisko return (ENXIO); 1015665484d8SDoug Ambrisko } 1016665484d8SDoug Ambrisko 10178e727371SKashyap D Desai /* 10188071588dSKashyap D Desai * Interrupt config hook 10198071588dSKashyap D Desai */ 10208071588dSKashyap D Desai static void 10218071588dSKashyap D Desai mrsas_ich_startup(void *arg) 10228071588dSKashyap D Desai { 102379b4460bSKashyap D Desai int i = 0; 10248071588dSKashyap D Desai struct mrsas_softc *sc = (struct mrsas_softc *)arg; 10258071588dSKashyap D Desai 10268071588dSKashyap D Desai /* 10278071588dSKashyap D Desai * Intialize a counting Semaphore to take care no. of concurrent IOCTLs 10288071588dSKashyap D Desai */ 1029731b7561SKashyap D Desai sema_init(&sc->ioctl_count_sema, MRSAS_MAX_IOCTL_CMDS, 10308071588dSKashyap D Desai IOCTL_SEMA_DESCRIPTION); 10318071588dSKashyap D Desai 10328071588dSKashyap D Desai /* Create a /dev entry for mrsas controller. */ 10338071588dSKashyap D Desai sc->mrsas_cdev = make_dev(&mrsas_cdevsw, device_get_unit(sc->mrsas_dev), UID_ROOT, 10348071588dSKashyap D Desai GID_OPERATOR, (S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP), "mrsas%u", 10358071588dSKashyap D Desai device_get_unit(sc->mrsas_dev)); 10368071588dSKashyap D Desai 10378071588dSKashyap D Desai if (device_get_unit(sc->mrsas_dev) == 0) { 10388071588dSKashyap D Desai make_dev_alias_p(MAKEDEV_CHECKNAME, 10398071588dSKashyap D Desai &sc->mrsas_linux_emulator_cdev, sc->mrsas_cdev, 10408071588dSKashyap D Desai "megaraid_sas_ioctl_node"); 10418071588dSKashyap D Desai } 10428071588dSKashyap D Desai if (sc->mrsas_cdev) 10438071588dSKashyap D Desai sc->mrsas_cdev->si_drv1 = sc; 10448071588dSKashyap D Desai 10458071588dSKashyap D Desai /* 10468071588dSKashyap D Desai * Add this controller to mrsas_mgmt_info structure so that it can be 10478071588dSKashyap D Desai * exported to management applications 10488071588dSKashyap D Desai */ 10498071588dSKashyap D Desai if (device_get_unit(sc->mrsas_dev) == 0) 10508071588dSKashyap D Desai memset(&mrsas_mgmt_info, 0, sizeof(mrsas_mgmt_info)); 10518071588dSKashyap D Desai 10528071588dSKashyap D Desai mrsas_mgmt_info.count++; 10538071588dSKashyap D Desai mrsas_mgmt_info.sc_ptr[mrsas_mgmt_info.max_index] = sc; 10548071588dSKashyap D Desai mrsas_mgmt_info.max_index++; 10558071588dSKashyap D Desai 10568071588dSKashyap D Desai /* Enable Interrupts */ 10578071588dSKashyap D Desai mrsas_enable_intr(sc); 10588071588dSKashyap D Desai 105979b4460bSKashyap D Desai /* Call DCMD get_pd_info for all system PDs */ 106079b4460bSKashyap D Desai for (i = 0; i < MRSAS_MAX_PD; i++) { 106179b4460bSKashyap D Desai if ((sc->target_list[i].target_id != 0xffff) && 106279b4460bSKashyap D Desai sc->pd_info_mem) 106379b4460bSKashyap D Desai mrsas_get_pd_info(sc, sc->target_list[i].target_id); 106479b4460bSKashyap D Desai } 106579b4460bSKashyap D Desai 10668071588dSKashyap D Desai /* Initiate AEN (Asynchronous Event Notification) */ 10678071588dSKashyap D Desai if (mrsas_start_aen(sc)) { 10688071588dSKashyap D Desai device_printf(sc->mrsas_dev, "Error: AEN registration FAILED !!! " 10698071588dSKashyap D Desai "Further events from the controller will not be communicated.\n" 10708071588dSKashyap D Desai "Either there is some problem in the controller" 10718071588dSKashyap D Desai "or the controller does not support AEN.\n" 10728071588dSKashyap D Desai "Please contact to the SUPPORT TEAM if the problem persists\n"); 10738071588dSKashyap D Desai } 10748071588dSKashyap D Desai if (sc->mrsas_ich.ich_arg != NULL) { 10758071588dSKashyap D Desai device_printf(sc->mrsas_dev, "Disestablish mrsas intr hook\n"); 10768071588dSKashyap D Desai config_intrhook_disestablish(&sc->mrsas_ich); 10778071588dSKashyap D Desai sc->mrsas_ich.ich_arg = NULL; 10788071588dSKashyap D Desai } 10798071588dSKashyap D Desai } 10808071588dSKashyap D Desai 10818071588dSKashyap D Desai /* 1082665484d8SDoug Ambrisko * mrsas_detach: De-allocates and teardown resources 10838e727371SKashyap D Desai * input: pointer to device struct 1084665484d8SDoug Ambrisko * 10858e727371SKashyap D Desai * This function is the entry point for device disconnect and detach. 10868e727371SKashyap D Desai * It performs memory de-allocations, shutdown of the controller and various 1087665484d8SDoug Ambrisko * teardown and destroy resource functions. 1088665484d8SDoug Ambrisko */ 10898e727371SKashyap D Desai static int 10908e727371SKashyap D Desai mrsas_detach(device_t dev) 1091665484d8SDoug Ambrisko { 1092665484d8SDoug Ambrisko struct mrsas_softc *sc; 1093665484d8SDoug Ambrisko int i = 0; 1094665484d8SDoug Ambrisko 1095665484d8SDoug Ambrisko sc = device_get_softc(dev); 1096665484d8SDoug Ambrisko sc->remove_in_progress = 1; 1097536094dcSKashyap D Desai 1098839ee025SKashyap D Desai /* Destroy the character device so no other IOCTL will be handled */ 10998071588dSKashyap D Desai if ((device_get_unit(dev) == 0) && sc->mrsas_linux_emulator_cdev) 11008071588dSKashyap D Desai destroy_dev(sc->mrsas_linux_emulator_cdev); 1101839ee025SKashyap D Desai destroy_dev(sc->mrsas_cdev); 1102839ee025SKashyap D Desai 1103536094dcSKashyap D Desai /* 1104536094dcSKashyap D Desai * Take the instance off the instance array. Note that we will not 1105536094dcSKashyap D Desai * decrement the max_index. We let this array be sparse array 1106536094dcSKashyap D Desai */ 1107536094dcSKashyap D Desai for (i = 0; i < mrsas_mgmt_info.max_index; i++) { 1108536094dcSKashyap D Desai if (mrsas_mgmt_info.sc_ptr[i] == sc) { 1109536094dcSKashyap D Desai mrsas_mgmt_info.count--; 1110536094dcSKashyap D Desai mrsas_mgmt_info.sc_ptr[i] = NULL; 1111536094dcSKashyap D Desai break; 1112536094dcSKashyap D Desai } 1113536094dcSKashyap D Desai } 1114536094dcSKashyap D Desai 1115665484d8SDoug Ambrisko if (sc->ocr_thread_active) 1116665484d8SDoug Ambrisko wakeup(&sc->ocr_chan); 1117665484d8SDoug Ambrisko while (sc->reset_in_progress) { 1118665484d8SDoug Ambrisko i++; 1119665484d8SDoug Ambrisko if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) { 1120665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_INFO, 1121f0c7594bSKashyap D Desai "[%2d]waiting for OCR to be finished from %s\n", i, __func__); 1122665484d8SDoug Ambrisko } 1123665484d8SDoug Ambrisko pause("mr_shutdown", hz); 1124665484d8SDoug Ambrisko } 1125665484d8SDoug Ambrisko i = 0; 1126665484d8SDoug Ambrisko while (sc->ocr_thread_active) { 1127665484d8SDoug Ambrisko i++; 1128665484d8SDoug Ambrisko if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) { 1129665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_INFO, 1130665484d8SDoug Ambrisko "[%2d]waiting for " 1131665484d8SDoug Ambrisko "mrsas_ocr thread to quit ocr %d\n", i, 1132665484d8SDoug Ambrisko sc->ocr_thread_active); 1133665484d8SDoug Ambrisko } 1134665484d8SDoug Ambrisko pause("mr_shutdown", hz); 1135665484d8SDoug Ambrisko } 1136665484d8SDoug Ambrisko mrsas_flush_cache(sc); 1137665484d8SDoug Ambrisko mrsas_shutdown_ctlr(sc, MR_DCMD_CTRL_SHUTDOWN); 1138665484d8SDoug Ambrisko mrsas_disable_intr(sc); 1139821df4b9SKashyap D Desai 11402909aab4SKashyap D Desai if ((sc->is_ventura || sc->is_aero) && sc->streamDetectByLD) { 1141821df4b9SKashyap D Desai for (i = 0; i < MAX_LOGICAL_DRIVES_EXT; ++i) 1142821df4b9SKashyap D Desai free(sc->streamDetectByLD[i], M_MRSAS); 1143821df4b9SKashyap D Desai free(sc->streamDetectByLD, M_MRSAS); 1144821df4b9SKashyap D Desai sc->streamDetectByLD = NULL; 1145821df4b9SKashyap D Desai } 1146821df4b9SKashyap D Desai 1147665484d8SDoug Ambrisko mrsas_cam_detach(sc); 1148665484d8SDoug Ambrisko mrsas_teardown_intr(sc); 1149665484d8SDoug Ambrisko mrsas_free_mem(sc); 1150665484d8SDoug Ambrisko mtx_destroy(&sc->sim_lock); 1151665484d8SDoug Ambrisko mtx_destroy(&sc->aen_lock); 1152665484d8SDoug Ambrisko mtx_destroy(&sc->pci_lock); 1153665484d8SDoug Ambrisko mtx_destroy(&sc->io_lock); 1154665484d8SDoug Ambrisko mtx_destroy(&sc->ioctl_lock); 1155665484d8SDoug Ambrisko mtx_destroy(&sc->mpt_cmd_pool_lock); 1156665484d8SDoug Ambrisko mtx_destroy(&sc->mfi_cmd_pool_lock); 1157665484d8SDoug Ambrisko mtx_destroy(&sc->raidmap_lock); 1158821df4b9SKashyap D Desai mtx_destroy(&sc->stream_lock); 1159839ee025SKashyap D Desai 1160839ee025SKashyap D Desai /* Wait for all the semaphores to be released */ 1161731b7561SKashyap D Desai while (sema_value(&sc->ioctl_count_sema) != MRSAS_MAX_IOCTL_CMDS) 1162839ee025SKashyap D Desai pause("mr_shutdown", hz); 1163839ee025SKashyap D Desai 1164839ee025SKashyap D Desai /* Destroy the counting semaphore created for Ioctl */ 1165839ee025SKashyap D Desai sema_destroy(&sc->ioctl_count_sema); 1166839ee025SKashyap D Desai 1167665484d8SDoug Ambrisko if (sc->reg_res) { 1168665484d8SDoug Ambrisko bus_release_resource(sc->mrsas_dev, 1169665484d8SDoug Ambrisko SYS_RES_MEMORY, sc->reg_res_id, sc->reg_res); 1170665484d8SDoug Ambrisko } 1171665484d8SDoug Ambrisko if (sc->sysctl_tree != NULL) 1172665484d8SDoug Ambrisko sysctl_ctx_free(&sc->sysctl_ctx); 1173839ee025SKashyap D Desai 1174665484d8SDoug Ambrisko return (0); 1175665484d8SDoug Ambrisko } 1176665484d8SDoug Ambrisko 1177f28ecf2bSAndriy Gapon static int 1178f28ecf2bSAndriy Gapon mrsas_shutdown(device_t dev) 1179f28ecf2bSAndriy Gapon { 1180f28ecf2bSAndriy Gapon struct mrsas_softc *sc; 1181f28ecf2bSAndriy Gapon int i; 1182f28ecf2bSAndriy Gapon 1183f28ecf2bSAndriy Gapon sc = device_get_softc(dev); 1184f28ecf2bSAndriy Gapon sc->remove_in_progress = 1; 1185879e0604SMateusz Guzik if (!KERNEL_PANICKED()) { 1186f28ecf2bSAndriy Gapon if (sc->ocr_thread_active) 1187f28ecf2bSAndriy Gapon wakeup(&sc->ocr_chan); 1188f28ecf2bSAndriy Gapon i = 0; 1189f28ecf2bSAndriy Gapon while (sc->reset_in_progress && i < 15) { 1190f28ecf2bSAndriy Gapon i++; 1191f28ecf2bSAndriy Gapon if ((i % MRSAS_RESET_NOTICE_INTERVAL) == 0) { 1192f28ecf2bSAndriy Gapon mrsas_dprint(sc, MRSAS_INFO, 1193f28ecf2bSAndriy Gapon "[%2d]waiting for OCR to be finished " 1194f28ecf2bSAndriy Gapon "from %s\n", i, __func__); 1195f28ecf2bSAndriy Gapon } 1196f28ecf2bSAndriy Gapon pause("mr_shutdown", hz); 1197f28ecf2bSAndriy Gapon } 1198f28ecf2bSAndriy Gapon if (sc->reset_in_progress) { 1199f28ecf2bSAndriy Gapon mrsas_dprint(sc, MRSAS_INFO, 1200f28ecf2bSAndriy Gapon "gave up waiting for OCR to be finished\n"); 1201f28ecf2bSAndriy Gapon } 1202f28ecf2bSAndriy Gapon } 1203f28ecf2bSAndriy Gapon 1204f28ecf2bSAndriy Gapon mrsas_flush_cache(sc); 1205f28ecf2bSAndriy Gapon mrsas_shutdown_ctlr(sc, MR_DCMD_CTRL_SHUTDOWN); 1206f28ecf2bSAndriy Gapon mrsas_disable_intr(sc); 1207f28ecf2bSAndriy Gapon return (0); 1208f28ecf2bSAndriy Gapon } 1209f28ecf2bSAndriy Gapon 12108e727371SKashyap D Desai /* 1211665484d8SDoug Ambrisko * mrsas_free_mem: Frees allocated memory 1212665484d8SDoug Ambrisko * input: Adapter instance soft state 1213665484d8SDoug Ambrisko * 1214665484d8SDoug Ambrisko * This function is called from mrsas_detach() to free previously allocated 1215665484d8SDoug Ambrisko * memory. 1216665484d8SDoug Ambrisko */ 12178e727371SKashyap D Desai void 12188e727371SKashyap D Desai mrsas_free_mem(struct mrsas_softc *sc) 1219665484d8SDoug Ambrisko { 1220665484d8SDoug Ambrisko int i; 12212a1d3bcdSKashyap D Desai u_int32_t max_fw_cmds; 1222665484d8SDoug Ambrisko struct mrsas_mfi_cmd *mfi_cmd; 1223665484d8SDoug Ambrisko struct mrsas_mpt_cmd *mpt_cmd; 1224665484d8SDoug Ambrisko 1225665484d8SDoug Ambrisko /* 1226665484d8SDoug Ambrisko * Free RAID map memory 1227665484d8SDoug Ambrisko */ 12288e727371SKashyap D Desai for (i = 0; i < 2; i++) { 1229665484d8SDoug Ambrisko if (sc->raidmap_phys_addr[i]) 1230665484d8SDoug Ambrisko bus_dmamap_unload(sc->raidmap_tag[i], sc->raidmap_dmamap[i]); 1231665484d8SDoug Ambrisko if (sc->raidmap_mem[i] != NULL) 1232665484d8SDoug Ambrisko bus_dmamem_free(sc->raidmap_tag[i], sc->raidmap_mem[i], sc->raidmap_dmamap[i]); 1233665484d8SDoug Ambrisko if (sc->raidmap_tag[i] != NULL) 1234665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->raidmap_tag[i]); 12354799d485SKashyap D Desai 12364799d485SKashyap D Desai if (sc->ld_drv_map[i] != NULL) 12374799d485SKashyap D Desai free(sc->ld_drv_map[i], M_MRSAS); 1238665484d8SDoug Ambrisko } 1239a688fcd0SKashyap D Desai for (i = 0; i < 2; i++) { 1240a688fcd0SKashyap D Desai if (sc->jbodmap_phys_addr[i]) 1241a688fcd0SKashyap D Desai bus_dmamap_unload(sc->jbodmap_tag[i], sc->jbodmap_dmamap[i]); 1242a688fcd0SKashyap D Desai if (sc->jbodmap_mem[i] != NULL) 1243a688fcd0SKashyap D Desai bus_dmamem_free(sc->jbodmap_tag[i], sc->jbodmap_mem[i], sc->jbodmap_dmamap[i]); 1244a688fcd0SKashyap D Desai if (sc->jbodmap_tag[i] != NULL) 1245a688fcd0SKashyap D Desai bus_dma_tag_destroy(sc->jbodmap_tag[i]); 1246a688fcd0SKashyap D Desai } 1247665484d8SDoug Ambrisko /* 1248453130d9SPedro F. Giffuni * Free version buffer memory 1249665484d8SDoug Ambrisko */ 1250665484d8SDoug Ambrisko if (sc->verbuf_phys_addr) 1251665484d8SDoug Ambrisko bus_dmamap_unload(sc->verbuf_tag, sc->verbuf_dmamap); 1252665484d8SDoug Ambrisko if (sc->verbuf_mem != NULL) 1253665484d8SDoug Ambrisko bus_dmamem_free(sc->verbuf_tag, sc->verbuf_mem, sc->verbuf_dmamap); 1254665484d8SDoug Ambrisko if (sc->verbuf_tag != NULL) 1255665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->verbuf_tag); 1256665484d8SDoug Ambrisko 1257665484d8SDoug Ambrisko /* 1258665484d8SDoug Ambrisko * Free sense buffer memory 1259665484d8SDoug Ambrisko */ 1260665484d8SDoug Ambrisko if (sc->sense_phys_addr) 1261665484d8SDoug Ambrisko bus_dmamap_unload(sc->sense_tag, sc->sense_dmamap); 1262665484d8SDoug Ambrisko if (sc->sense_mem != NULL) 1263665484d8SDoug Ambrisko bus_dmamem_free(sc->sense_tag, sc->sense_mem, sc->sense_dmamap); 1264665484d8SDoug Ambrisko if (sc->sense_tag != NULL) 1265665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->sense_tag); 1266665484d8SDoug Ambrisko 1267665484d8SDoug Ambrisko /* 1268665484d8SDoug Ambrisko * Free chain frame memory 1269665484d8SDoug Ambrisko */ 1270665484d8SDoug Ambrisko if (sc->chain_frame_phys_addr) 1271665484d8SDoug Ambrisko bus_dmamap_unload(sc->chain_frame_tag, sc->chain_frame_dmamap); 1272665484d8SDoug Ambrisko if (sc->chain_frame_mem != NULL) 1273665484d8SDoug Ambrisko bus_dmamem_free(sc->chain_frame_tag, sc->chain_frame_mem, sc->chain_frame_dmamap); 1274665484d8SDoug Ambrisko if (sc->chain_frame_tag != NULL) 1275665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->chain_frame_tag); 1276665484d8SDoug Ambrisko 1277665484d8SDoug Ambrisko /* 1278665484d8SDoug Ambrisko * Free IO Request memory 1279665484d8SDoug Ambrisko */ 1280665484d8SDoug Ambrisko if (sc->io_request_phys_addr) 1281665484d8SDoug Ambrisko bus_dmamap_unload(sc->io_request_tag, sc->io_request_dmamap); 1282665484d8SDoug Ambrisko if (sc->io_request_mem != NULL) 1283665484d8SDoug Ambrisko bus_dmamem_free(sc->io_request_tag, sc->io_request_mem, sc->io_request_dmamap); 1284665484d8SDoug Ambrisko if (sc->io_request_tag != NULL) 1285665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->io_request_tag); 1286665484d8SDoug Ambrisko 1287665484d8SDoug Ambrisko /* 1288665484d8SDoug Ambrisko * Free Reply Descriptor memory 1289665484d8SDoug Ambrisko */ 1290665484d8SDoug Ambrisko if (sc->reply_desc_phys_addr) 1291665484d8SDoug Ambrisko bus_dmamap_unload(sc->reply_desc_tag, sc->reply_desc_dmamap); 1292665484d8SDoug Ambrisko if (sc->reply_desc_mem != NULL) 1293665484d8SDoug Ambrisko bus_dmamem_free(sc->reply_desc_tag, sc->reply_desc_mem, sc->reply_desc_dmamap); 1294665484d8SDoug Ambrisko if (sc->reply_desc_tag != NULL) 1295665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->reply_desc_tag); 1296665484d8SDoug Ambrisko 1297665484d8SDoug Ambrisko /* 1298665484d8SDoug Ambrisko * Free event detail memory 1299665484d8SDoug Ambrisko */ 1300665484d8SDoug Ambrisko if (sc->evt_detail_phys_addr) 1301665484d8SDoug Ambrisko bus_dmamap_unload(sc->evt_detail_tag, sc->evt_detail_dmamap); 1302665484d8SDoug Ambrisko if (sc->evt_detail_mem != NULL) 1303665484d8SDoug Ambrisko bus_dmamem_free(sc->evt_detail_tag, sc->evt_detail_mem, sc->evt_detail_dmamap); 1304665484d8SDoug Ambrisko if (sc->evt_detail_tag != NULL) 1305665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->evt_detail_tag); 1306665484d8SDoug Ambrisko 1307665484d8SDoug Ambrisko /* 130879b4460bSKashyap D Desai * Free PD info memory 130979b4460bSKashyap D Desai */ 131079b4460bSKashyap D Desai if (sc->pd_info_phys_addr) 131179b4460bSKashyap D Desai bus_dmamap_unload(sc->pd_info_tag, sc->pd_info_dmamap); 131279b4460bSKashyap D Desai if (sc->pd_info_mem != NULL) 131379b4460bSKashyap D Desai bus_dmamem_free(sc->pd_info_tag, sc->pd_info_mem, sc->pd_info_dmamap); 131479b4460bSKashyap D Desai if (sc->pd_info_tag != NULL) 131579b4460bSKashyap D Desai bus_dma_tag_destroy(sc->pd_info_tag); 131679b4460bSKashyap D Desai 131779b4460bSKashyap D Desai /* 1318665484d8SDoug Ambrisko * Free MFI frames 1319665484d8SDoug Ambrisko */ 1320665484d8SDoug Ambrisko if (sc->mfi_cmd_list) { 1321665484d8SDoug Ambrisko for (i = 0; i < MRSAS_MAX_MFI_CMDS; i++) { 1322665484d8SDoug Ambrisko mfi_cmd = sc->mfi_cmd_list[i]; 1323665484d8SDoug Ambrisko mrsas_free_frame(sc, mfi_cmd); 1324665484d8SDoug Ambrisko } 1325665484d8SDoug Ambrisko } 1326665484d8SDoug Ambrisko if (sc->mficmd_frame_tag != NULL) 1327665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->mficmd_frame_tag); 1328665484d8SDoug Ambrisko 1329665484d8SDoug Ambrisko /* 1330665484d8SDoug Ambrisko * Free MPT internal command list 1331665484d8SDoug Ambrisko */ 13322a1d3bcdSKashyap D Desai max_fw_cmds = sc->max_fw_cmds; 1333665484d8SDoug Ambrisko if (sc->mpt_cmd_list) { 13342a1d3bcdSKashyap D Desai for (i = 0; i < max_fw_cmds; i++) { 1335665484d8SDoug Ambrisko mpt_cmd = sc->mpt_cmd_list[i]; 1336665484d8SDoug Ambrisko bus_dmamap_destroy(sc->data_tag, mpt_cmd->data_dmamap); 1337665484d8SDoug Ambrisko free(sc->mpt_cmd_list[i], M_MRSAS); 1338665484d8SDoug Ambrisko } 1339665484d8SDoug Ambrisko free(sc->mpt_cmd_list, M_MRSAS); 1340665484d8SDoug Ambrisko sc->mpt_cmd_list = NULL; 1341665484d8SDoug Ambrisko } 1342665484d8SDoug Ambrisko /* 1343665484d8SDoug Ambrisko * Free MFI internal command list 1344665484d8SDoug Ambrisko */ 1345665484d8SDoug Ambrisko 1346665484d8SDoug Ambrisko if (sc->mfi_cmd_list) { 1347665484d8SDoug Ambrisko for (i = 0; i < MRSAS_MAX_MFI_CMDS; i++) { 1348665484d8SDoug Ambrisko free(sc->mfi_cmd_list[i], M_MRSAS); 1349665484d8SDoug Ambrisko } 1350665484d8SDoug Ambrisko free(sc->mfi_cmd_list, M_MRSAS); 1351665484d8SDoug Ambrisko sc->mfi_cmd_list = NULL; 1352665484d8SDoug Ambrisko } 1353665484d8SDoug Ambrisko /* 1354665484d8SDoug Ambrisko * Free request descriptor memory 1355665484d8SDoug Ambrisko */ 1356665484d8SDoug Ambrisko free(sc->req_desc, M_MRSAS); 1357665484d8SDoug Ambrisko sc->req_desc = NULL; 1358665484d8SDoug Ambrisko 1359665484d8SDoug Ambrisko /* 1360665484d8SDoug Ambrisko * Destroy parent tag 1361665484d8SDoug Ambrisko */ 1362665484d8SDoug Ambrisko if (sc->mrsas_parent_tag != NULL) 1363665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->mrsas_parent_tag); 1364af51c29fSKashyap D Desai 1365af51c29fSKashyap D Desai /* 1366af51c29fSKashyap D Desai * Free ctrl_info memory 1367af51c29fSKashyap D Desai */ 1368af51c29fSKashyap D Desai if (sc->ctrl_info != NULL) 1369af51c29fSKashyap D Desai free(sc->ctrl_info, M_MRSAS); 1370665484d8SDoug Ambrisko } 1371665484d8SDoug Ambrisko 13728e727371SKashyap D Desai /* 1373665484d8SDoug Ambrisko * mrsas_teardown_intr: Teardown interrupt 1374665484d8SDoug Ambrisko * input: Adapter instance soft state 1375665484d8SDoug Ambrisko * 13768e727371SKashyap D Desai * This function is called from mrsas_detach() to teardown and release bus 13778e727371SKashyap D Desai * interrupt resourse. 1378665484d8SDoug Ambrisko */ 13798e727371SKashyap D Desai void 13808e727371SKashyap D Desai mrsas_teardown_intr(struct mrsas_softc *sc) 1381665484d8SDoug Ambrisko { 1382d18d1b47SKashyap D Desai int i; 13838e727371SKashyap D Desai 1384d18d1b47SKashyap D Desai if (!sc->msix_enable) { 1385d18d1b47SKashyap D Desai if (sc->intr_handle[0]) 1386d18d1b47SKashyap D Desai bus_teardown_intr(sc->mrsas_dev, sc->mrsas_irq[0], sc->intr_handle[0]); 1387d18d1b47SKashyap D Desai if (sc->mrsas_irq[0] != NULL) 13888e727371SKashyap D Desai bus_release_resource(sc->mrsas_dev, SYS_RES_IRQ, 13898e727371SKashyap D Desai sc->irq_id[0], sc->mrsas_irq[0]); 1390d18d1b47SKashyap D Desai sc->intr_handle[0] = NULL; 1391d18d1b47SKashyap D Desai } else { 1392d18d1b47SKashyap D Desai for (i = 0; i < sc->msix_vectors; i++) { 1393d18d1b47SKashyap D Desai if (sc->intr_handle[i]) 1394d18d1b47SKashyap D Desai bus_teardown_intr(sc->mrsas_dev, sc->mrsas_irq[i], 1395d18d1b47SKashyap D Desai sc->intr_handle[i]); 1396d18d1b47SKashyap D Desai 1397d18d1b47SKashyap D Desai if (sc->mrsas_irq[i] != NULL) 1398d18d1b47SKashyap D Desai bus_release_resource(sc->mrsas_dev, SYS_RES_IRQ, 1399d18d1b47SKashyap D Desai sc->irq_id[i], sc->mrsas_irq[i]); 1400d18d1b47SKashyap D Desai 1401d18d1b47SKashyap D Desai sc->intr_handle[i] = NULL; 1402d18d1b47SKashyap D Desai } 1403d18d1b47SKashyap D Desai pci_release_msi(sc->mrsas_dev); 1404d18d1b47SKashyap D Desai } 1405d18d1b47SKashyap D Desai 1406665484d8SDoug Ambrisko } 1407665484d8SDoug Ambrisko 14088e727371SKashyap D Desai /* 1409665484d8SDoug Ambrisko * mrsas_suspend: Suspend entry point 1410665484d8SDoug Ambrisko * input: Device struct pointer 1411665484d8SDoug Ambrisko * 1412665484d8SDoug Ambrisko * This function is the entry point for system suspend from the OS. 1413665484d8SDoug Ambrisko */ 14148e727371SKashyap D Desai static int 14158e727371SKashyap D Desai mrsas_suspend(device_t dev) 1416665484d8SDoug Ambrisko { 14174bb0a4f0SKashyap D Desai /* This will be filled when the driver will have hibernation support */ 1418665484d8SDoug Ambrisko return (0); 1419665484d8SDoug Ambrisko } 1420665484d8SDoug Ambrisko 14218e727371SKashyap D Desai /* 1422665484d8SDoug Ambrisko * mrsas_resume: Resume entry point 1423665484d8SDoug Ambrisko * input: Device struct pointer 1424665484d8SDoug Ambrisko * 1425665484d8SDoug Ambrisko * This function is the entry point for system resume from the OS. 1426665484d8SDoug Ambrisko */ 14278e727371SKashyap D Desai static int 14288e727371SKashyap D Desai mrsas_resume(device_t dev) 1429665484d8SDoug Ambrisko { 14304bb0a4f0SKashyap D Desai /* This will be filled when the driver will have hibernation support */ 1431665484d8SDoug Ambrisko return (0); 1432665484d8SDoug Ambrisko } 1433665484d8SDoug Ambrisko 14345844115eSKashyap D Desai /** 14355844115eSKashyap D Desai * mrsas_get_softc_instance: Find softc instance based on cmd type 14365844115eSKashyap D Desai * 14375844115eSKashyap D Desai * This function will return softc instance based on cmd type. 14385844115eSKashyap D Desai * In some case, application fire ioctl on required management instance and 14395844115eSKashyap D Desai * do not provide host_no. Use cdev->si_drv1 to get softc instance for those 14405844115eSKashyap D Desai * case, else get the softc instance from host_no provided by application in 14415844115eSKashyap D Desai * user data. 14425844115eSKashyap D Desai */ 14435844115eSKashyap D Desai 14445844115eSKashyap D Desai static struct mrsas_softc * 14455844115eSKashyap D Desai mrsas_get_softc_instance(struct cdev *dev, u_long cmd, caddr_t arg) 14465844115eSKashyap D Desai { 14475844115eSKashyap D Desai struct mrsas_softc *sc = NULL; 14485844115eSKashyap D Desai struct mrsas_iocpacket *user_ioc = (struct mrsas_iocpacket *)arg; 1449dbcc81dfSKashyap D Desai 14505844115eSKashyap D Desai if (cmd == MRSAS_IOC_GET_PCI_INFO) { 14515844115eSKashyap D Desai sc = dev->si_drv1; 14525844115eSKashyap D Desai } else { 1453dbcc81dfSKashyap D Desai /* 1454dbcc81dfSKashyap D Desai * get the Host number & the softc from data sent by the 1455dbcc81dfSKashyap D Desai * Application 1456dbcc81dfSKashyap D Desai */ 14575844115eSKashyap D Desai sc = mrsas_mgmt_info.sc_ptr[user_ioc->host_no]; 14585844115eSKashyap D Desai if (sc == NULL) 14595bae00d6SSteven Hartland printf("There is no Controller number %d\n", 14605bae00d6SSteven Hartland user_ioc->host_no); 14615bae00d6SSteven Hartland else if (user_ioc->host_no >= mrsas_mgmt_info.max_index) 14625844115eSKashyap D Desai mrsas_dprint(sc, MRSAS_FAULT, 14635bae00d6SSteven Hartland "Invalid Controller number %d\n", user_ioc->host_no); 14645844115eSKashyap D Desai } 14655844115eSKashyap D Desai 14665844115eSKashyap D Desai return sc; 14675844115eSKashyap D Desai } 14685844115eSKashyap D Desai 14698e727371SKashyap D Desai /* 1470665484d8SDoug Ambrisko * mrsas_ioctl: IOCtl commands entry point. 1471665484d8SDoug Ambrisko * 1472665484d8SDoug Ambrisko * This function is the entry point for IOCtls from the OS. It calls the 1473665484d8SDoug Ambrisko * appropriate function for processing depending on the command received. 1474665484d8SDoug Ambrisko */ 1475665484d8SDoug Ambrisko static int 14767fc5f329SJohn Baldwin mrsas_ioctl(struct cdev *dev, u_long cmd, caddr_t arg, int flag, 14777fc5f329SJohn Baldwin struct thread *td) 1478665484d8SDoug Ambrisko { 1479665484d8SDoug Ambrisko struct mrsas_softc *sc; 1480665484d8SDoug Ambrisko int ret = 0, i = 0; 14815844115eSKashyap D Desai MRSAS_DRV_PCI_INFORMATION *pciDrvInfo; 1482665484d8SDoug Ambrisko 14835844115eSKashyap D Desai sc = mrsas_get_softc_instance(dev, cmd, arg); 14845844115eSKashyap D Desai if (!sc) 1485536094dcSKashyap D Desai return ENOENT; 14865844115eSKashyap D Desai 1487808517a4SKashyap D Desai if (sc->remove_in_progress || 1488808517a4SKashyap D Desai (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR)) { 1489665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_INFO, 1490808517a4SKashyap D Desai "Either driver remove or shutdown called or " 1491808517a4SKashyap D Desai "HW is in unrecoverable critical error state.\n"); 1492665484d8SDoug Ambrisko return ENOENT; 1493665484d8SDoug Ambrisko } 1494665484d8SDoug Ambrisko mtx_lock_spin(&sc->ioctl_lock); 1495665484d8SDoug Ambrisko if (!sc->reset_in_progress) { 1496665484d8SDoug Ambrisko mtx_unlock_spin(&sc->ioctl_lock); 1497665484d8SDoug Ambrisko goto do_ioctl; 1498665484d8SDoug Ambrisko } 1499665484d8SDoug Ambrisko mtx_unlock_spin(&sc->ioctl_lock); 1500665484d8SDoug Ambrisko while (sc->reset_in_progress) { 1501665484d8SDoug Ambrisko i++; 1502665484d8SDoug Ambrisko if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) { 1503665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_INFO, 1504f0c7594bSKashyap D Desai "[%2d]waiting for OCR to be finished from %s\n", i, __func__); 1505665484d8SDoug Ambrisko } 1506665484d8SDoug Ambrisko pause("mr_ioctl", hz); 1507665484d8SDoug Ambrisko } 1508665484d8SDoug Ambrisko 1509665484d8SDoug Ambrisko do_ioctl: 1510665484d8SDoug Ambrisko switch (cmd) { 1511536094dcSKashyap D Desai case MRSAS_IOC_FIRMWARE_PASS_THROUGH64: 1512536094dcSKashyap D Desai #ifdef COMPAT_FREEBSD32 1513536094dcSKashyap D Desai case MRSAS_IOC_FIRMWARE_PASS_THROUGH32: 1514536094dcSKashyap D Desai #endif 15158e727371SKashyap D Desai /* 15168e727371SKashyap D Desai * Decrement the Ioctl counting Semaphore before getting an 15178e727371SKashyap D Desai * mfi command 15188e727371SKashyap D Desai */ 1519839ee025SKashyap D Desai sema_wait(&sc->ioctl_count_sema); 1520839ee025SKashyap D Desai 1521536094dcSKashyap D Desai ret = mrsas_passthru(sc, (void *)arg, cmd); 1522839ee025SKashyap D Desai 1523839ee025SKashyap D Desai /* Increment the Ioctl counting semaphore value */ 1524839ee025SKashyap D Desai sema_post(&sc->ioctl_count_sema); 1525839ee025SKashyap D Desai 1526665484d8SDoug Ambrisko break; 1527665484d8SDoug Ambrisko case MRSAS_IOC_SCAN_BUS: 1528665484d8SDoug Ambrisko ret = mrsas_bus_scan(sc); 1529665484d8SDoug Ambrisko break; 15305844115eSKashyap D Desai 15315844115eSKashyap D Desai case MRSAS_IOC_GET_PCI_INFO: 15325844115eSKashyap D Desai pciDrvInfo = (MRSAS_DRV_PCI_INFORMATION *) arg; 15335844115eSKashyap D Desai memset(pciDrvInfo, 0, sizeof(MRSAS_DRV_PCI_INFORMATION)); 15345844115eSKashyap D Desai pciDrvInfo->busNumber = pci_get_bus(sc->mrsas_dev); 15355844115eSKashyap D Desai pciDrvInfo->deviceNumber = pci_get_slot(sc->mrsas_dev); 15365844115eSKashyap D Desai pciDrvInfo->functionNumber = pci_get_function(sc->mrsas_dev); 15375844115eSKashyap D Desai pciDrvInfo->domainID = pci_get_domain(sc->mrsas_dev); 15385844115eSKashyap D Desai mrsas_dprint(sc, MRSAS_INFO, "pci bus no: %d," 15395844115eSKashyap D Desai "pci device no: %d, pci function no: %d," 15405844115eSKashyap D Desai "pci domain ID: %d\n", 15415844115eSKashyap D Desai pciDrvInfo->busNumber, pciDrvInfo->deviceNumber, 15425844115eSKashyap D Desai pciDrvInfo->functionNumber, pciDrvInfo->domainID); 15435844115eSKashyap D Desai ret = 0; 15445844115eSKashyap D Desai break; 15455844115eSKashyap D Desai 1546536094dcSKashyap D Desai default: 1547536094dcSKashyap D Desai mrsas_dprint(sc, MRSAS_TRACE, "IOCTL command 0x%lx is not handled\n", cmd); 1548839ee025SKashyap D Desai ret = ENOENT; 1549665484d8SDoug Ambrisko } 1550665484d8SDoug Ambrisko 1551665484d8SDoug Ambrisko return (ret); 1552665484d8SDoug Ambrisko } 1553665484d8SDoug Ambrisko 15548e727371SKashyap D Desai /* 1555da011113SKashyap D Desai * mrsas_poll: poll entry point for mrsas driver fd 1556da011113SKashyap D Desai * 15578e727371SKashyap D Desai * This function is the entry point for poll from the OS. It waits for some AEN 15588e727371SKashyap D Desai * events to be triggered from the controller and notifies back. 1559da011113SKashyap D Desai */ 1560da011113SKashyap D Desai static int 1561da011113SKashyap D Desai mrsas_poll(struct cdev *dev, int poll_events, struct thread *td) 1562da011113SKashyap D Desai { 1563da011113SKashyap D Desai struct mrsas_softc *sc; 1564da011113SKashyap D Desai int revents = 0; 1565da011113SKashyap D Desai 1566da011113SKashyap D Desai sc = dev->si_drv1; 1567da011113SKashyap D Desai 1568da011113SKashyap D Desai if (poll_events & (POLLIN | POLLRDNORM)) { 1569da011113SKashyap D Desai if (sc->mrsas_aen_triggered) { 1570da011113SKashyap D Desai revents |= poll_events & (POLLIN | POLLRDNORM); 1571da011113SKashyap D Desai } 1572da011113SKashyap D Desai } 1573da011113SKashyap D Desai if (revents == 0) { 1574da011113SKashyap D Desai if (poll_events & (POLLIN | POLLRDNORM)) { 1575ecea5be4SKashyap D Desai mtx_lock(&sc->aen_lock); 1576da011113SKashyap D Desai sc->mrsas_poll_waiting = 1; 1577da011113SKashyap D Desai selrecord(td, &sc->mrsas_select); 1578ecea5be4SKashyap D Desai mtx_unlock(&sc->aen_lock); 1579da011113SKashyap D Desai } 1580da011113SKashyap D Desai } 1581da011113SKashyap D Desai return revents; 1582da011113SKashyap D Desai } 1583da011113SKashyap D Desai 15848e727371SKashyap D Desai /* 15858e727371SKashyap D Desai * mrsas_setup_irq: Set up interrupt 1586665484d8SDoug Ambrisko * input: Adapter instance soft state 1587665484d8SDoug Ambrisko * 1588665484d8SDoug Ambrisko * This function sets up interrupts as a bus resource, with flags indicating 1589665484d8SDoug Ambrisko * resource permitting contemporaneous sharing and for resource to activate 1590665484d8SDoug Ambrisko * atomically. 1591665484d8SDoug Ambrisko */ 15928e727371SKashyap D Desai static int 15938e727371SKashyap D Desai mrsas_setup_irq(struct mrsas_softc *sc) 1594665484d8SDoug Ambrisko { 1595d18d1b47SKashyap D Desai if (sc->msix_enable && (mrsas_setup_msix(sc) == SUCCESS)) 1596d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "MSI-x interrupts setup success\n"); 1597665484d8SDoug Ambrisko 1598d18d1b47SKashyap D Desai else { 1599d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "Fall back to legacy interrupt\n"); 1600d18d1b47SKashyap D Desai sc->irq_context[0].sc = sc; 1601d18d1b47SKashyap D Desai sc->irq_context[0].MSIxIndex = 0; 1602d18d1b47SKashyap D Desai sc->irq_id[0] = 0; 1603d18d1b47SKashyap D Desai sc->mrsas_irq[0] = bus_alloc_resource_any(sc->mrsas_dev, 1604d18d1b47SKashyap D Desai SYS_RES_IRQ, &sc->irq_id[0], RF_SHAREABLE | RF_ACTIVE); 1605d18d1b47SKashyap D Desai if (sc->mrsas_irq[0] == NULL) { 1606d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "Cannot allocate legcay" 1607d18d1b47SKashyap D Desai "interrupt\n"); 1608d18d1b47SKashyap D Desai return (FAIL); 1609d18d1b47SKashyap D Desai } 1610d18d1b47SKashyap D Desai if (bus_setup_intr(sc->mrsas_dev, sc->mrsas_irq[0], 1611d18d1b47SKashyap D Desai INTR_MPSAFE | INTR_TYPE_CAM, NULL, mrsas_isr, 1612d18d1b47SKashyap D Desai &sc->irq_context[0], &sc->intr_handle[0])) { 1613d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "Cannot set up legacy" 1614d18d1b47SKashyap D Desai "interrupt\n"); 1615d18d1b47SKashyap D Desai return (FAIL); 1616d18d1b47SKashyap D Desai } 1617d18d1b47SKashyap D Desai } 1618665484d8SDoug Ambrisko return (0); 1619665484d8SDoug Ambrisko } 1620665484d8SDoug Ambrisko 1621665484d8SDoug Ambrisko /* 1622665484d8SDoug Ambrisko * mrsas_isr: ISR entry point 1623665484d8SDoug Ambrisko * input: argument pointer 1624665484d8SDoug Ambrisko * 16258e727371SKashyap D Desai * This function is the interrupt service routine entry point. There are two 16268e727371SKashyap D Desai * types of interrupts, state change interrupt and response interrupt. If an 16278e727371SKashyap D Desai * interrupt is not ours, we just return. 1628665484d8SDoug Ambrisko */ 16298e727371SKashyap D Desai void 16308e727371SKashyap D Desai mrsas_isr(void *arg) 1631665484d8SDoug Ambrisko { 1632d18d1b47SKashyap D Desai struct mrsas_irq_context *irq_context = (struct mrsas_irq_context *)arg; 1633d18d1b47SKashyap D Desai struct mrsas_softc *sc = irq_context->sc; 1634d18d1b47SKashyap D Desai int status = 0; 1635665484d8SDoug Ambrisko 16362f863eb8SKashyap D Desai if (sc->mask_interrupts) 16372f863eb8SKashyap D Desai return; 16382f863eb8SKashyap D Desai 1639d18d1b47SKashyap D Desai if (!sc->msix_vectors) { 1640665484d8SDoug Ambrisko status = mrsas_clear_intr(sc); 1641665484d8SDoug Ambrisko if (!status) 1642665484d8SDoug Ambrisko return; 1643d18d1b47SKashyap D Desai } 1644665484d8SDoug Ambrisko /* If we are resetting, bail */ 1645f5fb2237SKashyap D Desai if (mrsas_test_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags)) { 1646665484d8SDoug Ambrisko printf(" Entered into ISR when OCR is going active. \n"); 1647665484d8SDoug Ambrisko mrsas_clear_intr(sc); 1648665484d8SDoug Ambrisko return; 1649665484d8SDoug Ambrisko } 1650665484d8SDoug Ambrisko /* Process for reply request and clear response interrupt */ 1651d18d1b47SKashyap D Desai if (mrsas_complete_cmd(sc, irq_context->MSIxIndex) != SUCCESS) 1652665484d8SDoug Ambrisko mrsas_clear_intr(sc); 1653665484d8SDoug Ambrisko 1654665484d8SDoug Ambrisko return; 1655665484d8SDoug Ambrisko } 1656665484d8SDoug Ambrisko 1657665484d8SDoug Ambrisko /* 1658665484d8SDoug Ambrisko * mrsas_complete_cmd: Process reply request 1659665484d8SDoug Ambrisko * input: Adapter instance soft state 1660665484d8SDoug Ambrisko * 16618e727371SKashyap D Desai * This function is called from mrsas_isr() to process reply request and clear 16628e727371SKashyap D Desai * response interrupt. Processing of the reply request entails walking 16638e727371SKashyap D Desai * through the reply descriptor array for the command request pended from 16648e727371SKashyap D Desai * Firmware. We look at the Function field to determine the command type and 16658e727371SKashyap D Desai * perform the appropriate action. Before we return, we clear the response 16668e727371SKashyap D Desai * interrupt. 1667665484d8SDoug Ambrisko */ 16684bb0a4f0SKashyap D Desai int 16698e727371SKashyap D Desai mrsas_complete_cmd(struct mrsas_softc *sc, u_int32_t MSIxIndex) 1670665484d8SDoug Ambrisko { 1671665484d8SDoug Ambrisko Mpi2ReplyDescriptorsUnion_t *desc; 1672665484d8SDoug Ambrisko MPI2_SCSI_IO_SUCCESS_REPLY_DESCRIPTOR *reply_desc; 1673665484d8SDoug Ambrisko MRSAS_RAID_SCSI_IO_REQUEST *scsi_io_req; 16742a1d3bcdSKashyap D Desai struct mrsas_mpt_cmd *cmd_mpt, *r1_cmd = NULL; 1675665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd_mfi; 16762a1d3bcdSKashyap D Desai u_int8_t reply_descript_type, *sense; 1677665484d8SDoug Ambrisko u_int16_t smid, num_completed; 1678665484d8SDoug Ambrisko u_int8_t status, extStatus; 1679665484d8SDoug Ambrisko union desc_value desc_val; 1680665484d8SDoug Ambrisko PLD_LOAD_BALANCE_INFO lbinfo; 16812a1d3bcdSKashyap D Desai u_int32_t device_id, data_length; 1682665484d8SDoug Ambrisko int threshold_reply_count = 0; 16838bb601acSKashyap D Desai #if TM_DEBUG 16848bb601acSKashyap D Desai MR_TASK_MANAGE_REQUEST *mr_tm_req; 16858bb601acSKashyap D Desai MPI2_SCSI_TASK_MANAGE_REQUEST *mpi_tm_req; 16868bb601acSKashyap D Desai #endif 1687665484d8SDoug Ambrisko 1688665484d8SDoug Ambrisko /* If we have a hardware error, not need to continue */ 1689665484d8SDoug Ambrisko if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) 1690665484d8SDoug Ambrisko return (DONE); 1691665484d8SDoug Ambrisko 1692665484d8SDoug Ambrisko desc = sc->reply_desc_mem; 1693d18d1b47SKashyap D Desai desc += ((MSIxIndex * sc->reply_alloc_sz) / sizeof(MPI2_REPLY_DESCRIPTORS_UNION)) 1694d18d1b47SKashyap D Desai + sc->last_reply_idx[MSIxIndex]; 1695665484d8SDoug Ambrisko 1696665484d8SDoug Ambrisko reply_desc = (MPI2_SCSI_IO_SUCCESS_REPLY_DESCRIPTOR *) desc; 1697665484d8SDoug Ambrisko 1698665484d8SDoug Ambrisko desc_val.word = desc->Words; 1699665484d8SDoug Ambrisko num_completed = 0; 1700665484d8SDoug Ambrisko 1701665484d8SDoug Ambrisko reply_descript_type = reply_desc->ReplyFlags & MPI2_RPY_DESCRIPT_FLAGS_TYPE_MASK; 1702665484d8SDoug Ambrisko 1703665484d8SDoug Ambrisko /* Find our reply descriptor for the command and process */ 17048e727371SKashyap D Desai while ((desc_val.u.low != 0xFFFFFFFF) && (desc_val.u.high != 0xFFFFFFFF)) { 1705*e34a057cSAlfredo Dal'Ava Junior smid = le16toh(reply_desc->SMID); 1706665484d8SDoug Ambrisko cmd_mpt = sc->mpt_cmd_list[smid - 1]; 1707665484d8SDoug Ambrisko scsi_io_req = (MRSAS_RAID_SCSI_IO_REQUEST *) cmd_mpt->io_request; 1708665484d8SDoug Ambrisko 1709503c4f8dSKashyap D Desai status = scsi_io_req->RaidContext.raid_context.status; 1710503c4f8dSKashyap D Desai extStatus = scsi_io_req->RaidContext.raid_context.exStatus; 17112a1d3bcdSKashyap D Desai sense = cmd_mpt->sense; 17122a1d3bcdSKashyap D Desai data_length = scsi_io_req->DataLength; 1713665484d8SDoug Ambrisko 17148e727371SKashyap D Desai switch (scsi_io_req->Function) { 17158bb601acSKashyap D Desai case MPI2_FUNCTION_SCSI_TASK_MGMT: 17168bb601acSKashyap D Desai #if TM_DEBUG 17178bb601acSKashyap D Desai mr_tm_req = (MR_TASK_MANAGE_REQUEST *) cmd_mpt->io_request; 17188bb601acSKashyap D Desai mpi_tm_req = (MPI2_SCSI_TASK_MANAGE_REQUEST *) 17198bb601acSKashyap D Desai &mr_tm_req->TmRequest; 17208bb601acSKashyap D Desai device_printf(sc->mrsas_dev, "TM completion type 0x%X, " 17218bb601acSKashyap D Desai "TaskMID: 0x%X", mpi_tm_req->TaskType, mpi_tm_req->TaskMID); 17228bb601acSKashyap D Desai #endif 17238bb601acSKashyap D Desai wakeup_one((void *)&sc->ocr_chan); 17248bb601acSKashyap D Desai break; 1725665484d8SDoug Ambrisko case MPI2_FUNCTION_SCSI_IO_REQUEST: /* Fast Path IO. */ 1726665484d8SDoug Ambrisko device_id = cmd_mpt->ccb_ptr->ccb_h.target_id; 1727665484d8SDoug Ambrisko lbinfo = &sc->load_balance_info[device_id]; 17282a1d3bcdSKashyap D Desai /* R1 load balancing for READ */ 1729665484d8SDoug Ambrisko if (cmd_mpt->load_balance == MRSAS_LOAD_BALANCE_FLAG) { 173016dc2814SKashyap D Desai mrsas_atomic_dec(&lbinfo->scsi_pending_cmds[cmd_mpt->pd_r1_lb]); 1731665484d8SDoug Ambrisko cmd_mpt->load_balance &= ~MRSAS_LOAD_BALANCE_FLAG; 1732665484d8SDoug Ambrisko } 17338e727371SKashyap D Desai /* Fall thru and complete IO */ 1734665484d8SDoug Ambrisko case MRSAS_MPI2_FUNCTION_LD_IO_REQUEST: 17352a1d3bcdSKashyap D Desai if (cmd_mpt->r1_alt_dev_handle == MR_DEVHANDLE_INVALID) { 17362a1d3bcdSKashyap D Desai mrsas_map_mpt_cmd_status(cmd_mpt, cmd_mpt->ccb_ptr, status, 1737*e34a057cSAlfredo Dal'Ava Junior extStatus, le32toh(data_length), sense); 1738665484d8SDoug Ambrisko mrsas_cmd_done(sc, cmd_mpt); 17395437c8b8SKashyap D Desai mrsas_atomic_dec(&sc->fw_outstanding); 17402a1d3bcdSKashyap D Desai } else { 17412a1d3bcdSKashyap D Desai /* 17422a1d3bcdSKashyap D Desai * If the peer Raid 1/10 fast path failed, 17432a1d3bcdSKashyap D Desai * mark IO as failed to the scsi layer. 17442a1d3bcdSKashyap D Desai * Overwrite the current status by the failed status 17452a1d3bcdSKashyap D Desai * and make sure that if any command fails, 17462a1d3bcdSKashyap D Desai * driver returns fail status to CAM. 17472a1d3bcdSKashyap D Desai */ 17482a1d3bcdSKashyap D Desai cmd_mpt->cmd_completed = 1; 17492a1d3bcdSKashyap D Desai r1_cmd = cmd_mpt->peer_cmd; 17502a1d3bcdSKashyap D Desai if (r1_cmd->cmd_completed) { 17512a1d3bcdSKashyap D Desai if (r1_cmd->io_request->RaidContext.raid_context.status != MFI_STAT_OK) { 17522a1d3bcdSKashyap D Desai status = r1_cmd->io_request->RaidContext.raid_context.status; 17532a1d3bcdSKashyap D Desai extStatus = r1_cmd->io_request->RaidContext.raid_context.exStatus; 17542a1d3bcdSKashyap D Desai data_length = r1_cmd->io_request->DataLength; 17552a1d3bcdSKashyap D Desai sense = r1_cmd->sense; 17562a1d3bcdSKashyap D Desai } 17572a1d3bcdSKashyap D Desai r1_cmd->ccb_ptr = NULL; 17582a1d3bcdSKashyap D Desai if (r1_cmd->callout_owner) { 17592a1d3bcdSKashyap D Desai callout_stop(&r1_cmd->cm_callout); 17602a1d3bcdSKashyap D Desai r1_cmd->callout_owner = false; 17612a1d3bcdSKashyap D Desai } 17622a1d3bcdSKashyap D Desai mrsas_release_mpt_cmd(r1_cmd); 17635437c8b8SKashyap D Desai mrsas_atomic_dec(&sc->fw_outstanding); 17642a1d3bcdSKashyap D Desai mrsas_map_mpt_cmd_status(cmd_mpt, cmd_mpt->ccb_ptr, status, 1765*e34a057cSAlfredo Dal'Ava Junior extStatus, le32toh(data_length), sense); 17662a1d3bcdSKashyap D Desai mrsas_cmd_done(sc, cmd_mpt); 1767f5fb2237SKashyap D Desai mrsas_atomic_dec(&sc->fw_outstanding); 17685437c8b8SKashyap D Desai } 17695437c8b8SKashyap D Desai } 1770665484d8SDoug Ambrisko break; 1771665484d8SDoug Ambrisko case MRSAS_MPI2_FUNCTION_PASSTHRU_IO_REQUEST: /* MFI command */ 1772665484d8SDoug Ambrisko cmd_mfi = sc->mfi_cmd_list[cmd_mpt->sync_cmd_idx]; 1773731b7561SKashyap D Desai /* 1774731b7561SKashyap D Desai * Make sure NOT TO release the mfi command from the called 1775731b7561SKashyap D Desai * function's context if it is fired with issue_polled call. 1776731b7561SKashyap D Desai * And also make sure that the issue_polled call should only be 1777731b7561SKashyap D Desai * used if INTERRUPT IS DISABLED. 1778731b7561SKashyap D Desai */ 1779*e34a057cSAlfredo Dal'Ava Junior if (cmd_mfi->frame->hdr.flags & htole16(MFI_FRAME_DONT_POST_IN_REPLY_QUEUE)) 1780731b7561SKashyap D Desai mrsas_release_mfi_cmd(cmd_mfi); 1781731b7561SKashyap D Desai else 1782665484d8SDoug Ambrisko mrsas_complete_mptmfi_passthru(sc, cmd_mfi, status); 1783665484d8SDoug Ambrisko break; 1784665484d8SDoug Ambrisko } 1785665484d8SDoug Ambrisko 1786d18d1b47SKashyap D Desai sc->last_reply_idx[MSIxIndex]++; 1787d18d1b47SKashyap D Desai if (sc->last_reply_idx[MSIxIndex] >= sc->reply_q_depth) 1788d18d1b47SKashyap D Desai sc->last_reply_idx[MSIxIndex] = 0; 1789665484d8SDoug Ambrisko 17908e727371SKashyap D Desai desc->Words = ~((uint64_t)0x00); /* set it back to all 17918e727371SKashyap D Desai * 0xFFFFFFFFs */ 1792665484d8SDoug Ambrisko num_completed++; 1793665484d8SDoug Ambrisko threshold_reply_count++; 1794665484d8SDoug Ambrisko 1795665484d8SDoug Ambrisko /* Get the next reply descriptor */ 1796d18d1b47SKashyap D Desai if (!sc->last_reply_idx[MSIxIndex]) { 1797665484d8SDoug Ambrisko desc = sc->reply_desc_mem; 1798d18d1b47SKashyap D Desai desc += ((MSIxIndex * sc->reply_alloc_sz) / sizeof(MPI2_REPLY_DESCRIPTORS_UNION)); 1799d18d1b47SKashyap D Desai } else 1800665484d8SDoug Ambrisko desc++; 1801665484d8SDoug Ambrisko 1802665484d8SDoug Ambrisko reply_desc = (MPI2_SCSI_IO_SUCCESS_REPLY_DESCRIPTOR *) desc; 1803665484d8SDoug Ambrisko desc_val.word = desc->Words; 1804665484d8SDoug Ambrisko 1805665484d8SDoug Ambrisko reply_descript_type = reply_desc->ReplyFlags & MPI2_RPY_DESCRIPT_FLAGS_TYPE_MASK; 1806665484d8SDoug Ambrisko 1807665484d8SDoug Ambrisko if (reply_descript_type == MPI2_RPY_DESCRIPT_FLAGS_UNUSED) 1808665484d8SDoug Ambrisko break; 1809665484d8SDoug Ambrisko 1810665484d8SDoug Ambrisko /* 18118e727371SKashyap D Desai * Write to reply post index after completing threshold reply 18128e727371SKashyap D Desai * count and still there are more replies in reply queue 18138e727371SKashyap D Desai * pending to be completed. 1814665484d8SDoug Ambrisko */ 1815665484d8SDoug Ambrisko if (threshold_reply_count >= THRESHOLD_REPLY_COUNT) { 1816d18d1b47SKashyap D Desai if (sc->msix_enable) { 18177aade8bfSKashyap D Desai if (sc->msix_combined) 1818d18d1b47SKashyap D Desai mrsas_write_reg(sc, sc->msix_reg_offset[MSIxIndex / 8], 1819d18d1b47SKashyap D Desai ((MSIxIndex & 0x7) << 24) | 1820d18d1b47SKashyap D Desai sc->last_reply_idx[MSIxIndex]); 1821d18d1b47SKashyap D Desai else 1822d18d1b47SKashyap D Desai mrsas_write_reg(sc, sc->msix_reg_offset[0], (MSIxIndex << 24) | 1823d18d1b47SKashyap D Desai sc->last_reply_idx[MSIxIndex]); 1824d18d1b47SKashyap D Desai } else 1825d18d1b47SKashyap D Desai mrsas_write_reg(sc, offsetof(mrsas_reg_set, 1826d18d1b47SKashyap D Desai reply_post_host_index), sc->last_reply_idx[0]); 1827d18d1b47SKashyap D Desai 1828665484d8SDoug Ambrisko threshold_reply_count = 0; 1829665484d8SDoug Ambrisko } 1830665484d8SDoug Ambrisko } 1831665484d8SDoug Ambrisko 1832665484d8SDoug Ambrisko /* No match, just return */ 1833665484d8SDoug Ambrisko if (num_completed == 0) 1834665484d8SDoug Ambrisko return (DONE); 1835665484d8SDoug Ambrisko 1836665484d8SDoug Ambrisko /* Clear response interrupt */ 1837d18d1b47SKashyap D Desai if (sc->msix_enable) { 18387aade8bfSKashyap D Desai if (sc->msix_combined) { 1839d18d1b47SKashyap D Desai mrsas_write_reg(sc, sc->msix_reg_offset[MSIxIndex / 8], 1840d18d1b47SKashyap D Desai ((MSIxIndex & 0x7) << 24) | 1841d18d1b47SKashyap D Desai sc->last_reply_idx[MSIxIndex]); 1842d18d1b47SKashyap D Desai } else 1843d18d1b47SKashyap D Desai mrsas_write_reg(sc, sc->msix_reg_offset[0], (MSIxIndex << 24) | 1844d18d1b47SKashyap D Desai sc->last_reply_idx[MSIxIndex]); 1845d18d1b47SKashyap D Desai } else 1846d18d1b47SKashyap D Desai mrsas_write_reg(sc, offsetof(mrsas_reg_set, 1847d18d1b47SKashyap D Desai reply_post_host_index), sc->last_reply_idx[0]); 1848665484d8SDoug Ambrisko 1849665484d8SDoug Ambrisko return (0); 1850665484d8SDoug Ambrisko } 1851665484d8SDoug Ambrisko 1852665484d8SDoug Ambrisko /* 1853665484d8SDoug Ambrisko * mrsas_map_mpt_cmd_status: Allocate DMAable memory. 1854665484d8SDoug Ambrisko * input: Adapter instance soft state 1855665484d8SDoug Ambrisko * 1856665484d8SDoug Ambrisko * This function is called from mrsas_complete_cmd(), for LD IO and FastPath IO. 18578e727371SKashyap D Desai * It checks the command status and maps the appropriate CAM status for the 18588e727371SKashyap D Desai * CCB. 1859665484d8SDoug Ambrisko */ 18608e727371SKashyap D Desai void 18612a1d3bcdSKashyap D Desai mrsas_map_mpt_cmd_status(struct mrsas_mpt_cmd *cmd, union ccb *ccb_ptr, u_int8_t status, 18622a1d3bcdSKashyap D Desai u_int8_t extStatus, u_int32_t data_length, u_int8_t *sense) 1863665484d8SDoug Ambrisko { 1864665484d8SDoug Ambrisko struct mrsas_softc *sc = cmd->sc; 1865665484d8SDoug Ambrisko u_int8_t *sense_data; 1866665484d8SDoug Ambrisko 1867665484d8SDoug Ambrisko switch (status) { 1868665484d8SDoug Ambrisko case MFI_STAT_OK: 18692a1d3bcdSKashyap D Desai ccb_ptr->ccb_h.status = CAM_REQ_CMP; 1870665484d8SDoug Ambrisko break; 1871665484d8SDoug Ambrisko case MFI_STAT_SCSI_IO_FAILED: 1872665484d8SDoug Ambrisko case MFI_STAT_SCSI_DONE_WITH_ERROR: 18732a1d3bcdSKashyap D Desai ccb_ptr->ccb_h.status = CAM_SCSI_STATUS_ERROR; 18742a1d3bcdSKashyap D Desai sense_data = (u_int8_t *)&ccb_ptr->csio.sense_data; 1875665484d8SDoug Ambrisko if (sense_data) { 1876665484d8SDoug Ambrisko /* For now just copy 18 bytes back */ 18772a1d3bcdSKashyap D Desai memcpy(sense_data, sense, 18); 18782a1d3bcdSKashyap D Desai ccb_ptr->csio.sense_len = 18; 18792a1d3bcdSKashyap D Desai ccb_ptr->ccb_h.status |= CAM_AUTOSNS_VALID; 1880665484d8SDoug Ambrisko } 1881665484d8SDoug Ambrisko break; 1882665484d8SDoug Ambrisko case MFI_STAT_LD_OFFLINE: 1883665484d8SDoug Ambrisko case MFI_STAT_DEVICE_NOT_FOUND: 18842a1d3bcdSKashyap D Desai if (ccb_ptr->ccb_h.target_lun) 18852a1d3bcdSKashyap D Desai ccb_ptr->ccb_h.status |= CAM_LUN_INVALID; 1886665484d8SDoug Ambrisko else 18872a1d3bcdSKashyap D Desai ccb_ptr->ccb_h.status |= CAM_DEV_NOT_THERE; 1888665484d8SDoug Ambrisko break; 1889665484d8SDoug Ambrisko case MFI_STAT_CONFIG_SEQ_MISMATCH: 18902a1d3bcdSKashyap D Desai ccb_ptr->ccb_h.status |= CAM_REQUEUE_REQ; 1891665484d8SDoug Ambrisko break; 1892665484d8SDoug Ambrisko default: 1893665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "FW cmd complete status %x\n", status); 18942a1d3bcdSKashyap D Desai ccb_ptr->ccb_h.status = CAM_REQ_CMP_ERR; 18952a1d3bcdSKashyap D Desai ccb_ptr->csio.scsi_status = status; 1896665484d8SDoug Ambrisko } 1897665484d8SDoug Ambrisko return; 1898665484d8SDoug Ambrisko } 1899665484d8SDoug Ambrisko 1900665484d8SDoug Ambrisko /* 19018e727371SKashyap D Desai * mrsas_alloc_mem: Allocate DMAable memory 1902665484d8SDoug Ambrisko * input: Adapter instance soft state 1903665484d8SDoug Ambrisko * 19048e727371SKashyap D Desai * This function creates the parent DMA tag and allocates DMAable memory. DMA 19058e727371SKashyap D Desai * tag describes constraints of DMA mapping. Memory allocated is mapped into 19068e727371SKashyap D Desai * Kernel virtual address. Callback argument is physical memory address. 1907665484d8SDoug Ambrisko */ 19088e727371SKashyap D Desai static int 19098e727371SKashyap D Desai mrsas_alloc_mem(struct mrsas_softc *sc) 1910665484d8SDoug Ambrisko { 19114ad83576SKashyap D Desai u_int32_t verbuf_size, io_req_size, reply_desc_size, sense_size, chain_frame_size, 191279b4460bSKashyap D Desai evt_detail_size, count, pd_info_size; 1913665484d8SDoug Ambrisko 1914665484d8SDoug Ambrisko /* 1915665484d8SDoug Ambrisko * Allocate parent DMA tag 1916665484d8SDoug Ambrisko */ 1917665484d8SDoug Ambrisko if (bus_dma_tag_create(NULL, /* parent */ 1918665484d8SDoug Ambrisko 1, /* alignment */ 1919665484d8SDoug Ambrisko 0, /* boundary */ 1920665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, /* lowaddr */ 1921665484d8SDoug Ambrisko BUS_SPACE_MAXADDR, /* highaddr */ 1922665484d8SDoug Ambrisko NULL, NULL, /* filter, filterarg */ 1923cd853791SKonstantin Belousov maxphys, /* maxsize */ 19243a3fc6cbSKashyap D Desai sc->max_num_sge, /* nsegments */ 1925cd853791SKonstantin Belousov maxphys, /* maxsegsize */ 1926665484d8SDoug Ambrisko 0, /* flags */ 1927665484d8SDoug Ambrisko NULL, NULL, /* lockfunc, lockarg */ 1928665484d8SDoug Ambrisko &sc->mrsas_parent_tag /* tag */ 1929665484d8SDoug Ambrisko )) { 1930665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate parent DMA tag\n"); 1931665484d8SDoug Ambrisko return (ENOMEM); 1932665484d8SDoug Ambrisko } 1933665484d8SDoug Ambrisko /* 1934665484d8SDoug Ambrisko * Allocate for version buffer 1935665484d8SDoug Ambrisko */ 1936665484d8SDoug Ambrisko verbuf_size = MRSAS_MAX_NAME_LENGTH * (sizeof(bus_addr_t)); 19378e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 19388e727371SKashyap D Desai 1, 0, 19398e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 19408e727371SKashyap D Desai BUS_SPACE_MAXADDR, 19418e727371SKashyap D Desai NULL, NULL, 19428e727371SKashyap D Desai verbuf_size, 19438e727371SKashyap D Desai 1, 19448e727371SKashyap D Desai verbuf_size, 19458e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 19468e727371SKashyap D Desai NULL, NULL, 1947665484d8SDoug Ambrisko &sc->verbuf_tag)) { 1948665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate verbuf DMA tag\n"); 1949665484d8SDoug Ambrisko return (ENOMEM); 1950665484d8SDoug Ambrisko } 1951665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->verbuf_tag, (void **)&sc->verbuf_mem, 1952665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->verbuf_dmamap)) { 1953665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate verbuf memory\n"); 1954665484d8SDoug Ambrisko return (ENOMEM); 1955665484d8SDoug Ambrisko } 1956665484d8SDoug Ambrisko bzero(sc->verbuf_mem, verbuf_size); 1957665484d8SDoug Ambrisko if (bus_dmamap_load(sc->verbuf_tag, sc->verbuf_dmamap, sc->verbuf_mem, 19588e727371SKashyap D Desai verbuf_size, mrsas_addr_cb, &sc->verbuf_phys_addr, 19598e727371SKashyap D Desai BUS_DMA_NOWAIT)) { 1960665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load verbuf DMA map\n"); 1961665484d8SDoug Ambrisko return (ENOMEM); 1962665484d8SDoug Ambrisko } 1963665484d8SDoug Ambrisko /* 1964665484d8SDoug Ambrisko * Allocate IO Request Frames 1965665484d8SDoug Ambrisko */ 1966665484d8SDoug Ambrisko io_req_size = sc->io_frames_alloc_sz; 19678e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 19688e727371SKashyap D Desai 16, 0, 19698e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 19708e727371SKashyap D Desai BUS_SPACE_MAXADDR, 19718e727371SKashyap D Desai NULL, NULL, 19728e727371SKashyap D Desai io_req_size, 19738e727371SKashyap D Desai 1, 19748e727371SKashyap D Desai io_req_size, 19758e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 19768e727371SKashyap D Desai NULL, NULL, 1977665484d8SDoug Ambrisko &sc->io_request_tag)) { 1978665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create IO request tag\n"); 1979665484d8SDoug Ambrisko return (ENOMEM); 1980665484d8SDoug Ambrisko } 1981665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->io_request_tag, (void **)&sc->io_request_mem, 1982665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->io_request_dmamap)) { 1983665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc IO request memory\n"); 1984665484d8SDoug Ambrisko return (ENOMEM); 1985665484d8SDoug Ambrisko } 1986665484d8SDoug Ambrisko bzero(sc->io_request_mem, io_req_size); 1987665484d8SDoug Ambrisko if (bus_dmamap_load(sc->io_request_tag, sc->io_request_dmamap, 1988665484d8SDoug Ambrisko sc->io_request_mem, io_req_size, mrsas_addr_cb, 1989665484d8SDoug Ambrisko &sc->io_request_phys_addr, BUS_DMA_NOWAIT)) { 1990665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load IO request memory\n"); 1991665484d8SDoug Ambrisko return (ENOMEM); 1992665484d8SDoug Ambrisko } 1993665484d8SDoug Ambrisko /* 1994665484d8SDoug Ambrisko * Allocate Chain Frames 1995665484d8SDoug Ambrisko */ 1996665484d8SDoug Ambrisko chain_frame_size = sc->chain_frames_alloc_sz; 19978e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 19988e727371SKashyap D Desai 4, 0, 19998e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 20008e727371SKashyap D Desai BUS_SPACE_MAXADDR, 20018e727371SKashyap D Desai NULL, NULL, 20028e727371SKashyap D Desai chain_frame_size, 20038e727371SKashyap D Desai 1, 20048e727371SKashyap D Desai chain_frame_size, 20058e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 20068e727371SKashyap D Desai NULL, NULL, 2007665484d8SDoug Ambrisko &sc->chain_frame_tag)) { 2008665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create chain frame tag\n"); 2009665484d8SDoug Ambrisko return (ENOMEM); 2010665484d8SDoug Ambrisko } 2011665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->chain_frame_tag, (void **)&sc->chain_frame_mem, 2012665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->chain_frame_dmamap)) { 2013665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc chain frame memory\n"); 2014665484d8SDoug Ambrisko return (ENOMEM); 2015665484d8SDoug Ambrisko } 2016665484d8SDoug Ambrisko bzero(sc->chain_frame_mem, chain_frame_size); 2017665484d8SDoug Ambrisko if (bus_dmamap_load(sc->chain_frame_tag, sc->chain_frame_dmamap, 2018665484d8SDoug Ambrisko sc->chain_frame_mem, chain_frame_size, mrsas_addr_cb, 2019665484d8SDoug Ambrisko &sc->chain_frame_phys_addr, BUS_DMA_NOWAIT)) { 2020665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load chain frame memory\n"); 2021665484d8SDoug Ambrisko return (ENOMEM); 2022665484d8SDoug Ambrisko } 2023d18d1b47SKashyap D Desai count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 2024665484d8SDoug Ambrisko /* 2025665484d8SDoug Ambrisko * Allocate Reply Descriptor Array 2026665484d8SDoug Ambrisko */ 2027d18d1b47SKashyap D Desai reply_desc_size = sc->reply_alloc_sz * count; 20288e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 20298e727371SKashyap D Desai 16, 0, 20308e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 20318e727371SKashyap D Desai BUS_SPACE_MAXADDR, 20328e727371SKashyap D Desai NULL, NULL, 20338e727371SKashyap D Desai reply_desc_size, 20348e727371SKashyap D Desai 1, 20358e727371SKashyap D Desai reply_desc_size, 20368e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 20378e727371SKashyap D Desai NULL, NULL, 2038665484d8SDoug Ambrisko &sc->reply_desc_tag)) { 2039665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create reply descriptor tag\n"); 2040665484d8SDoug Ambrisko return (ENOMEM); 2041665484d8SDoug Ambrisko } 2042665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->reply_desc_tag, (void **)&sc->reply_desc_mem, 2043665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->reply_desc_dmamap)) { 2044665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc reply descriptor memory\n"); 2045665484d8SDoug Ambrisko return (ENOMEM); 2046665484d8SDoug Ambrisko } 2047665484d8SDoug Ambrisko if (bus_dmamap_load(sc->reply_desc_tag, sc->reply_desc_dmamap, 2048665484d8SDoug Ambrisko sc->reply_desc_mem, reply_desc_size, mrsas_addr_cb, 2049665484d8SDoug Ambrisko &sc->reply_desc_phys_addr, BUS_DMA_NOWAIT)) { 2050665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load reply descriptor memory\n"); 2051665484d8SDoug Ambrisko return (ENOMEM); 2052665484d8SDoug Ambrisko } 2053665484d8SDoug Ambrisko /* 2054665484d8SDoug Ambrisko * Allocate Sense Buffer Array. Keep in lower 4GB 2055665484d8SDoug Ambrisko */ 2056665484d8SDoug Ambrisko sense_size = sc->max_fw_cmds * MRSAS_SENSE_LEN; 20578e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 20588e727371SKashyap D Desai 64, 0, 20598e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 20608e727371SKashyap D Desai BUS_SPACE_MAXADDR, 20618e727371SKashyap D Desai NULL, NULL, 20628e727371SKashyap D Desai sense_size, 20638e727371SKashyap D Desai 1, 20648e727371SKashyap D Desai sense_size, 20658e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 20668e727371SKashyap D Desai NULL, NULL, 2067665484d8SDoug Ambrisko &sc->sense_tag)) { 2068665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate sense buf tag\n"); 2069665484d8SDoug Ambrisko return (ENOMEM); 2070665484d8SDoug Ambrisko } 2071665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->sense_tag, (void **)&sc->sense_mem, 2072665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->sense_dmamap)) { 2073665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate sense buf memory\n"); 2074665484d8SDoug Ambrisko return (ENOMEM); 2075665484d8SDoug Ambrisko } 2076665484d8SDoug Ambrisko if (bus_dmamap_load(sc->sense_tag, sc->sense_dmamap, 2077665484d8SDoug Ambrisko sc->sense_mem, sense_size, mrsas_addr_cb, &sc->sense_phys_addr, 2078665484d8SDoug Ambrisko BUS_DMA_NOWAIT)) { 2079665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load sense buf memory\n"); 2080665484d8SDoug Ambrisko return (ENOMEM); 2081665484d8SDoug Ambrisko } 20822a1d3bcdSKashyap D Desai 2083665484d8SDoug Ambrisko /* 2084665484d8SDoug Ambrisko * Allocate for Event detail structure 2085665484d8SDoug Ambrisko */ 2086665484d8SDoug Ambrisko evt_detail_size = sizeof(struct mrsas_evt_detail); 20878e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 20888e727371SKashyap D Desai 1, 0, 20898e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 20908e727371SKashyap D Desai BUS_SPACE_MAXADDR, 20918e727371SKashyap D Desai NULL, NULL, 20928e727371SKashyap D Desai evt_detail_size, 20938e727371SKashyap D Desai 1, 20948e727371SKashyap D Desai evt_detail_size, 20958e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 20968e727371SKashyap D Desai NULL, NULL, 2097665484d8SDoug Ambrisko &sc->evt_detail_tag)) { 2098665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create Event detail tag\n"); 2099665484d8SDoug Ambrisko return (ENOMEM); 2100665484d8SDoug Ambrisko } 2101665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->evt_detail_tag, (void **)&sc->evt_detail_mem, 2102665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->evt_detail_dmamap)) { 2103665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc Event detail buffer memory\n"); 2104665484d8SDoug Ambrisko return (ENOMEM); 2105665484d8SDoug Ambrisko } 2106665484d8SDoug Ambrisko bzero(sc->evt_detail_mem, evt_detail_size); 2107665484d8SDoug Ambrisko if (bus_dmamap_load(sc->evt_detail_tag, sc->evt_detail_dmamap, 2108665484d8SDoug Ambrisko sc->evt_detail_mem, evt_detail_size, mrsas_addr_cb, 2109665484d8SDoug Ambrisko &sc->evt_detail_phys_addr, BUS_DMA_NOWAIT)) { 2110665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load Event detail buffer memory\n"); 2111665484d8SDoug Ambrisko return (ENOMEM); 2112665484d8SDoug Ambrisko } 211379b4460bSKashyap D Desai 211479b4460bSKashyap D Desai /* 211579b4460bSKashyap D Desai * Allocate for PD INFO structure 211679b4460bSKashyap D Desai */ 211779b4460bSKashyap D Desai pd_info_size = sizeof(struct mrsas_pd_info); 211879b4460bSKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 211979b4460bSKashyap D Desai 1, 0, 212079b4460bSKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 212179b4460bSKashyap D Desai BUS_SPACE_MAXADDR, 212279b4460bSKashyap D Desai NULL, NULL, 212379b4460bSKashyap D Desai pd_info_size, 212479b4460bSKashyap D Desai 1, 212579b4460bSKashyap D Desai pd_info_size, 212679b4460bSKashyap D Desai BUS_DMA_ALLOCNOW, 212779b4460bSKashyap D Desai NULL, NULL, 212879b4460bSKashyap D Desai &sc->pd_info_tag)) { 212979b4460bSKashyap D Desai device_printf(sc->mrsas_dev, "Cannot create PD INFO tag\n"); 213079b4460bSKashyap D Desai return (ENOMEM); 213179b4460bSKashyap D Desai } 213279b4460bSKashyap D Desai if (bus_dmamem_alloc(sc->pd_info_tag, (void **)&sc->pd_info_mem, 213379b4460bSKashyap D Desai BUS_DMA_NOWAIT, &sc->pd_info_dmamap)) { 213479b4460bSKashyap D Desai device_printf(sc->mrsas_dev, "Cannot alloc PD INFO buffer memory\n"); 213579b4460bSKashyap D Desai return (ENOMEM); 213679b4460bSKashyap D Desai } 213779b4460bSKashyap D Desai bzero(sc->pd_info_mem, pd_info_size); 213879b4460bSKashyap D Desai if (bus_dmamap_load(sc->pd_info_tag, sc->pd_info_dmamap, 213979b4460bSKashyap D Desai sc->pd_info_mem, pd_info_size, mrsas_addr_cb, 214079b4460bSKashyap D Desai &sc->pd_info_phys_addr, BUS_DMA_NOWAIT)) { 214179b4460bSKashyap D Desai device_printf(sc->mrsas_dev, "Cannot load PD INFO buffer memory\n"); 214279b4460bSKashyap D Desai return (ENOMEM); 214379b4460bSKashyap D Desai } 214479b4460bSKashyap D Desai 2145665484d8SDoug Ambrisko /* 2146665484d8SDoug Ambrisko * Create a dma tag for data buffers; size will be the maximum 2147665484d8SDoug Ambrisko * possible I/O size (280kB). 2148665484d8SDoug Ambrisko */ 21498e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 21508e727371SKashyap D Desai 1, 21518e727371SKashyap D Desai 0, 21528e727371SKashyap D Desai BUS_SPACE_MAXADDR, 21538e727371SKashyap D Desai BUS_SPACE_MAXADDR, 21548e727371SKashyap D Desai NULL, NULL, 2155cd853791SKonstantin Belousov maxphys, 21563a3fc6cbSKashyap D Desai sc->max_num_sge, /* nsegments */ 2157cd853791SKonstantin Belousov maxphys, 21588e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 21598e727371SKashyap D Desai busdma_lock_mutex, 21608e727371SKashyap D Desai &sc->io_lock, 2161665484d8SDoug Ambrisko &sc->data_tag)) { 2162665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot create data dma tag\n"); 2163665484d8SDoug Ambrisko return (ENOMEM); 2164665484d8SDoug Ambrisko } 2165665484d8SDoug Ambrisko return (0); 2166665484d8SDoug Ambrisko } 2167665484d8SDoug Ambrisko 2168665484d8SDoug Ambrisko /* 2169665484d8SDoug Ambrisko * mrsas_addr_cb: Callback function of bus_dmamap_load() 21708e727371SKashyap D Desai * input: callback argument, machine dependent type 21718e727371SKashyap D Desai * that describes DMA segments, number of segments, error code 2172665484d8SDoug Ambrisko * 21738e727371SKashyap D Desai * This function is for the driver to receive mapping information resultant of 21748e727371SKashyap D Desai * the bus_dmamap_load(). The information is actually not being used, but the 21758e727371SKashyap D Desai * address is saved anyway. 2176665484d8SDoug Ambrisko */ 2177665484d8SDoug Ambrisko void 2178665484d8SDoug Ambrisko mrsas_addr_cb(void *arg, bus_dma_segment_t *segs, int nsegs, int error) 2179665484d8SDoug Ambrisko { 2180665484d8SDoug Ambrisko bus_addr_t *addr; 2181665484d8SDoug Ambrisko 2182665484d8SDoug Ambrisko addr = arg; 2183665484d8SDoug Ambrisko *addr = segs[0].ds_addr; 2184665484d8SDoug Ambrisko } 2185665484d8SDoug Ambrisko 2186665484d8SDoug Ambrisko /* 2187665484d8SDoug Ambrisko * mrsas_setup_raidmap: Set up RAID map. 2188665484d8SDoug Ambrisko * input: Adapter instance soft state 2189665484d8SDoug Ambrisko * 2190665484d8SDoug Ambrisko * Allocate DMA memory for the RAID maps and perform setup. 2191665484d8SDoug Ambrisko */ 21928e727371SKashyap D Desai static int 21938e727371SKashyap D Desai mrsas_setup_raidmap(struct mrsas_softc *sc) 2194665484d8SDoug Ambrisko { 21954799d485SKashyap D Desai int i; 21964799d485SKashyap D Desai 21974799d485SKashyap D Desai for (i = 0; i < 2; i++) { 21984799d485SKashyap D Desai sc->ld_drv_map[i] = 21994799d485SKashyap D Desai (void *)malloc(sc->drv_map_sz, M_MRSAS, M_NOWAIT); 22004799d485SKashyap D Desai /* Do Error handling */ 22014799d485SKashyap D Desai if (!sc->ld_drv_map[i]) { 22024799d485SKashyap D Desai device_printf(sc->mrsas_dev, "Could not allocate memory for local map"); 22034799d485SKashyap D Desai 22044799d485SKashyap D Desai if (i == 1) 22054799d485SKashyap D Desai free(sc->ld_drv_map[0], M_MRSAS); 22068e727371SKashyap D Desai /* ABORT driver initialization */ 22074799d485SKashyap D Desai goto ABORT; 22084799d485SKashyap D Desai } 22094799d485SKashyap D Desai } 22104799d485SKashyap D Desai 22118e727371SKashyap D Desai for (int i = 0; i < 2; i++) { 22128e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 22138e727371SKashyap D Desai 4, 0, 22148e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 22158e727371SKashyap D Desai BUS_SPACE_MAXADDR, 22168e727371SKashyap D Desai NULL, NULL, 22178e727371SKashyap D Desai sc->max_map_sz, 22188e727371SKashyap D Desai 1, 22198e727371SKashyap D Desai sc->max_map_sz, 22208e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 22218e727371SKashyap D Desai NULL, NULL, 2222665484d8SDoug Ambrisko &sc->raidmap_tag[i])) { 22234799d485SKashyap D Desai device_printf(sc->mrsas_dev, 22244799d485SKashyap D Desai "Cannot allocate raid map tag.\n"); 2225665484d8SDoug Ambrisko return (ENOMEM); 2226665484d8SDoug Ambrisko } 22274799d485SKashyap D Desai if (bus_dmamem_alloc(sc->raidmap_tag[i], 22284799d485SKashyap D Desai (void **)&sc->raidmap_mem[i], 2229665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->raidmap_dmamap[i])) { 22304799d485SKashyap D Desai device_printf(sc->mrsas_dev, 22314799d485SKashyap D Desai "Cannot allocate raidmap memory.\n"); 2232665484d8SDoug Ambrisko return (ENOMEM); 2233665484d8SDoug Ambrisko } 22344799d485SKashyap D Desai bzero(sc->raidmap_mem[i], sc->max_map_sz); 22354799d485SKashyap D Desai 2236665484d8SDoug Ambrisko if (bus_dmamap_load(sc->raidmap_tag[i], sc->raidmap_dmamap[i], 22374799d485SKashyap D Desai sc->raidmap_mem[i], sc->max_map_sz, 22384799d485SKashyap D Desai mrsas_addr_cb, &sc->raidmap_phys_addr[i], 2239665484d8SDoug Ambrisko BUS_DMA_NOWAIT)) { 2240665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load raidmap memory.\n"); 2241665484d8SDoug Ambrisko return (ENOMEM); 2242665484d8SDoug Ambrisko } 2243665484d8SDoug Ambrisko if (!sc->raidmap_mem[i]) { 22444799d485SKashyap D Desai device_printf(sc->mrsas_dev, 22454799d485SKashyap D Desai "Cannot allocate memory for raid map.\n"); 2246665484d8SDoug Ambrisko return (ENOMEM); 2247665484d8SDoug Ambrisko } 2248665484d8SDoug Ambrisko } 2249665484d8SDoug Ambrisko 2250665484d8SDoug Ambrisko if (!mrsas_get_map_info(sc)) 2251665484d8SDoug Ambrisko mrsas_sync_map_info(sc); 2252665484d8SDoug Ambrisko 2253665484d8SDoug Ambrisko return (0); 22544799d485SKashyap D Desai 22554799d485SKashyap D Desai ABORT: 22564799d485SKashyap D Desai return (1); 2257665484d8SDoug Ambrisko } 2258665484d8SDoug Ambrisko 2259a688fcd0SKashyap D Desai /** 2260a688fcd0SKashyap D Desai * megasas_setup_jbod_map - setup jbod map for FP seq_number. 2261a688fcd0SKashyap D Desai * @sc: Adapter soft state 2262a688fcd0SKashyap D Desai * 2263a688fcd0SKashyap D Desai * Return 0 on success. 2264a688fcd0SKashyap D Desai */ 2265a688fcd0SKashyap D Desai void 2266a688fcd0SKashyap D Desai megasas_setup_jbod_map(struct mrsas_softc *sc) 2267a688fcd0SKashyap D Desai { 2268a688fcd0SKashyap D Desai int i; 2269a688fcd0SKashyap D Desai uint32_t pd_seq_map_sz; 2270a688fcd0SKashyap D Desai 2271a688fcd0SKashyap D Desai pd_seq_map_sz = sizeof(struct MR_PD_CFG_SEQ_NUM_SYNC) + 2272a688fcd0SKashyap D Desai (sizeof(struct MR_PD_CFG_SEQ) * (MAX_PHYSICAL_DEVICES - 1)); 2273a688fcd0SKashyap D Desai 2274a688fcd0SKashyap D Desai if (!sc->ctrl_info->adapterOperations3.useSeqNumJbodFP) { 2275a688fcd0SKashyap D Desai sc->use_seqnum_jbod_fp = 0; 2276a688fcd0SKashyap D Desai return; 2277a688fcd0SKashyap D Desai } 2278a688fcd0SKashyap D Desai if (sc->jbodmap_mem[0]) 2279a688fcd0SKashyap D Desai goto skip_alloc; 2280a688fcd0SKashyap D Desai 2281a688fcd0SKashyap D Desai for (i = 0; i < 2; i++) { 2282a688fcd0SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 2283a688fcd0SKashyap D Desai 4, 0, 2284a688fcd0SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 2285a688fcd0SKashyap D Desai BUS_SPACE_MAXADDR, 2286a688fcd0SKashyap D Desai NULL, NULL, 2287a688fcd0SKashyap D Desai pd_seq_map_sz, 2288a688fcd0SKashyap D Desai 1, 2289a688fcd0SKashyap D Desai pd_seq_map_sz, 2290a688fcd0SKashyap D Desai BUS_DMA_ALLOCNOW, 2291a688fcd0SKashyap D Desai NULL, NULL, 2292a688fcd0SKashyap D Desai &sc->jbodmap_tag[i])) { 2293a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, 2294a688fcd0SKashyap D Desai "Cannot allocate jbod map tag.\n"); 2295a688fcd0SKashyap D Desai return; 2296a688fcd0SKashyap D Desai } 2297a688fcd0SKashyap D Desai if (bus_dmamem_alloc(sc->jbodmap_tag[i], 2298a688fcd0SKashyap D Desai (void **)&sc->jbodmap_mem[i], 2299a688fcd0SKashyap D Desai BUS_DMA_NOWAIT, &sc->jbodmap_dmamap[i])) { 2300a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, 2301a688fcd0SKashyap D Desai "Cannot allocate jbod map memory.\n"); 2302a688fcd0SKashyap D Desai return; 2303a688fcd0SKashyap D Desai } 2304a688fcd0SKashyap D Desai bzero(sc->jbodmap_mem[i], pd_seq_map_sz); 2305a688fcd0SKashyap D Desai 2306a688fcd0SKashyap D Desai if (bus_dmamap_load(sc->jbodmap_tag[i], sc->jbodmap_dmamap[i], 2307a688fcd0SKashyap D Desai sc->jbodmap_mem[i], pd_seq_map_sz, 2308a688fcd0SKashyap D Desai mrsas_addr_cb, &sc->jbodmap_phys_addr[i], 2309a688fcd0SKashyap D Desai BUS_DMA_NOWAIT)) { 2310a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, "Cannot load jbod map memory.\n"); 2311a688fcd0SKashyap D Desai return; 2312a688fcd0SKashyap D Desai } 2313a688fcd0SKashyap D Desai if (!sc->jbodmap_mem[i]) { 2314a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, 2315a688fcd0SKashyap D Desai "Cannot allocate memory for jbod map.\n"); 2316a688fcd0SKashyap D Desai sc->use_seqnum_jbod_fp = 0; 2317a688fcd0SKashyap D Desai return; 2318a688fcd0SKashyap D Desai } 2319a688fcd0SKashyap D Desai } 2320a688fcd0SKashyap D Desai 2321a688fcd0SKashyap D Desai skip_alloc: 2322a688fcd0SKashyap D Desai if (!megasas_sync_pd_seq_num(sc, false) && 2323a688fcd0SKashyap D Desai !megasas_sync_pd_seq_num(sc, true)) 2324a688fcd0SKashyap D Desai sc->use_seqnum_jbod_fp = 1; 2325a688fcd0SKashyap D Desai else 2326a688fcd0SKashyap D Desai sc->use_seqnum_jbod_fp = 0; 2327a688fcd0SKashyap D Desai 2328a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, "Jbod map is supported\n"); 2329a688fcd0SKashyap D Desai } 2330a688fcd0SKashyap D Desai 23318e727371SKashyap D Desai /* 2332665484d8SDoug Ambrisko * mrsas_init_fw: Initialize Firmware 2333665484d8SDoug Ambrisko * input: Adapter soft state 2334665484d8SDoug Ambrisko * 23358e727371SKashyap D Desai * Calls transition_to_ready() to make sure Firmware is in operational state and 23368e727371SKashyap D Desai * calls mrsas_init_adapter() to send IOC_INIT command to Firmware. It 23378e727371SKashyap D Desai * issues internal commands to get the controller info after the IOC_INIT 23388e727371SKashyap D Desai * command response is received by Firmware. Note: code relating to 23398e727371SKashyap D Desai * get_pdlist, get_ld_list and max_sectors are currently not being used, it 23408e727371SKashyap D Desai * is left here as placeholder. 2341665484d8SDoug Ambrisko */ 23428e727371SKashyap D Desai static int 23438e727371SKashyap D Desai mrsas_init_fw(struct mrsas_softc *sc) 2344665484d8SDoug Ambrisko { 2345d18d1b47SKashyap D Desai 2346d18d1b47SKashyap D Desai int ret, loop, ocr = 0; 2347665484d8SDoug Ambrisko u_int32_t max_sectors_1; 2348665484d8SDoug Ambrisko u_int32_t max_sectors_2; 2349665484d8SDoug Ambrisko u_int32_t tmp_sectors; 23503d273176SKashyap D Desai u_int32_t scratch_pad_2, scratch_pad_3, scratch_pad_4; 2351d18d1b47SKashyap D Desai int msix_enable = 0; 2352d18d1b47SKashyap D Desai int fw_msix_count = 0; 2353821df4b9SKashyap D Desai int i, j; 2354665484d8SDoug Ambrisko 2355665484d8SDoug Ambrisko /* Make sure Firmware is ready */ 2356665484d8SDoug Ambrisko ret = mrsas_transition_to_ready(sc, ocr); 2357665484d8SDoug Ambrisko if (ret != SUCCESS) { 2358665484d8SDoug Ambrisko return (ret); 2359665484d8SDoug Ambrisko } 23602909aab4SKashyap D Desai if (sc->is_ventura || sc->is_aero) { 2361e315cf4dSKashyap D Desai scratch_pad_3 = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, outbound_scratch_pad_3)); 23624ad83576SKashyap D Desai #if VD_EXT_DEBUG 23634ad83576SKashyap D Desai device_printf(sc->mrsas_dev, "scratch_pad_3 0x%x\n", scratch_pad_3); 23644ad83576SKashyap D Desai #endif 23654ad83576SKashyap D Desai sc->maxRaidMapSize = ((scratch_pad_3 >> 23664ad83576SKashyap D Desai MR_MAX_RAID_MAP_SIZE_OFFSET_SHIFT) & 23674ad83576SKashyap D Desai MR_MAX_RAID_MAP_SIZE_MASK); 23684ad83576SKashyap D Desai } 2369d18d1b47SKashyap D Desai /* MSI-x index 0- reply post host index register */ 2370d18d1b47SKashyap D Desai sc->msix_reg_offset[0] = MPI2_REPLY_POST_HOST_INDEX_OFFSET; 2371d18d1b47SKashyap D Desai /* Check if MSI-X is supported while in ready state */ 2372e315cf4dSKashyap D Desai msix_enable = (mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, outbound_scratch_pad)) & 0x4000000) >> 0x1a; 2373d18d1b47SKashyap D Desai 2374d18d1b47SKashyap D Desai if (msix_enable) { 2375e315cf4dSKashyap D Desai scratch_pad_2 = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, 2376d18d1b47SKashyap D Desai outbound_scratch_pad_2)); 2377d18d1b47SKashyap D Desai 2378d18d1b47SKashyap D Desai /* Check max MSI-X vectors */ 2379d18d1b47SKashyap D Desai if (sc->device_id == MRSAS_TBOLT) { 2380d18d1b47SKashyap D Desai sc->msix_vectors = (scratch_pad_2 2381d18d1b47SKashyap D Desai & MR_MAX_REPLY_QUEUES_OFFSET) + 1; 2382d18d1b47SKashyap D Desai fw_msix_count = sc->msix_vectors; 2383d18d1b47SKashyap D Desai } else { 2384d18d1b47SKashyap D Desai /* Invader/Fury supports 96 MSI-X vectors */ 2385d18d1b47SKashyap D Desai sc->msix_vectors = ((scratch_pad_2 2386d18d1b47SKashyap D Desai & MR_MAX_REPLY_QUEUES_EXT_OFFSET) 2387d18d1b47SKashyap D Desai >> MR_MAX_REPLY_QUEUES_EXT_OFFSET_SHIFT) + 1; 2388d18d1b47SKashyap D Desai fw_msix_count = sc->msix_vectors; 2389d18d1b47SKashyap D Desai 23907aade8bfSKashyap D Desai if ((sc->mrsas_gen3_ctrl && (sc->msix_vectors > 8)) || 23912909aab4SKashyap D Desai ((sc->is_ventura || sc->is_aero) && (sc->msix_vectors > 16))) 23927aade8bfSKashyap D Desai sc->msix_combined = true; 23937aade8bfSKashyap D Desai /* 23947aade8bfSKashyap D Desai * Save 1-15 reply post index 23957aade8bfSKashyap D Desai * address to local memory Index 0 23967aade8bfSKashyap D Desai * is already saved from reg offset 23977aade8bfSKashyap D Desai * MPI2_REPLY_POST_HOST_INDEX_OFFSET 23987aade8bfSKashyap D Desai */ 2399d18d1b47SKashyap D Desai for (loop = 1; loop < MR_MAX_MSIX_REG_ARRAY; 2400d18d1b47SKashyap D Desai loop++) { 2401d18d1b47SKashyap D Desai sc->msix_reg_offset[loop] = 2402d18d1b47SKashyap D Desai MPI2_SUP_REPLY_POST_HOST_INDEX_OFFSET + 2403d18d1b47SKashyap D Desai (loop * 0x10); 2404d18d1b47SKashyap D Desai } 2405d18d1b47SKashyap D Desai } 2406d18d1b47SKashyap D Desai 2407d18d1b47SKashyap D Desai /* Don't bother allocating more MSI-X vectors than cpus */ 2408d18d1b47SKashyap D Desai sc->msix_vectors = min(sc->msix_vectors, 2409d18d1b47SKashyap D Desai mp_ncpus); 2410d18d1b47SKashyap D Desai 2411d18d1b47SKashyap D Desai /* Allocate MSI-x vectors */ 2412d18d1b47SKashyap D Desai if (mrsas_allocate_msix(sc) == SUCCESS) 2413d18d1b47SKashyap D Desai sc->msix_enable = 1; 2414d18d1b47SKashyap D Desai else 2415d18d1b47SKashyap D Desai sc->msix_enable = 0; 2416d18d1b47SKashyap D Desai 2417d18d1b47SKashyap D Desai device_printf(sc->mrsas_dev, "FW supports <%d> MSIX vector," 2418d18d1b47SKashyap D Desai "Online CPU %d Current MSIX <%d>\n", 2419d18d1b47SKashyap D Desai fw_msix_count, mp_ncpus, sc->msix_vectors); 2420d18d1b47SKashyap D Desai } 24217aade8bfSKashyap D Desai /* 24227aade8bfSKashyap D Desai * MSI-X host index 0 is common for all adapter. 24237aade8bfSKashyap D Desai * It is used for all MPT based Adapters. 24247aade8bfSKashyap D Desai */ 24257aade8bfSKashyap D Desai if (sc->msix_combined) { 24267aade8bfSKashyap D Desai sc->msix_reg_offset[0] = 24277aade8bfSKashyap D Desai MPI2_SUP_REPLY_POST_HOST_INDEX_OFFSET; 24287aade8bfSKashyap D Desai } 2429665484d8SDoug Ambrisko if (mrsas_init_adapter(sc) != SUCCESS) { 2430665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Adapter initialize Fail.\n"); 2431665484d8SDoug Ambrisko return (1); 2432665484d8SDoug Ambrisko } 243379b4460bSKashyap D Desai 24342909aab4SKashyap D Desai if (sc->is_ventura || sc->is_aero) { 2435e315cf4dSKashyap D Desai scratch_pad_4 = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, 24363d273176SKashyap D Desai outbound_scratch_pad_4)); 24373d273176SKashyap D Desai if ((scratch_pad_4 & MR_NVME_PAGE_SIZE_MASK) >= MR_DEFAULT_NVME_PAGE_SHIFT) 24383d273176SKashyap D Desai sc->nvme_page_size = 1 << (scratch_pad_4 & MR_NVME_PAGE_SIZE_MASK); 24393d273176SKashyap D Desai 24403d273176SKashyap D Desai device_printf(sc->mrsas_dev, "NVME page size\t: (%d)\n", sc->nvme_page_size); 24413d273176SKashyap D Desai } 24423d273176SKashyap D Desai 2443665484d8SDoug Ambrisko /* Allocate internal commands for pass-thru */ 2444665484d8SDoug Ambrisko if (mrsas_alloc_mfi_cmds(sc) != SUCCESS) { 2445665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Allocate MFI cmd failed.\n"); 2446665484d8SDoug Ambrisko return (1); 2447665484d8SDoug Ambrisko } 2448af51c29fSKashyap D Desai sc->ctrl_info = malloc(sizeof(struct mrsas_ctrl_info), M_MRSAS, M_NOWAIT); 2449af51c29fSKashyap D Desai if (!sc->ctrl_info) { 2450af51c29fSKashyap D Desai device_printf(sc->mrsas_dev, "Malloc for ctrl_info failed.\n"); 2451af51c29fSKashyap D Desai return (1); 2452af51c29fSKashyap D Desai } 24534799d485SKashyap D Desai /* 24548e727371SKashyap D Desai * Get the controller info from FW, so that the MAX VD support 24558e727371SKashyap D Desai * availability can be decided. 24564799d485SKashyap D Desai */ 2457af51c29fSKashyap D Desai if (mrsas_get_ctrl_info(sc)) { 24584799d485SKashyap D Desai device_printf(sc->mrsas_dev, "Unable to get FW ctrl_info.\n"); 2459af51c29fSKashyap D Desai return (1); 24604799d485SKashyap D Desai } 246177cf7df8SKashyap D Desai sc->secure_jbod_support = 2462af51c29fSKashyap D Desai (u_int8_t)sc->ctrl_info->adapterOperations3.supportSecurityonJBOD; 246377cf7df8SKashyap D Desai 246477cf7df8SKashyap D Desai if (sc->secure_jbod_support) 246577cf7df8SKashyap D Desai device_printf(sc->mrsas_dev, "FW supports SED \n"); 246677cf7df8SKashyap D Desai 2467a688fcd0SKashyap D Desai if (sc->use_seqnum_jbod_fp) 2468a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, "FW supports JBOD Map \n"); 2469a688fcd0SKashyap D Desai 2470c376f864SKashyap D Desai if (sc->support_morethan256jbod) 2471c376f864SKashyap D Desai device_printf(sc->mrsas_dev, "FW supports JBOD Map Ext \n"); 2472c376f864SKashyap D Desai 2473665484d8SDoug Ambrisko if (mrsas_setup_raidmap(sc) != SUCCESS) { 2474a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, "Error: RAID map setup FAILED !!! " 2475a688fcd0SKashyap D Desai "There seems to be some problem in the controller\n" 2476a688fcd0SKashyap D Desai "Please contact to the SUPPORT TEAM if the problem persists\n"); 2477665484d8SDoug Ambrisko } 2478a688fcd0SKashyap D Desai megasas_setup_jbod_map(sc); 2479a688fcd0SKashyap D Desai 248079b4460bSKashyap D Desai memset(sc->target_list, 0, 248179b4460bSKashyap D Desai MRSAS_MAX_TM_TARGETS * sizeof(struct mrsas_target)); 248279b4460bSKashyap D Desai for (i = 0; i < MRSAS_MAX_TM_TARGETS; i++) 248379b4460bSKashyap D Desai sc->target_list[i].target_id = 0xffff; 248479b4460bSKashyap D Desai 2485665484d8SDoug Ambrisko /* For pass-thru, get PD/LD list and controller info */ 24864799d485SKashyap D Desai memset(sc->pd_list, 0, 24874799d485SKashyap D Desai MRSAS_MAX_PD * sizeof(struct mrsas_pd_list)); 2488a688fcd0SKashyap D Desai if (mrsas_get_pd_list(sc) != SUCCESS) { 2489a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, "Get PD list failed.\n"); 2490a688fcd0SKashyap D Desai return (1); 2491a688fcd0SKashyap D Desai } 24924799d485SKashyap D Desai memset(sc->ld_ids, 0xff, MRSAS_MAX_LD_IDS); 2493a688fcd0SKashyap D Desai if (mrsas_get_ld_list(sc) != SUCCESS) { 2494a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, "Get LD lsit failed.\n"); 2495a688fcd0SKashyap D Desai return (1); 2496a688fcd0SKashyap D Desai } 2497821df4b9SKashyap D Desai 24982909aab4SKashyap D Desai if ((sc->is_ventura || sc->is_aero) && sc->drv_stream_detection) { 2499821df4b9SKashyap D Desai sc->streamDetectByLD = malloc(sizeof(PTR_LD_STREAM_DETECT) * 2500821df4b9SKashyap D Desai MAX_LOGICAL_DRIVES_EXT, M_MRSAS, M_NOWAIT); 2501821df4b9SKashyap D Desai if (!sc->streamDetectByLD) { 2502821df4b9SKashyap D Desai device_printf(sc->mrsas_dev, 2503821df4b9SKashyap D Desai "unable to allocate stream detection for pool of LDs\n"); 2504821df4b9SKashyap D Desai return (1); 2505821df4b9SKashyap D Desai } 2506821df4b9SKashyap D Desai for (i = 0; i < MAX_LOGICAL_DRIVES_EXT; ++i) { 2507821df4b9SKashyap D Desai sc->streamDetectByLD[i] = malloc(sizeof(LD_STREAM_DETECT), M_MRSAS, M_NOWAIT); 2508821df4b9SKashyap D Desai if (!sc->streamDetectByLD[i]) { 2509821df4b9SKashyap D Desai device_printf(sc->mrsas_dev, "unable to allocate stream detect by LD\n"); 2510821df4b9SKashyap D Desai for (j = 0; j < i; ++j) 2511821df4b9SKashyap D Desai free(sc->streamDetectByLD[j], M_MRSAS); 2512821df4b9SKashyap D Desai free(sc->streamDetectByLD, M_MRSAS); 2513821df4b9SKashyap D Desai sc->streamDetectByLD = NULL; 2514821df4b9SKashyap D Desai return (1); 2515821df4b9SKashyap D Desai } 2516821df4b9SKashyap D Desai memset(sc->streamDetectByLD[i], 0, sizeof(LD_STREAM_DETECT)); 2517821df4b9SKashyap D Desai sc->streamDetectByLD[i]->mruBitMap = MR_STREAM_BITMAP; 2518821df4b9SKashyap D Desai } 2519821df4b9SKashyap D Desai } 2520821df4b9SKashyap D Desai 2521665484d8SDoug Ambrisko /* 25228e727371SKashyap D Desai * Compute the max allowed sectors per IO: The controller info has 25238e727371SKashyap D Desai * two limits on max sectors. Driver should use the minimum of these 25248e727371SKashyap D Desai * two. 2525665484d8SDoug Ambrisko * 2526665484d8SDoug Ambrisko * 1 << stripe_sz_ops.min = max sectors per strip 2527665484d8SDoug Ambrisko * 25288e727371SKashyap D Desai * Note that older firmwares ( < FW ver 30) didn't report information to 25298e727371SKashyap D Desai * calculate max_sectors_1. So the number ended up as zero always. 2530665484d8SDoug Ambrisko */ 2531665484d8SDoug Ambrisko tmp_sectors = 0; 2532af51c29fSKashyap D Desai max_sectors_1 = (1 << sc->ctrl_info->stripe_sz_ops.min) * 2533af51c29fSKashyap D Desai sc->ctrl_info->max_strips_per_io; 2534af51c29fSKashyap D Desai max_sectors_2 = sc->ctrl_info->max_request_size; 2535665484d8SDoug Ambrisko tmp_sectors = min(max_sectors_1, max_sectors_2); 25364799d485SKashyap D Desai sc->max_sectors_per_req = sc->max_num_sge * MRSAS_PAGE_SIZE / 512; 25374799d485SKashyap D Desai 25384799d485SKashyap D Desai if (tmp_sectors && (sc->max_sectors_per_req > tmp_sectors)) 25394799d485SKashyap D Desai sc->max_sectors_per_req = tmp_sectors; 25404799d485SKashyap D Desai 2541665484d8SDoug Ambrisko sc->disableOnlineCtrlReset = 2542af51c29fSKashyap D Desai sc->ctrl_info->properties.OnOffProperties.disableOnlineCtrlReset; 2543665484d8SDoug Ambrisko sc->UnevenSpanSupport = 2544af51c29fSKashyap D Desai sc->ctrl_info->adapterOperations2.supportUnevenSpans; 2545665484d8SDoug Ambrisko if (sc->UnevenSpanSupport) { 25468e727371SKashyap D Desai device_printf(sc->mrsas_dev, "FW supports: UnevenSpanSupport=%x\n\n", 2547665484d8SDoug Ambrisko sc->UnevenSpanSupport); 25484799d485SKashyap D Desai 2549665484d8SDoug Ambrisko if (MR_ValidateMapInfo(sc)) 2550665484d8SDoug Ambrisko sc->fast_path_io = 1; 2551665484d8SDoug Ambrisko else 2552665484d8SDoug Ambrisko sc->fast_path_io = 0; 2553665484d8SDoug Ambrisko } 25545437c8b8SKashyap D Desai 25555437c8b8SKashyap D Desai device_printf(sc->mrsas_dev, "max_fw_cmds: %u max_scsi_cmds: %u\n", 25565437c8b8SKashyap D Desai sc->max_fw_cmds, sc->max_scsi_cmds); 2557665484d8SDoug Ambrisko return (0); 2558665484d8SDoug Ambrisko } 2559665484d8SDoug Ambrisko 25608e727371SKashyap D Desai /* 2561665484d8SDoug Ambrisko * mrsas_init_adapter: Initializes the adapter/controller 2562665484d8SDoug Ambrisko * input: Adapter soft state 2563665484d8SDoug Ambrisko * 2564665484d8SDoug Ambrisko * Prepares for the issuing of the IOC Init cmd to FW for initializing the 2565665484d8SDoug Ambrisko * ROC/controller. The FW register is read to determined the number of 2566665484d8SDoug Ambrisko * commands that is supported. All memory allocations for IO is based on 2567665484d8SDoug Ambrisko * max_cmd. Appropriate calculations are performed in this function. 2568665484d8SDoug Ambrisko */ 25698e727371SKashyap D Desai int 25708e727371SKashyap D Desai mrsas_init_adapter(struct mrsas_softc *sc) 2571665484d8SDoug Ambrisko { 2572665484d8SDoug Ambrisko uint32_t status; 25732a1d3bcdSKashyap D Desai u_int32_t scratch_pad_2; 2574665484d8SDoug Ambrisko int ret; 2575d18d1b47SKashyap D Desai int i = 0; 2576665484d8SDoug Ambrisko 2577665484d8SDoug Ambrisko /* Read FW status register */ 2578e315cf4dSKashyap D Desai status = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, outbound_scratch_pad)); 2579665484d8SDoug Ambrisko 2580665484d8SDoug Ambrisko sc->max_fw_cmds = status & MRSAS_FWSTATE_MAXCMD_MASK; 2581665484d8SDoug Ambrisko 2582665484d8SDoug Ambrisko /* Decrement the max supported by 1, to correlate with FW */ 2583665484d8SDoug Ambrisko sc->max_fw_cmds = sc->max_fw_cmds - 1; 258454f784f5SKashyap D Desai sc->max_scsi_cmds = sc->max_fw_cmds - MRSAS_MAX_MFI_CMDS; 2585665484d8SDoug Ambrisko 2586665484d8SDoug Ambrisko /* Determine allocation size of command frames */ 25872a1d3bcdSKashyap D Desai sc->reply_q_depth = ((sc->max_fw_cmds + 1 + 15) / 16 * 16) * 2; 25882a1d3bcdSKashyap D Desai sc->request_alloc_sz = sizeof(MRSAS_REQUEST_DESCRIPTOR_UNION) * sc->max_fw_cmds; 2589665484d8SDoug Ambrisko sc->reply_alloc_sz = sizeof(MPI2_REPLY_DESCRIPTORS_UNION) * (sc->reply_q_depth); 25902a1d3bcdSKashyap D Desai sc->io_frames_alloc_sz = MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE + 25912a1d3bcdSKashyap D Desai (MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE * (sc->max_fw_cmds + 1)); 2592e315cf4dSKashyap D Desai scratch_pad_2 = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, 25933a3fc6cbSKashyap D Desai outbound_scratch_pad_2)); 2594*e34a057cSAlfredo Dal'Ava Junior 2595*e34a057cSAlfredo Dal'Ava Junior mrsas_dprint(sc, MRSAS_TRACE, "%s: sc->reply_q_depth 0x%x," 2596*e34a057cSAlfredo Dal'Ava Junior "sc->request_alloc_sz 0x%x, sc->reply_alloc_sz 0x%x," 2597*e34a057cSAlfredo Dal'Ava Junior "sc->io_frames_alloc_sz 0x%x\n", __func__, 2598*e34a057cSAlfredo Dal'Ava Junior sc->reply_q_depth, sc->request_alloc_sz, 2599*e34a057cSAlfredo Dal'Ava Junior sc->reply_alloc_sz, sc->io_frames_alloc_sz); 2600*e34a057cSAlfredo Dal'Ava Junior 26013a3fc6cbSKashyap D Desai /* 26023a3fc6cbSKashyap D Desai * If scratch_pad_2 & MEGASAS_MAX_CHAIN_SIZE_UNITS_MASK is set, 26033a3fc6cbSKashyap D Desai * Firmware support extended IO chain frame which is 4 time more 26043a3fc6cbSKashyap D Desai * than legacy Firmware. Legacy Firmware - Frame size is (8 * 128) = 26053a3fc6cbSKashyap D Desai * 1K 1M IO Firmware - Frame size is (8 * 128 * 4) = 4K 26063a3fc6cbSKashyap D Desai */ 26073a3fc6cbSKashyap D Desai if (scratch_pad_2 & MEGASAS_MAX_CHAIN_SIZE_UNITS_MASK) 26083a3fc6cbSKashyap D Desai sc->max_chain_frame_sz = 26093a3fc6cbSKashyap D Desai ((scratch_pad_2 & MEGASAS_MAX_CHAIN_SIZE_MASK) >> 5) 26103a3fc6cbSKashyap D Desai * MEGASAS_1MB_IO; 26113a3fc6cbSKashyap D Desai else 26123a3fc6cbSKashyap D Desai sc->max_chain_frame_sz = 26133a3fc6cbSKashyap D Desai ((scratch_pad_2 & MEGASAS_MAX_CHAIN_SIZE_MASK) >> 5) 26143a3fc6cbSKashyap D Desai * MEGASAS_256K_IO; 26153a3fc6cbSKashyap D Desai 26162a1d3bcdSKashyap D Desai sc->chain_frames_alloc_sz = sc->max_chain_frame_sz * sc->max_fw_cmds; 2617665484d8SDoug Ambrisko sc->max_sge_in_main_msg = (MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE - 2618665484d8SDoug Ambrisko offsetof(MRSAS_RAID_SCSI_IO_REQUEST, SGL)) / 16; 2619665484d8SDoug Ambrisko 26203a3fc6cbSKashyap D Desai sc->max_sge_in_chain = sc->max_chain_frame_sz / sizeof(MPI2_SGE_IO_UNION); 2621665484d8SDoug Ambrisko sc->max_num_sge = sc->max_sge_in_main_msg + sc->max_sge_in_chain - 2; 2622665484d8SDoug Ambrisko 26232a1d3bcdSKashyap D Desai mrsas_dprint(sc, MRSAS_INFO, 26242a1d3bcdSKashyap D Desai "max sge: 0x%x, max chain frame size: 0x%x, " 2625*e34a057cSAlfredo Dal'Ava Junior "max fw cmd: 0x%x sc->chain_frames_alloc_sz: 0x%x\n", 2626*e34a057cSAlfredo Dal'Ava Junior sc->max_num_sge, 2627*e34a057cSAlfredo Dal'Ava Junior sc->max_chain_frame_sz, sc->max_fw_cmds, 2628*e34a057cSAlfredo Dal'Ava Junior sc->chain_frames_alloc_sz); 26293a3fc6cbSKashyap D Desai 2630665484d8SDoug Ambrisko /* Used for pass thru MFI frame (DCMD) */ 2631665484d8SDoug Ambrisko sc->chain_offset_mfi_pthru = offsetof(MRSAS_RAID_SCSI_IO_REQUEST, SGL) / 16; 2632665484d8SDoug Ambrisko 2633665484d8SDoug Ambrisko sc->chain_offset_io_request = (MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE - 2634665484d8SDoug Ambrisko sizeof(MPI2_SGE_IO_UNION)) / 16; 2635665484d8SDoug Ambrisko 2636d18d1b47SKashyap D Desai int count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 26378e727371SKashyap D Desai 2638d18d1b47SKashyap D Desai for (i = 0; i < count; i++) 2639d18d1b47SKashyap D Desai sc->last_reply_idx[i] = 0; 2640665484d8SDoug Ambrisko 2641665484d8SDoug Ambrisko ret = mrsas_alloc_mem(sc); 2642665484d8SDoug Ambrisko if (ret != SUCCESS) 2643665484d8SDoug Ambrisko return (ret); 2644665484d8SDoug Ambrisko 2645665484d8SDoug Ambrisko ret = mrsas_alloc_mpt_cmds(sc); 2646665484d8SDoug Ambrisko if (ret != SUCCESS) 2647665484d8SDoug Ambrisko return (ret); 2648665484d8SDoug Ambrisko 2649665484d8SDoug Ambrisko ret = mrsas_ioc_init(sc); 2650665484d8SDoug Ambrisko if (ret != SUCCESS) 2651665484d8SDoug Ambrisko return (ret); 2652665484d8SDoug Ambrisko 2653665484d8SDoug Ambrisko return (0); 2654665484d8SDoug Ambrisko } 2655665484d8SDoug Ambrisko 26568e727371SKashyap D Desai /* 2657665484d8SDoug Ambrisko * mrsas_alloc_ioc_cmd: Allocates memory for IOC Init command 2658665484d8SDoug Ambrisko * input: Adapter soft state 2659665484d8SDoug Ambrisko * 2660665484d8SDoug Ambrisko * Allocates for the IOC Init cmd to FW to initialize the ROC/controller. 2661665484d8SDoug Ambrisko */ 26628e727371SKashyap D Desai int 26638e727371SKashyap D Desai mrsas_alloc_ioc_cmd(struct mrsas_softc *sc) 2664665484d8SDoug Ambrisko { 2665665484d8SDoug Ambrisko int ioc_init_size; 2666665484d8SDoug Ambrisko 2667665484d8SDoug Ambrisko /* Allocate IOC INIT command */ 2668665484d8SDoug Ambrisko ioc_init_size = 1024 + sizeof(MPI2_IOC_INIT_REQUEST); 26698e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 26708e727371SKashyap D Desai 1, 0, 26718e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 26728e727371SKashyap D Desai BUS_SPACE_MAXADDR, 26738e727371SKashyap D Desai NULL, NULL, 26748e727371SKashyap D Desai ioc_init_size, 26758e727371SKashyap D Desai 1, 26768e727371SKashyap D Desai ioc_init_size, 26778e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 26788e727371SKashyap D Desai NULL, NULL, 2679665484d8SDoug Ambrisko &sc->ioc_init_tag)) { 2680665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate ioc init tag\n"); 2681665484d8SDoug Ambrisko return (ENOMEM); 2682665484d8SDoug Ambrisko } 2683665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->ioc_init_tag, (void **)&sc->ioc_init_mem, 2684665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->ioc_init_dmamap)) { 2685665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate ioc init cmd mem\n"); 2686665484d8SDoug Ambrisko return (ENOMEM); 2687665484d8SDoug Ambrisko } 2688665484d8SDoug Ambrisko bzero(sc->ioc_init_mem, ioc_init_size); 2689665484d8SDoug Ambrisko if (bus_dmamap_load(sc->ioc_init_tag, sc->ioc_init_dmamap, 2690665484d8SDoug Ambrisko sc->ioc_init_mem, ioc_init_size, mrsas_addr_cb, 2691665484d8SDoug Ambrisko &sc->ioc_init_phys_mem, BUS_DMA_NOWAIT)) { 2692665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load ioc init cmd mem\n"); 2693665484d8SDoug Ambrisko return (ENOMEM); 2694665484d8SDoug Ambrisko } 2695665484d8SDoug Ambrisko return (0); 2696665484d8SDoug Ambrisko } 2697665484d8SDoug Ambrisko 26988e727371SKashyap D Desai /* 2699665484d8SDoug Ambrisko * mrsas_free_ioc_cmd: Allocates memory for IOC Init command 2700665484d8SDoug Ambrisko * input: Adapter soft state 2701665484d8SDoug Ambrisko * 2702665484d8SDoug Ambrisko * Deallocates memory of the IOC Init cmd. 2703665484d8SDoug Ambrisko */ 27048e727371SKashyap D Desai void 27058e727371SKashyap D Desai mrsas_free_ioc_cmd(struct mrsas_softc *sc) 2706665484d8SDoug Ambrisko { 2707665484d8SDoug Ambrisko if (sc->ioc_init_phys_mem) 2708665484d8SDoug Ambrisko bus_dmamap_unload(sc->ioc_init_tag, sc->ioc_init_dmamap); 2709665484d8SDoug Ambrisko if (sc->ioc_init_mem != NULL) 2710665484d8SDoug Ambrisko bus_dmamem_free(sc->ioc_init_tag, sc->ioc_init_mem, sc->ioc_init_dmamap); 2711665484d8SDoug Ambrisko if (sc->ioc_init_tag != NULL) 2712665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->ioc_init_tag); 2713665484d8SDoug Ambrisko } 2714665484d8SDoug Ambrisko 27158e727371SKashyap D Desai /* 2716665484d8SDoug Ambrisko * mrsas_ioc_init: Sends IOC Init command to FW 2717665484d8SDoug Ambrisko * input: Adapter soft state 2718665484d8SDoug Ambrisko * 2719665484d8SDoug Ambrisko * Issues the IOC Init cmd to FW to initialize the ROC/controller. 2720665484d8SDoug Ambrisko */ 27218e727371SKashyap D Desai int 27228e727371SKashyap D Desai mrsas_ioc_init(struct mrsas_softc *sc) 2723665484d8SDoug Ambrisko { 2724665484d8SDoug Ambrisko struct mrsas_init_frame *init_frame; 2725665484d8SDoug Ambrisko pMpi2IOCInitRequest_t IOCInitMsg; 2726665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION req_desc; 2727e80341d5SKashyap D Desai u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME; 2728665484d8SDoug Ambrisko bus_addr_t phys_addr; 2729665484d8SDoug Ambrisko int i, retcode = 0; 2730d993dd83SKashyap D Desai u_int32_t scratch_pad_2; 2731665484d8SDoug Ambrisko 2732665484d8SDoug Ambrisko /* Allocate memory for the IOC INIT command */ 2733665484d8SDoug Ambrisko if (mrsas_alloc_ioc_cmd(sc)) { 2734665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate IOC command.\n"); 2735665484d8SDoug Ambrisko return (1); 2736665484d8SDoug Ambrisko } 2737d993dd83SKashyap D Desai 2738d993dd83SKashyap D Desai if (!sc->block_sync_cache) { 2739e315cf4dSKashyap D Desai scratch_pad_2 = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, 2740d993dd83SKashyap D Desai outbound_scratch_pad_2)); 2741d993dd83SKashyap D Desai sc->fw_sync_cache_support = (scratch_pad_2 & 2742d993dd83SKashyap D Desai MR_CAN_HANDLE_SYNC_CACHE_OFFSET) ? 1 : 0; 2743d993dd83SKashyap D Desai } 2744d993dd83SKashyap D Desai 2745665484d8SDoug Ambrisko IOCInitMsg = (pMpi2IOCInitRequest_t)(((char *)sc->ioc_init_mem) + 1024); 2746665484d8SDoug Ambrisko IOCInitMsg->Function = MPI2_FUNCTION_IOC_INIT; 2747665484d8SDoug Ambrisko IOCInitMsg->WhoInit = MPI2_WHOINIT_HOST_DRIVER; 2748*e34a057cSAlfredo Dal'Ava Junior IOCInitMsg->MsgVersion = htole16(MPI2_VERSION); 2749*e34a057cSAlfredo Dal'Ava Junior IOCInitMsg->HeaderVersion = htole16(MPI2_HEADER_VERSION); 2750*e34a057cSAlfredo Dal'Ava Junior IOCInitMsg->SystemRequestFrameSize = htole16(MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE / 4); 2751*e34a057cSAlfredo Dal'Ava Junior IOCInitMsg->ReplyDescriptorPostQueueDepth = htole16(sc->reply_q_depth); 2752*e34a057cSAlfredo Dal'Ava Junior IOCInitMsg->ReplyDescriptorPostQueueAddress = htole64(sc->reply_desc_phys_addr); 2753*e34a057cSAlfredo Dal'Ava Junior IOCInitMsg->SystemRequestFrameBaseAddress = htole64(sc->io_request_phys_addr); 2754d18d1b47SKashyap D Desai IOCInitMsg->HostMSIxVectors = (sc->msix_vectors > 0 ? sc->msix_vectors : 0); 27553d273176SKashyap D Desai IOCInitMsg->HostPageSize = MR_DEFAULT_NVME_PAGE_SHIFT; 2756665484d8SDoug Ambrisko 2757665484d8SDoug Ambrisko init_frame = (struct mrsas_init_frame *)sc->ioc_init_mem; 2758665484d8SDoug Ambrisko init_frame->cmd = MFI_CMD_INIT; 2759665484d8SDoug Ambrisko init_frame->cmd_status = 0xFF; 2760*e34a057cSAlfredo Dal'Ava Junior init_frame->flags |= htole16(MFI_FRAME_DONT_POST_IN_REPLY_QUEUE); 2761665484d8SDoug Ambrisko 2762d18d1b47SKashyap D Desai /* driver support Extended MSIX */ 27632909aab4SKashyap D Desai if (sc->mrsas_gen3_ctrl || sc->is_ventura || sc->is_aero) { 2764d18d1b47SKashyap D Desai init_frame->driver_operations. 2765d18d1b47SKashyap D Desai mfi_capabilities.support_additional_msix = 1; 2766d18d1b47SKashyap D Desai } 2767665484d8SDoug Ambrisko if (sc->verbuf_mem) { 2768665484d8SDoug Ambrisko snprintf((char *)sc->verbuf_mem, strlen(MRSAS_VERSION) + 2, "%s\n", 2769665484d8SDoug Ambrisko MRSAS_VERSION); 2770665484d8SDoug Ambrisko init_frame->driver_ver_lo = (bus_addr_t)sc->verbuf_phys_addr; 2771665484d8SDoug Ambrisko init_frame->driver_ver_hi = 0; 2772665484d8SDoug Ambrisko } 277316dc2814SKashyap D Desai init_frame->driver_operations.mfi_capabilities.support_ndrive_r1_lb = 1; 27744799d485SKashyap D Desai init_frame->driver_operations.mfi_capabilities.support_max_255lds = 1; 277577cf7df8SKashyap D Desai init_frame->driver_operations.mfi_capabilities.security_protocol_cmds_fw = 1; 27763a3fc6cbSKashyap D Desai if (sc->max_chain_frame_sz > MEGASAS_CHAIN_FRAME_SZ_MIN) 27773a3fc6cbSKashyap D Desai init_frame->driver_operations.mfi_capabilities.support_ext_io_size = 1; 2778665484d8SDoug Ambrisko 2779*e34a057cSAlfredo Dal'Ava Junior init_frame->driver_operations.reg = htole32(init_frame->driver_operations.reg); 2780*e34a057cSAlfredo Dal'Ava Junior 2781*e34a057cSAlfredo Dal'Ava Junior phys_addr = (bus_addr_t)sc->ioc_init_phys_mem + 1024; 2782*e34a057cSAlfredo Dal'Ava Junior init_frame->queue_info_new_phys_addr_lo = htole32(phys_addr); 2783*e34a057cSAlfredo Dal'Ava Junior init_frame->data_xfer_len = htole32(sizeof(Mpi2IOCInitRequest_t)); 2784*e34a057cSAlfredo Dal'Ava Junior 2785*e34a057cSAlfredo Dal'Ava Junior req_desc.addr.u.low = htole32((bus_addr_t)sc->ioc_init_phys_mem & 0xFFFFFFFF); 2786*e34a057cSAlfredo Dal'Ava Junior req_desc.addr.u.high = htole32((bus_addr_t)sc->ioc_init_phys_mem >> 32); 2787*e34a057cSAlfredo Dal'Ava Junior 2788665484d8SDoug Ambrisko req_desc.MFAIo.RequestFlags = 2789665484d8SDoug Ambrisko (MRSAS_REQ_DESCRIPT_FLAGS_MFA << MRSAS_REQ_DESCRIPT_FLAGS_TYPE_SHIFT); 2790665484d8SDoug Ambrisko 2791665484d8SDoug Ambrisko mrsas_disable_intr(sc); 2792665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "Issuing IOC INIT command to FW.\n"); 2793b518670cSKashyap D Desai mrsas_write_64bit_req_desc(sc, req_desc.addr.u.low, req_desc.addr.u.high); 2794665484d8SDoug Ambrisko 2795665484d8SDoug Ambrisko /* 2796665484d8SDoug Ambrisko * Poll response timer to wait for Firmware response. While this 2797665484d8SDoug Ambrisko * timer with the DELAY call could block CPU, the time interval for 2798665484d8SDoug Ambrisko * this is only 1 millisecond. 2799665484d8SDoug Ambrisko */ 2800665484d8SDoug Ambrisko if (init_frame->cmd_status == 0xFF) { 2801665484d8SDoug Ambrisko for (i = 0; i < (max_wait * 1000); i++) { 2802665484d8SDoug Ambrisko if (init_frame->cmd_status == 0xFF) 2803665484d8SDoug Ambrisko DELAY(1000); 2804665484d8SDoug Ambrisko else 2805665484d8SDoug Ambrisko break; 2806665484d8SDoug Ambrisko } 2807665484d8SDoug Ambrisko } 2808665484d8SDoug Ambrisko if (init_frame->cmd_status == 0) 2809665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 2810665484d8SDoug Ambrisko "IOC INIT response received from FW.\n"); 28118e727371SKashyap D Desai else { 2812665484d8SDoug Ambrisko if (init_frame->cmd_status == 0xFF) 2813665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "IOC Init timed out after %d seconds.\n", max_wait); 2814665484d8SDoug Ambrisko else 2815665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "IOC Init failed, status = 0x%x\n", init_frame->cmd_status); 2816665484d8SDoug Ambrisko retcode = 1; 2817665484d8SDoug Ambrisko } 2818665484d8SDoug Ambrisko 2819b518670cSKashyap D Desai if (sc->is_aero) { 2820e315cf4dSKashyap D Desai scratch_pad_2 = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, 2821b518670cSKashyap D Desai outbound_scratch_pad_2)); 2822b518670cSKashyap D Desai sc->atomic_desc_support = (scratch_pad_2 & 2823b518670cSKashyap D Desai MR_ATOMIC_DESCRIPTOR_SUPPORT_OFFSET) ? 1 : 0; 2824b518670cSKashyap D Desai device_printf(sc->mrsas_dev, "FW supports atomic descriptor: %s\n", 2825b518670cSKashyap D Desai sc->atomic_desc_support ? "Yes" : "No"); 2826b518670cSKashyap D Desai } 2827b518670cSKashyap D Desai 2828665484d8SDoug Ambrisko mrsas_free_ioc_cmd(sc); 2829665484d8SDoug Ambrisko return (retcode); 2830665484d8SDoug Ambrisko } 2831665484d8SDoug Ambrisko 28328e727371SKashyap D Desai /* 2833665484d8SDoug Ambrisko * mrsas_alloc_mpt_cmds: Allocates the command packets 2834665484d8SDoug Ambrisko * input: Adapter instance soft state 2835665484d8SDoug Ambrisko * 2836665484d8SDoug Ambrisko * This function allocates the internal commands for IOs. Each command that is 28378e727371SKashyap D Desai * issued to FW is wrapped in a local data structure called mrsas_mpt_cmd. An 28388e727371SKashyap D Desai * array is allocated with mrsas_mpt_cmd context. The free commands are 2839665484d8SDoug Ambrisko * maintained in a linked list (cmd pool). SMID value range is from 1 to 2840665484d8SDoug Ambrisko * max_fw_cmds. 2841665484d8SDoug Ambrisko */ 28428e727371SKashyap D Desai int 28438e727371SKashyap D Desai mrsas_alloc_mpt_cmds(struct mrsas_softc *sc) 2844665484d8SDoug Ambrisko { 2845665484d8SDoug Ambrisko int i, j; 28462a1d3bcdSKashyap D Desai u_int32_t max_fw_cmds, count; 2847665484d8SDoug Ambrisko struct mrsas_mpt_cmd *cmd; 2848665484d8SDoug Ambrisko pMpi2ReplyDescriptorsUnion_t reply_desc; 2849665484d8SDoug Ambrisko u_int32_t offset, chain_offset, sense_offset; 2850665484d8SDoug Ambrisko bus_addr_t io_req_base_phys, chain_frame_base_phys, sense_base_phys; 2851665484d8SDoug Ambrisko u_int8_t *io_req_base, *chain_frame_base, *sense_base; 2852665484d8SDoug Ambrisko 28532a1d3bcdSKashyap D Desai max_fw_cmds = sc->max_fw_cmds; 2854665484d8SDoug Ambrisko 2855665484d8SDoug Ambrisko sc->req_desc = malloc(sc->request_alloc_sz, M_MRSAS, M_NOWAIT); 2856665484d8SDoug Ambrisko if (!sc->req_desc) { 2857665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Out of memory, cannot alloc req desc\n"); 2858665484d8SDoug Ambrisko return (ENOMEM); 2859665484d8SDoug Ambrisko } 2860665484d8SDoug Ambrisko memset(sc->req_desc, 0, sc->request_alloc_sz); 2861665484d8SDoug Ambrisko 2862665484d8SDoug Ambrisko /* 28638e727371SKashyap D Desai * sc->mpt_cmd_list is an array of struct mrsas_mpt_cmd pointers. 28648e727371SKashyap D Desai * Allocate the dynamic array first and then allocate individual 28658e727371SKashyap D Desai * commands. 2866665484d8SDoug Ambrisko */ 28672a1d3bcdSKashyap D Desai sc->mpt_cmd_list = malloc(sizeof(struct mrsas_mpt_cmd *) * max_fw_cmds, 28682a1d3bcdSKashyap D Desai M_MRSAS, M_NOWAIT); 2869665484d8SDoug Ambrisko if (!sc->mpt_cmd_list) { 2870665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc memory for mpt_cmd_list.\n"); 2871665484d8SDoug Ambrisko return (ENOMEM); 2872665484d8SDoug Ambrisko } 28732a1d3bcdSKashyap D Desai memset(sc->mpt_cmd_list, 0, sizeof(struct mrsas_mpt_cmd *) * max_fw_cmds); 28742a1d3bcdSKashyap D Desai for (i = 0; i < max_fw_cmds; i++) { 2875665484d8SDoug Ambrisko sc->mpt_cmd_list[i] = malloc(sizeof(struct mrsas_mpt_cmd), 2876665484d8SDoug Ambrisko M_MRSAS, M_NOWAIT); 2877665484d8SDoug Ambrisko if (!sc->mpt_cmd_list[i]) { 2878665484d8SDoug Ambrisko for (j = 0; j < i; j++) 2879665484d8SDoug Ambrisko free(sc->mpt_cmd_list[j], M_MRSAS); 2880665484d8SDoug Ambrisko free(sc->mpt_cmd_list, M_MRSAS); 2881665484d8SDoug Ambrisko sc->mpt_cmd_list = NULL; 2882665484d8SDoug Ambrisko return (ENOMEM); 2883665484d8SDoug Ambrisko } 2884665484d8SDoug Ambrisko } 2885665484d8SDoug Ambrisko 2886665484d8SDoug Ambrisko io_req_base = (u_int8_t *)sc->io_request_mem + MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE; 2887665484d8SDoug Ambrisko io_req_base_phys = (bus_addr_t)sc->io_request_phys_addr + MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE; 2888665484d8SDoug Ambrisko chain_frame_base = (u_int8_t *)sc->chain_frame_mem; 2889665484d8SDoug Ambrisko chain_frame_base_phys = (bus_addr_t)sc->chain_frame_phys_addr; 2890665484d8SDoug Ambrisko sense_base = (u_int8_t *)sc->sense_mem; 2891665484d8SDoug Ambrisko sense_base_phys = (bus_addr_t)sc->sense_phys_addr; 28922a1d3bcdSKashyap D Desai for (i = 0; i < max_fw_cmds; i++) { 2893665484d8SDoug Ambrisko cmd = sc->mpt_cmd_list[i]; 2894665484d8SDoug Ambrisko offset = MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE * i; 28953a3fc6cbSKashyap D Desai chain_offset = sc->max_chain_frame_sz * i; 2896665484d8SDoug Ambrisko sense_offset = MRSAS_SENSE_LEN * i; 2897665484d8SDoug Ambrisko memset(cmd, 0, sizeof(struct mrsas_mpt_cmd)); 2898665484d8SDoug Ambrisko cmd->index = i + 1; 2899665484d8SDoug Ambrisko cmd->ccb_ptr = NULL; 29002a1d3bcdSKashyap D Desai cmd->r1_alt_dev_handle = MR_DEVHANDLE_INVALID; 29018bb601acSKashyap D Desai callout_init_mtx(&cmd->cm_callout, &sc->sim_lock, 0); 2902665484d8SDoug Ambrisko cmd->sync_cmd_idx = (u_int32_t)MRSAS_ULONG_MAX; 2903665484d8SDoug Ambrisko cmd->sc = sc; 2904665484d8SDoug Ambrisko cmd->io_request = (MRSAS_RAID_SCSI_IO_REQUEST *) (io_req_base + offset); 2905665484d8SDoug Ambrisko memset(cmd->io_request, 0, sizeof(MRSAS_RAID_SCSI_IO_REQUEST)); 2906665484d8SDoug Ambrisko cmd->io_request_phys_addr = io_req_base_phys + offset; 2907665484d8SDoug Ambrisko cmd->chain_frame = (MPI2_SGE_IO_UNION *) (chain_frame_base + chain_offset); 2908665484d8SDoug Ambrisko cmd->chain_frame_phys_addr = chain_frame_base_phys + chain_offset; 2909665484d8SDoug Ambrisko cmd->sense = sense_base + sense_offset; 2910665484d8SDoug Ambrisko cmd->sense_phys_addr = sense_base_phys + sense_offset; 2911665484d8SDoug Ambrisko if (bus_dmamap_create(sc->data_tag, 0, &cmd->data_dmamap)) { 2912665484d8SDoug Ambrisko return (FAIL); 2913665484d8SDoug Ambrisko } 2914665484d8SDoug Ambrisko TAILQ_INSERT_TAIL(&(sc->mrsas_mpt_cmd_list_head), cmd, next); 2915665484d8SDoug Ambrisko } 2916665484d8SDoug Ambrisko 2917665484d8SDoug Ambrisko /* Initialize reply descriptor array to 0xFFFFFFFF */ 2918665484d8SDoug Ambrisko reply_desc = sc->reply_desc_mem; 2919d18d1b47SKashyap D Desai count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 2920d18d1b47SKashyap D Desai for (i = 0; i < sc->reply_q_depth * count; i++, reply_desc++) { 2921665484d8SDoug Ambrisko reply_desc->Words = MRSAS_ULONG_MAX; 2922665484d8SDoug Ambrisko } 2923665484d8SDoug Ambrisko return (0); 2924665484d8SDoug Ambrisko } 2925665484d8SDoug Ambrisko 29268e727371SKashyap D Desai /* 2927b518670cSKashyap D Desai * mrsas_write_64bit_req_dsc: Writes 64 bit request descriptor to FW 2928b518670cSKashyap D Desai * input: Adapter softstate 2929b518670cSKashyap D Desai * request descriptor address low 2930b518670cSKashyap D Desai * request descriptor address high 2931b518670cSKashyap D Desai */ 2932b518670cSKashyap D Desai void 2933b518670cSKashyap D Desai mrsas_write_64bit_req_desc(struct mrsas_softc *sc, u_int32_t req_desc_lo, 2934b518670cSKashyap D Desai u_int32_t req_desc_hi) 2935b518670cSKashyap D Desai { 2936b518670cSKashyap D Desai mtx_lock(&sc->pci_lock); 2937b518670cSKashyap D Desai mrsas_write_reg(sc, offsetof(mrsas_reg_set, inbound_low_queue_port), 2938*e34a057cSAlfredo Dal'Ava Junior le32toh(req_desc_lo)); 2939b518670cSKashyap D Desai mrsas_write_reg(sc, offsetof(mrsas_reg_set, inbound_high_queue_port), 2940*e34a057cSAlfredo Dal'Ava Junior le32toh(req_desc_hi)); 2941b518670cSKashyap D Desai mtx_unlock(&sc->pci_lock); 2942b518670cSKashyap D Desai } 2943b518670cSKashyap D Desai 2944b518670cSKashyap D Desai /* 2945665484d8SDoug Ambrisko * mrsas_fire_cmd: Sends command to FW 2946665484d8SDoug Ambrisko * input: Adapter softstate 2947665484d8SDoug Ambrisko * request descriptor address low 2948665484d8SDoug Ambrisko * request descriptor address high 2949665484d8SDoug Ambrisko * 2950665484d8SDoug Ambrisko * This functions fires the command to Firmware by writing to the 2951665484d8SDoug Ambrisko * inbound_low_queue_port and inbound_high_queue_port. 2952665484d8SDoug Ambrisko */ 29538e727371SKashyap D Desai void 29548e727371SKashyap D Desai mrsas_fire_cmd(struct mrsas_softc *sc, u_int32_t req_desc_lo, 2955665484d8SDoug Ambrisko u_int32_t req_desc_hi) 2956665484d8SDoug Ambrisko { 2957b518670cSKashyap D Desai if (sc->atomic_desc_support) 2958b518670cSKashyap D Desai mrsas_write_reg(sc, offsetof(mrsas_reg_set, inbound_single_queue_port), 2959*e34a057cSAlfredo Dal'Ava Junior le32toh(req_desc_lo)); 2960b518670cSKashyap D Desai else 2961b518670cSKashyap D Desai mrsas_write_64bit_req_desc(sc, req_desc_lo, req_desc_hi); 2962665484d8SDoug Ambrisko } 2963665484d8SDoug Ambrisko 29648e727371SKashyap D Desai /* 29658e727371SKashyap D Desai * mrsas_transition_to_ready: Move FW to Ready state input: 29668e727371SKashyap D Desai * Adapter instance soft state 2967665484d8SDoug Ambrisko * 29688e727371SKashyap D Desai * During the initialization, FW passes can potentially be in any one of several 29698e727371SKashyap D Desai * possible states. If the FW in operational, waiting-for-handshake states, 29708e727371SKashyap D Desai * driver must take steps to bring it to ready state. Otherwise, it has to 29718e727371SKashyap D Desai * wait for the ready state. 2972665484d8SDoug Ambrisko */ 29738e727371SKashyap D Desai int 29748e727371SKashyap D Desai mrsas_transition_to_ready(struct mrsas_softc *sc, int ocr) 2975665484d8SDoug Ambrisko { 2976665484d8SDoug Ambrisko int i; 2977665484d8SDoug Ambrisko u_int8_t max_wait; 2978665484d8SDoug Ambrisko u_int32_t val, fw_state; 2979665484d8SDoug Ambrisko u_int32_t cur_state; 2980665484d8SDoug Ambrisko u_int32_t abs_state, curr_abs_state; 2981665484d8SDoug Ambrisko 2982e315cf4dSKashyap D Desai val = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, outbound_scratch_pad)); 2983665484d8SDoug Ambrisko fw_state = val & MFI_STATE_MASK; 2984665484d8SDoug Ambrisko max_wait = MRSAS_RESET_WAIT_TIME; 2985665484d8SDoug Ambrisko 2986665484d8SDoug Ambrisko if (fw_state != MFI_STATE_READY) 2987665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Waiting for FW to come to ready state\n"); 2988665484d8SDoug Ambrisko 2989665484d8SDoug Ambrisko while (fw_state != MFI_STATE_READY) { 2990e315cf4dSKashyap D Desai abs_state = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, outbound_scratch_pad)); 2991665484d8SDoug Ambrisko switch (fw_state) { 2992665484d8SDoug Ambrisko case MFI_STATE_FAULT: 2993665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "FW is in FAULT state!!\n"); 2994665484d8SDoug Ambrisko if (ocr) { 2995665484d8SDoug Ambrisko cur_state = MFI_STATE_FAULT; 2996665484d8SDoug Ambrisko break; 29978e727371SKashyap D Desai } else 2998665484d8SDoug Ambrisko return -ENODEV; 2999665484d8SDoug Ambrisko case MFI_STATE_WAIT_HANDSHAKE: 3000665484d8SDoug Ambrisko /* Set the CLR bit in inbound doorbell */ 3001665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell), 3002665484d8SDoug Ambrisko MFI_INIT_CLEAR_HANDSHAKE | MFI_INIT_HOTPLUG); 3003665484d8SDoug Ambrisko cur_state = MFI_STATE_WAIT_HANDSHAKE; 3004665484d8SDoug Ambrisko break; 3005665484d8SDoug Ambrisko case MFI_STATE_BOOT_MESSAGE_PENDING: 3006665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell), 3007665484d8SDoug Ambrisko MFI_INIT_HOTPLUG); 3008665484d8SDoug Ambrisko cur_state = MFI_STATE_BOOT_MESSAGE_PENDING; 3009665484d8SDoug Ambrisko break; 3010665484d8SDoug Ambrisko case MFI_STATE_OPERATIONAL: 30118e727371SKashyap D Desai /* 30128e727371SKashyap D Desai * Bring it to READY state; assuming max wait 10 30138e727371SKashyap D Desai * secs 30148e727371SKashyap D Desai */ 3015665484d8SDoug Ambrisko mrsas_disable_intr(sc); 3016665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell), MFI_RESET_FLAGS); 3017665484d8SDoug Ambrisko for (i = 0; i < max_wait * 1000; i++) { 3018e315cf4dSKashyap D Desai if (mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, doorbell)) & 1) 3019665484d8SDoug Ambrisko DELAY(1000); 3020665484d8SDoug Ambrisko else 3021665484d8SDoug Ambrisko break; 3022665484d8SDoug Ambrisko } 3023665484d8SDoug Ambrisko cur_state = MFI_STATE_OPERATIONAL; 3024665484d8SDoug Ambrisko break; 3025665484d8SDoug Ambrisko case MFI_STATE_UNDEFINED: 30268e727371SKashyap D Desai /* 30278e727371SKashyap D Desai * This state should not last for more than 2 30288e727371SKashyap D Desai * seconds 30298e727371SKashyap D Desai */ 3030665484d8SDoug Ambrisko cur_state = MFI_STATE_UNDEFINED; 3031665484d8SDoug Ambrisko break; 3032665484d8SDoug Ambrisko case MFI_STATE_BB_INIT: 3033665484d8SDoug Ambrisko cur_state = MFI_STATE_BB_INIT; 3034665484d8SDoug Ambrisko break; 3035665484d8SDoug Ambrisko case MFI_STATE_FW_INIT: 3036665484d8SDoug Ambrisko cur_state = MFI_STATE_FW_INIT; 3037665484d8SDoug Ambrisko break; 3038665484d8SDoug Ambrisko case MFI_STATE_FW_INIT_2: 3039665484d8SDoug Ambrisko cur_state = MFI_STATE_FW_INIT_2; 3040665484d8SDoug Ambrisko break; 3041665484d8SDoug Ambrisko case MFI_STATE_DEVICE_SCAN: 3042665484d8SDoug Ambrisko cur_state = MFI_STATE_DEVICE_SCAN; 3043665484d8SDoug Ambrisko break; 3044665484d8SDoug Ambrisko case MFI_STATE_FLUSH_CACHE: 3045665484d8SDoug Ambrisko cur_state = MFI_STATE_FLUSH_CACHE; 3046665484d8SDoug Ambrisko break; 3047665484d8SDoug Ambrisko default: 3048665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Unknown state 0x%x\n", fw_state); 3049665484d8SDoug Ambrisko return -ENODEV; 3050665484d8SDoug Ambrisko } 3051665484d8SDoug Ambrisko 3052665484d8SDoug Ambrisko /* 3053665484d8SDoug Ambrisko * The cur_state should not last for more than max_wait secs 3054665484d8SDoug Ambrisko */ 3055665484d8SDoug Ambrisko for (i = 0; i < (max_wait * 1000); i++) { 3056e315cf4dSKashyap D Desai fw_state = (mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, 3057665484d8SDoug Ambrisko outbound_scratch_pad)) & MFI_STATE_MASK); 3058e315cf4dSKashyap D Desai curr_abs_state = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, 3059665484d8SDoug Ambrisko outbound_scratch_pad)); 3060665484d8SDoug Ambrisko if (abs_state == curr_abs_state) 3061665484d8SDoug Ambrisko DELAY(1000); 3062665484d8SDoug Ambrisko else 3063665484d8SDoug Ambrisko break; 3064665484d8SDoug Ambrisko } 3065665484d8SDoug Ambrisko 3066665484d8SDoug Ambrisko /* 3067665484d8SDoug Ambrisko * Return error if fw_state hasn't changed after max_wait 3068665484d8SDoug Ambrisko */ 3069665484d8SDoug Ambrisko if (curr_abs_state == abs_state) { 3070665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "FW state [%d] hasn't changed " 3071665484d8SDoug Ambrisko "in %d secs\n", fw_state, max_wait); 3072665484d8SDoug Ambrisko return -ENODEV; 3073665484d8SDoug Ambrisko } 3074665484d8SDoug Ambrisko } 3075665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "FW now in Ready state\n"); 3076665484d8SDoug Ambrisko return 0; 3077665484d8SDoug Ambrisko } 3078665484d8SDoug Ambrisko 30798e727371SKashyap D Desai /* 3080665484d8SDoug Ambrisko * mrsas_get_mfi_cmd: Get a cmd from free command pool 3081665484d8SDoug Ambrisko * input: Adapter soft state 3082665484d8SDoug Ambrisko * 3083665484d8SDoug Ambrisko * This function removes an MFI command from the command list. 3084665484d8SDoug Ambrisko */ 30858e727371SKashyap D Desai struct mrsas_mfi_cmd * 30868e727371SKashyap D Desai mrsas_get_mfi_cmd(struct mrsas_softc *sc) 3087665484d8SDoug Ambrisko { 3088665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd = NULL; 3089665484d8SDoug Ambrisko 3090665484d8SDoug Ambrisko mtx_lock(&sc->mfi_cmd_pool_lock); 3091665484d8SDoug Ambrisko if (!TAILQ_EMPTY(&sc->mrsas_mfi_cmd_list_head)) { 3092665484d8SDoug Ambrisko cmd = TAILQ_FIRST(&sc->mrsas_mfi_cmd_list_head); 3093665484d8SDoug Ambrisko TAILQ_REMOVE(&sc->mrsas_mfi_cmd_list_head, cmd, next); 3094665484d8SDoug Ambrisko } 3095665484d8SDoug Ambrisko mtx_unlock(&sc->mfi_cmd_pool_lock); 3096665484d8SDoug Ambrisko 3097665484d8SDoug Ambrisko return cmd; 3098665484d8SDoug Ambrisko } 3099665484d8SDoug Ambrisko 31008e727371SKashyap D Desai /* 31018e727371SKashyap D Desai * mrsas_ocr_thread: Thread to handle OCR/Kill Adapter. 3102665484d8SDoug Ambrisko * input: Adapter Context. 3103665484d8SDoug Ambrisko * 31048e727371SKashyap D Desai * This function will check FW status register and flag do_timeout_reset flag. 31058e727371SKashyap D Desai * It will do OCR/Kill adapter if FW is in fault state or IO timed out has 31068e727371SKashyap D Desai * trigger reset. 3107665484d8SDoug Ambrisko */ 3108665484d8SDoug Ambrisko static void 3109665484d8SDoug Ambrisko mrsas_ocr_thread(void *arg) 3110665484d8SDoug Ambrisko { 3111665484d8SDoug Ambrisko struct mrsas_softc *sc; 3112665484d8SDoug Ambrisko u_int32_t fw_status, fw_state; 31138bb601acSKashyap D Desai u_int8_t tm_target_reset_failed = 0; 3114665484d8SDoug Ambrisko 3115665484d8SDoug Ambrisko sc = (struct mrsas_softc *)arg; 3116665484d8SDoug Ambrisko 3117665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_TRACE, "%s\n", __func__); 3118665484d8SDoug Ambrisko sc->ocr_thread_active = 1; 3119665484d8SDoug Ambrisko mtx_lock(&sc->sim_lock); 3120665484d8SDoug Ambrisko for (;;) { 3121665484d8SDoug Ambrisko /* Sleep for 1 second and check the queue status */ 3122665484d8SDoug Ambrisko msleep(&sc->ocr_chan, &sc->sim_lock, PRIBIO, 3123665484d8SDoug Ambrisko "mrsas_ocr", sc->mrsas_fw_fault_check_delay * hz); 3124f0c7594bSKashyap D Desai if (sc->remove_in_progress || 3125f0c7594bSKashyap D Desai sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) { 3126665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 3127f0c7594bSKashyap D Desai "Exit due to %s from %s\n", 3128f0c7594bSKashyap D Desai sc->remove_in_progress ? "Shutdown" : 3129f0c7594bSKashyap D Desai "Hardware critical error", __func__); 3130665484d8SDoug Ambrisko break; 3131665484d8SDoug Ambrisko } 3132e315cf4dSKashyap D Desai fw_status = mrsas_read_reg_with_retries(sc, 3133665484d8SDoug Ambrisko offsetof(mrsas_reg_set, outbound_scratch_pad)); 3134665484d8SDoug Ambrisko fw_state = fw_status & MFI_STATE_MASK; 31358bb601acSKashyap D Desai if (fw_state == MFI_STATE_FAULT || sc->do_timedout_reset || 31368bb601acSKashyap D Desai mrsas_atomic_read(&sc->target_reset_outstanding)) { 31378bb601acSKashyap D Desai /* First, freeze further IOs to come to the SIM */ 31388bb601acSKashyap D Desai mrsas_xpt_freeze(sc); 31398bb601acSKashyap D Desai 31408bb601acSKashyap D Desai /* If this is an IO timeout then go for target reset */ 31418bb601acSKashyap D Desai if (mrsas_atomic_read(&sc->target_reset_outstanding)) { 31428bb601acSKashyap D Desai device_printf(sc->mrsas_dev, "Initiating Target RESET " 31438bb601acSKashyap D Desai "because of SCSI IO timeout!\n"); 31448bb601acSKashyap D Desai 31458bb601acSKashyap D Desai /* Let the remaining IOs to complete */ 31468bb601acSKashyap D Desai msleep(&sc->ocr_chan, &sc->sim_lock, PRIBIO, 31478bb601acSKashyap D Desai "mrsas_reset_targets", 5 * hz); 31488bb601acSKashyap D Desai 31498bb601acSKashyap D Desai /* Try to reset the target device */ 31508bb601acSKashyap D Desai if (mrsas_reset_targets(sc) == FAIL) 31518bb601acSKashyap D Desai tm_target_reset_failed = 1; 31528bb601acSKashyap D Desai } 31538bb601acSKashyap D Desai 31548bb601acSKashyap D Desai /* If this is a DCMD timeout or FW fault, 31558bb601acSKashyap D Desai * then go for controller reset 31568bb601acSKashyap D Desai */ 31578bb601acSKashyap D Desai if (fw_state == MFI_STATE_FAULT || tm_target_reset_failed || 31588bb601acSKashyap D Desai (sc->do_timedout_reset == MFI_DCMD_TIMEOUT_OCR)) { 31598bb601acSKashyap D Desai if (tm_target_reset_failed) 31608bb601acSKashyap D Desai device_printf(sc->mrsas_dev, "Initiaiting OCR because of " 31618bb601acSKashyap D Desai "TM FAILURE!\n"); 31628bb601acSKashyap D Desai else 31638bb601acSKashyap D Desai device_printf(sc->mrsas_dev, "Initiaiting OCR " 31648bb601acSKashyap D Desai "because of %s!\n", sc->do_timedout_reset ? 31658bb601acSKashyap D Desai "DCMD IO Timeout" : "FW fault"); 31668bb601acSKashyap D Desai 3167665484d8SDoug Ambrisko mtx_lock_spin(&sc->ioctl_lock); 3168665484d8SDoug Ambrisko sc->reset_in_progress = 1; 3169665484d8SDoug Ambrisko mtx_unlock_spin(&sc->ioctl_lock); 31708bb601acSKashyap D Desai sc->reset_count++; 31718bb601acSKashyap D Desai 317285c0a961SKashyap D Desai /* 317385c0a961SKashyap D Desai * Wait for the AEN task to be completed if it is running. 317485c0a961SKashyap D Desai */ 317585c0a961SKashyap D Desai mtx_unlock(&sc->sim_lock); 317685c0a961SKashyap D Desai taskqueue_drain(sc->ev_tq, &sc->ev_task); 317785c0a961SKashyap D Desai mtx_lock(&sc->sim_lock); 317885c0a961SKashyap D Desai 317985c0a961SKashyap D Desai taskqueue_block(sc->ev_tq); 31808bb601acSKashyap D Desai /* Try to reset the controller */ 3181f0c7594bSKashyap D Desai mrsas_reset_ctrl(sc, sc->do_timedout_reset); 31828bb601acSKashyap D Desai 3183665484d8SDoug Ambrisko sc->do_timedout_reset = 0; 31848bb601acSKashyap D Desai sc->reset_in_progress = 0; 31858bb601acSKashyap D Desai tm_target_reset_failed = 0; 31868bb601acSKashyap D Desai mrsas_atomic_set(&sc->target_reset_outstanding, 0); 31878bb601acSKashyap D Desai memset(sc->target_reset_pool, 0, 31888bb601acSKashyap D Desai sizeof(sc->target_reset_pool)); 318985c0a961SKashyap D Desai taskqueue_unblock(sc->ev_tq); 31908bb601acSKashyap D Desai } 31918bb601acSKashyap D Desai 31928bb601acSKashyap D Desai /* Now allow IOs to come to the SIM */ 31938bb601acSKashyap D Desai mrsas_xpt_release(sc); 3194665484d8SDoug Ambrisko } 3195665484d8SDoug Ambrisko } 3196665484d8SDoug Ambrisko mtx_unlock(&sc->sim_lock); 3197665484d8SDoug Ambrisko sc->ocr_thread_active = 0; 3198665484d8SDoug Ambrisko mrsas_kproc_exit(0); 3199665484d8SDoug Ambrisko } 3200665484d8SDoug Ambrisko 32018e727371SKashyap D Desai /* 32028e727371SKashyap D Desai * mrsas_reset_reply_desc: Reset Reply descriptor as part of OCR. 3203665484d8SDoug Ambrisko * input: Adapter Context. 3204665484d8SDoug Ambrisko * 32058e727371SKashyap D Desai * This function will clear reply descriptor so that post OCR driver and FW will 32068e727371SKashyap D Desai * lost old history. 3207665484d8SDoug Ambrisko */ 32088e727371SKashyap D Desai void 32098e727371SKashyap D Desai mrsas_reset_reply_desc(struct mrsas_softc *sc) 3210665484d8SDoug Ambrisko { 3211d18d1b47SKashyap D Desai int i, count; 3212665484d8SDoug Ambrisko pMpi2ReplyDescriptorsUnion_t reply_desc; 3213665484d8SDoug Ambrisko 3214d18d1b47SKashyap D Desai count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 3215d18d1b47SKashyap D Desai for (i = 0; i < count; i++) 3216d18d1b47SKashyap D Desai sc->last_reply_idx[i] = 0; 3217d18d1b47SKashyap D Desai 3218665484d8SDoug Ambrisko reply_desc = sc->reply_desc_mem; 3219665484d8SDoug Ambrisko for (i = 0; i < sc->reply_q_depth; i++, reply_desc++) { 3220665484d8SDoug Ambrisko reply_desc->Words = MRSAS_ULONG_MAX; 3221665484d8SDoug Ambrisko } 3222665484d8SDoug Ambrisko } 3223665484d8SDoug Ambrisko 32248e727371SKashyap D Desai /* 32258e727371SKashyap D Desai * mrsas_reset_ctrl: Core function to OCR/Kill adapter. 3226665484d8SDoug Ambrisko * input: Adapter Context. 3227665484d8SDoug Ambrisko * 32288e727371SKashyap D Desai * This function will run from thread context so that it can sleep. 1. Do not 32298e727371SKashyap D Desai * handle OCR if FW is in HW critical error. 2. Wait for outstanding command 32308e727371SKashyap D Desai * to complete for 180 seconds. 3. If #2 does not find any outstanding 32318e727371SKashyap D Desai * command Controller is in working state, so skip OCR. Otherwise, do 32328e727371SKashyap D Desai * OCR/kill Adapter based on flag disableOnlineCtrlReset. 4. Start of the 32338e727371SKashyap D Desai * OCR, return all SCSI command back to CAM layer which has ccb_ptr. 5. Post 3234453130d9SPedro F. Giffuni * OCR, Re-fire Management command and move Controller to Operation state. 3235665484d8SDoug Ambrisko */ 32368e727371SKashyap D Desai int 3237f0c7594bSKashyap D Desai mrsas_reset_ctrl(struct mrsas_softc *sc, u_int8_t reset_reason) 3238665484d8SDoug Ambrisko { 3239665484d8SDoug Ambrisko int retval = SUCCESS, i, j, retry = 0; 3240665484d8SDoug Ambrisko u_int32_t host_diag, abs_state, status_reg, reset_adapter; 3241665484d8SDoug Ambrisko union ccb *ccb; 3242665484d8SDoug Ambrisko struct mrsas_mfi_cmd *mfi_cmd; 3243665484d8SDoug Ambrisko struct mrsas_mpt_cmd *mpt_cmd; 3244f0c7594bSKashyap D Desai union mrsas_evt_class_locale class_locale; 32452d53b485SKashyap D Desai MRSAS_REQUEST_DESCRIPTOR_UNION *req_desc; 3246665484d8SDoug Ambrisko 3247665484d8SDoug Ambrisko if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) { 3248665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, 3249665484d8SDoug Ambrisko "mrsas: Hardware critical error, returning FAIL.\n"); 3250665484d8SDoug Ambrisko return FAIL; 3251665484d8SDoug Ambrisko } 3252f5fb2237SKashyap D Desai mrsas_set_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags); 3253665484d8SDoug Ambrisko sc->adprecovery = MRSAS_ADPRESET_SM_INFAULT; 3254665484d8SDoug Ambrisko mrsas_disable_intr(sc); 3255f0c7594bSKashyap D Desai msleep(&sc->ocr_chan, &sc->sim_lock, PRIBIO, "mrsas_ocr", 3256f0c7594bSKashyap D Desai sc->mrsas_fw_fault_check_delay * hz); 3257665484d8SDoug Ambrisko 3258665484d8SDoug Ambrisko /* First try waiting for commands to complete */ 3259f0c7594bSKashyap D Desai if (mrsas_wait_for_outstanding(sc, reset_reason)) { 3260665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 3261665484d8SDoug Ambrisko "resetting adapter from %s.\n", 3262665484d8SDoug Ambrisko __func__); 3263665484d8SDoug Ambrisko /* Now return commands back to the CAM layer */ 32645b2490f8SKashyap D Desai mtx_unlock(&sc->sim_lock); 3265665484d8SDoug Ambrisko for (i = 0; i < sc->max_fw_cmds; i++) { 3266665484d8SDoug Ambrisko mpt_cmd = sc->mpt_cmd_list[i]; 32672a1d3bcdSKashyap D Desai 32682a1d3bcdSKashyap D Desai if (mpt_cmd->peer_cmd) { 32692a1d3bcdSKashyap D Desai mrsas_dprint(sc, MRSAS_OCR, 32702a1d3bcdSKashyap D Desai "R1 FP command [%d] - (mpt_cmd) %p, (peer_cmd) %p\n", 32712a1d3bcdSKashyap D Desai i, mpt_cmd, mpt_cmd->peer_cmd); 32722a1d3bcdSKashyap D Desai } 32732a1d3bcdSKashyap D Desai 3274665484d8SDoug Ambrisko if (mpt_cmd->ccb_ptr) { 32752a1d3bcdSKashyap D Desai if (mpt_cmd->callout_owner) { 3276665484d8SDoug Ambrisko ccb = (union ccb *)(mpt_cmd->ccb_ptr); 3277665484d8SDoug Ambrisko ccb->ccb_h.status = CAM_SCSI_BUS_RESET; 3278665484d8SDoug Ambrisko mrsas_cmd_done(sc, mpt_cmd); 32792a1d3bcdSKashyap D Desai } else { 32802a1d3bcdSKashyap D Desai mpt_cmd->ccb_ptr = NULL; 32812a1d3bcdSKashyap D Desai mrsas_release_mpt_cmd(mpt_cmd); 3282665484d8SDoug Ambrisko } 3283665484d8SDoug Ambrisko } 32842a1d3bcdSKashyap D Desai } 32852a1d3bcdSKashyap D Desai 32862a1d3bcdSKashyap D Desai mrsas_atomic_set(&sc->fw_outstanding, 0); 32872a1d3bcdSKashyap D Desai 32885b2490f8SKashyap D Desai mtx_lock(&sc->sim_lock); 3289665484d8SDoug Ambrisko 3290e315cf4dSKashyap D Desai status_reg = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, 3291665484d8SDoug Ambrisko outbound_scratch_pad)); 3292665484d8SDoug Ambrisko abs_state = status_reg & MFI_STATE_MASK; 3293665484d8SDoug Ambrisko reset_adapter = status_reg & MFI_RESET_ADAPTER; 3294665484d8SDoug Ambrisko if (sc->disableOnlineCtrlReset || 3295665484d8SDoug Ambrisko (abs_state == MFI_STATE_FAULT && !reset_adapter)) { 3296665484d8SDoug Ambrisko /* Reset not supported, kill adapter */ 3297665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "Reset not supported, killing adapter.\n"); 3298665484d8SDoug Ambrisko mrsas_kill_hba(sc); 3299665484d8SDoug Ambrisko retval = FAIL; 3300665484d8SDoug Ambrisko goto out; 3301665484d8SDoug Ambrisko } 3302665484d8SDoug Ambrisko /* Now try to reset the chip */ 3303665484d8SDoug Ambrisko for (i = 0; i < MRSAS_FUSION_MAX_RESET_TRIES; i++) { 3304665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 3305665484d8SDoug Ambrisko MPI2_WRSEQ_FLUSH_KEY_VALUE); 3306665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 3307665484d8SDoug Ambrisko MPI2_WRSEQ_1ST_KEY_VALUE); 3308665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 3309665484d8SDoug Ambrisko MPI2_WRSEQ_2ND_KEY_VALUE); 3310665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 3311665484d8SDoug Ambrisko MPI2_WRSEQ_3RD_KEY_VALUE); 3312665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 3313665484d8SDoug Ambrisko MPI2_WRSEQ_4TH_KEY_VALUE); 3314665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 3315665484d8SDoug Ambrisko MPI2_WRSEQ_5TH_KEY_VALUE); 3316665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset), 3317665484d8SDoug Ambrisko MPI2_WRSEQ_6TH_KEY_VALUE); 3318665484d8SDoug Ambrisko 3319665484d8SDoug Ambrisko /* Check that the diag write enable (DRWE) bit is on */ 3320e315cf4dSKashyap D Desai host_diag = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, 3321665484d8SDoug Ambrisko fusion_host_diag)); 3322665484d8SDoug Ambrisko retry = 0; 3323665484d8SDoug Ambrisko while (!(host_diag & HOST_DIAG_WRITE_ENABLE)) { 3324665484d8SDoug Ambrisko DELAY(100 * 1000); 3325e315cf4dSKashyap D Desai host_diag = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, 3326665484d8SDoug Ambrisko fusion_host_diag)); 3327665484d8SDoug Ambrisko if (retry++ == 100) { 3328665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 3329665484d8SDoug Ambrisko "Host diag unlock failed!\n"); 3330665484d8SDoug Ambrisko break; 3331665484d8SDoug Ambrisko } 3332665484d8SDoug Ambrisko } 3333665484d8SDoug Ambrisko if (!(host_diag & HOST_DIAG_WRITE_ENABLE)) 3334665484d8SDoug Ambrisko continue; 3335665484d8SDoug Ambrisko 3336665484d8SDoug Ambrisko /* Send chip reset command */ 3337665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_host_diag), 3338665484d8SDoug Ambrisko host_diag | HOST_DIAG_RESET_ADAPTER); 3339665484d8SDoug Ambrisko DELAY(3000 * 1000); 3340665484d8SDoug Ambrisko 3341665484d8SDoug Ambrisko /* Make sure reset adapter bit is cleared */ 3342e315cf4dSKashyap D Desai host_diag = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, 3343665484d8SDoug Ambrisko fusion_host_diag)); 3344665484d8SDoug Ambrisko retry = 0; 3345665484d8SDoug Ambrisko while (host_diag & HOST_DIAG_RESET_ADAPTER) { 3346665484d8SDoug Ambrisko DELAY(100 * 1000); 3347e315cf4dSKashyap D Desai host_diag = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, 3348665484d8SDoug Ambrisko fusion_host_diag)); 3349665484d8SDoug Ambrisko if (retry++ == 1000) { 3350665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 3351665484d8SDoug Ambrisko "Diag reset adapter never cleared!\n"); 3352665484d8SDoug Ambrisko break; 3353665484d8SDoug Ambrisko } 3354665484d8SDoug Ambrisko } 3355665484d8SDoug Ambrisko if (host_diag & HOST_DIAG_RESET_ADAPTER) 3356665484d8SDoug Ambrisko continue; 3357665484d8SDoug Ambrisko 3358e315cf4dSKashyap D Desai abs_state = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, 3359665484d8SDoug Ambrisko outbound_scratch_pad)) & MFI_STATE_MASK; 3360665484d8SDoug Ambrisko retry = 0; 3361665484d8SDoug Ambrisko 3362665484d8SDoug Ambrisko while ((abs_state <= MFI_STATE_FW_INIT) && (retry++ < 1000)) { 3363665484d8SDoug Ambrisko DELAY(100 * 1000); 3364e315cf4dSKashyap D Desai abs_state = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, 3365665484d8SDoug Ambrisko outbound_scratch_pad)) & MFI_STATE_MASK; 3366665484d8SDoug Ambrisko } 3367665484d8SDoug Ambrisko if (abs_state <= MFI_STATE_FW_INIT) { 3368665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "firmware state < MFI_STATE_FW_INIT," 3369665484d8SDoug Ambrisko " state = 0x%x\n", abs_state); 3370665484d8SDoug Ambrisko continue; 3371665484d8SDoug Ambrisko } 3372665484d8SDoug Ambrisko /* Wait for FW to become ready */ 3373665484d8SDoug Ambrisko if (mrsas_transition_to_ready(sc, 1)) { 3374665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 3375665484d8SDoug Ambrisko "mrsas: Failed to transition controller to ready.\n"); 3376665484d8SDoug Ambrisko continue; 3377665484d8SDoug Ambrisko } 3378665484d8SDoug Ambrisko mrsas_reset_reply_desc(sc); 3379665484d8SDoug Ambrisko if (mrsas_ioc_init(sc)) { 3380665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "mrsas_ioc_init() failed!\n"); 3381665484d8SDoug Ambrisko continue; 3382665484d8SDoug Ambrisko } 3383665484d8SDoug Ambrisko for (j = 0; j < sc->max_fw_cmds; j++) { 3384665484d8SDoug Ambrisko mpt_cmd = sc->mpt_cmd_list[j]; 3385665484d8SDoug Ambrisko if (mpt_cmd->sync_cmd_idx != (u_int32_t)MRSAS_ULONG_MAX) { 3386665484d8SDoug Ambrisko mfi_cmd = sc->mfi_cmd_list[mpt_cmd->sync_cmd_idx]; 33872d53b485SKashyap D Desai /* If not an IOCTL then release the command else re-fire */ 33882d53b485SKashyap D Desai if (!mfi_cmd->sync_cmd) { 3389665484d8SDoug Ambrisko mrsas_release_mfi_cmd(mfi_cmd); 33902d53b485SKashyap D Desai } else { 33912d53b485SKashyap D Desai req_desc = mrsas_get_request_desc(sc, 33922d53b485SKashyap D Desai mfi_cmd->cmd_id.context.smid - 1); 33932d53b485SKashyap D Desai mrsas_dprint(sc, MRSAS_OCR, 33942d53b485SKashyap D Desai "Re-fire command DCMD opcode 0x%x index %d\n ", 33952d53b485SKashyap D Desai mfi_cmd->frame->dcmd.opcode, j); 33962d53b485SKashyap D Desai if (!req_desc) 33972d53b485SKashyap D Desai device_printf(sc->mrsas_dev, 33982d53b485SKashyap D Desai "Cannot build MPT cmd.\n"); 33992d53b485SKashyap D Desai else 34002d53b485SKashyap D Desai mrsas_fire_cmd(sc, req_desc->addr.u.low, 34012d53b485SKashyap D Desai req_desc->addr.u.high); 34022d53b485SKashyap D Desai } 3403665484d8SDoug Ambrisko } 3404665484d8SDoug Ambrisko } 3405f0c7594bSKashyap D Desai 3406665484d8SDoug Ambrisko /* Reset load balance info */ 3407665484d8SDoug Ambrisko memset(sc->load_balance_info, 0, 34084799d485SKashyap D Desai sizeof(LD_LOAD_BALANCE_INFO) * MAX_LOGICAL_DRIVES_EXT); 3409665484d8SDoug Ambrisko 3410af51c29fSKashyap D Desai if (mrsas_get_ctrl_info(sc)) { 3411af51c29fSKashyap D Desai mrsas_kill_hba(sc); 34122f863eb8SKashyap D Desai retval = FAIL; 34132f863eb8SKashyap D Desai goto out; 3414af51c29fSKashyap D Desai } 3415665484d8SDoug Ambrisko if (!mrsas_get_map_info(sc)) 3416665484d8SDoug Ambrisko mrsas_sync_map_info(sc); 3417665484d8SDoug Ambrisko 3418a688fcd0SKashyap D Desai megasas_setup_jbod_map(sc); 3419a688fcd0SKashyap D Desai 34202909aab4SKashyap D Desai if ((sc->is_ventura || sc->is_aero) && sc->streamDetectByLD) { 3421821df4b9SKashyap D Desai for (j = 0; j < MAX_LOGICAL_DRIVES_EXT; ++j) { 3422821df4b9SKashyap D Desai memset(sc->streamDetectByLD[i], 0, sizeof(LD_STREAM_DETECT)); 3423821df4b9SKashyap D Desai sc->streamDetectByLD[i]->mruBitMap = MR_STREAM_BITMAP; 3424821df4b9SKashyap D Desai } 3425821df4b9SKashyap D Desai } 3426821df4b9SKashyap D Desai 34272f863eb8SKashyap D Desai mrsas_clear_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags); 34282f863eb8SKashyap D Desai mrsas_enable_intr(sc); 34292f863eb8SKashyap D Desai sc->adprecovery = MRSAS_HBA_OPERATIONAL; 34302f863eb8SKashyap D Desai 3431f0c7594bSKashyap D Desai /* Register AEN with FW for last sequence number */ 3432f0c7594bSKashyap D Desai class_locale.members.reserved = 0; 3433f0c7594bSKashyap D Desai class_locale.members.locale = MR_EVT_LOCALE_ALL; 3434f0c7594bSKashyap D Desai class_locale.members.class = MR_EVT_CLASS_DEBUG; 3435f0c7594bSKashyap D Desai 34362d53b485SKashyap D Desai mtx_unlock(&sc->sim_lock); 3437f0c7594bSKashyap D Desai if (mrsas_register_aen(sc, sc->last_seq_num, 3438f0c7594bSKashyap D Desai class_locale.word)) { 3439f0c7594bSKashyap D Desai device_printf(sc->mrsas_dev, 3440f0c7594bSKashyap D Desai "ERROR: AEN registration FAILED from OCR !!! " 3441f0c7594bSKashyap D Desai "Further events from the controller cannot be notified." 3442f0c7594bSKashyap D Desai "Either there is some problem in the controller" 3443f0c7594bSKashyap D Desai "or the controller does not support AEN.\n" 3444f0c7594bSKashyap D Desai "Please contact to the SUPPORT TEAM if the problem persists\n"); 3445f0c7594bSKashyap D Desai } 34462d53b485SKashyap D Desai mtx_lock(&sc->sim_lock); 34472d53b485SKashyap D Desai 3448665484d8SDoug Ambrisko /* Adapter reset completed successfully */ 3449665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Reset successful\n"); 3450665484d8SDoug Ambrisko retval = SUCCESS; 3451665484d8SDoug Ambrisko goto out; 3452665484d8SDoug Ambrisko } 3453665484d8SDoug Ambrisko /* Reset failed, kill the adapter */ 3454665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Reset failed, killing adapter.\n"); 3455665484d8SDoug Ambrisko mrsas_kill_hba(sc); 3456665484d8SDoug Ambrisko retval = FAIL; 3457665484d8SDoug Ambrisko } else { 3458f5fb2237SKashyap D Desai mrsas_clear_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags); 3459665484d8SDoug Ambrisko mrsas_enable_intr(sc); 3460665484d8SDoug Ambrisko sc->adprecovery = MRSAS_HBA_OPERATIONAL; 3461665484d8SDoug Ambrisko } 3462665484d8SDoug Ambrisko out: 3463f5fb2237SKashyap D Desai mrsas_clear_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags); 3464665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 3465665484d8SDoug Ambrisko "Reset Exit with %d.\n", retval); 3466665484d8SDoug Ambrisko return retval; 3467665484d8SDoug Ambrisko } 3468665484d8SDoug Ambrisko 34698e727371SKashyap D Desai /* 34708e727371SKashyap D Desai * mrsas_kill_hba: Kill HBA when OCR is not supported 3471665484d8SDoug Ambrisko * input: Adapter Context. 3472665484d8SDoug Ambrisko * 3473665484d8SDoug Ambrisko * This function will kill HBA when OCR is not supported. 3474665484d8SDoug Ambrisko */ 34758e727371SKashyap D Desai void 34768e727371SKashyap D Desai mrsas_kill_hba(struct mrsas_softc *sc) 3477665484d8SDoug Ambrisko { 3478daeed973SKashyap D Desai sc->adprecovery = MRSAS_HW_CRITICAL_ERROR; 3479f0c7594bSKashyap D Desai DELAY(1000 * 1000); 3480665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "%s\n", __func__); 3481665484d8SDoug Ambrisko mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell), 3482665484d8SDoug Ambrisko MFI_STOP_ADP); 3483665484d8SDoug Ambrisko /* Flush */ 3484665484d8SDoug Ambrisko mrsas_read_reg(sc, offsetof(mrsas_reg_set, doorbell)); 3485daeed973SKashyap D Desai mrsas_complete_outstanding_ioctls(sc); 3486daeed973SKashyap D Desai } 3487daeed973SKashyap D Desai 3488daeed973SKashyap D Desai /** 3489daeed973SKashyap D Desai * mrsas_complete_outstanding_ioctls Complete pending IOCTLS after kill_hba 3490daeed973SKashyap D Desai * input: Controller softc 3491daeed973SKashyap D Desai * 3492daeed973SKashyap D Desai * Returns void 3493daeed973SKashyap D Desai */ 3494dbcc81dfSKashyap D Desai void 3495dbcc81dfSKashyap D Desai mrsas_complete_outstanding_ioctls(struct mrsas_softc *sc) 3496dbcc81dfSKashyap D Desai { 3497daeed973SKashyap D Desai int i; 3498daeed973SKashyap D Desai struct mrsas_mpt_cmd *cmd_mpt; 3499daeed973SKashyap D Desai struct mrsas_mfi_cmd *cmd_mfi; 3500daeed973SKashyap D Desai u_int32_t count, MSIxIndex; 3501daeed973SKashyap D Desai 3502daeed973SKashyap D Desai count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 3503daeed973SKashyap D Desai for (i = 0; i < sc->max_fw_cmds; i++) { 3504daeed973SKashyap D Desai cmd_mpt = sc->mpt_cmd_list[i]; 3505daeed973SKashyap D Desai 3506daeed973SKashyap D Desai if (cmd_mpt->sync_cmd_idx != (u_int32_t)MRSAS_ULONG_MAX) { 3507daeed973SKashyap D Desai cmd_mfi = sc->mfi_cmd_list[cmd_mpt->sync_cmd_idx]; 3508daeed973SKashyap D Desai if (cmd_mfi->sync_cmd && cmd_mfi->frame->hdr.cmd != MFI_CMD_ABORT) { 3509daeed973SKashyap D Desai for (MSIxIndex = 0; MSIxIndex < count; MSIxIndex++) 3510daeed973SKashyap D Desai mrsas_complete_mptmfi_passthru(sc, cmd_mfi, 3511503c4f8dSKashyap D Desai cmd_mpt->io_request->RaidContext.raid_context.status); 3512daeed973SKashyap D Desai } 3513daeed973SKashyap D Desai } 3514daeed973SKashyap D Desai } 3515665484d8SDoug Ambrisko } 3516665484d8SDoug Ambrisko 35178e727371SKashyap D Desai /* 35188e727371SKashyap D Desai * mrsas_wait_for_outstanding: Wait for outstanding commands 3519665484d8SDoug Ambrisko * input: Adapter Context. 3520665484d8SDoug Ambrisko * 35218e727371SKashyap D Desai * This function will wait for 180 seconds for outstanding commands to be 35228e727371SKashyap D Desai * completed. 3523665484d8SDoug Ambrisko */ 35248e727371SKashyap D Desai int 3525f0c7594bSKashyap D Desai mrsas_wait_for_outstanding(struct mrsas_softc *sc, u_int8_t check_reason) 3526665484d8SDoug Ambrisko { 3527665484d8SDoug Ambrisko int i, outstanding, retval = 0; 3528d18d1b47SKashyap D Desai u_int32_t fw_state, count, MSIxIndex; 3529d18d1b47SKashyap D Desai 3530665484d8SDoug Ambrisko for (i = 0; i < MRSAS_RESET_WAIT_TIME; i++) { 3531665484d8SDoug Ambrisko if (sc->remove_in_progress) { 3532665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 3533665484d8SDoug Ambrisko "Driver remove or shutdown called.\n"); 3534665484d8SDoug Ambrisko retval = 1; 3535665484d8SDoug Ambrisko goto out; 3536665484d8SDoug Ambrisko } 3537665484d8SDoug Ambrisko /* Check if firmware is in fault state */ 3538e315cf4dSKashyap D Desai fw_state = mrsas_read_reg_with_retries(sc, offsetof(mrsas_reg_set, 3539665484d8SDoug Ambrisko outbound_scratch_pad)) & MFI_STATE_MASK; 3540665484d8SDoug Ambrisko if (fw_state == MFI_STATE_FAULT) { 3541665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 3542665484d8SDoug Ambrisko "Found FW in FAULT state, will reset adapter.\n"); 3543e2e8afb1SKashyap D Desai count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 3544e2e8afb1SKashyap D Desai mtx_unlock(&sc->sim_lock); 3545e2e8afb1SKashyap D Desai for (MSIxIndex = 0; MSIxIndex < count; MSIxIndex++) 3546e2e8afb1SKashyap D Desai mrsas_complete_cmd(sc, MSIxIndex); 3547e2e8afb1SKashyap D Desai mtx_lock(&sc->sim_lock); 3548665484d8SDoug Ambrisko retval = 1; 3549665484d8SDoug Ambrisko goto out; 3550665484d8SDoug Ambrisko } 3551f0c7594bSKashyap D Desai if (check_reason == MFI_DCMD_TIMEOUT_OCR) { 3552f0c7594bSKashyap D Desai mrsas_dprint(sc, MRSAS_OCR, 3553f0c7594bSKashyap D Desai "DCMD IO TIMEOUT detected, will reset adapter.\n"); 3554f0c7594bSKashyap D Desai retval = 1; 3555f0c7594bSKashyap D Desai goto out; 3556f0c7594bSKashyap D Desai } 3557f5fb2237SKashyap D Desai outstanding = mrsas_atomic_read(&sc->fw_outstanding); 3558665484d8SDoug Ambrisko if (!outstanding) 3559665484d8SDoug Ambrisko goto out; 3560665484d8SDoug Ambrisko 3561665484d8SDoug Ambrisko if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) { 3562665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, "[%2d]waiting for %d " 3563665484d8SDoug Ambrisko "commands to complete\n", i, outstanding); 3564d18d1b47SKashyap D Desai count = sc->msix_vectors > 0 ? sc->msix_vectors : 1; 35652d53b485SKashyap D Desai mtx_unlock(&sc->sim_lock); 3566d18d1b47SKashyap D Desai for (MSIxIndex = 0; MSIxIndex < count; MSIxIndex++) 3567d18d1b47SKashyap D Desai mrsas_complete_cmd(sc, MSIxIndex); 35682d53b485SKashyap D Desai mtx_lock(&sc->sim_lock); 3569665484d8SDoug Ambrisko } 3570665484d8SDoug Ambrisko DELAY(1000 * 1000); 3571665484d8SDoug Ambrisko } 3572665484d8SDoug Ambrisko 3573f5fb2237SKashyap D Desai if (mrsas_atomic_read(&sc->fw_outstanding)) { 3574665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_OCR, 3575665484d8SDoug Ambrisko " pending commands remain after waiting," 3576665484d8SDoug Ambrisko " will reset adapter.\n"); 3577665484d8SDoug Ambrisko retval = 1; 3578665484d8SDoug Ambrisko } 3579665484d8SDoug Ambrisko out: 3580665484d8SDoug Ambrisko return retval; 3581665484d8SDoug Ambrisko } 3582665484d8SDoug Ambrisko 35838e727371SKashyap D Desai /* 3584665484d8SDoug Ambrisko * mrsas_release_mfi_cmd: Return a cmd to free command pool 3585665484d8SDoug Ambrisko * input: Command packet for return to free cmd pool 3586665484d8SDoug Ambrisko * 3587731b7561SKashyap D Desai * This function returns the MFI & MPT command to the command list. 3588665484d8SDoug Ambrisko */ 35898e727371SKashyap D Desai void 3590731b7561SKashyap D Desai mrsas_release_mfi_cmd(struct mrsas_mfi_cmd *cmd_mfi) 3591665484d8SDoug Ambrisko { 3592731b7561SKashyap D Desai struct mrsas_softc *sc = cmd_mfi->sc; 3593731b7561SKashyap D Desai struct mrsas_mpt_cmd *cmd_mpt; 3594731b7561SKashyap D Desai 3595665484d8SDoug Ambrisko mtx_lock(&sc->mfi_cmd_pool_lock); 3596731b7561SKashyap D Desai /* 3597731b7561SKashyap D Desai * Release the mpt command (if at all it is allocated 3598731b7561SKashyap D Desai * associated with the mfi command 3599731b7561SKashyap D Desai */ 3600731b7561SKashyap D Desai if (cmd_mfi->cmd_id.context.smid) { 3601731b7561SKashyap D Desai mtx_lock(&sc->mpt_cmd_pool_lock); 3602731b7561SKashyap D Desai /* Get the mpt cmd from mfi cmd frame's smid value */ 3603731b7561SKashyap D Desai cmd_mpt = sc->mpt_cmd_list[cmd_mfi->cmd_id.context.smid-1]; 3604731b7561SKashyap D Desai cmd_mpt->flags = 0; 3605731b7561SKashyap D Desai cmd_mpt->sync_cmd_idx = (u_int32_t)MRSAS_ULONG_MAX; 3606731b7561SKashyap D Desai TAILQ_INSERT_HEAD(&(sc->mrsas_mpt_cmd_list_head), cmd_mpt, next); 3607731b7561SKashyap D Desai mtx_unlock(&sc->mpt_cmd_pool_lock); 3608731b7561SKashyap D Desai } 3609731b7561SKashyap D Desai /* Release the mfi command */ 3610731b7561SKashyap D Desai cmd_mfi->ccb_ptr = NULL; 3611731b7561SKashyap D Desai cmd_mfi->cmd_id.frame_count = 0; 3612731b7561SKashyap D Desai TAILQ_INSERT_HEAD(&(sc->mrsas_mfi_cmd_list_head), cmd_mfi, next); 3613665484d8SDoug Ambrisko mtx_unlock(&sc->mfi_cmd_pool_lock); 3614665484d8SDoug Ambrisko 3615665484d8SDoug Ambrisko return; 3616665484d8SDoug Ambrisko } 3617665484d8SDoug Ambrisko 36188e727371SKashyap D Desai /* 36198e727371SKashyap D Desai * mrsas_get_controller_info: Returns FW's controller structure 3620665484d8SDoug Ambrisko * input: Adapter soft state 3621665484d8SDoug Ambrisko * Controller information structure 3622665484d8SDoug Ambrisko * 36238e727371SKashyap D Desai * Issues an internal command (DCMD) to get the FW's controller structure. This 36248e727371SKashyap D Desai * information is mainly used to find out the maximum IO transfer per command 36258e727371SKashyap D Desai * supported by the FW. 3626665484d8SDoug Ambrisko */ 36278e727371SKashyap D Desai static int 3628af51c29fSKashyap D Desai mrsas_get_ctrl_info(struct mrsas_softc *sc) 3629665484d8SDoug Ambrisko { 3630665484d8SDoug Ambrisko int retcode = 0; 3631f0c7594bSKashyap D Desai u_int8_t do_ocr = 1; 3632665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 3633665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 3634665484d8SDoug Ambrisko 3635665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 3636665484d8SDoug Ambrisko 3637665484d8SDoug Ambrisko if (!cmd) { 3638665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Failed to get a free cmd\n"); 3639665484d8SDoug Ambrisko return -ENOMEM; 3640665484d8SDoug Ambrisko } 3641665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 3642665484d8SDoug Ambrisko 3643665484d8SDoug Ambrisko if (mrsas_alloc_ctlr_info_cmd(sc) != SUCCESS) { 3644665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate get ctlr info cmd\n"); 3645665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 3646665484d8SDoug Ambrisko return -ENOMEM; 3647665484d8SDoug Ambrisko } 3648665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 3649665484d8SDoug Ambrisko 3650665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 3651665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 3652665484d8SDoug Ambrisko dcmd->sge_count = 1; 3653665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 3654665484d8SDoug Ambrisko dcmd->timeout = 0; 3655665484d8SDoug Ambrisko dcmd->pad_0 = 0; 3656*e34a057cSAlfredo Dal'Ava Junior dcmd->data_xfer_len = htole32(sizeof(struct mrsas_ctrl_info)); 3657*e34a057cSAlfredo Dal'Ava Junior dcmd->opcode = htole32(MR_DCMD_CTRL_GET_INFO); 3658*e34a057cSAlfredo Dal'Ava Junior dcmd->sgl.sge32[0].phys_addr = htole32(sc->ctlr_info_phys_addr & 0xFFFFFFFF); 3659*e34a057cSAlfredo Dal'Ava Junior dcmd->sgl.sge32[0].length = htole32(sizeof(struct mrsas_ctrl_info)); 3660665484d8SDoug Ambrisko 36618bc320adSKashyap D Desai if (!sc->mask_interrupts) 36628bc320adSKashyap D Desai retcode = mrsas_issue_blocked_cmd(sc, cmd); 36638bc320adSKashyap D Desai else 3664f0c7594bSKashyap D Desai retcode = mrsas_issue_polled(sc, cmd); 36658bc320adSKashyap D Desai 3666f0c7594bSKashyap D Desai if (retcode == ETIMEDOUT) 3667f0c7594bSKashyap D Desai goto dcmd_timeout; 3668*e34a057cSAlfredo Dal'Ava Junior else { 3669f0c7594bSKashyap D Desai memcpy(sc->ctrl_info, sc->ctlr_info_mem, sizeof(struct mrsas_ctrl_info)); 3670*e34a057cSAlfredo Dal'Ava Junior le32_to_cpus(&sc->ctrl_info->properties.OnOffProperties); 3671*e34a057cSAlfredo Dal'Ava Junior le32_to_cpus(&sc->ctrl_info->adapterOperations2); 3672*e34a057cSAlfredo Dal'Ava Junior le32_to_cpus(&sc->ctrl_info->adapterOperations3); 3673*e34a057cSAlfredo Dal'Ava Junior le16_to_cpus(&sc->ctrl_info->adapterOperations4); 3674*e34a057cSAlfredo Dal'Ava Junior } 3675665484d8SDoug Ambrisko 3676f0c7594bSKashyap D Desai do_ocr = 0; 3677af51c29fSKashyap D Desai mrsas_update_ext_vd_details(sc); 3678af51c29fSKashyap D Desai 3679a688fcd0SKashyap D Desai sc->use_seqnum_jbod_fp = 3680a688fcd0SKashyap D Desai sc->ctrl_info->adapterOperations3.useSeqNumJbodFP; 3681c376f864SKashyap D Desai sc->support_morethan256jbod = 3682c376f864SKashyap D Desai sc->ctrl_info->adapterOperations4.supportPdMapTargetId; 3683c376f864SKashyap D Desai 36848bc320adSKashyap D Desai sc->disableOnlineCtrlReset = 36858bc320adSKashyap D Desai sc->ctrl_info->properties.OnOffProperties.disableOnlineCtrlReset; 3686a688fcd0SKashyap D Desai 3687f0c7594bSKashyap D Desai dcmd_timeout: 3688665484d8SDoug Ambrisko mrsas_free_ctlr_info_cmd(sc); 3689f0c7594bSKashyap D Desai 3690f0c7594bSKashyap D Desai if (do_ocr) 3691f0c7594bSKashyap D Desai sc->do_timedout_reset = MFI_DCMD_TIMEOUT_OCR; 3692f0c7594bSKashyap D Desai 36938bc320adSKashyap D Desai if (!sc->mask_interrupts) 36948bc320adSKashyap D Desai mrsas_release_mfi_cmd(cmd); 36958bc320adSKashyap D Desai 3696665484d8SDoug Ambrisko return (retcode); 3697665484d8SDoug Ambrisko } 3698665484d8SDoug Ambrisko 36998e727371SKashyap D Desai /* 3700af51c29fSKashyap D Desai * mrsas_update_ext_vd_details : Update details w.r.t Extended VD 3701af51c29fSKashyap D Desai * input: 3702af51c29fSKashyap D Desai * sc - Controller's softc 3703af51c29fSKashyap D Desai */ 3704dbcc81dfSKashyap D Desai static void 3705dbcc81dfSKashyap D Desai mrsas_update_ext_vd_details(struct mrsas_softc *sc) 3706af51c29fSKashyap D Desai { 37074ad83576SKashyap D Desai u_int32_t ventura_map_sz = 0; 3708af51c29fSKashyap D Desai sc->max256vdSupport = 3709af51c29fSKashyap D Desai sc->ctrl_info->adapterOperations3.supportMaxExtLDs; 37104ad83576SKashyap D Desai 3711af51c29fSKashyap D Desai /* Below is additional check to address future FW enhancement */ 3712af51c29fSKashyap D Desai if (sc->ctrl_info->max_lds > 64) 3713af51c29fSKashyap D Desai sc->max256vdSupport = 1; 3714af51c29fSKashyap D Desai 3715af51c29fSKashyap D Desai sc->drv_supported_vd_count = MRSAS_MAX_LD_CHANNELS 3716af51c29fSKashyap D Desai * MRSAS_MAX_DEV_PER_CHANNEL; 3717af51c29fSKashyap D Desai sc->drv_supported_pd_count = MRSAS_MAX_PD_CHANNELS 3718af51c29fSKashyap D Desai * MRSAS_MAX_DEV_PER_CHANNEL; 3719af51c29fSKashyap D Desai if (sc->max256vdSupport) { 3720af51c29fSKashyap D Desai sc->fw_supported_vd_count = MAX_LOGICAL_DRIVES_EXT; 3721af51c29fSKashyap D Desai sc->fw_supported_pd_count = MAX_PHYSICAL_DEVICES; 3722af51c29fSKashyap D Desai } else { 3723af51c29fSKashyap D Desai sc->fw_supported_vd_count = MAX_LOGICAL_DRIVES; 3724af51c29fSKashyap D Desai sc->fw_supported_pd_count = MAX_PHYSICAL_DEVICES; 3725af51c29fSKashyap D Desai } 3726af51c29fSKashyap D Desai 37274ad83576SKashyap D Desai if (sc->maxRaidMapSize) { 37284ad83576SKashyap D Desai ventura_map_sz = sc->maxRaidMapSize * 37294ad83576SKashyap D Desai MR_MIN_MAP_SIZE; 37304ad83576SKashyap D Desai sc->current_map_sz = ventura_map_sz; 37314ad83576SKashyap D Desai sc->max_map_sz = ventura_map_sz; 37324ad83576SKashyap D Desai } else { 3733af51c29fSKashyap D Desai sc->old_map_sz = sizeof(MR_FW_RAID_MAP) + 37344ad83576SKashyap D Desai (sizeof(MR_LD_SPAN_MAP) * (sc->fw_supported_vd_count - 1)); 3735af51c29fSKashyap D Desai sc->new_map_sz = sizeof(MR_FW_RAID_MAP_EXT); 3736af51c29fSKashyap D Desai sc->max_map_sz = max(sc->old_map_sz, sc->new_map_sz); 3737af51c29fSKashyap D Desai if (sc->max256vdSupport) 3738af51c29fSKashyap D Desai sc->current_map_sz = sc->new_map_sz; 3739af51c29fSKashyap D Desai else 3740af51c29fSKashyap D Desai sc->current_map_sz = sc->old_map_sz; 3741af51c29fSKashyap D Desai } 3742af51c29fSKashyap D Desai 37434ad83576SKashyap D Desai sc->drv_map_sz = sizeof(MR_DRV_RAID_MAP_ALL); 37444ad83576SKashyap D Desai #if VD_EXT_DEBUG 37454ad83576SKashyap D Desai device_printf(sc->mrsas_dev, "sc->maxRaidMapSize 0x%x \n", 37464ad83576SKashyap D Desai sc->maxRaidMapSize); 37474ad83576SKashyap D Desai device_printf(sc->mrsas_dev, 37484ad83576SKashyap D Desai "new_map_sz = 0x%x, old_map_sz = 0x%x, " 37494ad83576SKashyap D Desai "ventura_map_sz = 0x%x, current_map_sz = 0x%x " 37504ad83576SKashyap D Desai "fusion->drv_map_sz =0x%x, size of driver raid map 0x%lx \n", 37514ad83576SKashyap D Desai sc->new_map_sz, sc->old_map_sz, ventura_map_sz, 37524ad83576SKashyap D Desai sc->current_map_sz, sc->drv_map_sz, sizeof(MR_DRV_RAID_MAP_ALL)); 37534ad83576SKashyap D Desai #endif 37544ad83576SKashyap D Desai } 37554ad83576SKashyap D Desai 3756af51c29fSKashyap D Desai /* 3757665484d8SDoug Ambrisko * mrsas_alloc_ctlr_info_cmd: Allocates memory for controller info command 3758665484d8SDoug Ambrisko * input: Adapter soft state 3759665484d8SDoug Ambrisko * 3760665484d8SDoug Ambrisko * Allocates DMAable memory for the controller info internal command. 3761665484d8SDoug Ambrisko */ 37628e727371SKashyap D Desai int 37638e727371SKashyap D Desai mrsas_alloc_ctlr_info_cmd(struct mrsas_softc *sc) 3764665484d8SDoug Ambrisko { 3765665484d8SDoug Ambrisko int ctlr_info_size; 3766665484d8SDoug Ambrisko 3767665484d8SDoug Ambrisko /* Allocate get controller info command */ 3768665484d8SDoug Ambrisko ctlr_info_size = sizeof(struct mrsas_ctrl_info); 37698e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 37708e727371SKashyap D Desai 1, 0, 37718e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 37728e727371SKashyap D Desai BUS_SPACE_MAXADDR, 37738e727371SKashyap D Desai NULL, NULL, 37748e727371SKashyap D Desai ctlr_info_size, 37758e727371SKashyap D Desai 1, 37768e727371SKashyap D Desai ctlr_info_size, 37778e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 37788e727371SKashyap D Desai NULL, NULL, 3779665484d8SDoug Ambrisko &sc->ctlr_info_tag)) { 3780665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate ctlr info tag\n"); 3781665484d8SDoug Ambrisko return (ENOMEM); 3782665484d8SDoug Ambrisko } 3783665484d8SDoug Ambrisko if (bus_dmamem_alloc(sc->ctlr_info_tag, (void **)&sc->ctlr_info_mem, 3784665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &sc->ctlr_info_dmamap)) { 3785665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate ctlr info cmd mem\n"); 3786665484d8SDoug Ambrisko return (ENOMEM); 3787665484d8SDoug Ambrisko } 3788665484d8SDoug Ambrisko if (bus_dmamap_load(sc->ctlr_info_tag, sc->ctlr_info_dmamap, 3789665484d8SDoug Ambrisko sc->ctlr_info_mem, ctlr_info_size, mrsas_addr_cb, 3790665484d8SDoug Ambrisko &sc->ctlr_info_phys_addr, BUS_DMA_NOWAIT)) { 3791665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load ctlr info cmd mem\n"); 3792665484d8SDoug Ambrisko return (ENOMEM); 3793665484d8SDoug Ambrisko } 3794665484d8SDoug Ambrisko memset(sc->ctlr_info_mem, 0, ctlr_info_size); 3795665484d8SDoug Ambrisko return (0); 3796665484d8SDoug Ambrisko } 3797665484d8SDoug Ambrisko 37988e727371SKashyap D Desai /* 3799665484d8SDoug Ambrisko * mrsas_free_ctlr_info_cmd: Free memory for controller info command 3800665484d8SDoug Ambrisko * input: Adapter soft state 3801665484d8SDoug Ambrisko * 3802665484d8SDoug Ambrisko * Deallocates memory of the get controller info cmd. 3803665484d8SDoug Ambrisko */ 38048e727371SKashyap D Desai void 38058e727371SKashyap D Desai mrsas_free_ctlr_info_cmd(struct mrsas_softc *sc) 3806665484d8SDoug Ambrisko { 3807665484d8SDoug Ambrisko if (sc->ctlr_info_phys_addr) 3808665484d8SDoug Ambrisko bus_dmamap_unload(sc->ctlr_info_tag, sc->ctlr_info_dmamap); 3809665484d8SDoug Ambrisko if (sc->ctlr_info_mem != NULL) 3810665484d8SDoug Ambrisko bus_dmamem_free(sc->ctlr_info_tag, sc->ctlr_info_mem, sc->ctlr_info_dmamap); 3811665484d8SDoug Ambrisko if (sc->ctlr_info_tag != NULL) 3812665484d8SDoug Ambrisko bus_dma_tag_destroy(sc->ctlr_info_tag); 3813665484d8SDoug Ambrisko } 3814665484d8SDoug Ambrisko 38158e727371SKashyap D Desai /* 3816665484d8SDoug Ambrisko * mrsas_issue_polled: Issues a polling command 3817665484d8SDoug Ambrisko * inputs: Adapter soft state 3818665484d8SDoug Ambrisko * Command packet to be issued 3819665484d8SDoug Ambrisko * 38208e727371SKashyap D Desai * This function is for posting of internal commands to Firmware. MFI requires 38218e727371SKashyap D Desai * the cmd_status to be set to 0xFF before posting. The maximun wait time of 38228e727371SKashyap D Desai * the poll response timer is 180 seconds. 3823665484d8SDoug Ambrisko */ 38248e727371SKashyap D Desai int 38258e727371SKashyap D Desai mrsas_issue_polled(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 3826665484d8SDoug Ambrisko { 3827665484d8SDoug Ambrisko struct mrsas_header *frame_hdr = &cmd->frame->hdr; 3828665484d8SDoug Ambrisko u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME; 3829f0c7594bSKashyap D Desai int i, retcode = SUCCESS; 3830665484d8SDoug Ambrisko 3831665484d8SDoug Ambrisko frame_hdr->cmd_status = 0xFF; 3832*e34a057cSAlfredo Dal'Ava Junior frame_hdr->flags |= htole16(MFI_FRAME_DONT_POST_IN_REPLY_QUEUE); 3833665484d8SDoug Ambrisko 3834665484d8SDoug Ambrisko /* Issue the frame using inbound queue port */ 3835665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)) { 3836665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot issue DCMD internal command.\n"); 3837665484d8SDoug Ambrisko return (1); 3838665484d8SDoug Ambrisko } 3839665484d8SDoug Ambrisko /* 3840665484d8SDoug Ambrisko * Poll response timer to wait for Firmware response. While this 3841665484d8SDoug Ambrisko * timer with the DELAY call could block CPU, the time interval for 3842665484d8SDoug Ambrisko * this is only 1 millisecond. 3843665484d8SDoug Ambrisko */ 3844665484d8SDoug Ambrisko if (frame_hdr->cmd_status == 0xFF) { 3845665484d8SDoug Ambrisko for (i = 0; i < (max_wait * 1000); i++) { 3846665484d8SDoug Ambrisko if (frame_hdr->cmd_status == 0xFF) 3847665484d8SDoug Ambrisko DELAY(1000); 3848665484d8SDoug Ambrisko else 3849665484d8SDoug Ambrisko break; 3850665484d8SDoug Ambrisko } 3851665484d8SDoug Ambrisko } 3852f0c7594bSKashyap D Desai if (frame_hdr->cmd_status == 0xFF) { 3853f0c7594bSKashyap D Desai device_printf(sc->mrsas_dev, "DCMD timed out after %d " 3854f0c7594bSKashyap D Desai "seconds from %s\n", max_wait, __func__); 3855f0c7594bSKashyap D Desai device_printf(sc->mrsas_dev, "DCMD opcode 0x%X\n", 3856f0c7594bSKashyap D Desai cmd->frame->dcmd.opcode); 3857f0c7594bSKashyap D Desai retcode = ETIMEDOUT; 3858665484d8SDoug Ambrisko } 3859665484d8SDoug Ambrisko return (retcode); 3860665484d8SDoug Ambrisko } 3861665484d8SDoug Ambrisko 38628e727371SKashyap D Desai /* 38638e727371SKashyap D Desai * mrsas_issue_dcmd: Issues a MFI Pass thru cmd 38648e727371SKashyap D Desai * input: Adapter soft state mfi cmd pointer 3865665484d8SDoug Ambrisko * 3866665484d8SDoug Ambrisko * This function is called by mrsas_issued_blocked_cmd() and 38678e727371SKashyap D Desai * mrsas_issued_polled(), to build the MPT command and then fire the command 38688e727371SKashyap D Desai * to Firmware. 3869665484d8SDoug Ambrisko */ 3870665484d8SDoug Ambrisko int 3871665484d8SDoug Ambrisko mrsas_issue_dcmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 3872665484d8SDoug Ambrisko { 3873665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION *req_desc; 3874665484d8SDoug Ambrisko 3875665484d8SDoug Ambrisko req_desc = mrsas_build_mpt_cmd(sc, cmd); 3876665484d8SDoug Ambrisko if (!req_desc) { 3877665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot build MPT cmd.\n"); 3878665484d8SDoug Ambrisko return (1); 3879665484d8SDoug Ambrisko } 3880665484d8SDoug Ambrisko mrsas_fire_cmd(sc, req_desc->addr.u.low, req_desc->addr.u.high); 3881665484d8SDoug Ambrisko 3882665484d8SDoug Ambrisko return (0); 3883665484d8SDoug Ambrisko } 3884665484d8SDoug Ambrisko 38858e727371SKashyap D Desai /* 38868e727371SKashyap D Desai * mrsas_build_mpt_cmd: Calls helper function to build Passthru cmd 38878e727371SKashyap D Desai * input: Adapter soft state mfi cmd to build 3888665484d8SDoug Ambrisko * 38898e727371SKashyap D Desai * This function is called by mrsas_issue_cmd() to build the MPT-MFI passthru 38908e727371SKashyap D Desai * command and prepares the MPT command to send to Firmware. 3891665484d8SDoug Ambrisko */ 3892665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION * 3893665484d8SDoug Ambrisko mrsas_build_mpt_cmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 3894665484d8SDoug Ambrisko { 3895665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION *req_desc; 3896665484d8SDoug Ambrisko u_int16_t index; 3897665484d8SDoug Ambrisko 3898665484d8SDoug Ambrisko if (mrsas_build_mptmfi_passthru(sc, cmd)) { 3899665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot build MPT-MFI passthru cmd.\n"); 3900665484d8SDoug Ambrisko return NULL; 3901665484d8SDoug Ambrisko } 3902665484d8SDoug Ambrisko index = cmd->cmd_id.context.smid; 3903665484d8SDoug Ambrisko 3904665484d8SDoug Ambrisko req_desc = mrsas_get_request_desc(sc, index - 1); 3905665484d8SDoug Ambrisko if (!req_desc) 3906665484d8SDoug Ambrisko return NULL; 3907665484d8SDoug Ambrisko 3908665484d8SDoug Ambrisko req_desc->addr.Words = 0; 3909665484d8SDoug Ambrisko req_desc->SCSIIO.RequestFlags = (MPI2_REQ_DESCRIPT_FLAGS_SCSI_IO << MRSAS_REQ_DESCRIPT_FLAGS_TYPE_SHIFT); 3910665484d8SDoug Ambrisko 3911*e34a057cSAlfredo Dal'Ava Junior req_desc->SCSIIO.SMID = htole16(index); 3912665484d8SDoug Ambrisko 3913665484d8SDoug Ambrisko return (req_desc); 3914665484d8SDoug Ambrisko } 3915665484d8SDoug Ambrisko 39168e727371SKashyap D Desai /* 39178e727371SKashyap D Desai * mrsas_build_mptmfi_passthru: Builds a MPT MFI Passthru command 39188e727371SKashyap D Desai * input: Adapter soft state mfi cmd pointer 3919665484d8SDoug Ambrisko * 39208e727371SKashyap D Desai * The MPT command and the io_request are setup as a passthru command. The SGE 39218e727371SKashyap D Desai * chain address is set to frame_phys_addr of the MFI command. 3922665484d8SDoug Ambrisko */ 3923665484d8SDoug Ambrisko u_int8_t 3924665484d8SDoug Ambrisko mrsas_build_mptmfi_passthru(struct mrsas_softc *sc, struct mrsas_mfi_cmd *mfi_cmd) 3925665484d8SDoug Ambrisko { 3926665484d8SDoug Ambrisko MPI25_IEEE_SGE_CHAIN64 *mpi25_ieee_chain; 3927665484d8SDoug Ambrisko PTR_MRSAS_RAID_SCSI_IO_REQUEST io_req; 3928665484d8SDoug Ambrisko struct mrsas_mpt_cmd *mpt_cmd; 3929665484d8SDoug Ambrisko struct mrsas_header *frame_hdr = &mfi_cmd->frame->hdr; 3930665484d8SDoug Ambrisko 3931665484d8SDoug Ambrisko mpt_cmd = mrsas_get_mpt_cmd(sc); 3932665484d8SDoug Ambrisko if (!mpt_cmd) 3933665484d8SDoug Ambrisko return (1); 3934665484d8SDoug Ambrisko 3935665484d8SDoug Ambrisko /* Save the smid. To be used for returning the cmd */ 3936665484d8SDoug Ambrisko mfi_cmd->cmd_id.context.smid = mpt_cmd->index; 3937665484d8SDoug Ambrisko 3938665484d8SDoug Ambrisko mpt_cmd->sync_cmd_idx = mfi_cmd->index; 3939665484d8SDoug Ambrisko 3940665484d8SDoug Ambrisko /* 39418e727371SKashyap D Desai * For cmds where the flag is set, store the flag and check on 39428e727371SKashyap D Desai * completion. For cmds with this flag, don't call 3943665484d8SDoug Ambrisko * mrsas_complete_cmd. 3944665484d8SDoug Ambrisko */ 3945665484d8SDoug Ambrisko 3946*e34a057cSAlfredo Dal'Ava Junior if (frame_hdr->flags & htole16(MFI_FRAME_DONT_POST_IN_REPLY_QUEUE)) 3947665484d8SDoug Ambrisko mpt_cmd->flags = MFI_FRAME_DONT_POST_IN_REPLY_QUEUE; 3948665484d8SDoug Ambrisko 3949665484d8SDoug Ambrisko io_req = mpt_cmd->io_request; 3950665484d8SDoug Ambrisko 39512909aab4SKashyap D Desai if (sc->mrsas_gen3_ctrl || sc->is_ventura || sc->is_aero) { 3952665484d8SDoug Ambrisko pMpi25IeeeSgeChain64_t sgl_ptr_end = (pMpi25IeeeSgeChain64_t)&io_req->SGL; 39538e727371SKashyap D Desai 3954665484d8SDoug Ambrisko sgl_ptr_end += sc->max_sge_in_main_msg - 1; 3955665484d8SDoug Ambrisko sgl_ptr_end->Flags = 0; 3956665484d8SDoug Ambrisko } 3957665484d8SDoug Ambrisko mpi25_ieee_chain = (MPI25_IEEE_SGE_CHAIN64 *) & io_req->SGL.IeeeChain; 3958665484d8SDoug Ambrisko 3959665484d8SDoug Ambrisko io_req->Function = MRSAS_MPI2_FUNCTION_PASSTHRU_IO_REQUEST; 3960665484d8SDoug Ambrisko io_req->SGLOffset0 = offsetof(MRSAS_RAID_SCSI_IO_REQUEST, SGL) / 4; 3961665484d8SDoug Ambrisko io_req->ChainOffset = sc->chain_offset_mfi_pthru; 3962665484d8SDoug Ambrisko 3963*e34a057cSAlfredo Dal'Ava Junior mpi25_ieee_chain->Address = htole64(mfi_cmd->frame_phys_addr); 3964665484d8SDoug Ambrisko 3965665484d8SDoug Ambrisko mpi25_ieee_chain->Flags = IEEE_SGE_FLAGS_CHAIN_ELEMENT | 3966665484d8SDoug Ambrisko MPI2_IEEE_SGE_FLAGS_IOCPLBNTA_ADDR; 3967665484d8SDoug Ambrisko 3968*e34a057cSAlfredo Dal'Ava Junior mpi25_ieee_chain->Length = htole32(sc->max_chain_frame_sz); 3969665484d8SDoug Ambrisko 3970665484d8SDoug Ambrisko return (0); 3971665484d8SDoug Ambrisko } 3972665484d8SDoug Ambrisko 39738e727371SKashyap D Desai /* 39748e727371SKashyap D Desai * mrsas_issue_blocked_cmd: Synchronous wrapper around regular FW cmds 39758e727371SKashyap D Desai * input: Adapter soft state Command to be issued 3976665484d8SDoug Ambrisko * 39778e727371SKashyap D Desai * This function waits on an event for the command to be returned from the ISR. 39788e727371SKashyap D Desai * Max wait time is MRSAS_INTERNAL_CMD_WAIT_TIME secs. Used for issuing 39798e727371SKashyap D Desai * internal and ioctl commands. 3980665484d8SDoug Ambrisko */ 39818e727371SKashyap D Desai int 39828e727371SKashyap D Desai mrsas_issue_blocked_cmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 3983665484d8SDoug Ambrisko { 3984665484d8SDoug Ambrisko u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME; 3985665484d8SDoug Ambrisko unsigned long total_time = 0; 3986f0c7594bSKashyap D Desai int retcode = SUCCESS; 3987665484d8SDoug Ambrisko 3988665484d8SDoug Ambrisko /* Initialize cmd_status */ 3989f0c7594bSKashyap D Desai cmd->cmd_status = 0xFF; 3990665484d8SDoug Ambrisko 3991665484d8SDoug Ambrisko /* Build MPT-MFI command for issue to FW */ 3992665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)) { 3993665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot issue DCMD internal command.\n"); 3994665484d8SDoug Ambrisko return (1); 3995665484d8SDoug Ambrisko } 3996665484d8SDoug Ambrisko sc->chan = (void *)&cmd; 3997665484d8SDoug Ambrisko 3998665484d8SDoug Ambrisko while (1) { 3999f0c7594bSKashyap D Desai if (cmd->cmd_status == 0xFF) { 4000665484d8SDoug Ambrisko tsleep((void *)&sc->chan, 0, "mrsas_sleep", hz); 40018e727371SKashyap D Desai } else 4002665484d8SDoug Ambrisko break; 4003f0c7594bSKashyap D Desai 4004f0c7594bSKashyap D Desai if (!cmd->sync_cmd) { /* cmd->sync will be set for an IOCTL 4005f0c7594bSKashyap D Desai * command */ 4006665484d8SDoug Ambrisko total_time++; 4007665484d8SDoug Ambrisko if (total_time >= max_wait) { 40088e727371SKashyap D Desai device_printf(sc->mrsas_dev, 40098e727371SKashyap D Desai "Internal command timed out after %d seconds.\n", max_wait); 4010665484d8SDoug Ambrisko retcode = 1; 4011665484d8SDoug Ambrisko break; 4012665484d8SDoug Ambrisko } 4013665484d8SDoug Ambrisko } 4014f0c7594bSKashyap D Desai } 4015f0c7594bSKashyap D Desai 4016f0c7594bSKashyap D Desai if (cmd->cmd_status == 0xFF) { 4017f0c7594bSKashyap D Desai device_printf(sc->mrsas_dev, "DCMD timed out after %d " 4018f0c7594bSKashyap D Desai "seconds from %s\n", max_wait, __func__); 4019f0c7594bSKashyap D Desai device_printf(sc->mrsas_dev, "DCMD opcode 0x%X\n", 4020f0c7594bSKashyap D Desai cmd->frame->dcmd.opcode); 4021f0c7594bSKashyap D Desai retcode = ETIMEDOUT; 4022f0c7594bSKashyap D Desai } 4023665484d8SDoug Ambrisko return (retcode); 4024665484d8SDoug Ambrisko } 4025665484d8SDoug Ambrisko 40268e727371SKashyap D Desai /* 40278e727371SKashyap D Desai * mrsas_complete_mptmfi_passthru: Completes a command 40288e727371SKashyap D Desai * input: @sc: Adapter soft state 40298e727371SKashyap D Desai * @cmd: Command to be completed 40308e727371SKashyap D Desai * @status: cmd completion status 4031665484d8SDoug Ambrisko * 40328e727371SKashyap D Desai * This function is called from mrsas_complete_cmd() after an interrupt is 40338e727371SKashyap D Desai * received from Firmware, and io_request->Function is 4034665484d8SDoug Ambrisko * MRSAS_MPI2_FUNCTION_PASSTHRU_IO_REQUEST. 4035665484d8SDoug Ambrisko */ 4036665484d8SDoug Ambrisko void 4037665484d8SDoug Ambrisko mrsas_complete_mptmfi_passthru(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd, 4038665484d8SDoug Ambrisko u_int8_t status) 4039665484d8SDoug Ambrisko { 4040665484d8SDoug Ambrisko struct mrsas_header *hdr = &cmd->frame->hdr; 4041665484d8SDoug Ambrisko u_int8_t cmd_status = cmd->frame->hdr.cmd_status; 4042665484d8SDoug Ambrisko 4043665484d8SDoug Ambrisko /* Reset the retry counter for future re-tries */ 4044665484d8SDoug Ambrisko cmd->retry_for_fw_reset = 0; 4045665484d8SDoug Ambrisko 4046665484d8SDoug Ambrisko if (cmd->ccb_ptr) 4047665484d8SDoug Ambrisko cmd->ccb_ptr = NULL; 4048665484d8SDoug Ambrisko 4049665484d8SDoug Ambrisko switch (hdr->cmd) { 4050665484d8SDoug Ambrisko case MFI_CMD_INVALID: 4051665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "MFI_CMD_INVALID command.\n"); 4052665484d8SDoug Ambrisko break; 4053665484d8SDoug Ambrisko case MFI_CMD_PD_SCSI_IO: 4054665484d8SDoug Ambrisko case MFI_CMD_LD_SCSI_IO: 4055665484d8SDoug Ambrisko /* 4056665484d8SDoug Ambrisko * MFI_CMD_PD_SCSI_IO and MFI_CMD_LD_SCSI_IO could have been 4057665484d8SDoug Ambrisko * issued either through an IO path or an IOCTL path. If it 4058665484d8SDoug Ambrisko * was via IOCTL, we will send it to internal completion. 4059665484d8SDoug Ambrisko */ 4060665484d8SDoug Ambrisko if (cmd->sync_cmd) { 4061665484d8SDoug Ambrisko cmd->sync_cmd = 0; 4062665484d8SDoug Ambrisko mrsas_wakeup(sc, cmd); 4063665484d8SDoug Ambrisko break; 4064665484d8SDoug Ambrisko } 4065665484d8SDoug Ambrisko case MFI_CMD_SMP: 4066665484d8SDoug Ambrisko case MFI_CMD_STP: 4067665484d8SDoug Ambrisko case MFI_CMD_DCMD: 4068665484d8SDoug Ambrisko /* Check for LD map update */ 4069665484d8SDoug Ambrisko if ((cmd->frame->dcmd.opcode == MR_DCMD_LD_MAP_GET_INFO) && 4070665484d8SDoug Ambrisko (cmd->frame->dcmd.mbox.b[1] == 1)) { 4071665484d8SDoug Ambrisko sc->fast_path_io = 0; 4072665484d8SDoug Ambrisko mtx_lock(&sc->raidmap_lock); 4073f0c7594bSKashyap D Desai sc->map_update_cmd = NULL; 4074665484d8SDoug Ambrisko if (cmd_status != 0) { 4075665484d8SDoug Ambrisko if (cmd_status != MFI_STAT_NOT_FOUND) 4076665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "map sync failed, status=%x\n", cmd_status); 4077665484d8SDoug Ambrisko else { 4078665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 4079665484d8SDoug Ambrisko mtx_unlock(&sc->raidmap_lock); 4080665484d8SDoug Ambrisko break; 4081665484d8SDoug Ambrisko } 40828e727371SKashyap D Desai } else 4083665484d8SDoug Ambrisko sc->map_id++; 4084665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 4085665484d8SDoug Ambrisko if (MR_ValidateMapInfo(sc)) 4086665484d8SDoug Ambrisko sc->fast_path_io = 0; 4087665484d8SDoug Ambrisko else 4088665484d8SDoug Ambrisko sc->fast_path_io = 1; 4089665484d8SDoug Ambrisko mrsas_sync_map_info(sc); 4090665484d8SDoug Ambrisko mtx_unlock(&sc->raidmap_lock); 4091665484d8SDoug Ambrisko break; 4092665484d8SDoug Ambrisko } 4093665484d8SDoug Ambrisko if (cmd->frame->dcmd.opcode == MR_DCMD_CTRL_EVENT_GET_INFO || 4094665484d8SDoug Ambrisko cmd->frame->dcmd.opcode == MR_DCMD_CTRL_EVENT_GET) { 4095da011113SKashyap D Desai sc->mrsas_aen_triggered = 0; 4096665484d8SDoug Ambrisko } 4097a688fcd0SKashyap D Desai /* FW has an updated PD sequence */ 4098a688fcd0SKashyap D Desai if ((cmd->frame->dcmd.opcode == 4099a688fcd0SKashyap D Desai MR_DCMD_SYSTEM_PD_MAP_GET_INFO) && 4100a688fcd0SKashyap D Desai (cmd->frame->dcmd.mbox.b[0] == 1)) { 4101a688fcd0SKashyap D Desai mtx_lock(&sc->raidmap_lock); 4102a688fcd0SKashyap D Desai sc->jbod_seq_cmd = NULL; 4103a688fcd0SKashyap D Desai mrsas_release_mfi_cmd(cmd); 4104a688fcd0SKashyap D Desai 4105a688fcd0SKashyap D Desai if (cmd_status == MFI_STAT_OK) { 4106a688fcd0SKashyap D Desai sc->pd_seq_map_id++; 4107a688fcd0SKashyap D Desai /* Re-register a pd sync seq num cmd */ 4108a688fcd0SKashyap D Desai if (megasas_sync_pd_seq_num(sc, true)) 4109a688fcd0SKashyap D Desai sc->use_seqnum_jbod_fp = 0; 4110a688fcd0SKashyap D Desai } else { 4111a688fcd0SKashyap D Desai sc->use_seqnum_jbod_fp = 0; 4112a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, 4113a688fcd0SKashyap D Desai "Jbod map sync failed, status=%x\n", cmd_status); 4114a688fcd0SKashyap D Desai } 4115a688fcd0SKashyap D Desai mtx_unlock(&sc->raidmap_lock); 4116a688fcd0SKashyap D Desai break; 4117a688fcd0SKashyap D Desai } 4118665484d8SDoug Ambrisko /* See if got an event notification */ 4119*e34a057cSAlfredo Dal'Ava Junior if (le32toh(cmd->frame->dcmd.opcode) == MR_DCMD_CTRL_EVENT_WAIT) 4120665484d8SDoug Ambrisko mrsas_complete_aen(sc, cmd); 4121665484d8SDoug Ambrisko else 4122665484d8SDoug Ambrisko mrsas_wakeup(sc, cmd); 4123665484d8SDoug Ambrisko break; 4124665484d8SDoug Ambrisko case MFI_CMD_ABORT: 4125665484d8SDoug Ambrisko /* Command issued to abort another cmd return */ 4126665484d8SDoug Ambrisko mrsas_complete_abort(sc, cmd); 4127665484d8SDoug Ambrisko break; 4128665484d8SDoug Ambrisko default: 4129665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Unknown command completed! [0x%X]\n", hdr->cmd); 4130665484d8SDoug Ambrisko break; 4131665484d8SDoug Ambrisko } 4132665484d8SDoug Ambrisko } 4133665484d8SDoug Ambrisko 41348e727371SKashyap D Desai /* 41358e727371SKashyap D Desai * mrsas_wakeup: Completes an internal command 4136665484d8SDoug Ambrisko * input: Adapter soft state 4137665484d8SDoug Ambrisko * Command to be completed 4138665484d8SDoug Ambrisko * 41398e727371SKashyap D Desai * In mrsas_issue_blocked_cmd(), after a command is issued to Firmware, a wait 41408e727371SKashyap D Desai * timer is started. This function is called from 41418e727371SKashyap D Desai * mrsas_complete_mptmfi_passthru() as it completes the command, to wake up 41428e727371SKashyap D Desai * from the command wait. 4143665484d8SDoug Ambrisko */ 41448e727371SKashyap D Desai void 41458e727371SKashyap D Desai mrsas_wakeup(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 4146665484d8SDoug Ambrisko { 4147665484d8SDoug Ambrisko cmd->cmd_status = cmd->frame->io.cmd_status; 4148665484d8SDoug Ambrisko 4149f0c7594bSKashyap D Desai if (cmd->cmd_status == 0xFF) 4150665484d8SDoug Ambrisko cmd->cmd_status = 0; 4151665484d8SDoug Ambrisko 4152665484d8SDoug Ambrisko sc->chan = (void *)&cmd; 4153665484d8SDoug Ambrisko wakeup_one((void *)&sc->chan); 4154665484d8SDoug Ambrisko return; 4155665484d8SDoug Ambrisko } 4156665484d8SDoug Ambrisko 41578e727371SKashyap D Desai /* 41588e727371SKashyap D Desai * mrsas_shutdown_ctlr: Instructs FW to shutdown the controller input: 41598e727371SKashyap D Desai * Adapter soft state Shutdown/Hibernate 4160665484d8SDoug Ambrisko * 41618e727371SKashyap D Desai * This function issues a DCMD internal command to Firmware to initiate shutdown 41628e727371SKashyap D Desai * of the controller. 4163665484d8SDoug Ambrisko */ 41648e727371SKashyap D Desai static void 41658e727371SKashyap D Desai mrsas_shutdown_ctlr(struct mrsas_softc *sc, u_int32_t opcode) 4166665484d8SDoug Ambrisko { 4167665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 4168665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 4169665484d8SDoug Ambrisko 4170665484d8SDoug Ambrisko if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) 4171665484d8SDoug Ambrisko return; 4172665484d8SDoug Ambrisko 4173665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 4174665484d8SDoug Ambrisko if (!cmd) { 4175665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate for shutdown cmd.\n"); 4176665484d8SDoug Ambrisko return; 4177665484d8SDoug Ambrisko } 4178665484d8SDoug Ambrisko if (sc->aen_cmd) 4179665484d8SDoug Ambrisko mrsas_issue_blocked_abort_cmd(sc, sc->aen_cmd); 4180665484d8SDoug Ambrisko if (sc->map_update_cmd) 4181665484d8SDoug Ambrisko mrsas_issue_blocked_abort_cmd(sc, sc->map_update_cmd); 4182a688fcd0SKashyap D Desai if (sc->jbod_seq_cmd) 4183a688fcd0SKashyap D Desai mrsas_issue_blocked_abort_cmd(sc, sc->jbod_seq_cmd); 4184665484d8SDoug Ambrisko 4185665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 4186665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 4187665484d8SDoug Ambrisko 4188665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 4189665484d8SDoug Ambrisko dcmd->cmd_status = 0x0; 4190665484d8SDoug Ambrisko dcmd->sge_count = 0; 4191665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_NONE; 4192665484d8SDoug Ambrisko dcmd->timeout = 0; 4193665484d8SDoug Ambrisko dcmd->pad_0 = 0; 4194665484d8SDoug Ambrisko dcmd->data_xfer_len = 0; 4195665484d8SDoug Ambrisko dcmd->opcode = opcode; 4196665484d8SDoug Ambrisko 4197665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Preparing to shut down controller.\n"); 4198665484d8SDoug Ambrisko 4199665484d8SDoug Ambrisko mrsas_issue_blocked_cmd(sc, cmd); 4200665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 4201665484d8SDoug Ambrisko 4202665484d8SDoug Ambrisko return; 4203665484d8SDoug Ambrisko } 4204665484d8SDoug Ambrisko 42058e727371SKashyap D Desai /* 42068e727371SKashyap D Desai * mrsas_flush_cache: Requests FW to flush all its caches input: 42078e727371SKashyap D Desai * Adapter soft state 4208665484d8SDoug Ambrisko * 4209665484d8SDoug Ambrisko * This function is issues a DCMD internal command to Firmware to initiate 4210665484d8SDoug Ambrisko * flushing of all caches. 4211665484d8SDoug Ambrisko */ 42128e727371SKashyap D Desai static void 42138e727371SKashyap D Desai mrsas_flush_cache(struct mrsas_softc *sc) 4214665484d8SDoug Ambrisko { 4215665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 4216665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 4217665484d8SDoug Ambrisko 4218665484d8SDoug Ambrisko if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) 4219665484d8SDoug Ambrisko return; 4220665484d8SDoug Ambrisko 4221665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 4222665484d8SDoug Ambrisko if (!cmd) { 4223665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate for flush cache cmd.\n"); 4224665484d8SDoug Ambrisko return; 4225665484d8SDoug Ambrisko } 4226665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 4227665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 4228665484d8SDoug Ambrisko 4229665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 4230665484d8SDoug Ambrisko dcmd->cmd_status = 0x0; 4231665484d8SDoug Ambrisko dcmd->sge_count = 0; 4232665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_NONE; 4233665484d8SDoug Ambrisko dcmd->timeout = 0; 4234665484d8SDoug Ambrisko dcmd->pad_0 = 0; 4235665484d8SDoug Ambrisko dcmd->data_xfer_len = 0; 4236665484d8SDoug Ambrisko dcmd->opcode = MR_DCMD_CTRL_CACHE_FLUSH; 4237665484d8SDoug Ambrisko dcmd->mbox.b[0] = MR_FLUSH_CTRL_CACHE | MR_FLUSH_DISK_CACHE; 4238665484d8SDoug Ambrisko 4239665484d8SDoug Ambrisko mrsas_issue_blocked_cmd(sc, cmd); 4240665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 4241665484d8SDoug Ambrisko 4242665484d8SDoug Ambrisko return; 4243665484d8SDoug Ambrisko } 4244665484d8SDoug Ambrisko 4245a688fcd0SKashyap D Desai int 4246a688fcd0SKashyap D Desai megasas_sync_pd_seq_num(struct mrsas_softc *sc, boolean_t pend) 4247a688fcd0SKashyap D Desai { 4248a688fcd0SKashyap D Desai int retcode = 0; 4249a688fcd0SKashyap D Desai u_int8_t do_ocr = 1; 4250a688fcd0SKashyap D Desai struct mrsas_mfi_cmd *cmd; 4251a688fcd0SKashyap D Desai struct mrsas_dcmd_frame *dcmd; 4252a688fcd0SKashyap D Desai uint32_t pd_seq_map_sz; 4253a688fcd0SKashyap D Desai struct MR_PD_CFG_SEQ_NUM_SYNC *pd_sync; 4254a688fcd0SKashyap D Desai bus_addr_t pd_seq_h; 4255a688fcd0SKashyap D Desai 4256a688fcd0SKashyap D Desai pd_seq_map_sz = sizeof(struct MR_PD_CFG_SEQ_NUM_SYNC) + 4257a688fcd0SKashyap D Desai (sizeof(struct MR_PD_CFG_SEQ) * 4258a688fcd0SKashyap D Desai (MAX_PHYSICAL_DEVICES - 1)); 4259a688fcd0SKashyap D Desai 4260a688fcd0SKashyap D Desai cmd = mrsas_get_mfi_cmd(sc); 4261a688fcd0SKashyap D Desai if (!cmd) { 4262a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, 4263a688fcd0SKashyap D Desai "Cannot alloc for ld map info cmd.\n"); 4264a688fcd0SKashyap D Desai return 1; 4265a688fcd0SKashyap D Desai } 4266a688fcd0SKashyap D Desai dcmd = &cmd->frame->dcmd; 4267a688fcd0SKashyap D Desai 4268a688fcd0SKashyap D Desai pd_sync = (void *)sc->jbodmap_mem[(sc->pd_seq_map_id & 1)]; 4269a688fcd0SKashyap D Desai pd_seq_h = sc->jbodmap_phys_addr[(sc->pd_seq_map_id & 1)]; 4270a688fcd0SKashyap D Desai if (!pd_sync) { 4271a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, 4272a688fcd0SKashyap D Desai "Failed to alloc mem for jbod map info.\n"); 4273a688fcd0SKashyap D Desai mrsas_release_mfi_cmd(cmd); 4274a688fcd0SKashyap D Desai return (ENOMEM); 4275a688fcd0SKashyap D Desai } 4276a688fcd0SKashyap D Desai memset(pd_sync, 0, pd_seq_map_sz); 4277a688fcd0SKashyap D Desai memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 4278a688fcd0SKashyap D Desai dcmd->cmd = MFI_CMD_DCMD; 4279a688fcd0SKashyap D Desai dcmd->cmd_status = 0xFF; 4280a688fcd0SKashyap D Desai dcmd->sge_count = 1; 4281a688fcd0SKashyap D Desai dcmd->timeout = 0; 4282a688fcd0SKashyap D Desai dcmd->pad_0 = 0; 4283*e34a057cSAlfredo Dal'Ava Junior dcmd->data_xfer_len = htole32(pd_seq_map_sz); 4284*e34a057cSAlfredo Dal'Ava Junior dcmd->opcode = htole32(MR_DCMD_SYSTEM_PD_MAP_GET_INFO); 4285*e34a057cSAlfredo Dal'Ava Junior dcmd->sgl.sge32[0].phys_addr = htole32(pd_seq_h & 0xFFFFFFFF); 4286*e34a057cSAlfredo Dal'Ava Junior dcmd->sgl.sge32[0].length = htole32(pd_seq_map_sz); 4287a688fcd0SKashyap D Desai 4288a688fcd0SKashyap D Desai if (pend) { 4289a688fcd0SKashyap D Desai dcmd->mbox.b[0] = MRSAS_DCMD_MBOX_PEND_FLAG; 4290*e34a057cSAlfredo Dal'Ava Junior dcmd->flags = htole16(MFI_FRAME_DIR_WRITE); 4291a688fcd0SKashyap D Desai sc->jbod_seq_cmd = cmd; 4292a688fcd0SKashyap D Desai if (mrsas_issue_dcmd(sc, cmd)) { 4293a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, 4294a688fcd0SKashyap D Desai "Fail to send sync map info command.\n"); 4295a688fcd0SKashyap D Desai return 1; 4296a688fcd0SKashyap D Desai } else 4297a688fcd0SKashyap D Desai return 0; 4298a688fcd0SKashyap D Desai } else 4299*e34a057cSAlfredo Dal'Ava Junior dcmd->flags = htole16(MFI_FRAME_DIR_READ); 4300a688fcd0SKashyap D Desai 4301a688fcd0SKashyap D Desai retcode = mrsas_issue_polled(sc, cmd); 4302a688fcd0SKashyap D Desai if (retcode == ETIMEDOUT) 4303a688fcd0SKashyap D Desai goto dcmd_timeout; 4304a688fcd0SKashyap D Desai 4305*e34a057cSAlfredo Dal'Ava Junior if (le32toh(pd_sync->count) > MAX_PHYSICAL_DEVICES) { 4306a688fcd0SKashyap D Desai device_printf(sc->mrsas_dev, 4307a688fcd0SKashyap D Desai "driver supports max %d JBOD, but FW reports %d\n", 4308a688fcd0SKashyap D Desai MAX_PHYSICAL_DEVICES, pd_sync->count); 4309a688fcd0SKashyap D Desai retcode = -EINVAL; 4310a688fcd0SKashyap D Desai } 4311a688fcd0SKashyap D Desai if (!retcode) 4312a688fcd0SKashyap D Desai sc->pd_seq_map_id++; 4313a688fcd0SKashyap D Desai do_ocr = 0; 4314a688fcd0SKashyap D Desai 4315a688fcd0SKashyap D Desai dcmd_timeout: 4316a688fcd0SKashyap D Desai if (do_ocr) 4317a688fcd0SKashyap D Desai sc->do_timedout_reset = MFI_DCMD_TIMEOUT_OCR; 4318a688fcd0SKashyap D Desai 4319a688fcd0SKashyap D Desai return (retcode); 4320a688fcd0SKashyap D Desai } 4321a688fcd0SKashyap D Desai 43228e727371SKashyap D Desai /* 43238e727371SKashyap D Desai * mrsas_get_map_info: Load and validate RAID map input: 43248e727371SKashyap D Desai * Adapter instance soft state 4325665484d8SDoug Ambrisko * 43268e727371SKashyap D Desai * This function calls mrsas_get_ld_map_info() and MR_ValidateMapInfo() to load 43278e727371SKashyap D Desai * and validate RAID map. It returns 0 if successful, 1 other- wise. 4328665484d8SDoug Ambrisko */ 43298e727371SKashyap D Desai static int 43308e727371SKashyap D Desai mrsas_get_map_info(struct mrsas_softc *sc) 4331665484d8SDoug Ambrisko { 4332665484d8SDoug Ambrisko uint8_t retcode = 0; 4333665484d8SDoug Ambrisko 4334665484d8SDoug Ambrisko sc->fast_path_io = 0; 4335665484d8SDoug Ambrisko if (!mrsas_get_ld_map_info(sc)) { 4336665484d8SDoug Ambrisko retcode = MR_ValidateMapInfo(sc); 4337665484d8SDoug Ambrisko if (retcode == 0) { 4338665484d8SDoug Ambrisko sc->fast_path_io = 1; 4339665484d8SDoug Ambrisko return 0; 4340665484d8SDoug Ambrisko } 4341665484d8SDoug Ambrisko } 4342665484d8SDoug Ambrisko return 1; 4343665484d8SDoug Ambrisko } 4344665484d8SDoug Ambrisko 43458e727371SKashyap D Desai /* 43468e727371SKashyap D Desai * mrsas_get_ld_map_info: Get FW's ld_map structure input: 43478e727371SKashyap D Desai * Adapter instance soft state 4348665484d8SDoug Ambrisko * 43498e727371SKashyap D Desai * Issues an internal command (DCMD) to get the FW's controller PD list 43508e727371SKashyap D Desai * structure. 4351665484d8SDoug Ambrisko */ 43528e727371SKashyap D Desai static int 43538e727371SKashyap D Desai mrsas_get_ld_map_info(struct mrsas_softc *sc) 4354665484d8SDoug Ambrisko { 4355665484d8SDoug Ambrisko int retcode = 0; 4356665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 4357665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 43584799d485SKashyap D Desai void *map; 4359665484d8SDoug Ambrisko bus_addr_t map_phys_addr = 0; 4360665484d8SDoug Ambrisko 4361665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 4362665484d8SDoug Ambrisko if (!cmd) { 43634799d485SKashyap D Desai device_printf(sc->mrsas_dev, 43644799d485SKashyap D Desai "Cannot alloc for ld map info cmd.\n"); 4365665484d8SDoug Ambrisko return 1; 4366665484d8SDoug Ambrisko } 4367665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 4368665484d8SDoug Ambrisko 43694799d485SKashyap D Desai map = (void *)sc->raidmap_mem[(sc->map_id & 1)]; 4370665484d8SDoug Ambrisko map_phys_addr = sc->raidmap_phys_addr[(sc->map_id & 1)]; 4371665484d8SDoug Ambrisko if (!map) { 43724799d485SKashyap D Desai device_printf(sc->mrsas_dev, 43734799d485SKashyap D Desai "Failed to alloc mem for ld map info.\n"); 4374665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 4375665484d8SDoug Ambrisko return (ENOMEM); 4376665484d8SDoug Ambrisko } 43774799d485SKashyap D Desai memset(map, 0, sizeof(sc->max_map_sz)); 4378665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 4379665484d8SDoug Ambrisko 4380665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 4381665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 4382665484d8SDoug Ambrisko dcmd->sge_count = 1; 4383*e34a057cSAlfredo Dal'Ava Junior dcmd->flags = htole16(MFI_FRAME_DIR_READ); 4384665484d8SDoug Ambrisko dcmd->timeout = 0; 4385665484d8SDoug Ambrisko dcmd->pad_0 = 0; 4386*e34a057cSAlfredo Dal'Ava Junior dcmd->data_xfer_len = htole32(sc->current_map_sz); 4387*e34a057cSAlfredo Dal'Ava Junior dcmd->opcode = htole32(MR_DCMD_LD_MAP_GET_INFO); 4388*e34a057cSAlfredo Dal'Ava Junior dcmd->sgl.sge32[0].phys_addr = htole32(map_phys_addr & 0xFFFFFFFF); 4389*e34a057cSAlfredo Dal'Ava Junior dcmd->sgl.sge32[0].length = htole32(sc->current_map_sz); 43904799d485SKashyap D Desai 4391f0c7594bSKashyap D Desai retcode = mrsas_issue_polled(sc, cmd); 4392f0c7594bSKashyap D Desai if (retcode == ETIMEDOUT) 4393f0c7594bSKashyap D Desai sc->do_timedout_reset = MFI_DCMD_TIMEOUT_OCR; 43944799d485SKashyap D Desai 4395665484d8SDoug Ambrisko return (retcode); 4396665484d8SDoug Ambrisko } 4397665484d8SDoug Ambrisko 43988e727371SKashyap D Desai /* 43998e727371SKashyap D Desai * mrsas_sync_map_info: Get FW's ld_map structure input: 44008e727371SKashyap D Desai * Adapter instance soft state 4401665484d8SDoug Ambrisko * 44028e727371SKashyap D Desai * Issues an internal command (DCMD) to get the FW's controller PD list 44038e727371SKashyap D Desai * structure. 4404665484d8SDoug Ambrisko */ 44058e727371SKashyap D Desai static int 44068e727371SKashyap D Desai mrsas_sync_map_info(struct mrsas_softc *sc) 4407665484d8SDoug Ambrisko { 4408665484d8SDoug Ambrisko int retcode = 0, i; 4409665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 4410665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 4411665484d8SDoug Ambrisko uint32_t size_sync_info, num_lds; 4412665484d8SDoug Ambrisko MR_LD_TARGET_SYNC *target_map = NULL; 44134799d485SKashyap D Desai MR_DRV_RAID_MAP_ALL *map; 4414665484d8SDoug Ambrisko MR_LD_RAID *raid; 4415665484d8SDoug Ambrisko MR_LD_TARGET_SYNC *ld_sync; 4416665484d8SDoug Ambrisko bus_addr_t map_phys_addr = 0; 4417665484d8SDoug Ambrisko 4418665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 4419665484d8SDoug Ambrisko if (!cmd) { 4420731b7561SKashyap D Desai device_printf(sc->mrsas_dev, "Cannot alloc for sync map info cmd\n"); 4421731b7561SKashyap D Desai return ENOMEM; 4422665484d8SDoug Ambrisko } 44234799d485SKashyap D Desai map = sc->ld_drv_map[sc->map_id & 1]; 4424665484d8SDoug Ambrisko num_lds = map->raidMap.ldCount; 4425665484d8SDoug Ambrisko 4426665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 4427665484d8SDoug Ambrisko size_sync_info = sizeof(MR_LD_TARGET_SYNC) * num_lds; 4428665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 4429665484d8SDoug Ambrisko 44308e727371SKashyap D Desai target_map = (MR_LD_TARGET_SYNC *) sc->raidmap_mem[(sc->map_id - 1) & 1]; 44314799d485SKashyap D Desai memset(target_map, 0, sc->max_map_sz); 4432665484d8SDoug Ambrisko 4433665484d8SDoug Ambrisko map_phys_addr = sc->raidmap_phys_addr[(sc->map_id - 1) & 1]; 4434665484d8SDoug Ambrisko 4435665484d8SDoug Ambrisko ld_sync = (MR_LD_TARGET_SYNC *) target_map; 4436665484d8SDoug Ambrisko 4437665484d8SDoug Ambrisko for (i = 0; i < num_lds; i++, ld_sync++) { 4438665484d8SDoug Ambrisko raid = MR_LdRaidGet(i, map); 4439665484d8SDoug Ambrisko ld_sync->targetId = MR_GetLDTgtId(i, map); 4440665484d8SDoug Ambrisko ld_sync->seqNum = raid->seqNum; 4441665484d8SDoug Ambrisko } 4442665484d8SDoug Ambrisko 4443665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 4444665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 4445665484d8SDoug Ambrisko dcmd->sge_count = 1; 4446*e34a057cSAlfredo Dal'Ava Junior dcmd->flags = htole16(MFI_FRAME_DIR_WRITE); 4447665484d8SDoug Ambrisko dcmd->timeout = 0; 4448665484d8SDoug Ambrisko dcmd->pad_0 = 0; 4449*e34a057cSAlfredo Dal'Ava Junior dcmd->data_xfer_len = htole32(sc->current_map_sz); 4450665484d8SDoug Ambrisko dcmd->mbox.b[0] = num_lds; 4451665484d8SDoug Ambrisko dcmd->mbox.b[1] = MRSAS_DCMD_MBOX_PEND_FLAG; 4452*e34a057cSAlfredo Dal'Ava Junior dcmd->opcode = htole32(MR_DCMD_LD_MAP_GET_INFO); 4453*e34a057cSAlfredo Dal'Ava Junior dcmd->sgl.sge32[0].phys_addr = htole32(map_phys_addr & 0xFFFFFFFF); 4454*e34a057cSAlfredo Dal'Ava Junior dcmd->sgl.sge32[0].length = htole32(sc->current_map_sz); 4455665484d8SDoug Ambrisko 4456665484d8SDoug Ambrisko sc->map_update_cmd = cmd; 4457665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)) { 44584799d485SKashyap D Desai device_printf(sc->mrsas_dev, 44594799d485SKashyap D Desai "Fail to send sync map info command.\n"); 4460665484d8SDoug Ambrisko return (1); 4461665484d8SDoug Ambrisko } 4462665484d8SDoug Ambrisko return (retcode); 4463665484d8SDoug Ambrisko } 4464665484d8SDoug Ambrisko 446579b4460bSKashyap D Desai /* Input: dcmd.opcode - MR_DCMD_PD_GET_INFO 446679b4460bSKashyap D Desai * dcmd.mbox.s[0] - deviceId for this physical drive 446779b4460bSKashyap D Desai * dcmd.sge IN - ptr to returned MR_PD_INFO structure 446879b4460bSKashyap D Desai * Desc: Firmware return the physical drive info structure 446979b4460bSKashyap D Desai * 447079b4460bSKashyap D Desai */ 447179b4460bSKashyap D Desai static void 447279b4460bSKashyap D Desai mrsas_get_pd_info(struct mrsas_softc *sc, u_int16_t device_id) 447379b4460bSKashyap D Desai { 447479b4460bSKashyap D Desai int retcode; 447579b4460bSKashyap D Desai u_int8_t do_ocr = 1; 447679b4460bSKashyap D Desai struct mrsas_mfi_cmd *cmd; 447779b4460bSKashyap D Desai struct mrsas_dcmd_frame *dcmd; 447879b4460bSKashyap D Desai 447979b4460bSKashyap D Desai cmd = mrsas_get_mfi_cmd(sc); 448079b4460bSKashyap D Desai 448179b4460bSKashyap D Desai if (!cmd) { 448279b4460bSKashyap D Desai device_printf(sc->mrsas_dev, 448379b4460bSKashyap D Desai "Cannot alloc for get PD info cmd\n"); 448479b4460bSKashyap D Desai return; 448579b4460bSKashyap D Desai } 448679b4460bSKashyap D Desai dcmd = &cmd->frame->dcmd; 448779b4460bSKashyap D Desai 448879b4460bSKashyap D Desai memset(sc->pd_info_mem, 0, sizeof(struct mrsas_pd_info)); 448979b4460bSKashyap D Desai memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 449079b4460bSKashyap D Desai 4491*e34a057cSAlfredo Dal'Ava Junior dcmd->mbox.s[0] = htole16(device_id); 449279b4460bSKashyap D Desai dcmd->cmd = MFI_CMD_DCMD; 449379b4460bSKashyap D Desai dcmd->cmd_status = 0xFF; 449479b4460bSKashyap D Desai dcmd->sge_count = 1; 449579b4460bSKashyap D Desai dcmd->flags = MFI_FRAME_DIR_READ; 449679b4460bSKashyap D Desai dcmd->timeout = 0; 449779b4460bSKashyap D Desai dcmd->pad_0 = 0; 4498*e34a057cSAlfredo Dal'Ava Junior dcmd->data_xfer_len = htole32(sizeof(struct mrsas_pd_info)); 4499*e34a057cSAlfredo Dal'Ava Junior dcmd->opcode = htole32(MR_DCMD_PD_GET_INFO); 4500*e34a057cSAlfredo Dal'Ava Junior dcmd->sgl.sge32[0].phys_addr = htole32((u_int32_t)sc->pd_info_phys_addr & 0xFFFFFFFF); 4501*e34a057cSAlfredo Dal'Ava Junior dcmd->sgl.sge32[0].length = htole32(sizeof(struct mrsas_pd_info)); 450279b4460bSKashyap D Desai 450379b4460bSKashyap D Desai if (!sc->mask_interrupts) 450479b4460bSKashyap D Desai retcode = mrsas_issue_blocked_cmd(sc, cmd); 450579b4460bSKashyap D Desai else 450679b4460bSKashyap D Desai retcode = mrsas_issue_polled(sc, cmd); 450779b4460bSKashyap D Desai 450879b4460bSKashyap D Desai if (retcode == ETIMEDOUT) 450979b4460bSKashyap D Desai goto dcmd_timeout; 451079b4460bSKashyap D Desai 451179b4460bSKashyap D Desai sc->target_list[device_id].interface_type = 4512*e34a057cSAlfredo Dal'Ava Junior le16toh(sc->pd_info_mem->state.ddf.pdType.intf); 451379b4460bSKashyap D Desai 451479b4460bSKashyap D Desai do_ocr = 0; 451579b4460bSKashyap D Desai 451679b4460bSKashyap D Desai dcmd_timeout: 451779b4460bSKashyap D Desai 451879b4460bSKashyap D Desai if (do_ocr) 451979b4460bSKashyap D Desai sc->do_timedout_reset = MFI_DCMD_TIMEOUT_OCR; 452079b4460bSKashyap D Desai 452179b4460bSKashyap D Desai if (!sc->mask_interrupts) 452279b4460bSKashyap D Desai mrsas_release_mfi_cmd(cmd); 452379b4460bSKashyap D Desai } 452479b4460bSKashyap D Desai 452579b4460bSKashyap D Desai /* 452679b4460bSKashyap D Desai * mrsas_add_target: Add target ID of system PD/VD to driver's data structure. 452779b4460bSKashyap D Desai * sc: Adapter's soft state 452879b4460bSKashyap D Desai * target_id: Unique target id per controller(managed by driver) 452979b4460bSKashyap D Desai * for system PDs- target ID ranges from 0 to (MRSAS_MAX_PD - 1) 453079b4460bSKashyap D Desai * for VDs- target ID ranges from MRSAS_MAX_PD to MRSAS_MAX_TM_TARGETS 453179b4460bSKashyap D Desai * return: void 453279b4460bSKashyap D Desai * Descripton: This function will be called whenever system PD or VD is created. 453379b4460bSKashyap D Desai */ 453479b4460bSKashyap D Desai static void mrsas_add_target(struct mrsas_softc *sc, 453579b4460bSKashyap D Desai u_int16_t target_id) 453679b4460bSKashyap D Desai { 453779b4460bSKashyap D Desai sc->target_list[target_id].target_id = target_id; 453879b4460bSKashyap D Desai 453979b4460bSKashyap D Desai device_printf(sc->mrsas_dev, 454079b4460bSKashyap D Desai "%s created target ID: 0x%x\n", 454179b4460bSKashyap D Desai (target_id < MRSAS_MAX_PD ? "System PD" : "VD"), 454279b4460bSKashyap D Desai (target_id < MRSAS_MAX_PD ? target_id : (target_id - MRSAS_MAX_PD))); 454379b4460bSKashyap D Desai /* 454479b4460bSKashyap D Desai * If interrupts are enabled, then only fire DCMD to get pd_info 454579b4460bSKashyap D Desai * for system PDs 454679b4460bSKashyap D Desai */ 454779b4460bSKashyap D Desai if (!sc->mask_interrupts && sc->pd_info_mem && 454879b4460bSKashyap D Desai (target_id < MRSAS_MAX_PD)) 454979b4460bSKashyap D Desai mrsas_get_pd_info(sc, target_id); 455079b4460bSKashyap D Desai 455179b4460bSKashyap D Desai } 455279b4460bSKashyap D Desai 455379b4460bSKashyap D Desai /* 455479b4460bSKashyap D Desai * mrsas_remove_target: Remove target ID of system PD/VD from driver's data structure. 455579b4460bSKashyap D Desai * sc: Adapter's soft state 455679b4460bSKashyap D Desai * target_id: Unique target id per controller(managed by driver) 455779b4460bSKashyap D Desai * for system PDs- target ID ranges from 0 to (MRSAS_MAX_PD - 1) 455879b4460bSKashyap D Desai * for VDs- target ID ranges from MRSAS_MAX_PD to MRSAS_MAX_TM_TARGETS 455979b4460bSKashyap D Desai * return: void 456079b4460bSKashyap D Desai * Descripton: This function will be called whenever system PD or VD is deleted 456179b4460bSKashyap D Desai */ 456279b4460bSKashyap D Desai static void mrsas_remove_target(struct mrsas_softc *sc, 456379b4460bSKashyap D Desai u_int16_t target_id) 456479b4460bSKashyap D Desai { 456579b4460bSKashyap D Desai sc->target_list[target_id].target_id = 0xffff; 456679b4460bSKashyap D Desai device_printf(sc->mrsas_dev, 456779b4460bSKashyap D Desai "%s deleted target ID: 0x%x\n", 456879b4460bSKashyap D Desai (target_id < MRSAS_MAX_PD ? "System PD" : "VD"), 456979b4460bSKashyap D Desai (target_id < MRSAS_MAX_PD ? target_id : (target_id - MRSAS_MAX_PD))); 457079b4460bSKashyap D Desai } 457179b4460bSKashyap D Desai 45728e727371SKashyap D Desai /* 45738e727371SKashyap D Desai * mrsas_get_pd_list: Returns FW's PD list structure input: 45748e727371SKashyap D Desai * Adapter soft state 4575665484d8SDoug Ambrisko * 45768e727371SKashyap D Desai * Issues an internal command (DCMD) to get the FW's controller PD list 45778e727371SKashyap D Desai * structure. This information is mainly used to find out about system 45788e727371SKashyap D Desai * supported by Firmware. 4579665484d8SDoug Ambrisko */ 45808e727371SKashyap D Desai static int 45818e727371SKashyap D Desai mrsas_get_pd_list(struct mrsas_softc *sc) 4582665484d8SDoug Ambrisko { 4583665484d8SDoug Ambrisko int retcode = 0, pd_index = 0, pd_count = 0, pd_list_size; 4584f0c7594bSKashyap D Desai u_int8_t do_ocr = 1; 4585665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 4586665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 4587665484d8SDoug Ambrisko struct MR_PD_LIST *pd_list_mem; 4588665484d8SDoug Ambrisko struct MR_PD_ADDRESS *pd_addr; 4589665484d8SDoug Ambrisko bus_addr_t pd_list_phys_addr = 0; 4590665484d8SDoug Ambrisko struct mrsas_tmp_dcmd *tcmd; 4591*e34a057cSAlfredo Dal'Ava Junior u_int16_t dev_id; 4592665484d8SDoug Ambrisko 4593665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 4594665484d8SDoug Ambrisko if (!cmd) { 45954799d485SKashyap D Desai device_printf(sc->mrsas_dev, 45964799d485SKashyap D Desai "Cannot alloc for get PD list cmd\n"); 4597665484d8SDoug Ambrisko return 1; 4598665484d8SDoug Ambrisko } 4599665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 4600665484d8SDoug Ambrisko 4601665484d8SDoug Ambrisko tcmd = malloc(sizeof(struct mrsas_tmp_dcmd), M_MRSAS, M_NOWAIT); 4602665484d8SDoug Ambrisko pd_list_size = MRSAS_MAX_PD * sizeof(struct MR_PD_LIST); 4603665484d8SDoug Ambrisko if (mrsas_alloc_tmp_dcmd(sc, tcmd, pd_list_size) != SUCCESS) { 46044799d485SKashyap D Desai device_printf(sc->mrsas_dev, 46054799d485SKashyap D Desai "Cannot alloc dmamap for get PD list cmd\n"); 4606665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 4607f0c7594bSKashyap D Desai mrsas_free_tmp_dcmd(tcmd); 4608f0c7594bSKashyap D Desai free(tcmd, M_MRSAS); 4609665484d8SDoug Ambrisko return (ENOMEM); 46108e727371SKashyap D Desai } else { 4611665484d8SDoug Ambrisko pd_list_mem = tcmd->tmp_dcmd_mem; 4612665484d8SDoug Ambrisko pd_list_phys_addr = tcmd->tmp_dcmd_phys_addr; 4613665484d8SDoug Ambrisko } 4614665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 4615665484d8SDoug Ambrisko 4616665484d8SDoug Ambrisko dcmd->mbox.b[0] = MR_PD_QUERY_TYPE_EXPOSED_TO_HOST; 4617665484d8SDoug Ambrisko dcmd->mbox.b[1] = 0; 4618665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 4619665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 4620665484d8SDoug Ambrisko dcmd->sge_count = 1; 4621*e34a057cSAlfredo Dal'Ava Junior dcmd->flags = htole16(MFI_FRAME_DIR_READ); 4622665484d8SDoug Ambrisko dcmd->timeout = 0; 4623665484d8SDoug Ambrisko dcmd->pad_0 = 0; 4624*e34a057cSAlfredo Dal'Ava Junior dcmd->data_xfer_len = htole32(MRSAS_MAX_PD * sizeof(struct MR_PD_LIST)); 4625*e34a057cSAlfredo Dal'Ava Junior dcmd->opcode = htole32(MR_DCMD_PD_LIST_QUERY); 4626*e34a057cSAlfredo Dal'Ava Junior dcmd->sgl.sge32[0].phys_addr = htole32(pd_list_phys_addr & 0xFFFFFFFF); 4627*e34a057cSAlfredo Dal'Ava Junior dcmd->sgl.sge32[0].length = htole32(MRSAS_MAX_PD * sizeof(struct MR_PD_LIST)); 4628665484d8SDoug Ambrisko 4629731b7561SKashyap D Desai if (!sc->mask_interrupts) 4630731b7561SKashyap D Desai retcode = mrsas_issue_blocked_cmd(sc, cmd); 4631731b7561SKashyap D Desai else 4632f0c7594bSKashyap D Desai retcode = mrsas_issue_polled(sc, cmd); 4633731b7561SKashyap D Desai 4634f0c7594bSKashyap D Desai if (retcode == ETIMEDOUT) 4635f0c7594bSKashyap D Desai goto dcmd_timeout; 4636665484d8SDoug Ambrisko 4637665484d8SDoug Ambrisko /* Get the instance PD list */ 4638665484d8SDoug Ambrisko pd_count = MRSAS_MAX_PD; 4639665484d8SDoug Ambrisko pd_addr = pd_list_mem->addr; 4640*e34a057cSAlfredo Dal'Ava Junior if (le32toh(pd_list_mem->count) < pd_count) { 46414799d485SKashyap D Desai memset(sc->local_pd_list, 0, 46424799d485SKashyap D Desai MRSAS_MAX_PD * sizeof(struct mrsas_pd_list)); 4643*e34a057cSAlfredo Dal'Ava Junior for (pd_index = 0; pd_index < le32toh(pd_list_mem->count); pd_index++) { 4644*e34a057cSAlfredo Dal'Ava Junior dev_id = le16toh(pd_addr->deviceId); 4645*e34a057cSAlfredo Dal'Ava Junior sc->local_pd_list[dev_id].tid = dev_id; 4646*e34a057cSAlfredo Dal'Ava Junior sc->local_pd_list[dev_id].driveType = 4647*e34a057cSAlfredo Dal'Ava Junior le16toh(pd_addr->scsiDevType); 4648*e34a057cSAlfredo Dal'Ava Junior sc->local_pd_list[dev_id].driveState = 46494799d485SKashyap D Desai MR_PD_STATE_SYSTEM; 4650*e34a057cSAlfredo Dal'Ava Junior if (sc->target_list[dev_id].target_id == 0xffff) 4651*e34a057cSAlfredo Dal'Ava Junior mrsas_add_target(sc, dev_id); 4652665484d8SDoug Ambrisko pd_addr++; 4653665484d8SDoug Ambrisko } 465479b4460bSKashyap D Desai for (pd_index = 0; pd_index < MRSAS_MAX_PD; pd_index++) { 465579b4460bSKashyap D Desai if ((sc->local_pd_list[pd_index].driveState != 465679b4460bSKashyap D Desai MR_PD_STATE_SYSTEM) && 465779b4460bSKashyap D Desai (sc->target_list[pd_index].target_id != 465879b4460bSKashyap D Desai 0xffff)) { 465979b4460bSKashyap D Desai mrsas_remove_target(sc, pd_index); 466079b4460bSKashyap D Desai } 466179b4460bSKashyap D Desai } 46628e727371SKashyap D Desai /* 46638e727371SKashyap D Desai * Use mutext/spinlock if pd_list component size increase more than 46648e727371SKashyap D Desai * 32 bit. 46658e727371SKashyap D Desai */ 4666665484d8SDoug Ambrisko memcpy(sc->pd_list, sc->local_pd_list, sizeof(sc->local_pd_list)); 4667f0c7594bSKashyap D Desai do_ocr = 0; 4668f0c7594bSKashyap D Desai } 4669f0c7594bSKashyap D Desai dcmd_timeout: 4670665484d8SDoug Ambrisko mrsas_free_tmp_dcmd(tcmd); 4671665484d8SDoug Ambrisko free(tcmd, M_MRSAS); 4672f0c7594bSKashyap D Desai 4673f0c7594bSKashyap D Desai if (do_ocr) 4674f0c7594bSKashyap D Desai sc->do_timedout_reset = MFI_DCMD_TIMEOUT_OCR; 4675731b7561SKashyap D Desai 4676731b7561SKashyap D Desai if (!sc->mask_interrupts) 4677f0c7594bSKashyap D Desai mrsas_release_mfi_cmd(cmd); 4678f0c7594bSKashyap D Desai 4679665484d8SDoug Ambrisko return (retcode); 4680665484d8SDoug Ambrisko } 4681665484d8SDoug Ambrisko 46828e727371SKashyap D Desai /* 46838e727371SKashyap D Desai * mrsas_get_ld_list: Returns FW's LD list structure input: 46848e727371SKashyap D Desai * Adapter soft state 4685665484d8SDoug Ambrisko * 46868e727371SKashyap D Desai * Issues an internal command (DCMD) to get the FW's controller PD list 46878e727371SKashyap D Desai * structure. This information is mainly used to find out about supported by 46888e727371SKashyap D Desai * the FW. 4689665484d8SDoug Ambrisko */ 46908e727371SKashyap D Desai static int 46918e727371SKashyap D Desai mrsas_get_ld_list(struct mrsas_softc *sc) 4692665484d8SDoug Ambrisko { 469379b4460bSKashyap D Desai int ld_list_size, retcode = 0, ld_index = 0, ids = 0, drv_tgt_id; 4694f0c7594bSKashyap D Desai u_int8_t do_ocr = 1; 4695665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 4696665484d8SDoug Ambrisko struct mrsas_dcmd_frame *dcmd; 4697665484d8SDoug Ambrisko struct MR_LD_LIST *ld_list_mem; 4698665484d8SDoug Ambrisko bus_addr_t ld_list_phys_addr = 0; 4699665484d8SDoug Ambrisko struct mrsas_tmp_dcmd *tcmd; 4700665484d8SDoug Ambrisko 4701665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 4702665484d8SDoug Ambrisko if (!cmd) { 47034799d485SKashyap D Desai device_printf(sc->mrsas_dev, 47044799d485SKashyap D Desai "Cannot alloc for get LD list cmd\n"); 4705665484d8SDoug Ambrisko return 1; 4706665484d8SDoug Ambrisko } 4707665484d8SDoug Ambrisko dcmd = &cmd->frame->dcmd; 4708665484d8SDoug Ambrisko 4709665484d8SDoug Ambrisko tcmd = malloc(sizeof(struct mrsas_tmp_dcmd), M_MRSAS, M_NOWAIT); 4710665484d8SDoug Ambrisko ld_list_size = sizeof(struct MR_LD_LIST); 4711665484d8SDoug Ambrisko if (mrsas_alloc_tmp_dcmd(sc, tcmd, ld_list_size) != SUCCESS) { 47124799d485SKashyap D Desai device_printf(sc->mrsas_dev, 47134799d485SKashyap D Desai "Cannot alloc dmamap for get LD list cmd\n"); 4714665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 4715f0c7594bSKashyap D Desai mrsas_free_tmp_dcmd(tcmd); 4716f0c7594bSKashyap D Desai free(tcmd, M_MRSAS); 4717665484d8SDoug Ambrisko return (ENOMEM); 47188e727371SKashyap D Desai } else { 4719665484d8SDoug Ambrisko ld_list_mem = tcmd->tmp_dcmd_mem; 4720665484d8SDoug Ambrisko ld_list_phys_addr = tcmd->tmp_dcmd_phys_addr; 4721665484d8SDoug Ambrisko } 4722665484d8SDoug Ambrisko memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE); 4723665484d8SDoug Ambrisko 47244799d485SKashyap D Desai if (sc->max256vdSupport) 47254799d485SKashyap D Desai dcmd->mbox.b[0] = 1; 47264799d485SKashyap D Desai 4727665484d8SDoug Ambrisko dcmd->cmd = MFI_CMD_DCMD; 4728665484d8SDoug Ambrisko dcmd->cmd_status = 0xFF; 4729665484d8SDoug Ambrisko dcmd->sge_count = 1; 4730665484d8SDoug Ambrisko dcmd->flags = MFI_FRAME_DIR_READ; 4731665484d8SDoug Ambrisko dcmd->timeout = 0; 4732*e34a057cSAlfredo Dal'Ava Junior dcmd->data_xfer_len = htole32(sizeof(struct MR_LD_LIST)); 4733*e34a057cSAlfredo Dal'Ava Junior dcmd->opcode = htole32(MR_DCMD_LD_GET_LIST); 4734*e34a057cSAlfredo Dal'Ava Junior dcmd->sgl.sge32[0].phys_addr = htole32(ld_list_phys_addr); 4735*e34a057cSAlfredo Dal'Ava Junior dcmd->sgl.sge32[0].length = htole32(sizeof(struct MR_LD_LIST)); 4736665484d8SDoug Ambrisko dcmd->pad_0 = 0; 4737665484d8SDoug Ambrisko 4738731b7561SKashyap D Desai if (!sc->mask_interrupts) 4739731b7561SKashyap D Desai retcode = mrsas_issue_blocked_cmd(sc, cmd); 4740731b7561SKashyap D Desai else 4741f0c7594bSKashyap D Desai retcode = mrsas_issue_polled(sc, cmd); 4742731b7561SKashyap D Desai 4743f0c7594bSKashyap D Desai if (retcode == ETIMEDOUT) 4744f0c7594bSKashyap D Desai goto dcmd_timeout; 4745665484d8SDoug Ambrisko 47464799d485SKashyap D Desai #if VD_EXT_DEBUG 47474799d485SKashyap D Desai printf("Number of LDs %d\n", ld_list_mem->ldCount); 47484799d485SKashyap D Desai #endif 47494799d485SKashyap D Desai 4750665484d8SDoug Ambrisko /* Get the instance LD list */ 4751*e34a057cSAlfredo Dal'Ava Junior if (le32toh(ld_list_mem->ldCount) <= sc->fw_supported_vd_count) { 4752*e34a057cSAlfredo Dal'Ava Junior sc->CurLdCount = le32toh(ld_list_mem->ldCount); 47534799d485SKashyap D Desai memset(sc->ld_ids, 0xff, MAX_LOGICAL_DRIVES_EXT); 4754*e34a057cSAlfredo Dal'Ava Junior for (ld_index = 0; ld_index < le32toh(ld_list_mem->ldCount); ld_index++) { 4755665484d8SDoug Ambrisko ids = ld_list_mem->ldList[ld_index].ref.ld_context.targetId; 475679b4460bSKashyap D Desai drv_tgt_id = ids + MRSAS_MAX_PD; 475779b4460bSKashyap D Desai if (ld_list_mem->ldList[ld_index].state != 0) { 4758665484d8SDoug Ambrisko sc->ld_ids[ids] = ld_list_mem->ldList[ld_index].ref.ld_context.targetId; 475979b4460bSKashyap D Desai if (sc->target_list[drv_tgt_id].target_id == 476079b4460bSKashyap D Desai 0xffff) 476179b4460bSKashyap D Desai mrsas_add_target(sc, drv_tgt_id); 476279b4460bSKashyap D Desai } else { 476379b4460bSKashyap D Desai if (sc->target_list[drv_tgt_id].target_id != 476479b4460bSKashyap D Desai 0xffff) 476579b4460bSKashyap D Desai mrsas_remove_target(sc, 476679b4460bSKashyap D Desai drv_tgt_id); 4767665484d8SDoug Ambrisko } 4768665484d8SDoug Ambrisko } 476979b4460bSKashyap D Desai 4770f0c7594bSKashyap D Desai do_ocr = 0; 4771665484d8SDoug Ambrisko } 4772f0c7594bSKashyap D Desai dcmd_timeout: 4773665484d8SDoug Ambrisko mrsas_free_tmp_dcmd(tcmd); 4774665484d8SDoug Ambrisko free(tcmd, M_MRSAS); 4775f0c7594bSKashyap D Desai 4776f0c7594bSKashyap D Desai if (do_ocr) 4777f0c7594bSKashyap D Desai sc->do_timedout_reset = MFI_DCMD_TIMEOUT_OCR; 4778731b7561SKashyap D Desai if (!sc->mask_interrupts) 4779f0c7594bSKashyap D Desai mrsas_release_mfi_cmd(cmd); 4780f0c7594bSKashyap D Desai 4781665484d8SDoug Ambrisko return (retcode); 4782665484d8SDoug Ambrisko } 4783665484d8SDoug Ambrisko 47848e727371SKashyap D Desai /* 47858e727371SKashyap D Desai * mrsas_alloc_tmp_dcmd: Allocates memory for temporary command input: 47868e727371SKashyap D Desai * Adapter soft state Temp command Size of alloction 4787665484d8SDoug Ambrisko * 4788665484d8SDoug Ambrisko * Allocates DMAable memory for a temporary internal command. The allocated 4789665484d8SDoug Ambrisko * memory is initialized to all zeros upon successful loading of the dma 4790665484d8SDoug Ambrisko * mapped memory. 4791665484d8SDoug Ambrisko */ 47928e727371SKashyap D Desai int 47938e727371SKashyap D Desai mrsas_alloc_tmp_dcmd(struct mrsas_softc *sc, 47948e727371SKashyap D Desai struct mrsas_tmp_dcmd *tcmd, int size) 4795665484d8SDoug Ambrisko { 47968e727371SKashyap D Desai if (bus_dma_tag_create(sc->mrsas_parent_tag, 47978e727371SKashyap D Desai 1, 0, 47988e727371SKashyap D Desai BUS_SPACE_MAXADDR_32BIT, 47998e727371SKashyap D Desai BUS_SPACE_MAXADDR, 48008e727371SKashyap D Desai NULL, NULL, 48018e727371SKashyap D Desai size, 48028e727371SKashyap D Desai 1, 48038e727371SKashyap D Desai size, 48048e727371SKashyap D Desai BUS_DMA_ALLOCNOW, 48058e727371SKashyap D Desai NULL, NULL, 4806665484d8SDoug Ambrisko &tcmd->tmp_dcmd_tag)) { 4807665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate tmp dcmd tag\n"); 4808665484d8SDoug Ambrisko return (ENOMEM); 4809665484d8SDoug Ambrisko } 4810665484d8SDoug Ambrisko if (bus_dmamem_alloc(tcmd->tmp_dcmd_tag, (void **)&tcmd->tmp_dcmd_mem, 4811665484d8SDoug Ambrisko BUS_DMA_NOWAIT, &tcmd->tmp_dcmd_dmamap)) { 4812665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot allocate tmp dcmd mem\n"); 4813665484d8SDoug Ambrisko return (ENOMEM); 4814665484d8SDoug Ambrisko } 4815665484d8SDoug Ambrisko if (bus_dmamap_load(tcmd->tmp_dcmd_tag, tcmd->tmp_dcmd_dmamap, 4816665484d8SDoug Ambrisko tcmd->tmp_dcmd_mem, size, mrsas_addr_cb, 4817665484d8SDoug Ambrisko &tcmd->tmp_dcmd_phys_addr, BUS_DMA_NOWAIT)) { 4818665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot load tmp dcmd mem\n"); 4819665484d8SDoug Ambrisko return (ENOMEM); 4820665484d8SDoug Ambrisko } 4821665484d8SDoug Ambrisko memset(tcmd->tmp_dcmd_mem, 0, size); 4822665484d8SDoug Ambrisko return (0); 4823665484d8SDoug Ambrisko } 4824665484d8SDoug Ambrisko 48258e727371SKashyap D Desai /* 48268e727371SKashyap D Desai * mrsas_free_tmp_dcmd: Free memory for temporary command input: 48278e727371SKashyap D Desai * temporary dcmd pointer 4828665484d8SDoug Ambrisko * 48298e727371SKashyap D Desai * Deallocates memory of the temporary command for use in the construction of 48308e727371SKashyap D Desai * the internal DCMD. 4831665484d8SDoug Ambrisko */ 48328e727371SKashyap D Desai void 48338e727371SKashyap D Desai mrsas_free_tmp_dcmd(struct mrsas_tmp_dcmd *tmp) 4834665484d8SDoug Ambrisko { 4835665484d8SDoug Ambrisko if (tmp->tmp_dcmd_phys_addr) 4836665484d8SDoug Ambrisko bus_dmamap_unload(tmp->tmp_dcmd_tag, tmp->tmp_dcmd_dmamap); 4837665484d8SDoug Ambrisko if (tmp->tmp_dcmd_mem != NULL) 4838665484d8SDoug Ambrisko bus_dmamem_free(tmp->tmp_dcmd_tag, tmp->tmp_dcmd_mem, tmp->tmp_dcmd_dmamap); 4839665484d8SDoug Ambrisko if (tmp->tmp_dcmd_tag != NULL) 4840665484d8SDoug Ambrisko bus_dma_tag_destroy(tmp->tmp_dcmd_tag); 4841665484d8SDoug Ambrisko } 4842665484d8SDoug Ambrisko 48438e727371SKashyap D Desai /* 48448e727371SKashyap D Desai * mrsas_issue_blocked_abort_cmd: Aborts previously issued cmd input: 48458e727371SKashyap D Desai * Adapter soft state Previously issued cmd to be aborted 4846665484d8SDoug Ambrisko * 4847665484d8SDoug Ambrisko * This function is used to abort previously issued commands, such as AEN and 4848665484d8SDoug Ambrisko * RAID map sync map commands. The abort command is sent as a DCMD internal 4849665484d8SDoug Ambrisko * command and subsequently the driver will wait for a return status. The 4850665484d8SDoug Ambrisko * max wait time is MRSAS_INTERNAL_CMD_WAIT_TIME seconds. 4851665484d8SDoug Ambrisko */ 48528e727371SKashyap D Desai static int 48538e727371SKashyap D Desai mrsas_issue_blocked_abort_cmd(struct mrsas_softc *sc, 4854665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd_to_abort) 4855665484d8SDoug Ambrisko { 4856665484d8SDoug Ambrisko struct mrsas_mfi_cmd *cmd; 4857665484d8SDoug Ambrisko struct mrsas_abort_frame *abort_fr; 4858665484d8SDoug Ambrisko u_int8_t retcode = 0; 4859665484d8SDoug Ambrisko unsigned long total_time = 0; 4860665484d8SDoug Ambrisko u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME; 4861665484d8SDoug Ambrisko 4862665484d8SDoug Ambrisko cmd = mrsas_get_mfi_cmd(sc); 4863665484d8SDoug Ambrisko if (!cmd) { 4864665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Cannot alloc for abort cmd\n"); 4865665484d8SDoug Ambrisko return (1); 4866665484d8SDoug Ambrisko } 4867665484d8SDoug Ambrisko abort_fr = &cmd->frame->abort; 4868665484d8SDoug Ambrisko 4869665484d8SDoug Ambrisko /* Prepare and issue the abort frame */ 4870665484d8SDoug Ambrisko abort_fr->cmd = MFI_CMD_ABORT; 4871665484d8SDoug Ambrisko abort_fr->cmd_status = 0xFF; 4872665484d8SDoug Ambrisko abort_fr->flags = 0; 4873665484d8SDoug Ambrisko abort_fr->abort_context = cmd_to_abort->index; 4874665484d8SDoug Ambrisko abort_fr->abort_mfi_phys_addr_lo = cmd_to_abort->frame_phys_addr; 4875665484d8SDoug Ambrisko abort_fr->abort_mfi_phys_addr_hi = 0; 4876665484d8SDoug Ambrisko 4877665484d8SDoug Ambrisko cmd->sync_cmd = 1; 4878665484d8SDoug Ambrisko cmd->cmd_status = 0xFF; 4879665484d8SDoug Ambrisko 4880665484d8SDoug Ambrisko if (mrsas_issue_dcmd(sc, cmd)) { 4881665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Fail to send abort command.\n"); 4882665484d8SDoug Ambrisko return (1); 4883665484d8SDoug Ambrisko } 4884665484d8SDoug Ambrisko /* Wait for this cmd to complete */ 4885665484d8SDoug Ambrisko sc->chan = (void *)&cmd; 4886665484d8SDoug Ambrisko while (1) { 4887665484d8SDoug Ambrisko if (cmd->cmd_status == 0xFF) { 4888665484d8SDoug Ambrisko tsleep((void *)&sc->chan, 0, "mrsas_sleep", hz); 48898e727371SKashyap D Desai } else 4890665484d8SDoug Ambrisko break; 4891665484d8SDoug Ambrisko total_time++; 4892665484d8SDoug Ambrisko if (total_time >= max_wait) { 4893665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "Abort cmd timed out after %d sec.\n", max_wait); 4894665484d8SDoug Ambrisko retcode = 1; 4895665484d8SDoug Ambrisko break; 4896665484d8SDoug Ambrisko } 4897665484d8SDoug Ambrisko } 4898665484d8SDoug Ambrisko 4899665484d8SDoug Ambrisko cmd->sync_cmd = 0; 4900665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 4901665484d8SDoug Ambrisko return (retcode); 4902665484d8SDoug Ambrisko } 4903665484d8SDoug Ambrisko 49048e727371SKashyap D Desai /* 49058e727371SKashyap D Desai * mrsas_complete_abort: Completes aborting a command input: 49068e727371SKashyap D Desai * Adapter soft state Cmd that was issued to abort another cmd 4907665484d8SDoug Ambrisko * 49088e727371SKashyap D Desai * The mrsas_issue_blocked_abort_cmd() function waits for the command status to 49098e727371SKashyap D Desai * change after sending the command. This function is called from 4910665484d8SDoug Ambrisko * mrsas_complete_mptmfi_passthru() to wake up the sleep thread associated. 4911665484d8SDoug Ambrisko */ 49128e727371SKashyap D Desai void 49138e727371SKashyap D Desai mrsas_complete_abort(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 4914665484d8SDoug Ambrisko { 4915665484d8SDoug Ambrisko if (cmd->sync_cmd) { 4916665484d8SDoug Ambrisko cmd->sync_cmd = 0; 4917665484d8SDoug Ambrisko cmd->cmd_status = 0; 4918665484d8SDoug Ambrisko sc->chan = (void *)&cmd; 4919665484d8SDoug Ambrisko wakeup_one((void *)&sc->chan); 4920665484d8SDoug Ambrisko } 4921665484d8SDoug Ambrisko return; 4922665484d8SDoug Ambrisko } 4923665484d8SDoug Ambrisko 49248e727371SKashyap D Desai /* 49258e727371SKashyap D Desai * mrsas_aen_handler: AEN processing callback function from thread context 4926665484d8SDoug Ambrisko * input: Adapter soft state 4927665484d8SDoug Ambrisko * 49288e727371SKashyap D Desai * Asynchronous event handler 4929665484d8SDoug Ambrisko */ 49308e727371SKashyap D Desai void 49318e727371SKashyap D Desai mrsas_aen_handler(struct mrsas_softc *sc) 4932665484d8SDoug Ambrisko { 4933665484d8SDoug Ambrisko union mrsas_evt_class_locale class_locale; 4934665484d8SDoug Ambrisko int doscan = 0; 4935665484d8SDoug Ambrisko u_int32_t seq_num; 4936f0c7594bSKashyap D Desai int error, fail_aen = 0; 4937665484d8SDoug Ambrisko 49385bae00d6SSteven Hartland if (sc == NULL) { 49395bae00d6SSteven Hartland printf("invalid instance!\n"); 4940665484d8SDoug Ambrisko return; 4941665484d8SDoug Ambrisko } 494285c0a961SKashyap D Desai if (sc->remove_in_progress || sc->reset_in_progress) { 494385c0a961SKashyap D Desai device_printf(sc->mrsas_dev, "Returning from %s, line no %d\n", 494485c0a961SKashyap D Desai __func__, __LINE__); 494585c0a961SKashyap D Desai return; 494685c0a961SKashyap D Desai } 4947665484d8SDoug Ambrisko if (sc->evt_detail_mem) { 4948665484d8SDoug Ambrisko switch (sc->evt_detail_mem->code) { 4949665484d8SDoug Ambrisko case MR_EVT_PD_INSERTED: 4950f0c7594bSKashyap D Desai fail_aen = mrsas_get_pd_list(sc); 4951f0c7594bSKashyap D Desai if (!fail_aen) 4952665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_1); 4953f0c7594bSKashyap D Desai else 4954f0c7594bSKashyap D Desai goto skip_register_aen; 4955665484d8SDoug Ambrisko break; 4956665484d8SDoug Ambrisko case MR_EVT_PD_REMOVED: 4957f0c7594bSKashyap D Desai fail_aen = mrsas_get_pd_list(sc); 4958f0c7594bSKashyap D Desai if (!fail_aen) 4959665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_1); 4960f0c7594bSKashyap D Desai else 4961f0c7594bSKashyap D Desai goto skip_register_aen; 4962665484d8SDoug Ambrisko break; 4963665484d8SDoug Ambrisko case MR_EVT_LD_OFFLINE: 4964665484d8SDoug Ambrisko case MR_EVT_CFG_CLEARED: 4965665484d8SDoug Ambrisko case MR_EVT_LD_DELETED: 4966665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_0); 4967665484d8SDoug Ambrisko break; 4968665484d8SDoug Ambrisko case MR_EVT_LD_CREATED: 4969f0c7594bSKashyap D Desai fail_aen = mrsas_get_ld_list(sc); 4970f0c7594bSKashyap D Desai if (!fail_aen) 4971665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_0); 4972f0c7594bSKashyap D Desai else 4973f0c7594bSKashyap D Desai goto skip_register_aen; 4974665484d8SDoug Ambrisko break; 4975665484d8SDoug Ambrisko case MR_EVT_CTRL_HOST_BUS_SCAN_REQUESTED: 4976665484d8SDoug Ambrisko case MR_EVT_FOREIGN_CFG_IMPORTED: 4977665484d8SDoug Ambrisko case MR_EVT_LD_STATE_CHANGE: 4978665484d8SDoug Ambrisko doscan = 1; 4979665484d8SDoug Ambrisko break; 49808bc320adSKashyap D Desai case MR_EVT_CTRL_PROP_CHANGED: 49818bc320adSKashyap D Desai fail_aen = mrsas_get_ctrl_info(sc); 49828bc320adSKashyap D Desai if (fail_aen) 49838bc320adSKashyap D Desai goto skip_register_aen; 49848bc320adSKashyap D Desai break; 4985665484d8SDoug Ambrisko default: 4986665484d8SDoug Ambrisko break; 4987665484d8SDoug Ambrisko } 4988665484d8SDoug Ambrisko } else { 4989665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "invalid evt_detail\n"); 4990665484d8SDoug Ambrisko return; 4991665484d8SDoug Ambrisko } 4992665484d8SDoug Ambrisko if (doscan) { 4993f0c7594bSKashyap D Desai fail_aen = mrsas_get_pd_list(sc); 4994f0c7594bSKashyap D Desai if (!fail_aen) { 4995665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_AEN, "scanning ...sim 1\n"); 4996665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_1); 4997f0c7594bSKashyap D Desai } else 4998f0c7594bSKashyap D Desai goto skip_register_aen; 4999f0c7594bSKashyap D Desai 5000f0c7594bSKashyap D Desai fail_aen = mrsas_get_ld_list(sc); 5001f0c7594bSKashyap D Desai if (!fail_aen) { 5002665484d8SDoug Ambrisko mrsas_dprint(sc, MRSAS_AEN, "scanning ...sim 0\n"); 5003665484d8SDoug Ambrisko mrsas_bus_scan_sim(sc, sc->sim_0); 5004f0c7594bSKashyap D Desai } else 5005f0c7594bSKashyap D Desai goto skip_register_aen; 5006665484d8SDoug Ambrisko } 5007665484d8SDoug Ambrisko seq_num = sc->evt_detail_mem->seq_num + 1; 5008665484d8SDoug Ambrisko 50098e727371SKashyap D Desai /* Register AEN with FW for latest sequence number plus 1 */ 5010665484d8SDoug Ambrisko class_locale.members.reserved = 0; 5011665484d8SDoug Ambrisko class_locale.members.locale = MR_EVT_LOCALE_ALL; 5012665484d8SDoug Ambrisko class_locale.members.class = MR_EVT_CLASS_DEBUG; 5013665484d8SDoug Ambrisko 5014665484d8SDoug Ambrisko if (sc->aen_cmd != NULL) 5015665484d8SDoug Ambrisko return; 5016665484d8SDoug Ambrisko 5017665484d8SDoug Ambrisko mtx_lock(&sc->aen_lock); 5018665484d8SDoug Ambrisko error = mrsas_register_aen(sc, seq_num, 5019665484d8SDoug Ambrisko class_locale.word); 5020665484d8SDoug Ambrisko mtx_unlock(&sc->aen_lock); 5021665484d8SDoug Ambrisko 5022665484d8SDoug Ambrisko if (error) 5023665484d8SDoug Ambrisko device_printf(sc->mrsas_dev, "register aen failed error %x\n", error); 5024665484d8SDoug Ambrisko 5025f0c7594bSKashyap D Desai skip_register_aen: 5026f0c7594bSKashyap D Desai return; 5027f0c7594bSKashyap D Desai 5028665484d8SDoug Ambrisko } 5029665484d8SDoug Ambrisko 50308e727371SKashyap D Desai /* 5031665484d8SDoug Ambrisko * mrsas_complete_aen: Completes AEN command 5032665484d8SDoug Ambrisko * input: Adapter soft state 5033665484d8SDoug Ambrisko * Cmd that was issued to abort another cmd 5034665484d8SDoug Ambrisko * 50358e727371SKashyap D Desai * This function will be called from ISR and will continue event processing from 50368e727371SKashyap D Desai * thread context by enqueuing task in ev_tq (callback function 50378e727371SKashyap D Desai * "mrsas_aen_handler"). 5038665484d8SDoug Ambrisko */ 50398e727371SKashyap D Desai void 50408e727371SKashyap D Desai mrsas_complete_aen(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd) 5041665484d8SDoug Ambrisko { 5042665484d8SDoug Ambrisko /* 50438e727371SKashyap D Desai * Don't signal app if it is just an aborted previously registered 50448e727371SKashyap D Desai * aen 5045665484d8SDoug Ambrisko */ 5046665484d8SDoug Ambrisko if ((!cmd->abort_aen) && (sc->remove_in_progress == 0)) { 5047da011113SKashyap D Desai sc->mrsas_aen_triggered = 1; 5048ecea5be4SKashyap D Desai mtx_lock(&sc->aen_lock); 5049da011113SKashyap D Desai if (sc->mrsas_poll_waiting) { 5050da011113SKashyap D Desai sc->mrsas_poll_waiting = 0; 5051da011113SKashyap D Desai selwakeup(&sc->mrsas_select); 5052da011113SKashyap D Desai } 5053ecea5be4SKashyap D Desai mtx_unlock(&sc->aen_lock); 50548e727371SKashyap D Desai } else 5055665484d8SDoug Ambrisko cmd->abort_aen = 0; 5056665484d8SDoug Ambrisko 5057665484d8SDoug Ambrisko sc->aen_cmd = NULL; 5058665484d8SDoug Ambrisko mrsas_release_mfi_cmd(cmd); 5059665484d8SDoug Ambrisko 5060665484d8SDoug Ambrisko taskqueue_enqueue(sc->ev_tq, &sc->ev_task); 5061665484d8SDoug Ambrisko 5062665484d8SDoug Ambrisko return; 5063665484d8SDoug Ambrisko } 5064665484d8SDoug Ambrisko 5065665484d8SDoug Ambrisko static device_method_t mrsas_methods[] = { 5066665484d8SDoug Ambrisko DEVMETHOD(device_probe, mrsas_probe), 5067665484d8SDoug Ambrisko DEVMETHOD(device_attach, mrsas_attach), 5068665484d8SDoug Ambrisko DEVMETHOD(device_detach, mrsas_detach), 5069f28ecf2bSAndriy Gapon DEVMETHOD(device_shutdown, mrsas_shutdown), 5070665484d8SDoug Ambrisko DEVMETHOD(device_suspend, mrsas_suspend), 5071665484d8SDoug Ambrisko DEVMETHOD(device_resume, mrsas_resume), 5072665484d8SDoug Ambrisko DEVMETHOD(bus_print_child, bus_generic_print_child), 5073665484d8SDoug Ambrisko DEVMETHOD(bus_driver_added, bus_generic_driver_added), 5074665484d8SDoug Ambrisko {0, 0} 5075665484d8SDoug Ambrisko }; 5076665484d8SDoug Ambrisko 5077665484d8SDoug Ambrisko static driver_t mrsas_driver = { 5078665484d8SDoug Ambrisko "mrsas", 5079665484d8SDoug Ambrisko mrsas_methods, 5080665484d8SDoug Ambrisko sizeof(struct mrsas_softc) 5081665484d8SDoug Ambrisko }; 5082665484d8SDoug Ambrisko 5083665484d8SDoug Ambrisko static devclass_t mrsas_devclass; 50848e727371SKashyap D Desai 5085665484d8SDoug Ambrisko DRIVER_MODULE(mrsas, pci, mrsas_driver, mrsas_devclass, 0, 0); 5086665484d8SDoug Ambrisko MODULE_DEPEND(mrsas, cam, 1, 1, 1); 5087