xref: /freebsd/sys/dev/mrsas/mrsas.c (revision e34a057ca6ebdf8e30ec8b0dc21d18eb450bf36a)
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