xref: /freebsd/sys/dev/mrsas/mrsas.c (revision 77cf7df80453bed271e94284831ceb7bdc580486)
1665484d8SDoug Ambrisko /*
28e727371SKashyap D Desai  * Copyright (c) 2014, LSI Corp. All rights reserved. Author: Marian Choy
3665484d8SDoug Ambrisko  * Support: freebsdraid@lsi.com
4665484d8SDoug Ambrisko  *
5665484d8SDoug Ambrisko  * Redistribution and use in source and binary forms, with or without
68e727371SKashyap D Desai  * modification, are permitted provided that the following conditions are
78e727371SKashyap D Desai  * met:
8665484d8SDoug Ambrisko  *
98e727371SKashyap D Desai  * 1. Redistributions of source code must retain the above copyright notice,
108e727371SKashyap D Desai  * this list of conditions and the following disclaimer. 2. Redistributions
118e727371SKashyap D Desai  * in binary form must reproduce the above copyright notice, this list of
128e727371SKashyap D Desai  * conditions and the following disclaimer in the documentation and/or other
138e727371SKashyap D Desai  * materials provided with the distribution. 3. Neither the name of the
148e727371SKashyap D Desai  * <ORGANIZATION> nor the names of its contributors may be used to endorse or
158e727371SKashyap D Desai  * promote products derived from this software without specific prior written
168e727371SKashyap D Desai  * permission.
17665484d8SDoug Ambrisko  *
188e727371SKashyap D Desai  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
198e727371SKashyap D Desai  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
208e727371SKashyap D Desai  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
218e727371SKashyap D Desai  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
228e727371SKashyap D Desai  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
238e727371SKashyap D Desai  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
248e727371SKashyap D Desai  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
258e727371SKashyap D Desai  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
268e727371SKashyap D Desai  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
278e727371SKashyap D Desai  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
28665484d8SDoug Ambrisko  * POSSIBILITY OF SUCH DAMAGE.
29665484d8SDoug Ambrisko  *
308e727371SKashyap D Desai  * The views and conclusions contained in the software and documentation are
318e727371SKashyap D Desai  * those of the authors and should not be interpreted as representing
32665484d8SDoug Ambrisko  * official policies,either expressed or implied, of the FreeBSD Project.
33665484d8SDoug Ambrisko  *
348e727371SKashyap D Desai  * Send feedback to: <megaraidfbsd@lsi.com> Mail to: LSI Corporation, 1621
358e727371SKashyap D Desai  * Barber Lane, Milpitas, CA 95035 ATTN: MegaRaid FreeBSD
36665484d8SDoug Ambrisko  *
37665484d8SDoug Ambrisko  */
38665484d8SDoug Ambrisko 
39665484d8SDoug Ambrisko #include <sys/cdefs.h>
40665484d8SDoug Ambrisko __FBSDID("$FreeBSD$");
41665484d8SDoug Ambrisko 
42665484d8SDoug Ambrisko #include <dev/mrsas/mrsas.h>
43665484d8SDoug Ambrisko #include <dev/mrsas/mrsas_ioctl.h>
44665484d8SDoug Ambrisko 
45665484d8SDoug Ambrisko #include <cam/cam.h>
46665484d8SDoug Ambrisko #include <cam/cam_ccb.h>
47665484d8SDoug Ambrisko 
48665484d8SDoug Ambrisko #include <sys/sysctl.h>
49665484d8SDoug Ambrisko #include <sys/types.h>
50665484d8SDoug Ambrisko #include <sys/kthread.h>
51665484d8SDoug Ambrisko #include <sys/taskqueue.h>
52d18d1b47SKashyap D Desai #include <sys/smp.h>
53665484d8SDoug Ambrisko 
54665484d8SDoug Ambrisko 
55665484d8SDoug Ambrisko /*
56665484d8SDoug Ambrisko  * Function prototypes
57665484d8SDoug Ambrisko  */
58665484d8SDoug Ambrisko static d_open_t mrsas_open;
59665484d8SDoug Ambrisko static d_close_t mrsas_close;
60665484d8SDoug Ambrisko static d_read_t mrsas_read;
61665484d8SDoug Ambrisko static d_write_t mrsas_write;
62665484d8SDoug Ambrisko static d_ioctl_t mrsas_ioctl;
63da011113SKashyap D Desai static d_poll_t mrsas_poll;
64665484d8SDoug Ambrisko 
65536094dcSKashyap D Desai static struct mrsas_mgmt_info mrsas_mgmt_info;
66665484d8SDoug Ambrisko static struct mrsas_ident *mrsas_find_ident(device_t);
67d18d1b47SKashyap D Desai static int mrsas_setup_msix(struct mrsas_softc *sc);
68d18d1b47SKashyap D Desai static int mrsas_allocate_msix(struct mrsas_softc *sc);
69665484d8SDoug Ambrisko static void mrsas_shutdown_ctlr(struct mrsas_softc *sc, u_int32_t opcode);
70665484d8SDoug Ambrisko static void mrsas_flush_cache(struct mrsas_softc *sc);
71665484d8SDoug Ambrisko static void mrsas_reset_reply_desc(struct mrsas_softc *sc);
72665484d8SDoug Ambrisko static void mrsas_ocr_thread(void *arg);
73665484d8SDoug Ambrisko static int mrsas_get_map_info(struct mrsas_softc *sc);
74665484d8SDoug Ambrisko static int mrsas_get_ld_map_info(struct mrsas_softc *sc);
75665484d8SDoug Ambrisko static int mrsas_sync_map_info(struct mrsas_softc *sc);
76665484d8SDoug Ambrisko static int mrsas_get_pd_list(struct mrsas_softc *sc);
77665484d8SDoug Ambrisko static int mrsas_get_ld_list(struct mrsas_softc *sc);
78665484d8SDoug Ambrisko static int mrsas_setup_irq(struct mrsas_softc *sc);
79665484d8SDoug Ambrisko static int mrsas_alloc_mem(struct mrsas_softc *sc);
80665484d8SDoug Ambrisko static int mrsas_init_fw(struct mrsas_softc *sc);
81665484d8SDoug Ambrisko static int mrsas_setup_raidmap(struct mrsas_softc *sc);
82d18d1b47SKashyap D Desai static int mrsas_complete_cmd(struct mrsas_softc *sc, u_int32_t MSIxIndex);
83665484d8SDoug Ambrisko static int mrsas_clear_intr(struct mrsas_softc *sc);
848e727371SKashyap D Desai static int
858e727371SKashyap D Desai mrsas_get_ctrl_info(struct mrsas_softc *sc,
86665484d8SDoug Ambrisko     struct mrsas_ctrl_info *ctrl_info);
878e727371SKashyap D Desai static int
888e727371SKashyap D Desai mrsas_issue_blocked_abort_cmd(struct mrsas_softc *sc,
89665484d8SDoug Ambrisko     struct mrsas_mfi_cmd *cmd_to_abort);
905844115eSKashyap D Desai static struct mrsas_softc *mrsas_get_softc_instance(struct cdev *dev,
915844115eSKashyap D Desai 						u_long cmd, caddr_t arg);
92665484d8SDoug Ambrisko u_int32_t mrsas_read_reg(struct mrsas_softc *sc, int offset);
938e727371SKashyap D Desai u_int8_t
948e727371SKashyap D Desai mrsas_build_mptmfi_passthru(struct mrsas_softc *sc,
95665484d8SDoug Ambrisko     struct mrsas_mfi_cmd *mfi_cmd);
96665484d8SDoug Ambrisko int	mrsas_transition_to_ready(struct mrsas_softc *sc, int ocr);
97665484d8SDoug Ambrisko int	mrsas_init_adapter(struct mrsas_softc *sc);
98665484d8SDoug Ambrisko int	mrsas_alloc_mpt_cmds(struct mrsas_softc *sc);
99665484d8SDoug Ambrisko int	mrsas_alloc_ioc_cmd(struct mrsas_softc *sc);
100665484d8SDoug Ambrisko int	mrsas_alloc_ctlr_info_cmd(struct mrsas_softc *sc);
101665484d8SDoug Ambrisko int	mrsas_ioc_init(struct mrsas_softc *sc);
102665484d8SDoug Ambrisko int	mrsas_bus_scan(struct mrsas_softc *sc);
103665484d8SDoug Ambrisko int	mrsas_issue_dcmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd);
104665484d8SDoug Ambrisko int	mrsas_issue_polled(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd);
105665484d8SDoug Ambrisko int	mrsas_reset_ctrl(struct mrsas_softc *sc);
106665484d8SDoug Ambrisko int	mrsas_wait_for_outstanding(struct mrsas_softc *sc);
1078e727371SKashyap D Desai int
1088e727371SKashyap D Desai mrsas_issue_blocked_cmd(struct mrsas_softc *sc,
109665484d8SDoug Ambrisko     struct mrsas_mfi_cmd *cmd);
1108e727371SKashyap D Desai int
1118e727371SKashyap D Desai mrsas_alloc_tmp_dcmd(struct mrsas_softc *sc, struct mrsas_tmp_dcmd *tcmd,
112665484d8SDoug Ambrisko     int size);
113665484d8SDoug Ambrisko void	mrsas_release_mfi_cmd(struct mrsas_mfi_cmd *cmd);
114665484d8SDoug Ambrisko void	mrsas_wakeup(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd);
115665484d8SDoug Ambrisko void	mrsas_complete_aen(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd);
116665484d8SDoug Ambrisko void	mrsas_complete_abort(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd);
117665484d8SDoug Ambrisko void	mrsas_disable_intr(struct mrsas_softc *sc);
118665484d8SDoug Ambrisko void	mrsas_enable_intr(struct mrsas_softc *sc);
119665484d8SDoug Ambrisko void	mrsas_free_ioc_cmd(struct mrsas_softc *sc);
120665484d8SDoug Ambrisko void	mrsas_free_mem(struct mrsas_softc *sc);
121665484d8SDoug Ambrisko void	mrsas_free_tmp_dcmd(struct mrsas_tmp_dcmd *tmp);
122665484d8SDoug Ambrisko void	mrsas_isr(void *arg);
123665484d8SDoug Ambrisko void	mrsas_teardown_intr(struct mrsas_softc *sc);
124665484d8SDoug Ambrisko void	mrsas_addr_cb(void *arg, bus_dma_segment_t *segs, int nsegs, int error);
125665484d8SDoug Ambrisko void	mrsas_kill_hba(struct mrsas_softc *sc);
126665484d8SDoug Ambrisko void	mrsas_aen_handler(struct mrsas_softc *sc);
1278e727371SKashyap D Desai void
1288e727371SKashyap D Desai mrsas_write_reg(struct mrsas_softc *sc, int offset,
129665484d8SDoug Ambrisko     u_int32_t value);
1308e727371SKashyap D Desai void
1318e727371SKashyap D Desai mrsas_fire_cmd(struct mrsas_softc *sc, u_int32_t req_desc_lo,
132665484d8SDoug Ambrisko     u_int32_t req_desc_hi);
133665484d8SDoug Ambrisko void	mrsas_free_ctlr_info_cmd(struct mrsas_softc *sc);
1348e727371SKashyap D Desai void
1358e727371SKashyap D Desai mrsas_complete_mptmfi_passthru(struct mrsas_softc *sc,
136665484d8SDoug Ambrisko     struct mrsas_mfi_cmd *cmd, u_int8_t status);
1378e727371SKashyap D Desai void
1388e727371SKashyap D Desai mrsas_map_mpt_cmd_status(struct mrsas_mpt_cmd *cmd, u_int8_t status,
139665484d8SDoug Ambrisko     u_int8_t extStatus);
140665484d8SDoug Ambrisko struct mrsas_mfi_cmd *mrsas_get_mfi_cmd(struct mrsas_softc *sc);
1418e727371SKashyap D Desai 
1428e727371SKashyap D Desai MRSAS_REQUEST_DESCRIPTOR_UNION *mrsas_build_mpt_cmd
1438e727371SKashyap D Desai         (struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd);
144665484d8SDoug Ambrisko 
145665484d8SDoug Ambrisko extern int mrsas_cam_attach(struct mrsas_softc *sc);
146665484d8SDoug Ambrisko extern void mrsas_cam_detach(struct mrsas_softc *sc);
147665484d8SDoug Ambrisko extern void mrsas_cmd_done(struct mrsas_softc *sc, struct mrsas_mpt_cmd *cmd);
148665484d8SDoug Ambrisko extern void mrsas_free_frame(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd);
149665484d8SDoug Ambrisko extern int mrsas_alloc_mfi_cmds(struct mrsas_softc *sc);
150665484d8SDoug Ambrisko extern void mrsas_release_mpt_cmd(struct mrsas_mpt_cmd *cmd);
151665484d8SDoug Ambrisko extern struct mrsas_mpt_cmd *mrsas_get_mpt_cmd(struct mrsas_softc *sc);
152536094dcSKashyap D Desai extern int mrsas_passthru(struct mrsas_softc *sc, void *arg, u_long ioctlCmd);
153665484d8SDoug Ambrisko extern uint8_t MR_ValidateMapInfo(struct mrsas_softc *sc);
1544799d485SKashyap D Desai extern u_int16_t MR_GetLDTgtId(u_int32_t ld, MR_DRV_RAID_MAP_ALL * map);
1554799d485SKashyap D Desai extern MR_LD_RAID *MR_LdRaidGet(u_int32_t ld, MR_DRV_RAID_MAP_ALL * map);
156665484d8SDoug Ambrisko extern void mrsas_xpt_freeze(struct mrsas_softc *sc);
157665484d8SDoug Ambrisko extern void mrsas_xpt_release(struct mrsas_softc *sc);
1588e727371SKashyap D Desai extern MRSAS_REQUEST_DESCRIPTOR_UNION *
1598e727371SKashyap D Desai mrsas_get_request_desc(struct mrsas_softc *sc,
160665484d8SDoug Ambrisko     u_int16_t index);
161665484d8SDoug Ambrisko extern int mrsas_bus_scan_sim(struct mrsas_softc *sc, struct cam_sim *sim);
162665484d8SDoug Ambrisko static int mrsas_alloc_evt_log_info_cmd(struct mrsas_softc *sc);
163665484d8SDoug Ambrisko static void mrsas_free_evt_log_info_cmd(struct mrsas_softc *sc);
1648e727371SKashyap D Desai 
165665484d8SDoug Ambrisko SYSCTL_NODE(_hw, OID_AUTO, mrsas, CTLFLAG_RD, 0, "MRSAS Driver Parameters");
166665484d8SDoug Ambrisko 
1678e727371SKashyap D Desai /*
168665484d8SDoug Ambrisko  * PCI device struct and table
169665484d8SDoug Ambrisko  *
170665484d8SDoug Ambrisko  */
171665484d8SDoug Ambrisko typedef struct mrsas_ident {
172665484d8SDoug Ambrisko 	uint16_t vendor;
173665484d8SDoug Ambrisko 	uint16_t device;
174665484d8SDoug Ambrisko 	uint16_t subvendor;
175665484d8SDoug Ambrisko 	uint16_t subdevice;
176665484d8SDoug Ambrisko 	const char *desc;
177665484d8SDoug Ambrisko }	MRSAS_CTLR_ID;
178665484d8SDoug Ambrisko 
179665484d8SDoug Ambrisko MRSAS_CTLR_ID device_table[] = {
180665484d8SDoug Ambrisko 	{0x1000, MRSAS_TBOLT, 0xffff, 0xffff, "LSI Thunderbolt SAS Controller"},
181665484d8SDoug Ambrisko 	{0x1000, MRSAS_INVADER, 0xffff, 0xffff, "LSI Invader SAS Controller"},
182665484d8SDoug Ambrisko 	{0x1000, MRSAS_FURY, 0xffff, 0xffff, "LSI Fury SAS Controller"},
183665484d8SDoug Ambrisko 	{0, 0, 0, 0, NULL}
184665484d8SDoug Ambrisko };
185665484d8SDoug Ambrisko 
1868e727371SKashyap D Desai /*
187665484d8SDoug Ambrisko  * Character device entry points
188665484d8SDoug Ambrisko  *
189665484d8SDoug Ambrisko  */
190665484d8SDoug Ambrisko static struct cdevsw mrsas_cdevsw = {
191665484d8SDoug Ambrisko 	.d_version = D_VERSION,
192665484d8SDoug Ambrisko 	.d_open = mrsas_open,
193665484d8SDoug Ambrisko 	.d_close = mrsas_close,
194665484d8SDoug Ambrisko 	.d_read = mrsas_read,
195665484d8SDoug Ambrisko 	.d_write = mrsas_write,
196665484d8SDoug Ambrisko 	.d_ioctl = mrsas_ioctl,
197da011113SKashyap D Desai 	.d_poll = mrsas_poll,
198665484d8SDoug Ambrisko 	.d_name = "mrsas",
199665484d8SDoug Ambrisko };
200665484d8SDoug Ambrisko 
201665484d8SDoug Ambrisko MALLOC_DEFINE(M_MRSAS, "mrsasbuf", "Buffers for the MRSAS driver");
202665484d8SDoug Ambrisko 
2038e727371SKashyap D Desai /*
2048e727371SKashyap D Desai  * In the cdevsw routines, we find our softc by using the si_drv1 member of
2058e727371SKashyap D Desai  * struct cdev.  We set this variable to point to our softc in our attach
2068e727371SKashyap D Desai  * routine when we create the /dev entry.
207665484d8SDoug Ambrisko  */
208665484d8SDoug Ambrisko int
2097fc5f329SJohn Baldwin mrsas_open(struct cdev *dev, int oflags, int devtype, struct thread *td)
210665484d8SDoug Ambrisko {
211665484d8SDoug Ambrisko 	struct mrsas_softc *sc;
212665484d8SDoug Ambrisko 
213665484d8SDoug Ambrisko 	sc = dev->si_drv1;
214665484d8SDoug Ambrisko 	return (0);
215665484d8SDoug Ambrisko }
216665484d8SDoug Ambrisko 
217665484d8SDoug Ambrisko int
2187fc5f329SJohn Baldwin mrsas_close(struct cdev *dev, int fflag, int devtype, struct thread *td)
219665484d8SDoug Ambrisko {
220665484d8SDoug Ambrisko 	struct mrsas_softc *sc;
221665484d8SDoug Ambrisko 
222665484d8SDoug Ambrisko 	sc = dev->si_drv1;
223665484d8SDoug Ambrisko 	return (0);
224665484d8SDoug Ambrisko }
225665484d8SDoug Ambrisko 
226665484d8SDoug Ambrisko int
227665484d8SDoug Ambrisko mrsas_read(struct cdev *dev, struct uio *uio, int ioflag)
228665484d8SDoug Ambrisko {
229665484d8SDoug Ambrisko 	struct mrsas_softc *sc;
230665484d8SDoug Ambrisko 
231665484d8SDoug Ambrisko 	sc = dev->si_drv1;
232665484d8SDoug Ambrisko 	return (0);
233665484d8SDoug Ambrisko }
234665484d8SDoug Ambrisko int
235665484d8SDoug Ambrisko mrsas_write(struct cdev *dev, struct uio *uio, int ioflag)
236665484d8SDoug Ambrisko {
237665484d8SDoug Ambrisko 	struct mrsas_softc *sc;
238665484d8SDoug Ambrisko 
239665484d8SDoug Ambrisko 	sc = dev->si_drv1;
240665484d8SDoug Ambrisko 	return (0);
241665484d8SDoug Ambrisko }
242665484d8SDoug Ambrisko 
2438e727371SKashyap D Desai /*
244665484d8SDoug Ambrisko  * Register Read/Write Functions
245665484d8SDoug Ambrisko  *
246665484d8SDoug Ambrisko  */
247665484d8SDoug Ambrisko void
248665484d8SDoug Ambrisko mrsas_write_reg(struct mrsas_softc *sc, int offset,
249665484d8SDoug Ambrisko     u_int32_t value)
250665484d8SDoug Ambrisko {
251665484d8SDoug Ambrisko 	bus_space_tag_t bus_tag = sc->bus_tag;
252665484d8SDoug Ambrisko 	bus_space_handle_t bus_handle = sc->bus_handle;
253665484d8SDoug Ambrisko 
254665484d8SDoug Ambrisko 	bus_space_write_4(bus_tag, bus_handle, offset, value);
255665484d8SDoug Ambrisko }
256665484d8SDoug Ambrisko 
257665484d8SDoug Ambrisko u_int32_t
258665484d8SDoug Ambrisko mrsas_read_reg(struct mrsas_softc *sc, int offset)
259665484d8SDoug Ambrisko {
260665484d8SDoug Ambrisko 	bus_space_tag_t bus_tag = sc->bus_tag;
261665484d8SDoug Ambrisko 	bus_space_handle_t bus_handle = sc->bus_handle;
262665484d8SDoug Ambrisko 
263665484d8SDoug Ambrisko 	return ((u_int32_t)bus_space_read_4(bus_tag, bus_handle, offset));
264665484d8SDoug Ambrisko }
265665484d8SDoug Ambrisko 
266665484d8SDoug Ambrisko 
2678e727371SKashyap D Desai /*
268665484d8SDoug Ambrisko  * Interrupt Disable/Enable/Clear Functions
269665484d8SDoug Ambrisko  *
270665484d8SDoug Ambrisko  */
2718e727371SKashyap D Desai void
2728e727371SKashyap D Desai mrsas_disable_intr(struct mrsas_softc *sc)
273665484d8SDoug Ambrisko {
274665484d8SDoug Ambrisko 	u_int32_t mask = 0xFFFFFFFF;
275665484d8SDoug Ambrisko 	u_int32_t status;
276665484d8SDoug Ambrisko 
277665484d8SDoug Ambrisko 	mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask), mask);
278665484d8SDoug Ambrisko 	/* Dummy read to force pci flush */
279665484d8SDoug Ambrisko 	status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask));
280665484d8SDoug Ambrisko }
281665484d8SDoug Ambrisko 
2828e727371SKashyap D Desai void
2838e727371SKashyap D Desai mrsas_enable_intr(struct mrsas_softc *sc)
284665484d8SDoug Ambrisko {
285665484d8SDoug Ambrisko 	u_int32_t mask = MFI_FUSION_ENABLE_INTERRUPT_MASK;
286665484d8SDoug Ambrisko 	u_int32_t status;
287665484d8SDoug Ambrisko 
288665484d8SDoug Ambrisko 	mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status), ~0);
289665484d8SDoug Ambrisko 	status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status));
290665484d8SDoug Ambrisko 
291665484d8SDoug Ambrisko 	mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask), ~mask);
292665484d8SDoug Ambrisko 	status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask));
293665484d8SDoug Ambrisko }
294665484d8SDoug Ambrisko 
2958e727371SKashyap D Desai static int
2968e727371SKashyap D Desai mrsas_clear_intr(struct mrsas_softc *sc)
297665484d8SDoug Ambrisko {
298665484d8SDoug Ambrisko 	u_int32_t status, fw_status, fw_state;
299665484d8SDoug Ambrisko 
300665484d8SDoug Ambrisko 	/* Read received interrupt */
301665484d8SDoug Ambrisko 	status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status));
302665484d8SDoug Ambrisko 
3038e727371SKashyap D Desai 	/*
3048e727371SKashyap D Desai 	 * If FW state change interrupt is received, write to it again to
3058e727371SKashyap D Desai 	 * clear
3068e727371SKashyap D Desai 	 */
307665484d8SDoug Ambrisko 	if (status & MRSAS_FW_STATE_CHNG_INTERRUPT) {
308665484d8SDoug Ambrisko 		fw_status = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
309665484d8SDoug Ambrisko 		    outbound_scratch_pad));
310665484d8SDoug Ambrisko 		fw_state = fw_status & MFI_STATE_MASK;
311665484d8SDoug Ambrisko 		if (fw_state == MFI_STATE_FAULT) {
312665484d8SDoug Ambrisko 			device_printf(sc->mrsas_dev, "FW is in FAULT state!\n");
313665484d8SDoug Ambrisko 			if (sc->ocr_thread_active)
314665484d8SDoug Ambrisko 				wakeup(&sc->ocr_chan);
315665484d8SDoug Ambrisko 		}
316665484d8SDoug Ambrisko 		mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status), status);
317665484d8SDoug Ambrisko 		mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status));
318665484d8SDoug Ambrisko 		return (1);
319665484d8SDoug Ambrisko 	}
320665484d8SDoug Ambrisko 	/* Not our interrupt, so just return */
321665484d8SDoug Ambrisko 	if (!(status & MFI_FUSION_ENABLE_INTERRUPT_MASK))
322665484d8SDoug Ambrisko 		return (0);
323665484d8SDoug Ambrisko 
324665484d8SDoug Ambrisko 	/* We got a reply interrupt */
325665484d8SDoug Ambrisko 	return (1);
326665484d8SDoug Ambrisko }
327665484d8SDoug Ambrisko 
3288e727371SKashyap D Desai /*
329665484d8SDoug Ambrisko  * PCI Support Functions
330665484d8SDoug Ambrisko  *
331665484d8SDoug Ambrisko  */
3328e727371SKashyap D Desai static struct mrsas_ident *
3338e727371SKashyap D Desai mrsas_find_ident(device_t dev)
334665484d8SDoug Ambrisko {
335665484d8SDoug Ambrisko 	struct mrsas_ident *pci_device;
336665484d8SDoug Ambrisko 
3378e727371SKashyap D Desai 	for (pci_device = device_table; pci_device->vendor != 0; pci_device++) {
338665484d8SDoug Ambrisko 		if ((pci_device->vendor == pci_get_vendor(dev)) &&
339665484d8SDoug Ambrisko 		    (pci_device->device == pci_get_device(dev)) &&
340665484d8SDoug Ambrisko 		    ((pci_device->subvendor == pci_get_subvendor(dev)) ||
341665484d8SDoug Ambrisko 		    (pci_device->subvendor == 0xffff)) &&
342665484d8SDoug Ambrisko 		    ((pci_device->subdevice == pci_get_subdevice(dev)) ||
343665484d8SDoug Ambrisko 		    (pci_device->subdevice == 0xffff)))
344665484d8SDoug Ambrisko 			return (pci_device);
345665484d8SDoug Ambrisko 	}
346665484d8SDoug Ambrisko 	return (NULL);
347665484d8SDoug Ambrisko }
348665484d8SDoug Ambrisko 
3498e727371SKashyap D Desai static int
3508e727371SKashyap D Desai mrsas_probe(device_t dev)
351665484d8SDoug Ambrisko {
352665484d8SDoug Ambrisko 	static u_int8_t first_ctrl = 1;
353665484d8SDoug Ambrisko 	struct mrsas_ident *id;
354665484d8SDoug Ambrisko 
355665484d8SDoug Ambrisko 	if ((id = mrsas_find_ident(dev)) != NULL) {
356665484d8SDoug Ambrisko 		if (first_ctrl) {
3578e727371SKashyap D Desai 			printf("LSI MegaRAID SAS FreeBSD mrsas driver version: %s\n",
3588e727371SKashyap D Desai 			    MRSAS_VERSION);
359665484d8SDoug Ambrisko 			first_ctrl = 0;
360665484d8SDoug Ambrisko 		}
361665484d8SDoug Ambrisko 		device_set_desc(dev, id->desc);
362665484d8SDoug Ambrisko 		/* between BUS_PROBE_DEFAULT and BUS_PROBE_LOW_PRIORITY */
363665484d8SDoug Ambrisko 		return (-30);
364665484d8SDoug Ambrisko 	}
365665484d8SDoug Ambrisko 	return (ENXIO);
366665484d8SDoug Ambrisko }
367665484d8SDoug Ambrisko 
3688e727371SKashyap D Desai /*
369665484d8SDoug Ambrisko  * mrsas_setup_sysctl:	setup sysctl values for mrsas
370665484d8SDoug Ambrisko  * input:				Adapter instance soft state
371665484d8SDoug Ambrisko  *
372665484d8SDoug Ambrisko  * Setup sysctl entries for mrsas driver.
373665484d8SDoug Ambrisko  */
374665484d8SDoug Ambrisko static void
375665484d8SDoug Ambrisko mrsas_setup_sysctl(struct mrsas_softc *sc)
376665484d8SDoug Ambrisko {
377665484d8SDoug Ambrisko 	struct sysctl_ctx_list *sysctl_ctx = NULL;
378665484d8SDoug Ambrisko 	struct sysctl_oid *sysctl_tree = NULL;
379665484d8SDoug Ambrisko 	char tmpstr[80], tmpstr2[80];
380665484d8SDoug Ambrisko 
381665484d8SDoug Ambrisko 	/*
382665484d8SDoug Ambrisko 	 * Setup the sysctl variable so the user can change the debug level
383665484d8SDoug Ambrisko 	 * on the fly.
384665484d8SDoug Ambrisko 	 */
385665484d8SDoug Ambrisko 	snprintf(tmpstr, sizeof(tmpstr), "MRSAS controller %d",
386665484d8SDoug Ambrisko 	    device_get_unit(sc->mrsas_dev));
387665484d8SDoug Ambrisko 	snprintf(tmpstr2, sizeof(tmpstr2), "%d", device_get_unit(sc->mrsas_dev));
388665484d8SDoug Ambrisko 
389665484d8SDoug Ambrisko 	sysctl_ctx = device_get_sysctl_ctx(sc->mrsas_dev);
390665484d8SDoug Ambrisko 	if (sysctl_ctx != NULL)
391665484d8SDoug Ambrisko 		sysctl_tree = device_get_sysctl_tree(sc->mrsas_dev);
392665484d8SDoug Ambrisko 
393665484d8SDoug Ambrisko 	if (sysctl_tree == NULL) {
394665484d8SDoug Ambrisko 		sysctl_ctx_init(&sc->sysctl_ctx);
395665484d8SDoug Ambrisko 		sc->sysctl_tree = SYSCTL_ADD_NODE(&sc->sysctl_ctx,
396665484d8SDoug Ambrisko 		    SYSCTL_STATIC_CHILDREN(_hw_mrsas), OID_AUTO, tmpstr2,
397665484d8SDoug Ambrisko 		    CTLFLAG_RD, 0, tmpstr);
398665484d8SDoug Ambrisko 		if (sc->sysctl_tree == NULL)
399665484d8SDoug Ambrisko 			return;
400665484d8SDoug Ambrisko 		sysctl_ctx = &sc->sysctl_ctx;
401665484d8SDoug Ambrisko 		sysctl_tree = sc->sysctl_tree;
402665484d8SDoug Ambrisko 	}
403665484d8SDoug Ambrisko 	SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree),
404665484d8SDoug Ambrisko 	    OID_AUTO, "disable_ocr", CTLFLAG_RW, &sc->disableOnlineCtrlReset, 0,
405665484d8SDoug Ambrisko 	    "Disable the use of OCR");
406665484d8SDoug Ambrisko 
407665484d8SDoug Ambrisko 	SYSCTL_ADD_STRING(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree),
408665484d8SDoug Ambrisko 	    OID_AUTO, "driver_version", CTLFLAG_RD, MRSAS_VERSION,
409665484d8SDoug Ambrisko 	    strlen(MRSAS_VERSION), "driver version");
410665484d8SDoug Ambrisko 
411665484d8SDoug Ambrisko 	SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree),
412665484d8SDoug Ambrisko 	    OID_AUTO, "reset_count", CTLFLAG_RD,
413665484d8SDoug Ambrisko 	    &sc->reset_count, 0, "number of ocr from start of the day");
414665484d8SDoug Ambrisko 
415665484d8SDoug Ambrisko 	SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree),
416665484d8SDoug Ambrisko 	    OID_AUTO, "fw_outstanding", CTLFLAG_RD,
417f0188618SHans Petter Selasky 	    &sc->fw_outstanding.val_rdonly, 0, "FW outstanding commands");
418665484d8SDoug Ambrisko 
419665484d8SDoug Ambrisko 	SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree),
420665484d8SDoug Ambrisko 	    OID_AUTO, "io_cmds_highwater", CTLFLAG_RD,
421665484d8SDoug Ambrisko 	    &sc->io_cmds_highwater, 0, "Max FW outstanding commands");
422665484d8SDoug Ambrisko 
423665484d8SDoug Ambrisko 	SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree),
424665484d8SDoug Ambrisko 	    OID_AUTO, "mrsas_debug", CTLFLAG_RW, &sc->mrsas_debug, 0,
425665484d8SDoug Ambrisko 	    "Driver debug level");
426665484d8SDoug Ambrisko 
427665484d8SDoug Ambrisko 	SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree),
428665484d8SDoug Ambrisko 	    OID_AUTO, "mrsas_io_timeout", CTLFLAG_RW, &sc->mrsas_io_timeout,
429665484d8SDoug Ambrisko 	    0, "Driver IO timeout value in mili-second.");
430665484d8SDoug Ambrisko 
431665484d8SDoug Ambrisko 	SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree),
432665484d8SDoug Ambrisko 	    OID_AUTO, "mrsas_fw_fault_check_delay", CTLFLAG_RW,
433665484d8SDoug Ambrisko 	    &sc->mrsas_fw_fault_check_delay,
434665484d8SDoug Ambrisko 	    0, "FW fault check thread delay in seconds. <default is 1 sec>");
435665484d8SDoug Ambrisko 
436665484d8SDoug Ambrisko 	SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree),
437665484d8SDoug Ambrisko 	    OID_AUTO, "reset_in_progress", CTLFLAG_RD,
438665484d8SDoug Ambrisko 	    &sc->reset_in_progress, 0, "ocr in progress status");
439665484d8SDoug Ambrisko 
440665484d8SDoug Ambrisko }
441665484d8SDoug Ambrisko 
4428e727371SKashyap D Desai /*
443665484d8SDoug Ambrisko  * mrsas_get_tunables:	get tunable parameters.
444665484d8SDoug Ambrisko  * input:				Adapter instance soft state
445665484d8SDoug Ambrisko  *
446665484d8SDoug Ambrisko  * Get tunable parameters. This will help to debug driver at boot time.
447665484d8SDoug Ambrisko  */
448665484d8SDoug Ambrisko static void
449665484d8SDoug Ambrisko mrsas_get_tunables(struct mrsas_softc *sc)
450665484d8SDoug Ambrisko {
451665484d8SDoug Ambrisko 	char tmpstr[80];
452665484d8SDoug Ambrisko 
453665484d8SDoug Ambrisko 	/* XXX default to some debugging for now */
454665484d8SDoug Ambrisko 	sc->mrsas_debug = MRSAS_FAULT;
455665484d8SDoug Ambrisko 	sc->mrsas_io_timeout = MRSAS_IO_TIMEOUT;
456665484d8SDoug Ambrisko 	sc->mrsas_fw_fault_check_delay = 1;
457665484d8SDoug Ambrisko 	sc->reset_count = 0;
458665484d8SDoug Ambrisko 	sc->reset_in_progress = 0;
459665484d8SDoug Ambrisko 
460665484d8SDoug Ambrisko 	/*
461665484d8SDoug Ambrisko 	 * Grab the global variables.
462665484d8SDoug Ambrisko 	 */
463665484d8SDoug Ambrisko 	TUNABLE_INT_FETCH("hw.mrsas.debug_level", &sc->mrsas_debug);
464665484d8SDoug Ambrisko 
465665484d8SDoug Ambrisko 	/* Grab the unit-instance variables */
466665484d8SDoug Ambrisko 	snprintf(tmpstr, sizeof(tmpstr), "dev.mrsas.%d.debug_level",
467665484d8SDoug Ambrisko 	    device_get_unit(sc->mrsas_dev));
468665484d8SDoug Ambrisko 	TUNABLE_INT_FETCH(tmpstr, &sc->mrsas_debug);
469665484d8SDoug Ambrisko }
470665484d8SDoug Ambrisko 
4718e727371SKashyap D Desai /*
472665484d8SDoug Ambrisko  * mrsas_alloc_evt_log_info cmd: Allocates memory to get event log information.
473665484d8SDoug Ambrisko  * Used to get sequence number at driver load time.
474665484d8SDoug Ambrisko  * input:		Adapter soft state
475665484d8SDoug Ambrisko  *
476665484d8SDoug Ambrisko  * Allocates DMAable memory for the event log info internal command.
477665484d8SDoug Ambrisko  */
4788e727371SKashyap D Desai int
4798e727371SKashyap D Desai mrsas_alloc_evt_log_info_cmd(struct mrsas_softc *sc)
480665484d8SDoug Ambrisko {
481665484d8SDoug Ambrisko 	int el_info_size;
482665484d8SDoug Ambrisko 
483665484d8SDoug Ambrisko 	/* Allocate get event log info command */
484665484d8SDoug Ambrisko 	el_info_size = sizeof(struct mrsas_evt_log_info);
4858e727371SKashyap D Desai 	if (bus_dma_tag_create(sc->mrsas_parent_tag,
4868e727371SKashyap D Desai 	    1, 0,
4878e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR_32BIT,
4888e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR,
4898e727371SKashyap D Desai 	    NULL, NULL,
4908e727371SKashyap D Desai 	    el_info_size,
4918e727371SKashyap D Desai 	    1,
4928e727371SKashyap D Desai 	    el_info_size,
4938e727371SKashyap D Desai 	    BUS_DMA_ALLOCNOW,
4948e727371SKashyap D Desai 	    NULL, NULL,
495665484d8SDoug Ambrisko 	    &sc->el_info_tag)) {
496665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate event log info tag\n");
497665484d8SDoug Ambrisko 		return (ENOMEM);
498665484d8SDoug Ambrisko 	}
499665484d8SDoug Ambrisko 	if (bus_dmamem_alloc(sc->el_info_tag, (void **)&sc->el_info_mem,
500665484d8SDoug Ambrisko 	    BUS_DMA_NOWAIT, &sc->el_info_dmamap)) {
501665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate event log info cmd mem\n");
502665484d8SDoug Ambrisko 		return (ENOMEM);
503665484d8SDoug Ambrisko 	}
504665484d8SDoug Ambrisko 	if (bus_dmamap_load(sc->el_info_tag, sc->el_info_dmamap,
505665484d8SDoug Ambrisko 	    sc->el_info_mem, el_info_size, mrsas_addr_cb,
506665484d8SDoug Ambrisko 	    &sc->el_info_phys_addr, BUS_DMA_NOWAIT)) {
507665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot load event log info cmd mem\n");
508665484d8SDoug Ambrisko 		return (ENOMEM);
509665484d8SDoug Ambrisko 	}
510665484d8SDoug Ambrisko 	memset(sc->el_info_mem, 0, el_info_size);
511665484d8SDoug Ambrisko 	return (0);
512665484d8SDoug Ambrisko }
513665484d8SDoug Ambrisko 
5148e727371SKashyap D Desai /*
515665484d8SDoug Ambrisko  * mrsas_free_evt_info_cmd:	Free memory for Event log info command
516665484d8SDoug Ambrisko  * input:					Adapter soft state
517665484d8SDoug Ambrisko  *
518665484d8SDoug Ambrisko  * Deallocates memory for the event log info internal command.
519665484d8SDoug Ambrisko  */
5208e727371SKashyap D Desai void
5218e727371SKashyap D Desai mrsas_free_evt_log_info_cmd(struct mrsas_softc *sc)
522665484d8SDoug Ambrisko {
523665484d8SDoug Ambrisko 	if (sc->el_info_phys_addr)
524665484d8SDoug Ambrisko 		bus_dmamap_unload(sc->el_info_tag, sc->el_info_dmamap);
525665484d8SDoug Ambrisko 	if (sc->el_info_mem != NULL)
526665484d8SDoug Ambrisko 		bus_dmamem_free(sc->el_info_tag, sc->el_info_mem, sc->el_info_dmamap);
527665484d8SDoug Ambrisko 	if (sc->el_info_tag != NULL)
528665484d8SDoug Ambrisko 		bus_dma_tag_destroy(sc->el_info_tag);
529665484d8SDoug Ambrisko }
530665484d8SDoug Ambrisko 
5318e727371SKashyap D Desai /*
532665484d8SDoug Ambrisko  *  mrsas_get_seq_num:	Get latest event sequence number
533665484d8SDoug Ambrisko  *  @sc:				Adapter soft state
534665484d8SDoug Ambrisko  *  @eli:				Firmware event log sequence number information.
5358e727371SKashyap D Desai  *
536665484d8SDoug Ambrisko  * Firmware maintains a log of all events in a non-volatile area.
537665484d8SDoug Ambrisko  * Driver get the sequence number using DCMD
538665484d8SDoug Ambrisko  * "MR_DCMD_CTRL_EVENT_GET_INFO" at driver load time.
539665484d8SDoug Ambrisko  */
540665484d8SDoug Ambrisko 
541665484d8SDoug Ambrisko static int
542665484d8SDoug Ambrisko mrsas_get_seq_num(struct mrsas_softc *sc,
543665484d8SDoug Ambrisko     struct mrsas_evt_log_info *eli)
544665484d8SDoug Ambrisko {
545665484d8SDoug Ambrisko 	struct mrsas_mfi_cmd *cmd;
546665484d8SDoug Ambrisko 	struct mrsas_dcmd_frame *dcmd;
547665484d8SDoug Ambrisko 
548665484d8SDoug Ambrisko 	cmd = mrsas_get_mfi_cmd(sc);
549665484d8SDoug Ambrisko 
550665484d8SDoug Ambrisko 	if (!cmd) {
551665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Failed to get a free cmd\n");
552665484d8SDoug Ambrisko 		return -ENOMEM;
553665484d8SDoug Ambrisko 	}
554665484d8SDoug Ambrisko 	dcmd = &cmd->frame->dcmd;
555665484d8SDoug Ambrisko 
556665484d8SDoug Ambrisko 	if (mrsas_alloc_evt_log_info_cmd(sc) != SUCCESS) {
557665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate evt log info cmd\n");
558665484d8SDoug Ambrisko 		mrsas_release_mfi_cmd(cmd);
559665484d8SDoug Ambrisko 		return -ENOMEM;
560665484d8SDoug Ambrisko 	}
561665484d8SDoug Ambrisko 	memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE);
562665484d8SDoug Ambrisko 
563665484d8SDoug Ambrisko 	dcmd->cmd = MFI_CMD_DCMD;
564665484d8SDoug Ambrisko 	dcmd->cmd_status = 0x0;
565665484d8SDoug Ambrisko 	dcmd->sge_count = 1;
566665484d8SDoug Ambrisko 	dcmd->flags = MFI_FRAME_DIR_READ;
567665484d8SDoug Ambrisko 	dcmd->timeout = 0;
568665484d8SDoug Ambrisko 	dcmd->pad_0 = 0;
569665484d8SDoug Ambrisko 	dcmd->data_xfer_len = sizeof(struct mrsas_evt_log_info);
570665484d8SDoug Ambrisko 	dcmd->opcode = MR_DCMD_CTRL_EVENT_GET_INFO;
571665484d8SDoug Ambrisko 	dcmd->sgl.sge32[0].phys_addr = sc->el_info_phys_addr;
572665484d8SDoug Ambrisko 	dcmd->sgl.sge32[0].length = sizeof(struct mrsas_evt_log_info);
573665484d8SDoug Ambrisko 
574665484d8SDoug Ambrisko 	mrsas_issue_blocked_cmd(sc, cmd);
575665484d8SDoug Ambrisko 
576665484d8SDoug Ambrisko 	/*
577665484d8SDoug Ambrisko 	 * Copy the data back into callers buffer
578665484d8SDoug Ambrisko 	 */
579665484d8SDoug Ambrisko 	memcpy(eli, sc->el_info_mem, sizeof(struct mrsas_evt_log_info));
580665484d8SDoug Ambrisko 	mrsas_free_evt_log_info_cmd(sc);
581665484d8SDoug Ambrisko 	mrsas_release_mfi_cmd(cmd);
582665484d8SDoug Ambrisko 
583665484d8SDoug Ambrisko 	return 0;
584665484d8SDoug Ambrisko }
585665484d8SDoug Ambrisko 
586665484d8SDoug Ambrisko 
5878e727371SKashyap D Desai /*
588665484d8SDoug Ambrisko  *  mrsas_register_aen:		Register for asynchronous event notification
589665484d8SDoug Ambrisko  *  @sc:			Adapter soft state
590665484d8SDoug Ambrisko  *  @seq_num:			Starting sequence number
591665484d8SDoug Ambrisko  *  @class_locale:		Class of the event
5928e727371SKashyap D Desai  *
593665484d8SDoug Ambrisko  *  This function subscribes for events beyond the @seq_num
594665484d8SDoug Ambrisko  *  and type @class_locale.
595665484d8SDoug Ambrisko  *
5968e727371SKashyap D Desai  */
597665484d8SDoug Ambrisko static int
598665484d8SDoug Ambrisko mrsas_register_aen(struct mrsas_softc *sc, u_int32_t seq_num,
599665484d8SDoug Ambrisko     u_int32_t class_locale_word)
600665484d8SDoug Ambrisko {
601665484d8SDoug Ambrisko 	int ret_val;
602665484d8SDoug Ambrisko 	struct mrsas_mfi_cmd *cmd;
603665484d8SDoug Ambrisko 	struct mrsas_dcmd_frame *dcmd;
604665484d8SDoug Ambrisko 	union mrsas_evt_class_locale curr_aen;
605665484d8SDoug Ambrisko 	union mrsas_evt_class_locale prev_aen;
606665484d8SDoug Ambrisko 
607665484d8SDoug Ambrisko 	/*
608665484d8SDoug Ambrisko 	 * If there an AEN pending already (aen_cmd), check if the
6098e727371SKashyap D Desai 	 * class_locale of that pending AEN is inclusive of the new AEN
6108e727371SKashyap D Desai 	 * request we currently have. If it is, then we don't have to do
6118e727371SKashyap D Desai 	 * anything. In other words, whichever events the current AEN request
6128e727371SKashyap D Desai 	 * is subscribing to, have already been subscribed to. If the old_cmd
6138e727371SKashyap D Desai 	 * is _not_ inclusive, then we have to abort that command, form a
6148e727371SKashyap D Desai 	 * class_locale that is superset of both old and current and re-issue
6158e727371SKashyap D Desai 	 * to the FW
6168e727371SKashyap D Desai 	 */
617665484d8SDoug Ambrisko 
618665484d8SDoug Ambrisko 	curr_aen.word = class_locale_word;
619665484d8SDoug Ambrisko 
620665484d8SDoug Ambrisko 	if (sc->aen_cmd) {
621665484d8SDoug Ambrisko 
622665484d8SDoug Ambrisko 		prev_aen.word = sc->aen_cmd->frame->dcmd.mbox.w[1];
623665484d8SDoug Ambrisko 
624665484d8SDoug Ambrisko 		/*
625665484d8SDoug Ambrisko 		 * A class whose enum value is smaller is inclusive of all
626665484d8SDoug Ambrisko 		 * higher values. If a PROGRESS (= -1) was previously
627665484d8SDoug Ambrisko 		 * registered, then a new registration requests for higher
628665484d8SDoug Ambrisko 		 * classes need not be sent to FW. They are automatically
6298e727371SKashyap D Desai 		 * included. Locale numbers don't have such hierarchy. They
6308e727371SKashyap D Desai 		 * are bitmap values
631665484d8SDoug Ambrisko 		 */
632665484d8SDoug Ambrisko 		if ((prev_aen.members.class <= curr_aen.members.class) &&
633665484d8SDoug Ambrisko 		    !((prev_aen.members.locale & curr_aen.members.locale) ^
634665484d8SDoug Ambrisko 		    curr_aen.members.locale)) {
635665484d8SDoug Ambrisko 			/*
636665484d8SDoug Ambrisko 			 * Previously issued event registration includes
637665484d8SDoug Ambrisko 			 * current request. Nothing to do.
638665484d8SDoug Ambrisko 			 */
639665484d8SDoug Ambrisko 			return 0;
640665484d8SDoug Ambrisko 		} else {
641665484d8SDoug Ambrisko 			curr_aen.members.locale |= prev_aen.members.locale;
642665484d8SDoug Ambrisko 
643665484d8SDoug Ambrisko 			if (prev_aen.members.class < curr_aen.members.class)
644665484d8SDoug Ambrisko 				curr_aen.members.class = prev_aen.members.class;
645665484d8SDoug Ambrisko 
646665484d8SDoug Ambrisko 			sc->aen_cmd->abort_aen = 1;
647665484d8SDoug Ambrisko 			ret_val = mrsas_issue_blocked_abort_cmd(sc,
648665484d8SDoug Ambrisko 			    sc->aen_cmd);
649665484d8SDoug Ambrisko 
650665484d8SDoug Ambrisko 			if (ret_val) {
651665484d8SDoug Ambrisko 				printf("mrsas: Failed to abort "
652665484d8SDoug Ambrisko 				    "previous AEN command\n");
653665484d8SDoug Ambrisko 				return ret_val;
654665484d8SDoug Ambrisko 			}
655665484d8SDoug Ambrisko 		}
656665484d8SDoug Ambrisko 	}
657665484d8SDoug Ambrisko 	cmd = mrsas_get_mfi_cmd(sc);
658665484d8SDoug Ambrisko 
659665484d8SDoug Ambrisko 	if (!cmd)
660665484d8SDoug Ambrisko 		return -ENOMEM;
661665484d8SDoug Ambrisko 
662665484d8SDoug Ambrisko 	dcmd = &cmd->frame->dcmd;
663665484d8SDoug Ambrisko 
664665484d8SDoug Ambrisko 	memset(sc->evt_detail_mem, 0, sizeof(struct mrsas_evt_detail));
665665484d8SDoug Ambrisko 
666665484d8SDoug Ambrisko 	/*
667665484d8SDoug Ambrisko 	 * Prepare DCMD for aen registration
668665484d8SDoug Ambrisko 	 */
669665484d8SDoug Ambrisko 	memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE);
670665484d8SDoug Ambrisko 
671665484d8SDoug Ambrisko 	dcmd->cmd = MFI_CMD_DCMD;
672665484d8SDoug Ambrisko 	dcmd->cmd_status = 0x0;
673665484d8SDoug Ambrisko 	dcmd->sge_count = 1;
674665484d8SDoug Ambrisko 	dcmd->flags = MFI_FRAME_DIR_READ;
675665484d8SDoug Ambrisko 	dcmd->timeout = 0;
676665484d8SDoug Ambrisko 	dcmd->pad_0 = 0;
677665484d8SDoug Ambrisko 	dcmd->data_xfer_len = sizeof(struct mrsas_evt_detail);
678665484d8SDoug Ambrisko 	dcmd->opcode = MR_DCMD_CTRL_EVENT_WAIT;
679665484d8SDoug Ambrisko 	dcmd->mbox.w[0] = seq_num;
680665484d8SDoug Ambrisko 	sc->last_seq_num = seq_num;
681665484d8SDoug Ambrisko 	dcmd->mbox.w[1] = curr_aen.word;
682665484d8SDoug Ambrisko 	dcmd->sgl.sge32[0].phys_addr = (u_int32_t)sc->evt_detail_phys_addr;
683665484d8SDoug Ambrisko 	dcmd->sgl.sge32[0].length = sizeof(struct mrsas_evt_detail);
684665484d8SDoug Ambrisko 
685665484d8SDoug Ambrisko 	if (sc->aen_cmd != NULL) {
686665484d8SDoug Ambrisko 		mrsas_release_mfi_cmd(cmd);
687665484d8SDoug Ambrisko 		return 0;
688665484d8SDoug Ambrisko 	}
689665484d8SDoug Ambrisko 	/*
690665484d8SDoug Ambrisko 	 * Store reference to the cmd used to register for AEN. When an
691665484d8SDoug Ambrisko 	 * application wants us to register for AEN, we have to abort this
692665484d8SDoug Ambrisko 	 * cmd and re-register with a new EVENT LOCALE supplied by that app
693665484d8SDoug Ambrisko 	 */
694665484d8SDoug Ambrisko 	sc->aen_cmd = cmd;
695665484d8SDoug Ambrisko 
696665484d8SDoug Ambrisko 	/*
6978e727371SKashyap D Desai 	 * Issue the aen registration frame
698665484d8SDoug Ambrisko 	 */
699665484d8SDoug Ambrisko 	if (mrsas_issue_dcmd(sc, cmd)) {
700665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot issue AEN DCMD command.\n");
701665484d8SDoug Ambrisko 		return (1);
702665484d8SDoug Ambrisko 	}
703665484d8SDoug Ambrisko 	return 0;
704665484d8SDoug Ambrisko }
7058e727371SKashyap D Desai 
7068e727371SKashyap D Desai /*
7078e727371SKashyap D Desai  * mrsas_start_aen:	Subscribes to AEN during driver load time
708665484d8SDoug Ambrisko  * @instance:		Adapter soft state
709665484d8SDoug Ambrisko  */
7108e727371SKashyap D Desai static int
7118e727371SKashyap D Desai mrsas_start_aen(struct mrsas_softc *sc)
712665484d8SDoug Ambrisko {
713665484d8SDoug Ambrisko 	struct mrsas_evt_log_info eli;
714665484d8SDoug Ambrisko 	union mrsas_evt_class_locale class_locale;
715665484d8SDoug Ambrisko 
716665484d8SDoug Ambrisko 
717665484d8SDoug Ambrisko 	/* Get the latest sequence number from FW */
718665484d8SDoug Ambrisko 
719665484d8SDoug Ambrisko 	memset(&eli, 0, sizeof(eli));
720665484d8SDoug Ambrisko 
721665484d8SDoug Ambrisko 	if (mrsas_get_seq_num(sc, &eli))
722665484d8SDoug Ambrisko 		return -1;
723665484d8SDoug Ambrisko 
724665484d8SDoug Ambrisko 	/* Register AEN with FW for latest sequence number plus 1 */
725665484d8SDoug Ambrisko 	class_locale.members.reserved = 0;
726665484d8SDoug Ambrisko 	class_locale.members.locale = MR_EVT_LOCALE_ALL;
727665484d8SDoug Ambrisko 	class_locale.members.class = MR_EVT_CLASS_DEBUG;
728665484d8SDoug Ambrisko 
729665484d8SDoug Ambrisko 	return mrsas_register_aen(sc, eli.newest_seq_num + 1,
730665484d8SDoug Ambrisko 	    class_locale.word);
731d18d1b47SKashyap D Desai 
732665484d8SDoug Ambrisko }
733665484d8SDoug Ambrisko 
7348e727371SKashyap D Desai /*
735d18d1b47SKashyap D Desai  * mrsas_setup_msix:	Allocate MSI-x vectors
7368e727371SKashyap D Desai  * @sc:					adapter soft state
737d18d1b47SKashyap D Desai  */
7388e727371SKashyap D Desai static int
7398e727371SKashyap D Desai mrsas_setup_msix(struct mrsas_softc *sc)
740d18d1b47SKashyap D Desai {
741d18d1b47SKashyap D Desai 	int i;
7428e727371SKashyap D Desai 
743d18d1b47SKashyap D Desai 	for (i = 0; i < sc->msix_vectors; i++) {
744d18d1b47SKashyap D Desai 		sc->irq_context[i].sc = sc;
745d18d1b47SKashyap D Desai 		sc->irq_context[i].MSIxIndex = i;
746d18d1b47SKashyap D Desai 		sc->irq_id[i] = i + 1;
747d18d1b47SKashyap D Desai 		sc->mrsas_irq[i] = bus_alloc_resource_any
748d18d1b47SKashyap D Desai 		    (sc->mrsas_dev, SYS_RES_IRQ, &sc->irq_id[i]
749d18d1b47SKashyap D Desai 		    ,RF_ACTIVE);
750d18d1b47SKashyap D Desai 		if (sc->mrsas_irq[i] == NULL) {
751d18d1b47SKashyap D Desai 			device_printf(sc->mrsas_dev, "Can't allocate MSI-x\n");
752d18d1b47SKashyap D Desai 			goto irq_alloc_failed;
753d18d1b47SKashyap D Desai 		}
754d18d1b47SKashyap D Desai 		if (bus_setup_intr(sc->mrsas_dev,
755d18d1b47SKashyap D Desai 		    sc->mrsas_irq[i],
756d18d1b47SKashyap D Desai 		    INTR_MPSAFE | INTR_TYPE_CAM,
757d18d1b47SKashyap D Desai 		    NULL, mrsas_isr, &sc->irq_context[i],
758d18d1b47SKashyap D Desai 		    &sc->intr_handle[i])) {
759d18d1b47SKashyap D Desai 			device_printf(sc->mrsas_dev,
760d18d1b47SKashyap D Desai 			    "Cannot set up MSI-x interrupt handler\n");
761d18d1b47SKashyap D Desai 			goto irq_alloc_failed;
762d18d1b47SKashyap D Desai 		}
763d18d1b47SKashyap D Desai 	}
764d18d1b47SKashyap D Desai 	return SUCCESS;
765d18d1b47SKashyap D Desai 
766d18d1b47SKashyap D Desai irq_alloc_failed:
767d18d1b47SKashyap D Desai 	mrsas_teardown_intr(sc);
768d18d1b47SKashyap D Desai 	return (FAIL);
769d18d1b47SKashyap D Desai }
770d18d1b47SKashyap D Desai 
7718e727371SKashyap D Desai /*
772d18d1b47SKashyap D Desai  * mrsas_allocate_msix:		Setup MSI-x vectors
7738e727371SKashyap D Desai  * @sc:						adapter soft state
774d18d1b47SKashyap D Desai  */
7758e727371SKashyap D Desai static int
7768e727371SKashyap D Desai mrsas_allocate_msix(struct mrsas_softc *sc)
777d18d1b47SKashyap D Desai {
778d18d1b47SKashyap D Desai 	if (pci_alloc_msix(sc->mrsas_dev, &sc->msix_vectors) == 0) {
779d18d1b47SKashyap D Desai 		device_printf(sc->mrsas_dev, "Using MSI-X with %d number"
780d18d1b47SKashyap D Desai 		    " of vectors\n", sc->msix_vectors);
781d18d1b47SKashyap D Desai 	} else {
782d18d1b47SKashyap D Desai 		device_printf(sc->mrsas_dev, "MSI-x setup failed\n");
783d18d1b47SKashyap D Desai 		goto irq_alloc_failed;
784d18d1b47SKashyap D Desai 	}
785d18d1b47SKashyap D Desai 	return SUCCESS;
786d18d1b47SKashyap D Desai 
787d18d1b47SKashyap D Desai irq_alloc_failed:
788d18d1b47SKashyap D Desai 	mrsas_teardown_intr(sc);
789d18d1b47SKashyap D Desai 	return (FAIL);
790d18d1b47SKashyap D Desai }
7918e727371SKashyap D Desai 
7928e727371SKashyap D Desai /*
793665484d8SDoug Ambrisko  * mrsas_attach:	PCI entry point
7948e727371SKashyap D Desai  * input:			pointer to device struct
795665484d8SDoug Ambrisko  *
7968e727371SKashyap D Desai  * Performs setup of PCI and registers, initializes mutexes and linked lists,
7978e727371SKashyap D Desai  * registers interrupts and CAM, and initializes   the adapter/controller to
7988e727371SKashyap D Desai  * its proper state.
799665484d8SDoug Ambrisko  */
8008e727371SKashyap D Desai static int
8018e727371SKashyap D Desai mrsas_attach(device_t dev)
802665484d8SDoug Ambrisko {
803665484d8SDoug Ambrisko 	struct mrsas_softc *sc = device_get_softc(dev);
804665484d8SDoug Ambrisko 	uint32_t cmd, bar, error;
805665484d8SDoug Ambrisko 
806665484d8SDoug Ambrisko 	/* Look up our softc and initialize its fields. */
807665484d8SDoug Ambrisko 	sc->mrsas_dev = dev;
808665484d8SDoug Ambrisko 	sc->device_id = pci_get_device(dev);
809665484d8SDoug Ambrisko 
810665484d8SDoug Ambrisko 	mrsas_get_tunables(sc);
811665484d8SDoug Ambrisko 
812665484d8SDoug Ambrisko 	/*
813665484d8SDoug Ambrisko 	 * Set up PCI and registers
814665484d8SDoug Ambrisko 	 */
815665484d8SDoug Ambrisko 	cmd = pci_read_config(dev, PCIR_COMMAND, 2);
816665484d8SDoug Ambrisko 	if ((cmd & PCIM_CMD_PORTEN) == 0) {
817665484d8SDoug Ambrisko 		return (ENXIO);
818665484d8SDoug Ambrisko 	}
819665484d8SDoug Ambrisko 	/* Force the busmaster enable bit on. */
820665484d8SDoug Ambrisko 	cmd |= PCIM_CMD_BUSMASTEREN;
821665484d8SDoug Ambrisko 	pci_write_config(dev, PCIR_COMMAND, cmd, 2);
822665484d8SDoug Ambrisko 
823665484d8SDoug Ambrisko 	bar = pci_read_config(dev, MRSAS_PCI_BAR1, 4);
824665484d8SDoug Ambrisko 
825665484d8SDoug Ambrisko 	sc->reg_res_id = MRSAS_PCI_BAR1;/* BAR1 offset */
826665484d8SDoug Ambrisko 	if ((sc->reg_res = bus_alloc_resource(dev, SYS_RES_MEMORY,
827665484d8SDoug Ambrisko 	    &(sc->reg_res_id), 0, ~0, 1, RF_ACTIVE))
828665484d8SDoug Ambrisko 	    == NULL) {
829665484d8SDoug Ambrisko 		device_printf(dev, "Cannot allocate PCI registers\n");
830665484d8SDoug Ambrisko 		goto attach_fail;
831665484d8SDoug Ambrisko 	}
832665484d8SDoug Ambrisko 	sc->bus_tag = rman_get_bustag(sc->reg_res);
833665484d8SDoug Ambrisko 	sc->bus_handle = rman_get_bushandle(sc->reg_res);
834665484d8SDoug Ambrisko 
835665484d8SDoug Ambrisko 	/* Intialize mutexes */
836665484d8SDoug Ambrisko 	mtx_init(&sc->sim_lock, "mrsas_sim_lock", NULL, MTX_DEF);
837665484d8SDoug Ambrisko 	mtx_init(&sc->pci_lock, "mrsas_pci_lock", NULL, MTX_DEF);
838665484d8SDoug Ambrisko 	mtx_init(&sc->io_lock, "mrsas_io_lock", NULL, MTX_DEF);
839665484d8SDoug Ambrisko 	mtx_init(&sc->aen_lock, "mrsas_aen_lock", NULL, MTX_DEF);
840665484d8SDoug Ambrisko 	mtx_init(&sc->ioctl_lock, "mrsas_ioctl_lock", NULL, MTX_SPIN);
841665484d8SDoug Ambrisko 	mtx_init(&sc->mpt_cmd_pool_lock, "mrsas_mpt_cmd_pool_lock", NULL, MTX_DEF);
842665484d8SDoug Ambrisko 	mtx_init(&sc->mfi_cmd_pool_lock, "mrsas_mfi_cmd_pool_lock", NULL, MTX_DEF);
843665484d8SDoug Ambrisko 	mtx_init(&sc->raidmap_lock, "mrsas_raidmap_lock", NULL, MTX_DEF);
844665484d8SDoug Ambrisko 
8458e727371SKashyap D Desai 	/*
8468e727371SKashyap D Desai 	 * Intialize a counting Semaphore to take care no. of concurrent
8478e727371SKashyap D Desai 	 * IOCTLs
8488e727371SKashyap D Desai 	 */
849839ee025SKashyap D Desai 	sema_init(&sc->ioctl_count_sema, MRSAS_MAX_MFI_CMDS - 5, IOCTL_SEMA_DESCRIPTION);
850839ee025SKashyap D Desai 
851665484d8SDoug Ambrisko 	/* Intialize linked list */
852665484d8SDoug Ambrisko 	TAILQ_INIT(&sc->mrsas_mpt_cmd_list_head);
853665484d8SDoug Ambrisko 	TAILQ_INIT(&sc->mrsas_mfi_cmd_list_head);
854665484d8SDoug Ambrisko 
855f5fb2237SKashyap D Desai 	mrsas_atomic_set(&sc->fw_outstanding, 0);
856665484d8SDoug Ambrisko 
857665484d8SDoug Ambrisko 	sc->io_cmds_highwater = 0;
858665484d8SDoug Ambrisko 
859665484d8SDoug Ambrisko 	/* Create a /dev entry for this device. */
860665484d8SDoug Ambrisko 	sc->mrsas_cdev = make_dev(&mrsas_cdevsw, device_get_unit(dev), UID_ROOT,
861665484d8SDoug Ambrisko 	    GID_OPERATOR, (S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP), "mrsas%u",
862665484d8SDoug Ambrisko 	    device_get_unit(dev));
863536094dcSKashyap D Desai 	if (device_get_unit(dev) == 0)
864536094dcSKashyap D Desai 		make_dev_alias(sc->mrsas_cdev, "megaraid_sas_ioctl_node");
865665484d8SDoug Ambrisko 	if (sc->mrsas_cdev)
866665484d8SDoug Ambrisko 		sc->mrsas_cdev->si_drv1 = sc;
867665484d8SDoug Ambrisko 
868665484d8SDoug Ambrisko 	sc->adprecovery = MRSAS_HBA_OPERATIONAL;
869665484d8SDoug Ambrisko 	sc->UnevenSpanSupport = 0;
870665484d8SDoug Ambrisko 
871d18d1b47SKashyap D Desai 	sc->msix_enable = 0;
872d18d1b47SKashyap D Desai 
873665484d8SDoug Ambrisko 	/* Initialize Firmware */
874665484d8SDoug Ambrisko 	if (mrsas_init_fw(sc) != SUCCESS) {
875665484d8SDoug Ambrisko 		goto attach_fail_fw;
876665484d8SDoug Ambrisko 	}
877665484d8SDoug Ambrisko 	/* Register SCSI mid-layer */
878665484d8SDoug Ambrisko 	if ((mrsas_cam_attach(sc) != SUCCESS)) {
879665484d8SDoug Ambrisko 		goto attach_fail_cam;
880665484d8SDoug Ambrisko 	}
881665484d8SDoug Ambrisko 	/* Register IRQs */
882665484d8SDoug Ambrisko 	if (mrsas_setup_irq(sc) != SUCCESS) {
883665484d8SDoug Ambrisko 		goto attach_fail_irq;
884665484d8SDoug Ambrisko 	}
885665484d8SDoug Ambrisko 	/* Enable Interrupts */
886665484d8SDoug Ambrisko 	mrsas_enable_intr(sc);
887665484d8SDoug Ambrisko 
888665484d8SDoug Ambrisko 	error = mrsas_kproc_create(mrsas_ocr_thread, sc,
889665484d8SDoug Ambrisko 	    &sc->ocr_thread, 0, 0, "mrsas_ocr%d",
890665484d8SDoug Ambrisko 	    device_get_unit(sc->mrsas_dev));
891665484d8SDoug Ambrisko 	if (error) {
892665484d8SDoug Ambrisko 		printf("Error %d starting rescan thread\n", error);
893665484d8SDoug Ambrisko 		goto attach_fail_irq;
894665484d8SDoug Ambrisko 	}
895665484d8SDoug Ambrisko 	mrsas_setup_sysctl(sc);
896665484d8SDoug Ambrisko 
897665484d8SDoug Ambrisko 	/* Initiate AEN (Asynchronous Event Notification) */
898665484d8SDoug Ambrisko 
899665484d8SDoug Ambrisko 	if (mrsas_start_aen(sc)) {
900665484d8SDoug Ambrisko 		printf("Error: start aen failed\n");
901665484d8SDoug Ambrisko 		goto fail_start_aen;
902665484d8SDoug Ambrisko 	}
903536094dcSKashyap D Desai 	/*
9048e727371SKashyap D Desai 	 * Add this controller to mrsas_mgmt_info structure so that it can be
9058e727371SKashyap D Desai 	 * exported to management applications
906536094dcSKashyap D Desai 	 */
907536094dcSKashyap D Desai 	if (device_get_unit(dev) == 0)
908536094dcSKashyap D Desai 		memset(&mrsas_mgmt_info, 0, sizeof(mrsas_mgmt_info));
909536094dcSKashyap D Desai 
910536094dcSKashyap D Desai 	mrsas_mgmt_info.count++;
911536094dcSKashyap D Desai 	mrsas_mgmt_info.sc_ptr[mrsas_mgmt_info.max_index] = sc;
912536094dcSKashyap D Desai 	mrsas_mgmt_info.max_index++;
913536094dcSKashyap D Desai 
914665484d8SDoug Ambrisko 	return (0);
915665484d8SDoug Ambrisko 
916665484d8SDoug Ambrisko fail_start_aen:
917665484d8SDoug Ambrisko attach_fail_irq:
918665484d8SDoug Ambrisko 	mrsas_teardown_intr(sc);
919665484d8SDoug Ambrisko attach_fail_cam:
920665484d8SDoug Ambrisko 	mrsas_cam_detach(sc);
921665484d8SDoug Ambrisko attach_fail_fw:
922d18d1b47SKashyap D Desai 	/* if MSIX vector is allocated and FW Init FAILED then release MSIX */
923d18d1b47SKashyap D Desai 	if (sc->msix_enable == 1)
924d18d1b47SKashyap D Desai 		pci_release_msi(sc->mrsas_dev);
925665484d8SDoug Ambrisko 	mrsas_free_mem(sc);
926665484d8SDoug Ambrisko 	mtx_destroy(&sc->sim_lock);
927665484d8SDoug Ambrisko 	mtx_destroy(&sc->aen_lock);
928665484d8SDoug Ambrisko 	mtx_destroy(&sc->pci_lock);
929665484d8SDoug Ambrisko 	mtx_destroy(&sc->io_lock);
930665484d8SDoug Ambrisko 	mtx_destroy(&sc->ioctl_lock);
931665484d8SDoug Ambrisko 	mtx_destroy(&sc->mpt_cmd_pool_lock);
932665484d8SDoug Ambrisko 	mtx_destroy(&sc->mfi_cmd_pool_lock);
933665484d8SDoug Ambrisko 	mtx_destroy(&sc->raidmap_lock);
934839ee025SKashyap D Desai 	/* Destroy the counting semaphore created for Ioctl */
935839ee025SKashyap D Desai 	sema_destroy(&sc->ioctl_count_sema);
936665484d8SDoug Ambrisko attach_fail:
937665484d8SDoug Ambrisko 	destroy_dev(sc->mrsas_cdev);
938665484d8SDoug Ambrisko 	if (sc->reg_res) {
939665484d8SDoug Ambrisko 		bus_release_resource(sc->mrsas_dev, SYS_RES_MEMORY,
940665484d8SDoug Ambrisko 		    sc->reg_res_id, sc->reg_res);
941665484d8SDoug Ambrisko 	}
942665484d8SDoug Ambrisko 	return (ENXIO);
943665484d8SDoug Ambrisko }
944665484d8SDoug Ambrisko 
9458e727371SKashyap D Desai /*
946665484d8SDoug Ambrisko  * mrsas_detach:	De-allocates and teardown resources
9478e727371SKashyap D Desai  * input:			pointer to device struct
948665484d8SDoug Ambrisko  *
9498e727371SKashyap D Desai  * This function is the entry point for device disconnect and detach.
9508e727371SKashyap D Desai  * It performs memory de-allocations, shutdown of the controller and various
951665484d8SDoug Ambrisko  * teardown and destroy resource functions.
952665484d8SDoug Ambrisko  */
9538e727371SKashyap D Desai static int
9548e727371SKashyap D Desai mrsas_detach(device_t dev)
955665484d8SDoug Ambrisko {
956665484d8SDoug Ambrisko 	struct mrsas_softc *sc;
957665484d8SDoug Ambrisko 	int i = 0;
958665484d8SDoug Ambrisko 
959665484d8SDoug Ambrisko 	sc = device_get_softc(dev);
960665484d8SDoug Ambrisko 	sc->remove_in_progress = 1;
961536094dcSKashyap D Desai 
962839ee025SKashyap D Desai 	/* Destroy the character device so no other IOCTL will be handled */
963839ee025SKashyap D Desai 	destroy_dev(sc->mrsas_cdev);
964839ee025SKashyap D Desai 
965536094dcSKashyap D Desai 	/*
966536094dcSKashyap D Desai 	 * Take the instance off the instance array. Note that we will not
967536094dcSKashyap D Desai 	 * decrement the max_index. We let this array be sparse array
968536094dcSKashyap D Desai 	 */
969536094dcSKashyap D Desai 	for (i = 0; i < mrsas_mgmt_info.max_index; i++) {
970536094dcSKashyap D Desai 		if (mrsas_mgmt_info.sc_ptr[i] == sc) {
971536094dcSKashyap D Desai 			mrsas_mgmt_info.count--;
972536094dcSKashyap D Desai 			mrsas_mgmt_info.sc_ptr[i] = NULL;
973536094dcSKashyap D Desai 			break;
974536094dcSKashyap D Desai 		}
975536094dcSKashyap D Desai 	}
976536094dcSKashyap D Desai 
977665484d8SDoug Ambrisko 	if (sc->ocr_thread_active)
978665484d8SDoug Ambrisko 		wakeup(&sc->ocr_chan);
979665484d8SDoug Ambrisko 	while (sc->reset_in_progress) {
980665484d8SDoug Ambrisko 		i++;
981665484d8SDoug Ambrisko 		if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) {
982665484d8SDoug Ambrisko 			mrsas_dprint(sc, MRSAS_INFO,
983665484d8SDoug Ambrisko 			    "[%2d]waiting for ocr to be finished\n", i);
984665484d8SDoug Ambrisko 		}
985665484d8SDoug Ambrisko 		pause("mr_shutdown", hz);
986665484d8SDoug Ambrisko 	}
987665484d8SDoug Ambrisko 	i = 0;
988665484d8SDoug Ambrisko 	while (sc->ocr_thread_active) {
989665484d8SDoug Ambrisko 		i++;
990665484d8SDoug Ambrisko 		if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) {
991665484d8SDoug Ambrisko 			mrsas_dprint(sc, MRSAS_INFO,
992665484d8SDoug Ambrisko 			    "[%2d]waiting for "
993665484d8SDoug Ambrisko 			    "mrsas_ocr thread to quit ocr %d\n", i,
994665484d8SDoug Ambrisko 			    sc->ocr_thread_active);
995665484d8SDoug Ambrisko 		}
996665484d8SDoug Ambrisko 		pause("mr_shutdown", hz);
997665484d8SDoug Ambrisko 	}
998665484d8SDoug Ambrisko 	mrsas_flush_cache(sc);
999665484d8SDoug Ambrisko 	mrsas_shutdown_ctlr(sc, MR_DCMD_CTRL_SHUTDOWN);
1000665484d8SDoug Ambrisko 	mrsas_disable_intr(sc);
1001665484d8SDoug Ambrisko 	mrsas_cam_detach(sc);
1002665484d8SDoug Ambrisko 	mrsas_teardown_intr(sc);
1003665484d8SDoug Ambrisko 	mrsas_free_mem(sc);
1004665484d8SDoug Ambrisko 	mtx_destroy(&sc->sim_lock);
1005665484d8SDoug Ambrisko 	mtx_destroy(&sc->aen_lock);
1006665484d8SDoug Ambrisko 	mtx_destroy(&sc->pci_lock);
1007665484d8SDoug Ambrisko 	mtx_destroy(&sc->io_lock);
1008665484d8SDoug Ambrisko 	mtx_destroy(&sc->ioctl_lock);
1009665484d8SDoug Ambrisko 	mtx_destroy(&sc->mpt_cmd_pool_lock);
1010665484d8SDoug Ambrisko 	mtx_destroy(&sc->mfi_cmd_pool_lock);
1011665484d8SDoug Ambrisko 	mtx_destroy(&sc->raidmap_lock);
1012839ee025SKashyap D Desai 
1013839ee025SKashyap D Desai 	/* Wait for all the semaphores to be released */
1014839ee025SKashyap D Desai 	while (sema_value(&sc->ioctl_count_sema) != (MRSAS_MAX_MFI_CMDS - 5))
1015839ee025SKashyap D Desai 		pause("mr_shutdown", hz);
1016839ee025SKashyap D Desai 
1017839ee025SKashyap D Desai 	/* Destroy the counting semaphore created for Ioctl */
1018839ee025SKashyap D Desai 	sema_destroy(&sc->ioctl_count_sema);
1019839ee025SKashyap D Desai 
1020665484d8SDoug Ambrisko 	if (sc->reg_res) {
1021665484d8SDoug Ambrisko 		bus_release_resource(sc->mrsas_dev,
1022665484d8SDoug Ambrisko 		    SYS_RES_MEMORY, sc->reg_res_id, sc->reg_res);
1023665484d8SDoug Ambrisko 	}
1024665484d8SDoug Ambrisko 	if (sc->sysctl_tree != NULL)
1025665484d8SDoug Ambrisko 		sysctl_ctx_free(&sc->sysctl_ctx);
1026839ee025SKashyap D Desai 
1027665484d8SDoug Ambrisko 	return (0);
1028665484d8SDoug Ambrisko }
1029665484d8SDoug Ambrisko 
10308e727371SKashyap D Desai /*
1031665484d8SDoug Ambrisko  * mrsas_free_mem:		Frees allocated memory
1032665484d8SDoug Ambrisko  * input:				Adapter instance soft state
1033665484d8SDoug Ambrisko  *
1034665484d8SDoug Ambrisko  * This function is called from mrsas_detach() to free previously allocated
1035665484d8SDoug Ambrisko  * memory.
1036665484d8SDoug Ambrisko  */
10378e727371SKashyap D Desai void
10388e727371SKashyap D Desai mrsas_free_mem(struct mrsas_softc *sc)
1039665484d8SDoug Ambrisko {
1040665484d8SDoug Ambrisko 	int i;
1041665484d8SDoug Ambrisko 	u_int32_t max_cmd;
1042665484d8SDoug Ambrisko 	struct mrsas_mfi_cmd *mfi_cmd;
1043665484d8SDoug Ambrisko 	struct mrsas_mpt_cmd *mpt_cmd;
1044665484d8SDoug Ambrisko 
1045665484d8SDoug Ambrisko 	/*
1046665484d8SDoug Ambrisko 	 * Free RAID map memory
1047665484d8SDoug Ambrisko 	 */
10488e727371SKashyap D Desai 	for (i = 0; i < 2; i++) {
1049665484d8SDoug Ambrisko 		if (sc->raidmap_phys_addr[i])
1050665484d8SDoug Ambrisko 			bus_dmamap_unload(sc->raidmap_tag[i], sc->raidmap_dmamap[i]);
1051665484d8SDoug Ambrisko 		if (sc->raidmap_mem[i] != NULL)
1052665484d8SDoug Ambrisko 			bus_dmamem_free(sc->raidmap_tag[i], sc->raidmap_mem[i], sc->raidmap_dmamap[i]);
1053665484d8SDoug Ambrisko 		if (sc->raidmap_tag[i] != NULL)
1054665484d8SDoug Ambrisko 			bus_dma_tag_destroy(sc->raidmap_tag[i]);
10554799d485SKashyap D Desai 
10564799d485SKashyap D Desai 		if (sc->ld_drv_map[i] != NULL)
10574799d485SKashyap D Desai 			free(sc->ld_drv_map[i], M_MRSAS);
1058665484d8SDoug Ambrisko 	}
1059665484d8SDoug Ambrisko 
1060665484d8SDoug Ambrisko 	/*
1061665484d8SDoug Ambrisko 	 * Free version buffer memroy
1062665484d8SDoug Ambrisko 	 */
1063665484d8SDoug Ambrisko 	if (sc->verbuf_phys_addr)
1064665484d8SDoug Ambrisko 		bus_dmamap_unload(sc->verbuf_tag, sc->verbuf_dmamap);
1065665484d8SDoug Ambrisko 	if (sc->verbuf_mem != NULL)
1066665484d8SDoug Ambrisko 		bus_dmamem_free(sc->verbuf_tag, sc->verbuf_mem, sc->verbuf_dmamap);
1067665484d8SDoug Ambrisko 	if (sc->verbuf_tag != NULL)
1068665484d8SDoug Ambrisko 		bus_dma_tag_destroy(sc->verbuf_tag);
1069665484d8SDoug Ambrisko 
1070665484d8SDoug Ambrisko 
1071665484d8SDoug Ambrisko 	/*
1072665484d8SDoug Ambrisko 	 * Free sense buffer memory
1073665484d8SDoug Ambrisko 	 */
1074665484d8SDoug Ambrisko 	if (sc->sense_phys_addr)
1075665484d8SDoug Ambrisko 		bus_dmamap_unload(sc->sense_tag, sc->sense_dmamap);
1076665484d8SDoug Ambrisko 	if (sc->sense_mem != NULL)
1077665484d8SDoug Ambrisko 		bus_dmamem_free(sc->sense_tag, sc->sense_mem, sc->sense_dmamap);
1078665484d8SDoug Ambrisko 	if (sc->sense_tag != NULL)
1079665484d8SDoug Ambrisko 		bus_dma_tag_destroy(sc->sense_tag);
1080665484d8SDoug Ambrisko 
1081665484d8SDoug Ambrisko 	/*
1082665484d8SDoug Ambrisko 	 * Free chain frame memory
1083665484d8SDoug Ambrisko 	 */
1084665484d8SDoug Ambrisko 	if (sc->chain_frame_phys_addr)
1085665484d8SDoug Ambrisko 		bus_dmamap_unload(sc->chain_frame_tag, sc->chain_frame_dmamap);
1086665484d8SDoug Ambrisko 	if (sc->chain_frame_mem != NULL)
1087665484d8SDoug Ambrisko 		bus_dmamem_free(sc->chain_frame_tag, sc->chain_frame_mem, sc->chain_frame_dmamap);
1088665484d8SDoug Ambrisko 	if (sc->chain_frame_tag != NULL)
1089665484d8SDoug Ambrisko 		bus_dma_tag_destroy(sc->chain_frame_tag);
1090665484d8SDoug Ambrisko 
1091665484d8SDoug Ambrisko 	/*
1092665484d8SDoug Ambrisko 	 * Free IO Request memory
1093665484d8SDoug Ambrisko 	 */
1094665484d8SDoug Ambrisko 	if (sc->io_request_phys_addr)
1095665484d8SDoug Ambrisko 		bus_dmamap_unload(sc->io_request_tag, sc->io_request_dmamap);
1096665484d8SDoug Ambrisko 	if (sc->io_request_mem != NULL)
1097665484d8SDoug Ambrisko 		bus_dmamem_free(sc->io_request_tag, sc->io_request_mem, sc->io_request_dmamap);
1098665484d8SDoug Ambrisko 	if (sc->io_request_tag != NULL)
1099665484d8SDoug Ambrisko 		bus_dma_tag_destroy(sc->io_request_tag);
1100665484d8SDoug Ambrisko 
1101665484d8SDoug Ambrisko 	/*
1102665484d8SDoug Ambrisko 	 * Free Reply Descriptor memory
1103665484d8SDoug Ambrisko 	 */
1104665484d8SDoug Ambrisko 	if (sc->reply_desc_phys_addr)
1105665484d8SDoug Ambrisko 		bus_dmamap_unload(sc->reply_desc_tag, sc->reply_desc_dmamap);
1106665484d8SDoug Ambrisko 	if (sc->reply_desc_mem != NULL)
1107665484d8SDoug Ambrisko 		bus_dmamem_free(sc->reply_desc_tag, sc->reply_desc_mem, sc->reply_desc_dmamap);
1108665484d8SDoug Ambrisko 	if (sc->reply_desc_tag != NULL)
1109665484d8SDoug Ambrisko 		bus_dma_tag_destroy(sc->reply_desc_tag);
1110665484d8SDoug Ambrisko 
1111665484d8SDoug Ambrisko 	/*
1112665484d8SDoug Ambrisko 	 * Free event detail memory
1113665484d8SDoug Ambrisko 	 */
1114665484d8SDoug Ambrisko 	if (sc->evt_detail_phys_addr)
1115665484d8SDoug Ambrisko 		bus_dmamap_unload(sc->evt_detail_tag, sc->evt_detail_dmamap);
1116665484d8SDoug Ambrisko 	if (sc->evt_detail_mem != NULL)
1117665484d8SDoug Ambrisko 		bus_dmamem_free(sc->evt_detail_tag, sc->evt_detail_mem, sc->evt_detail_dmamap);
1118665484d8SDoug Ambrisko 	if (sc->evt_detail_tag != NULL)
1119665484d8SDoug Ambrisko 		bus_dma_tag_destroy(sc->evt_detail_tag);
1120665484d8SDoug Ambrisko 
1121665484d8SDoug Ambrisko 	/*
1122665484d8SDoug Ambrisko 	 * Free MFI frames
1123665484d8SDoug Ambrisko 	 */
1124665484d8SDoug Ambrisko 	if (sc->mfi_cmd_list) {
1125665484d8SDoug Ambrisko 		for (i = 0; i < MRSAS_MAX_MFI_CMDS; i++) {
1126665484d8SDoug Ambrisko 			mfi_cmd = sc->mfi_cmd_list[i];
1127665484d8SDoug Ambrisko 			mrsas_free_frame(sc, mfi_cmd);
1128665484d8SDoug Ambrisko 		}
1129665484d8SDoug Ambrisko 	}
1130665484d8SDoug Ambrisko 	if (sc->mficmd_frame_tag != NULL)
1131665484d8SDoug Ambrisko 		bus_dma_tag_destroy(sc->mficmd_frame_tag);
1132665484d8SDoug Ambrisko 
1133665484d8SDoug Ambrisko 	/*
1134665484d8SDoug Ambrisko 	 * Free MPT internal command list
1135665484d8SDoug Ambrisko 	 */
1136665484d8SDoug Ambrisko 	max_cmd = sc->max_fw_cmds;
1137665484d8SDoug Ambrisko 	if (sc->mpt_cmd_list) {
1138665484d8SDoug Ambrisko 		for (i = 0; i < max_cmd; i++) {
1139665484d8SDoug Ambrisko 			mpt_cmd = sc->mpt_cmd_list[i];
1140665484d8SDoug Ambrisko 			bus_dmamap_destroy(sc->data_tag, mpt_cmd->data_dmamap);
1141665484d8SDoug Ambrisko 			free(sc->mpt_cmd_list[i], M_MRSAS);
1142665484d8SDoug Ambrisko 		}
1143665484d8SDoug Ambrisko 		free(sc->mpt_cmd_list, M_MRSAS);
1144665484d8SDoug Ambrisko 		sc->mpt_cmd_list = NULL;
1145665484d8SDoug Ambrisko 	}
1146665484d8SDoug Ambrisko 	/*
1147665484d8SDoug Ambrisko 	 * Free MFI internal command list
1148665484d8SDoug Ambrisko 	 */
1149665484d8SDoug Ambrisko 
1150665484d8SDoug Ambrisko 	if (sc->mfi_cmd_list) {
1151665484d8SDoug Ambrisko 		for (i = 0; i < MRSAS_MAX_MFI_CMDS; i++) {
1152665484d8SDoug Ambrisko 			free(sc->mfi_cmd_list[i], M_MRSAS);
1153665484d8SDoug Ambrisko 		}
1154665484d8SDoug Ambrisko 		free(sc->mfi_cmd_list, M_MRSAS);
1155665484d8SDoug Ambrisko 		sc->mfi_cmd_list = NULL;
1156665484d8SDoug Ambrisko 	}
1157665484d8SDoug Ambrisko 	/*
1158665484d8SDoug Ambrisko 	 * Free request descriptor memory
1159665484d8SDoug Ambrisko 	 */
1160665484d8SDoug Ambrisko 	free(sc->req_desc, M_MRSAS);
1161665484d8SDoug Ambrisko 	sc->req_desc = NULL;
1162665484d8SDoug Ambrisko 
1163665484d8SDoug Ambrisko 	/*
1164665484d8SDoug Ambrisko 	 * Destroy parent tag
1165665484d8SDoug Ambrisko 	 */
1166665484d8SDoug Ambrisko 	if (sc->mrsas_parent_tag != NULL)
1167665484d8SDoug Ambrisko 		bus_dma_tag_destroy(sc->mrsas_parent_tag);
1168665484d8SDoug Ambrisko }
1169665484d8SDoug Ambrisko 
11708e727371SKashyap D Desai /*
1171665484d8SDoug Ambrisko  * mrsas_teardown_intr:	Teardown interrupt
1172665484d8SDoug Ambrisko  * input:				Adapter instance soft state
1173665484d8SDoug Ambrisko  *
11748e727371SKashyap D Desai  * This function is called from mrsas_detach() to teardown and release bus
11758e727371SKashyap D Desai  * interrupt resourse.
1176665484d8SDoug Ambrisko  */
11778e727371SKashyap D Desai void
11788e727371SKashyap D Desai mrsas_teardown_intr(struct mrsas_softc *sc)
1179665484d8SDoug Ambrisko {
1180d18d1b47SKashyap D Desai 	int i;
11818e727371SKashyap D Desai 
1182d18d1b47SKashyap D Desai 	if (!sc->msix_enable) {
1183d18d1b47SKashyap D Desai 		if (sc->intr_handle[0])
1184d18d1b47SKashyap D Desai 			bus_teardown_intr(sc->mrsas_dev, sc->mrsas_irq[0], sc->intr_handle[0]);
1185d18d1b47SKashyap D Desai 		if (sc->mrsas_irq[0] != NULL)
11868e727371SKashyap D Desai 			bus_release_resource(sc->mrsas_dev, SYS_RES_IRQ,
11878e727371SKashyap D Desai 			    sc->irq_id[0], sc->mrsas_irq[0]);
1188d18d1b47SKashyap D Desai 		sc->intr_handle[0] = NULL;
1189d18d1b47SKashyap D Desai 	} else {
1190d18d1b47SKashyap D Desai 		for (i = 0; i < sc->msix_vectors; i++) {
1191d18d1b47SKashyap D Desai 			if (sc->intr_handle[i])
1192d18d1b47SKashyap D Desai 				bus_teardown_intr(sc->mrsas_dev, sc->mrsas_irq[i],
1193d18d1b47SKashyap D Desai 				    sc->intr_handle[i]);
1194d18d1b47SKashyap D Desai 
1195d18d1b47SKashyap D Desai 			if (sc->mrsas_irq[i] != NULL)
1196d18d1b47SKashyap D Desai 				bus_release_resource(sc->mrsas_dev, SYS_RES_IRQ,
1197d18d1b47SKashyap D Desai 				    sc->irq_id[i], sc->mrsas_irq[i]);
1198d18d1b47SKashyap D Desai 
1199d18d1b47SKashyap D Desai 			sc->intr_handle[i] = NULL;
1200d18d1b47SKashyap D Desai 		}
1201d18d1b47SKashyap D Desai 		pci_release_msi(sc->mrsas_dev);
1202d18d1b47SKashyap D Desai 	}
1203d18d1b47SKashyap D Desai 
1204665484d8SDoug Ambrisko }
1205665484d8SDoug Ambrisko 
12068e727371SKashyap D Desai /*
1207665484d8SDoug Ambrisko  * mrsas_suspend:	Suspend entry point
1208665484d8SDoug Ambrisko  * input:			Device struct pointer
1209665484d8SDoug Ambrisko  *
1210665484d8SDoug Ambrisko  * This function is the entry point for system suspend from the OS.
1211665484d8SDoug Ambrisko  */
12128e727371SKashyap D Desai static int
12138e727371SKashyap D Desai mrsas_suspend(device_t dev)
1214665484d8SDoug Ambrisko {
1215665484d8SDoug Ambrisko 	struct mrsas_softc *sc;
1216665484d8SDoug Ambrisko 
1217665484d8SDoug Ambrisko 	sc = device_get_softc(dev);
1218665484d8SDoug Ambrisko 	return (0);
1219665484d8SDoug Ambrisko }
1220665484d8SDoug Ambrisko 
12218e727371SKashyap D Desai /*
1222665484d8SDoug Ambrisko  * mrsas_resume:	Resume entry point
1223665484d8SDoug Ambrisko  * input:			Device struct pointer
1224665484d8SDoug Ambrisko  *
1225665484d8SDoug Ambrisko  * This function is the entry point for system resume from the OS.
1226665484d8SDoug Ambrisko  */
12278e727371SKashyap D Desai static int
12288e727371SKashyap D Desai mrsas_resume(device_t dev)
1229665484d8SDoug Ambrisko {
1230665484d8SDoug Ambrisko 	struct mrsas_softc *sc;
1231665484d8SDoug Ambrisko 
1232665484d8SDoug Ambrisko 	sc = device_get_softc(dev);
1233665484d8SDoug Ambrisko 	return (0);
1234665484d8SDoug Ambrisko }
1235665484d8SDoug Ambrisko 
12365844115eSKashyap D Desai /**
12375844115eSKashyap D Desai  * mrsas_get_softc_instance:    Find softc instance based on cmd type
12385844115eSKashyap D Desai  *
12395844115eSKashyap D Desai  * This function will return softc instance based on cmd type.
12405844115eSKashyap D Desai  * In some case, application fire ioctl on required management instance and
12415844115eSKashyap D Desai  * do not provide host_no. Use cdev->si_drv1 to get softc instance for those
12425844115eSKashyap D Desai  * case, else get the softc instance from host_no provided by application in
12435844115eSKashyap D Desai  * user data.
12445844115eSKashyap D Desai  */
12455844115eSKashyap D Desai 
12465844115eSKashyap D Desai static struct mrsas_softc *
12475844115eSKashyap D Desai mrsas_get_softc_instance(struct cdev *dev, u_long cmd, caddr_t arg)
12485844115eSKashyap D Desai {
12495844115eSKashyap D Desai 	struct mrsas_softc *sc = NULL;
12505844115eSKashyap D Desai 	struct mrsas_iocpacket *user_ioc = (struct mrsas_iocpacket *)arg;
12515844115eSKashyap D Desai 	if (cmd == MRSAS_IOC_GET_PCI_INFO){
12525844115eSKashyap D Desai 	sc = dev->si_drv1;
12535844115eSKashyap D Desai 	} else {
12545844115eSKashyap D Desai 	/* get the Host number & the softc from data sent by the Application */
12555844115eSKashyap D Desai 	sc = mrsas_mgmt_info.sc_ptr[user_ioc->host_no];
12565844115eSKashyap D Desai 		if ((user_ioc->host_no >= mrsas_mgmt_info.max_index) || (sc == NULL)) {
12575844115eSKashyap D Desai 			if (sc == NULL)
12585844115eSKashyap D Desai 				mrsas_dprint(sc, MRSAS_FAULT,
12595844115eSKashyap D Desai 					"There is no Controller number %d .\n", user_ioc->host_no);
12605844115eSKashyap D Desai 			else
12615844115eSKashyap D Desai 				mrsas_dprint(sc, MRSAS_FAULT,
12625844115eSKashyap D Desai 					"Invalid Controller number %d .\n", user_ioc->host_no);
12635844115eSKashyap D Desai 		}
12645844115eSKashyap D Desai 	}
12655844115eSKashyap D Desai 
12665844115eSKashyap D Desai 	return sc;
12675844115eSKashyap D Desai }
12685844115eSKashyap D Desai 
12698e727371SKashyap D Desai /*
1270665484d8SDoug Ambrisko  * mrsas_ioctl:	IOCtl commands entry point.
1271665484d8SDoug Ambrisko  *
1272665484d8SDoug Ambrisko  * This function is the entry point for IOCtls from the OS.  It calls the
1273665484d8SDoug Ambrisko  * appropriate function for processing depending on the command received.
1274665484d8SDoug Ambrisko  */
1275665484d8SDoug Ambrisko static int
12767fc5f329SJohn Baldwin mrsas_ioctl(struct cdev *dev, u_long cmd, caddr_t arg, int flag,
12777fc5f329SJohn Baldwin     struct thread *td)
1278665484d8SDoug Ambrisko {
1279665484d8SDoug Ambrisko 	struct mrsas_softc *sc;
1280665484d8SDoug Ambrisko 	int ret = 0, i = 0;
12815844115eSKashyap D Desai 	MRSAS_DRV_PCI_INFORMATION *pciDrvInfo;
1282665484d8SDoug Ambrisko 
12835844115eSKashyap D Desai 	sc = mrsas_get_softc_instance(dev, cmd, arg);
12845844115eSKashyap D Desai 	if (!sc)
1285536094dcSKashyap D Desai 		return ENOENT;
12865844115eSKashyap D Desai 
1287665484d8SDoug Ambrisko 	if (sc->remove_in_progress) {
1288665484d8SDoug Ambrisko 		mrsas_dprint(sc, MRSAS_INFO,
1289665484d8SDoug Ambrisko 		    "Driver remove or shutdown called.\n");
1290665484d8SDoug Ambrisko 		return ENOENT;
1291665484d8SDoug Ambrisko 	}
1292665484d8SDoug Ambrisko 	mtx_lock_spin(&sc->ioctl_lock);
1293665484d8SDoug Ambrisko 	if (!sc->reset_in_progress) {
1294665484d8SDoug Ambrisko 		mtx_unlock_spin(&sc->ioctl_lock);
1295665484d8SDoug Ambrisko 		goto do_ioctl;
1296665484d8SDoug Ambrisko 	}
1297665484d8SDoug Ambrisko 	mtx_unlock_spin(&sc->ioctl_lock);
1298665484d8SDoug Ambrisko 	while (sc->reset_in_progress) {
1299665484d8SDoug Ambrisko 		i++;
1300665484d8SDoug Ambrisko 		if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) {
1301665484d8SDoug Ambrisko 			mrsas_dprint(sc, MRSAS_INFO,
1302665484d8SDoug Ambrisko 			    "[%2d]waiting for "
1303665484d8SDoug Ambrisko 			    "OCR to be finished %d\n", i,
1304665484d8SDoug Ambrisko 			    sc->ocr_thread_active);
1305665484d8SDoug Ambrisko 		}
1306665484d8SDoug Ambrisko 		pause("mr_ioctl", hz);
1307665484d8SDoug Ambrisko 	}
1308665484d8SDoug Ambrisko 
1309665484d8SDoug Ambrisko do_ioctl:
1310665484d8SDoug Ambrisko 	switch (cmd) {
1311536094dcSKashyap D Desai 	case MRSAS_IOC_FIRMWARE_PASS_THROUGH64:
1312536094dcSKashyap D Desai #ifdef COMPAT_FREEBSD32
1313536094dcSKashyap D Desai 	case MRSAS_IOC_FIRMWARE_PASS_THROUGH32:
1314536094dcSKashyap D Desai #endif
13158e727371SKashyap D Desai 		/*
13168e727371SKashyap D Desai 		 * Decrement the Ioctl counting Semaphore before getting an
13178e727371SKashyap D Desai 		 * mfi command
13188e727371SKashyap D Desai 		 */
1319839ee025SKashyap D Desai 		sema_wait(&sc->ioctl_count_sema);
1320839ee025SKashyap D Desai 
1321536094dcSKashyap D Desai 		ret = mrsas_passthru(sc, (void *)arg, cmd);
1322839ee025SKashyap D Desai 
1323839ee025SKashyap D Desai 		/* Increment the Ioctl counting semaphore value */
1324839ee025SKashyap D Desai 		sema_post(&sc->ioctl_count_sema);
1325839ee025SKashyap D Desai 
1326665484d8SDoug Ambrisko 		break;
1327665484d8SDoug Ambrisko 	case MRSAS_IOC_SCAN_BUS:
1328665484d8SDoug Ambrisko 		ret = mrsas_bus_scan(sc);
1329665484d8SDoug Ambrisko 		break;
13305844115eSKashyap D Desai 
13315844115eSKashyap D Desai 	case MRSAS_IOC_GET_PCI_INFO:
13325844115eSKashyap D Desai 		pciDrvInfo = (MRSAS_DRV_PCI_INFORMATION *)arg;
13335844115eSKashyap D Desai 		memset (pciDrvInfo, 0, sizeof(MRSAS_DRV_PCI_INFORMATION));
13345844115eSKashyap D Desai 		pciDrvInfo->busNumber = pci_get_bus(sc->mrsas_dev);
13355844115eSKashyap D Desai 		pciDrvInfo->deviceNumber = pci_get_slot(sc->mrsas_dev);
13365844115eSKashyap D Desai 		pciDrvInfo->functionNumber = pci_get_function(sc->mrsas_dev);
13375844115eSKashyap D Desai 		pciDrvInfo->domainID = pci_get_domain(sc->mrsas_dev);
13385844115eSKashyap D Desai 		mrsas_dprint (sc, MRSAS_INFO, "pci bus no: %d,"
13395844115eSKashyap D Desai 			"pci device no: %d, pci function no: %d,"
13405844115eSKashyap D Desai 			"pci domain ID: %d\n",
13415844115eSKashyap D Desai 			pciDrvInfo->busNumber, pciDrvInfo->deviceNumber,
13425844115eSKashyap D Desai 			pciDrvInfo->functionNumber, pciDrvInfo->domainID);
13435844115eSKashyap D Desai 		ret = 0;
13445844115eSKashyap D Desai 		break;
13455844115eSKashyap D Desai 
1346536094dcSKashyap D Desai 	default:
1347536094dcSKashyap D Desai 		mrsas_dprint(sc, MRSAS_TRACE, "IOCTL command 0x%lx is not handled\n", cmd);
1348839ee025SKashyap D Desai 		ret = ENOENT;
1349665484d8SDoug Ambrisko 	}
1350665484d8SDoug Ambrisko 
1351665484d8SDoug Ambrisko 	return (ret);
1352665484d8SDoug Ambrisko }
1353665484d8SDoug Ambrisko 
13548e727371SKashyap D Desai /*
1355da011113SKashyap D Desai  * mrsas_poll:	poll entry point for mrsas driver fd
1356da011113SKashyap D Desai  *
13578e727371SKashyap D Desai  * This function is the entry point for poll from the OS.  It waits for some AEN
13588e727371SKashyap D Desai  * events to be triggered from the controller and notifies back.
1359da011113SKashyap D Desai  */
1360da011113SKashyap D Desai static int
1361da011113SKashyap D Desai mrsas_poll(struct cdev *dev, int poll_events, struct thread *td)
1362da011113SKashyap D Desai {
1363da011113SKashyap D Desai 	struct mrsas_softc *sc;
1364da011113SKashyap D Desai 	int revents = 0;
1365da011113SKashyap D Desai 
1366da011113SKashyap D Desai 	sc = dev->si_drv1;
1367da011113SKashyap D Desai 
1368da011113SKashyap D Desai 	if (poll_events & (POLLIN | POLLRDNORM)) {
1369da011113SKashyap D Desai 		if (sc->mrsas_aen_triggered) {
1370da011113SKashyap D Desai 			revents |= poll_events & (POLLIN | POLLRDNORM);
1371da011113SKashyap D Desai 		}
1372da011113SKashyap D Desai 	}
1373da011113SKashyap D Desai 	if (revents == 0) {
1374da011113SKashyap D Desai 		if (poll_events & (POLLIN | POLLRDNORM)) {
1375da011113SKashyap D Desai 			sc->mrsas_poll_waiting = 1;
1376da011113SKashyap D Desai 			selrecord(td, &sc->mrsas_select);
1377da011113SKashyap D Desai 		}
1378da011113SKashyap D Desai 	}
1379da011113SKashyap D Desai 	return revents;
1380da011113SKashyap D Desai }
1381da011113SKashyap D Desai 
13828e727371SKashyap D Desai /*
13838e727371SKashyap D Desai  * mrsas_setup_irq:	Set up interrupt
1384665484d8SDoug Ambrisko  * input:			Adapter instance soft state
1385665484d8SDoug Ambrisko  *
1386665484d8SDoug Ambrisko  * This function sets up interrupts as a bus resource, with flags indicating
1387665484d8SDoug Ambrisko  * resource permitting contemporaneous sharing and for resource to activate
1388665484d8SDoug Ambrisko  * atomically.
1389665484d8SDoug Ambrisko  */
13908e727371SKashyap D Desai static int
13918e727371SKashyap D Desai mrsas_setup_irq(struct mrsas_softc *sc)
1392665484d8SDoug Ambrisko {
1393d18d1b47SKashyap D Desai 	if (sc->msix_enable && (mrsas_setup_msix(sc) == SUCCESS))
1394d18d1b47SKashyap D Desai 		device_printf(sc->mrsas_dev, "MSI-x interrupts setup success\n");
1395665484d8SDoug Ambrisko 
1396d18d1b47SKashyap D Desai 	else {
1397d18d1b47SKashyap D Desai 		device_printf(sc->mrsas_dev, "Fall back to legacy interrupt\n");
1398d18d1b47SKashyap D Desai 		sc->irq_context[0].sc = sc;
1399d18d1b47SKashyap D Desai 		sc->irq_context[0].MSIxIndex = 0;
1400d18d1b47SKashyap D Desai 		sc->irq_id[0] = 0;
1401d18d1b47SKashyap D Desai 		sc->mrsas_irq[0] = bus_alloc_resource_any(sc->mrsas_dev,
1402d18d1b47SKashyap D Desai 		    SYS_RES_IRQ, &sc->irq_id[0], RF_SHAREABLE | RF_ACTIVE);
1403d18d1b47SKashyap D Desai 		if (sc->mrsas_irq[0] == NULL) {
1404d18d1b47SKashyap D Desai 			device_printf(sc->mrsas_dev, "Cannot allocate legcay"
1405d18d1b47SKashyap D Desai 			    "interrupt\n");
1406d18d1b47SKashyap D Desai 			return (FAIL);
1407d18d1b47SKashyap D Desai 		}
1408d18d1b47SKashyap D Desai 		if (bus_setup_intr(sc->mrsas_dev, sc->mrsas_irq[0],
1409d18d1b47SKashyap D Desai 		    INTR_MPSAFE | INTR_TYPE_CAM, NULL, mrsas_isr,
1410d18d1b47SKashyap D Desai 		    &sc->irq_context[0], &sc->intr_handle[0])) {
1411d18d1b47SKashyap D Desai 			device_printf(sc->mrsas_dev, "Cannot set up legacy"
1412d18d1b47SKashyap D Desai 			    "interrupt\n");
1413d18d1b47SKashyap D Desai 			return (FAIL);
1414d18d1b47SKashyap D Desai 		}
1415d18d1b47SKashyap D Desai 	}
1416665484d8SDoug Ambrisko 	return (0);
1417665484d8SDoug Ambrisko }
1418665484d8SDoug Ambrisko 
1419665484d8SDoug Ambrisko /*
1420665484d8SDoug Ambrisko  * mrsas_isr:	ISR entry point
1421665484d8SDoug Ambrisko  * input:		argument pointer
1422665484d8SDoug Ambrisko  *
14238e727371SKashyap D Desai  * This function is the interrupt service routine entry point.  There are two
14248e727371SKashyap D Desai  * types of interrupts, state change interrupt and response interrupt.  If an
14258e727371SKashyap D Desai  * interrupt is not ours, we just return.
1426665484d8SDoug Ambrisko  */
14278e727371SKashyap D Desai void
14288e727371SKashyap D Desai mrsas_isr(void *arg)
1429665484d8SDoug Ambrisko {
1430d18d1b47SKashyap D Desai 	struct mrsas_irq_context *irq_context = (struct mrsas_irq_context *)arg;
1431d18d1b47SKashyap D Desai 	struct mrsas_softc *sc = irq_context->sc;
1432d18d1b47SKashyap D Desai 	int status = 0;
1433665484d8SDoug Ambrisko 
1434d18d1b47SKashyap D Desai 	if (!sc->msix_vectors) {
1435665484d8SDoug Ambrisko 		status = mrsas_clear_intr(sc);
1436665484d8SDoug Ambrisko 		if (!status)
1437665484d8SDoug Ambrisko 			return;
1438d18d1b47SKashyap D Desai 	}
1439665484d8SDoug Ambrisko 	/* If we are resetting, bail */
1440f5fb2237SKashyap D Desai 	if (mrsas_test_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags)) {
1441665484d8SDoug Ambrisko 		printf(" Entered into ISR when OCR is going active. \n");
1442665484d8SDoug Ambrisko 		mrsas_clear_intr(sc);
1443665484d8SDoug Ambrisko 		return;
1444665484d8SDoug Ambrisko 	}
1445665484d8SDoug Ambrisko 	/* Process for reply request and clear response interrupt */
1446d18d1b47SKashyap D Desai 	if (mrsas_complete_cmd(sc, irq_context->MSIxIndex) != SUCCESS)
1447665484d8SDoug Ambrisko 		mrsas_clear_intr(sc);
1448665484d8SDoug Ambrisko 
1449665484d8SDoug Ambrisko 	return;
1450665484d8SDoug Ambrisko }
1451665484d8SDoug Ambrisko 
1452665484d8SDoug Ambrisko /*
1453665484d8SDoug Ambrisko  * mrsas_complete_cmd:	Process reply request
1454665484d8SDoug Ambrisko  * input:				Adapter instance soft state
1455665484d8SDoug Ambrisko  *
14568e727371SKashyap D Desai  * This function is called from mrsas_isr() to process reply request and clear
14578e727371SKashyap D Desai  * response interrupt. Processing of the reply request entails walking
14588e727371SKashyap D Desai  * through the reply descriptor array for the command request  pended from
14598e727371SKashyap D Desai  * Firmware.  We look at the Function field to determine the command type and
14608e727371SKashyap D Desai  * perform the appropriate action.  Before we return, we clear the response
14618e727371SKashyap D Desai  * interrupt.
1462665484d8SDoug Ambrisko  */
14638e727371SKashyap D Desai static int
14648e727371SKashyap D Desai mrsas_complete_cmd(struct mrsas_softc *sc, u_int32_t MSIxIndex)
1465665484d8SDoug Ambrisko {
1466665484d8SDoug Ambrisko 	Mpi2ReplyDescriptorsUnion_t *desc;
1467665484d8SDoug Ambrisko 	MPI2_SCSI_IO_SUCCESS_REPLY_DESCRIPTOR *reply_desc;
1468665484d8SDoug Ambrisko 	MRSAS_RAID_SCSI_IO_REQUEST *scsi_io_req;
1469665484d8SDoug Ambrisko 	struct mrsas_mpt_cmd *cmd_mpt;
1470665484d8SDoug Ambrisko 	struct mrsas_mfi_cmd *cmd_mfi;
1471665484d8SDoug Ambrisko 	u_int8_t arm, reply_descript_type;
1472665484d8SDoug Ambrisko 	u_int16_t smid, num_completed;
1473665484d8SDoug Ambrisko 	u_int8_t status, extStatus;
1474665484d8SDoug Ambrisko 	union desc_value desc_val;
1475665484d8SDoug Ambrisko 	PLD_LOAD_BALANCE_INFO lbinfo;
1476665484d8SDoug Ambrisko 	u_int32_t device_id;
1477665484d8SDoug Ambrisko 	int threshold_reply_count = 0;
1478665484d8SDoug Ambrisko 
1479665484d8SDoug Ambrisko 
1480665484d8SDoug Ambrisko 	/* If we have a hardware error, not need to continue */
1481665484d8SDoug Ambrisko 	if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR)
1482665484d8SDoug Ambrisko 		return (DONE);
1483665484d8SDoug Ambrisko 
1484665484d8SDoug Ambrisko 	desc = sc->reply_desc_mem;
1485d18d1b47SKashyap D Desai 	desc += ((MSIxIndex * sc->reply_alloc_sz) / sizeof(MPI2_REPLY_DESCRIPTORS_UNION))
1486d18d1b47SKashyap D Desai 	    + sc->last_reply_idx[MSIxIndex];
1487665484d8SDoug Ambrisko 
1488665484d8SDoug Ambrisko 	reply_desc = (MPI2_SCSI_IO_SUCCESS_REPLY_DESCRIPTOR *) desc;
1489665484d8SDoug Ambrisko 
1490665484d8SDoug Ambrisko 	desc_val.word = desc->Words;
1491665484d8SDoug Ambrisko 	num_completed = 0;
1492665484d8SDoug Ambrisko 
1493665484d8SDoug Ambrisko 	reply_descript_type = reply_desc->ReplyFlags & MPI2_RPY_DESCRIPT_FLAGS_TYPE_MASK;
1494665484d8SDoug Ambrisko 
1495665484d8SDoug Ambrisko 	/* Find our reply descriptor for the command and process */
14968e727371SKashyap D Desai 	while ((desc_val.u.low != 0xFFFFFFFF) && (desc_val.u.high != 0xFFFFFFFF)) {
1497665484d8SDoug Ambrisko 		smid = reply_desc->SMID;
1498665484d8SDoug Ambrisko 		cmd_mpt = sc->mpt_cmd_list[smid - 1];
1499665484d8SDoug Ambrisko 		scsi_io_req = (MRSAS_RAID_SCSI_IO_REQUEST *) cmd_mpt->io_request;
1500665484d8SDoug Ambrisko 
1501665484d8SDoug Ambrisko 		status = scsi_io_req->RaidContext.status;
1502665484d8SDoug Ambrisko 		extStatus = scsi_io_req->RaidContext.exStatus;
1503665484d8SDoug Ambrisko 
15048e727371SKashyap D Desai 		switch (scsi_io_req->Function) {
1505665484d8SDoug Ambrisko 		case MPI2_FUNCTION_SCSI_IO_REQUEST:	/* Fast Path IO. */
1506665484d8SDoug Ambrisko 			device_id = cmd_mpt->ccb_ptr->ccb_h.target_id;
1507665484d8SDoug Ambrisko 			lbinfo = &sc->load_balance_info[device_id];
1508665484d8SDoug Ambrisko 			if (cmd_mpt->load_balance == MRSAS_LOAD_BALANCE_FLAG) {
1509665484d8SDoug Ambrisko 				arm = lbinfo->raid1DevHandle[0] == scsi_io_req->DevHandle ? 0 : 1;
1510f5fb2237SKashyap D Desai 				mrsas_atomic_dec(&lbinfo->scsi_pending_cmds[arm]);
1511665484d8SDoug Ambrisko 				cmd_mpt->load_balance &= ~MRSAS_LOAD_BALANCE_FLAG;
1512665484d8SDoug Ambrisko 			}
15138e727371SKashyap D Desai 			/* Fall thru and complete IO */
1514665484d8SDoug Ambrisko 		case MRSAS_MPI2_FUNCTION_LD_IO_REQUEST:
1515665484d8SDoug Ambrisko 			mrsas_map_mpt_cmd_status(cmd_mpt, status, extStatus);
1516665484d8SDoug Ambrisko 			mrsas_cmd_done(sc, cmd_mpt);
1517665484d8SDoug Ambrisko 			scsi_io_req->RaidContext.status = 0;
1518665484d8SDoug Ambrisko 			scsi_io_req->RaidContext.exStatus = 0;
1519f5fb2237SKashyap D Desai 			mrsas_atomic_dec(&sc->fw_outstanding);
1520665484d8SDoug Ambrisko 			break;
1521665484d8SDoug Ambrisko 		case MRSAS_MPI2_FUNCTION_PASSTHRU_IO_REQUEST:	/* MFI command */
1522665484d8SDoug Ambrisko 			cmd_mfi = sc->mfi_cmd_list[cmd_mpt->sync_cmd_idx];
1523665484d8SDoug Ambrisko 			mrsas_complete_mptmfi_passthru(sc, cmd_mfi, status);
1524665484d8SDoug Ambrisko 			cmd_mpt->flags = 0;
1525665484d8SDoug Ambrisko 			mrsas_release_mpt_cmd(cmd_mpt);
1526665484d8SDoug Ambrisko 			break;
1527665484d8SDoug Ambrisko 		}
1528665484d8SDoug Ambrisko 
1529d18d1b47SKashyap D Desai 		sc->last_reply_idx[MSIxIndex]++;
1530d18d1b47SKashyap D Desai 		if (sc->last_reply_idx[MSIxIndex] >= sc->reply_q_depth)
1531d18d1b47SKashyap D Desai 			sc->last_reply_idx[MSIxIndex] = 0;
1532665484d8SDoug Ambrisko 
15338e727371SKashyap D Desai 		desc->Words = ~((uint64_t)0x00);	/* set it back to all
15348e727371SKashyap D Desai 							 * 0xFFFFFFFFs */
1535665484d8SDoug Ambrisko 		num_completed++;
1536665484d8SDoug Ambrisko 		threshold_reply_count++;
1537665484d8SDoug Ambrisko 
1538665484d8SDoug Ambrisko 		/* Get the next reply descriptor */
1539d18d1b47SKashyap D Desai 		if (!sc->last_reply_idx[MSIxIndex]) {
1540665484d8SDoug Ambrisko 			desc = sc->reply_desc_mem;
1541d18d1b47SKashyap D Desai 			desc += ((MSIxIndex * sc->reply_alloc_sz) / sizeof(MPI2_REPLY_DESCRIPTORS_UNION));
1542d18d1b47SKashyap D Desai 		} else
1543665484d8SDoug Ambrisko 			desc++;
1544665484d8SDoug Ambrisko 
1545665484d8SDoug Ambrisko 		reply_desc = (MPI2_SCSI_IO_SUCCESS_REPLY_DESCRIPTOR *) desc;
1546665484d8SDoug Ambrisko 		desc_val.word = desc->Words;
1547665484d8SDoug Ambrisko 
1548665484d8SDoug Ambrisko 		reply_descript_type = reply_desc->ReplyFlags & MPI2_RPY_DESCRIPT_FLAGS_TYPE_MASK;
1549665484d8SDoug Ambrisko 
1550665484d8SDoug Ambrisko 		if (reply_descript_type == MPI2_RPY_DESCRIPT_FLAGS_UNUSED)
1551665484d8SDoug Ambrisko 			break;
1552665484d8SDoug Ambrisko 
1553665484d8SDoug Ambrisko 		/*
15548e727371SKashyap D Desai 		 * Write to reply post index after completing threshold reply
15558e727371SKashyap D Desai 		 * count and still there are more replies in reply queue
15568e727371SKashyap D Desai 		 * pending to be completed.
1557665484d8SDoug Ambrisko 		 */
1558665484d8SDoug Ambrisko 		if (threshold_reply_count >= THRESHOLD_REPLY_COUNT) {
1559d18d1b47SKashyap D Desai 			if (sc->msix_enable) {
1560d18d1b47SKashyap D Desai 				if ((sc->device_id == MRSAS_INVADER) ||
1561d18d1b47SKashyap D Desai 				    (sc->device_id == MRSAS_FURY))
1562d18d1b47SKashyap D Desai 					mrsas_write_reg(sc, sc->msix_reg_offset[MSIxIndex / 8],
1563d18d1b47SKashyap D Desai 					    ((MSIxIndex & 0x7) << 24) |
1564d18d1b47SKashyap D Desai 					    sc->last_reply_idx[MSIxIndex]);
1565d18d1b47SKashyap D Desai 				else
1566d18d1b47SKashyap D Desai 					mrsas_write_reg(sc, sc->msix_reg_offset[0], (MSIxIndex << 24) |
1567d18d1b47SKashyap D Desai 					    sc->last_reply_idx[MSIxIndex]);
1568d18d1b47SKashyap D Desai 			} else
1569d18d1b47SKashyap D Desai 				mrsas_write_reg(sc, offsetof(mrsas_reg_set,
1570d18d1b47SKashyap D Desai 				    reply_post_host_index), sc->last_reply_idx[0]);
1571d18d1b47SKashyap D Desai 
1572665484d8SDoug Ambrisko 			threshold_reply_count = 0;
1573665484d8SDoug Ambrisko 		}
1574665484d8SDoug Ambrisko 	}
1575665484d8SDoug Ambrisko 
1576665484d8SDoug Ambrisko 	/* No match, just return */
1577665484d8SDoug Ambrisko 	if (num_completed == 0)
1578665484d8SDoug Ambrisko 		return (DONE);
1579665484d8SDoug Ambrisko 
1580665484d8SDoug Ambrisko 	/* Clear response interrupt */
1581d18d1b47SKashyap D Desai 	if (sc->msix_enable) {
1582d18d1b47SKashyap D Desai 		if ((sc->device_id == MRSAS_INVADER) ||
1583d18d1b47SKashyap D Desai 		    (sc->device_id == MRSAS_FURY)) {
1584d18d1b47SKashyap D Desai 			mrsas_write_reg(sc, sc->msix_reg_offset[MSIxIndex / 8],
1585d18d1b47SKashyap D Desai 			    ((MSIxIndex & 0x7) << 24) |
1586d18d1b47SKashyap D Desai 			    sc->last_reply_idx[MSIxIndex]);
1587d18d1b47SKashyap D Desai 		} else
1588d18d1b47SKashyap D Desai 			mrsas_write_reg(sc, sc->msix_reg_offset[0], (MSIxIndex << 24) |
1589d18d1b47SKashyap D Desai 			    sc->last_reply_idx[MSIxIndex]);
1590d18d1b47SKashyap D Desai 	} else
1591d18d1b47SKashyap D Desai 		mrsas_write_reg(sc, offsetof(mrsas_reg_set,
1592d18d1b47SKashyap D Desai 		    reply_post_host_index), sc->last_reply_idx[0]);
1593665484d8SDoug Ambrisko 
1594665484d8SDoug Ambrisko 	return (0);
1595665484d8SDoug Ambrisko }
1596665484d8SDoug Ambrisko 
1597665484d8SDoug Ambrisko /*
1598665484d8SDoug Ambrisko  * mrsas_map_mpt_cmd_status:	Allocate DMAable memory.
1599665484d8SDoug Ambrisko  * input:						Adapter instance soft state
1600665484d8SDoug Ambrisko  *
1601665484d8SDoug Ambrisko  * This function is called from mrsas_complete_cmd(), for LD IO and FastPath IO.
16028e727371SKashyap D Desai  * It checks the command status and maps the appropriate CAM status for the
16038e727371SKashyap D Desai  * CCB.
1604665484d8SDoug Ambrisko  */
16058e727371SKashyap D Desai void
16068e727371SKashyap D Desai mrsas_map_mpt_cmd_status(struct mrsas_mpt_cmd *cmd, u_int8_t status, u_int8_t extStatus)
1607665484d8SDoug Ambrisko {
1608665484d8SDoug Ambrisko 	struct mrsas_softc *sc = cmd->sc;
1609665484d8SDoug Ambrisko 	u_int8_t *sense_data;
1610665484d8SDoug Ambrisko 
1611665484d8SDoug Ambrisko 	switch (status) {
1612665484d8SDoug Ambrisko 	case MFI_STAT_OK:
1613665484d8SDoug Ambrisko 		cmd->ccb_ptr->ccb_h.status = CAM_REQ_CMP;
1614665484d8SDoug Ambrisko 		break;
1615665484d8SDoug Ambrisko 	case MFI_STAT_SCSI_IO_FAILED:
1616665484d8SDoug Ambrisko 	case MFI_STAT_SCSI_DONE_WITH_ERROR:
1617665484d8SDoug Ambrisko 		cmd->ccb_ptr->ccb_h.status = CAM_SCSI_STATUS_ERROR;
1618665484d8SDoug Ambrisko 		sense_data = (u_int8_t *)&cmd->ccb_ptr->csio.sense_data;
1619665484d8SDoug Ambrisko 		if (sense_data) {
1620665484d8SDoug Ambrisko 			/* For now just copy 18 bytes back */
1621665484d8SDoug Ambrisko 			memcpy(sense_data, cmd->sense, 18);
1622665484d8SDoug Ambrisko 			cmd->ccb_ptr->csio.sense_len = 18;
1623665484d8SDoug Ambrisko 			cmd->ccb_ptr->ccb_h.status |= CAM_AUTOSNS_VALID;
1624665484d8SDoug Ambrisko 		}
1625665484d8SDoug Ambrisko 		break;
1626665484d8SDoug Ambrisko 	case MFI_STAT_LD_OFFLINE:
1627665484d8SDoug Ambrisko 	case MFI_STAT_DEVICE_NOT_FOUND:
1628665484d8SDoug Ambrisko 		if (cmd->ccb_ptr->ccb_h.target_lun)
1629665484d8SDoug Ambrisko 			cmd->ccb_ptr->ccb_h.status |= CAM_LUN_INVALID;
1630665484d8SDoug Ambrisko 		else
1631665484d8SDoug Ambrisko 			cmd->ccb_ptr->ccb_h.status |= CAM_DEV_NOT_THERE;
1632665484d8SDoug Ambrisko 		break;
1633665484d8SDoug Ambrisko 	case MFI_STAT_CONFIG_SEQ_MISMATCH:
1634665484d8SDoug Ambrisko 		cmd->ccb_ptr->ccb_h.status |= CAM_REQUEUE_REQ;
1635665484d8SDoug Ambrisko 		break;
1636665484d8SDoug Ambrisko 	default:
1637665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "FW cmd complete status %x\n", status);
1638665484d8SDoug Ambrisko 		cmd->ccb_ptr->ccb_h.status = CAM_REQ_CMP_ERR;
1639665484d8SDoug Ambrisko 		cmd->ccb_ptr->csio.scsi_status = status;
1640665484d8SDoug Ambrisko 	}
1641665484d8SDoug Ambrisko 	return;
1642665484d8SDoug Ambrisko }
1643665484d8SDoug Ambrisko 
1644665484d8SDoug Ambrisko /*
16458e727371SKashyap D Desai  * mrsas_alloc_mem:	Allocate DMAable memory
1646665484d8SDoug Ambrisko  * input:			Adapter instance soft state
1647665484d8SDoug Ambrisko  *
16488e727371SKashyap D Desai  * This function creates the parent DMA tag and allocates DMAable memory. DMA
16498e727371SKashyap D Desai  * tag describes constraints of DMA mapping. Memory allocated is mapped into
16508e727371SKashyap D Desai  * Kernel virtual address. Callback argument is physical memory address.
1651665484d8SDoug Ambrisko  */
16528e727371SKashyap D Desai static int
16538e727371SKashyap D Desai mrsas_alloc_mem(struct mrsas_softc *sc)
1654665484d8SDoug Ambrisko {
16558e727371SKashyap D Desai 	u_int32_t verbuf_size, io_req_size, reply_desc_size, sense_size, chain_frame_size,
16568e727371SKashyap D Desai 	          evt_detail_size, count;
1657665484d8SDoug Ambrisko 
1658665484d8SDoug Ambrisko 	/*
1659665484d8SDoug Ambrisko 	 * Allocate parent DMA tag
1660665484d8SDoug Ambrisko 	 */
1661665484d8SDoug Ambrisko 	if (bus_dma_tag_create(NULL,	/* parent */
1662665484d8SDoug Ambrisko 	    1,				/* alignment */
1663665484d8SDoug Ambrisko 	    0,				/* boundary */
1664665484d8SDoug Ambrisko 	    BUS_SPACE_MAXADDR,		/* lowaddr */
1665665484d8SDoug Ambrisko 	    BUS_SPACE_MAXADDR,		/* highaddr */
1666665484d8SDoug Ambrisko 	    NULL, NULL,			/* filter, filterarg */
1667665484d8SDoug Ambrisko 	    MRSAS_MAX_IO_SIZE,		/* maxsize */
1668665484d8SDoug Ambrisko 	    MRSAS_MAX_SGL,		/* nsegments */
1669665484d8SDoug Ambrisko 	    MRSAS_MAX_IO_SIZE,		/* maxsegsize */
1670665484d8SDoug Ambrisko 	    0,				/* flags */
1671665484d8SDoug Ambrisko 	    NULL, NULL,			/* lockfunc, lockarg */
1672665484d8SDoug Ambrisko 	    &sc->mrsas_parent_tag	/* tag */
1673665484d8SDoug Ambrisko 	    )) {
1674665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate parent DMA tag\n");
1675665484d8SDoug Ambrisko 		return (ENOMEM);
1676665484d8SDoug Ambrisko 	}
1677665484d8SDoug Ambrisko 	/*
1678665484d8SDoug Ambrisko 	 * Allocate for version buffer
1679665484d8SDoug Ambrisko 	 */
1680665484d8SDoug Ambrisko 	verbuf_size = MRSAS_MAX_NAME_LENGTH * (sizeof(bus_addr_t));
16818e727371SKashyap D Desai 	if (bus_dma_tag_create(sc->mrsas_parent_tag,
16828e727371SKashyap D Desai 	    1, 0,
16838e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR_32BIT,
16848e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR,
16858e727371SKashyap D Desai 	    NULL, NULL,
16868e727371SKashyap D Desai 	    verbuf_size,
16878e727371SKashyap D Desai 	    1,
16888e727371SKashyap D Desai 	    verbuf_size,
16898e727371SKashyap D Desai 	    BUS_DMA_ALLOCNOW,
16908e727371SKashyap D Desai 	    NULL, NULL,
1691665484d8SDoug Ambrisko 	    &sc->verbuf_tag)) {
1692665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate verbuf DMA tag\n");
1693665484d8SDoug Ambrisko 		return (ENOMEM);
1694665484d8SDoug Ambrisko 	}
1695665484d8SDoug Ambrisko 	if (bus_dmamem_alloc(sc->verbuf_tag, (void **)&sc->verbuf_mem,
1696665484d8SDoug Ambrisko 	    BUS_DMA_NOWAIT, &sc->verbuf_dmamap)) {
1697665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate verbuf memory\n");
1698665484d8SDoug Ambrisko 		return (ENOMEM);
1699665484d8SDoug Ambrisko 	}
1700665484d8SDoug Ambrisko 	bzero(sc->verbuf_mem, verbuf_size);
1701665484d8SDoug Ambrisko 	if (bus_dmamap_load(sc->verbuf_tag, sc->verbuf_dmamap, sc->verbuf_mem,
17028e727371SKashyap D Desai 	    verbuf_size, mrsas_addr_cb, &sc->verbuf_phys_addr,
17038e727371SKashyap D Desai 	    BUS_DMA_NOWAIT)) {
1704665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot load verbuf DMA map\n");
1705665484d8SDoug Ambrisko 		return (ENOMEM);
1706665484d8SDoug Ambrisko 	}
1707665484d8SDoug Ambrisko 	/*
1708665484d8SDoug Ambrisko 	 * Allocate IO Request Frames
1709665484d8SDoug Ambrisko 	 */
1710665484d8SDoug Ambrisko 	io_req_size = sc->io_frames_alloc_sz;
17118e727371SKashyap D Desai 	if (bus_dma_tag_create(sc->mrsas_parent_tag,
17128e727371SKashyap D Desai 	    16, 0,
17138e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR_32BIT,
17148e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR,
17158e727371SKashyap D Desai 	    NULL, NULL,
17168e727371SKashyap D Desai 	    io_req_size,
17178e727371SKashyap D Desai 	    1,
17188e727371SKashyap D Desai 	    io_req_size,
17198e727371SKashyap D Desai 	    BUS_DMA_ALLOCNOW,
17208e727371SKashyap D Desai 	    NULL, NULL,
1721665484d8SDoug Ambrisko 	    &sc->io_request_tag)) {
1722665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot create IO request tag\n");
1723665484d8SDoug Ambrisko 		return (ENOMEM);
1724665484d8SDoug Ambrisko 	}
1725665484d8SDoug Ambrisko 	if (bus_dmamem_alloc(sc->io_request_tag, (void **)&sc->io_request_mem,
1726665484d8SDoug Ambrisko 	    BUS_DMA_NOWAIT, &sc->io_request_dmamap)) {
1727665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot alloc IO request memory\n");
1728665484d8SDoug Ambrisko 		return (ENOMEM);
1729665484d8SDoug Ambrisko 	}
1730665484d8SDoug Ambrisko 	bzero(sc->io_request_mem, io_req_size);
1731665484d8SDoug Ambrisko 	if (bus_dmamap_load(sc->io_request_tag, sc->io_request_dmamap,
1732665484d8SDoug Ambrisko 	    sc->io_request_mem, io_req_size, mrsas_addr_cb,
1733665484d8SDoug Ambrisko 	    &sc->io_request_phys_addr, BUS_DMA_NOWAIT)) {
1734665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot load IO request memory\n");
1735665484d8SDoug Ambrisko 		return (ENOMEM);
1736665484d8SDoug Ambrisko 	}
1737665484d8SDoug Ambrisko 	/*
1738665484d8SDoug Ambrisko 	 * Allocate Chain Frames
1739665484d8SDoug Ambrisko 	 */
1740665484d8SDoug Ambrisko 	chain_frame_size = sc->chain_frames_alloc_sz;
17418e727371SKashyap D Desai 	if (bus_dma_tag_create(sc->mrsas_parent_tag,
17428e727371SKashyap D Desai 	    4, 0,
17438e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR_32BIT,
17448e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR,
17458e727371SKashyap D Desai 	    NULL, NULL,
17468e727371SKashyap D Desai 	    chain_frame_size,
17478e727371SKashyap D Desai 	    1,
17488e727371SKashyap D Desai 	    chain_frame_size,
17498e727371SKashyap D Desai 	    BUS_DMA_ALLOCNOW,
17508e727371SKashyap D Desai 	    NULL, NULL,
1751665484d8SDoug Ambrisko 	    &sc->chain_frame_tag)) {
1752665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot create chain frame tag\n");
1753665484d8SDoug Ambrisko 		return (ENOMEM);
1754665484d8SDoug Ambrisko 	}
1755665484d8SDoug Ambrisko 	if (bus_dmamem_alloc(sc->chain_frame_tag, (void **)&sc->chain_frame_mem,
1756665484d8SDoug Ambrisko 	    BUS_DMA_NOWAIT, &sc->chain_frame_dmamap)) {
1757665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot alloc chain frame memory\n");
1758665484d8SDoug Ambrisko 		return (ENOMEM);
1759665484d8SDoug Ambrisko 	}
1760665484d8SDoug Ambrisko 	bzero(sc->chain_frame_mem, chain_frame_size);
1761665484d8SDoug Ambrisko 	if (bus_dmamap_load(sc->chain_frame_tag, sc->chain_frame_dmamap,
1762665484d8SDoug Ambrisko 	    sc->chain_frame_mem, chain_frame_size, mrsas_addr_cb,
1763665484d8SDoug Ambrisko 	    &sc->chain_frame_phys_addr, BUS_DMA_NOWAIT)) {
1764665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot load chain frame memory\n");
1765665484d8SDoug Ambrisko 		return (ENOMEM);
1766665484d8SDoug Ambrisko 	}
1767d18d1b47SKashyap D Desai 	count = sc->msix_vectors > 0 ? sc->msix_vectors : 1;
1768665484d8SDoug Ambrisko 	/*
1769665484d8SDoug Ambrisko 	 * Allocate Reply Descriptor Array
1770665484d8SDoug Ambrisko 	 */
1771d18d1b47SKashyap D Desai 	reply_desc_size = sc->reply_alloc_sz * count;
17728e727371SKashyap D Desai 	if (bus_dma_tag_create(sc->mrsas_parent_tag,
17738e727371SKashyap D Desai 	    16, 0,
17748e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR_32BIT,
17758e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR,
17768e727371SKashyap D Desai 	    NULL, NULL,
17778e727371SKashyap D Desai 	    reply_desc_size,
17788e727371SKashyap D Desai 	    1,
17798e727371SKashyap D Desai 	    reply_desc_size,
17808e727371SKashyap D Desai 	    BUS_DMA_ALLOCNOW,
17818e727371SKashyap D Desai 	    NULL, NULL,
1782665484d8SDoug Ambrisko 	    &sc->reply_desc_tag)) {
1783665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot create reply descriptor tag\n");
1784665484d8SDoug Ambrisko 		return (ENOMEM);
1785665484d8SDoug Ambrisko 	}
1786665484d8SDoug Ambrisko 	if (bus_dmamem_alloc(sc->reply_desc_tag, (void **)&sc->reply_desc_mem,
1787665484d8SDoug Ambrisko 	    BUS_DMA_NOWAIT, &sc->reply_desc_dmamap)) {
1788665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot alloc reply descriptor memory\n");
1789665484d8SDoug Ambrisko 		return (ENOMEM);
1790665484d8SDoug Ambrisko 	}
1791665484d8SDoug Ambrisko 	if (bus_dmamap_load(sc->reply_desc_tag, sc->reply_desc_dmamap,
1792665484d8SDoug Ambrisko 	    sc->reply_desc_mem, reply_desc_size, mrsas_addr_cb,
1793665484d8SDoug Ambrisko 	    &sc->reply_desc_phys_addr, BUS_DMA_NOWAIT)) {
1794665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot load reply descriptor memory\n");
1795665484d8SDoug Ambrisko 		return (ENOMEM);
1796665484d8SDoug Ambrisko 	}
1797665484d8SDoug Ambrisko 	/*
1798665484d8SDoug Ambrisko 	 * Allocate Sense Buffer Array.  Keep in lower 4GB
1799665484d8SDoug Ambrisko 	 */
1800665484d8SDoug Ambrisko 	sense_size = sc->max_fw_cmds * MRSAS_SENSE_LEN;
18018e727371SKashyap D Desai 	if (bus_dma_tag_create(sc->mrsas_parent_tag,
18028e727371SKashyap D Desai 	    64, 0,
18038e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR_32BIT,
18048e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR,
18058e727371SKashyap D Desai 	    NULL, NULL,
18068e727371SKashyap D Desai 	    sense_size,
18078e727371SKashyap D Desai 	    1,
18088e727371SKashyap D Desai 	    sense_size,
18098e727371SKashyap D Desai 	    BUS_DMA_ALLOCNOW,
18108e727371SKashyap D Desai 	    NULL, NULL,
1811665484d8SDoug Ambrisko 	    &sc->sense_tag)) {
1812665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate sense buf tag\n");
1813665484d8SDoug Ambrisko 		return (ENOMEM);
1814665484d8SDoug Ambrisko 	}
1815665484d8SDoug Ambrisko 	if (bus_dmamem_alloc(sc->sense_tag, (void **)&sc->sense_mem,
1816665484d8SDoug Ambrisko 	    BUS_DMA_NOWAIT, &sc->sense_dmamap)) {
1817665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate sense buf memory\n");
1818665484d8SDoug Ambrisko 		return (ENOMEM);
1819665484d8SDoug Ambrisko 	}
1820665484d8SDoug Ambrisko 	if (bus_dmamap_load(sc->sense_tag, sc->sense_dmamap,
1821665484d8SDoug Ambrisko 	    sc->sense_mem, sense_size, mrsas_addr_cb, &sc->sense_phys_addr,
1822665484d8SDoug Ambrisko 	    BUS_DMA_NOWAIT)) {
1823665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot load sense buf memory\n");
1824665484d8SDoug Ambrisko 		return (ENOMEM);
1825665484d8SDoug Ambrisko 	}
1826665484d8SDoug Ambrisko 	/*
1827665484d8SDoug Ambrisko 	 * Allocate for Event detail structure
1828665484d8SDoug Ambrisko 	 */
1829665484d8SDoug Ambrisko 	evt_detail_size = sizeof(struct mrsas_evt_detail);
18308e727371SKashyap D Desai 	if (bus_dma_tag_create(sc->mrsas_parent_tag,
18318e727371SKashyap D Desai 	    1, 0,
18328e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR_32BIT,
18338e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR,
18348e727371SKashyap D Desai 	    NULL, NULL,
18358e727371SKashyap D Desai 	    evt_detail_size,
18368e727371SKashyap D Desai 	    1,
18378e727371SKashyap D Desai 	    evt_detail_size,
18388e727371SKashyap D Desai 	    BUS_DMA_ALLOCNOW,
18398e727371SKashyap D Desai 	    NULL, NULL,
1840665484d8SDoug Ambrisko 	    &sc->evt_detail_tag)) {
1841665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot create Event detail tag\n");
1842665484d8SDoug Ambrisko 		return (ENOMEM);
1843665484d8SDoug Ambrisko 	}
1844665484d8SDoug Ambrisko 	if (bus_dmamem_alloc(sc->evt_detail_tag, (void **)&sc->evt_detail_mem,
1845665484d8SDoug Ambrisko 	    BUS_DMA_NOWAIT, &sc->evt_detail_dmamap)) {
1846665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot alloc Event detail buffer memory\n");
1847665484d8SDoug Ambrisko 		return (ENOMEM);
1848665484d8SDoug Ambrisko 	}
1849665484d8SDoug Ambrisko 	bzero(sc->evt_detail_mem, evt_detail_size);
1850665484d8SDoug Ambrisko 	if (bus_dmamap_load(sc->evt_detail_tag, sc->evt_detail_dmamap,
1851665484d8SDoug Ambrisko 	    sc->evt_detail_mem, evt_detail_size, mrsas_addr_cb,
1852665484d8SDoug Ambrisko 	    &sc->evt_detail_phys_addr, BUS_DMA_NOWAIT)) {
1853665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot load Event detail buffer memory\n");
1854665484d8SDoug Ambrisko 		return (ENOMEM);
1855665484d8SDoug Ambrisko 	}
1856665484d8SDoug Ambrisko 	/*
1857665484d8SDoug Ambrisko 	 * Create a dma tag for data buffers; size will be the maximum
1858665484d8SDoug Ambrisko 	 * possible I/O size (280kB).
1859665484d8SDoug Ambrisko 	 */
18608e727371SKashyap D Desai 	if (bus_dma_tag_create(sc->mrsas_parent_tag,
18618e727371SKashyap D Desai 	    1,
18628e727371SKashyap D Desai 	    0,
18638e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR,
18648e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR,
18658e727371SKashyap D Desai 	    NULL, NULL,
18668e727371SKashyap D Desai 	    MRSAS_MAX_IO_SIZE,
18678e727371SKashyap D Desai 	    MRSAS_MAX_SGL,
18688e727371SKashyap D Desai 	    MRSAS_MAX_IO_SIZE,
18698e727371SKashyap D Desai 	    BUS_DMA_ALLOCNOW,
18708e727371SKashyap D Desai 	    busdma_lock_mutex,
18718e727371SKashyap D Desai 	    &sc->io_lock,
1872665484d8SDoug Ambrisko 	    &sc->data_tag)) {
1873665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot create data dma tag\n");
1874665484d8SDoug Ambrisko 		return (ENOMEM);
1875665484d8SDoug Ambrisko 	}
1876665484d8SDoug Ambrisko 	return (0);
1877665484d8SDoug Ambrisko }
1878665484d8SDoug Ambrisko 
1879665484d8SDoug Ambrisko /*
1880665484d8SDoug Ambrisko  * mrsas_addr_cb:	Callback function of bus_dmamap_load()
18818e727371SKashyap D Desai  * input:			callback argument, machine dependent type
18828e727371SKashyap D Desai  * 					that describes DMA segments, number of segments, error code
1883665484d8SDoug Ambrisko  *
18848e727371SKashyap D Desai  * This function is for the driver to receive mapping information resultant of
18858e727371SKashyap D Desai  * the bus_dmamap_load(). The information is actually not being used, but the
18868e727371SKashyap D Desai  * address is saved anyway.
1887665484d8SDoug Ambrisko  */
1888665484d8SDoug Ambrisko void
1889665484d8SDoug Ambrisko mrsas_addr_cb(void *arg, bus_dma_segment_t *segs, int nsegs, int error)
1890665484d8SDoug Ambrisko {
1891665484d8SDoug Ambrisko 	bus_addr_t *addr;
1892665484d8SDoug Ambrisko 
1893665484d8SDoug Ambrisko 	addr = arg;
1894665484d8SDoug Ambrisko 	*addr = segs[0].ds_addr;
1895665484d8SDoug Ambrisko }
1896665484d8SDoug Ambrisko 
1897665484d8SDoug Ambrisko /*
1898665484d8SDoug Ambrisko  * mrsas_setup_raidmap:	Set up RAID map.
1899665484d8SDoug Ambrisko  * input:				Adapter instance soft state
1900665484d8SDoug Ambrisko  *
1901665484d8SDoug Ambrisko  * Allocate DMA memory for the RAID maps and perform setup.
1902665484d8SDoug Ambrisko  */
19038e727371SKashyap D Desai static int
19048e727371SKashyap D Desai mrsas_setup_raidmap(struct mrsas_softc *sc)
1905665484d8SDoug Ambrisko {
19064799d485SKashyap D Desai 	int i;
19074799d485SKashyap D Desai 
19084799d485SKashyap D Desai 	sc->drv_supported_vd_count =
19094799d485SKashyap D Desai 	    MRSAS_MAX_LD_CHANNELS * MRSAS_MAX_DEV_PER_CHANNEL;
19104799d485SKashyap D Desai 	sc->drv_supported_pd_count =
19114799d485SKashyap D Desai 	    MRSAS_MAX_PD_CHANNELS * MRSAS_MAX_DEV_PER_CHANNEL;
19124799d485SKashyap D Desai 
19134799d485SKashyap D Desai 	if (sc->max256vdSupport) {
19144799d485SKashyap D Desai 		sc->fw_supported_vd_count = MAX_LOGICAL_DRIVES_EXT;
19154799d485SKashyap D Desai 		sc->fw_supported_pd_count = MAX_PHYSICAL_DEVICES;
19164799d485SKashyap D Desai 	} else {
19174799d485SKashyap D Desai 		sc->fw_supported_vd_count = MAX_LOGICAL_DRIVES;
19184799d485SKashyap D Desai 		sc->fw_supported_pd_count = MAX_PHYSICAL_DEVICES;
19194799d485SKashyap D Desai 	}
19204799d485SKashyap D Desai 
19214799d485SKashyap D Desai #if VD_EXT_DEBUG
19224799d485SKashyap D Desai 	device_printf(sc->mrsas_dev, "FW supports: max256vdSupport = %s\n",
19234799d485SKashyap D Desai 	    sc->max256vdSupport ? "YES" : "NO");
19244799d485SKashyap D Desai 	device_printf(sc->mrsas_dev, "FW supports %dVDs %dPDs\n"
19254799d485SKashyap D Desai 	    "DRIVER supports %dVDs  %dPDs \n",
19264799d485SKashyap D Desai 	    sc->fw_supported_vd_count, sc->fw_supported_pd_count,
19274799d485SKashyap D Desai 	    sc->drv_supported_vd_count, sc->drv_supported_pd_count);
19284799d485SKashyap D Desai #endif
19294799d485SKashyap D Desai 
19304799d485SKashyap D Desai 	sc->old_map_sz = sizeof(MR_FW_RAID_MAP) +
19314799d485SKashyap D Desai 	    (sizeof(MR_LD_SPAN_MAP) * (sc->fw_supported_vd_count - 1));
19324799d485SKashyap D Desai 	sc->new_map_sz = sizeof(MR_FW_RAID_MAP_EXT);
19334799d485SKashyap D Desai 	sc->drv_map_sz = sizeof(MR_DRV_RAID_MAP) +
19344799d485SKashyap D Desai 	    (sizeof(MR_LD_SPAN_MAP) * (sc->drv_supported_vd_count - 1));
19354799d485SKashyap D Desai 
19364799d485SKashyap D Desai 	for (i = 0; i < 2; i++) {
19374799d485SKashyap D Desai 		sc->ld_drv_map[i] =
19384799d485SKashyap D Desai 		    (void *)malloc(sc->drv_map_sz, M_MRSAS, M_NOWAIT);
19394799d485SKashyap D Desai 		/* Do Error handling */
19404799d485SKashyap D Desai 		if (!sc->ld_drv_map[i]) {
19414799d485SKashyap D Desai 			device_printf(sc->mrsas_dev, "Could not allocate memory for local map");
19424799d485SKashyap D Desai 
19434799d485SKashyap D Desai 			if (i == 1)
19444799d485SKashyap D Desai 				free(sc->ld_drv_map[0], M_MRSAS);
19458e727371SKashyap D Desai 			/* ABORT driver initialization */
19464799d485SKashyap D Desai 			goto ABORT;
19474799d485SKashyap D Desai 		}
19484799d485SKashyap D Desai 	}
19494799d485SKashyap D Desai 
19504799d485SKashyap D Desai 	sc->max_map_sz = max(sc->old_map_sz, sc->new_map_sz);
19514799d485SKashyap D Desai 
19524799d485SKashyap D Desai 	if (sc->max256vdSupport)
19534799d485SKashyap D Desai 		sc->current_map_sz = sc->new_map_sz;
19544799d485SKashyap D Desai 	else
19554799d485SKashyap D Desai 		sc->current_map_sz = sc->old_map_sz;
19564799d485SKashyap D Desai 
1957665484d8SDoug Ambrisko 
19588e727371SKashyap D Desai 	for (int i = 0; i < 2; i++) {
19598e727371SKashyap D Desai 		if (bus_dma_tag_create(sc->mrsas_parent_tag,
19608e727371SKashyap D Desai 		    4, 0,
19618e727371SKashyap D Desai 		    BUS_SPACE_MAXADDR_32BIT,
19628e727371SKashyap D Desai 		    BUS_SPACE_MAXADDR,
19638e727371SKashyap D Desai 		    NULL, NULL,
19648e727371SKashyap D Desai 		    sc->max_map_sz,
19658e727371SKashyap D Desai 		    1,
19668e727371SKashyap D Desai 		    sc->max_map_sz,
19678e727371SKashyap D Desai 		    BUS_DMA_ALLOCNOW,
19688e727371SKashyap D Desai 		    NULL, NULL,
1969665484d8SDoug Ambrisko 		    &sc->raidmap_tag[i])) {
19704799d485SKashyap D Desai 			device_printf(sc->mrsas_dev,
19714799d485SKashyap D Desai 			    "Cannot allocate raid map tag.\n");
1972665484d8SDoug Ambrisko 			return (ENOMEM);
1973665484d8SDoug Ambrisko 		}
19744799d485SKashyap D Desai 		if (bus_dmamem_alloc(sc->raidmap_tag[i],
19754799d485SKashyap D Desai 		    (void **)&sc->raidmap_mem[i],
1976665484d8SDoug Ambrisko 		    BUS_DMA_NOWAIT, &sc->raidmap_dmamap[i])) {
19774799d485SKashyap D Desai 			device_printf(sc->mrsas_dev,
19784799d485SKashyap D Desai 			    "Cannot allocate raidmap memory.\n");
1979665484d8SDoug Ambrisko 			return (ENOMEM);
1980665484d8SDoug Ambrisko 		}
19814799d485SKashyap D Desai 		bzero(sc->raidmap_mem[i], sc->max_map_sz);
19824799d485SKashyap D Desai 
1983665484d8SDoug Ambrisko 		if (bus_dmamap_load(sc->raidmap_tag[i], sc->raidmap_dmamap[i],
19844799d485SKashyap D Desai 		    sc->raidmap_mem[i], sc->max_map_sz,
19854799d485SKashyap D Desai 		    mrsas_addr_cb, &sc->raidmap_phys_addr[i],
1986665484d8SDoug Ambrisko 		    BUS_DMA_NOWAIT)) {
1987665484d8SDoug Ambrisko 			device_printf(sc->mrsas_dev, "Cannot load raidmap memory.\n");
1988665484d8SDoug Ambrisko 			return (ENOMEM);
1989665484d8SDoug Ambrisko 		}
1990665484d8SDoug Ambrisko 		if (!sc->raidmap_mem[i]) {
19914799d485SKashyap D Desai 			device_printf(sc->mrsas_dev,
19924799d485SKashyap D Desai 			    "Cannot allocate memory for raid map.\n");
1993665484d8SDoug Ambrisko 			return (ENOMEM);
1994665484d8SDoug Ambrisko 		}
1995665484d8SDoug Ambrisko 	}
1996665484d8SDoug Ambrisko 
1997665484d8SDoug Ambrisko 	if (!mrsas_get_map_info(sc))
1998665484d8SDoug Ambrisko 		mrsas_sync_map_info(sc);
1999665484d8SDoug Ambrisko 
2000665484d8SDoug Ambrisko 	return (0);
20014799d485SKashyap D Desai 
20024799d485SKashyap D Desai ABORT:
20034799d485SKashyap D Desai 	return (1);
2004665484d8SDoug Ambrisko }
2005665484d8SDoug Ambrisko 
20068e727371SKashyap D Desai /*
2007665484d8SDoug Ambrisko  * mrsas_init_fw:	Initialize Firmware
2008665484d8SDoug Ambrisko  * input:			Adapter soft state
2009665484d8SDoug Ambrisko  *
20108e727371SKashyap D Desai  * Calls transition_to_ready() to make sure Firmware is in operational state and
20118e727371SKashyap D Desai  * calls mrsas_init_adapter() to send IOC_INIT command to Firmware.  It
20128e727371SKashyap D Desai  * issues internal commands to get the controller info after the IOC_INIT
20138e727371SKashyap D Desai  * command response is received by Firmware.  Note:  code relating to
20148e727371SKashyap D Desai  * get_pdlist, get_ld_list and max_sectors are currently not being used, it
20158e727371SKashyap D Desai  * is left here as placeholder.
2016665484d8SDoug Ambrisko  */
20178e727371SKashyap D Desai static int
20188e727371SKashyap D Desai mrsas_init_fw(struct mrsas_softc *sc)
2019665484d8SDoug Ambrisko {
2020d18d1b47SKashyap D Desai 
2021d18d1b47SKashyap D Desai 	int ret, loop, ocr = 0;
2022665484d8SDoug Ambrisko 	u_int32_t max_sectors_1;
2023665484d8SDoug Ambrisko 	u_int32_t max_sectors_2;
2024665484d8SDoug Ambrisko 	u_int32_t tmp_sectors;
2025665484d8SDoug Ambrisko 	struct mrsas_ctrl_info *ctrl_info;
2026d18d1b47SKashyap D Desai 	u_int32_t scratch_pad_2;
2027d18d1b47SKashyap D Desai 	int msix_enable = 0;
2028d18d1b47SKashyap D Desai 	int fw_msix_count = 0;
2029665484d8SDoug Ambrisko 
2030665484d8SDoug Ambrisko 	/* Make sure Firmware is ready */
2031665484d8SDoug Ambrisko 	ret = mrsas_transition_to_ready(sc, ocr);
2032665484d8SDoug Ambrisko 	if (ret != SUCCESS) {
2033665484d8SDoug Ambrisko 		return (ret);
2034665484d8SDoug Ambrisko 	}
2035d18d1b47SKashyap D Desai 	/* MSI-x index 0- reply post host index register */
2036d18d1b47SKashyap D Desai 	sc->msix_reg_offset[0] = MPI2_REPLY_POST_HOST_INDEX_OFFSET;
2037d18d1b47SKashyap D Desai 	/* Check if MSI-X is supported while in ready state */
2038d18d1b47SKashyap D Desai 	msix_enable = (mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad)) & 0x4000000) >> 0x1a;
2039d18d1b47SKashyap D Desai 
2040d18d1b47SKashyap D Desai 	if (msix_enable) {
2041d18d1b47SKashyap D Desai 		scratch_pad_2 = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2042d18d1b47SKashyap D Desai 		    outbound_scratch_pad_2));
2043d18d1b47SKashyap D Desai 
2044d18d1b47SKashyap D Desai 		/* Check max MSI-X vectors */
2045d18d1b47SKashyap D Desai 		if (sc->device_id == MRSAS_TBOLT) {
2046d18d1b47SKashyap D Desai 			sc->msix_vectors = (scratch_pad_2
2047d18d1b47SKashyap D Desai 			    & MR_MAX_REPLY_QUEUES_OFFSET) + 1;
2048d18d1b47SKashyap D Desai 			fw_msix_count = sc->msix_vectors;
2049d18d1b47SKashyap D Desai 		} else {
2050d18d1b47SKashyap D Desai 			/* Invader/Fury supports 96 MSI-X vectors */
2051d18d1b47SKashyap D Desai 			sc->msix_vectors = ((scratch_pad_2
2052d18d1b47SKashyap D Desai 			    & MR_MAX_REPLY_QUEUES_EXT_OFFSET)
2053d18d1b47SKashyap D Desai 			    >> MR_MAX_REPLY_QUEUES_EXT_OFFSET_SHIFT) + 1;
2054d18d1b47SKashyap D Desai 			fw_msix_count = sc->msix_vectors;
2055d18d1b47SKashyap D Desai 
2056d18d1b47SKashyap D Desai 			for (loop = 1; loop < MR_MAX_MSIX_REG_ARRAY;
2057d18d1b47SKashyap D Desai 			    loop++) {
2058d18d1b47SKashyap D Desai 				sc->msix_reg_offset[loop] =
2059d18d1b47SKashyap D Desai 				    MPI2_SUP_REPLY_POST_HOST_INDEX_OFFSET +
2060d18d1b47SKashyap D Desai 				    (loop * 0x10);
2061d18d1b47SKashyap D Desai 			}
2062d18d1b47SKashyap D Desai 		}
2063d18d1b47SKashyap D Desai 
2064d18d1b47SKashyap D Desai 		/* Don't bother allocating more MSI-X vectors than cpus */
2065d18d1b47SKashyap D Desai 		sc->msix_vectors = min(sc->msix_vectors,
2066d18d1b47SKashyap D Desai 		    mp_ncpus);
2067d18d1b47SKashyap D Desai 
2068d18d1b47SKashyap D Desai 		/* Allocate MSI-x vectors */
2069d18d1b47SKashyap D Desai 		if (mrsas_allocate_msix(sc) == SUCCESS)
2070d18d1b47SKashyap D Desai 			sc->msix_enable = 1;
2071d18d1b47SKashyap D Desai 		else
2072d18d1b47SKashyap D Desai 			sc->msix_enable = 0;
2073d18d1b47SKashyap D Desai 
2074d18d1b47SKashyap D Desai 		device_printf(sc->mrsas_dev, "FW supports <%d> MSIX vector,"
2075d18d1b47SKashyap D Desai 		    "Online CPU %d Current MSIX <%d>\n",
2076d18d1b47SKashyap D Desai 		    fw_msix_count, mp_ncpus, sc->msix_vectors);
2077d18d1b47SKashyap D Desai 	}
2078665484d8SDoug Ambrisko 	if (mrsas_init_adapter(sc) != SUCCESS) {
2079665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Adapter initialize Fail.\n");
2080665484d8SDoug Ambrisko 		return (1);
2081665484d8SDoug Ambrisko 	}
2082665484d8SDoug Ambrisko 	/* Allocate internal commands for pass-thru */
2083665484d8SDoug Ambrisko 	if (mrsas_alloc_mfi_cmds(sc) != SUCCESS) {
2084665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Allocate MFI cmd failed.\n");
2085665484d8SDoug Ambrisko 		return (1);
2086665484d8SDoug Ambrisko 	}
20874799d485SKashyap D Desai 	/*
20888e727371SKashyap D Desai 	 * Get the controller info from FW, so that the MAX VD support
20898e727371SKashyap D Desai 	 * availability can be decided.
20904799d485SKashyap D Desai 	 */
20914799d485SKashyap D Desai 	ctrl_info = malloc(sizeof(struct mrsas_ctrl_info), M_MRSAS, M_NOWAIT);
20924799d485SKashyap D Desai 	if (!ctrl_info)
20934799d485SKashyap D Desai 		device_printf(sc->mrsas_dev, "Malloc for ctrl_info failed.\n");
20944799d485SKashyap D Desai 
20954799d485SKashyap D Desai 	if (mrsas_get_ctrl_info(sc, ctrl_info)) {
20964799d485SKashyap D Desai 		device_printf(sc->mrsas_dev, "Unable to get FW ctrl_info.\n");
20974799d485SKashyap D Desai 	}
2098*77cf7df8SKashyap D Desai 	sc->secure_jbod_support =
2099*77cf7df8SKashyap D Desai 		(u_int8_t) ctrl_info->adapterOperations3.supportSecurityonJBOD;
2100*77cf7df8SKashyap D Desai 
2101*77cf7df8SKashyap D Desai 	if (sc->secure_jbod_support)
2102*77cf7df8SKashyap D Desai 		device_printf(sc->mrsas_dev, "FW supports SED \n");
2103*77cf7df8SKashyap D Desai 
21044799d485SKashyap D Desai 	sc->max256vdSupport =
21054799d485SKashyap D Desai 	    (u_int8_t)ctrl_info->adapterOperations3.supportMaxExtLDs;
21064799d485SKashyap D Desai 
21074799d485SKashyap D Desai 	if (ctrl_info->max_lds > 64) {
21084799d485SKashyap D Desai 		sc->max256vdSupport = 1;
21094799d485SKashyap D Desai 	}
2110665484d8SDoug Ambrisko 	if (mrsas_setup_raidmap(sc) != SUCCESS) {
2111665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Set up RAID map failed.\n");
2112665484d8SDoug Ambrisko 		return (1);
2113665484d8SDoug Ambrisko 	}
2114665484d8SDoug Ambrisko 	/* For pass-thru, get PD/LD list and controller info */
21154799d485SKashyap D Desai 	memset(sc->pd_list, 0,
21164799d485SKashyap D Desai 	    MRSAS_MAX_PD * sizeof(struct mrsas_pd_list));
2117665484d8SDoug Ambrisko 	mrsas_get_pd_list(sc);
2118665484d8SDoug Ambrisko 
21194799d485SKashyap D Desai 	memset(sc->ld_ids, 0xff, MRSAS_MAX_LD_IDS);
2120665484d8SDoug Ambrisko 	mrsas_get_ld_list(sc);
2121665484d8SDoug Ambrisko 
2122665484d8SDoug Ambrisko 	/*
21238e727371SKashyap D Desai 	 * Compute the max allowed sectors per IO: The controller info has
21248e727371SKashyap D Desai 	 * two limits on max sectors. Driver should use the minimum of these
21258e727371SKashyap D Desai 	 * two.
2126665484d8SDoug Ambrisko 	 *
2127665484d8SDoug Ambrisko 	 * 1 << stripe_sz_ops.min = max sectors per strip
2128665484d8SDoug Ambrisko 	 *
21298e727371SKashyap D Desai 	 * Note that older firmwares ( < FW ver 30) didn't report information to
21308e727371SKashyap D Desai 	 * calculate max_sectors_1. So the number ended up as zero always.
2131665484d8SDoug Ambrisko 	 */
2132665484d8SDoug Ambrisko 	tmp_sectors = 0;
2133665484d8SDoug Ambrisko 	max_sectors_1 = (1 << ctrl_info->stripe_sz_ops.min) *
2134665484d8SDoug Ambrisko 	    ctrl_info->max_strips_per_io;
2135665484d8SDoug Ambrisko 	max_sectors_2 = ctrl_info->max_request_size;
2136665484d8SDoug Ambrisko 	tmp_sectors = min(max_sectors_1, max_sectors_2);
21374799d485SKashyap D Desai 	sc->max_sectors_per_req = sc->max_num_sge * MRSAS_PAGE_SIZE / 512;
21384799d485SKashyap D Desai 
21394799d485SKashyap D Desai 	if (tmp_sectors && (sc->max_sectors_per_req > tmp_sectors))
21404799d485SKashyap D Desai 		sc->max_sectors_per_req = tmp_sectors;
21414799d485SKashyap D Desai 
2142665484d8SDoug Ambrisko 	sc->disableOnlineCtrlReset =
2143665484d8SDoug Ambrisko 	    ctrl_info->properties.OnOffProperties.disableOnlineCtrlReset;
2144665484d8SDoug Ambrisko 	sc->UnevenSpanSupport =
2145665484d8SDoug Ambrisko 	    ctrl_info->adapterOperations2.supportUnevenSpans;
2146665484d8SDoug Ambrisko 	if (sc->UnevenSpanSupport) {
21478e727371SKashyap D Desai 		device_printf(sc->mrsas_dev, "FW supports: UnevenSpanSupport=%x\n\n",
2148665484d8SDoug Ambrisko 		    sc->UnevenSpanSupport);
21494799d485SKashyap D Desai 
2150665484d8SDoug Ambrisko 		if (MR_ValidateMapInfo(sc))
2151665484d8SDoug Ambrisko 			sc->fast_path_io = 1;
2152665484d8SDoug Ambrisko 		else
2153665484d8SDoug Ambrisko 			sc->fast_path_io = 0;
2154665484d8SDoug Ambrisko 	}
2155665484d8SDoug Ambrisko 	if (ctrl_info)
2156665484d8SDoug Ambrisko 		free(ctrl_info, M_MRSAS);
2157665484d8SDoug Ambrisko 
2158665484d8SDoug Ambrisko 	return (0);
2159665484d8SDoug Ambrisko }
2160665484d8SDoug Ambrisko 
21618e727371SKashyap D Desai /*
2162665484d8SDoug Ambrisko  * mrsas_init_adapter:	Initializes the adapter/controller
2163665484d8SDoug Ambrisko  * input:				Adapter soft state
2164665484d8SDoug Ambrisko  *
2165665484d8SDoug Ambrisko  * Prepares for the issuing of the IOC Init cmd to FW for initializing the
2166665484d8SDoug Ambrisko  * ROC/controller.  The FW register is read to determined the number of
2167665484d8SDoug Ambrisko  * commands that is supported.  All memory allocations for IO is based on
2168665484d8SDoug Ambrisko  * max_cmd.  Appropriate calculations are performed in this function.
2169665484d8SDoug Ambrisko  */
21708e727371SKashyap D Desai int
21718e727371SKashyap D Desai mrsas_init_adapter(struct mrsas_softc *sc)
2172665484d8SDoug Ambrisko {
2173665484d8SDoug Ambrisko 	uint32_t status;
2174665484d8SDoug Ambrisko 	u_int32_t max_cmd;
2175665484d8SDoug Ambrisko 	int ret;
2176d18d1b47SKashyap D Desai 	int i = 0;
2177665484d8SDoug Ambrisko 
2178665484d8SDoug Ambrisko 	/* Read FW status register */
2179665484d8SDoug Ambrisko 	status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad));
2180665484d8SDoug Ambrisko 
2181665484d8SDoug Ambrisko 	/* Get operational params from status register */
2182665484d8SDoug Ambrisko 	sc->max_fw_cmds = status & MRSAS_FWSTATE_MAXCMD_MASK;
2183665484d8SDoug Ambrisko 
2184665484d8SDoug Ambrisko 	/* Decrement the max supported by 1, to correlate with FW */
2185665484d8SDoug Ambrisko 	sc->max_fw_cmds = sc->max_fw_cmds - 1;
2186665484d8SDoug Ambrisko 	max_cmd = sc->max_fw_cmds;
2187665484d8SDoug Ambrisko 
2188665484d8SDoug Ambrisko 	/* Determine allocation size of command frames */
2189d18d1b47SKashyap D Desai 	sc->reply_q_depth = ((max_cmd + 1 + 15) / 16 * 16);
2190665484d8SDoug Ambrisko 	sc->request_alloc_sz = sizeof(MRSAS_REQUEST_DESCRIPTOR_UNION) * max_cmd;
2191665484d8SDoug Ambrisko 	sc->reply_alloc_sz = sizeof(MPI2_REPLY_DESCRIPTORS_UNION) * (sc->reply_q_depth);
2192665484d8SDoug Ambrisko 	sc->io_frames_alloc_sz = MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE + (MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE * (max_cmd + 1));
2193665484d8SDoug Ambrisko 	sc->chain_frames_alloc_sz = 1024 * max_cmd;
2194665484d8SDoug Ambrisko 	sc->max_sge_in_main_msg = (MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE -
2195665484d8SDoug Ambrisko 	    offsetof(MRSAS_RAID_SCSI_IO_REQUEST, SGL)) / 16;
2196665484d8SDoug Ambrisko 
2197665484d8SDoug Ambrisko 	sc->max_sge_in_chain = MRSAS_MAX_SZ_CHAIN_FRAME / sizeof(MPI2_SGE_IO_UNION);
2198665484d8SDoug Ambrisko 	sc->max_num_sge = sc->max_sge_in_main_msg + sc->max_sge_in_chain - 2;
2199665484d8SDoug Ambrisko 
2200665484d8SDoug Ambrisko 	/* Used for pass thru MFI frame (DCMD) */
2201665484d8SDoug Ambrisko 	sc->chain_offset_mfi_pthru = offsetof(MRSAS_RAID_SCSI_IO_REQUEST, SGL) / 16;
2202665484d8SDoug Ambrisko 
2203665484d8SDoug Ambrisko 	sc->chain_offset_io_request = (MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE -
2204665484d8SDoug Ambrisko 	    sizeof(MPI2_SGE_IO_UNION)) / 16;
2205665484d8SDoug Ambrisko 
2206d18d1b47SKashyap D Desai 	int count = sc->msix_vectors > 0 ? sc->msix_vectors : 1;
22078e727371SKashyap D Desai 
2208d18d1b47SKashyap D Desai 	for (i = 0; i < count; i++)
2209d18d1b47SKashyap D Desai 		sc->last_reply_idx[i] = 0;
2210665484d8SDoug Ambrisko 
2211665484d8SDoug Ambrisko 	ret = mrsas_alloc_mem(sc);
2212665484d8SDoug Ambrisko 	if (ret != SUCCESS)
2213665484d8SDoug Ambrisko 		return (ret);
2214665484d8SDoug Ambrisko 
2215665484d8SDoug Ambrisko 	ret = mrsas_alloc_mpt_cmds(sc);
2216665484d8SDoug Ambrisko 	if (ret != SUCCESS)
2217665484d8SDoug Ambrisko 		return (ret);
2218665484d8SDoug Ambrisko 
2219665484d8SDoug Ambrisko 	ret = mrsas_ioc_init(sc);
2220665484d8SDoug Ambrisko 	if (ret != SUCCESS)
2221665484d8SDoug Ambrisko 		return (ret);
2222665484d8SDoug Ambrisko 
2223665484d8SDoug Ambrisko 	return (0);
2224665484d8SDoug Ambrisko }
2225665484d8SDoug Ambrisko 
22268e727371SKashyap D Desai /*
2227665484d8SDoug Ambrisko  * mrsas_alloc_ioc_cmd:	Allocates memory for IOC Init command
2228665484d8SDoug Ambrisko  * input:				Adapter soft state
2229665484d8SDoug Ambrisko  *
2230665484d8SDoug Ambrisko  * Allocates for the IOC Init cmd to FW to initialize the ROC/controller.
2231665484d8SDoug Ambrisko  */
22328e727371SKashyap D Desai int
22338e727371SKashyap D Desai mrsas_alloc_ioc_cmd(struct mrsas_softc *sc)
2234665484d8SDoug Ambrisko {
2235665484d8SDoug Ambrisko 	int ioc_init_size;
2236665484d8SDoug Ambrisko 
2237665484d8SDoug Ambrisko 	/* Allocate IOC INIT command */
2238665484d8SDoug Ambrisko 	ioc_init_size = 1024 + sizeof(MPI2_IOC_INIT_REQUEST);
22398e727371SKashyap D Desai 	if (bus_dma_tag_create(sc->mrsas_parent_tag,
22408e727371SKashyap D Desai 	    1, 0,
22418e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR_32BIT,
22428e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR,
22438e727371SKashyap D Desai 	    NULL, NULL,
22448e727371SKashyap D Desai 	    ioc_init_size,
22458e727371SKashyap D Desai 	    1,
22468e727371SKashyap D Desai 	    ioc_init_size,
22478e727371SKashyap D Desai 	    BUS_DMA_ALLOCNOW,
22488e727371SKashyap D Desai 	    NULL, NULL,
2249665484d8SDoug Ambrisko 	    &sc->ioc_init_tag)) {
2250665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate ioc init tag\n");
2251665484d8SDoug Ambrisko 		return (ENOMEM);
2252665484d8SDoug Ambrisko 	}
2253665484d8SDoug Ambrisko 	if (bus_dmamem_alloc(sc->ioc_init_tag, (void **)&sc->ioc_init_mem,
2254665484d8SDoug Ambrisko 	    BUS_DMA_NOWAIT, &sc->ioc_init_dmamap)) {
2255665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate ioc init cmd mem\n");
2256665484d8SDoug Ambrisko 		return (ENOMEM);
2257665484d8SDoug Ambrisko 	}
2258665484d8SDoug Ambrisko 	bzero(sc->ioc_init_mem, ioc_init_size);
2259665484d8SDoug Ambrisko 	if (bus_dmamap_load(sc->ioc_init_tag, sc->ioc_init_dmamap,
2260665484d8SDoug Ambrisko 	    sc->ioc_init_mem, ioc_init_size, mrsas_addr_cb,
2261665484d8SDoug Ambrisko 	    &sc->ioc_init_phys_mem, BUS_DMA_NOWAIT)) {
2262665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot load ioc init cmd mem\n");
2263665484d8SDoug Ambrisko 		return (ENOMEM);
2264665484d8SDoug Ambrisko 	}
2265665484d8SDoug Ambrisko 	return (0);
2266665484d8SDoug Ambrisko }
2267665484d8SDoug Ambrisko 
22688e727371SKashyap D Desai /*
2269665484d8SDoug Ambrisko  * mrsas_free_ioc_cmd:	Allocates memory for IOC Init command
2270665484d8SDoug Ambrisko  * input:				Adapter soft state
2271665484d8SDoug Ambrisko  *
2272665484d8SDoug Ambrisko  * Deallocates memory of the IOC Init cmd.
2273665484d8SDoug Ambrisko  */
22748e727371SKashyap D Desai void
22758e727371SKashyap D Desai mrsas_free_ioc_cmd(struct mrsas_softc *sc)
2276665484d8SDoug Ambrisko {
2277665484d8SDoug Ambrisko 	if (sc->ioc_init_phys_mem)
2278665484d8SDoug Ambrisko 		bus_dmamap_unload(sc->ioc_init_tag, sc->ioc_init_dmamap);
2279665484d8SDoug Ambrisko 	if (sc->ioc_init_mem != NULL)
2280665484d8SDoug Ambrisko 		bus_dmamem_free(sc->ioc_init_tag, sc->ioc_init_mem, sc->ioc_init_dmamap);
2281665484d8SDoug Ambrisko 	if (sc->ioc_init_tag != NULL)
2282665484d8SDoug Ambrisko 		bus_dma_tag_destroy(sc->ioc_init_tag);
2283665484d8SDoug Ambrisko }
2284665484d8SDoug Ambrisko 
22858e727371SKashyap D Desai /*
2286665484d8SDoug Ambrisko  * mrsas_ioc_init:	Sends IOC Init command to FW
2287665484d8SDoug Ambrisko  * input:			Adapter soft state
2288665484d8SDoug Ambrisko  *
2289665484d8SDoug Ambrisko  * Issues the IOC Init cmd to FW to initialize the ROC/controller.
2290665484d8SDoug Ambrisko  */
22918e727371SKashyap D Desai int
22928e727371SKashyap D Desai mrsas_ioc_init(struct mrsas_softc *sc)
2293665484d8SDoug Ambrisko {
2294665484d8SDoug Ambrisko 	struct mrsas_init_frame *init_frame;
2295665484d8SDoug Ambrisko 	pMpi2IOCInitRequest_t IOCInitMsg;
2296665484d8SDoug Ambrisko 	MRSAS_REQUEST_DESCRIPTOR_UNION req_desc;
2297665484d8SDoug Ambrisko 	u_int8_t max_wait = MRSAS_IOC_INIT_WAIT_TIME;
2298665484d8SDoug Ambrisko 	bus_addr_t phys_addr;
2299665484d8SDoug Ambrisko 	int i, retcode = 0;
2300665484d8SDoug Ambrisko 
2301665484d8SDoug Ambrisko 	/* Allocate memory for the IOC INIT command */
2302665484d8SDoug Ambrisko 	if (mrsas_alloc_ioc_cmd(sc)) {
2303665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate IOC command.\n");
2304665484d8SDoug Ambrisko 		return (1);
2305665484d8SDoug Ambrisko 	}
2306665484d8SDoug Ambrisko 	IOCInitMsg = (pMpi2IOCInitRequest_t)(((char *)sc->ioc_init_mem) + 1024);
2307665484d8SDoug Ambrisko 	IOCInitMsg->Function = MPI2_FUNCTION_IOC_INIT;
2308665484d8SDoug Ambrisko 	IOCInitMsg->WhoInit = MPI2_WHOINIT_HOST_DRIVER;
2309665484d8SDoug Ambrisko 	IOCInitMsg->MsgVersion = MPI2_VERSION;
2310665484d8SDoug Ambrisko 	IOCInitMsg->HeaderVersion = MPI2_HEADER_VERSION;
2311665484d8SDoug Ambrisko 	IOCInitMsg->SystemRequestFrameSize = MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE / 4;
2312665484d8SDoug Ambrisko 	IOCInitMsg->ReplyDescriptorPostQueueDepth = sc->reply_q_depth;
2313665484d8SDoug Ambrisko 	IOCInitMsg->ReplyDescriptorPostQueueAddress = sc->reply_desc_phys_addr;
2314665484d8SDoug Ambrisko 	IOCInitMsg->SystemRequestFrameBaseAddress = sc->io_request_phys_addr;
2315d18d1b47SKashyap D Desai 	IOCInitMsg->HostMSIxVectors = (sc->msix_vectors > 0 ? sc->msix_vectors : 0);
2316665484d8SDoug Ambrisko 
2317665484d8SDoug Ambrisko 	init_frame = (struct mrsas_init_frame *)sc->ioc_init_mem;
2318665484d8SDoug Ambrisko 	init_frame->cmd = MFI_CMD_INIT;
2319665484d8SDoug Ambrisko 	init_frame->cmd_status = 0xFF;
2320665484d8SDoug Ambrisko 	init_frame->flags |= MFI_FRAME_DONT_POST_IN_REPLY_QUEUE;
2321665484d8SDoug Ambrisko 
2322d18d1b47SKashyap D Desai 	/* driver support Extended MSIX */
2323d18d1b47SKashyap D Desai 	if ((sc->device_id == MRSAS_INVADER) ||
2324d18d1b47SKashyap D Desai 	    (sc->device_id == MRSAS_FURY)) {
2325d18d1b47SKashyap D Desai 		init_frame->driver_operations.
2326d18d1b47SKashyap D Desai 		    mfi_capabilities.support_additional_msix = 1;
2327d18d1b47SKashyap D Desai 	}
2328665484d8SDoug Ambrisko 	if (sc->verbuf_mem) {
2329665484d8SDoug Ambrisko 		snprintf((char *)sc->verbuf_mem, strlen(MRSAS_VERSION) + 2, "%s\n",
2330665484d8SDoug Ambrisko 		    MRSAS_VERSION);
2331665484d8SDoug Ambrisko 		init_frame->driver_ver_lo = (bus_addr_t)sc->verbuf_phys_addr;
2332665484d8SDoug Ambrisko 		init_frame->driver_ver_hi = 0;
2333665484d8SDoug Ambrisko 	}
23344799d485SKashyap D Desai 	init_frame->driver_operations.mfi_capabilities.support_max_255lds = 1;
2335*77cf7df8SKashyap D Desai 	init_frame->driver_operations.mfi_capabilities.security_protocol_cmds_fw = 1;
2336665484d8SDoug Ambrisko 	phys_addr = (bus_addr_t)sc->ioc_init_phys_mem + 1024;
2337665484d8SDoug Ambrisko 	init_frame->queue_info_new_phys_addr_lo = phys_addr;
2338665484d8SDoug Ambrisko 	init_frame->data_xfer_len = sizeof(Mpi2IOCInitRequest_t);
2339665484d8SDoug Ambrisko 
2340665484d8SDoug Ambrisko 	req_desc.addr.Words = (bus_addr_t)sc->ioc_init_phys_mem;
2341665484d8SDoug Ambrisko 	req_desc.MFAIo.RequestFlags =
2342665484d8SDoug Ambrisko 	    (MRSAS_REQ_DESCRIPT_FLAGS_MFA << MRSAS_REQ_DESCRIPT_FLAGS_TYPE_SHIFT);
2343665484d8SDoug Ambrisko 
2344665484d8SDoug Ambrisko 	mrsas_disable_intr(sc);
2345665484d8SDoug Ambrisko 	mrsas_dprint(sc, MRSAS_OCR, "Issuing IOC INIT command to FW.\n");
2346665484d8SDoug Ambrisko 	mrsas_fire_cmd(sc, req_desc.addr.u.low, req_desc.addr.u.high);
2347665484d8SDoug Ambrisko 
2348665484d8SDoug Ambrisko 	/*
2349665484d8SDoug Ambrisko 	 * Poll response timer to wait for Firmware response.  While this
2350665484d8SDoug Ambrisko 	 * timer with the DELAY call could block CPU, the time interval for
2351665484d8SDoug Ambrisko 	 * this is only 1 millisecond.
2352665484d8SDoug Ambrisko 	 */
2353665484d8SDoug Ambrisko 	if (init_frame->cmd_status == 0xFF) {
2354665484d8SDoug Ambrisko 		for (i = 0; i < (max_wait * 1000); i++) {
2355665484d8SDoug Ambrisko 			if (init_frame->cmd_status == 0xFF)
2356665484d8SDoug Ambrisko 				DELAY(1000);
2357665484d8SDoug Ambrisko 			else
2358665484d8SDoug Ambrisko 				break;
2359665484d8SDoug Ambrisko 		}
2360665484d8SDoug Ambrisko 	}
2361665484d8SDoug Ambrisko 	if (init_frame->cmd_status == 0)
2362665484d8SDoug Ambrisko 		mrsas_dprint(sc, MRSAS_OCR,
2363665484d8SDoug Ambrisko 		    "IOC INIT response received from FW.\n");
23648e727371SKashyap D Desai 	else {
2365665484d8SDoug Ambrisko 		if (init_frame->cmd_status == 0xFF)
2366665484d8SDoug Ambrisko 			device_printf(sc->mrsas_dev, "IOC Init timed out after %d seconds.\n", max_wait);
2367665484d8SDoug Ambrisko 		else
2368665484d8SDoug Ambrisko 			device_printf(sc->mrsas_dev, "IOC Init failed, status = 0x%x\n", init_frame->cmd_status);
2369665484d8SDoug Ambrisko 		retcode = 1;
2370665484d8SDoug Ambrisko 	}
2371665484d8SDoug Ambrisko 
2372665484d8SDoug Ambrisko 	mrsas_free_ioc_cmd(sc);
2373665484d8SDoug Ambrisko 	return (retcode);
2374665484d8SDoug Ambrisko }
2375665484d8SDoug Ambrisko 
23768e727371SKashyap D Desai /*
2377665484d8SDoug Ambrisko  * mrsas_alloc_mpt_cmds:	Allocates the command packets
2378665484d8SDoug Ambrisko  * input:					Adapter instance soft state
2379665484d8SDoug Ambrisko  *
2380665484d8SDoug Ambrisko  * This function allocates the internal commands for IOs. Each command that is
23818e727371SKashyap D Desai  * issued to FW is wrapped in a local data structure called mrsas_mpt_cmd. An
23828e727371SKashyap D Desai  * array is allocated with mrsas_mpt_cmd context.  The free commands are
2383665484d8SDoug Ambrisko  * maintained in a linked list (cmd pool). SMID value range is from 1 to
2384665484d8SDoug Ambrisko  * max_fw_cmds.
2385665484d8SDoug Ambrisko  */
23868e727371SKashyap D Desai int
23878e727371SKashyap D Desai mrsas_alloc_mpt_cmds(struct mrsas_softc *sc)
2388665484d8SDoug Ambrisko {
2389665484d8SDoug Ambrisko 	int i, j;
2390d18d1b47SKashyap D Desai 	u_int32_t max_cmd, count;
2391665484d8SDoug Ambrisko 	struct mrsas_mpt_cmd *cmd;
2392665484d8SDoug Ambrisko 	pMpi2ReplyDescriptorsUnion_t reply_desc;
2393665484d8SDoug Ambrisko 	u_int32_t offset, chain_offset, sense_offset;
2394665484d8SDoug Ambrisko 	bus_addr_t io_req_base_phys, chain_frame_base_phys, sense_base_phys;
2395665484d8SDoug Ambrisko 	u_int8_t *io_req_base, *chain_frame_base, *sense_base;
2396665484d8SDoug Ambrisko 
2397665484d8SDoug Ambrisko 	max_cmd = sc->max_fw_cmds;
2398665484d8SDoug Ambrisko 
2399665484d8SDoug Ambrisko 	sc->req_desc = malloc(sc->request_alloc_sz, M_MRSAS, M_NOWAIT);
2400665484d8SDoug Ambrisko 	if (!sc->req_desc) {
2401665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Out of memory, cannot alloc req desc\n");
2402665484d8SDoug Ambrisko 		return (ENOMEM);
2403665484d8SDoug Ambrisko 	}
2404665484d8SDoug Ambrisko 	memset(sc->req_desc, 0, sc->request_alloc_sz);
2405665484d8SDoug Ambrisko 
2406665484d8SDoug Ambrisko 	/*
24078e727371SKashyap D Desai 	 * sc->mpt_cmd_list is an array of struct mrsas_mpt_cmd pointers.
24088e727371SKashyap D Desai 	 * Allocate the dynamic array first and then allocate individual
24098e727371SKashyap D Desai 	 * commands.
2410665484d8SDoug Ambrisko 	 */
2411665484d8SDoug Ambrisko 	sc->mpt_cmd_list = malloc(sizeof(struct mrsas_mpt_cmd *) * max_cmd, M_MRSAS, M_NOWAIT);
2412665484d8SDoug Ambrisko 	if (!sc->mpt_cmd_list) {
2413665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot alloc memory for mpt_cmd_list.\n");
2414665484d8SDoug Ambrisko 		return (ENOMEM);
2415665484d8SDoug Ambrisko 	}
2416665484d8SDoug Ambrisko 	memset(sc->mpt_cmd_list, 0, sizeof(struct mrsas_mpt_cmd *) * max_cmd);
2417665484d8SDoug Ambrisko 	for (i = 0; i < max_cmd; i++) {
2418665484d8SDoug Ambrisko 		sc->mpt_cmd_list[i] = malloc(sizeof(struct mrsas_mpt_cmd),
2419665484d8SDoug Ambrisko 		    M_MRSAS, M_NOWAIT);
2420665484d8SDoug Ambrisko 		if (!sc->mpt_cmd_list[i]) {
2421665484d8SDoug Ambrisko 			for (j = 0; j < i; j++)
2422665484d8SDoug Ambrisko 				free(sc->mpt_cmd_list[j], M_MRSAS);
2423665484d8SDoug Ambrisko 			free(sc->mpt_cmd_list, M_MRSAS);
2424665484d8SDoug Ambrisko 			sc->mpt_cmd_list = NULL;
2425665484d8SDoug Ambrisko 			return (ENOMEM);
2426665484d8SDoug Ambrisko 		}
2427665484d8SDoug Ambrisko 	}
2428665484d8SDoug Ambrisko 
2429665484d8SDoug Ambrisko 	io_req_base = (u_int8_t *)sc->io_request_mem + MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE;
2430665484d8SDoug Ambrisko 	io_req_base_phys = (bus_addr_t)sc->io_request_phys_addr + MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE;
2431665484d8SDoug Ambrisko 	chain_frame_base = (u_int8_t *)sc->chain_frame_mem;
2432665484d8SDoug Ambrisko 	chain_frame_base_phys = (bus_addr_t)sc->chain_frame_phys_addr;
2433665484d8SDoug Ambrisko 	sense_base = (u_int8_t *)sc->sense_mem;
2434665484d8SDoug Ambrisko 	sense_base_phys = (bus_addr_t)sc->sense_phys_addr;
2435665484d8SDoug Ambrisko 	for (i = 0; i < max_cmd; i++) {
2436665484d8SDoug Ambrisko 		cmd = sc->mpt_cmd_list[i];
2437665484d8SDoug Ambrisko 		offset = MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE * i;
2438665484d8SDoug Ambrisko 		chain_offset = 1024 * i;
2439665484d8SDoug Ambrisko 		sense_offset = MRSAS_SENSE_LEN * i;
2440665484d8SDoug Ambrisko 		memset(cmd, 0, sizeof(struct mrsas_mpt_cmd));
2441665484d8SDoug Ambrisko 		cmd->index = i + 1;
2442665484d8SDoug Ambrisko 		cmd->ccb_ptr = NULL;
2443665484d8SDoug Ambrisko 		callout_init(&cmd->cm_callout, 0);
2444665484d8SDoug Ambrisko 		cmd->sync_cmd_idx = (u_int32_t)MRSAS_ULONG_MAX;
2445665484d8SDoug Ambrisko 		cmd->sc = sc;
2446665484d8SDoug Ambrisko 		cmd->io_request = (MRSAS_RAID_SCSI_IO_REQUEST *) (io_req_base + offset);
2447665484d8SDoug Ambrisko 		memset(cmd->io_request, 0, sizeof(MRSAS_RAID_SCSI_IO_REQUEST));
2448665484d8SDoug Ambrisko 		cmd->io_request_phys_addr = io_req_base_phys + offset;
2449665484d8SDoug Ambrisko 		cmd->chain_frame = (MPI2_SGE_IO_UNION *) (chain_frame_base + chain_offset);
2450665484d8SDoug Ambrisko 		cmd->chain_frame_phys_addr = chain_frame_base_phys + chain_offset;
2451665484d8SDoug Ambrisko 		cmd->sense = sense_base + sense_offset;
2452665484d8SDoug Ambrisko 		cmd->sense_phys_addr = sense_base_phys + sense_offset;
2453665484d8SDoug Ambrisko 		if (bus_dmamap_create(sc->data_tag, 0, &cmd->data_dmamap)) {
2454665484d8SDoug Ambrisko 			return (FAIL);
2455665484d8SDoug Ambrisko 		}
2456665484d8SDoug Ambrisko 		TAILQ_INSERT_TAIL(&(sc->mrsas_mpt_cmd_list_head), cmd, next);
2457665484d8SDoug Ambrisko 	}
2458665484d8SDoug Ambrisko 
2459665484d8SDoug Ambrisko 	/* Initialize reply descriptor array to 0xFFFFFFFF */
2460665484d8SDoug Ambrisko 	reply_desc = sc->reply_desc_mem;
2461d18d1b47SKashyap D Desai 	count = sc->msix_vectors > 0 ? sc->msix_vectors : 1;
2462d18d1b47SKashyap D Desai 	for (i = 0; i < sc->reply_q_depth * count; i++, reply_desc++) {
2463665484d8SDoug Ambrisko 		reply_desc->Words = MRSAS_ULONG_MAX;
2464665484d8SDoug Ambrisko 	}
2465665484d8SDoug Ambrisko 	return (0);
2466665484d8SDoug Ambrisko }
2467665484d8SDoug Ambrisko 
24688e727371SKashyap D Desai /*
2469665484d8SDoug Ambrisko  * mrsas_fire_cmd:	Sends command to FW
2470665484d8SDoug Ambrisko  * input:			Adapter softstate
2471665484d8SDoug Ambrisko  * 					request descriptor address low
2472665484d8SDoug Ambrisko  * 					request descriptor address high
2473665484d8SDoug Ambrisko  *
2474665484d8SDoug Ambrisko  * This functions fires the command to Firmware by writing to the
2475665484d8SDoug Ambrisko  * inbound_low_queue_port and inbound_high_queue_port.
2476665484d8SDoug Ambrisko  */
24778e727371SKashyap D Desai void
24788e727371SKashyap D Desai mrsas_fire_cmd(struct mrsas_softc *sc, u_int32_t req_desc_lo,
2479665484d8SDoug Ambrisko     u_int32_t req_desc_hi)
2480665484d8SDoug Ambrisko {
2481665484d8SDoug Ambrisko 	mtx_lock(&sc->pci_lock);
2482665484d8SDoug Ambrisko 	mrsas_write_reg(sc, offsetof(mrsas_reg_set, inbound_low_queue_port),
2483665484d8SDoug Ambrisko 	    req_desc_lo);
2484665484d8SDoug Ambrisko 	mrsas_write_reg(sc, offsetof(mrsas_reg_set, inbound_high_queue_port),
2485665484d8SDoug Ambrisko 	    req_desc_hi);
2486665484d8SDoug Ambrisko 	mtx_unlock(&sc->pci_lock);
2487665484d8SDoug Ambrisko }
2488665484d8SDoug Ambrisko 
24898e727371SKashyap D Desai /*
24908e727371SKashyap D Desai  * mrsas_transition_to_ready:  Move FW to Ready state input:
24918e727371SKashyap D Desai  * Adapter instance soft state
2492665484d8SDoug Ambrisko  *
24938e727371SKashyap D Desai  * During the initialization, FW passes can potentially be in any one of several
24948e727371SKashyap D Desai  * possible states. If the FW in operational, waiting-for-handshake states,
24958e727371SKashyap D Desai  * driver must take steps to bring it to ready state. Otherwise, it has to
24968e727371SKashyap D Desai  * wait for the ready state.
2497665484d8SDoug Ambrisko  */
24988e727371SKashyap D Desai int
24998e727371SKashyap D Desai mrsas_transition_to_ready(struct mrsas_softc *sc, int ocr)
2500665484d8SDoug Ambrisko {
2501665484d8SDoug Ambrisko 	int i;
2502665484d8SDoug Ambrisko 	u_int8_t max_wait;
2503665484d8SDoug Ambrisko 	u_int32_t val, fw_state;
2504665484d8SDoug Ambrisko 	u_int32_t cur_state;
2505665484d8SDoug Ambrisko 	u_int32_t abs_state, curr_abs_state;
2506665484d8SDoug Ambrisko 
2507665484d8SDoug Ambrisko 	val = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad));
2508665484d8SDoug Ambrisko 	fw_state = val & MFI_STATE_MASK;
2509665484d8SDoug Ambrisko 	max_wait = MRSAS_RESET_WAIT_TIME;
2510665484d8SDoug Ambrisko 
2511665484d8SDoug Ambrisko 	if (fw_state != MFI_STATE_READY)
2512665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Waiting for FW to come to ready state\n");
2513665484d8SDoug Ambrisko 
2514665484d8SDoug Ambrisko 	while (fw_state != MFI_STATE_READY) {
2515665484d8SDoug Ambrisko 		abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad));
2516665484d8SDoug Ambrisko 		switch (fw_state) {
2517665484d8SDoug Ambrisko 		case MFI_STATE_FAULT:
2518665484d8SDoug Ambrisko 			device_printf(sc->mrsas_dev, "FW is in FAULT state!!\n");
2519665484d8SDoug Ambrisko 			if (ocr) {
2520665484d8SDoug Ambrisko 				cur_state = MFI_STATE_FAULT;
2521665484d8SDoug Ambrisko 				break;
25228e727371SKashyap D Desai 			} else
2523665484d8SDoug Ambrisko 				return -ENODEV;
2524665484d8SDoug Ambrisko 		case MFI_STATE_WAIT_HANDSHAKE:
2525665484d8SDoug Ambrisko 			/* Set the CLR bit in inbound doorbell */
2526665484d8SDoug Ambrisko 			mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell),
2527665484d8SDoug Ambrisko 			    MFI_INIT_CLEAR_HANDSHAKE | MFI_INIT_HOTPLUG);
2528665484d8SDoug Ambrisko 			cur_state = MFI_STATE_WAIT_HANDSHAKE;
2529665484d8SDoug Ambrisko 			break;
2530665484d8SDoug Ambrisko 		case MFI_STATE_BOOT_MESSAGE_PENDING:
2531665484d8SDoug Ambrisko 			mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell),
2532665484d8SDoug Ambrisko 			    MFI_INIT_HOTPLUG);
2533665484d8SDoug Ambrisko 			cur_state = MFI_STATE_BOOT_MESSAGE_PENDING;
2534665484d8SDoug Ambrisko 			break;
2535665484d8SDoug Ambrisko 		case MFI_STATE_OPERATIONAL:
25368e727371SKashyap D Desai 			/*
25378e727371SKashyap D Desai 			 * Bring it to READY state; assuming max wait 10
25388e727371SKashyap D Desai 			 * secs
25398e727371SKashyap D Desai 			 */
2540665484d8SDoug Ambrisko 			mrsas_disable_intr(sc);
2541665484d8SDoug Ambrisko 			mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell), MFI_RESET_FLAGS);
2542665484d8SDoug Ambrisko 			for (i = 0; i < max_wait * 1000; i++) {
2543665484d8SDoug Ambrisko 				if (mrsas_read_reg(sc, offsetof(mrsas_reg_set, doorbell)) & 1)
2544665484d8SDoug Ambrisko 					DELAY(1000);
2545665484d8SDoug Ambrisko 				else
2546665484d8SDoug Ambrisko 					break;
2547665484d8SDoug Ambrisko 			}
2548665484d8SDoug Ambrisko 			cur_state = MFI_STATE_OPERATIONAL;
2549665484d8SDoug Ambrisko 			break;
2550665484d8SDoug Ambrisko 		case MFI_STATE_UNDEFINED:
25518e727371SKashyap D Desai 			/*
25528e727371SKashyap D Desai 			 * This state should not last for more than 2
25538e727371SKashyap D Desai 			 * seconds
25548e727371SKashyap D Desai 			 */
2555665484d8SDoug Ambrisko 			cur_state = MFI_STATE_UNDEFINED;
2556665484d8SDoug Ambrisko 			break;
2557665484d8SDoug Ambrisko 		case MFI_STATE_BB_INIT:
2558665484d8SDoug Ambrisko 			cur_state = MFI_STATE_BB_INIT;
2559665484d8SDoug Ambrisko 			break;
2560665484d8SDoug Ambrisko 		case MFI_STATE_FW_INIT:
2561665484d8SDoug Ambrisko 			cur_state = MFI_STATE_FW_INIT;
2562665484d8SDoug Ambrisko 			break;
2563665484d8SDoug Ambrisko 		case MFI_STATE_FW_INIT_2:
2564665484d8SDoug Ambrisko 			cur_state = MFI_STATE_FW_INIT_2;
2565665484d8SDoug Ambrisko 			break;
2566665484d8SDoug Ambrisko 		case MFI_STATE_DEVICE_SCAN:
2567665484d8SDoug Ambrisko 			cur_state = MFI_STATE_DEVICE_SCAN;
2568665484d8SDoug Ambrisko 			break;
2569665484d8SDoug Ambrisko 		case MFI_STATE_FLUSH_CACHE:
2570665484d8SDoug Ambrisko 			cur_state = MFI_STATE_FLUSH_CACHE;
2571665484d8SDoug Ambrisko 			break;
2572665484d8SDoug Ambrisko 		default:
2573665484d8SDoug Ambrisko 			device_printf(sc->mrsas_dev, "Unknown state 0x%x\n", fw_state);
2574665484d8SDoug Ambrisko 			return -ENODEV;
2575665484d8SDoug Ambrisko 		}
2576665484d8SDoug Ambrisko 
2577665484d8SDoug Ambrisko 		/*
2578665484d8SDoug Ambrisko 		 * The cur_state should not last for more than max_wait secs
2579665484d8SDoug Ambrisko 		 */
2580665484d8SDoug Ambrisko 		for (i = 0; i < (max_wait * 1000); i++) {
2581665484d8SDoug Ambrisko 			fw_state = (mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2582665484d8SDoug Ambrisko 			    outbound_scratch_pad)) & MFI_STATE_MASK);
2583665484d8SDoug Ambrisko 			curr_abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2584665484d8SDoug Ambrisko 			    outbound_scratch_pad));
2585665484d8SDoug Ambrisko 			if (abs_state == curr_abs_state)
2586665484d8SDoug Ambrisko 				DELAY(1000);
2587665484d8SDoug Ambrisko 			else
2588665484d8SDoug Ambrisko 				break;
2589665484d8SDoug Ambrisko 		}
2590665484d8SDoug Ambrisko 
2591665484d8SDoug Ambrisko 		/*
2592665484d8SDoug Ambrisko 		 * Return error if fw_state hasn't changed after max_wait
2593665484d8SDoug Ambrisko 		 */
2594665484d8SDoug Ambrisko 		if (curr_abs_state == abs_state) {
2595665484d8SDoug Ambrisko 			device_printf(sc->mrsas_dev, "FW state [%d] hasn't changed "
2596665484d8SDoug Ambrisko 			    "in %d secs\n", fw_state, max_wait);
2597665484d8SDoug Ambrisko 			return -ENODEV;
2598665484d8SDoug Ambrisko 		}
2599665484d8SDoug Ambrisko 	}
2600665484d8SDoug Ambrisko 	mrsas_dprint(sc, MRSAS_OCR, "FW now in Ready state\n");
2601665484d8SDoug Ambrisko 	return 0;
2602665484d8SDoug Ambrisko }
2603665484d8SDoug Ambrisko 
26048e727371SKashyap D Desai /*
2605665484d8SDoug Ambrisko  * mrsas_get_mfi_cmd:	Get a cmd from free command pool
2606665484d8SDoug Ambrisko  * input:				Adapter soft state
2607665484d8SDoug Ambrisko  *
2608665484d8SDoug Ambrisko  * This function removes an MFI command from the command list.
2609665484d8SDoug Ambrisko  */
26108e727371SKashyap D Desai struct mrsas_mfi_cmd *
26118e727371SKashyap D Desai mrsas_get_mfi_cmd(struct mrsas_softc *sc)
2612665484d8SDoug Ambrisko {
2613665484d8SDoug Ambrisko 	struct mrsas_mfi_cmd *cmd = NULL;
2614665484d8SDoug Ambrisko 
2615665484d8SDoug Ambrisko 	mtx_lock(&sc->mfi_cmd_pool_lock);
2616665484d8SDoug Ambrisko 	if (!TAILQ_EMPTY(&sc->mrsas_mfi_cmd_list_head)) {
2617665484d8SDoug Ambrisko 		cmd = TAILQ_FIRST(&sc->mrsas_mfi_cmd_list_head);
2618665484d8SDoug Ambrisko 		TAILQ_REMOVE(&sc->mrsas_mfi_cmd_list_head, cmd, next);
2619665484d8SDoug Ambrisko 	}
2620665484d8SDoug Ambrisko 	mtx_unlock(&sc->mfi_cmd_pool_lock);
2621665484d8SDoug Ambrisko 
2622665484d8SDoug Ambrisko 	return cmd;
2623665484d8SDoug Ambrisko }
2624665484d8SDoug Ambrisko 
26258e727371SKashyap D Desai /*
26268e727371SKashyap D Desai  * mrsas_ocr_thread:	Thread to handle OCR/Kill Adapter.
2627665484d8SDoug Ambrisko  * input:				Adapter Context.
2628665484d8SDoug Ambrisko  *
26298e727371SKashyap D Desai  * This function will check FW status register and flag do_timeout_reset flag.
26308e727371SKashyap D Desai  * It will do OCR/Kill adapter if FW is in fault state or IO timed out has
26318e727371SKashyap D Desai  * trigger reset.
2632665484d8SDoug Ambrisko  */
2633665484d8SDoug Ambrisko static void
2634665484d8SDoug Ambrisko mrsas_ocr_thread(void *arg)
2635665484d8SDoug Ambrisko {
2636665484d8SDoug Ambrisko 	struct mrsas_softc *sc;
2637665484d8SDoug Ambrisko 	u_int32_t fw_status, fw_state;
2638665484d8SDoug Ambrisko 
2639665484d8SDoug Ambrisko 	sc = (struct mrsas_softc *)arg;
2640665484d8SDoug Ambrisko 
2641665484d8SDoug Ambrisko 	mrsas_dprint(sc, MRSAS_TRACE, "%s\n", __func__);
2642665484d8SDoug Ambrisko 
2643665484d8SDoug Ambrisko 	sc->ocr_thread_active = 1;
2644665484d8SDoug Ambrisko 	mtx_lock(&sc->sim_lock);
2645665484d8SDoug Ambrisko 	for (;;) {
2646665484d8SDoug Ambrisko 		/* Sleep for 1 second and check the queue status */
2647665484d8SDoug Ambrisko 		msleep(&sc->ocr_chan, &sc->sim_lock, PRIBIO,
2648665484d8SDoug Ambrisko 		    "mrsas_ocr", sc->mrsas_fw_fault_check_delay * hz);
2649665484d8SDoug Ambrisko 		if (sc->remove_in_progress) {
2650665484d8SDoug Ambrisko 			mrsas_dprint(sc, MRSAS_OCR,
2651665484d8SDoug Ambrisko 			    "Exit due to shutdown from %s\n", __func__);
2652665484d8SDoug Ambrisko 			break;
2653665484d8SDoug Ambrisko 		}
2654665484d8SDoug Ambrisko 		fw_status = mrsas_read_reg(sc,
2655665484d8SDoug Ambrisko 		    offsetof(mrsas_reg_set, outbound_scratch_pad));
2656665484d8SDoug Ambrisko 		fw_state = fw_status & MFI_STATE_MASK;
2657665484d8SDoug Ambrisko 		if (fw_state == MFI_STATE_FAULT || sc->do_timedout_reset) {
2658665484d8SDoug Ambrisko 			device_printf(sc->mrsas_dev, "OCR started due to %s!\n",
2659665484d8SDoug Ambrisko 			    sc->do_timedout_reset ? "IO Timeout" :
2660665484d8SDoug Ambrisko 			    "FW fault detected");
2661665484d8SDoug Ambrisko 			mtx_lock_spin(&sc->ioctl_lock);
2662665484d8SDoug Ambrisko 			sc->reset_in_progress = 1;
2663665484d8SDoug Ambrisko 			sc->reset_count++;
2664665484d8SDoug Ambrisko 			mtx_unlock_spin(&sc->ioctl_lock);
2665665484d8SDoug Ambrisko 			mrsas_xpt_freeze(sc);
2666665484d8SDoug Ambrisko 			mrsas_reset_ctrl(sc);
2667665484d8SDoug Ambrisko 			mrsas_xpt_release(sc);
2668665484d8SDoug Ambrisko 			sc->reset_in_progress = 0;
2669665484d8SDoug Ambrisko 			sc->do_timedout_reset = 0;
2670665484d8SDoug Ambrisko 		}
2671665484d8SDoug Ambrisko 	}
2672665484d8SDoug Ambrisko 	mtx_unlock(&sc->sim_lock);
2673665484d8SDoug Ambrisko 	sc->ocr_thread_active = 0;
2674665484d8SDoug Ambrisko 	mrsas_kproc_exit(0);
2675665484d8SDoug Ambrisko }
2676665484d8SDoug Ambrisko 
26778e727371SKashyap D Desai /*
26788e727371SKashyap D Desai  * mrsas_reset_reply_desc:	Reset Reply descriptor as part of OCR.
2679665484d8SDoug Ambrisko  * input:					Adapter Context.
2680665484d8SDoug Ambrisko  *
26818e727371SKashyap D Desai  * This function will clear reply descriptor so that post OCR driver and FW will
26828e727371SKashyap D Desai  * lost old history.
2683665484d8SDoug Ambrisko  */
26848e727371SKashyap D Desai void
26858e727371SKashyap D Desai mrsas_reset_reply_desc(struct mrsas_softc *sc)
2686665484d8SDoug Ambrisko {
2687d18d1b47SKashyap D Desai 	int i, count;
2688665484d8SDoug Ambrisko 	pMpi2ReplyDescriptorsUnion_t reply_desc;
2689665484d8SDoug Ambrisko 
2690d18d1b47SKashyap D Desai 	count = sc->msix_vectors > 0 ? sc->msix_vectors : 1;
2691d18d1b47SKashyap D Desai 	for (i = 0; i < count; i++)
2692d18d1b47SKashyap D Desai 		sc->last_reply_idx[i] = 0;
2693d18d1b47SKashyap D Desai 
2694665484d8SDoug Ambrisko 	reply_desc = sc->reply_desc_mem;
2695665484d8SDoug Ambrisko 	for (i = 0; i < sc->reply_q_depth; i++, reply_desc++) {
2696665484d8SDoug Ambrisko 		reply_desc->Words = MRSAS_ULONG_MAX;
2697665484d8SDoug Ambrisko 	}
2698665484d8SDoug Ambrisko }
2699665484d8SDoug Ambrisko 
27008e727371SKashyap D Desai /*
27018e727371SKashyap D Desai  * mrsas_reset_ctrl:	Core function to OCR/Kill adapter.
2702665484d8SDoug Ambrisko  * input:				Adapter Context.
2703665484d8SDoug Ambrisko  *
27048e727371SKashyap D Desai  * This function will run from thread context so that it can sleep. 1. Do not
27058e727371SKashyap D Desai  * handle OCR if FW is in HW critical error. 2. Wait for outstanding command
27068e727371SKashyap D Desai  * to complete for 180 seconds. 3. If #2 does not find any outstanding
27078e727371SKashyap D Desai  * command Controller is in working state, so skip OCR. Otherwise, do
27088e727371SKashyap D Desai  * OCR/kill Adapter based on flag disableOnlineCtrlReset. 4. Start of the
27098e727371SKashyap D Desai  * OCR, return all SCSI command back to CAM layer which has ccb_ptr. 5. Post
27108e727371SKashyap D Desai  * OCR, Re-fire Managment command and move Controller to Operation state.
2711665484d8SDoug Ambrisko  */
27128e727371SKashyap D Desai int
27138e727371SKashyap D Desai mrsas_reset_ctrl(struct mrsas_softc *sc)
2714665484d8SDoug Ambrisko {
2715665484d8SDoug Ambrisko 	int retval = SUCCESS, i, j, retry = 0;
2716665484d8SDoug Ambrisko 	u_int32_t host_diag, abs_state, status_reg, reset_adapter;
2717665484d8SDoug Ambrisko 	union ccb *ccb;
2718665484d8SDoug Ambrisko 	struct mrsas_mfi_cmd *mfi_cmd;
2719665484d8SDoug Ambrisko 	struct mrsas_mpt_cmd *mpt_cmd;
2720665484d8SDoug Ambrisko 	MRSAS_REQUEST_DESCRIPTOR_UNION *req_desc;
2721665484d8SDoug Ambrisko 
2722665484d8SDoug Ambrisko 	if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) {
2723665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev,
2724665484d8SDoug Ambrisko 		    "mrsas: Hardware critical error, returning FAIL.\n");
2725665484d8SDoug Ambrisko 		return FAIL;
2726665484d8SDoug Ambrisko 	}
2727f5fb2237SKashyap D Desai 	mrsas_set_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags);
2728665484d8SDoug Ambrisko 	sc->adprecovery = MRSAS_ADPRESET_SM_INFAULT;
2729665484d8SDoug Ambrisko 	mrsas_disable_intr(sc);
2730665484d8SDoug Ambrisko 	DELAY(1000 * 1000);
2731665484d8SDoug Ambrisko 
2732665484d8SDoug Ambrisko 	/* First try waiting for commands to complete */
2733665484d8SDoug Ambrisko 	if (mrsas_wait_for_outstanding(sc)) {
2734665484d8SDoug Ambrisko 		mrsas_dprint(sc, MRSAS_OCR,
2735665484d8SDoug Ambrisko 		    "resetting adapter from %s.\n",
2736665484d8SDoug Ambrisko 		    __func__);
2737665484d8SDoug Ambrisko 		/* Now return commands back to the CAM layer */
2738665484d8SDoug Ambrisko 		for (i = 0; i < sc->max_fw_cmds; i++) {
2739665484d8SDoug Ambrisko 			mpt_cmd = sc->mpt_cmd_list[i];
2740665484d8SDoug Ambrisko 			if (mpt_cmd->ccb_ptr) {
2741665484d8SDoug Ambrisko 				ccb = (union ccb *)(mpt_cmd->ccb_ptr);
2742665484d8SDoug Ambrisko 				ccb->ccb_h.status = CAM_SCSI_BUS_RESET;
2743665484d8SDoug Ambrisko 				mrsas_cmd_done(sc, mpt_cmd);
2744f5fb2237SKashyap D Desai 				mrsas_atomic_dec(&sc->fw_outstanding);
2745665484d8SDoug Ambrisko 			}
2746665484d8SDoug Ambrisko 		}
2747665484d8SDoug Ambrisko 
2748665484d8SDoug Ambrisko 		status_reg = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2749665484d8SDoug Ambrisko 		    outbound_scratch_pad));
2750665484d8SDoug Ambrisko 		abs_state = status_reg & MFI_STATE_MASK;
2751665484d8SDoug Ambrisko 		reset_adapter = status_reg & MFI_RESET_ADAPTER;
2752665484d8SDoug Ambrisko 		if (sc->disableOnlineCtrlReset ||
2753665484d8SDoug Ambrisko 		    (abs_state == MFI_STATE_FAULT && !reset_adapter)) {
2754665484d8SDoug Ambrisko 			/* Reset not supported, kill adapter */
2755665484d8SDoug Ambrisko 			mrsas_dprint(sc, MRSAS_OCR, "Reset not supported, killing adapter.\n");
2756665484d8SDoug Ambrisko 			mrsas_kill_hba(sc);
2757665484d8SDoug Ambrisko 			sc->adprecovery = MRSAS_HW_CRITICAL_ERROR;
2758665484d8SDoug Ambrisko 			retval = FAIL;
2759665484d8SDoug Ambrisko 			goto out;
2760665484d8SDoug Ambrisko 		}
2761665484d8SDoug Ambrisko 		/* Now try to reset the chip */
2762665484d8SDoug Ambrisko 		for (i = 0; i < MRSAS_FUSION_MAX_RESET_TRIES; i++) {
2763665484d8SDoug Ambrisko 			mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset),
2764665484d8SDoug Ambrisko 			    MPI2_WRSEQ_FLUSH_KEY_VALUE);
2765665484d8SDoug Ambrisko 			mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset),
2766665484d8SDoug Ambrisko 			    MPI2_WRSEQ_1ST_KEY_VALUE);
2767665484d8SDoug Ambrisko 			mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset),
2768665484d8SDoug Ambrisko 			    MPI2_WRSEQ_2ND_KEY_VALUE);
2769665484d8SDoug Ambrisko 			mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset),
2770665484d8SDoug Ambrisko 			    MPI2_WRSEQ_3RD_KEY_VALUE);
2771665484d8SDoug Ambrisko 			mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset),
2772665484d8SDoug Ambrisko 			    MPI2_WRSEQ_4TH_KEY_VALUE);
2773665484d8SDoug Ambrisko 			mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset),
2774665484d8SDoug Ambrisko 			    MPI2_WRSEQ_5TH_KEY_VALUE);
2775665484d8SDoug Ambrisko 			mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset),
2776665484d8SDoug Ambrisko 			    MPI2_WRSEQ_6TH_KEY_VALUE);
2777665484d8SDoug Ambrisko 
2778665484d8SDoug Ambrisko 			/* Check that the diag write enable (DRWE) bit is on */
2779665484d8SDoug Ambrisko 			host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2780665484d8SDoug Ambrisko 			    fusion_host_diag));
2781665484d8SDoug Ambrisko 			retry = 0;
2782665484d8SDoug Ambrisko 			while (!(host_diag & HOST_DIAG_WRITE_ENABLE)) {
2783665484d8SDoug Ambrisko 				DELAY(100 * 1000);
2784665484d8SDoug Ambrisko 				host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2785665484d8SDoug Ambrisko 				    fusion_host_diag));
2786665484d8SDoug Ambrisko 				if (retry++ == 100) {
2787665484d8SDoug Ambrisko 					mrsas_dprint(sc, MRSAS_OCR,
2788665484d8SDoug Ambrisko 					    "Host diag unlock failed!\n");
2789665484d8SDoug Ambrisko 					break;
2790665484d8SDoug Ambrisko 				}
2791665484d8SDoug Ambrisko 			}
2792665484d8SDoug Ambrisko 			if (!(host_diag & HOST_DIAG_WRITE_ENABLE))
2793665484d8SDoug Ambrisko 				continue;
2794665484d8SDoug Ambrisko 
2795665484d8SDoug Ambrisko 			/* Send chip reset command */
2796665484d8SDoug Ambrisko 			mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_host_diag),
2797665484d8SDoug Ambrisko 			    host_diag | HOST_DIAG_RESET_ADAPTER);
2798665484d8SDoug Ambrisko 			DELAY(3000 * 1000);
2799665484d8SDoug Ambrisko 
2800665484d8SDoug Ambrisko 			/* Make sure reset adapter bit is cleared */
2801665484d8SDoug Ambrisko 			host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2802665484d8SDoug Ambrisko 			    fusion_host_diag));
2803665484d8SDoug Ambrisko 			retry = 0;
2804665484d8SDoug Ambrisko 			while (host_diag & HOST_DIAG_RESET_ADAPTER) {
2805665484d8SDoug Ambrisko 				DELAY(100 * 1000);
2806665484d8SDoug Ambrisko 				host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2807665484d8SDoug Ambrisko 				    fusion_host_diag));
2808665484d8SDoug Ambrisko 				if (retry++ == 1000) {
2809665484d8SDoug Ambrisko 					mrsas_dprint(sc, MRSAS_OCR,
2810665484d8SDoug Ambrisko 					    "Diag reset adapter never cleared!\n");
2811665484d8SDoug Ambrisko 					break;
2812665484d8SDoug Ambrisko 				}
2813665484d8SDoug Ambrisko 			}
2814665484d8SDoug Ambrisko 			if (host_diag & HOST_DIAG_RESET_ADAPTER)
2815665484d8SDoug Ambrisko 				continue;
2816665484d8SDoug Ambrisko 
2817665484d8SDoug Ambrisko 			abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2818665484d8SDoug Ambrisko 			    outbound_scratch_pad)) & MFI_STATE_MASK;
2819665484d8SDoug Ambrisko 			retry = 0;
2820665484d8SDoug Ambrisko 
2821665484d8SDoug Ambrisko 			while ((abs_state <= MFI_STATE_FW_INIT) && (retry++ < 1000)) {
2822665484d8SDoug Ambrisko 				DELAY(100 * 1000);
2823665484d8SDoug Ambrisko 				abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2824665484d8SDoug Ambrisko 				    outbound_scratch_pad)) & MFI_STATE_MASK;
2825665484d8SDoug Ambrisko 			}
2826665484d8SDoug Ambrisko 			if (abs_state <= MFI_STATE_FW_INIT) {
2827665484d8SDoug Ambrisko 				mrsas_dprint(sc, MRSAS_OCR, "firmware state < MFI_STATE_FW_INIT,"
2828665484d8SDoug Ambrisko 				    " state = 0x%x\n", abs_state);
2829665484d8SDoug Ambrisko 				continue;
2830665484d8SDoug Ambrisko 			}
2831665484d8SDoug Ambrisko 			/* Wait for FW to become ready */
2832665484d8SDoug Ambrisko 			if (mrsas_transition_to_ready(sc, 1)) {
2833665484d8SDoug Ambrisko 				mrsas_dprint(sc, MRSAS_OCR,
2834665484d8SDoug Ambrisko 				    "mrsas: Failed to transition controller to ready.\n");
2835665484d8SDoug Ambrisko 				continue;
2836665484d8SDoug Ambrisko 			}
2837665484d8SDoug Ambrisko 			mrsas_reset_reply_desc(sc);
2838665484d8SDoug Ambrisko 			if (mrsas_ioc_init(sc)) {
2839665484d8SDoug Ambrisko 				mrsas_dprint(sc, MRSAS_OCR, "mrsas_ioc_init() failed!\n");
2840665484d8SDoug Ambrisko 				continue;
2841665484d8SDoug Ambrisko 			}
2842f5fb2237SKashyap D Desai 			mrsas_clear_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags);
2843665484d8SDoug Ambrisko 			mrsas_enable_intr(sc);
2844665484d8SDoug Ambrisko 			sc->adprecovery = MRSAS_HBA_OPERATIONAL;
2845665484d8SDoug Ambrisko 
2846665484d8SDoug Ambrisko 			/* Re-fire management commands */
2847665484d8SDoug Ambrisko 			for (j = 0; j < sc->max_fw_cmds; j++) {
2848665484d8SDoug Ambrisko 				mpt_cmd = sc->mpt_cmd_list[j];
2849665484d8SDoug Ambrisko 				if (mpt_cmd->sync_cmd_idx != (u_int32_t)MRSAS_ULONG_MAX) {
2850665484d8SDoug Ambrisko 					mfi_cmd = sc->mfi_cmd_list[mpt_cmd->sync_cmd_idx];
2851665484d8SDoug Ambrisko 					if (mfi_cmd->frame->dcmd.opcode ==
2852665484d8SDoug Ambrisko 					    MR_DCMD_LD_MAP_GET_INFO) {
2853665484d8SDoug Ambrisko 						mrsas_release_mfi_cmd(mfi_cmd);
2854665484d8SDoug Ambrisko 						mrsas_release_mpt_cmd(mpt_cmd);
2855665484d8SDoug Ambrisko 					} else {
2856665484d8SDoug Ambrisko 						req_desc = mrsas_get_request_desc(sc,
2857665484d8SDoug Ambrisko 						    mfi_cmd->cmd_id.context.smid - 1);
2858665484d8SDoug Ambrisko 						mrsas_dprint(sc, MRSAS_OCR,
2859665484d8SDoug Ambrisko 						    "Re-fire command DCMD opcode 0x%x index %d\n ",
2860665484d8SDoug Ambrisko 						    mfi_cmd->frame->dcmd.opcode, j);
2861665484d8SDoug Ambrisko 						if (!req_desc)
2862665484d8SDoug Ambrisko 							device_printf(sc->mrsas_dev,
2863665484d8SDoug Ambrisko 							    "Cannot build MPT cmd.\n");
2864665484d8SDoug Ambrisko 						else
2865665484d8SDoug Ambrisko 							mrsas_fire_cmd(sc, req_desc->addr.u.low,
2866665484d8SDoug Ambrisko 							    req_desc->addr.u.high);
2867665484d8SDoug Ambrisko 					}
2868665484d8SDoug Ambrisko 				}
2869665484d8SDoug Ambrisko 			}
2870665484d8SDoug Ambrisko 
2871665484d8SDoug Ambrisko 			/* Reset load balance info */
2872665484d8SDoug Ambrisko 			memset(sc->load_balance_info, 0,
28734799d485SKashyap D Desai 			    sizeof(LD_LOAD_BALANCE_INFO) * MAX_LOGICAL_DRIVES_EXT);
2874665484d8SDoug Ambrisko 
2875665484d8SDoug Ambrisko 			if (!mrsas_get_map_info(sc))
2876665484d8SDoug Ambrisko 				mrsas_sync_map_info(sc);
2877665484d8SDoug Ambrisko 
2878665484d8SDoug Ambrisko 			/* Adapter reset completed successfully */
2879665484d8SDoug Ambrisko 			device_printf(sc->mrsas_dev, "Reset successful\n");
2880665484d8SDoug Ambrisko 			retval = SUCCESS;
2881665484d8SDoug Ambrisko 			goto out;
2882665484d8SDoug Ambrisko 		}
2883665484d8SDoug Ambrisko 		/* Reset failed, kill the adapter */
2884665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Reset failed, killing adapter.\n");
2885665484d8SDoug Ambrisko 		mrsas_kill_hba(sc);
2886665484d8SDoug Ambrisko 		retval = FAIL;
2887665484d8SDoug Ambrisko 	} else {
2888f5fb2237SKashyap D Desai 		mrsas_clear_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags);
2889665484d8SDoug Ambrisko 		mrsas_enable_intr(sc);
2890665484d8SDoug Ambrisko 		sc->adprecovery = MRSAS_HBA_OPERATIONAL;
2891665484d8SDoug Ambrisko 	}
2892665484d8SDoug Ambrisko out:
2893f5fb2237SKashyap D Desai 	mrsas_clear_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags);
2894665484d8SDoug Ambrisko 	mrsas_dprint(sc, MRSAS_OCR,
2895665484d8SDoug Ambrisko 	    "Reset Exit with %d.\n", retval);
2896665484d8SDoug Ambrisko 	return retval;
2897665484d8SDoug Ambrisko }
2898665484d8SDoug Ambrisko 
28998e727371SKashyap D Desai /*
29008e727371SKashyap D Desai  * mrsas_kill_hba:	Kill HBA when OCR is not supported
2901665484d8SDoug Ambrisko  * input:			Adapter Context.
2902665484d8SDoug Ambrisko  *
2903665484d8SDoug Ambrisko  * This function will kill HBA when OCR is not supported.
2904665484d8SDoug Ambrisko  */
29058e727371SKashyap D Desai void
29068e727371SKashyap D Desai mrsas_kill_hba(struct mrsas_softc *sc)
2907665484d8SDoug Ambrisko {
2908665484d8SDoug Ambrisko 	mrsas_dprint(sc, MRSAS_OCR, "%s\n", __func__);
2909665484d8SDoug Ambrisko 	mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell),
2910665484d8SDoug Ambrisko 	    MFI_STOP_ADP);
2911665484d8SDoug Ambrisko 	/* Flush */
2912665484d8SDoug Ambrisko 	mrsas_read_reg(sc, offsetof(mrsas_reg_set, doorbell));
2913665484d8SDoug Ambrisko }
2914665484d8SDoug Ambrisko 
29158e727371SKashyap D Desai /*
29168e727371SKashyap D Desai  * mrsas_wait_for_outstanding:	Wait for outstanding commands
2917665484d8SDoug Ambrisko  * input:						Adapter Context.
2918665484d8SDoug Ambrisko  *
29198e727371SKashyap D Desai  * This function will wait for 180 seconds for outstanding commands to be
29208e727371SKashyap D Desai  * completed.
2921665484d8SDoug Ambrisko  */
29228e727371SKashyap D Desai int
29238e727371SKashyap D Desai mrsas_wait_for_outstanding(struct mrsas_softc *sc)
2924665484d8SDoug Ambrisko {
2925665484d8SDoug Ambrisko 	int i, outstanding, retval = 0;
2926d18d1b47SKashyap D Desai 	u_int32_t fw_state, count, MSIxIndex;
2927d18d1b47SKashyap D Desai 
2928665484d8SDoug Ambrisko 
2929665484d8SDoug Ambrisko 	for (i = 0; i < MRSAS_RESET_WAIT_TIME; i++) {
2930665484d8SDoug Ambrisko 		if (sc->remove_in_progress) {
2931665484d8SDoug Ambrisko 			mrsas_dprint(sc, MRSAS_OCR,
2932665484d8SDoug Ambrisko 			    "Driver remove or shutdown called.\n");
2933665484d8SDoug Ambrisko 			retval = 1;
2934665484d8SDoug Ambrisko 			goto out;
2935665484d8SDoug Ambrisko 		}
2936665484d8SDoug Ambrisko 		/* Check if firmware is in fault state */
2937665484d8SDoug Ambrisko 		fw_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2938665484d8SDoug Ambrisko 		    outbound_scratch_pad)) & MFI_STATE_MASK;
2939665484d8SDoug Ambrisko 		if (fw_state == MFI_STATE_FAULT) {
2940665484d8SDoug Ambrisko 			mrsas_dprint(sc, MRSAS_OCR,
2941665484d8SDoug Ambrisko 			    "Found FW in FAULT state, will reset adapter.\n");
2942665484d8SDoug Ambrisko 			retval = 1;
2943665484d8SDoug Ambrisko 			goto out;
2944665484d8SDoug Ambrisko 		}
2945f5fb2237SKashyap D Desai 		outstanding = mrsas_atomic_read(&sc->fw_outstanding);
2946665484d8SDoug Ambrisko 		if (!outstanding)
2947665484d8SDoug Ambrisko 			goto out;
2948665484d8SDoug Ambrisko 
2949665484d8SDoug Ambrisko 		if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) {
2950665484d8SDoug Ambrisko 			mrsas_dprint(sc, MRSAS_OCR, "[%2d]waiting for %d "
2951665484d8SDoug Ambrisko 			    "commands to complete\n", i, outstanding);
2952d18d1b47SKashyap D Desai 			count = sc->msix_vectors > 0 ? sc->msix_vectors : 1;
2953d18d1b47SKashyap D Desai 			for (MSIxIndex = 0; MSIxIndex < count; MSIxIndex++)
2954d18d1b47SKashyap D Desai 				mrsas_complete_cmd(sc, MSIxIndex);
2955665484d8SDoug Ambrisko 		}
2956665484d8SDoug Ambrisko 		DELAY(1000 * 1000);
2957665484d8SDoug Ambrisko 	}
2958665484d8SDoug Ambrisko 
2959f5fb2237SKashyap D Desai 	if (mrsas_atomic_read(&sc->fw_outstanding)) {
2960665484d8SDoug Ambrisko 		mrsas_dprint(sc, MRSAS_OCR,
2961665484d8SDoug Ambrisko 		    " pending commands remain after waiting,"
2962665484d8SDoug Ambrisko 		    " will reset adapter.\n");
2963665484d8SDoug Ambrisko 		retval = 1;
2964665484d8SDoug Ambrisko 	}
2965665484d8SDoug Ambrisko out:
2966665484d8SDoug Ambrisko 	return retval;
2967665484d8SDoug Ambrisko }
2968665484d8SDoug Ambrisko 
29698e727371SKashyap D Desai /*
2970665484d8SDoug Ambrisko  * mrsas_release_mfi_cmd:	Return a cmd to free command pool
2971665484d8SDoug Ambrisko  * input:					Command packet for return to free cmd pool
2972665484d8SDoug Ambrisko  *
2973665484d8SDoug Ambrisko  * This function returns the MFI command to the command list.
2974665484d8SDoug Ambrisko  */
29758e727371SKashyap D Desai void
29768e727371SKashyap D Desai mrsas_release_mfi_cmd(struct mrsas_mfi_cmd *cmd)
2977665484d8SDoug Ambrisko {
2978665484d8SDoug Ambrisko 	struct mrsas_softc *sc = cmd->sc;
2979665484d8SDoug Ambrisko 
2980665484d8SDoug Ambrisko 	mtx_lock(&sc->mfi_cmd_pool_lock);
2981665484d8SDoug Ambrisko 	cmd->ccb_ptr = NULL;
2982665484d8SDoug Ambrisko 	cmd->cmd_id.frame_count = 0;
2983665484d8SDoug Ambrisko 	TAILQ_INSERT_TAIL(&(sc->mrsas_mfi_cmd_list_head), cmd, next);
2984665484d8SDoug Ambrisko 	mtx_unlock(&sc->mfi_cmd_pool_lock);
2985665484d8SDoug Ambrisko 
2986665484d8SDoug Ambrisko 	return;
2987665484d8SDoug Ambrisko }
2988665484d8SDoug Ambrisko 
29898e727371SKashyap D Desai /*
29908e727371SKashyap D Desai  * mrsas_get_controller_info:	Returns FW's controller structure
2991665484d8SDoug Ambrisko  * input:						Adapter soft state
2992665484d8SDoug Ambrisko  * 								Controller information structure
2993665484d8SDoug Ambrisko  *
29948e727371SKashyap D Desai  * Issues an internal command (DCMD) to get the FW's controller structure. This
29958e727371SKashyap D Desai  * information is mainly used to find out the maximum IO transfer per command
29968e727371SKashyap D Desai  * supported by the FW.
2997665484d8SDoug Ambrisko  */
29988e727371SKashyap D Desai static int
29998e727371SKashyap D Desai mrsas_get_ctrl_info(struct mrsas_softc *sc,
3000665484d8SDoug Ambrisko     struct mrsas_ctrl_info *ctrl_info)
3001665484d8SDoug Ambrisko {
3002665484d8SDoug Ambrisko 	int retcode = 0;
3003665484d8SDoug Ambrisko 	struct mrsas_mfi_cmd *cmd;
3004665484d8SDoug Ambrisko 	struct mrsas_dcmd_frame *dcmd;
3005665484d8SDoug Ambrisko 
3006665484d8SDoug Ambrisko 	cmd = mrsas_get_mfi_cmd(sc);
3007665484d8SDoug Ambrisko 
3008665484d8SDoug Ambrisko 	if (!cmd) {
3009665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Failed to get a free cmd\n");
3010665484d8SDoug Ambrisko 		return -ENOMEM;
3011665484d8SDoug Ambrisko 	}
3012665484d8SDoug Ambrisko 	dcmd = &cmd->frame->dcmd;
3013665484d8SDoug Ambrisko 
3014665484d8SDoug Ambrisko 	if (mrsas_alloc_ctlr_info_cmd(sc) != SUCCESS) {
3015665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate get ctlr info cmd\n");
3016665484d8SDoug Ambrisko 		mrsas_release_mfi_cmd(cmd);
3017665484d8SDoug Ambrisko 		return -ENOMEM;
3018665484d8SDoug Ambrisko 	}
3019665484d8SDoug Ambrisko 	memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE);
3020665484d8SDoug Ambrisko 
3021665484d8SDoug Ambrisko 	dcmd->cmd = MFI_CMD_DCMD;
3022665484d8SDoug Ambrisko 	dcmd->cmd_status = 0xFF;
3023665484d8SDoug Ambrisko 	dcmd->sge_count = 1;
3024665484d8SDoug Ambrisko 	dcmd->flags = MFI_FRAME_DIR_READ;
3025665484d8SDoug Ambrisko 	dcmd->timeout = 0;
3026665484d8SDoug Ambrisko 	dcmd->pad_0 = 0;
3027665484d8SDoug Ambrisko 	dcmd->data_xfer_len = sizeof(struct mrsas_ctrl_info);
3028665484d8SDoug Ambrisko 	dcmd->opcode = MR_DCMD_CTRL_GET_INFO;
3029665484d8SDoug Ambrisko 	dcmd->sgl.sge32[0].phys_addr = sc->ctlr_info_phys_addr;
3030665484d8SDoug Ambrisko 	dcmd->sgl.sge32[0].length = sizeof(struct mrsas_ctrl_info);
3031665484d8SDoug Ambrisko 
3032665484d8SDoug Ambrisko 	if (!mrsas_issue_polled(sc, cmd))
3033665484d8SDoug Ambrisko 		memcpy(ctrl_info, sc->ctlr_info_mem, sizeof(struct mrsas_ctrl_info));
3034665484d8SDoug Ambrisko 	else
3035665484d8SDoug Ambrisko 		retcode = 1;
3036665484d8SDoug Ambrisko 
3037665484d8SDoug Ambrisko 	mrsas_free_ctlr_info_cmd(sc);
3038665484d8SDoug Ambrisko 	mrsas_release_mfi_cmd(cmd);
3039665484d8SDoug Ambrisko 	return (retcode);
3040665484d8SDoug Ambrisko }
3041665484d8SDoug Ambrisko 
30428e727371SKashyap D Desai /*
3043665484d8SDoug Ambrisko  * mrsas_alloc_ctlr_info_cmd:	Allocates memory for controller info command
3044665484d8SDoug Ambrisko  * input:						Adapter soft state
3045665484d8SDoug Ambrisko  *
3046665484d8SDoug Ambrisko  * Allocates DMAable memory for the controller info internal command.
3047665484d8SDoug Ambrisko  */
30488e727371SKashyap D Desai int
30498e727371SKashyap D Desai mrsas_alloc_ctlr_info_cmd(struct mrsas_softc *sc)
3050665484d8SDoug Ambrisko {
3051665484d8SDoug Ambrisko 	int ctlr_info_size;
3052665484d8SDoug Ambrisko 
3053665484d8SDoug Ambrisko 	/* Allocate get controller info command */
3054665484d8SDoug Ambrisko 	ctlr_info_size = sizeof(struct mrsas_ctrl_info);
30558e727371SKashyap D Desai 	if (bus_dma_tag_create(sc->mrsas_parent_tag,
30568e727371SKashyap D Desai 	    1, 0,
30578e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR_32BIT,
30588e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR,
30598e727371SKashyap D Desai 	    NULL, NULL,
30608e727371SKashyap D Desai 	    ctlr_info_size,
30618e727371SKashyap D Desai 	    1,
30628e727371SKashyap D Desai 	    ctlr_info_size,
30638e727371SKashyap D Desai 	    BUS_DMA_ALLOCNOW,
30648e727371SKashyap D Desai 	    NULL, NULL,
3065665484d8SDoug Ambrisko 	    &sc->ctlr_info_tag)) {
3066665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate ctlr info tag\n");
3067665484d8SDoug Ambrisko 		return (ENOMEM);
3068665484d8SDoug Ambrisko 	}
3069665484d8SDoug Ambrisko 	if (bus_dmamem_alloc(sc->ctlr_info_tag, (void **)&sc->ctlr_info_mem,
3070665484d8SDoug Ambrisko 	    BUS_DMA_NOWAIT, &sc->ctlr_info_dmamap)) {
3071665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate ctlr info cmd mem\n");
3072665484d8SDoug Ambrisko 		return (ENOMEM);
3073665484d8SDoug Ambrisko 	}
3074665484d8SDoug Ambrisko 	if (bus_dmamap_load(sc->ctlr_info_tag, sc->ctlr_info_dmamap,
3075665484d8SDoug Ambrisko 	    sc->ctlr_info_mem, ctlr_info_size, mrsas_addr_cb,
3076665484d8SDoug Ambrisko 	    &sc->ctlr_info_phys_addr, BUS_DMA_NOWAIT)) {
3077665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot load ctlr info cmd mem\n");
3078665484d8SDoug Ambrisko 		return (ENOMEM);
3079665484d8SDoug Ambrisko 	}
3080665484d8SDoug Ambrisko 	memset(sc->ctlr_info_mem, 0, ctlr_info_size);
3081665484d8SDoug Ambrisko 	return (0);
3082665484d8SDoug Ambrisko }
3083665484d8SDoug Ambrisko 
30848e727371SKashyap D Desai /*
3085665484d8SDoug Ambrisko  * mrsas_free_ctlr_info_cmd:	Free memory for controller info command
3086665484d8SDoug Ambrisko  * input:						Adapter soft state
3087665484d8SDoug Ambrisko  *
3088665484d8SDoug Ambrisko  * Deallocates memory of the get controller info cmd.
3089665484d8SDoug Ambrisko  */
30908e727371SKashyap D Desai void
30918e727371SKashyap D Desai mrsas_free_ctlr_info_cmd(struct mrsas_softc *sc)
3092665484d8SDoug Ambrisko {
3093665484d8SDoug Ambrisko 	if (sc->ctlr_info_phys_addr)
3094665484d8SDoug Ambrisko 		bus_dmamap_unload(sc->ctlr_info_tag, sc->ctlr_info_dmamap);
3095665484d8SDoug Ambrisko 	if (sc->ctlr_info_mem != NULL)
3096665484d8SDoug Ambrisko 		bus_dmamem_free(sc->ctlr_info_tag, sc->ctlr_info_mem, sc->ctlr_info_dmamap);
3097665484d8SDoug Ambrisko 	if (sc->ctlr_info_tag != NULL)
3098665484d8SDoug Ambrisko 		bus_dma_tag_destroy(sc->ctlr_info_tag);
3099665484d8SDoug Ambrisko }
3100665484d8SDoug Ambrisko 
31018e727371SKashyap D Desai /*
3102665484d8SDoug Ambrisko  * mrsas_issue_polled:	Issues a polling command
3103665484d8SDoug Ambrisko  * inputs:				Adapter soft state
3104665484d8SDoug Ambrisko  * 						Command packet to be issued
3105665484d8SDoug Ambrisko  *
31068e727371SKashyap D Desai  * This function is for posting of internal commands to Firmware.  MFI requires
31078e727371SKashyap D Desai  * the cmd_status to be set to 0xFF before posting.  The maximun wait time of
31088e727371SKashyap D Desai  * the poll response timer is 180 seconds.
3109665484d8SDoug Ambrisko  */
31108e727371SKashyap D Desai int
31118e727371SKashyap D Desai mrsas_issue_polled(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd)
3112665484d8SDoug Ambrisko {
3113665484d8SDoug Ambrisko 	struct mrsas_header *frame_hdr = &cmd->frame->hdr;
3114665484d8SDoug Ambrisko 	u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME;
3115665484d8SDoug Ambrisko 	int i, retcode = 0;
3116665484d8SDoug Ambrisko 
3117665484d8SDoug Ambrisko 	frame_hdr->cmd_status = 0xFF;
3118665484d8SDoug Ambrisko 	frame_hdr->flags |= MFI_FRAME_DONT_POST_IN_REPLY_QUEUE;
3119665484d8SDoug Ambrisko 
3120665484d8SDoug Ambrisko 	/* Issue the frame using inbound queue port */
3121665484d8SDoug Ambrisko 	if (mrsas_issue_dcmd(sc, cmd)) {
3122665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot issue DCMD internal command.\n");
3123665484d8SDoug Ambrisko 		return (1);
3124665484d8SDoug Ambrisko 	}
3125665484d8SDoug Ambrisko 	/*
3126665484d8SDoug Ambrisko 	 * Poll response timer to wait for Firmware response.  While this
3127665484d8SDoug Ambrisko 	 * timer with the DELAY call could block CPU, the time interval for
3128665484d8SDoug Ambrisko 	 * this is only 1 millisecond.
3129665484d8SDoug Ambrisko 	 */
3130665484d8SDoug Ambrisko 	if (frame_hdr->cmd_status == 0xFF) {
3131665484d8SDoug Ambrisko 		for (i = 0; i < (max_wait * 1000); i++) {
3132665484d8SDoug Ambrisko 			if (frame_hdr->cmd_status == 0xFF)
3133665484d8SDoug Ambrisko 				DELAY(1000);
3134665484d8SDoug Ambrisko 			else
3135665484d8SDoug Ambrisko 				break;
3136665484d8SDoug Ambrisko 		}
3137665484d8SDoug Ambrisko 	}
31388e727371SKashyap D Desai 	if (frame_hdr->cmd_status != 0) {
3139665484d8SDoug Ambrisko 		if (frame_hdr->cmd_status == 0xFF)
3140665484d8SDoug Ambrisko 			device_printf(sc->mrsas_dev, "DCMD timed out after %d seconds.\n", max_wait);
3141665484d8SDoug Ambrisko 		else
3142665484d8SDoug Ambrisko 			device_printf(sc->mrsas_dev, "DCMD failed, status = 0x%x\n", frame_hdr->cmd_status);
3143665484d8SDoug Ambrisko 		retcode = 1;
3144665484d8SDoug Ambrisko 	}
3145665484d8SDoug Ambrisko 	return (retcode);
3146665484d8SDoug Ambrisko }
3147665484d8SDoug Ambrisko 
31488e727371SKashyap D Desai /*
31498e727371SKashyap D Desai  * mrsas_issue_dcmd:	Issues a MFI Pass thru cmd
31508e727371SKashyap D Desai  * input:				Adapter soft state mfi cmd pointer
3151665484d8SDoug Ambrisko  *
3152665484d8SDoug Ambrisko  * This function is called by mrsas_issued_blocked_cmd() and
31538e727371SKashyap D Desai  * mrsas_issued_polled(), to build the MPT command and then fire the command
31548e727371SKashyap D Desai  * to Firmware.
3155665484d8SDoug Ambrisko  */
3156665484d8SDoug Ambrisko int
3157665484d8SDoug Ambrisko mrsas_issue_dcmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd)
3158665484d8SDoug Ambrisko {
3159665484d8SDoug Ambrisko 	MRSAS_REQUEST_DESCRIPTOR_UNION *req_desc;
3160665484d8SDoug Ambrisko 
3161665484d8SDoug Ambrisko 	req_desc = mrsas_build_mpt_cmd(sc, cmd);
3162665484d8SDoug Ambrisko 	if (!req_desc) {
3163665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot build MPT cmd.\n");
3164665484d8SDoug Ambrisko 		return (1);
3165665484d8SDoug Ambrisko 	}
3166665484d8SDoug Ambrisko 	mrsas_fire_cmd(sc, req_desc->addr.u.low, req_desc->addr.u.high);
3167665484d8SDoug Ambrisko 
3168665484d8SDoug Ambrisko 	return (0);
3169665484d8SDoug Ambrisko }
3170665484d8SDoug Ambrisko 
31718e727371SKashyap D Desai /*
31728e727371SKashyap D Desai  * mrsas_build_mpt_cmd:	Calls helper function to build Passthru cmd
31738e727371SKashyap D Desai  * input:				Adapter soft state mfi cmd to build
3174665484d8SDoug Ambrisko  *
31758e727371SKashyap D Desai  * This function is called by mrsas_issue_cmd() to build the MPT-MFI passthru
31768e727371SKashyap D Desai  * command and prepares the MPT command to send to Firmware.
3177665484d8SDoug Ambrisko  */
3178665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION *
3179665484d8SDoug Ambrisko mrsas_build_mpt_cmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd)
3180665484d8SDoug Ambrisko {
3181665484d8SDoug Ambrisko 	MRSAS_REQUEST_DESCRIPTOR_UNION *req_desc;
3182665484d8SDoug Ambrisko 	u_int16_t index;
3183665484d8SDoug Ambrisko 
3184665484d8SDoug Ambrisko 	if (mrsas_build_mptmfi_passthru(sc, cmd)) {
3185665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot build MPT-MFI passthru cmd.\n");
3186665484d8SDoug Ambrisko 		return NULL;
3187665484d8SDoug Ambrisko 	}
3188665484d8SDoug Ambrisko 	index = cmd->cmd_id.context.smid;
3189665484d8SDoug Ambrisko 
3190665484d8SDoug Ambrisko 	req_desc = mrsas_get_request_desc(sc, index - 1);
3191665484d8SDoug Ambrisko 	if (!req_desc)
3192665484d8SDoug Ambrisko 		return NULL;
3193665484d8SDoug Ambrisko 
3194665484d8SDoug Ambrisko 	req_desc->addr.Words = 0;
3195665484d8SDoug Ambrisko 	req_desc->SCSIIO.RequestFlags = (MPI2_REQ_DESCRIPT_FLAGS_SCSI_IO << MRSAS_REQ_DESCRIPT_FLAGS_TYPE_SHIFT);
3196665484d8SDoug Ambrisko 
3197665484d8SDoug Ambrisko 	req_desc->SCSIIO.SMID = index;
3198665484d8SDoug Ambrisko 
3199665484d8SDoug Ambrisko 	return (req_desc);
3200665484d8SDoug Ambrisko }
3201665484d8SDoug Ambrisko 
32028e727371SKashyap D Desai /*
32038e727371SKashyap D Desai  * mrsas_build_mptmfi_passthru:	Builds a MPT MFI Passthru command
32048e727371SKashyap D Desai  * input:						Adapter soft state mfi cmd pointer
3205665484d8SDoug Ambrisko  *
32068e727371SKashyap D Desai  * The MPT command and the io_request are setup as a passthru command. The SGE
32078e727371SKashyap D Desai  * chain address is set to frame_phys_addr of the MFI command.
3208665484d8SDoug Ambrisko  */
3209665484d8SDoug Ambrisko u_int8_t
3210665484d8SDoug Ambrisko mrsas_build_mptmfi_passthru(struct mrsas_softc *sc, struct mrsas_mfi_cmd *mfi_cmd)
3211665484d8SDoug Ambrisko {
3212665484d8SDoug Ambrisko 	MPI25_IEEE_SGE_CHAIN64 *mpi25_ieee_chain;
3213665484d8SDoug Ambrisko 	PTR_MRSAS_RAID_SCSI_IO_REQUEST io_req;
3214665484d8SDoug Ambrisko 	struct mrsas_mpt_cmd *mpt_cmd;
3215665484d8SDoug Ambrisko 	struct mrsas_header *frame_hdr = &mfi_cmd->frame->hdr;
3216665484d8SDoug Ambrisko 
3217665484d8SDoug Ambrisko 	mpt_cmd = mrsas_get_mpt_cmd(sc);
3218665484d8SDoug Ambrisko 	if (!mpt_cmd)
3219665484d8SDoug Ambrisko 		return (1);
3220665484d8SDoug Ambrisko 
3221665484d8SDoug Ambrisko 	/* Save the smid. To be used for returning the cmd */
3222665484d8SDoug Ambrisko 	mfi_cmd->cmd_id.context.smid = mpt_cmd->index;
3223665484d8SDoug Ambrisko 
3224665484d8SDoug Ambrisko 	mpt_cmd->sync_cmd_idx = mfi_cmd->index;
3225665484d8SDoug Ambrisko 
3226665484d8SDoug Ambrisko 	/*
32278e727371SKashyap D Desai 	 * For cmds where the flag is set, store the flag and check on
32288e727371SKashyap D Desai 	 * completion. For cmds with this flag, don't call
3229665484d8SDoug Ambrisko 	 * mrsas_complete_cmd.
3230665484d8SDoug Ambrisko 	 */
3231665484d8SDoug Ambrisko 
3232665484d8SDoug Ambrisko 	if (frame_hdr->flags & MFI_FRAME_DONT_POST_IN_REPLY_QUEUE)
3233665484d8SDoug Ambrisko 		mpt_cmd->flags = MFI_FRAME_DONT_POST_IN_REPLY_QUEUE;
3234665484d8SDoug Ambrisko 
3235665484d8SDoug Ambrisko 	io_req = mpt_cmd->io_request;
3236665484d8SDoug Ambrisko 
3237665484d8SDoug Ambrisko 	if ((sc->device_id == MRSAS_INVADER) || (sc->device_id == MRSAS_FURY)) {
3238665484d8SDoug Ambrisko 		pMpi25IeeeSgeChain64_t sgl_ptr_end = (pMpi25IeeeSgeChain64_t)&io_req->SGL;
32398e727371SKashyap D Desai 
3240665484d8SDoug Ambrisko 		sgl_ptr_end += sc->max_sge_in_main_msg - 1;
3241665484d8SDoug Ambrisko 		sgl_ptr_end->Flags = 0;
3242665484d8SDoug Ambrisko 	}
3243665484d8SDoug Ambrisko 	mpi25_ieee_chain = (MPI25_IEEE_SGE_CHAIN64 *) & io_req->SGL.IeeeChain;
3244665484d8SDoug Ambrisko 
3245665484d8SDoug Ambrisko 	io_req->Function = MRSAS_MPI2_FUNCTION_PASSTHRU_IO_REQUEST;
3246665484d8SDoug Ambrisko 	io_req->SGLOffset0 = offsetof(MRSAS_RAID_SCSI_IO_REQUEST, SGL) / 4;
3247665484d8SDoug Ambrisko 	io_req->ChainOffset = sc->chain_offset_mfi_pthru;
3248665484d8SDoug Ambrisko 
3249665484d8SDoug Ambrisko 	mpi25_ieee_chain->Address = mfi_cmd->frame_phys_addr;
3250665484d8SDoug Ambrisko 
3251665484d8SDoug Ambrisko 	mpi25_ieee_chain->Flags = IEEE_SGE_FLAGS_CHAIN_ELEMENT |
3252665484d8SDoug Ambrisko 	    MPI2_IEEE_SGE_FLAGS_IOCPLBNTA_ADDR;
3253665484d8SDoug Ambrisko 
3254665484d8SDoug Ambrisko 	mpi25_ieee_chain->Length = MRSAS_MAX_SZ_CHAIN_FRAME;
3255665484d8SDoug Ambrisko 
3256665484d8SDoug Ambrisko 	return (0);
3257665484d8SDoug Ambrisko }
3258665484d8SDoug Ambrisko 
32598e727371SKashyap D Desai /*
32608e727371SKashyap D Desai  * mrsas_issue_blocked_cmd:	Synchronous wrapper around regular FW cmds
32618e727371SKashyap D Desai  * input:					Adapter soft state Command to be issued
3262665484d8SDoug Ambrisko  *
32638e727371SKashyap D Desai  * This function waits on an event for the command to be returned from the ISR.
32648e727371SKashyap D Desai  * Max wait time is MRSAS_INTERNAL_CMD_WAIT_TIME secs. Used for issuing
32658e727371SKashyap D Desai  * internal and ioctl commands.
3266665484d8SDoug Ambrisko  */
32678e727371SKashyap D Desai int
32688e727371SKashyap D Desai mrsas_issue_blocked_cmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd)
3269665484d8SDoug Ambrisko {
3270665484d8SDoug Ambrisko 	u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME;
3271665484d8SDoug Ambrisko 	unsigned long total_time = 0;
3272665484d8SDoug Ambrisko 	int retcode = 0;
3273665484d8SDoug Ambrisko 
3274665484d8SDoug Ambrisko 	/* Initialize cmd_status */
3275665484d8SDoug Ambrisko 	cmd->cmd_status = ECONNREFUSED;
3276665484d8SDoug Ambrisko 
3277665484d8SDoug Ambrisko 	/* Build MPT-MFI command for issue to FW */
3278665484d8SDoug Ambrisko 	if (mrsas_issue_dcmd(sc, cmd)) {
3279665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot issue DCMD internal command.\n");
3280665484d8SDoug Ambrisko 		return (1);
3281665484d8SDoug Ambrisko 	}
3282665484d8SDoug Ambrisko 	sc->chan = (void *)&cmd;
3283665484d8SDoug Ambrisko 
3284665484d8SDoug Ambrisko 	while (1) {
3285665484d8SDoug Ambrisko 		if (cmd->cmd_status == ECONNREFUSED) {
3286665484d8SDoug Ambrisko 			tsleep((void *)&sc->chan, 0, "mrsas_sleep", hz);
32878e727371SKashyap D Desai 		} else
3288665484d8SDoug Ambrisko 			break;
3289665484d8SDoug Ambrisko 		total_time++;
3290665484d8SDoug Ambrisko 		if (total_time >= max_wait) {
32918e727371SKashyap D Desai 			device_printf(sc->mrsas_dev,
32928e727371SKashyap D Desai 			    "Internal command timed out after %d seconds.\n", max_wait);
3293665484d8SDoug Ambrisko 			retcode = 1;
3294665484d8SDoug Ambrisko 			break;
3295665484d8SDoug Ambrisko 		}
3296665484d8SDoug Ambrisko 	}
3297665484d8SDoug Ambrisko 	return (retcode);
3298665484d8SDoug Ambrisko }
3299665484d8SDoug Ambrisko 
33008e727371SKashyap D Desai /*
33018e727371SKashyap D Desai  * mrsas_complete_mptmfi_passthru:	Completes a command
33028e727371SKashyap D Desai  * input:	@sc:					Adapter soft state
33038e727371SKashyap D Desai  * 			@cmd:					Command to be completed
33048e727371SKashyap D Desai  * 			@status:				cmd completion status
3305665484d8SDoug Ambrisko  *
33068e727371SKashyap D Desai  * This function is called from mrsas_complete_cmd() after an interrupt is
33078e727371SKashyap D Desai  * received from Firmware, and io_request->Function is
3308665484d8SDoug Ambrisko  * MRSAS_MPI2_FUNCTION_PASSTHRU_IO_REQUEST.
3309665484d8SDoug Ambrisko  */
3310665484d8SDoug Ambrisko void
3311665484d8SDoug Ambrisko mrsas_complete_mptmfi_passthru(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd,
3312665484d8SDoug Ambrisko     u_int8_t status)
3313665484d8SDoug Ambrisko {
3314665484d8SDoug Ambrisko 	struct mrsas_header *hdr = &cmd->frame->hdr;
3315665484d8SDoug Ambrisko 	u_int8_t cmd_status = cmd->frame->hdr.cmd_status;
3316665484d8SDoug Ambrisko 
3317665484d8SDoug Ambrisko 	/* Reset the retry counter for future re-tries */
3318665484d8SDoug Ambrisko 	cmd->retry_for_fw_reset = 0;
3319665484d8SDoug Ambrisko 
3320665484d8SDoug Ambrisko 	if (cmd->ccb_ptr)
3321665484d8SDoug Ambrisko 		cmd->ccb_ptr = NULL;
3322665484d8SDoug Ambrisko 
3323665484d8SDoug Ambrisko 	switch (hdr->cmd) {
3324665484d8SDoug Ambrisko 	case MFI_CMD_INVALID:
3325665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "MFI_CMD_INVALID command.\n");
3326665484d8SDoug Ambrisko 		break;
3327665484d8SDoug Ambrisko 	case MFI_CMD_PD_SCSI_IO:
3328665484d8SDoug Ambrisko 	case MFI_CMD_LD_SCSI_IO:
3329665484d8SDoug Ambrisko 		/*
3330665484d8SDoug Ambrisko 		 * MFI_CMD_PD_SCSI_IO and MFI_CMD_LD_SCSI_IO could have been
3331665484d8SDoug Ambrisko 		 * issued either through an IO path or an IOCTL path. If it
3332665484d8SDoug Ambrisko 		 * was via IOCTL, we will send it to internal completion.
3333665484d8SDoug Ambrisko 		 */
3334665484d8SDoug Ambrisko 		if (cmd->sync_cmd) {
3335665484d8SDoug Ambrisko 			cmd->sync_cmd = 0;
3336665484d8SDoug Ambrisko 			mrsas_wakeup(sc, cmd);
3337665484d8SDoug Ambrisko 			break;
3338665484d8SDoug Ambrisko 		}
3339665484d8SDoug Ambrisko 	case MFI_CMD_SMP:
3340665484d8SDoug Ambrisko 	case MFI_CMD_STP:
3341665484d8SDoug Ambrisko 	case MFI_CMD_DCMD:
3342665484d8SDoug Ambrisko 		/* Check for LD map update */
3343665484d8SDoug Ambrisko 		if ((cmd->frame->dcmd.opcode == MR_DCMD_LD_MAP_GET_INFO) &&
3344665484d8SDoug Ambrisko 		    (cmd->frame->dcmd.mbox.b[1] == 1)) {
3345665484d8SDoug Ambrisko 			sc->fast_path_io = 0;
3346665484d8SDoug Ambrisko 			mtx_lock(&sc->raidmap_lock);
3347665484d8SDoug Ambrisko 			if (cmd_status != 0) {
3348665484d8SDoug Ambrisko 				if (cmd_status != MFI_STAT_NOT_FOUND)
3349665484d8SDoug Ambrisko 					device_printf(sc->mrsas_dev, "map sync failed, status=%x\n", cmd_status);
3350665484d8SDoug Ambrisko 				else {
3351665484d8SDoug Ambrisko 					mrsas_release_mfi_cmd(cmd);
3352665484d8SDoug Ambrisko 					mtx_unlock(&sc->raidmap_lock);
3353665484d8SDoug Ambrisko 					break;
3354665484d8SDoug Ambrisko 				}
33558e727371SKashyap D Desai 			} else
3356665484d8SDoug Ambrisko 				sc->map_id++;
3357665484d8SDoug Ambrisko 			mrsas_release_mfi_cmd(cmd);
3358665484d8SDoug Ambrisko 			if (MR_ValidateMapInfo(sc))
3359665484d8SDoug Ambrisko 				sc->fast_path_io = 0;
3360665484d8SDoug Ambrisko 			else
3361665484d8SDoug Ambrisko 				sc->fast_path_io = 1;
3362665484d8SDoug Ambrisko 			mrsas_sync_map_info(sc);
3363665484d8SDoug Ambrisko 			mtx_unlock(&sc->raidmap_lock);
3364665484d8SDoug Ambrisko 			break;
3365665484d8SDoug Ambrisko 		}
3366665484d8SDoug Ambrisko 		if (cmd->frame->dcmd.opcode == MR_DCMD_CTRL_EVENT_GET_INFO ||
3367665484d8SDoug Ambrisko 		    cmd->frame->dcmd.opcode == MR_DCMD_CTRL_EVENT_GET) {
3368da011113SKashyap D Desai 			sc->mrsas_aen_triggered = 0;
3369665484d8SDoug Ambrisko 		}
3370665484d8SDoug Ambrisko 		/* See if got an event notification */
3371665484d8SDoug Ambrisko 		if (cmd->frame->dcmd.opcode == MR_DCMD_CTRL_EVENT_WAIT)
3372665484d8SDoug Ambrisko 			mrsas_complete_aen(sc, cmd);
3373665484d8SDoug Ambrisko 		else
3374665484d8SDoug Ambrisko 			mrsas_wakeup(sc, cmd);
3375665484d8SDoug Ambrisko 		break;
3376665484d8SDoug Ambrisko 	case MFI_CMD_ABORT:
3377665484d8SDoug Ambrisko 		/* Command issued to abort another cmd return */
3378665484d8SDoug Ambrisko 		mrsas_complete_abort(sc, cmd);
3379665484d8SDoug Ambrisko 		break;
3380665484d8SDoug Ambrisko 	default:
3381665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Unknown command completed! [0x%X]\n", hdr->cmd);
3382665484d8SDoug Ambrisko 		break;
3383665484d8SDoug Ambrisko 	}
3384665484d8SDoug Ambrisko }
3385665484d8SDoug Ambrisko 
33868e727371SKashyap D Desai /*
33878e727371SKashyap D Desai  * mrsas_wakeup:	Completes an internal command
3388665484d8SDoug Ambrisko  * input:			Adapter soft state
3389665484d8SDoug Ambrisko  * 					Command to be completed
3390665484d8SDoug Ambrisko  *
33918e727371SKashyap D Desai  * In mrsas_issue_blocked_cmd(), after a command is issued to Firmware, a wait
33928e727371SKashyap D Desai  * timer is started.  This function is called from
33938e727371SKashyap D Desai  * mrsas_complete_mptmfi_passthru() as it completes the command, to wake up
33948e727371SKashyap D Desai  * from the command wait.
3395665484d8SDoug Ambrisko  */
33968e727371SKashyap D Desai void
33978e727371SKashyap D Desai mrsas_wakeup(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd)
3398665484d8SDoug Ambrisko {
3399665484d8SDoug Ambrisko 	cmd->cmd_status = cmd->frame->io.cmd_status;
3400665484d8SDoug Ambrisko 
3401665484d8SDoug Ambrisko 	if (cmd->cmd_status == ECONNREFUSED)
3402665484d8SDoug Ambrisko 		cmd->cmd_status = 0;
3403665484d8SDoug Ambrisko 
3404665484d8SDoug Ambrisko 	sc->chan = (void *)&cmd;
3405665484d8SDoug Ambrisko 	wakeup_one((void *)&sc->chan);
3406665484d8SDoug Ambrisko 	return;
3407665484d8SDoug Ambrisko }
3408665484d8SDoug Ambrisko 
34098e727371SKashyap D Desai /*
34108e727371SKashyap D Desai  * mrsas_shutdown_ctlr:       Instructs FW to shutdown the controller input:
34118e727371SKashyap D Desai  * Adapter soft state Shutdown/Hibernate
3412665484d8SDoug Ambrisko  *
34138e727371SKashyap D Desai  * This function issues a DCMD internal command to Firmware to initiate shutdown
34148e727371SKashyap D Desai  * of the controller.
3415665484d8SDoug Ambrisko  */
34168e727371SKashyap D Desai static void
34178e727371SKashyap D Desai mrsas_shutdown_ctlr(struct mrsas_softc *sc, u_int32_t opcode)
3418665484d8SDoug Ambrisko {
3419665484d8SDoug Ambrisko 	struct mrsas_mfi_cmd *cmd;
3420665484d8SDoug Ambrisko 	struct mrsas_dcmd_frame *dcmd;
3421665484d8SDoug Ambrisko 
3422665484d8SDoug Ambrisko 	if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR)
3423665484d8SDoug Ambrisko 		return;
3424665484d8SDoug Ambrisko 
3425665484d8SDoug Ambrisko 	cmd = mrsas_get_mfi_cmd(sc);
3426665484d8SDoug Ambrisko 	if (!cmd) {
3427665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate for shutdown cmd.\n");
3428665484d8SDoug Ambrisko 		return;
3429665484d8SDoug Ambrisko 	}
3430665484d8SDoug Ambrisko 	if (sc->aen_cmd)
3431665484d8SDoug Ambrisko 		mrsas_issue_blocked_abort_cmd(sc, sc->aen_cmd);
3432665484d8SDoug Ambrisko 
3433665484d8SDoug Ambrisko 	if (sc->map_update_cmd)
3434665484d8SDoug Ambrisko 		mrsas_issue_blocked_abort_cmd(sc, sc->map_update_cmd);
3435665484d8SDoug Ambrisko 
3436665484d8SDoug Ambrisko 	dcmd = &cmd->frame->dcmd;
3437665484d8SDoug Ambrisko 	memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE);
3438665484d8SDoug Ambrisko 
3439665484d8SDoug Ambrisko 	dcmd->cmd = MFI_CMD_DCMD;
3440665484d8SDoug Ambrisko 	dcmd->cmd_status = 0x0;
3441665484d8SDoug Ambrisko 	dcmd->sge_count = 0;
3442665484d8SDoug Ambrisko 	dcmd->flags = MFI_FRAME_DIR_NONE;
3443665484d8SDoug Ambrisko 	dcmd->timeout = 0;
3444665484d8SDoug Ambrisko 	dcmd->pad_0 = 0;
3445665484d8SDoug Ambrisko 	dcmd->data_xfer_len = 0;
3446665484d8SDoug Ambrisko 	dcmd->opcode = opcode;
3447665484d8SDoug Ambrisko 
3448665484d8SDoug Ambrisko 	device_printf(sc->mrsas_dev, "Preparing to shut down controller.\n");
3449665484d8SDoug Ambrisko 
3450665484d8SDoug Ambrisko 	mrsas_issue_blocked_cmd(sc, cmd);
3451665484d8SDoug Ambrisko 	mrsas_release_mfi_cmd(cmd);
3452665484d8SDoug Ambrisko 
3453665484d8SDoug Ambrisko 	return;
3454665484d8SDoug Ambrisko }
3455665484d8SDoug Ambrisko 
34568e727371SKashyap D Desai /*
34578e727371SKashyap D Desai  * mrsas_flush_cache:         Requests FW to flush all its caches input:
34588e727371SKashyap D Desai  * Adapter soft state
3459665484d8SDoug Ambrisko  *
3460665484d8SDoug Ambrisko  * This function is issues a DCMD internal command to Firmware to initiate
3461665484d8SDoug Ambrisko  * flushing of all caches.
3462665484d8SDoug Ambrisko  */
34638e727371SKashyap D Desai static void
34648e727371SKashyap D Desai mrsas_flush_cache(struct mrsas_softc *sc)
3465665484d8SDoug Ambrisko {
3466665484d8SDoug Ambrisko 	struct mrsas_mfi_cmd *cmd;
3467665484d8SDoug Ambrisko 	struct mrsas_dcmd_frame *dcmd;
3468665484d8SDoug Ambrisko 
3469665484d8SDoug Ambrisko 	if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR)
3470665484d8SDoug Ambrisko 		return;
3471665484d8SDoug Ambrisko 
3472665484d8SDoug Ambrisko 	cmd = mrsas_get_mfi_cmd(sc);
3473665484d8SDoug Ambrisko 	if (!cmd) {
3474665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate for flush cache cmd.\n");
3475665484d8SDoug Ambrisko 		return;
3476665484d8SDoug Ambrisko 	}
3477665484d8SDoug Ambrisko 	dcmd = &cmd->frame->dcmd;
3478665484d8SDoug Ambrisko 	memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE);
3479665484d8SDoug Ambrisko 
3480665484d8SDoug Ambrisko 	dcmd->cmd = MFI_CMD_DCMD;
3481665484d8SDoug Ambrisko 	dcmd->cmd_status = 0x0;
3482665484d8SDoug Ambrisko 	dcmd->sge_count = 0;
3483665484d8SDoug Ambrisko 	dcmd->flags = MFI_FRAME_DIR_NONE;
3484665484d8SDoug Ambrisko 	dcmd->timeout = 0;
3485665484d8SDoug Ambrisko 	dcmd->pad_0 = 0;
3486665484d8SDoug Ambrisko 	dcmd->data_xfer_len = 0;
3487665484d8SDoug Ambrisko 	dcmd->opcode = MR_DCMD_CTRL_CACHE_FLUSH;
3488665484d8SDoug Ambrisko 	dcmd->mbox.b[0] = MR_FLUSH_CTRL_CACHE | MR_FLUSH_DISK_CACHE;
3489665484d8SDoug Ambrisko 
3490665484d8SDoug Ambrisko 	mrsas_issue_blocked_cmd(sc, cmd);
3491665484d8SDoug Ambrisko 	mrsas_release_mfi_cmd(cmd);
3492665484d8SDoug Ambrisko 
3493665484d8SDoug Ambrisko 	return;
3494665484d8SDoug Ambrisko }
3495665484d8SDoug Ambrisko 
34968e727371SKashyap D Desai /*
34978e727371SKashyap D Desai  * mrsas_get_map_info:        Load and validate RAID map input:
34988e727371SKashyap D Desai  * Adapter instance soft state
3499665484d8SDoug Ambrisko  *
35008e727371SKashyap D Desai  * This function calls mrsas_get_ld_map_info() and MR_ValidateMapInfo() to load
35018e727371SKashyap D Desai  * and validate RAID map.  It returns 0 if successful, 1 other- wise.
3502665484d8SDoug Ambrisko  */
35038e727371SKashyap D Desai static int
35048e727371SKashyap D Desai mrsas_get_map_info(struct mrsas_softc *sc)
3505665484d8SDoug Ambrisko {
3506665484d8SDoug Ambrisko 	uint8_t retcode = 0;
3507665484d8SDoug Ambrisko 
3508665484d8SDoug Ambrisko 	sc->fast_path_io = 0;
3509665484d8SDoug Ambrisko 	if (!mrsas_get_ld_map_info(sc)) {
3510665484d8SDoug Ambrisko 		retcode = MR_ValidateMapInfo(sc);
3511665484d8SDoug Ambrisko 		if (retcode == 0) {
3512665484d8SDoug Ambrisko 			sc->fast_path_io = 1;
3513665484d8SDoug Ambrisko 			return 0;
3514665484d8SDoug Ambrisko 		}
3515665484d8SDoug Ambrisko 	}
3516665484d8SDoug Ambrisko 	return 1;
3517665484d8SDoug Ambrisko }
3518665484d8SDoug Ambrisko 
35198e727371SKashyap D Desai /*
35208e727371SKashyap D Desai  * mrsas_get_ld_map_info:      Get FW's ld_map structure input:
35218e727371SKashyap D Desai  * Adapter instance soft state
3522665484d8SDoug Ambrisko  *
35238e727371SKashyap D Desai  * Issues an internal command (DCMD) to get the FW's controller PD list
35248e727371SKashyap D Desai  * structure.
3525665484d8SDoug Ambrisko  */
35268e727371SKashyap D Desai static int
35278e727371SKashyap D Desai mrsas_get_ld_map_info(struct mrsas_softc *sc)
3528665484d8SDoug Ambrisko {
3529665484d8SDoug Ambrisko 	int retcode = 0;
3530665484d8SDoug Ambrisko 	struct mrsas_mfi_cmd *cmd;
3531665484d8SDoug Ambrisko 	struct mrsas_dcmd_frame *dcmd;
35324799d485SKashyap D Desai 	void *map;
3533665484d8SDoug Ambrisko 	bus_addr_t map_phys_addr = 0;
3534665484d8SDoug Ambrisko 
3535665484d8SDoug Ambrisko 	cmd = mrsas_get_mfi_cmd(sc);
3536665484d8SDoug Ambrisko 	if (!cmd) {
35374799d485SKashyap D Desai 		device_printf(sc->mrsas_dev,
35384799d485SKashyap D Desai 		    "Cannot alloc for ld map info cmd.\n");
3539665484d8SDoug Ambrisko 		return 1;
3540665484d8SDoug Ambrisko 	}
3541665484d8SDoug Ambrisko 	dcmd = &cmd->frame->dcmd;
3542665484d8SDoug Ambrisko 
35434799d485SKashyap D Desai 	map = (void *)sc->raidmap_mem[(sc->map_id & 1)];
3544665484d8SDoug Ambrisko 	map_phys_addr = sc->raidmap_phys_addr[(sc->map_id & 1)];
3545665484d8SDoug Ambrisko 	if (!map) {
35464799d485SKashyap D Desai 		device_printf(sc->mrsas_dev,
35474799d485SKashyap D Desai 		    "Failed to alloc mem for ld map info.\n");
3548665484d8SDoug Ambrisko 		mrsas_release_mfi_cmd(cmd);
3549665484d8SDoug Ambrisko 		return (ENOMEM);
3550665484d8SDoug Ambrisko 	}
35514799d485SKashyap D Desai 	memset(map, 0, sizeof(sc->max_map_sz));
3552665484d8SDoug Ambrisko 	memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE);
3553665484d8SDoug Ambrisko 
3554665484d8SDoug Ambrisko 	dcmd->cmd = MFI_CMD_DCMD;
3555665484d8SDoug Ambrisko 	dcmd->cmd_status = 0xFF;
3556665484d8SDoug Ambrisko 	dcmd->sge_count = 1;
3557665484d8SDoug Ambrisko 	dcmd->flags = MFI_FRAME_DIR_READ;
3558665484d8SDoug Ambrisko 	dcmd->timeout = 0;
3559665484d8SDoug Ambrisko 	dcmd->pad_0 = 0;
35604799d485SKashyap D Desai 	dcmd->data_xfer_len = sc->current_map_sz;
3561665484d8SDoug Ambrisko 	dcmd->opcode = MR_DCMD_LD_MAP_GET_INFO;
3562665484d8SDoug Ambrisko 	dcmd->sgl.sge32[0].phys_addr = map_phys_addr;
35634799d485SKashyap D Desai 	dcmd->sgl.sge32[0].length = sc->current_map_sz;
35644799d485SKashyap D Desai 
3565665484d8SDoug Ambrisko 	if (!mrsas_issue_polled(sc, cmd))
3566665484d8SDoug Ambrisko 		retcode = 0;
35678e727371SKashyap D Desai 	else {
35684799d485SKashyap D Desai 		device_printf(sc->mrsas_dev,
35694799d485SKashyap D Desai 		    "Fail to send get LD map info cmd.\n");
3570665484d8SDoug Ambrisko 		retcode = 1;
3571665484d8SDoug Ambrisko 	}
3572665484d8SDoug Ambrisko 	mrsas_release_mfi_cmd(cmd);
35734799d485SKashyap D Desai 
3574665484d8SDoug Ambrisko 	return (retcode);
3575665484d8SDoug Ambrisko }
3576665484d8SDoug Ambrisko 
35778e727371SKashyap D Desai /*
35788e727371SKashyap D Desai  * mrsas_sync_map_info:        Get FW's ld_map structure input:
35798e727371SKashyap D Desai  * Adapter instance soft state
3580665484d8SDoug Ambrisko  *
35818e727371SKashyap D Desai  * Issues an internal command (DCMD) to get the FW's controller PD list
35828e727371SKashyap D Desai  * structure.
3583665484d8SDoug Ambrisko  */
35848e727371SKashyap D Desai static int
35858e727371SKashyap D Desai mrsas_sync_map_info(struct mrsas_softc *sc)
3586665484d8SDoug Ambrisko {
3587665484d8SDoug Ambrisko 	int retcode = 0, i;
3588665484d8SDoug Ambrisko 	struct mrsas_mfi_cmd *cmd;
3589665484d8SDoug Ambrisko 	struct mrsas_dcmd_frame *dcmd;
3590665484d8SDoug Ambrisko 	uint32_t size_sync_info, num_lds;
3591665484d8SDoug Ambrisko 	MR_LD_TARGET_SYNC *target_map = NULL;
35924799d485SKashyap D Desai 	MR_DRV_RAID_MAP_ALL *map;
3593665484d8SDoug Ambrisko 	MR_LD_RAID *raid;
3594665484d8SDoug Ambrisko 	MR_LD_TARGET_SYNC *ld_sync;
3595665484d8SDoug Ambrisko 	bus_addr_t map_phys_addr = 0;
3596665484d8SDoug Ambrisko 
3597665484d8SDoug Ambrisko 	cmd = mrsas_get_mfi_cmd(sc);
3598665484d8SDoug Ambrisko 	if (!cmd) {
35994799d485SKashyap D Desai 		device_printf(sc->mrsas_dev,
36004799d485SKashyap D Desai 		    "Cannot alloc for sync map info cmd\n");
3601665484d8SDoug Ambrisko 		return 1;
3602665484d8SDoug Ambrisko 	}
36034799d485SKashyap D Desai 	map = sc->ld_drv_map[sc->map_id & 1];
3604665484d8SDoug Ambrisko 	num_lds = map->raidMap.ldCount;
3605665484d8SDoug Ambrisko 
3606665484d8SDoug Ambrisko 	dcmd = &cmd->frame->dcmd;
3607665484d8SDoug Ambrisko 	size_sync_info = sizeof(MR_LD_TARGET_SYNC) * num_lds;
3608665484d8SDoug Ambrisko 	memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE);
3609665484d8SDoug Ambrisko 
36108e727371SKashyap D Desai 	target_map = (MR_LD_TARGET_SYNC *) sc->raidmap_mem[(sc->map_id - 1) & 1];
36114799d485SKashyap D Desai 	memset(target_map, 0, sc->max_map_sz);
3612665484d8SDoug Ambrisko 
3613665484d8SDoug Ambrisko 	map_phys_addr = sc->raidmap_phys_addr[(sc->map_id - 1) & 1];
3614665484d8SDoug Ambrisko 
3615665484d8SDoug Ambrisko 	ld_sync = (MR_LD_TARGET_SYNC *) target_map;
3616665484d8SDoug Ambrisko 
3617665484d8SDoug Ambrisko 	for (i = 0; i < num_lds; i++, ld_sync++) {
3618665484d8SDoug Ambrisko 		raid = MR_LdRaidGet(i, map);
3619665484d8SDoug Ambrisko 		ld_sync->targetId = MR_GetLDTgtId(i, map);
3620665484d8SDoug Ambrisko 		ld_sync->seqNum = raid->seqNum;
3621665484d8SDoug Ambrisko 	}
3622665484d8SDoug Ambrisko 
3623665484d8SDoug Ambrisko 	dcmd->cmd = MFI_CMD_DCMD;
3624665484d8SDoug Ambrisko 	dcmd->cmd_status = 0xFF;
3625665484d8SDoug Ambrisko 	dcmd->sge_count = 1;
3626665484d8SDoug Ambrisko 	dcmd->flags = MFI_FRAME_DIR_WRITE;
3627665484d8SDoug Ambrisko 	dcmd->timeout = 0;
3628665484d8SDoug Ambrisko 	dcmd->pad_0 = 0;
36294799d485SKashyap D Desai 	dcmd->data_xfer_len = sc->current_map_sz;
3630665484d8SDoug Ambrisko 	dcmd->mbox.b[0] = num_lds;
3631665484d8SDoug Ambrisko 	dcmd->mbox.b[1] = MRSAS_DCMD_MBOX_PEND_FLAG;
3632665484d8SDoug Ambrisko 	dcmd->opcode = MR_DCMD_LD_MAP_GET_INFO;
3633665484d8SDoug Ambrisko 	dcmd->sgl.sge32[0].phys_addr = map_phys_addr;
36344799d485SKashyap D Desai 	dcmd->sgl.sge32[0].length = sc->current_map_sz;
3635665484d8SDoug Ambrisko 
3636665484d8SDoug Ambrisko 	sc->map_update_cmd = cmd;
3637665484d8SDoug Ambrisko 	if (mrsas_issue_dcmd(sc, cmd)) {
36384799d485SKashyap D Desai 		device_printf(sc->mrsas_dev,
36394799d485SKashyap D Desai 		    "Fail to send sync map info command.\n");
3640665484d8SDoug Ambrisko 		return (1);
3641665484d8SDoug Ambrisko 	}
3642665484d8SDoug Ambrisko 	return (retcode);
3643665484d8SDoug Ambrisko }
3644665484d8SDoug Ambrisko 
36458e727371SKashyap D Desai /*
36468e727371SKashyap D Desai  * mrsas_get_pd_list:           Returns FW's PD list structure input:
36478e727371SKashyap D Desai  * Adapter soft state
3648665484d8SDoug Ambrisko  *
36498e727371SKashyap D Desai  * Issues an internal command (DCMD) to get the FW's controller PD list
36508e727371SKashyap D Desai  * structure.  This information is mainly used to find out about system
36518e727371SKashyap D Desai  * supported by Firmware.
3652665484d8SDoug Ambrisko  */
36538e727371SKashyap D Desai static int
36548e727371SKashyap D Desai mrsas_get_pd_list(struct mrsas_softc *sc)
3655665484d8SDoug Ambrisko {
3656665484d8SDoug Ambrisko 	int retcode = 0, pd_index = 0, pd_count = 0, pd_list_size;
3657665484d8SDoug Ambrisko 	struct mrsas_mfi_cmd *cmd;
3658665484d8SDoug Ambrisko 	struct mrsas_dcmd_frame *dcmd;
3659665484d8SDoug Ambrisko 	struct MR_PD_LIST *pd_list_mem;
3660665484d8SDoug Ambrisko 	struct MR_PD_ADDRESS *pd_addr;
3661665484d8SDoug Ambrisko 	bus_addr_t pd_list_phys_addr = 0;
3662665484d8SDoug Ambrisko 	struct mrsas_tmp_dcmd *tcmd;
3663665484d8SDoug Ambrisko 
3664665484d8SDoug Ambrisko 	cmd = mrsas_get_mfi_cmd(sc);
3665665484d8SDoug Ambrisko 	if (!cmd) {
36664799d485SKashyap D Desai 		device_printf(sc->mrsas_dev,
36674799d485SKashyap D Desai 		    "Cannot alloc for get PD list cmd\n");
3668665484d8SDoug Ambrisko 		return 1;
3669665484d8SDoug Ambrisko 	}
3670665484d8SDoug Ambrisko 	dcmd = &cmd->frame->dcmd;
3671665484d8SDoug Ambrisko 
3672665484d8SDoug Ambrisko 	tcmd = malloc(sizeof(struct mrsas_tmp_dcmd), M_MRSAS, M_NOWAIT);
3673665484d8SDoug Ambrisko 	pd_list_size = MRSAS_MAX_PD * sizeof(struct MR_PD_LIST);
3674665484d8SDoug Ambrisko 	if (mrsas_alloc_tmp_dcmd(sc, tcmd, pd_list_size) != SUCCESS) {
36754799d485SKashyap D Desai 		device_printf(sc->mrsas_dev,
36764799d485SKashyap D Desai 		    "Cannot alloc dmamap for get PD list cmd\n");
3677665484d8SDoug Ambrisko 		mrsas_release_mfi_cmd(cmd);
3678665484d8SDoug Ambrisko 		return (ENOMEM);
36798e727371SKashyap D Desai 	} else {
3680665484d8SDoug Ambrisko 		pd_list_mem = tcmd->tmp_dcmd_mem;
3681665484d8SDoug Ambrisko 		pd_list_phys_addr = tcmd->tmp_dcmd_phys_addr;
3682665484d8SDoug Ambrisko 	}
3683665484d8SDoug Ambrisko 	memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE);
3684665484d8SDoug Ambrisko 
3685665484d8SDoug Ambrisko 	dcmd->mbox.b[0] = MR_PD_QUERY_TYPE_EXPOSED_TO_HOST;
3686665484d8SDoug Ambrisko 	dcmd->mbox.b[1] = 0;
3687665484d8SDoug Ambrisko 	dcmd->cmd = MFI_CMD_DCMD;
3688665484d8SDoug Ambrisko 	dcmd->cmd_status = 0xFF;
3689665484d8SDoug Ambrisko 	dcmd->sge_count = 1;
3690665484d8SDoug Ambrisko 	dcmd->flags = MFI_FRAME_DIR_READ;
3691665484d8SDoug Ambrisko 	dcmd->timeout = 0;
3692665484d8SDoug Ambrisko 	dcmd->pad_0 = 0;
3693665484d8SDoug Ambrisko 	dcmd->data_xfer_len = MRSAS_MAX_PD * sizeof(struct MR_PD_LIST);
3694665484d8SDoug Ambrisko 	dcmd->opcode = MR_DCMD_PD_LIST_QUERY;
3695665484d8SDoug Ambrisko 	dcmd->sgl.sge32[0].phys_addr = pd_list_phys_addr;
3696665484d8SDoug Ambrisko 	dcmd->sgl.sge32[0].length = MRSAS_MAX_PD * sizeof(struct MR_PD_LIST);
3697665484d8SDoug Ambrisko 
3698665484d8SDoug Ambrisko 	if (!mrsas_issue_polled(sc, cmd))
3699665484d8SDoug Ambrisko 		retcode = 0;
3700665484d8SDoug Ambrisko 	else
3701665484d8SDoug Ambrisko 		retcode = 1;
3702665484d8SDoug Ambrisko 
3703665484d8SDoug Ambrisko 	/* Get the instance PD list */
3704665484d8SDoug Ambrisko 	pd_count = MRSAS_MAX_PD;
3705665484d8SDoug Ambrisko 	pd_addr = pd_list_mem->addr;
3706665484d8SDoug Ambrisko 	if (retcode == 0 && pd_list_mem->count < pd_count) {
37074799d485SKashyap D Desai 		memset(sc->local_pd_list, 0,
37084799d485SKashyap D Desai 		    MRSAS_MAX_PD * sizeof(struct mrsas_pd_list));
3709665484d8SDoug Ambrisko 		for (pd_index = 0; pd_index < pd_list_mem->count; pd_index++) {
3710665484d8SDoug Ambrisko 			sc->local_pd_list[pd_addr->deviceId].tid = pd_addr->deviceId;
37114799d485SKashyap D Desai 			sc->local_pd_list[pd_addr->deviceId].driveType =
37124799d485SKashyap D Desai 			    pd_addr->scsiDevType;
37134799d485SKashyap D Desai 			sc->local_pd_list[pd_addr->deviceId].driveState =
37144799d485SKashyap D Desai 			    MR_PD_STATE_SYSTEM;
3715665484d8SDoug Ambrisko 			pd_addr++;
3716665484d8SDoug Ambrisko 		}
3717665484d8SDoug Ambrisko 	}
37188e727371SKashyap D Desai 	/*
37198e727371SKashyap D Desai 	 * Use mutext/spinlock if pd_list component size increase more than
37208e727371SKashyap D Desai 	 * 32 bit.
37218e727371SKashyap D Desai 	 */
3722665484d8SDoug Ambrisko 	memcpy(sc->pd_list, sc->local_pd_list, sizeof(sc->local_pd_list));
3723665484d8SDoug Ambrisko 	mrsas_free_tmp_dcmd(tcmd);
3724665484d8SDoug Ambrisko 	mrsas_release_mfi_cmd(cmd);
3725665484d8SDoug Ambrisko 	free(tcmd, M_MRSAS);
3726665484d8SDoug Ambrisko 	return (retcode);
3727665484d8SDoug Ambrisko }
3728665484d8SDoug Ambrisko 
37298e727371SKashyap D Desai /*
37308e727371SKashyap D Desai  * mrsas_get_ld_list:           Returns FW's LD list structure input:
37318e727371SKashyap D Desai  * Adapter soft state
3732665484d8SDoug Ambrisko  *
37338e727371SKashyap D Desai  * Issues an internal command (DCMD) to get the FW's controller PD list
37348e727371SKashyap D Desai  * structure.  This information is mainly used to find out about supported by
37358e727371SKashyap D Desai  * the FW.
3736665484d8SDoug Ambrisko  */
37378e727371SKashyap D Desai static int
37388e727371SKashyap D Desai mrsas_get_ld_list(struct mrsas_softc *sc)
3739665484d8SDoug Ambrisko {
3740665484d8SDoug Ambrisko 	int ld_list_size, retcode = 0, ld_index = 0, ids = 0;
3741665484d8SDoug Ambrisko 	struct mrsas_mfi_cmd *cmd;
3742665484d8SDoug Ambrisko 	struct mrsas_dcmd_frame *dcmd;
3743665484d8SDoug Ambrisko 	struct MR_LD_LIST *ld_list_mem;
3744665484d8SDoug Ambrisko 	bus_addr_t ld_list_phys_addr = 0;
3745665484d8SDoug Ambrisko 	struct mrsas_tmp_dcmd *tcmd;
3746665484d8SDoug Ambrisko 
3747665484d8SDoug Ambrisko 	cmd = mrsas_get_mfi_cmd(sc);
3748665484d8SDoug Ambrisko 	if (!cmd) {
37494799d485SKashyap D Desai 		device_printf(sc->mrsas_dev,
37504799d485SKashyap D Desai 		    "Cannot alloc for get LD list cmd\n");
3751665484d8SDoug Ambrisko 		return 1;
3752665484d8SDoug Ambrisko 	}
3753665484d8SDoug Ambrisko 	dcmd = &cmd->frame->dcmd;
3754665484d8SDoug Ambrisko 
3755665484d8SDoug Ambrisko 	tcmd = malloc(sizeof(struct mrsas_tmp_dcmd), M_MRSAS, M_NOWAIT);
3756665484d8SDoug Ambrisko 	ld_list_size = sizeof(struct MR_LD_LIST);
3757665484d8SDoug Ambrisko 	if (mrsas_alloc_tmp_dcmd(sc, tcmd, ld_list_size) != SUCCESS) {
37584799d485SKashyap D Desai 		device_printf(sc->mrsas_dev,
37594799d485SKashyap D Desai 		    "Cannot alloc dmamap for get LD list cmd\n");
3760665484d8SDoug Ambrisko 		mrsas_release_mfi_cmd(cmd);
3761665484d8SDoug Ambrisko 		return (ENOMEM);
37628e727371SKashyap D Desai 	} else {
3763665484d8SDoug Ambrisko 		ld_list_mem = tcmd->tmp_dcmd_mem;
3764665484d8SDoug Ambrisko 		ld_list_phys_addr = tcmd->tmp_dcmd_phys_addr;
3765665484d8SDoug Ambrisko 	}
3766665484d8SDoug Ambrisko 	memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE);
3767665484d8SDoug Ambrisko 
37684799d485SKashyap D Desai 	if (sc->max256vdSupport)
37694799d485SKashyap D Desai 		dcmd->mbox.b[0] = 1;
37704799d485SKashyap D Desai 
3771665484d8SDoug Ambrisko 	dcmd->cmd = MFI_CMD_DCMD;
3772665484d8SDoug Ambrisko 	dcmd->cmd_status = 0xFF;
3773665484d8SDoug Ambrisko 	dcmd->sge_count = 1;
3774665484d8SDoug Ambrisko 	dcmd->flags = MFI_FRAME_DIR_READ;
3775665484d8SDoug Ambrisko 	dcmd->timeout = 0;
3776665484d8SDoug Ambrisko 	dcmd->data_xfer_len = sizeof(struct MR_LD_LIST);
3777665484d8SDoug Ambrisko 	dcmd->opcode = MR_DCMD_LD_GET_LIST;
3778665484d8SDoug Ambrisko 	dcmd->sgl.sge32[0].phys_addr = ld_list_phys_addr;
3779665484d8SDoug Ambrisko 	dcmd->sgl.sge32[0].length = sizeof(struct MR_LD_LIST);
3780665484d8SDoug Ambrisko 	dcmd->pad_0 = 0;
3781665484d8SDoug Ambrisko 
3782665484d8SDoug Ambrisko 	if (!mrsas_issue_polled(sc, cmd))
3783665484d8SDoug Ambrisko 		retcode = 0;
3784665484d8SDoug Ambrisko 	else
3785665484d8SDoug Ambrisko 		retcode = 1;
3786665484d8SDoug Ambrisko 
37874799d485SKashyap D Desai #if VD_EXT_DEBUG
37884799d485SKashyap D Desai 	printf("Number of LDs %d\n", ld_list_mem->ldCount);
37894799d485SKashyap D Desai #endif
37904799d485SKashyap D Desai 
3791665484d8SDoug Ambrisko 	/* Get the instance LD list */
37924799d485SKashyap D Desai 	if ((retcode == 0) &&
37934799d485SKashyap D Desai 	    (ld_list_mem->ldCount <= sc->fw_supported_vd_count)) {
3794665484d8SDoug Ambrisko 		sc->CurLdCount = ld_list_mem->ldCount;
37954799d485SKashyap D Desai 		memset(sc->ld_ids, 0xff, MAX_LOGICAL_DRIVES_EXT);
3796665484d8SDoug Ambrisko 		for (ld_index = 0; ld_index < ld_list_mem->ldCount; ld_index++) {
3797665484d8SDoug Ambrisko 			if (ld_list_mem->ldList[ld_index].state != 0) {
3798665484d8SDoug Ambrisko 				ids = ld_list_mem->ldList[ld_index].ref.ld_context.targetId;
3799665484d8SDoug Ambrisko 				sc->ld_ids[ids] = ld_list_mem->ldList[ld_index].ref.ld_context.targetId;
3800665484d8SDoug Ambrisko 			}
3801665484d8SDoug Ambrisko 		}
3802665484d8SDoug Ambrisko 	}
3803665484d8SDoug Ambrisko 	mrsas_free_tmp_dcmd(tcmd);
3804665484d8SDoug Ambrisko 	mrsas_release_mfi_cmd(cmd);
3805665484d8SDoug Ambrisko 	free(tcmd, M_MRSAS);
3806665484d8SDoug Ambrisko 	return (retcode);
3807665484d8SDoug Ambrisko }
3808665484d8SDoug Ambrisko 
38098e727371SKashyap D Desai /*
38108e727371SKashyap D Desai  * mrsas_alloc_tmp_dcmd:       Allocates memory for temporary command input:
38118e727371SKashyap D Desai  * Adapter soft state Temp command Size of alloction
3812665484d8SDoug Ambrisko  *
3813665484d8SDoug Ambrisko  * Allocates DMAable memory for a temporary internal command. The allocated
3814665484d8SDoug Ambrisko  * memory is initialized to all zeros upon successful loading of the dma
3815665484d8SDoug Ambrisko  * mapped memory.
3816665484d8SDoug Ambrisko  */
38178e727371SKashyap D Desai int
38188e727371SKashyap D Desai mrsas_alloc_tmp_dcmd(struct mrsas_softc *sc,
38198e727371SKashyap D Desai     struct mrsas_tmp_dcmd *tcmd, int size)
3820665484d8SDoug Ambrisko {
38218e727371SKashyap D Desai 	if (bus_dma_tag_create(sc->mrsas_parent_tag,
38228e727371SKashyap D Desai 	    1, 0,
38238e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR_32BIT,
38248e727371SKashyap D Desai 	    BUS_SPACE_MAXADDR,
38258e727371SKashyap D Desai 	    NULL, NULL,
38268e727371SKashyap D Desai 	    size,
38278e727371SKashyap D Desai 	    1,
38288e727371SKashyap D Desai 	    size,
38298e727371SKashyap D Desai 	    BUS_DMA_ALLOCNOW,
38308e727371SKashyap D Desai 	    NULL, NULL,
3831665484d8SDoug Ambrisko 	    &tcmd->tmp_dcmd_tag)) {
3832665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate tmp dcmd tag\n");
3833665484d8SDoug Ambrisko 		return (ENOMEM);
3834665484d8SDoug Ambrisko 	}
3835665484d8SDoug Ambrisko 	if (bus_dmamem_alloc(tcmd->tmp_dcmd_tag, (void **)&tcmd->tmp_dcmd_mem,
3836665484d8SDoug Ambrisko 	    BUS_DMA_NOWAIT, &tcmd->tmp_dcmd_dmamap)) {
3837665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate tmp dcmd mem\n");
3838665484d8SDoug Ambrisko 		return (ENOMEM);
3839665484d8SDoug Ambrisko 	}
3840665484d8SDoug Ambrisko 	if (bus_dmamap_load(tcmd->tmp_dcmd_tag, tcmd->tmp_dcmd_dmamap,
3841665484d8SDoug Ambrisko 	    tcmd->tmp_dcmd_mem, size, mrsas_addr_cb,
3842665484d8SDoug Ambrisko 	    &tcmd->tmp_dcmd_phys_addr, BUS_DMA_NOWAIT)) {
3843665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot load tmp dcmd mem\n");
3844665484d8SDoug Ambrisko 		return (ENOMEM);
3845665484d8SDoug Ambrisko 	}
3846665484d8SDoug Ambrisko 	memset(tcmd->tmp_dcmd_mem, 0, size);
3847665484d8SDoug Ambrisko 	return (0);
3848665484d8SDoug Ambrisko }
3849665484d8SDoug Ambrisko 
38508e727371SKashyap D Desai /*
38518e727371SKashyap D Desai  * mrsas_free_tmp_dcmd:      Free memory for temporary command input:
38528e727371SKashyap D Desai  * temporary dcmd pointer
3853665484d8SDoug Ambrisko  *
38548e727371SKashyap D Desai  * Deallocates memory of the temporary command for use in the construction of
38558e727371SKashyap D Desai  * the internal DCMD.
3856665484d8SDoug Ambrisko  */
38578e727371SKashyap D Desai void
38588e727371SKashyap D Desai mrsas_free_tmp_dcmd(struct mrsas_tmp_dcmd *tmp)
3859665484d8SDoug Ambrisko {
3860665484d8SDoug Ambrisko 	if (tmp->tmp_dcmd_phys_addr)
3861665484d8SDoug Ambrisko 		bus_dmamap_unload(tmp->tmp_dcmd_tag, tmp->tmp_dcmd_dmamap);
3862665484d8SDoug Ambrisko 	if (tmp->tmp_dcmd_mem != NULL)
3863665484d8SDoug Ambrisko 		bus_dmamem_free(tmp->tmp_dcmd_tag, tmp->tmp_dcmd_mem, tmp->tmp_dcmd_dmamap);
3864665484d8SDoug Ambrisko 	if (tmp->tmp_dcmd_tag != NULL)
3865665484d8SDoug Ambrisko 		bus_dma_tag_destroy(tmp->tmp_dcmd_tag);
3866665484d8SDoug Ambrisko }
3867665484d8SDoug Ambrisko 
38688e727371SKashyap D Desai /*
38698e727371SKashyap D Desai  * mrsas_issue_blocked_abort_cmd:       Aborts previously issued cmd input:
38708e727371SKashyap D Desai  * Adapter soft state Previously issued cmd to be aborted
3871665484d8SDoug Ambrisko  *
3872665484d8SDoug Ambrisko  * This function is used to abort previously issued commands, such as AEN and
3873665484d8SDoug Ambrisko  * RAID map sync map commands.  The abort command is sent as a DCMD internal
3874665484d8SDoug Ambrisko  * command and subsequently the driver will wait for a return status.  The
3875665484d8SDoug Ambrisko  * max wait time is MRSAS_INTERNAL_CMD_WAIT_TIME seconds.
3876665484d8SDoug Ambrisko  */
38778e727371SKashyap D Desai static int
38788e727371SKashyap D Desai mrsas_issue_blocked_abort_cmd(struct mrsas_softc *sc,
3879665484d8SDoug Ambrisko     struct mrsas_mfi_cmd *cmd_to_abort)
3880665484d8SDoug Ambrisko {
3881665484d8SDoug Ambrisko 	struct mrsas_mfi_cmd *cmd;
3882665484d8SDoug Ambrisko 	struct mrsas_abort_frame *abort_fr;
3883665484d8SDoug Ambrisko 	u_int8_t retcode = 0;
3884665484d8SDoug Ambrisko 	unsigned long total_time = 0;
3885665484d8SDoug Ambrisko 	u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME;
3886665484d8SDoug Ambrisko 
3887665484d8SDoug Ambrisko 	cmd = mrsas_get_mfi_cmd(sc);
3888665484d8SDoug Ambrisko 	if (!cmd) {
3889665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot alloc for abort cmd\n");
3890665484d8SDoug Ambrisko 		return (1);
3891665484d8SDoug Ambrisko 	}
3892665484d8SDoug Ambrisko 	abort_fr = &cmd->frame->abort;
3893665484d8SDoug Ambrisko 
3894665484d8SDoug Ambrisko 	/* Prepare and issue the abort frame */
3895665484d8SDoug Ambrisko 	abort_fr->cmd = MFI_CMD_ABORT;
3896665484d8SDoug Ambrisko 	abort_fr->cmd_status = 0xFF;
3897665484d8SDoug Ambrisko 	abort_fr->flags = 0;
3898665484d8SDoug Ambrisko 	abort_fr->abort_context = cmd_to_abort->index;
3899665484d8SDoug Ambrisko 	abort_fr->abort_mfi_phys_addr_lo = cmd_to_abort->frame_phys_addr;
3900665484d8SDoug Ambrisko 	abort_fr->abort_mfi_phys_addr_hi = 0;
3901665484d8SDoug Ambrisko 
3902665484d8SDoug Ambrisko 	cmd->sync_cmd = 1;
3903665484d8SDoug Ambrisko 	cmd->cmd_status = 0xFF;
3904665484d8SDoug Ambrisko 
3905665484d8SDoug Ambrisko 	if (mrsas_issue_dcmd(sc, cmd)) {
3906665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Fail to send abort command.\n");
3907665484d8SDoug Ambrisko 		return (1);
3908665484d8SDoug Ambrisko 	}
3909665484d8SDoug Ambrisko 	/* Wait for this cmd to complete */
3910665484d8SDoug Ambrisko 	sc->chan = (void *)&cmd;
3911665484d8SDoug Ambrisko 	while (1) {
3912665484d8SDoug Ambrisko 		if (cmd->cmd_status == 0xFF) {
3913665484d8SDoug Ambrisko 			tsleep((void *)&sc->chan, 0, "mrsas_sleep", hz);
39148e727371SKashyap D Desai 		} else
3915665484d8SDoug Ambrisko 			break;
3916665484d8SDoug Ambrisko 		total_time++;
3917665484d8SDoug Ambrisko 		if (total_time >= max_wait) {
3918665484d8SDoug Ambrisko 			device_printf(sc->mrsas_dev, "Abort cmd timed out after %d sec.\n", max_wait);
3919665484d8SDoug Ambrisko 			retcode = 1;
3920665484d8SDoug Ambrisko 			break;
3921665484d8SDoug Ambrisko 		}
3922665484d8SDoug Ambrisko 	}
3923665484d8SDoug Ambrisko 
3924665484d8SDoug Ambrisko 	cmd->sync_cmd = 0;
3925665484d8SDoug Ambrisko 	mrsas_release_mfi_cmd(cmd);
3926665484d8SDoug Ambrisko 	return (retcode);
3927665484d8SDoug Ambrisko }
3928665484d8SDoug Ambrisko 
39298e727371SKashyap D Desai /*
39308e727371SKashyap D Desai  * mrsas_complete_abort:      Completes aborting a command input:
39318e727371SKashyap D Desai  * Adapter soft state Cmd that was issued to abort another cmd
3932665484d8SDoug Ambrisko  *
39338e727371SKashyap D Desai  * The mrsas_issue_blocked_abort_cmd() function waits for the command status to
39348e727371SKashyap D Desai  * change after sending the command.  This function is called from
3935665484d8SDoug Ambrisko  * mrsas_complete_mptmfi_passthru() to wake up the sleep thread associated.
3936665484d8SDoug Ambrisko  */
39378e727371SKashyap D Desai void
39388e727371SKashyap D Desai mrsas_complete_abort(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd)
3939665484d8SDoug Ambrisko {
3940665484d8SDoug Ambrisko 	if (cmd->sync_cmd) {
3941665484d8SDoug Ambrisko 		cmd->sync_cmd = 0;
3942665484d8SDoug Ambrisko 		cmd->cmd_status = 0;
3943665484d8SDoug Ambrisko 		sc->chan = (void *)&cmd;
3944665484d8SDoug Ambrisko 		wakeup_one((void *)&sc->chan);
3945665484d8SDoug Ambrisko 	}
3946665484d8SDoug Ambrisko 	return;
3947665484d8SDoug Ambrisko }
3948665484d8SDoug Ambrisko 
39498e727371SKashyap D Desai /*
39508e727371SKashyap D Desai  * mrsas_aen_handler:	AEN processing callback function from thread context
3951665484d8SDoug Ambrisko  * input:				Adapter soft state
3952665484d8SDoug Ambrisko  *
39538e727371SKashyap D Desai  * Asynchronous event handler
3954665484d8SDoug Ambrisko  */
39558e727371SKashyap D Desai void
39568e727371SKashyap D Desai mrsas_aen_handler(struct mrsas_softc *sc)
3957665484d8SDoug Ambrisko {
3958665484d8SDoug Ambrisko 	union mrsas_evt_class_locale class_locale;
3959665484d8SDoug Ambrisko 	int doscan = 0;
3960665484d8SDoug Ambrisko 	u_int32_t seq_num;
3961665484d8SDoug Ambrisko 	int error;
3962665484d8SDoug Ambrisko 
3963665484d8SDoug Ambrisko 	if (!sc) {
3964665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "invalid instance!\n");
3965665484d8SDoug Ambrisko 		return;
3966665484d8SDoug Ambrisko 	}
3967665484d8SDoug Ambrisko 	if (sc->evt_detail_mem) {
3968665484d8SDoug Ambrisko 		switch (sc->evt_detail_mem->code) {
3969665484d8SDoug Ambrisko 		case MR_EVT_PD_INSERTED:
3970665484d8SDoug Ambrisko 			mrsas_get_pd_list(sc);
3971665484d8SDoug Ambrisko 			mrsas_bus_scan_sim(sc, sc->sim_1);
3972665484d8SDoug Ambrisko 			doscan = 0;
3973665484d8SDoug Ambrisko 			break;
3974665484d8SDoug Ambrisko 		case MR_EVT_PD_REMOVED:
3975665484d8SDoug Ambrisko 			mrsas_get_pd_list(sc);
3976665484d8SDoug Ambrisko 			mrsas_bus_scan_sim(sc, sc->sim_1);
3977665484d8SDoug Ambrisko 			doscan = 0;
3978665484d8SDoug Ambrisko 			break;
3979665484d8SDoug Ambrisko 		case MR_EVT_LD_OFFLINE:
3980665484d8SDoug Ambrisko 		case MR_EVT_CFG_CLEARED:
3981665484d8SDoug Ambrisko 		case MR_EVT_LD_DELETED:
3982665484d8SDoug Ambrisko 			mrsas_bus_scan_sim(sc, sc->sim_0);
3983665484d8SDoug Ambrisko 			doscan = 0;
3984665484d8SDoug Ambrisko 			break;
3985665484d8SDoug Ambrisko 		case MR_EVT_LD_CREATED:
3986665484d8SDoug Ambrisko 			mrsas_get_ld_list(sc);
3987665484d8SDoug Ambrisko 			mrsas_bus_scan_sim(sc, sc->sim_0);
3988665484d8SDoug Ambrisko 			doscan = 0;
3989665484d8SDoug Ambrisko 			break;
3990665484d8SDoug Ambrisko 		case MR_EVT_CTRL_HOST_BUS_SCAN_REQUESTED:
3991665484d8SDoug Ambrisko 		case MR_EVT_FOREIGN_CFG_IMPORTED:
3992665484d8SDoug Ambrisko 		case MR_EVT_LD_STATE_CHANGE:
3993665484d8SDoug Ambrisko 			doscan = 1;
3994665484d8SDoug Ambrisko 			break;
3995665484d8SDoug Ambrisko 		default:
3996665484d8SDoug Ambrisko 			doscan = 0;
3997665484d8SDoug Ambrisko 			break;
3998665484d8SDoug Ambrisko 		}
3999665484d8SDoug Ambrisko 	} else {
4000665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "invalid evt_detail\n");
4001665484d8SDoug Ambrisko 		return;
4002665484d8SDoug Ambrisko 	}
4003665484d8SDoug Ambrisko 	if (doscan) {
4004665484d8SDoug Ambrisko 		mrsas_get_pd_list(sc);
4005665484d8SDoug Ambrisko 		mrsas_dprint(sc, MRSAS_AEN, "scanning ...sim 1\n");
4006665484d8SDoug Ambrisko 		mrsas_bus_scan_sim(sc, sc->sim_1);
4007665484d8SDoug Ambrisko 		mrsas_get_ld_list(sc);
4008665484d8SDoug Ambrisko 		mrsas_dprint(sc, MRSAS_AEN, "scanning ...sim 0\n");
4009665484d8SDoug Ambrisko 		mrsas_bus_scan_sim(sc, sc->sim_0);
4010665484d8SDoug Ambrisko 	}
4011665484d8SDoug Ambrisko 	seq_num = sc->evt_detail_mem->seq_num + 1;
4012665484d8SDoug Ambrisko 
40138e727371SKashyap D Desai 	/* Register AEN with FW for latest sequence number plus 1 */
4014665484d8SDoug Ambrisko 	class_locale.members.reserved = 0;
4015665484d8SDoug Ambrisko 	class_locale.members.locale = MR_EVT_LOCALE_ALL;
4016665484d8SDoug Ambrisko 	class_locale.members.class = MR_EVT_CLASS_DEBUG;
4017665484d8SDoug Ambrisko 
4018665484d8SDoug Ambrisko 	if (sc->aen_cmd != NULL)
4019665484d8SDoug Ambrisko 		return;
4020665484d8SDoug Ambrisko 
4021665484d8SDoug Ambrisko 	mtx_lock(&sc->aen_lock);
4022665484d8SDoug Ambrisko 	error = mrsas_register_aen(sc, seq_num,
4023665484d8SDoug Ambrisko 	    class_locale.word);
4024665484d8SDoug Ambrisko 	mtx_unlock(&sc->aen_lock);
4025665484d8SDoug Ambrisko 
4026665484d8SDoug Ambrisko 	if (error)
4027665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "register aen failed error %x\n", error);
4028665484d8SDoug Ambrisko 
4029665484d8SDoug Ambrisko }
4030665484d8SDoug Ambrisko 
4031665484d8SDoug Ambrisko 
40328e727371SKashyap D Desai /*
4033665484d8SDoug Ambrisko  * mrsas_complete_aen:	Completes AEN command
4034665484d8SDoug Ambrisko  * input:				Adapter soft state
4035665484d8SDoug Ambrisko  * 						Cmd that was issued to abort another cmd
4036665484d8SDoug Ambrisko  *
40378e727371SKashyap D Desai  * This function will be called from ISR and will continue event processing from
40388e727371SKashyap D Desai  * thread context by enqueuing task in ev_tq (callback function
40398e727371SKashyap D Desai  * "mrsas_aen_handler").
4040665484d8SDoug Ambrisko  */
40418e727371SKashyap D Desai void
40428e727371SKashyap D Desai mrsas_complete_aen(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd)
4043665484d8SDoug Ambrisko {
4044665484d8SDoug Ambrisko 	/*
40458e727371SKashyap D Desai 	 * Don't signal app if it is just an aborted previously registered
40468e727371SKashyap D Desai 	 * aen
4047665484d8SDoug Ambrisko 	 */
4048665484d8SDoug Ambrisko 	if ((!cmd->abort_aen) && (sc->remove_in_progress == 0)) {
4049da011113SKashyap D Desai 		sc->mrsas_aen_triggered = 1;
4050da011113SKashyap D Desai 		if (sc->mrsas_poll_waiting) {
4051da011113SKashyap D Desai 			sc->mrsas_poll_waiting = 0;
4052da011113SKashyap D Desai 			selwakeup(&sc->mrsas_select);
4053da011113SKashyap D Desai 		}
40548e727371SKashyap D Desai 	} else
4055665484d8SDoug Ambrisko 		cmd->abort_aen = 0;
4056665484d8SDoug Ambrisko 
4057665484d8SDoug Ambrisko 	sc->aen_cmd = NULL;
4058665484d8SDoug Ambrisko 	mrsas_release_mfi_cmd(cmd);
4059665484d8SDoug Ambrisko 
4060665484d8SDoug Ambrisko 	if (!sc->remove_in_progress)
4061665484d8SDoug Ambrisko 		taskqueue_enqueue(sc->ev_tq, &sc->ev_task);
4062665484d8SDoug Ambrisko 
4063665484d8SDoug Ambrisko 	return;
4064665484d8SDoug Ambrisko }
4065665484d8SDoug Ambrisko 
4066665484d8SDoug Ambrisko static device_method_t mrsas_methods[] = {
4067665484d8SDoug Ambrisko 	DEVMETHOD(device_probe, mrsas_probe),
4068665484d8SDoug Ambrisko 	DEVMETHOD(device_attach, mrsas_attach),
4069665484d8SDoug Ambrisko 	DEVMETHOD(device_detach, mrsas_detach),
4070665484d8SDoug Ambrisko 	DEVMETHOD(device_suspend, mrsas_suspend),
4071665484d8SDoug Ambrisko 	DEVMETHOD(device_resume, mrsas_resume),
4072665484d8SDoug Ambrisko 	DEVMETHOD(bus_print_child, bus_generic_print_child),
4073665484d8SDoug Ambrisko 	DEVMETHOD(bus_driver_added, bus_generic_driver_added),
4074665484d8SDoug Ambrisko 	{0, 0}
4075665484d8SDoug Ambrisko };
4076665484d8SDoug Ambrisko 
4077665484d8SDoug Ambrisko static driver_t mrsas_driver = {
4078665484d8SDoug Ambrisko 	"mrsas",
4079665484d8SDoug Ambrisko 	mrsas_methods,
4080665484d8SDoug Ambrisko 	sizeof(struct mrsas_softc)
4081665484d8SDoug Ambrisko };
4082665484d8SDoug Ambrisko 
4083665484d8SDoug Ambrisko static devclass_t mrsas_devclass;
40848e727371SKashyap D Desai 
4085665484d8SDoug Ambrisko DRIVER_MODULE(mrsas, pci, mrsas_driver, mrsas_devclass, 0, 0);
4086665484d8SDoug Ambrisko MODULE_DEPEND(mrsas, cam, 1, 1, 1);
4087