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