xref: /freebsd/sys/dev/mrsas/mrsas.c (revision d18d1b472d32109f251c8af231bc415dcb0d4bd8)
1665484d8SDoug Ambrisko /*
2665484d8SDoug Ambrisko  * Copyright (c) 2014, LSI Corp.
3665484d8SDoug Ambrisko  * All rights reserved.
4665484d8SDoug Ambrisko  * Author: Marian Choy
5665484d8SDoug Ambrisko  * Support: freebsdraid@lsi.com
6665484d8SDoug Ambrisko  *
7665484d8SDoug Ambrisko  * Redistribution and use in source and binary forms, with or without
8665484d8SDoug Ambrisko  * modification, are permitted provided that the following conditions
9665484d8SDoug Ambrisko  * are met:
10665484d8SDoug Ambrisko  *
11665484d8SDoug Ambrisko  * 1. Redistributions of source code must retain the above copyright
12665484d8SDoug Ambrisko  *    notice, this list of conditions and the following disclaimer.
13665484d8SDoug Ambrisko  * 2. Redistributions in binary form must reproduce the above copyright
14665484d8SDoug Ambrisko  *    notice, this list of conditions and the following disclaimer in
15665484d8SDoug Ambrisko  *    the documentation and/or other materials provided with the
16665484d8SDoug Ambrisko  *    distribution.
17665484d8SDoug Ambrisko  * 3. Neither the name of the <ORGANIZATION> nor the names of its
18665484d8SDoug Ambrisko  *    contributors may be used to endorse or promote products derived
19665484d8SDoug Ambrisko  *    from this software without specific prior written permission.
20665484d8SDoug Ambrisko  *
21665484d8SDoug Ambrisko  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
22665484d8SDoug Ambrisko  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
23665484d8SDoug Ambrisko  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
24665484d8SDoug Ambrisko  * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
25665484d8SDoug Ambrisko  * COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
26665484d8SDoug Ambrisko  * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
27665484d8SDoug Ambrisko  * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
28665484d8SDoug Ambrisko  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
29665484d8SDoug Ambrisko  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30665484d8SDoug Ambrisko  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
31665484d8SDoug Ambrisko  * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
32665484d8SDoug Ambrisko  * POSSIBILITY OF SUCH DAMAGE.
33665484d8SDoug Ambrisko  *
34665484d8SDoug Ambrisko  * The views and conclusions contained in the software and documentation
35665484d8SDoug Ambrisko  * are those of the authors and should not be interpreted as representing
36665484d8SDoug Ambrisko  * official policies,either expressed or implied, of the FreeBSD Project.
37665484d8SDoug Ambrisko  *
38665484d8SDoug Ambrisko  * Send feedback to: <megaraidfbsd@lsi.com>
39665484d8SDoug Ambrisko  * Mail to: LSI Corporation, 1621 Barber Lane, Milpitas, CA 95035
40665484d8SDoug Ambrisko  *    ATTN: MegaRaid FreeBSD
41665484d8SDoug Ambrisko  *
42665484d8SDoug Ambrisko  */
43665484d8SDoug Ambrisko 
44665484d8SDoug Ambrisko #include <sys/cdefs.h>
45665484d8SDoug Ambrisko __FBSDID("$FreeBSD$");
46665484d8SDoug Ambrisko 
47665484d8SDoug Ambrisko #include <dev/mrsas/mrsas.h>
48665484d8SDoug Ambrisko #include <dev/mrsas/mrsas_ioctl.h>
49665484d8SDoug Ambrisko 
50665484d8SDoug Ambrisko #include <cam/cam.h>
51665484d8SDoug Ambrisko #include <cam/cam_ccb.h>
52665484d8SDoug Ambrisko 
53665484d8SDoug Ambrisko #include <sys/sysctl.h>
54665484d8SDoug Ambrisko #include <sys/types.h>
55665484d8SDoug Ambrisko #include <sys/kthread.h>
56665484d8SDoug Ambrisko #include <sys/taskqueue.h>
57*d18d1b47SKashyap D Desai #include <sys/smp.h>
58665484d8SDoug Ambrisko 
59665484d8SDoug Ambrisko 
60665484d8SDoug Ambrisko /*
61665484d8SDoug Ambrisko  * Function prototypes
62665484d8SDoug Ambrisko  */
63665484d8SDoug Ambrisko static d_open_t     mrsas_open;
64665484d8SDoug Ambrisko static d_close_t    mrsas_close;
65665484d8SDoug Ambrisko static d_read_t     mrsas_read;
66665484d8SDoug Ambrisko static d_write_t    mrsas_write;
67665484d8SDoug Ambrisko static d_ioctl_t    mrsas_ioctl;
68665484d8SDoug Ambrisko 
69536094dcSKashyap D Desai static struct mrsas_mgmt_info mrsas_mgmt_info;
70665484d8SDoug Ambrisko static struct mrsas_ident *mrsas_find_ident(device_t);
71*d18d1b47SKashyap D Desai static int mrsas_setup_msix(struct mrsas_softc *sc);
72*d18d1b47SKashyap D Desai static int mrsas_allocate_msix(struct mrsas_softc *sc);
73665484d8SDoug Ambrisko static void mrsas_shutdown_ctlr(struct mrsas_softc *sc, u_int32_t opcode);
74665484d8SDoug Ambrisko static void mrsas_flush_cache(struct mrsas_softc *sc);
75665484d8SDoug Ambrisko static void mrsas_reset_reply_desc(struct mrsas_softc *sc);
76665484d8SDoug Ambrisko static void mrsas_ocr_thread(void *arg);
77665484d8SDoug Ambrisko static int mrsas_get_map_info(struct mrsas_softc *sc);
78665484d8SDoug Ambrisko static int mrsas_get_ld_map_info(struct mrsas_softc *sc);
79665484d8SDoug Ambrisko static int mrsas_sync_map_info(struct mrsas_softc *sc);
80665484d8SDoug Ambrisko static int mrsas_get_pd_list(struct mrsas_softc *sc);
81665484d8SDoug Ambrisko static int mrsas_get_ld_list(struct mrsas_softc *sc);
82665484d8SDoug Ambrisko static int mrsas_setup_irq(struct mrsas_softc *sc);
83665484d8SDoug Ambrisko static int mrsas_alloc_mem(struct mrsas_softc *sc);
84665484d8SDoug Ambrisko static int mrsas_init_fw(struct mrsas_softc *sc);
85665484d8SDoug Ambrisko static int mrsas_setup_raidmap(struct mrsas_softc *sc);
86*d18d1b47SKashyap D Desai static int mrsas_complete_cmd(struct mrsas_softc *sc, u_int32_t MSIxIndex);
87665484d8SDoug Ambrisko static int mrsas_clear_intr(struct mrsas_softc *sc);
88665484d8SDoug Ambrisko static int mrsas_get_ctrl_info(struct mrsas_softc *sc,
89665484d8SDoug Ambrisko                           struct mrsas_ctrl_info *ctrl_info);
90665484d8SDoug Ambrisko static int mrsas_issue_blocked_abort_cmd(struct mrsas_softc *sc,
91665484d8SDoug Ambrisko                          struct mrsas_mfi_cmd *cmd_to_abort);
92665484d8SDoug Ambrisko u_int32_t mrsas_read_reg(struct mrsas_softc *sc, int offset);
93665484d8SDoug Ambrisko u_int8_t mrsas_build_mptmfi_passthru(struct mrsas_softc *sc,
94665484d8SDoug Ambrisko                          struct mrsas_mfi_cmd *mfi_cmd);
95665484d8SDoug Ambrisko int mrsas_transition_to_ready(struct mrsas_softc *sc, int ocr);
96665484d8SDoug Ambrisko int mrsas_init_adapter(struct mrsas_softc *sc);
97665484d8SDoug Ambrisko int mrsas_alloc_mpt_cmds(struct mrsas_softc *sc);
98665484d8SDoug Ambrisko int mrsas_alloc_ioc_cmd(struct mrsas_softc *sc);
99665484d8SDoug Ambrisko int mrsas_alloc_ctlr_info_cmd(struct mrsas_softc *sc);
100665484d8SDoug Ambrisko int mrsas_ioc_init(struct mrsas_softc *sc);
101665484d8SDoug Ambrisko int mrsas_bus_scan(struct mrsas_softc *sc);
102665484d8SDoug Ambrisko int mrsas_issue_dcmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd);
103665484d8SDoug Ambrisko int mrsas_issue_polled(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd);
104665484d8SDoug Ambrisko int mrsas_reset_ctrl(struct mrsas_softc *sc);
105665484d8SDoug Ambrisko int mrsas_wait_for_outstanding(struct mrsas_softc *sc);
106665484d8SDoug Ambrisko int mrsas_issue_blocked_cmd(struct mrsas_softc *sc,
107665484d8SDoug Ambrisko                           struct mrsas_mfi_cmd *cmd);
108665484d8SDoug Ambrisko int mrsas_alloc_tmp_dcmd(struct mrsas_softc *sc, struct mrsas_tmp_dcmd *tcmd,
109665484d8SDoug Ambrisko                           int size);
110665484d8SDoug Ambrisko void mrsas_release_mfi_cmd(struct mrsas_mfi_cmd *cmd);
111665484d8SDoug Ambrisko void mrsas_wakeup(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd);
112665484d8SDoug Ambrisko void mrsas_complete_aen(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd);
113665484d8SDoug Ambrisko void mrsas_complete_abort(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd);
114665484d8SDoug Ambrisko void mrsas_disable_intr(struct mrsas_softc *sc);
115665484d8SDoug Ambrisko void mrsas_enable_intr(struct mrsas_softc *sc);
116665484d8SDoug Ambrisko void mrsas_free_ioc_cmd(struct mrsas_softc *sc);
117665484d8SDoug Ambrisko void mrsas_free_mem(struct mrsas_softc *sc);
118665484d8SDoug Ambrisko void mrsas_free_tmp_dcmd(struct mrsas_tmp_dcmd *tmp);
119665484d8SDoug Ambrisko void mrsas_isr(void *arg);
120665484d8SDoug Ambrisko void mrsas_teardown_intr(struct mrsas_softc *sc);
121665484d8SDoug Ambrisko void mrsas_addr_cb(void *arg, bus_dma_segment_t *segs, int nsegs, int error);
122665484d8SDoug Ambrisko void mrsas_kill_hba (struct mrsas_softc *sc);
123665484d8SDoug Ambrisko void mrsas_aen_handler(struct mrsas_softc *sc);
124665484d8SDoug Ambrisko void mrsas_write_reg(struct mrsas_softc *sc, int offset,
125665484d8SDoug Ambrisko                           u_int32_t value);
126665484d8SDoug Ambrisko void mrsas_fire_cmd(struct mrsas_softc *sc, u_int32_t req_desc_lo,
127665484d8SDoug Ambrisko                           u_int32_t req_desc_hi);
128665484d8SDoug Ambrisko void mrsas_free_ctlr_info_cmd(struct mrsas_softc *sc);
129665484d8SDoug Ambrisko void mrsas_complete_mptmfi_passthru(struct mrsas_softc *sc,
130665484d8SDoug Ambrisko                           struct mrsas_mfi_cmd *cmd, u_int8_t status);
131665484d8SDoug Ambrisko void mrsas_map_mpt_cmd_status(struct mrsas_mpt_cmd *cmd, u_int8_t status,
132665484d8SDoug Ambrisko                           u_int8_t extStatus);
133665484d8SDoug Ambrisko struct mrsas_mfi_cmd* mrsas_get_mfi_cmd(struct mrsas_softc *sc);
134665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION * mrsas_build_mpt_cmd(struct mrsas_softc *sc,
135665484d8SDoug Ambrisko                           struct mrsas_mfi_cmd *cmd);
136665484d8SDoug Ambrisko 
137665484d8SDoug Ambrisko extern int mrsas_cam_attach(struct mrsas_softc *sc);
138665484d8SDoug Ambrisko extern void mrsas_cam_detach(struct mrsas_softc *sc);
139665484d8SDoug Ambrisko extern void mrsas_cmd_done(struct mrsas_softc *sc, struct mrsas_mpt_cmd *cmd);
140665484d8SDoug Ambrisko extern void mrsas_free_frame(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd);
141665484d8SDoug Ambrisko extern int mrsas_alloc_mfi_cmds(struct mrsas_softc *sc);
142665484d8SDoug Ambrisko extern void mrsas_release_mpt_cmd(struct mrsas_mpt_cmd *cmd);
143665484d8SDoug Ambrisko extern struct mrsas_mpt_cmd *mrsas_get_mpt_cmd(struct mrsas_softc *sc);
144536094dcSKashyap D Desai extern int mrsas_passthru(struct mrsas_softc *sc, void *arg, u_long ioctlCmd);
145665484d8SDoug Ambrisko extern uint8_t MR_ValidateMapInfo(struct mrsas_softc *sc);
1464799d485SKashyap D Desai extern u_int16_t MR_GetLDTgtId(u_int32_t ld, MR_DRV_RAID_MAP_ALL *map);
1474799d485SKashyap D Desai extern MR_LD_RAID *MR_LdRaidGet(u_int32_t ld, MR_DRV_RAID_MAP_ALL *map);
148665484d8SDoug Ambrisko extern void mrsas_xpt_freeze(struct mrsas_softc *sc);
149665484d8SDoug Ambrisko extern void mrsas_xpt_release(struct mrsas_softc *sc);
150665484d8SDoug Ambrisko extern MRSAS_REQUEST_DESCRIPTOR_UNION *mrsas_get_request_desc(struct mrsas_softc *sc,
151665484d8SDoug Ambrisko                          u_int16_t index);
152665484d8SDoug Ambrisko extern int mrsas_bus_scan_sim(struct mrsas_softc *sc, struct cam_sim *sim);
153665484d8SDoug Ambrisko static int mrsas_alloc_evt_log_info_cmd(struct mrsas_softc *sc);
154665484d8SDoug Ambrisko static void mrsas_free_evt_log_info_cmd(struct mrsas_softc *sc);
155665484d8SDoug Ambrisko SYSCTL_NODE(_hw, OID_AUTO, mrsas, CTLFLAG_RD, 0, "MRSAS Driver Parameters");
156665484d8SDoug Ambrisko 
157665484d8SDoug Ambrisko /**
158665484d8SDoug Ambrisko  * PCI device struct and table
159665484d8SDoug Ambrisko  *
160665484d8SDoug Ambrisko  */
161665484d8SDoug Ambrisko typedef struct mrsas_ident {
162665484d8SDoug Ambrisko     uint16_t    vendor;
163665484d8SDoug Ambrisko     uint16_t    device;
164665484d8SDoug Ambrisko     uint16_t    subvendor;
165665484d8SDoug Ambrisko     uint16_t    subdevice;
166665484d8SDoug Ambrisko     const char  *desc;
167665484d8SDoug Ambrisko } MRSAS_CTLR_ID;
168665484d8SDoug Ambrisko 
169665484d8SDoug Ambrisko MRSAS_CTLR_ID device_table[] = {
170665484d8SDoug Ambrisko     {0x1000, MRSAS_TBOLT, 0xffff, 0xffff, "LSI Thunderbolt SAS Controller"},
171665484d8SDoug Ambrisko     {0x1000, MRSAS_INVADER, 0xffff, 0xffff, "LSI Invader SAS Controller"},
172665484d8SDoug Ambrisko     {0x1000, MRSAS_FURY, 0xffff, 0xffff, "LSI Fury SAS Controller"},
173665484d8SDoug Ambrisko     {0, 0, 0, 0, NULL}
174665484d8SDoug Ambrisko };
175665484d8SDoug Ambrisko 
176665484d8SDoug Ambrisko /**
177665484d8SDoug Ambrisko  * Character device entry points
178665484d8SDoug Ambrisko  *
179665484d8SDoug Ambrisko  */
180665484d8SDoug Ambrisko static struct cdevsw mrsas_cdevsw = {
181665484d8SDoug Ambrisko     .d_version =    D_VERSION,
182665484d8SDoug Ambrisko     .d_open =   mrsas_open,
183665484d8SDoug Ambrisko     .d_close =  mrsas_close,
184665484d8SDoug Ambrisko     .d_read =   mrsas_read,
185665484d8SDoug Ambrisko     .d_write =  mrsas_write,
186665484d8SDoug Ambrisko     .d_ioctl =  mrsas_ioctl,
187665484d8SDoug Ambrisko     .d_name =   "mrsas",
188665484d8SDoug Ambrisko };
189665484d8SDoug Ambrisko 
190665484d8SDoug Ambrisko MALLOC_DEFINE(M_MRSAS, "mrsasbuf", "Buffers for the MRSAS driver");
191665484d8SDoug Ambrisko 
192665484d8SDoug Ambrisko /**
193665484d8SDoug Ambrisko  * In the cdevsw routines, we find our softc by using the si_drv1 member
194665484d8SDoug Ambrisko  * of struct cdev.  We set this variable to point to our softc in our
195665484d8SDoug Ambrisko  * attach routine when we create the /dev entry.
196665484d8SDoug Ambrisko  */
197665484d8SDoug Ambrisko int
198665484d8SDoug Ambrisko mrsas_open(struct cdev *dev, int oflags, int devtype, d_thread_t *td)
199665484d8SDoug Ambrisko {
200665484d8SDoug Ambrisko     struct mrsas_softc *sc;
201665484d8SDoug Ambrisko 
202665484d8SDoug Ambrisko     sc = dev->si_drv1;
203665484d8SDoug Ambrisko     return (0);
204665484d8SDoug Ambrisko }
205665484d8SDoug Ambrisko 
206665484d8SDoug Ambrisko int
207665484d8SDoug Ambrisko mrsas_close(struct cdev *dev, int fflag, int devtype, d_thread_t *td)
208665484d8SDoug Ambrisko {
209665484d8SDoug Ambrisko     struct mrsas_softc *sc;
210665484d8SDoug Ambrisko 
211665484d8SDoug Ambrisko     sc = dev->si_drv1;
212665484d8SDoug Ambrisko     return (0);
213665484d8SDoug Ambrisko }
214665484d8SDoug Ambrisko 
215665484d8SDoug Ambrisko int
216665484d8SDoug Ambrisko mrsas_read(struct cdev *dev, struct uio *uio, int ioflag)
217665484d8SDoug Ambrisko {
218665484d8SDoug Ambrisko     struct mrsas_softc *sc;
219665484d8SDoug Ambrisko 
220665484d8SDoug Ambrisko     sc = dev->si_drv1;
221665484d8SDoug Ambrisko     return (0);
222665484d8SDoug Ambrisko }
223665484d8SDoug Ambrisko int
224665484d8SDoug Ambrisko mrsas_write(struct cdev *dev, struct uio *uio, int ioflag)
225665484d8SDoug Ambrisko {
226665484d8SDoug Ambrisko     struct mrsas_softc *sc;
227665484d8SDoug Ambrisko 
228665484d8SDoug Ambrisko     sc = dev->si_drv1;
229665484d8SDoug Ambrisko     return (0);
230665484d8SDoug Ambrisko }
231665484d8SDoug Ambrisko 
232665484d8SDoug Ambrisko /**
233665484d8SDoug Ambrisko  * Register Read/Write Functions
234665484d8SDoug Ambrisko  *
235665484d8SDoug Ambrisko  */
236665484d8SDoug Ambrisko void
237665484d8SDoug Ambrisko mrsas_write_reg(struct mrsas_softc *sc, int offset,
238665484d8SDoug Ambrisko                   u_int32_t value)
239665484d8SDoug Ambrisko {
240665484d8SDoug Ambrisko     bus_space_tag_t         bus_tag = sc->bus_tag;
241665484d8SDoug Ambrisko     bus_space_handle_t      bus_handle = sc->bus_handle;
242665484d8SDoug Ambrisko 
243665484d8SDoug Ambrisko     bus_space_write_4(bus_tag, bus_handle, offset, value);
244665484d8SDoug Ambrisko }
245665484d8SDoug Ambrisko 
246665484d8SDoug Ambrisko u_int32_t
247665484d8SDoug Ambrisko mrsas_read_reg(struct mrsas_softc *sc, int offset)
248665484d8SDoug Ambrisko {
249665484d8SDoug Ambrisko     bus_space_tag_t bus_tag = sc->bus_tag;
250665484d8SDoug Ambrisko     bus_space_handle_t bus_handle = sc->bus_handle;
251665484d8SDoug Ambrisko 
252665484d8SDoug Ambrisko     return((u_int32_t)bus_space_read_4(bus_tag, bus_handle, offset));
253665484d8SDoug Ambrisko }
254665484d8SDoug Ambrisko 
255665484d8SDoug Ambrisko 
256665484d8SDoug Ambrisko /**
257665484d8SDoug Ambrisko  * Interrupt Disable/Enable/Clear Functions
258665484d8SDoug Ambrisko  *
259665484d8SDoug Ambrisko  */
260665484d8SDoug Ambrisko void mrsas_disable_intr(struct mrsas_softc *sc)
261665484d8SDoug Ambrisko {
262665484d8SDoug Ambrisko     u_int32_t mask = 0xFFFFFFFF;
263665484d8SDoug Ambrisko     u_int32_t status;
264665484d8SDoug Ambrisko 
265665484d8SDoug Ambrisko     mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask), mask);
266665484d8SDoug Ambrisko     /* Dummy read to force pci flush */
267665484d8SDoug Ambrisko     status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask));
268665484d8SDoug Ambrisko }
269665484d8SDoug Ambrisko 
270665484d8SDoug Ambrisko void mrsas_enable_intr(struct mrsas_softc *sc)
271665484d8SDoug Ambrisko {
272665484d8SDoug Ambrisko     u_int32_t mask = MFI_FUSION_ENABLE_INTERRUPT_MASK;
273665484d8SDoug Ambrisko     u_int32_t status;
274665484d8SDoug Ambrisko 
275665484d8SDoug Ambrisko     mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status), ~0);
276665484d8SDoug Ambrisko     status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status));
277665484d8SDoug Ambrisko 
278665484d8SDoug Ambrisko     mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask), ~mask);
279665484d8SDoug Ambrisko     status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_mask));
280665484d8SDoug Ambrisko }
281665484d8SDoug Ambrisko 
282665484d8SDoug Ambrisko static int mrsas_clear_intr(struct mrsas_softc *sc)
283665484d8SDoug Ambrisko {
284665484d8SDoug Ambrisko     u_int32_t status, fw_status, fw_state;
285665484d8SDoug Ambrisko 
286665484d8SDoug Ambrisko     /* Read received interrupt */
287665484d8SDoug Ambrisko     status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status));
288665484d8SDoug Ambrisko 
289665484d8SDoug Ambrisko     /* If FW state change interrupt is received, write to it again to clear */
290665484d8SDoug Ambrisko     if (status & MRSAS_FW_STATE_CHNG_INTERRUPT) {
291665484d8SDoug Ambrisko         fw_status = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
292665484d8SDoug Ambrisko                                    outbound_scratch_pad));
293665484d8SDoug Ambrisko         fw_state = fw_status & MFI_STATE_MASK;
294665484d8SDoug Ambrisko         if (fw_state == MFI_STATE_FAULT) {
295665484d8SDoug Ambrisko             device_printf(sc->mrsas_dev, "FW is in FAULT state!\n");
296665484d8SDoug Ambrisko             if(sc->ocr_thread_active)
297665484d8SDoug Ambrisko                 wakeup(&sc->ocr_chan);
298665484d8SDoug Ambrisko         }
299665484d8SDoug Ambrisko         mrsas_write_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status), status);
300665484d8SDoug Ambrisko         mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_intr_status));
301665484d8SDoug Ambrisko         return(1);
302665484d8SDoug Ambrisko     }
303665484d8SDoug Ambrisko 
304665484d8SDoug Ambrisko     /* Not our interrupt, so just return */
305665484d8SDoug Ambrisko     if (!(status & MFI_FUSION_ENABLE_INTERRUPT_MASK))
306665484d8SDoug Ambrisko         return(0);
307665484d8SDoug Ambrisko 
308665484d8SDoug Ambrisko     /* We got a reply interrupt */
309665484d8SDoug Ambrisko     return(1);
310665484d8SDoug Ambrisko }
311665484d8SDoug Ambrisko 
312665484d8SDoug Ambrisko /**
313665484d8SDoug Ambrisko  * PCI Support Functions
314665484d8SDoug Ambrisko  *
315665484d8SDoug Ambrisko  */
316665484d8SDoug Ambrisko static struct mrsas_ident * mrsas_find_ident(device_t dev)
317665484d8SDoug Ambrisko {
318665484d8SDoug Ambrisko     struct mrsas_ident *pci_device;
319665484d8SDoug Ambrisko 
320665484d8SDoug Ambrisko     for (pci_device=device_table; pci_device->vendor != 0; pci_device++)
321665484d8SDoug Ambrisko     {
322665484d8SDoug Ambrisko         if ((pci_device->vendor == pci_get_vendor(dev)) &&
323665484d8SDoug Ambrisko             (pci_device->device == pci_get_device(dev)) &&
324665484d8SDoug Ambrisko             ((pci_device->subvendor == pci_get_subvendor(dev)) ||
325665484d8SDoug Ambrisko             (pci_device->subvendor == 0xffff)) &&
326665484d8SDoug Ambrisko             ((pci_device->subdevice == pci_get_subdevice(dev)) ||
327665484d8SDoug Ambrisko             (pci_device->subdevice == 0xffff)))
328665484d8SDoug Ambrisko         return (pci_device);
329665484d8SDoug Ambrisko     }
330665484d8SDoug Ambrisko     return (NULL);
331665484d8SDoug Ambrisko }
332665484d8SDoug Ambrisko 
333665484d8SDoug Ambrisko static int mrsas_probe(device_t dev)
334665484d8SDoug Ambrisko {
335665484d8SDoug Ambrisko     static u_int8_t first_ctrl = 1;
336665484d8SDoug Ambrisko     struct mrsas_ident *id;
337665484d8SDoug Ambrisko 
338665484d8SDoug Ambrisko     if ((id = mrsas_find_ident(dev)) != NULL) {
339665484d8SDoug Ambrisko         if (first_ctrl) {
340665484d8SDoug Ambrisko             printf("LSI MegaRAID SAS FreeBSD mrsas driver version: %s\n", MRSAS_VERSION);
341665484d8SDoug Ambrisko             first_ctrl = 0;
342665484d8SDoug Ambrisko         }
343665484d8SDoug Ambrisko         device_set_desc(dev, id->desc);
344665484d8SDoug Ambrisko     	/* between BUS_PROBE_DEFAULT and BUS_PROBE_LOW_PRIORITY */
345665484d8SDoug Ambrisko     	return (-30);
346665484d8SDoug Ambrisko     }
347665484d8SDoug Ambrisko     return (ENXIO);
348665484d8SDoug Ambrisko }
349665484d8SDoug Ambrisko 
350665484d8SDoug Ambrisko /**
351665484d8SDoug Ambrisko  * mrsas_setup_sysctl:  setup sysctl values for mrsas
352665484d8SDoug Ambrisko  * input:               Adapter instance soft state
353665484d8SDoug Ambrisko  *
354665484d8SDoug Ambrisko  * Setup sysctl entries for mrsas driver.
355665484d8SDoug Ambrisko  */
356665484d8SDoug Ambrisko static void
357665484d8SDoug Ambrisko mrsas_setup_sysctl(struct mrsas_softc *sc)
358665484d8SDoug Ambrisko {
359665484d8SDoug Ambrisko     struct sysctl_ctx_list  *sysctl_ctx = NULL;
360665484d8SDoug Ambrisko     struct sysctl_oid       *sysctl_tree = NULL;
361665484d8SDoug Ambrisko     char tmpstr[80], tmpstr2[80];
362665484d8SDoug Ambrisko 
363665484d8SDoug Ambrisko     /*
364665484d8SDoug Ambrisko      * Setup the sysctl variable so the user can change the debug level
365665484d8SDoug Ambrisko      * on the fly.
366665484d8SDoug Ambrisko      */
367665484d8SDoug Ambrisko     snprintf(tmpstr, sizeof(tmpstr), "MRSAS controller %d",
368665484d8SDoug Ambrisko     device_get_unit(sc->mrsas_dev));
369665484d8SDoug Ambrisko     snprintf(tmpstr2, sizeof(tmpstr2), "%d", device_get_unit(sc->mrsas_dev));
370665484d8SDoug Ambrisko 
371665484d8SDoug Ambrisko     sysctl_ctx = device_get_sysctl_ctx(sc->mrsas_dev);
372665484d8SDoug Ambrisko     if (sysctl_ctx != NULL)
373665484d8SDoug Ambrisko         sysctl_tree = device_get_sysctl_tree(sc->mrsas_dev);
374665484d8SDoug Ambrisko 
375665484d8SDoug Ambrisko     if (sysctl_tree == NULL) {
376665484d8SDoug Ambrisko         sysctl_ctx_init(&sc->sysctl_ctx);
377665484d8SDoug Ambrisko         sc->sysctl_tree = SYSCTL_ADD_NODE(&sc->sysctl_ctx,
378665484d8SDoug Ambrisko             SYSCTL_STATIC_CHILDREN(_hw_mrsas), OID_AUTO, tmpstr2,
379665484d8SDoug Ambrisko             CTLFLAG_RD, 0, tmpstr);
380665484d8SDoug Ambrisko         if (sc->sysctl_tree == NULL)
381665484d8SDoug Ambrisko              return;
382665484d8SDoug Ambrisko         sysctl_ctx = &sc->sysctl_ctx;
383665484d8SDoug Ambrisko         sysctl_tree = sc->sysctl_tree;
384665484d8SDoug Ambrisko     }
385665484d8SDoug Ambrisko     SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree),
386665484d8SDoug Ambrisko         OID_AUTO, "disable_ocr", CTLFLAG_RW, &sc->disableOnlineCtrlReset, 0,
387665484d8SDoug Ambrisko         "Disable the use of OCR");
388665484d8SDoug Ambrisko 
389665484d8SDoug Ambrisko     SYSCTL_ADD_STRING(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree),
390665484d8SDoug Ambrisko         OID_AUTO, "driver_version", CTLFLAG_RD, MRSAS_VERSION,
391665484d8SDoug Ambrisko         strlen(MRSAS_VERSION), "driver version");
392665484d8SDoug Ambrisko 
393665484d8SDoug Ambrisko     SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree),
394665484d8SDoug Ambrisko         OID_AUTO, "reset_count", CTLFLAG_RD,
395665484d8SDoug Ambrisko         &sc->reset_count, 0, "number of ocr from start of the day");
396665484d8SDoug Ambrisko 
397665484d8SDoug Ambrisko     SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree),
398665484d8SDoug Ambrisko         OID_AUTO, "fw_outstanding", CTLFLAG_RD,
399665484d8SDoug Ambrisko         &sc->fw_outstanding, 0, "FW outstanding commands");
400665484d8SDoug Ambrisko 
401665484d8SDoug Ambrisko 	SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree),
402665484d8SDoug Ambrisko         OID_AUTO, "io_cmds_highwater", CTLFLAG_RD,
403665484d8SDoug Ambrisko         &sc->io_cmds_highwater, 0, "Max FW outstanding commands");
404665484d8SDoug Ambrisko 
405665484d8SDoug Ambrisko     SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree),
406665484d8SDoug Ambrisko         OID_AUTO, "mrsas_debug", CTLFLAG_RW, &sc->mrsas_debug, 0,
407665484d8SDoug Ambrisko         "Driver debug level");
408665484d8SDoug Ambrisko 
409665484d8SDoug Ambrisko     SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree),
410665484d8SDoug Ambrisko         OID_AUTO, "mrsas_io_timeout", CTLFLAG_RW, &sc->mrsas_io_timeout,
411665484d8SDoug Ambrisko         0, "Driver IO timeout value in mili-second.");
412665484d8SDoug Ambrisko 
413665484d8SDoug Ambrisko     SYSCTL_ADD_UINT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree),
414665484d8SDoug Ambrisko         OID_AUTO, "mrsas_fw_fault_check_delay", CTLFLAG_RW,
415665484d8SDoug Ambrisko         &sc->mrsas_fw_fault_check_delay,
416665484d8SDoug Ambrisko         0, "FW fault check thread delay in seconds. <default is 1 sec>");
417665484d8SDoug Ambrisko 
418665484d8SDoug Ambrisko     SYSCTL_ADD_INT(sysctl_ctx, SYSCTL_CHILDREN(sysctl_tree),
419665484d8SDoug Ambrisko         OID_AUTO, "reset_in_progress", CTLFLAG_RD,
420665484d8SDoug Ambrisko         &sc->reset_in_progress, 0, "ocr in progress status");
421665484d8SDoug Ambrisko 
422665484d8SDoug Ambrisko }
423665484d8SDoug Ambrisko 
424665484d8SDoug Ambrisko /**
425665484d8SDoug Ambrisko  * mrsas_get_tunables:  get tunable parameters.
426665484d8SDoug Ambrisko  * input:               Adapter instance soft state
427665484d8SDoug Ambrisko  *
428665484d8SDoug Ambrisko  * Get tunable parameters. This will help to debug driver at boot time.
429665484d8SDoug Ambrisko  */
430665484d8SDoug Ambrisko static void
431665484d8SDoug Ambrisko mrsas_get_tunables(struct mrsas_softc *sc)
432665484d8SDoug Ambrisko {
433665484d8SDoug Ambrisko     char tmpstr[80];
434665484d8SDoug Ambrisko 
435665484d8SDoug Ambrisko     /* XXX default to some debugging for now */
436665484d8SDoug Ambrisko     sc->mrsas_debug = MRSAS_FAULT;
437665484d8SDoug Ambrisko     sc->mrsas_io_timeout = MRSAS_IO_TIMEOUT;
438665484d8SDoug Ambrisko     sc->mrsas_fw_fault_check_delay = 1;
439665484d8SDoug Ambrisko     sc->reset_count = 0;
440665484d8SDoug Ambrisko     sc->reset_in_progress = 0;
441665484d8SDoug Ambrisko 
442665484d8SDoug Ambrisko     /*
443665484d8SDoug Ambrisko      * Grab the global variables.
444665484d8SDoug Ambrisko      */
445665484d8SDoug Ambrisko     TUNABLE_INT_FETCH("hw.mrsas.debug_level", &sc->mrsas_debug);
446665484d8SDoug Ambrisko 
447665484d8SDoug Ambrisko     /* Grab the unit-instance variables */
448665484d8SDoug Ambrisko     snprintf(tmpstr, sizeof(tmpstr), "dev.mrsas.%d.debug_level",
449665484d8SDoug Ambrisko         device_get_unit(sc->mrsas_dev));
450665484d8SDoug Ambrisko     TUNABLE_INT_FETCH(tmpstr, &sc->mrsas_debug);
451665484d8SDoug Ambrisko }
452665484d8SDoug Ambrisko 
453665484d8SDoug Ambrisko /**
454665484d8SDoug Ambrisko  * mrsas_alloc_evt_log_info cmd:	Allocates memory to get event log information.
455665484d8SDoug Ambrisko  * 								  	Used to get sequence number at driver load time.
456665484d8SDoug Ambrisko  * input:                      	  	Adapter soft state
457665484d8SDoug Ambrisko  *
458665484d8SDoug Ambrisko  * Allocates DMAable memory for the event log info internal command.
459665484d8SDoug Ambrisko  */
460665484d8SDoug Ambrisko int mrsas_alloc_evt_log_info_cmd(struct mrsas_softc *sc)
461665484d8SDoug Ambrisko {
462665484d8SDoug Ambrisko     int el_info_size;
463665484d8SDoug Ambrisko 
464665484d8SDoug Ambrisko     /* Allocate get event log info command */
465665484d8SDoug Ambrisko     el_info_size = sizeof(struct mrsas_evt_log_info);
466665484d8SDoug Ambrisko     if (bus_dma_tag_create( sc->mrsas_parent_tag,   // parent
467665484d8SDoug Ambrisko                             1, 0,                   // algnmnt, boundary
468665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR_32BIT,// lowaddr
469665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR,      // highaddr
470665484d8SDoug Ambrisko                             NULL, NULL,             // filter, filterarg
471665484d8SDoug Ambrisko                             el_info_size,          // maxsize
472665484d8SDoug Ambrisko                             1,                      // msegments
473665484d8SDoug Ambrisko                             el_info_size,          // maxsegsize
474665484d8SDoug Ambrisko                             BUS_DMA_ALLOCNOW,       // flags
475665484d8SDoug Ambrisko                             NULL, NULL,             // lockfunc, lockarg
476665484d8SDoug Ambrisko                             &sc->el_info_tag)) {
477665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot allocate event log info tag\n");
478665484d8SDoug Ambrisko         return (ENOMEM);
479665484d8SDoug Ambrisko     }
480665484d8SDoug Ambrisko     if (bus_dmamem_alloc(sc->el_info_tag, (void **)&sc->el_info_mem,
481665484d8SDoug Ambrisko             BUS_DMA_NOWAIT, &sc->el_info_dmamap)) {
482665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot allocate event log info cmd mem\n");
483665484d8SDoug Ambrisko         return (ENOMEM);
484665484d8SDoug Ambrisko     }
485665484d8SDoug Ambrisko     if (bus_dmamap_load(sc->el_info_tag, sc->el_info_dmamap,
486665484d8SDoug Ambrisko             sc->el_info_mem, el_info_size, mrsas_addr_cb,
487665484d8SDoug Ambrisko             &sc->el_info_phys_addr, BUS_DMA_NOWAIT)) {
488665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot load event log info cmd mem\n");
489665484d8SDoug Ambrisko         return (ENOMEM);
490665484d8SDoug Ambrisko     }
491665484d8SDoug Ambrisko 
492665484d8SDoug Ambrisko     memset(sc->el_info_mem, 0, el_info_size);
493665484d8SDoug Ambrisko     return (0);
494665484d8SDoug Ambrisko }
495665484d8SDoug Ambrisko 
496665484d8SDoug Ambrisko /**
497665484d8SDoug Ambrisko  * mrsas_free_evt_info_cmd: 	Free memory for Event log info command
498665484d8SDoug Ambrisko  * input:                    	Adapter soft state
499665484d8SDoug Ambrisko  *
500665484d8SDoug Ambrisko  * Deallocates memory for the event log info internal command.
501665484d8SDoug Ambrisko  */
502665484d8SDoug Ambrisko void mrsas_free_evt_log_info_cmd(struct mrsas_softc *sc)
503665484d8SDoug Ambrisko {
504665484d8SDoug Ambrisko     if (sc->el_info_phys_addr)
505665484d8SDoug Ambrisko         bus_dmamap_unload(sc->el_info_tag, sc->el_info_dmamap);
506665484d8SDoug Ambrisko     if (sc->el_info_mem != NULL)
507665484d8SDoug Ambrisko         bus_dmamem_free(sc->el_info_tag, sc->el_info_mem, sc->el_info_dmamap);
508665484d8SDoug Ambrisko     if (sc->el_info_tag != NULL)
509665484d8SDoug Ambrisko         bus_dma_tag_destroy(sc->el_info_tag);
510665484d8SDoug Ambrisko }
511665484d8SDoug Ambrisko 
512665484d8SDoug Ambrisko /**
513665484d8SDoug Ambrisko  *  mrsas_get_seq_num:	Get latest event sequence number
514665484d8SDoug Ambrisko  *  @sc:				Adapter soft state
515665484d8SDoug Ambrisko  *  @eli:				Firmware event log sequence number information.
516665484d8SDoug Ambrisko  *						Firmware maintains a log of all events in a non-volatile area.
517665484d8SDoug Ambrisko  *						Driver get the sequence number using DCMD
518665484d8SDoug Ambrisko  *						"MR_DCMD_CTRL_EVENT_GET_INFO" at driver load time.
519665484d8SDoug Ambrisko  */
520665484d8SDoug Ambrisko 
521665484d8SDoug Ambrisko static int
522665484d8SDoug Ambrisko mrsas_get_seq_num(struct mrsas_softc *sc,
523665484d8SDoug Ambrisko 		    struct mrsas_evt_log_info *eli)
524665484d8SDoug Ambrisko {
525665484d8SDoug Ambrisko 	struct mrsas_mfi_cmd *cmd;
526665484d8SDoug Ambrisko 	struct mrsas_dcmd_frame *dcmd;
527665484d8SDoug Ambrisko 
528665484d8SDoug Ambrisko 	cmd =  mrsas_get_mfi_cmd(sc);
529665484d8SDoug Ambrisko 
530665484d8SDoug Ambrisko 	if (!cmd) {
531665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Failed to get a free cmd\n");
532665484d8SDoug Ambrisko 		return -ENOMEM;
533665484d8SDoug Ambrisko 	}
534665484d8SDoug Ambrisko 
535665484d8SDoug Ambrisko 	dcmd = &cmd->frame->dcmd;
536665484d8SDoug Ambrisko 
537665484d8SDoug Ambrisko 	if (mrsas_alloc_evt_log_info_cmd(sc) != SUCCESS) {
538665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Cannot allocate evt log info cmd\n");
539665484d8SDoug Ambrisko 		mrsas_release_mfi_cmd(cmd);
540665484d8SDoug Ambrisko 		return -ENOMEM;
541665484d8SDoug Ambrisko 	}
542665484d8SDoug Ambrisko 
543665484d8SDoug Ambrisko 	memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE);
544665484d8SDoug Ambrisko 
545665484d8SDoug Ambrisko 	dcmd->cmd = MFI_CMD_DCMD;
546665484d8SDoug Ambrisko 	dcmd->cmd_status = 0x0;
547665484d8SDoug Ambrisko 	dcmd->sge_count = 1;
548665484d8SDoug Ambrisko 	dcmd->flags = MFI_FRAME_DIR_READ;
549665484d8SDoug Ambrisko 	dcmd->timeout = 0;
550665484d8SDoug Ambrisko 	dcmd->pad_0 = 0;
551665484d8SDoug Ambrisko 	dcmd->data_xfer_len = sizeof(struct mrsas_evt_log_info);
552665484d8SDoug Ambrisko 	dcmd->opcode = MR_DCMD_CTRL_EVENT_GET_INFO;
553665484d8SDoug Ambrisko 	dcmd->sgl.sge32[0].phys_addr = sc->el_info_phys_addr;
554665484d8SDoug Ambrisko 	dcmd->sgl.sge32[0].length = sizeof(struct mrsas_evt_log_info);
555665484d8SDoug Ambrisko 
556665484d8SDoug Ambrisko 	mrsas_issue_blocked_cmd(sc, cmd);
557665484d8SDoug Ambrisko 
558665484d8SDoug Ambrisko 	/*
559665484d8SDoug Ambrisko  	 * Copy the data back into callers buffer
560665484d8SDoug Ambrisko  	 */
561665484d8SDoug Ambrisko 	memcpy(eli, sc->el_info_mem, sizeof(struct mrsas_evt_log_info));
562665484d8SDoug Ambrisko 	mrsas_free_evt_log_info_cmd(sc);
563665484d8SDoug Ambrisko 	mrsas_release_mfi_cmd(cmd);
564665484d8SDoug Ambrisko 
565665484d8SDoug Ambrisko 	return 0;
566665484d8SDoug Ambrisko }
567665484d8SDoug Ambrisko 
568665484d8SDoug Ambrisko 
569665484d8SDoug Ambrisko /**
570665484d8SDoug Ambrisko  *  mrsas_register_aen:		Register for asynchronous event notification
571665484d8SDoug Ambrisko  *  @sc:					Adapter soft state
572665484d8SDoug Ambrisko  *  @seq_num:				Starting sequence number
573665484d8SDoug Ambrisko  *  @class_locale:			Class of the event
574665484d8SDoug Ambrisko  *  						This function subscribes for events beyond the @seq_num
575665484d8SDoug Ambrisko  *  						and type @class_locale.
576665484d8SDoug Ambrisko  *
577665484d8SDoug Ambrisko  * */
578665484d8SDoug Ambrisko static int
579665484d8SDoug Ambrisko mrsas_register_aen(struct mrsas_softc *sc, u_int32_t seq_num,
580665484d8SDoug Ambrisko 		     u_int32_t class_locale_word)
581665484d8SDoug Ambrisko {
582665484d8SDoug Ambrisko 	int ret_val;
583665484d8SDoug Ambrisko 	struct mrsas_mfi_cmd *cmd;
584665484d8SDoug Ambrisko 	struct mrsas_dcmd_frame *dcmd;
585665484d8SDoug Ambrisko 	union mrsas_evt_class_locale curr_aen;
586665484d8SDoug Ambrisko 	union mrsas_evt_class_locale prev_aen;
587665484d8SDoug Ambrisko 
588665484d8SDoug Ambrisko /*
589665484d8SDoug Ambrisko  *  If there an AEN pending already (aen_cmd), check if the
590665484d8SDoug Ambrisko  *  class_locale of that pending AEN is inclusive of the new
591665484d8SDoug Ambrisko  *  AEN request we currently have. If it is, then we don't have
592665484d8SDoug Ambrisko  *  to do anything. In other words, whichever events the current
593665484d8SDoug Ambrisko  *  AEN request is subscribing to, have already been subscribed
594665484d8SDoug Ambrisko  *  to.
595665484d8SDoug Ambrisko  *  If the old_cmd is _not_ inclusive, then we have to abort
596665484d8SDoug Ambrisko  *  that command, form a class_locale that is superset of both
597665484d8SDoug Ambrisko  *  old and current and re-issue to the FW
598665484d8SDoug Ambrisko  * */
599665484d8SDoug Ambrisko 
600665484d8SDoug Ambrisko 	curr_aen.word = class_locale_word;
601665484d8SDoug Ambrisko 
602665484d8SDoug Ambrisko 	if (sc->aen_cmd) {
603665484d8SDoug Ambrisko 
604665484d8SDoug Ambrisko 		prev_aen.word = sc->aen_cmd->frame->dcmd.mbox.w[1];
605665484d8SDoug Ambrisko 
606665484d8SDoug Ambrisko /*
607665484d8SDoug Ambrisko  * A class whose enum value is smaller is inclusive of all
608665484d8SDoug Ambrisko  * higher values. If a PROGRESS (= -1) was previously
609665484d8SDoug Ambrisko  * registered, then a new registration requests for higher
610665484d8SDoug Ambrisko  * classes need not be sent to FW. They are automatically
611665484d8SDoug Ambrisko  * included.
612665484d8SDoug Ambrisko  * Locale numbers don't have such hierarchy. They are bitmap values
613665484d8SDoug Ambrisko  */
614665484d8SDoug Ambrisko 		if ((prev_aen.members.class <= curr_aen.members.class) &&
615665484d8SDoug Ambrisko 	    	!((prev_aen.members.locale & curr_aen.members.locale) ^
616665484d8SDoug Ambrisko 	      	curr_aen.members.locale)) {
617665484d8SDoug Ambrisko 			/*
618665484d8SDoug Ambrisko   			 * Previously issued event registration includes
619665484d8SDoug Ambrisko   			 * current request. Nothing to do.
620665484d8SDoug Ambrisko   			 */
621665484d8SDoug Ambrisko 			return 0;
622665484d8SDoug Ambrisko 		} else {
623665484d8SDoug Ambrisko 			curr_aen.members.locale |= prev_aen.members.locale;
624665484d8SDoug Ambrisko 
625665484d8SDoug Ambrisko 			if (prev_aen.members.class < curr_aen.members.class)
626665484d8SDoug Ambrisko 				curr_aen.members.class = prev_aen.members.class;
627665484d8SDoug Ambrisko 
628665484d8SDoug Ambrisko 			sc->aen_cmd->abort_aen = 1;
629665484d8SDoug Ambrisko 			ret_val = mrsas_issue_blocked_abort_cmd(sc,
630665484d8SDoug Ambrisko 				  sc->aen_cmd);
631665484d8SDoug Ambrisko 
632665484d8SDoug Ambrisko 			if (ret_val) {
633665484d8SDoug Ambrisko 				printf("mrsas: Failed to abort "
634665484d8SDoug Ambrisko 					   "previous AEN command\n");
635665484d8SDoug Ambrisko 				return ret_val;
636665484d8SDoug Ambrisko 			}
637665484d8SDoug Ambrisko 		}
638665484d8SDoug Ambrisko 	}
639665484d8SDoug Ambrisko 
640665484d8SDoug Ambrisko 	cmd =  mrsas_get_mfi_cmd(sc);
641665484d8SDoug Ambrisko 
642665484d8SDoug Ambrisko 	if (!cmd)
643665484d8SDoug Ambrisko 		return -ENOMEM;
644665484d8SDoug Ambrisko 
645665484d8SDoug Ambrisko 	dcmd = &cmd->frame->dcmd;
646665484d8SDoug Ambrisko 
647665484d8SDoug Ambrisko 	memset(sc->evt_detail_mem, 0, sizeof(struct mrsas_evt_detail));
648665484d8SDoug Ambrisko 
649665484d8SDoug Ambrisko /*
650665484d8SDoug Ambrisko  * Prepare DCMD for aen registration
651665484d8SDoug Ambrisko  */
652665484d8SDoug Ambrisko 	memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE);
653665484d8SDoug Ambrisko 
654665484d8SDoug Ambrisko 	dcmd->cmd = MFI_CMD_DCMD;
655665484d8SDoug Ambrisko 	dcmd->cmd_status = 0x0;
656665484d8SDoug Ambrisko 	dcmd->sge_count = 1;
657665484d8SDoug Ambrisko 	dcmd->flags = MFI_FRAME_DIR_READ;
658665484d8SDoug Ambrisko 	dcmd->timeout = 0;
659665484d8SDoug Ambrisko 	dcmd->pad_0 = 0;
660665484d8SDoug Ambrisko 	dcmd->data_xfer_len = sizeof(struct mrsas_evt_detail);
661665484d8SDoug Ambrisko 	dcmd->opcode = MR_DCMD_CTRL_EVENT_WAIT;
662665484d8SDoug Ambrisko 	dcmd->mbox.w[0] = seq_num;
663665484d8SDoug Ambrisko 	sc->last_seq_num = seq_num;
664665484d8SDoug Ambrisko 	dcmd->mbox.w[1] = curr_aen.word;
665665484d8SDoug Ambrisko 	dcmd->sgl.sge32[0].phys_addr = (u_int32_t) sc->evt_detail_phys_addr;
666665484d8SDoug Ambrisko 	dcmd->sgl.sge32[0].length = sizeof(struct mrsas_evt_detail);
667665484d8SDoug Ambrisko 
668665484d8SDoug Ambrisko 	if (sc->aen_cmd != NULL) {
669665484d8SDoug Ambrisko 		mrsas_release_mfi_cmd(cmd);
670665484d8SDoug Ambrisko 		return 0;
671665484d8SDoug Ambrisko 	}
672665484d8SDoug Ambrisko 
673665484d8SDoug Ambrisko 	/*
674665484d8SDoug Ambrisko   	 * Store reference to the cmd used to register for AEN. When an
675665484d8SDoug Ambrisko   	 * application wants us to register for AEN, we have to abort this
676665484d8SDoug Ambrisko    	 * cmd and re-register with a new EVENT LOCALE supplied by that app
677665484d8SDoug Ambrisko   	 */
678665484d8SDoug Ambrisko 	sc->aen_cmd = cmd;
679665484d8SDoug Ambrisko 
680665484d8SDoug Ambrisko 	/*
681665484d8SDoug Ambrisko   	  Issue the aen registration frame
682665484d8SDoug Ambrisko   	*/
683665484d8SDoug Ambrisko   	if (mrsas_issue_dcmd(sc, cmd)){
684665484d8SDoug Ambrisko        	device_printf(sc->mrsas_dev, "Cannot issue AEN DCMD command.\n");
685665484d8SDoug Ambrisko        	return(1);
686665484d8SDoug Ambrisko    	}
687665484d8SDoug Ambrisko 
688665484d8SDoug Ambrisko 	return 0;
689665484d8SDoug Ambrisko }
690665484d8SDoug Ambrisko /**
691665484d8SDoug Ambrisko  * mrsas_start_aen -  Subscribes to AEN during driver load time
692665484d8SDoug Ambrisko  * @instance:           Adapter soft state
693665484d8SDoug Ambrisko  */
694665484d8SDoug Ambrisko static int mrsas_start_aen(struct mrsas_softc *sc)
695665484d8SDoug Ambrisko {
696665484d8SDoug Ambrisko 	struct mrsas_evt_log_info eli;
697665484d8SDoug Ambrisko 	union mrsas_evt_class_locale class_locale;
698665484d8SDoug Ambrisko 
699665484d8SDoug Ambrisko 
700665484d8SDoug Ambrisko 	/* Get the latest sequence number from FW*/
701665484d8SDoug Ambrisko 
702665484d8SDoug Ambrisko 	memset(&eli, 0, sizeof(eli));
703665484d8SDoug Ambrisko 
704665484d8SDoug Ambrisko 	if (mrsas_get_seq_num(sc, &eli))
705665484d8SDoug Ambrisko 		return -1;
706665484d8SDoug Ambrisko 
707665484d8SDoug Ambrisko 	/* Register AEN with FW for latest sequence number plus 1*/
708665484d8SDoug Ambrisko 	class_locale.members.reserved = 0;
709665484d8SDoug Ambrisko 	class_locale.members.locale = MR_EVT_LOCALE_ALL;
710665484d8SDoug Ambrisko 	class_locale.members.class = MR_EVT_CLASS_DEBUG;
711665484d8SDoug Ambrisko 
712665484d8SDoug Ambrisko 	return mrsas_register_aen(sc, eli.newest_seq_num + 1,
713665484d8SDoug Ambrisko 				class_locale.word);
714*d18d1b47SKashyap D Desai 
715665484d8SDoug Ambrisko }
716665484d8SDoug Ambrisko 
717665484d8SDoug Ambrisko /**
718*d18d1b47SKashyap D Desai  * mrsas_setup_msix:	Allocate MSI-x vectors
719*d18d1b47SKashyap D Desai  * @sc:					Adapter soft state
720*d18d1b47SKashyap D Desai  */
721*d18d1b47SKashyap D Desai static int mrsas_setup_msix(struct mrsas_softc *sc)
722*d18d1b47SKashyap D Desai {
723*d18d1b47SKashyap D Desai 	int i;
724*d18d1b47SKashyap D Desai 	for (i = 0; i < sc->msix_vectors; i++) {
725*d18d1b47SKashyap D Desai 		sc->irq_context[i].sc = sc;
726*d18d1b47SKashyap D Desai         sc->irq_context[i].MSIxIndex = i;
727*d18d1b47SKashyap D Desai 		sc->irq_id[i] = i + 1;
728*d18d1b47SKashyap D Desai 		sc->mrsas_irq[i] = bus_alloc_resource_any
729*d18d1b47SKashyap D Desai 			(sc->mrsas_dev, SYS_RES_IRQ, &sc->irq_id[i]
730*d18d1b47SKashyap D Desai 			 , RF_ACTIVE);
731*d18d1b47SKashyap D Desai 		if (sc->mrsas_irq[i] == NULL) {
732*d18d1b47SKashyap D Desai 			device_printf(sc->mrsas_dev, "Can't allocate MSI-x\n");
733*d18d1b47SKashyap D Desai 			goto irq_alloc_failed;
734*d18d1b47SKashyap D Desai 		}
735*d18d1b47SKashyap D Desai 		if (bus_setup_intr(sc->mrsas_dev,
736*d18d1b47SKashyap D Desai 			sc->mrsas_irq[i],
737*d18d1b47SKashyap D Desai 			INTR_MPSAFE|INTR_TYPE_CAM,
738*d18d1b47SKashyap D Desai 			NULL, mrsas_isr, &sc->irq_context[i],
739*d18d1b47SKashyap D Desai 			&sc->intr_handle[i])) {
740*d18d1b47SKashyap D Desai 			device_printf(sc->mrsas_dev,
741*d18d1b47SKashyap D Desai 				"Cannot set up MSI-x interrupt handler\n");
742*d18d1b47SKashyap D Desai 			goto irq_alloc_failed;
743*d18d1b47SKashyap D Desai 		}
744*d18d1b47SKashyap D Desai 	}
745*d18d1b47SKashyap D Desai 	return SUCCESS;
746*d18d1b47SKashyap D Desai 
747*d18d1b47SKashyap D Desai irq_alloc_failed:
748*d18d1b47SKashyap D Desai 	mrsas_teardown_intr(sc);
749*d18d1b47SKashyap D Desai 	return (FAIL);
750*d18d1b47SKashyap D Desai }
751*d18d1b47SKashyap D Desai 
752*d18d1b47SKashyap D Desai /**
753*d18d1b47SKashyap D Desai  * mrsas_allocate_msix:		Setup MSI-x vectors
754*d18d1b47SKashyap D Desai  * @sc:						Adapter soft state
755*d18d1b47SKashyap D Desai  */
756*d18d1b47SKashyap D Desai static int mrsas_allocate_msix(struct mrsas_softc *sc)
757*d18d1b47SKashyap D Desai {
758*d18d1b47SKashyap D Desai 	if (pci_alloc_msix(sc->mrsas_dev, &sc->msix_vectors) == 0) {
759*d18d1b47SKashyap D Desai 		device_printf(sc->mrsas_dev, "Using MSI-X with %d number"
760*d18d1b47SKashyap D Desai 				" of vectors\n", sc->msix_vectors);
761*d18d1b47SKashyap D Desai 	} else {
762*d18d1b47SKashyap D Desai 		device_printf(sc->mrsas_dev, "MSI-x setup failed\n");
763*d18d1b47SKashyap D Desai 		goto irq_alloc_failed;
764*d18d1b47SKashyap D Desai 	}
765*d18d1b47SKashyap D Desai 	return SUCCESS;
766*d18d1b47SKashyap D Desai 
767*d18d1b47SKashyap D Desai irq_alloc_failed:
768*d18d1b47SKashyap D Desai 	mrsas_teardown_intr(sc);
769*d18d1b47SKashyap D Desai 	return (FAIL);
770*d18d1b47SKashyap D Desai }
771*d18d1b47SKashyap D Desai /**
772665484d8SDoug Ambrisko  * mrsas_attach:            PCI entry point
773665484d8SDoug Ambrisko  * input:                   device struct pointer
774665484d8SDoug Ambrisko  *
775665484d8SDoug Ambrisko  * Performs setup of PCI and registers, initializes mutexes and
776665484d8SDoug Ambrisko  * linked lists, registers interrupts and CAM, and initializes
777665484d8SDoug Ambrisko  * the adapter/controller to its proper state.
778665484d8SDoug Ambrisko  */
779665484d8SDoug Ambrisko static int mrsas_attach(device_t dev)
780665484d8SDoug Ambrisko {
781665484d8SDoug Ambrisko     struct mrsas_softc *sc = device_get_softc(dev);
782665484d8SDoug Ambrisko     uint32_t cmd, bar, error;
783665484d8SDoug Ambrisko 
784665484d8SDoug Ambrisko     /* Look up our softc and initialize its fields. */
785665484d8SDoug Ambrisko     sc->mrsas_dev = dev;
786665484d8SDoug Ambrisko     sc->device_id = pci_get_device(dev);
787665484d8SDoug Ambrisko 
788665484d8SDoug Ambrisko     mrsas_get_tunables(sc);
789665484d8SDoug Ambrisko 
790665484d8SDoug Ambrisko     /*
791665484d8SDoug Ambrisko      * Set up PCI and registers
792665484d8SDoug Ambrisko      */
793665484d8SDoug Ambrisko     cmd = pci_read_config(dev, PCIR_COMMAND, 2);
794665484d8SDoug Ambrisko     if ( (cmd & PCIM_CMD_PORTEN) == 0) {
795665484d8SDoug Ambrisko         return (ENXIO);
796665484d8SDoug Ambrisko     }
797665484d8SDoug Ambrisko     /* Force the busmaster enable bit on. */
798665484d8SDoug Ambrisko     cmd |= PCIM_CMD_BUSMASTEREN;
799665484d8SDoug Ambrisko     pci_write_config(dev, PCIR_COMMAND, cmd, 2);
800665484d8SDoug Ambrisko 
801665484d8SDoug Ambrisko     //bar = pci_read_config(dev, MRSAS_PCI_BAR0, 4);
802665484d8SDoug Ambrisko     bar = pci_read_config(dev, MRSAS_PCI_BAR1, 4);
803665484d8SDoug Ambrisko 
804665484d8SDoug Ambrisko     sc->reg_res_id = MRSAS_PCI_BAR1; /* BAR1 offset */
805665484d8SDoug Ambrisko     if ((sc->reg_res = bus_alloc_resource(dev, SYS_RES_MEMORY,
806665484d8SDoug Ambrisko                                 &(sc->reg_res_id), 0, ~0, 1, RF_ACTIVE))
807665484d8SDoug Ambrisko                                 == NULL) {
808665484d8SDoug Ambrisko         device_printf(dev, "Cannot allocate PCI registers\n");
809665484d8SDoug Ambrisko         goto attach_fail;
810665484d8SDoug Ambrisko     }
811665484d8SDoug Ambrisko     sc->bus_tag = rman_get_bustag(sc->reg_res);
812665484d8SDoug Ambrisko     sc->bus_handle = rman_get_bushandle(sc->reg_res);
813665484d8SDoug Ambrisko 
814665484d8SDoug Ambrisko     /* Intialize mutexes */
815665484d8SDoug Ambrisko     mtx_init(&sc->sim_lock,  "mrsas_sim_lock", NULL, MTX_DEF);
816665484d8SDoug Ambrisko     mtx_init(&sc->pci_lock,  "mrsas_pci_lock", NULL, MTX_DEF);
817665484d8SDoug Ambrisko     mtx_init(&sc->io_lock,  "mrsas_io_lock", NULL, MTX_DEF);
818665484d8SDoug Ambrisko     mtx_init(&sc->aen_lock,  "mrsas_aen_lock", NULL, MTX_DEF);
819665484d8SDoug Ambrisko     mtx_init(&sc->ioctl_lock,  "mrsas_ioctl_lock", NULL, MTX_SPIN);
820665484d8SDoug Ambrisko     mtx_init(&sc->mpt_cmd_pool_lock, "mrsas_mpt_cmd_pool_lock", NULL, MTX_DEF);
821665484d8SDoug Ambrisko     mtx_init(&sc->mfi_cmd_pool_lock, "mrsas_mfi_cmd_pool_lock", NULL, MTX_DEF);
822665484d8SDoug Ambrisko     mtx_init(&sc->raidmap_lock, "mrsas_raidmap_lock", NULL, MTX_DEF);
823665484d8SDoug Ambrisko 
824665484d8SDoug Ambrisko     /* Intialize linked list */
825665484d8SDoug Ambrisko     TAILQ_INIT(&sc->mrsas_mpt_cmd_list_head);
826665484d8SDoug Ambrisko     TAILQ_INIT(&sc->mrsas_mfi_cmd_list_head);
827665484d8SDoug Ambrisko 
828665484d8SDoug Ambrisko     atomic_set(&sc->fw_outstanding,0);
829665484d8SDoug Ambrisko 
830665484d8SDoug Ambrisko 	sc->io_cmds_highwater = 0;
831665484d8SDoug Ambrisko 
832665484d8SDoug Ambrisko     /* Create a /dev entry for this device. */
833665484d8SDoug Ambrisko     sc->mrsas_cdev = make_dev(&mrsas_cdevsw, device_get_unit(dev), UID_ROOT,
834665484d8SDoug Ambrisko         GID_OPERATOR, (S_IRUSR | S_IWUSR | S_IRGRP | S_IWGRP), "mrsas%u",
835665484d8SDoug Ambrisko         device_get_unit(dev));
836536094dcSKashyap D Desai     if (device_get_unit(dev) == 0)
837536094dcSKashyap D Desai 		make_dev_alias(sc->mrsas_cdev, "megaraid_sas_ioctl_node");
838665484d8SDoug Ambrisko     if (sc->mrsas_cdev)
839665484d8SDoug Ambrisko     	sc->mrsas_cdev->si_drv1 = sc;
840665484d8SDoug Ambrisko 
841665484d8SDoug Ambrisko     sc->adprecovery = MRSAS_HBA_OPERATIONAL;
842665484d8SDoug Ambrisko 	sc->UnevenSpanSupport = 0;
843665484d8SDoug Ambrisko 
844*d18d1b47SKashyap D Desai     sc->msix_enable = 0;
845*d18d1b47SKashyap D Desai 
846665484d8SDoug Ambrisko     /* Initialize Firmware */
847665484d8SDoug Ambrisko     if (mrsas_init_fw(sc) != SUCCESS) {
848665484d8SDoug Ambrisko         goto attach_fail_fw;
849665484d8SDoug Ambrisko     }
850665484d8SDoug Ambrisko 
851665484d8SDoug Ambrisko     /* Register SCSI mid-layer */
852665484d8SDoug Ambrisko     if ((mrsas_cam_attach(sc) != SUCCESS)) {
853665484d8SDoug Ambrisko         goto attach_fail_cam;
854665484d8SDoug Ambrisko     }
855665484d8SDoug Ambrisko 
856*d18d1b47SKashyap D Desai 
857665484d8SDoug Ambrisko     /* Register IRQs */
858665484d8SDoug Ambrisko     if (mrsas_setup_irq(sc) != SUCCESS) {
859665484d8SDoug Ambrisko         goto attach_fail_irq;
860665484d8SDoug Ambrisko     }
861665484d8SDoug Ambrisko 
862665484d8SDoug Ambrisko     /* Enable Interrupts */
863665484d8SDoug Ambrisko     mrsas_enable_intr(sc);
864665484d8SDoug Ambrisko 
865665484d8SDoug Ambrisko     error = mrsas_kproc_create(mrsas_ocr_thread, sc,
866665484d8SDoug Ambrisko         &sc->ocr_thread, 0, 0, "mrsas_ocr%d",
867665484d8SDoug Ambrisko         device_get_unit(sc->mrsas_dev));
868665484d8SDoug Ambrisko     if (error) {
869665484d8SDoug Ambrisko         printf("Error %d starting rescan thread\n", error);
870665484d8SDoug Ambrisko         goto attach_fail_irq;
871665484d8SDoug Ambrisko     }
872665484d8SDoug Ambrisko 
873665484d8SDoug Ambrisko     mrsas_setup_sysctl(sc);
874665484d8SDoug Ambrisko 
875665484d8SDoug Ambrisko 	/* Initiate AEN (Asynchronous Event Notification)*/
876665484d8SDoug Ambrisko 
877665484d8SDoug Ambrisko 	if (mrsas_start_aen(sc)) {
878665484d8SDoug Ambrisko 		printf("Error: start aen failed\n");
879665484d8SDoug Ambrisko 		goto fail_start_aen;
880665484d8SDoug Ambrisko 	}
881665484d8SDoug Ambrisko 
882536094dcSKashyap D Desai     /*
883536094dcSKashyap D Desai      * Add this controller to mrsas_mgmt_info structure so that it
884536094dcSKashyap D Desai      * can be exported to management applications
885536094dcSKashyap D Desai      */
886536094dcSKashyap D Desai     if (device_get_unit(dev) == 0)
887536094dcSKashyap D Desai         memset(&mrsas_mgmt_info, 0, sizeof(mrsas_mgmt_info));
888536094dcSKashyap D Desai 
889536094dcSKashyap D Desai     mrsas_mgmt_info.count++;
890536094dcSKashyap D Desai     mrsas_mgmt_info.sc_ptr[mrsas_mgmt_info.max_index] = sc;
891536094dcSKashyap D Desai     mrsas_mgmt_info.max_index++;
892536094dcSKashyap D Desai 
893665484d8SDoug Ambrisko     return (0);
894665484d8SDoug Ambrisko 
895665484d8SDoug Ambrisko fail_start_aen:
896665484d8SDoug Ambrisko attach_fail_irq:
897665484d8SDoug Ambrisko     mrsas_teardown_intr(sc);
898665484d8SDoug Ambrisko attach_fail_cam:
899665484d8SDoug Ambrisko     mrsas_cam_detach(sc);
900665484d8SDoug Ambrisko attach_fail_fw:
901*d18d1b47SKashyap D Desai     /* if MSIX vector is allocated and FW Init FAILED then release MSIX */
902*d18d1b47SKashyap D Desai     if (sc->msix_enable == 1)
903*d18d1b47SKashyap D Desai 		pci_release_msi(sc->mrsas_dev);
904665484d8SDoug Ambrisko     mrsas_free_mem(sc);
905665484d8SDoug Ambrisko     mtx_destroy(&sc->sim_lock);
906665484d8SDoug Ambrisko     mtx_destroy(&sc->aen_lock);
907665484d8SDoug Ambrisko     mtx_destroy(&sc->pci_lock);
908665484d8SDoug Ambrisko     mtx_destroy(&sc->io_lock);
909665484d8SDoug Ambrisko     mtx_destroy(&sc->ioctl_lock);
910665484d8SDoug Ambrisko     mtx_destroy(&sc->mpt_cmd_pool_lock);
911665484d8SDoug Ambrisko     mtx_destroy(&sc->mfi_cmd_pool_lock);
912665484d8SDoug Ambrisko     mtx_destroy(&sc->raidmap_lock);
913665484d8SDoug Ambrisko attach_fail:
914665484d8SDoug Ambrisko     destroy_dev(sc->mrsas_cdev);
915665484d8SDoug Ambrisko     if (sc->reg_res){
916665484d8SDoug Ambrisko         bus_release_resource(sc->mrsas_dev, SYS_RES_MEMORY,
917665484d8SDoug Ambrisko                              sc->reg_res_id, sc->reg_res);
918665484d8SDoug Ambrisko     }
919665484d8SDoug Ambrisko     return (ENXIO);
920665484d8SDoug Ambrisko }
921665484d8SDoug Ambrisko 
922665484d8SDoug Ambrisko /**
923665484d8SDoug Ambrisko  * mrsas_detach:            De-allocates and teardown resources
924665484d8SDoug Ambrisko  * input:                   device struct pointer
925665484d8SDoug Ambrisko  *
926665484d8SDoug Ambrisko  * This function is the entry point for device disconnect and detach.  It
927665484d8SDoug Ambrisko  * performs memory de-allocations, shutdown of the controller and various
928665484d8SDoug Ambrisko  * teardown and destroy resource functions.
929665484d8SDoug Ambrisko  */
930665484d8SDoug Ambrisko static int mrsas_detach(device_t dev)
931665484d8SDoug Ambrisko {
932665484d8SDoug Ambrisko     struct mrsas_softc *sc;
933665484d8SDoug Ambrisko     int i = 0;
934665484d8SDoug Ambrisko 
935665484d8SDoug Ambrisko     sc = device_get_softc(dev);
936665484d8SDoug Ambrisko     sc->remove_in_progress = 1;
937536094dcSKashyap D Desai 
938536094dcSKashyap D Desai     /*
939536094dcSKashyap D Desai      * Take the instance off the instance array. Note that we will not
940536094dcSKashyap D Desai      * decrement the max_index. We let this array be sparse array
941536094dcSKashyap D Desai     */
942536094dcSKashyap D Desai     for (i = 0; i < mrsas_mgmt_info.max_index; i++) {
943536094dcSKashyap D Desai             if (mrsas_mgmt_info.sc_ptr[i] == sc) {
944536094dcSKashyap D Desai                     mrsas_mgmt_info.count--;
945536094dcSKashyap D Desai                     mrsas_mgmt_info.sc_ptr[i] = NULL;
946536094dcSKashyap D Desai                     break;
947536094dcSKashyap D Desai             }
948536094dcSKashyap D Desai     }
949536094dcSKashyap D Desai 
950665484d8SDoug Ambrisko     if(sc->ocr_thread_active)
951665484d8SDoug Ambrisko         wakeup(&sc->ocr_chan);
952665484d8SDoug Ambrisko     while(sc->reset_in_progress){
953665484d8SDoug Ambrisko         i++;
954665484d8SDoug Ambrisko         if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) {
955665484d8SDoug Ambrisko             mrsas_dprint(sc, MRSAS_INFO,
956665484d8SDoug Ambrisko                 "[%2d]waiting for ocr to be finished\n",i);
957665484d8SDoug Ambrisko         }
958665484d8SDoug Ambrisko         pause("mr_shutdown", hz);
959665484d8SDoug Ambrisko     }
960665484d8SDoug Ambrisko     i = 0;
961665484d8SDoug Ambrisko     while(sc->ocr_thread_active){
962665484d8SDoug Ambrisko         i++;
963665484d8SDoug Ambrisko         if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) {
964665484d8SDoug Ambrisko             mrsas_dprint(sc, MRSAS_INFO,
965665484d8SDoug Ambrisko                 "[%2d]waiting for "
966665484d8SDoug Ambrisko                 "mrsas_ocr thread to quit ocr %d\n",i,
967665484d8SDoug Ambrisko                  sc->ocr_thread_active);
968665484d8SDoug Ambrisko         }
969665484d8SDoug Ambrisko         pause("mr_shutdown", hz);
970665484d8SDoug Ambrisko     }
971665484d8SDoug Ambrisko     mrsas_flush_cache(sc);
972665484d8SDoug Ambrisko     mrsas_shutdown_ctlr(sc, MR_DCMD_CTRL_SHUTDOWN);
973665484d8SDoug Ambrisko     mrsas_disable_intr(sc);
974665484d8SDoug Ambrisko     mrsas_cam_detach(sc);
975665484d8SDoug Ambrisko     mrsas_teardown_intr(sc);
976665484d8SDoug Ambrisko     mrsas_free_mem(sc);
977665484d8SDoug Ambrisko     mtx_destroy(&sc->sim_lock);
978665484d8SDoug Ambrisko     mtx_destroy(&sc->aen_lock);
979665484d8SDoug Ambrisko     mtx_destroy(&sc->pci_lock);
980665484d8SDoug Ambrisko     mtx_destroy(&sc->io_lock);
981665484d8SDoug Ambrisko     mtx_destroy(&sc->ioctl_lock);
982665484d8SDoug Ambrisko     mtx_destroy(&sc->mpt_cmd_pool_lock);
983665484d8SDoug Ambrisko     mtx_destroy(&sc->mfi_cmd_pool_lock);
984665484d8SDoug Ambrisko     mtx_destroy(&sc->raidmap_lock);
985665484d8SDoug Ambrisko     if (sc->reg_res){
986665484d8SDoug Ambrisko         bus_release_resource(sc->mrsas_dev,
987665484d8SDoug Ambrisko                  SYS_RES_MEMORY, sc->reg_res_id, sc->reg_res);
988665484d8SDoug Ambrisko     }
989665484d8SDoug Ambrisko     destroy_dev(sc->mrsas_cdev);
990665484d8SDoug Ambrisko     if (sc->sysctl_tree != NULL)
991665484d8SDoug Ambrisko         sysctl_ctx_free(&sc->sysctl_ctx);
992665484d8SDoug Ambrisko     return (0);
993665484d8SDoug Ambrisko }
994665484d8SDoug Ambrisko 
995665484d8SDoug Ambrisko /**
996665484d8SDoug Ambrisko  * mrsas_free_mem:          Frees allocated memory
997665484d8SDoug Ambrisko  * input:                   Adapter instance soft state
998665484d8SDoug Ambrisko  *
999665484d8SDoug Ambrisko  * This function is called from mrsas_detach() to free previously allocated
1000665484d8SDoug Ambrisko  * memory.
1001665484d8SDoug Ambrisko  */
1002665484d8SDoug Ambrisko void mrsas_free_mem(struct mrsas_softc *sc)
1003665484d8SDoug Ambrisko {
1004665484d8SDoug Ambrisko     int i;
1005665484d8SDoug Ambrisko     u_int32_t max_cmd;
1006665484d8SDoug Ambrisko     struct mrsas_mfi_cmd *mfi_cmd;
1007665484d8SDoug Ambrisko     struct mrsas_mpt_cmd *mpt_cmd;
1008665484d8SDoug Ambrisko 
1009665484d8SDoug Ambrisko 	/*
1010665484d8SDoug Ambrisko      * Free RAID map memory
1011665484d8SDoug Ambrisko      */
1012665484d8SDoug Ambrisko     for (i=0; i < 2; i++)
1013665484d8SDoug Ambrisko     {
1014665484d8SDoug Ambrisko         if (sc->raidmap_phys_addr[i])
1015665484d8SDoug Ambrisko             bus_dmamap_unload(sc->raidmap_tag[i], sc->raidmap_dmamap[i]);
1016665484d8SDoug Ambrisko         if (sc->raidmap_mem[i] != NULL)
1017665484d8SDoug Ambrisko             bus_dmamem_free(sc->raidmap_tag[i], sc->raidmap_mem[i], sc->raidmap_dmamap[i]);
1018665484d8SDoug Ambrisko         if (sc->raidmap_tag[i] != NULL)
1019665484d8SDoug Ambrisko             bus_dma_tag_destroy(sc->raidmap_tag[i]);
10204799d485SKashyap D Desai 
10214799d485SKashyap D Desai 	if (sc->ld_drv_map[i] != NULL)
10224799d485SKashyap D Desai 		free(sc->ld_drv_map[i], M_MRSAS);
1023665484d8SDoug Ambrisko     }
1024665484d8SDoug Ambrisko 
1025665484d8SDoug Ambrisko     /*
1026665484d8SDoug Ambrisko      * Free version buffer memroy
1027665484d8SDoug Ambrisko      */
1028665484d8SDoug Ambrisko     if (sc->verbuf_phys_addr)
1029665484d8SDoug Ambrisko         bus_dmamap_unload(sc->verbuf_tag, sc->verbuf_dmamap);
1030665484d8SDoug Ambrisko     if (sc->verbuf_mem != NULL)
1031665484d8SDoug Ambrisko         bus_dmamem_free(sc->verbuf_tag, sc->verbuf_mem, sc->verbuf_dmamap);
1032665484d8SDoug Ambrisko     if (sc->verbuf_tag != NULL)
1033665484d8SDoug Ambrisko         bus_dma_tag_destroy(sc->verbuf_tag);
1034665484d8SDoug Ambrisko 
1035665484d8SDoug Ambrisko 
1036665484d8SDoug Ambrisko     /*
1037665484d8SDoug Ambrisko      * Free sense buffer memory
1038665484d8SDoug Ambrisko      */
1039665484d8SDoug Ambrisko     if (sc->sense_phys_addr)
1040665484d8SDoug Ambrisko         bus_dmamap_unload(sc->sense_tag, sc->sense_dmamap);
1041665484d8SDoug Ambrisko     if (sc->sense_mem != NULL)
1042665484d8SDoug Ambrisko         bus_dmamem_free(sc->sense_tag, sc->sense_mem, sc->sense_dmamap);
1043665484d8SDoug Ambrisko     if (sc->sense_tag != NULL)
1044665484d8SDoug Ambrisko         bus_dma_tag_destroy(sc->sense_tag);
1045665484d8SDoug Ambrisko 
1046665484d8SDoug Ambrisko     /*
1047665484d8SDoug Ambrisko      * Free chain frame memory
1048665484d8SDoug Ambrisko      */
1049665484d8SDoug Ambrisko     if (sc->chain_frame_phys_addr)
1050665484d8SDoug Ambrisko         bus_dmamap_unload(sc->chain_frame_tag, sc->chain_frame_dmamap);
1051665484d8SDoug Ambrisko     if (sc->chain_frame_mem != NULL)
1052665484d8SDoug Ambrisko         bus_dmamem_free(sc->chain_frame_tag, sc->chain_frame_mem, sc->chain_frame_dmamap);
1053665484d8SDoug Ambrisko     if (sc->chain_frame_tag != NULL)
1054665484d8SDoug Ambrisko         bus_dma_tag_destroy(sc->chain_frame_tag);
1055665484d8SDoug Ambrisko 
1056665484d8SDoug Ambrisko     /*
1057665484d8SDoug Ambrisko      * Free IO Request memory
1058665484d8SDoug Ambrisko      */
1059665484d8SDoug Ambrisko     if (sc->io_request_phys_addr)
1060665484d8SDoug Ambrisko         bus_dmamap_unload(sc->io_request_tag, sc->io_request_dmamap);
1061665484d8SDoug Ambrisko     if (sc->io_request_mem != NULL)
1062665484d8SDoug Ambrisko         bus_dmamem_free(sc->io_request_tag, sc->io_request_mem, sc->io_request_dmamap);
1063665484d8SDoug Ambrisko     if (sc->io_request_tag != NULL)
1064665484d8SDoug Ambrisko         bus_dma_tag_destroy(sc->io_request_tag);
1065665484d8SDoug Ambrisko 
1066665484d8SDoug Ambrisko     /*
1067665484d8SDoug Ambrisko      * Free Reply Descriptor memory
1068665484d8SDoug Ambrisko      */
1069665484d8SDoug Ambrisko     if (sc->reply_desc_phys_addr)
1070665484d8SDoug Ambrisko         bus_dmamap_unload(sc->reply_desc_tag, sc->reply_desc_dmamap);
1071665484d8SDoug Ambrisko     if (sc->reply_desc_mem != NULL)
1072665484d8SDoug Ambrisko         bus_dmamem_free(sc->reply_desc_tag, sc->reply_desc_mem, sc->reply_desc_dmamap);
1073665484d8SDoug Ambrisko     if (sc->reply_desc_tag != NULL)
1074665484d8SDoug Ambrisko         bus_dma_tag_destroy(sc->reply_desc_tag);
1075665484d8SDoug Ambrisko 
1076665484d8SDoug Ambrisko     /*
1077665484d8SDoug Ambrisko      * Free event detail memory
1078665484d8SDoug Ambrisko      */
1079665484d8SDoug Ambrisko     if (sc->evt_detail_phys_addr)
1080665484d8SDoug Ambrisko         bus_dmamap_unload(sc->evt_detail_tag, sc->evt_detail_dmamap);
1081665484d8SDoug Ambrisko     if (sc->evt_detail_mem != NULL)
1082665484d8SDoug Ambrisko         bus_dmamem_free(sc->evt_detail_tag, sc->evt_detail_mem, sc->evt_detail_dmamap);
1083665484d8SDoug Ambrisko     if (sc->evt_detail_tag != NULL)
1084665484d8SDoug Ambrisko         bus_dma_tag_destroy(sc->evt_detail_tag);
1085665484d8SDoug Ambrisko 
1086665484d8SDoug Ambrisko     /*
1087665484d8SDoug Ambrisko      * Free MFI frames
1088665484d8SDoug Ambrisko      */
1089665484d8SDoug Ambrisko 	if (sc->mfi_cmd_list) {
1090665484d8SDoug Ambrisko     	for (i = 0; i < MRSAS_MAX_MFI_CMDS; i++) {
1091665484d8SDoug Ambrisko         	mfi_cmd = sc->mfi_cmd_list[i];
1092665484d8SDoug Ambrisko         	mrsas_free_frame(sc, mfi_cmd);
1093665484d8SDoug Ambrisko 		}
1094665484d8SDoug Ambrisko     }
1095665484d8SDoug Ambrisko     if (sc->mficmd_frame_tag != NULL)
1096665484d8SDoug Ambrisko         bus_dma_tag_destroy(sc->mficmd_frame_tag);
1097665484d8SDoug Ambrisko 
1098665484d8SDoug Ambrisko     /*
1099665484d8SDoug Ambrisko      * Free MPT internal command list
1100665484d8SDoug Ambrisko      */
1101665484d8SDoug Ambrisko     max_cmd = sc->max_fw_cmds;
1102665484d8SDoug Ambrisko 	if (sc->mpt_cmd_list) {
1103665484d8SDoug Ambrisko     	for (i = 0; i < max_cmd; i++) {
1104665484d8SDoug Ambrisko         	mpt_cmd = sc->mpt_cmd_list[i];
1105665484d8SDoug Ambrisko         	bus_dmamap_destroy(sc->data_tag, mpt_cmd->data_dmamap);
1106665484d8SDoug Ambrisko         	free(sc->mpt_cmd_list[i], M_MRSAS);
1107665484d8SDoug Ambrisko     	}
1108665484d8SDoug Ambrisko     	free(sc->mpt_cmd_list, M_MRSAS);
1109665484d8SDoug Ambrisko     	sc->mpt_cmd_list = NULL;
1110665484d8SDoug Ambrisko 	}
1111665484d8SDoug Ambrisko 
1112665484d8SDoug Ambrisko     /*
1113665484d8SDoug Ambrisko      * Free MFI internal command list
1114665484d8SDoug Ambrisko      */
1115665484d8SDoug Ambrisko 
1116665484d8SDoug Ambrisko 	if (sc->mfi_cmd_list) {
1117665484d8SDoug Ambrisko     	for (i = 0; i < MRSAS_MAX_MFI_CMDS; i++) {
1118665484d8SDoug Ambrisko         	free(sc->mfi_cmd_list[i], M_MRSAS);
1119665484d8SDoug Ambrisko     	}
1120665484d8SDoug Ambrisko     	free(sc->mfi_cmd_list, M_MRSAS);
1121665484d8SDoug Ambrisko     	sc->mfi_cmd_list = NULL;
1122665484d8SDoug Ambrisko 	}
1123665484d8SDoug Ambrisko 
1124665484d8SDoug Ambrisko     /*
1125665484d8SDoug Ambrisko      * Free request descriptor memory
1126665484d8SDoug Ambrisko      */
1127665484d8SDoug Ambrisko     free(sc->req_desc, M_MRSAS);
1128665484d8SDoug Ambrisko     sc->req_desc = NULL;
1129665484d8SDoug Ambrisko 
1130665484d8SDoug Ambrisko     /*
1131665484d8SDoug Ambrisko      * Destroy parent tag
1132665484d8SDoug Ambrisko      */
1133665484d8SDoug Ambrisko     if (sc->mrsas_parent_tag != NULL)
1134665484d8SDoug Ambrisko         bus_dma_tag_destroy(sc->mrsas_parent_tag);
1135665484d8SDoug Ambrisko }
1136665484d8SDoug Ambrisko 
1137665484d8SDoug Ambrisko /**
1138665484d8SDoug Ambrisko  * mrsas_teardown_intr:        Teardown interrupt
1139665484d8SDoug Ambrisko  * input:                      Adapter instance soft state
1140665484d8SDoug Ambrisko  *
1141665484d8SDoug Ambrisko  * This function is called from mrsas_detach() to teardown and release
1142665484d8SDoug Ambrisko  * bus interrupt resourse.
1143665484d8SDoug Ambrisko  */
1144665484d8SDoug Ambrisko void mrsas_teardown_intr(struct mrsas_softc *sc)
1145665484d8SDoug Ambrisko {
1146*d18d1b47SKashyap D Desai     int i;
1147*d18d1b47SKashyap D Desai     if (!sc->msix_enable) {
1148*d18d1b47SKashyap D Desai 	    if (sc->intr_handle[0])
1149*d18d1b47SKashyap D Desai 		bus_teardown_intr(sc->mrsas_dev, sc->mrsas_irq[0], sc->intr_handle[0]);
1150*d18d1b47SKashyap D Desai 	    if (sc->mrsas_irq[0] != NULL)
1151*d18d1b47SKashyap D Desai 		bus_release_resource(sc->mrsas_dev, SYS_RES_IRQ, sc->irq_id[0], sc->mrsas_irq[0]);
1152*d18d1b47SKashyap D Desai 	    sc->intr_handle[0] = NULL;
1153*d18d1b47SKashyap D Desai     } else {
1154*d18d1b47SKashyap D Desai 		for (i = 0; i < sc->msix_vectors; i++) {
1155*d18d1b47SKashyap D Desai 			if (sc->intr_handle[i])
1156*d18d1b47SKashyap D Desai 				bus_teardown_intr(sc->mrsas_dev, sc->mrsas_irq[i],
1157*d18d1b47SKashyap D Desai 					sc->intr_handle[i]);
1158*d18d1b47SKashyap D Desai 
1159*d18d1b47SKashyap D Desai 			if (sc->mrsas_irq[i] != NULL)
1160*d18d1b47SKashyap D Desai 				bus_release_resource(sc->mrsas_dev, SYS_RES_IRQ,
1161*d18d1b47SKashyap D Desai 					sc->irq_id[i], sc->mrsas_irq[i]);
1162*d18d1b47SKashyap D Desai 
1163*d18d1b47SKashyap D Desai 			sc->intr_handle[i] = NULL;
1164*d18d1b47SKashyap D Desai 		}
1165*d18d1b47SKashyap D Desai 		pci_release_msi(sc->mrsas_dev);
1166*d18d1b47SKashyap D Desai 	}
1167*d18d1b47SKashyap D Desai 
1168665484d8SDoug Ambrisko }
1169665484d8SDoug Ambrisko 
1170665484d8SDoug Ambrisko /**
1171665484d8SDoug Ambrisko  * mrsas_suspend:          Suspend entry point
1172665484d8SDoug Ambrisko  * input:                  Device struct pointer
1173665484d8SDoug Ambrisko  *
1174665484d8SDoug Ambrisko  * This function is the entry point for system suspend from the OS.
1175665484d8SDoug Ambrisko  */
1176665484d8SDoug Ambrisko static int mrsas_suspend(device_t dev)
1177665484d8SDoug Ambrisko {
1178665484d8SDoug Ambrisko     struct mrsas_softc *sc;
1179665484d8SDoug Ambrisko 
1180665484d8SDoug Ambrisko     sc = device_get_softc(dev);
1181665484d8SDoug Ambrisko     return (0);
1182665484d8SDoug Ambrisko }
1183665484d8SDoug Ambrisko 
1184665484d8SDoug Ambrisko /**
1185665484d8SDoug Ambrisko  * mrsas_resume:           Resume entry point
1186665484d8SDoug Ambrisko  * input:                  Device struct pointer
1187665484d8SDoug Ambrisko  *
1188665484d8SDoug Ambrisko  * This function is the entry point for system resume from the OS.
1189665484d8SDoug Ambrisko  */
1190665484d8SDoug Ambrisko static int mrsas_resume(device_t dev)
1191665484d8SDoug Ambrisko {
1192665484d8SDoug Ambrisko     struct mrsas_softc *sc;
1193665484d8SDoug Ambrisko 
1194665484d8SDoug Ambrisko     sc = device_get_softc(dev);
1195665484d8SDoug Ambrisko     return (0);
1196665484d8SDoug Ambrisko }
1197665484d8SDoug Ambrisko 
1198665484d8SDoug Ambrisko /**
1199665484d8SDoug Ambrisko  * mrsas_ioctl:       IOCtl commands entry point.
1200665484d8SDoug Ambrisko  *
1201665484d8SDoug Ambrisko  * This function is the entry point for IOCtls from the OS.  It calls the
1202665484d8SDoug Ambrisko  * appropriate function for processing depending on the command received.
1203665484d8SDoug Ambrisko  */
1204665484d8SDoug Ambrisko static int
1205665484d8SDoug Ambrisko mrsas_ioctl(struct cdev *dev, u_long cmd, caddr_t arg, int flag, d_thread_t *td)
1206665484d8SDoug Ambrisko {
1207665484d8SDoug Ambrisko     struct mrsas_softc *sc;
1208665484d8SDoug Ambrisko     int ret = 0, i = 0;
1209665484d8SDoug Ambrisko 
1210536094dcSKashyap D Desai     struct mrsas_iocpacket *user_ioc = (struct mrsas_iocpacket *)arg;
1211536094dcSKashyap D Desai 
1212536094dcSKashyap D Desai     /* get the Host number & the softc from data sent by the Application */
1213536094dcSKashyap D Desai     sc = mrsas_mgmt_info.sc_ptr[user_ioc->host_no];
1214536094dcSKashyap D Desai 
1215536094dcSKashyap D Desai     if ((mrsas_mgmt_info.max_index == user_ioc->host_no) || (sc == NULL)) {
1216536094dcSKashyap D Desai             printf ("Please check the controller number\n");
1217536094dcSKashyap D Desai             if (sc == NULL)
1218536094dcSKashyap D Desai                     printf ("There is NO such Host no. %d\n", user_ioc->host_no);
1219536094dcSKashyap D Desai 
1220536094dcSKashyap D Desai             return ENOENT;
1221536094dcSKashyap D Desai     }
1222665484d8SDoug Ambrisko 
1223665484d8SDoug Ambrisko     if (sc->remove_in_progress) {
1224665484d8SDoug Ambrisko         mrsas_dprint(sc, MRSAS_INFO,
1225665484d8SDoug Ambrisko             "Driver remove or shutdown called.\n");
1226665484d8SDoug Ambrisko         return ENOENT;
1227665484d8SDoug Ambrisko     }
1228665484d8SDoug Ambrisko 
1229665484d8SDoug Ambrisko     mtx_lock_spin(&sc->ioctl_lock);
1230665484d8SDoug Ambrisko     if (!sc->reset_in_progress) {
1231665484d8SDoug Ambrisko         mtx_unlock_spin(&sc->ioctl_lock);
1232665484d8SDoug Ambrisko         goto do_ioctl;
1233665484d8SDoug Ambrisko     }
1234665484d8SDoug Ambrisko 
1235665484d8SDoug Ambrisko     /* Release ioclt_lock, and wait for OCR
1236665484d8SDoug Ambrisko      * to be finished */
1237665484d8SDoug Ambrisko     mtx_unlock_spin(&sc->ioctl_lock);
1238665484d8SDoug Ambrisko     while(sc->reset_in_progress){
1239665484d8SDoug Ambrisko         i++;
1240665484d8SDoug Ambrisko         if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) {
1241665484d8SDoug Ambrisko             mrsas_dprint(sc, MRSAS_INFO,
1242665484d8SDoug Ambrisko                 "[%2d]waiting for "
1243665484d8SDoug Ambrisko                 "OCR to be finished %d\n",i,
1244665484d8SDoug Ambrisko                  sc->ocr_thread_active);
1245665484d8SDoug Ambrisko         }
1246665484d8SDoug Ambrisko         pause("mr_ioctl", hz);
1247665484d8SDoug Ambrisko     }
1248665484d8SDoug Ambrisko 
1249665484d8SDoug Ambrisko do_ioctl:
1250665484d8SDoug Ambrisko     switch (cmd) {
1251536094dcSKashyap D Desai         case MRSAS_IOC_FIRMWARE_PASS_THROUGH64:
1252536094dcSKashyap D Desai #ifdef COMPAT_FREEBSD32
1253536094dcSKashyap D Desai         case MRSAS_IOC_FIRMWARE_PASS_THROUGH32:
1254536094dcSKashyap D Desai #endif
1255536094dcSKashyap D Desai             ret = mrsas_passthru(sc, (void *)arg, cmd);
1256665484d8SDoug Ambrisko             break;
1257665484d8SDoug Ambrisko         case MRSAS_IOC_SCAN_BUS:
1258665484d8SDoug Ambrisko             ret = mrsas_bus_scan(sc);
1259665484d8SDoug Ambrisko             break;
1260536094dcSKashyap D Desai 	default:
1261536094dcSKashyap D Desai 		mrsas_dprint(sc, MRSAS_TRACE, "IOCTL command 0x%lx is not handled\n", cmd);
1262665484d8SDoug Ambrisko     }
1263665484d8SDoug Ambrisko 
1264665484d8SDoug Ambrisko     return (ret);
1265665484d8SDoug Ambrisko }
1266665484d8SDoug Ambrisko 
1267665484d8SDoug Ambrisko /**
1268665484d8SDoug Ambrisko  * mrsas_setup_irq:   Set up interrupt.
1269665484d8SDoug Ambrisko  * input:             Adapter instance soft state
1270665484d8SDoug Ambrisko  *
1271665484d8SDoug Ambrisko  * This function sets up interrupts as a bus resource, with flags indicating
1272665484d8SDoug Ambrisko  * resource permitting contemporaneous sharing and for resource to activate
1273665484d8SDoug Ambrisko  * atomically.
1274665484d8SDoug Ambrisko  */
1275665484d8SDoug Ambrisko static int mrsas_setup_irq(struct mrsas_softc *sc)
1276665484d8SDoug Ambrisko {
1277*d18d1b47SKashyap D Desai     if (sc->msix_enable && (mrsas_setup_msix(sc) == SUCCESS))
1278*d18d1b47SKashyap D Desai 		device_printf(sc->mrsas_dev, "MSI-x interrupts setup success\n");
1279665484d8SDoug Ambrisko 
1280*d18d1b47SKashyap D Desai 	else {
1281*d18d1b47SKashyap D Desai 			device_printf(sc->mrsas_dev, "Fall back to legacy interrupt\n");
1282*d18d1b47SKashyap D Desai 			sc->irq_context[0].sc = sc;
1283*d18d1b47SKashyap D Desai 			sc->irq_context[0].MSIxIndex = 0;
1284*d18d1b47SKashyap D Desai 			sc->irq_id[0] = 0;
1285*d18d1b47SKashyap D Desai 			sc->mrsas_irq[0] = bus_alloc_resource_any(sc->mrsas_dev,
1286*d18d1b47SKashyap D Desai 				SYS_RES_IRQ, &sc->irq_id[0], RF_SHAREABLE | RF_ACTIVE);
1287*d18d1b47SKashyap D Desai 			if (sc->mrsas_irq[0] == NULL){
1288*d18d1b47SKashyap D Desai 				device_printf(sc->mrsas_dev, "Cannot allocate legcay"
1289*d18d1b47SKashyap D Desai 					"interrupt\n");
1290*d18d1b47SKashyap D Desai 				return (FAIL);
1291*d18d1b47SKashyap D Desai 			}
1292*d18d1b47SKashyap D Desai 			if (bus_setup_intr(sc->mrsas_dev, sc->mrsas_irq[0],
1293*d18d1b47SKashyap D Desai 					INTR_MPSAFE|INTR_TYPE_CAM, NULL, mrsas_isr,
1294*d18d1b47SKashyap D Desai 					&sc->irq_context[0], &sc->intr_handle[0])) {
1295*d18d1b47SKashyap D Desai 					device_printf(sc->mrsas_dev, "Cannot set up legacy"
1296*d18d1b47SKashyap D Desai 						"interrupt\n");
1297*d18d1b47SKashyap D Desai 				return (FAIL);
1298*d18d1b47SKashyap D Desai 			}
1299*d18d1b47SKashyap D Desai 	}
1300665484d8SDoug Ambrisko     return (0);
1301665484d8SDoug Ambrisko }
1302665484d8SDoug Ambrisko 
1303665484d8SDoug Ambrisko /*
1304665484d8SDoug Ambrisko  * mrsas_isr:        ISR entry point
1305665484d8SDoug Ambrisko  * input:            argument pointer
1306665484d8SDoug Ambrisko  *
1307665484d8SDoug Ambrisko  * This function is the interrupt service routine entry point.  There
1308665484d8SDoug Ambrisko  * are two types of interrupts, state change interrupt and response
1309665484d8SDoug Ambrisko  * interrupt.  If an interrupt is not ours, we just return.
1310665484d8SDoug Ambrisko  */
1311665484d8SDoug Ambrisko void mrsas_isr(void *arg)
1312665484d8SDoug Ambrisko {
1313*d18d1b47SKashyap D Desai     struct mrsas_irq_context *irq_context = (struct mrsas_irq_context *)arg;
1314*d18d1b47SKashyap D Desai     struct mrsas_softc *sc = irq_context->sc;
1315*d18d1b47SKashyap D Desai     int status = 0;
1316665484d8SDoug Ambrisko 
1317*d18d1b47SKashyap D Desai 	if (!sc->msix_vectors) {
1318665484d8SDoug Ambrisko 		status = mrsas_clear_intr(sc);
1319665484d8SDoug Ambrisko 			if (!status)
1320665484d8SDoug Ambrisko 				return;
1321*d18d1b47SKashyap D Desai 	}
1322665484d8SDoug Ambrisko 
1323665484d8SDoug Ambrisko     /* If we are resetting, bail */
1324665484d8SDoug Ambrisko     if (test_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags)) {
1325665484d8SDoug Ambrisko         printf(" Entered into ISR when OCR is going active. \n");
1326665484d8SDoug Ambrisko         mrsas_clear_intr(sc);
1327665484d8SDoug Ambrisko         return;
1328665484d8SDoug Ambrisko     }
1329665484d8SDoug Ambrisko     /* Process for reply request and clear response interrupt */
1330*d18d1b47SKashyap D Desai     if (mrsas_complete_cmd(sc, irq_context->MSIxIndex) != SUCCESS)
1331665484d8SDoug Ambrisko         mrsas_clear_intr(sc);
1332665484d8SDoug Ambrisko 
1333665484d8SDoug Ambrisko     return;
1334665484d8SDoug Ambrisko }
1335665484d8SDoug Ambrisko 
1336665484d8SDoug Ambrisko /*
1337665484d8SDoug Ambrisko  * mrsas_complete_cmd:        Process reply request
1338665484d8SDoug Ambrisko  * input:                     Adapter instance soft state
1339665484d8SDoug Ambrisko  *
1340665484d8SDoug Ambrisko  * This function is called from mrsas_isr() to process reply request and
1341665484d8SDoug Ambrisko  * clear response interrupt. Processing of the reply request entails
1342665484d8SDoug Ambrisko  * walking through the reply descriptor array for the command request
1343665484d8SDoug Ambrisko  * pended from Firmware.  We look at the Function field to determine
1344665484d8SDoug Ambrisko  * the command type and perform the appropriate action.  Before we
1345665484d8SDoug Ambrisko  * return, we clear the response interrupt.
1346665484d8SDoug Ambrisko  */
1347*d18d1b47SKashyap D Desai static int mrsas_complete_cmd(struct mrsas_softc *sc, u_int32_t MSIxIndex)
1348665484d8SDoug Ambrisko {
1349665484d8SDoug Ambrisko     Mpi2ReplyDescriptorsUnion_t *desc;
1350665484d8SDoug Ambrisko     MPI2_SCSI_IO_SUCCESS_REPLY_DESCRIPTOR *reply_desc;
1351665484d8SDoug Ambrisko     MRSAS_RAID_SCSI_IO_REQUEST  *scsi_io_req;
1352665484d8SDoug Ambrisko     struct mrsas_mpt_cmd *cmd_mpt;
1353665484d8SDoug Ambrisko     struct mrsas_mfi_cmd *cmd_mfi;
1354665484d8SDoug Ambrisko     u_int8_t arm, reply_descript_type;
1355665484d8SDoug Ambrisko     u_int16_t smid, num_completed;
1356665484d8SDoug Ambrisko     u_int8_t status, extStatus;
1357665484d8SDoug Ambrisko     union desc_value desc_val;
1358665484d8SDoug Ambrisko     PLD_LOAD_BALANCE_INFO lbinfo;
1359665484d8SDoug Ambrisko     u_int32_t device_id;
1360665484d8SDoug Ambrisko     int threshold_reply_count = 0;
1361665484d8SDoug Ambrisko 
1362665484d8SDoug Ambrisko 
1363665484d8SDoug Ambrisko     /* If we have a hardware error, not need to continue */
1364665484d8SDoug Ambrisko     if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR)
1365665484d8SDoug Ambrisko         return (DONE);
1366665484d8SDoug Ambrisko 
1367665484d8SDoug Ambrisko     desc = sc->reply_desc_mem;
1368*d18d1b47SKashyap D Desai     //desc += sc->last_reply_idx[0];
1369*d18d1b47SKashyap D Desai     desc += ((MSIxIndex * sc->reply_alloc_sz)/sizeof(MPI2_REPLY_DESCRIPTORS_UNION))
1370*d18d1b47SKashyap D Desai                + sc->last_reply_idx[MSIxIndex];
1371665484d8SDoug Ambrisko 
1372665484d8SDoug Ambrisko     reply_desc = (MPI2_SCSI_IO_SUCCESS_REPLY_DESCRIPTOR *)desc;
1373665484d8SDoug Ambrisko 
1374665484d8SDoug Ambrisko     desc_val.word = desc->Words;
1375665484d8SDoug Ambrisko     num_completed = 0;
1376665484d8SDoug Ambrisko 
1377665484d8SDoug Ambrisko     reply_descript_type = reply_desc->ReplyFlags & MPI2_RPY_DESCRIPT_FLAGS_TYPE_MASK;
1378665484d8SDoug Ambrisko 
1379665484d8SDoug Ambrisko     /* Find our reply descriptor for the command and process */
1380665484d8SDoug Ambrisko     while((desc_val.u.low != 0xFFFFFFFF) && (desc_val.u.high != 0xFFFFFFFF))
1381665484d8SDoug Ambrisko     {
1382665484d8SDoug Ambrisko         smid = reply_desc->SMID;
1383665484d8SDoug Ambrisko         cmd_mpt = sc->mpt_cmd_list[smid -1];
1384665484d8SDoug Ambrisko         scsi_io_req = (MRSAS_RAID_SCSI_IO_REQUEST *)cmd_mpt->io_request;
1385665484d8SDoug Ambrisko 
1386665484d8SDoug Ambrisko         status = scsi_io_req->RaidContext.status;
1387665484d8SDoug Ambrisko         extStatus = scsi_io_req->RaidContext.exStatus;
1388665484d8SDoug Ambrisko 
1389665484d8SDoug Ambrisko         switch (scsi_io_req->Function)
1390665484d8SDoug Ambrisko         {
1391665484d8SDoug Ambrisko             case MPI2_FUNCTION_SCSI_IO_REQUEST :  /*Fast Path IO.*/
1392665484d8SDoug Ambrisko                 device_id = cmd_mpt->ccb_ptr->ccb_h.target_id;
1393665484d8SDoug Ambrisko                 lbinfo = &sc->load_balance_info[device_id];
1394665484d8SDoug Ambrisko                 if (cmd_mpt->load_balance == MRSAS_LOAD_BALANCE_FLAG) {
1395665484d8SDoug Ambrisko                     arm = lbinfo->raid1DevHandle[0] == scsi_io_req->DevHandle ? 0 : 1;
1396665484d8SDoug Ambrisko                     atomic_dec(&lbinfo->scsi_pending_cmds[arm]);
1397665484d8SDoug Ambrisko                     cmd_mpt->load_balance &= ~MRSAS_LOAD_BALANCE_FLAG;
1398665484d8SDoug Ambrisko                 }
1399665484d8SDoug Ambrisko                 //Fall thru and complete IO
1400665484d8SDoug Ambrisko             case MRSAS_MPI2_FUNCTION_LD_IO_REQUEST:
1401665484d8SDoug Ambrisko                 mrsas_map_mpt_cmd_status(cmd_mpt, status, extStatus);
1402665484d8SDoug Ambrisko                 mrsas_cmd_done(sc, cmd_mpt);
1403665484d8SDoug Ambrisko                 scsi_io_req->RaidContext.status = 0;
1404665484d8SDoug Ambrisko                 scsi_io_req->RaidContext.exStatus = 0;
1405665484d8SDoug Ambrisko                 atomic_dec(&sc->fw_outstanding);
1406665484d8SDoug Ambrisko                 break;
1407665484d8SDoug Ambrisko             case MRSAS_MPI2_FUNCTION_PASSTHRU_IO_REQUEST: /*MFI command */
1408665484d8SDoug Ambrisko                 cmd_mfi = sc->mfi_cmd_list[cmd_mpt->sync_cmd_idx];
1409665484d8SDoug Ambrisko                 mrsas_complete_mptmfi_passthru(sc, cmd_mfi, status);
1410665484d8SDoug Ambrisko                 cmd_mpt->flags = 0;
1411665484d8SDoug Ambrisko                 mrsas_release_mpt_cmd(cmd_mpt);
1412665484d8SDoug Ambrisko                 break;
1413665484d8SDoug Ambrisko         }
1414665484d8SDoug Ambrisko 
1415*d18d1b47SKashyap D Desai         sc->last_reply_idx[MSIxIndex]++;
1416*d18d1b47SKashyap D Desai         if (sc->last_reply_idx[MSIxIndex] >= sc->reply_q_depth)
1417*d18d1b47SKashyap D Desai             sc->last_reply_idx[MSIxIndex] = 0;
1418665484d8SDoug Ambrisko 
1419665484d8SDoug Ambrisko         desc->Words = ~((uint64_t)0x00); /* set it back to all 0xFFFFFFFFs */
1420665484d8SDoug Ambrisko         num_completed++;
1421665484d8SDoug Ambrisko         threshold_reply_count++;
1422665484d8SDoug Ambrisko 
1423665484d8SDoug Ambrisko         /* Get the next reply descriptor */
1424*d18d1b47SKashyap D Desai         if (!sc->last_reply_idx[MSIxIndex]){
1425665484d8SDoug Ambrisko             desc = sc->reply_desc_mem;
1426*d18d1b47SKashyap D Desai             desc += ((MSIxIndex * sc->reply_alloc_sz)/sizeof(MPI2_REPLY_DESCRIPTORS_UNION));
1427*d18d1b47SKashyap D Desai         } else
1428665484d8SDoug Ambrisko             desc++;
1429665484d8SDoug Ambrisko 
1430665484d8SDoug Ambrisko         reply_desc = (MPI2_SCSI_IO_SUCCESS_REPLY_DESCRIPTOR *)desc;
1431665484d8SDoug Ambrisko         desc_val.word = desc->Words;
1432665484d8SDoug Ambrisko 
1433665484d8SDoug Ambrisko         reply_descript_type = reply_desc->ReplyFlags & MPI2_RPY_DESCRIPT_FLAGS_TYPE_MASK;
1434665484d8SDoug Ambrisko 
1435665484d8SDoug Ambrisko         if(reply_descript_type == MPI2_RPY_DESCRIPT_FLAGS_UNUSED)
1436665484d8SDoug Ambrisko             break;
1437665484d8SDoug Ambrisko 
1438665484d8SDoug Ambrisko         /*
1439665484d8SDoug Ambrisko          * Write to reply post index after completing threshold reply count
1440665484d8SDoug Ambrisko          * and still there are more replies in reply queue pending to be
1441665484d8SDoug Ambrisko          * completed.
1442665484d8SDoug Ambrisko          */
1443665484d8SDoug Ambrisko         if (threshold_reply_count >= THRESHOLD_REPLY_COUNT) {
1444*d18d1b47SKashyap D Desai 				if (sc->msix_enable) {
1445*d18d1b47SKashyap D Desai 					if ((sc->device_id == MRSAS_INVADER) ||
1446*d18d1b47SKashyap D Desai 						(sc->device_id == MRSAS_FURY))
1447*d18d1b47SKashyap D Desai 						mrsas_write_reg(sc, sc->msix_reg_offset[MSIxIndex/8],
1448*d18d1b47SKashyap D Desai 						   ((MSIxIndex & 0x7) << 24) |
1449*d18d1b47SKashyap D Desai 						   sc->last_reply_idx[MSIxIndex]);
1450*d18d1b47SKashyap D Desai 					 else
1451*d18d1b47SKashyap D Desai 						mrsas_write_reg(sc, sc->msix_reg_offset[0], (MSIxIndex << 24) |
1452*d18d1b47SKashyap D Desai 						   sc->last_reply_idx[MSIxIndex]);
1453*d18d1b47SKashyap D Desai 				 } else
1454*d18d1b47SKashyap D Desai 					mrsas_write_reg(sc, offsetof(mrsas_reg_set,
1455*d18d1b47SKashyap D Desai 					 reply_post_host_index),sc->last_reply_idx[0]);
1456*d18d1b47SKashyap D Desai 
1457665484d8SDoug Ambrisko 				 threshold_reply_count = 0;
1458665484d8SDoug Ambrisko 				}
1459665484d8SDoug Ambrisko         }
1460665484d8SDoug Ambrisko 
1461665484d8SDoug Ambrisko     /* No match, just return */
1462665484d8SDoug Ambrisko     if (num_completed == 0)
1463665484d8SDoug Ambrisko         return (DONE);
1464665484d8SDoug Ambrisko 
1465665484d8SDoug Ambrisko     /* Clear response interrupt */
1466*d18d1b47SKashyap D Desai      if (sc->msix_enable) {
1467*d18d1b47SKashyap D Desai 	     if ((sc->device_id == MRSAS_INVADER) ||
1468*d18d1b47SKashyap D Desai                     (sc->device_id == MRSAS_FURY)){
1469*d18d1b47SKashyap D Desai 		    mrsas_write_reg(sc, sc->msix_reg_offset[MSIxIndex/8],
1470*d18d1b47SKashyap D Desai                        ((MSIxIndex & 0x7) << 24) |
1471*d18d1b47SKashyap D Desai                        sc->last_reply_idx[MSIxIndex]);
1472*d18d1b47SKashyap D Desai 		} else
1473*d18d1b47SKashyap D Desai 		    mrsas_write_reg(sc, sc->msix_reg_offset[0], (MSIxIndex << 24) |
1474*d18d1b47SKashyap D Desai                        sc->last_reply_idx[MSIxIndex]);
1475*d18d1b47SKashyap D Desai      } else
1476*d18d1b47SKashyap D Desai             mrsas_write_reg(sc, offsetof(mrsas_reg_set,
1477*d18d1b47SKashyap D Desai                  reply_post_host_index),sc->last_reply_idx[0]);
1478665484d8SDoug Ambrisko 
1479665484d8SDoug Ambrisko     return(0);
1480665484d8SDoug Ambrisko }
1481665484d8SDoug Ambrisko 
1482665484d8SDoug Ambrisko /*
1483665484d8SDoug Ambrisko  * mrsas_map_mpt_cmd_status:  Allocate DMAable memory.
1484665484d8SDoug Ambrisko  * input:                     Adapter instance soft state
1485665484d8SDoug Ambrisko  *
1486665484d8SDoug Ambrisko  * This function is called from mrsas_complete_cmd(), for LD IO and FastPath IO.
1487665484d8SDoug Ambrisko  * It checks the command status and maps the appropriate CAM status for the CCB.
1488665484d8SDoug Ambrisko  */
1489665484d8SDoug Ambrisko void mrsas_map_mpt_cmd_status(struct mrsas_mpt_cmd *cmd, u_int8_t status, u_int8_t extStatus)
1490665484d8SDoug Ambrisko {
1491665484d8SDoug Ambrisko     struct mrsas_softc *sc = cmd->sc;
1492665484d8SDoug Ambrisko     u_int8_t *sense_data;
1493665484d8SDoug Ambrisko 
1494665484d8SDoug Ambrisko     switch (status) {
1495665484d8SDoug Ambrisko         case MFI_STAT_OK:
1496665484d8SDoug Ambrisko             cmd->ccb_ptr->ccb_h.status = CAM_REQ_CMP;
1497665484d8SDoug Ambrisko             break;
1498665484d8SDoug Ambrisko         case MFI_STAT_SCSI_IO_FAILED:
1499665484d8SDoug Ambrisko         case MFI_STAT_SCSI_DONE_WITH_ERROR:
1500665484d8SDoug Ambrisko             cmd->ccb_ptr->ccb_h.status = CAM_SCSI_STATUS_ERROR;
1501665484d8SDoug Ambrisko             sense_data = (u_int8_t *)&cmd->ccb_ptr->csio.sense_data;
1502665484d8SDoug Ambrisko             if (sense_data) {
1503665484d8SDoug Ambrisko                 /* For now just copy 18 bytes back */
1504665484d8SDoug Ambrisko                 memcpy(sense_data, cmd->sense, 18);
1505665484d8SDoug Ambrisko                 cmd->ccb_ptr->csio.sense_len = 18;
1506665484d8SDoug Ambrisko                 cmd->ccb_ptr->ccb_h.status |= CAM_AUTOSNS_VALID;
1507665484d8SDoug Ambrisko             }
1508665484d8SDoug Ambrisko             break;
1509665484d8SDoug Ambrisko         case MFI_STAT_LD_OFFLINE:
1510665484d8SDoug Ambrisko         case MFI_STAT_DEVICE_NOT_FOUND:
1511665484d8SDoug Ambrisko             if (cmd->ccb_ptr->ccb_h.target_lun)
1512665484d8SDoug Ambrisko                 cmd->ccb_ptr->ccb_h.status |= CAM_LUN_INVALID;
1513665484d8SDoug Ambrisko             else
1514665484d8SDoug Ambrisko                 cmd->ccb_ptr->ccb_h.status |= CAM_DEV_NOT_THERE;
1515665484d8SDoug Ambrisko             break;
1516665484d8SDoug Ambrisko         case MFI_STAT_CONFIG_SEQ_MISMATCH:
1517665484d8SDoug Ambrisko             /*send status to CAM layer to retry sending  command without
1518665484d8SDoug Ambrisko              * decrementing retry counter*/
1519665484d8SDoug Ambrisko             cmd->ccb_ptr->ccb_h.status |= CAM_REQUEUE_REQ;
1520665484d8SDoug Ambrisko             break;
1521665484d8SDoug Ambrisko         default:
1522665484d8SDoug Ambrisko             device_printf(sc->mrsas_dev, "FW cmd complete status %x\n", status);
1523665484d8SDoug Ambrisko             cmd->ccb_ptr->ccb_h.status = CAM_REQ_CMP_ERR;
1524665484d8SDoug Ambrisko             cmd->ccb_ptr->csio.scsi_status = status;
1525665484d8SDoug Ambrisko     }
1526665484d8SDoug Ambrisko     return;
1527665484d8SDoug Ambrisko }
1528665484d8SDoug Ambrisko 
1529665484d8SDoug Ambrisko /*
1530665484d8SDoug Ambrisko  * mrsas_alloc_mem:  Allocate DMAable memory.
1531665484d8SDoug Ambrisko  * input:            Adapter instance soft state
1532665484d8SDoug Ambrisko  *
1533665484d8SDoug Ambrisko  * This function creates the parent DMA tag and allocates DMAable memory.
1534665484d8SDoug Ambrisko  * DMA tag describes constraints of DMA mapping. Memory allocated is mapped
1535665484d8SDoug Ambrisko  * into Kernel virtual address. Callback argument is physical memory address.
1536665484d8SDoug Ambrisko  */
1537665484d8SDoug Ambrisko static int mrsas_alloc_mem(struct mrsas_softc *sc)
1538665484d8SDoug Ambrisko {
1539665484d8SDoug Ambrisko     u_int32_t verbuf_size, io_req_size, reply_desc_size, sense_size,
1540*d18d1b47SKashyap D Desai               chain_frame_size, evt_detail_size, count;
1541665484d8SDoug Ambrisko 
1542665484d8SDoug Ambrisko     /*
1543665484d8SDoug Ambrisko      * Allocate parent DMA tag
1544665484d8SDoug Ambrisko      */
1545665484d8SDoug Ambrisko     if (bus_dma_tag_create(NULL,                   /* parent */
1546665484d8SDoug Ambrisko                            1,         /* alignment */
1547665484d8SDoug Ambrisko                            0,                      /* boundary */
1548665484d8SDoug Ambrisko                            BUS_SPACE_MAXADDR,     /* lowaddr */
1549665484d8SDoug Ambrisko                            BUS_SPACE_MAXADDR,      /* highaddr */
1550665484d8SDoug Ambrisko                            NULL, NULL,             /* filter, filterarg */
1551665484d8SDoug Ambrisko                            MRSAS_MAX_IO_SIZE,/* maxsize */
1552665484d8SDoug Ambrisko                            MRSAS_MAX_SGL, /* nsegments */
1553665484d8SDoug Ambrisko                            MRSAS_MAX_IO_SIZE,/* maxsegsize */
1554665484d8SDoug Ambrisko                            0,                      /* flags */
1555665484d8SDoug Ambrisko                            NULL, NULL,             /* lockfunc, lockarg */
1556665484d8SDoug Ambrisko                            &sc->mrsas_parent_tag   /* tag */
1557665484d8SDoug Ambrisko                            )) {
1558665484d8SDoug Ambrisko            device_printf(sc->mrsas_dev, "Cannot allocate parent DMA tag\n");
1559665484d8SDoug Ambrisko            return(ENOMEM);
1560665484d8SDoug Ambrisko     }
1561665484d8SDoug Ambrisko 
1562665484d8SDoug Ambrisko     /*
1563665484d8SDoug Ambrisko      * Allocate for version buffer
1564665484d8SDoug Ambrisko      */
1565665484d8SDoug Ambrisko     verbuf_size = MRSAS_MAX_NAME_LENGTH*(sizeof(bus_addr_t));
1566665484d8SDoug Ambrisko     if (bus_dma_tag_create(sc->mrsas_parent_tag,   // parent
1567665484d8SDoug Ambrisko                            1, 0,                   // algnmnt, boundary
1568665484d8SDoug Ambrisko                            BUS_SPACE_MAXADDR_32BIT,// lowaddr
1569665484d8SDoug Ambrisko                            BUS_SPACE_MAXADDR,      // highaddr
1570665484d8SDoug Ambrisko                            NULL, NULL,             // filter, filterarg
1571665484d8SDoug Ambrisko                            verbuf_size,           // maxsize
1572665484d8SDoug Ambrisko                            1,                      // msegments
1573665484d8SDoug Ambrisko                            verbuf_size,           // maxsegsize
1574665484d8SDoug Ambrisko                            BUS_DMA_ALLOCNOW,       // flags
1575665484d8SDoug Ambrisko                            NULL, NULL,             // lockfunc, lockarg
1576665484d8SDoug Ambrisko                            &sc->verbuf_tag)) {
1577665484d8SDoug Ambrisko             device_printf(sc->mrsas_dev, "Cannot allocate verbuf DMA tag\n");
1578665484d8SDoug Ambrisko             return (ENOMEM);
1579665484d8SDoug Ambrisko     }
1580665484d8SDoug Ambrisko     if (bus_dmamem_alloc(sc->verbuf_tag, (void **)&sc->verbuf_mem,
1581665484d8SDoug Ambrisko         BUS_DMA_NOWAIT, &sc->verbuf_dmamap)) {
1582665484d8SDoug Ambrisko             device_printf(sc->mrsas_dev, "Cannot allocate verbuf memory\n");
1583665484d8SDoug Ambrisko             return (ENOMEM);
1584665484d8SDoug Ambrisko     }
1585665484d8SDoug Ambrisko     bzero(sc->verbuf_mem, verbuf_size);
1586665484d8SDoug Ambrisko     if (bus_dmamap_load(sc->verbuf_tag, sc->verbuf_dmamap, sc->verbuf_mem,
1587665484d8SDoug Ambrisko         verbuf_size, mrsas_addr_cb, &sc->verbuf_phys_addr, BUS_DMA_NOWAIT)){
1588665484d8SDoug Ambrisko             device_printf(sc->mrsas_dev, "Cannot load verbuf DMA map\n");
1589665484d8SDoug Ambrisko             return(ENOMEM);
1590665484d8SDoug Ambrisko     }
1591665484d8SDoug Ambrisko 
1592665484d8SDoug Ambrisko     /*
1593665484d8SDoug Ambrisko      * Allocate IO Request Frames
1594665484d8SDoug Ambrisko      */
1595665484d8SDoug Ambrisko     io_req_size = sc->io_frames_alloc_sz;
1596665484d8SDoug Ambrisko     if (bus_dma_tag_create( sc->mrsas_parent_tag,   // parent
1597665484d8SDoug Ambrisko                             16, 0,                   // algnmnt, boundary
1598665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR_32BIT,// lowaddr
1599665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR,      // highaddr
1600665484d8SDoug Ambrisko                             NULL, NULL,             // filter, filterarg
1601665484d8SDoug Ambrisko                             io_req_size,            // maxsize
1602665484d8SDoug Ambrisko                             1,                      // msegments
1603665484d8SDoug Ambrisko                             io_req_size,            // maxsegsize
1604665484d8SDoug Ambrisko                             BUS_DMA_ALLOCNOW,       // flags
1605665484d8SDoug Ambrisko                             NULL, NULL,             // lockfunc, lockarg
1606665484d8SDoug Ambrisko                             &sc->io_request_tag)) {
1607665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot create IO request tag\n");
1608665484d8SDoug Ambrisko         return (ENOMEM);
1609665484d8SDoug Ambrisko     }
1610665484d8SDoug Ambrisko     if (bus_dmamem_alloc(sc->io_request_tag, (void **)&sc->io_request_mem,
1611665484d8SDoug Ambrisko                     BUS_DMA_NOWAIT, &sc->io_request_dmamap)) {
1612665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot alloc IO request memory\n");
1613665484d8SDoug Ambrisko         return (ENOMEM);
1614665484d8SDoug Ambrisko     }
1615665484d8SDoug Ambrisko     bzero(sc->io_request_mem, io_req_size);
1616665484d8SDoug Ambrisko     if (bus_dmamap_load(sc->io_request_tag, sc->io_request_dmamap,
1617665484d8SDoug Ambrisko                         sc->io_request_mem, io_req_size, mrsas_addr_cb,
1618665484d8SDoug Ambrisko                         &sc->io_request_phys_addr, BUS_DMA_NOWAIT)) {
1619665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot load IO request memory\n");
1620665484d8SDoug Ambrisko         return (ENOMEM);
1621665484d8SDoug Ambrisko     }
1622665484d8SDoug Ambrisko 
1623665484d8SDoug Ambrisko     /*
1624665484d8SDoug Ambrisko      * Allocate Chain Frames
1625665484d8SDoug Ambrisko      */
1626665484d8SDoug Ambrisko     chain_frame_size = sc->chain_frames_alloc_sz;
1627665484d8SDoug Ambrisko     if (bus_dma_tag_create( sc->mrsas_parent_tag,   // parent
1628665484d8SDoug Ambrisko                             4, 0,                   // algnmnt, boundary
1629665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR_32BIT,// lowaddr
1630665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR,      // highaddr
1631665484d8SDoug Ambrisko                             NULL, NULL,             // filter, filterarg
1632665484d8SDoug Ambrisko                             chain_frame_size,       // maxsize
1633665484d8SDoug Ambrisko                             1,                      // msegments
1634665484d8SDoug Ambrisko                             chain_frame_size,       // maxsegsize
1635665484d8SDoug Ambrisko                             BUS_DMA_ALLOCNOW,       // flags
1636665484d8SDoug Ambrisko                             NULL, NULL,             // lockfunc, lockarg
1637665484d8SDoug Ambrisko                             &sc->chain_frame_tag)) {
1638665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot create chain frame tag\n");
1639665484d8SDoug Ambrisko         return (ENOMEM);
1640665484d8SDoug Ambrisko     }
1641665484d8SDoug Ambrisko     if (bus_dmamem_alloc(sc->chain_frame_tag, (void **)&sc->chain_frame_mem,
1642665484d8SDoug Ambrisko                     BUS_DMA_NOWAIT, &sc->chain_frame_dmamap)) {
1643665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot alloc chain frame memory\n");
1644665484d8SDoug Ambrisko         return (ENOMEM);
1645665484d8SDoug Ambrisko     }
1646665484d8SDoug Ambrisko     bzero(sc->chain_frame_mem, chain_frame_size);
1647665484d8SDoug Ambrisko     if (bus_dmamap_load(sc->chain_frame_tag, sc->chain_frame_dmamap,
1648665484d8SDoug Ambrisko                         sc->chain_frame_mem, chain_frame_size, mrsas_addr_cb,
1649665484d8SDoug Ambrisko                         &sc->chain_frame_phys_addr, BUS_DMA_NOWAIT)) {
1650665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot load chain frame memory\n");
1651665484d8SDoug Ambrisko         return (ENOMEM);
1652665484d8SDoug Ambrisko     }
1653665484d8SDoug Ambrisko 
1654*d18d1b47SKashyap D Desai     count = sc->msix_vectors > 0 ? sc->msix_vectors : 1;
1655665484d8SDoug Ambrisko     /*
1656665484d8SDoug Ambrisko      * Allocate Reply Descriptor Array
1657665484d8SDoug Ambrisko      */
1658*d18d1b47SKashyap D Desai     reply_desc_size = sc->reply_alloc_sz * count;
1659665484d8SDoug Ambrisko     if (bus_dma_tag_create( sc->mrsas_parent_tag,   // parent
1660665484d8SDoug Ambrisko                             16, 0,                   // algnmnt, boundary
1661665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR_32BIT,// lowaddr
1662665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR,      // highaddr
1663665484d8SDoug Ambrisko                             NULL, NULL,             // filter, filterarg
1664665484d8SDoug Ambrisko                             reply_desc_size,        // maxsize
1665665484d8SDoug Ambrisko                             1,                      // msegments
1666665484d8SDoug Ambrisko                             reply_desc_size,        // maxsegsize
1667665484d8SDoug Ambrisko                             BUS_DMA_ALLOCNOW,       // flags
1668665484d8SDoug Ambrisko                             NULL, NULL,             // lockfunc, lockarg
1669665484d8SDoug Ambrisko                             &sc->reply_desc_tag)) {
1670665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot create reply descriptor tag\n");
1671665484d8SDoug Ambrisko         return (ENOMEM);
1672665484d8SDoug Ambrisko     }
1673665484d8SDoug Ambrisko     if (bus_dmamem_alloc(sc->reply_desc_tag, (void **)&sc->reply_desc_mem,
1674665484d8SDoug Ambrisko                     BUS_DMA_NOWAIT, &sc->reply_desc_dmamap)) {
1675665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot alloc reply descriptor memory\n");
1676665484d8SDoug Ambrisko         return (ENOMEM);
1677665484d8SDoug Ambrisko     }
1678665484d8SDoug Ambrisko     if (bus_dmamap_load(sc->reply_desc_tag, sc->reply_desc_dmamap,
1679665484d8SDoug Ambrisko                         sc->reply_desc_mem, reply_desc_size, mrsas_addr_cb,
1680665484d8SDoug Ambrisko                         &sc->reply_desc_phys_addr, BUS_DMA_NOWAIT)) {
1681665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot load reply descriptor memory\n");
1682665484d8SDoug Ambrisko         return (ENOMEM);
1683665484d8SDoug Ambrisko     }
1684665484d8SDoug Ambrisko 
1685665484d8SDoug Ambrisko     /*
1686665484d8SDoug Ambrisko      * Allocate Sense Buffer Array.  Keep in lower 4GB
1687665484d8SDoug Ambrisko      */
1688665484d8SDoug Ambrisko     sense_size = sc->max_fw_cmds * MRSAS_SENSE_LEN;
1689665484d8SDoug Ambrisko     if (bus_dma_tag_create(sc->mrsas_parent_tag,    // parent
1690665484d8SDoug Ambrisko                             64, 0,                   // algnmnt, boundary
1691665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR_32BIT,// lowaddr
1692665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR,      // highaddr
1693665484d8SDoug Ambrisko                             NULL, NULL,             // filter, filterarg
1694665484d8SDoug Ambrisko                             sense_size,             // maxsize
1695665484d8SDoug Ambrisko                             1,                      // nsegments
1696665484d8SDoug Ambrisko                             sense_size,             // maxsegsize
1697665484d8SDoug Ambrisko                             BUS_DMA_ALLOCNOW,       // flags
1698665484d8SDoug Ambrisko                             NULL, NULL,             // lockfunc, lockarg
1699665484d8SDoug Ambrisko                             &sc->sense_tag)) {
1700665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot allocate sense buf tag\n");
1701665484d8SDoug Ambrisko         return (ENOMEM);
1702665484d8SDoug Ambrisko     }
1703665484d8SDoug Ambrisko     if (bus_dmamem_alloc(sc->sense_tag, (void **)&sc->sense_mem,
1704665484d8SDoug Ambrisko             BUS_DMA_NOWAIT, &sc->sense_dmamap)) {
1705665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot allocate sense buf memory\n");
1706665484d8SDoug Ambrisko         return (ENOMEM);
1707665484d8SDoug Ambrisko     }
1708665484d8SDoug Ambrisko     if (bus_dmamap_load(sc->sense_tag, sc->sense_dmamap,
1709665484d8SDoug Ambrisko             sc->sense_mem, sense_size, mrsas_addr_cb, &sc->sense_phys_addr,
1710665484d8SDoug Ambrisko             BUS_DMA_NOWAIT)){
1711665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot load sense buf memory\n");
1712665484d8SDoug Ambrisko         return (ENOMEM);
1713665484d8SDoug Ambrisko     }
1714665484d8SDoug Ambrisko 
1715665484d8SDoug Ambrisko     /*
1716665484d8SDoug Ambrisko      * Allocate for Event detail structure
1717665484d8SDoug Ambrisko      */
1718665484d8SDoug Ambrisko     evt_detail_size = sizeof(struct mrsas_evt_detail);
1719665484d8SDoug Ambrisko     if (bus_dma_tag_create( sc->mrsas_parent_tag,   // parent
1720665484d8SDoug Ambrisko                             1, 0,                   // algnmnt, boundary
1721665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR_32BIT,// lowaddr
1722665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR,      // highaddr
1723665484d8SDoug Ambrisko                             NULL, NULL,             // filter, filterarg
1724665484d8SDoug Ambrisko                             evt_detail_size,        // maxsize
1725665484d8SDoug Ambrisko                             1,                      // msegments
1726665484d8SDoug Ambrisko                             evt_detail_size,        // maxsegsize
1727665484d8SDoug Ambrisko                             BUS_DMA_ALLOCNOW,       // flags
1728665484d8SDoug Ambrisko                             NULL, NULL,             // lockfunc, lockarg
1729665484d8SDoug Ambrisko                             &sc->evt_detail_tag)) {
1730665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot create Event detail tag\n");
1731665484d8SDoug Ambrisko         return (ENOMEM);
1732665484d8SDoug Ambrisko     }
1733665484d8SDoug Ambrisko     if (bus_dmamem_alloc(sc->evt_detail_tag, (void **)&sc->evt_detail_mem,
1734665484d8SDoug Ambrisko                     BUS_DMA_NOWAIT, &sc->evt_detail_dmamap)) {
1735665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot alloc Event detail buffer memory\n");
1736665484d8SDoug Ambrisko         return (ENOMEM);
1737665484d8SDoug Ambrisko     }
1738665484d8SDoug Ambrisko     bzero(sc->evt_detail_mem, evt_detail_size);
1739665484d8SDoug Ambrisko     if (bus_dmamap_load(sc->evt_detail_tag, sc->evt_detail_dmamap,
1740665484d8SDoug Ambrisko                         sc->evt_detail_mem, evt_detail_size, mrsas_addr_cb,
1741665484d8SDoug Ambrisko                         &sc->evt_detail_phys_addr, BUS_DMA_NOWAIT)) {
1742665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot load Event detail buffer memory\n");
1743665484d8SDoug Ambrisko         return (ENOMEM);
1744665484d8SDoug Ambrisko     }
1745665484d8SDoug Ambrisko 
1746665484d8SDoug Ambrisko 
1747665484d8SDoug Ambrisko    /*
1748665484d8SDoug Ambrisko     * Create a dma tag for data buffers; size will be the maximum
1749665484d8SDoug Ambrisko     * possible I/O size (280kB).
1750665484d8SDoug Ambrisko     */
1751665484d8SDoug Ambrisko     if (bus_dma_tag_create(sc->mrsas_parent_tag,   // parent
1752665484d8SDoug Ambrisko                            1,         // alignment
1753665484d8SDoug Ambrisko                            0,                      // boundary
1754665484d8SDoug Ambrisko                            BUS_SPACE_MAXADDR,      // lowaddr
1755665484d8SDoug Ambrisko                            BUS_SPACE_MAXADDR,      // highaddr
1756665484d8SDoug Ambrisko                            NULL, NULL,             // filter, filterarg
1757665484d8SDoug Ambrisko                            MRSAS_MAX_IO_SIZE,      // maxsize
1758665484d8SDoug Ambrisko                            MRSAS_MAX_SGL,          // nsegments
1759665484d8SDoug Ambrisko                            MRSAS_MAX_IO_SIZE,      // maxsegsize
1760665484d8SDoug Ambrisko                            BUS_DMA_ALLOCNOW,       // flags
1761665484d8SDoug Ambrisko                            busdma_lock_mutex,      // lockfunc
1762665484d8SDoug Ambrisko                            &sc->io_lock,           // lockfuncarg
1763665484d8SDoug Ambrisko                            &sc->data_tag)) {
1764665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot create data dma tag\n");
1765665484d8SDoug Ambrisko         return(ENOMEM);
1766665484d8SDoug Ambrisko     }
1767665484d8SDoug Ambrisko 
1768665484d8SDoug Ambrisko     return(0);
1769665484d8SDoug Ambrisko }
1770665484d8SDoug Ambrisko 
1771665484d8SDoug Ambrisko /*
1772665484d8SDoug Ambrisko  * mrsas_addr_cb:   Callback function of bus_dmamap_load()
1773665484d8SDoug Ambrisko  * input:           callback argument,
1774665484d8SDoug Ambrisko  *                  machine dependent type that describes DMA segments,
1775665484d8SDoug Ambrisko  *                  number of segments,
1776665484d8SDoug Ambrisko  *                  error code.
1777665484d8SDoug Ambrisko  *
1778665484d8SDoug Ambrisko  * This function is for the driver to receive mapping information resultant
1779665484d8SDoug Ambrisko  * of the bus_dmamap_load(). The information is actually not being used,
1780665484d8SDoug Ambrisko  * but the address is saved anyway.
1781665484d8SDoug Ambrisko  */
1782665484d8SDoug Ambrisko void
1783665484d8SDoug Ambrisko mrsas_addr_cb(void *arg, bus_dma_segment_t *segs, int nsegs, int error)
1784665484d8SDoug Ambrisko {
1785665484d8SDoug Ambrisko     bus_addr_t *addr;
1786665484d8SDoug Ambrisko 
1787665484d8SDoug Ambrisko     addr = arg;
1788665484d8SDoug Ambrisko     *addr = segs[0].ds_addr;
1789665484d8SDoug Ambrisko }
1790665484d8SDoug Ambrisko 
1791665484d8SDoug Ambrisko /*
1792665484d8SDoug Ambrisko  * mrsas_setup_raidmap:  Set up RAID map.
1793665484d8SDoug Ambrisko  * input:                Adapter instance soft state
1794665484d8SDoug Ambrisko  *
1795665484d8SDoug Ambrisko  * Allocate DMA memory for the RAID maps and perform setup.
1796665484d8SDoug Ambrisko  */
1797665484d8SDoug Ambrisko static int mrsas_setup_raidmap(struct mrsas_softc *sc)
1798665484d8SDoug Ambrisko {
17994799d485SKashyap D Desai 	int i;
18004799d485SKashyap D Desai 
18014799d485SKashyap D Desai 	sc->drv_supported_vd_count =
18024799d485SKashyap D Desai 		MRSAS_MAX_LD_CHANNELS * MRSAS_MAX_DEV_PER_CHANNEL;
18034799d485SKashyap D Desai 	sc->drv_supported_pd_count =
18044799d485SKashyap D Desai 		MRSAS_MAX_PD_CHANNELS * MRSAS_MAX_DEV_PER_CHANNEL;
18054799d485SKashyap D Desai 
18064799d485SKashyap D Desai 	if(sc->max256vdSupport) {
18074799d485SKashyap D Desai 		sc->fw_supported_vd_count = MAX_LOGICAL_DRIVES_EXT;
18084799d485SKashyap D Desai 		sc->fw_supported_pd_count = MAX_PHYSICAL_DEVICES;
18094799d485SKashyap D Desai 	} else {
18104799d485SKashyap D Desai 		sc->fw_supported_vd_count = MAX_LOGICAL_DRIVES;
18114799d485SKashyap D Desai 		sc->fw_supported_pd_count = MAX_PHYSICAL_DEVICES;
18124799d485SKashyap D Desai 	}
18134799d485SKashyap D Desai 
18144799d485SKashyap D Desai #if VD_EXT_DEBUG
18154799d485SKashyap D Desai 	device_printf(sc->mrsas_dev, "FW supports: max256vdSupport = %s\n",
18164799d485SKashyap D Desai 			sc->max256vdSupport ? "YES":"NO");
18174799d485SKashyap D Desai 	device_printf(sc->mrsas_dev, "FW supports %dVDs %dPDs\n"
18184799d485SKashyap D Desai 		"DRIVER supports %dVDs  %dPDs \n",
18194799d485SKashyap D Desai 		sc->fw_supported_vd_count, sc->fw_supported_pd_count,
18204799d485SKashyap D Desai 		sc->drv_supported_vd_count, sc->drv_supported_pd_count);
18214799d485SKashyap D Desai #endif
18224799d485SKashyap D Desai 
18234799d485SKashyap D Desai 	sc->old_map_sz = sizeof(MR_FW_RAID_MAP) +
18244799d485SKashyap D Desai 		(sizeof(MR_LD_SPAN_MAP) * (sc->fw_supported_vd_count - 1));
18254799d485SKashyap D Desai 	sc->new_map_sz = sizeof(MR_FW_RAID_MAP_EXT);
18264799d485SKashyap D Desai 	sc->drv_map_sz = sizeof(MR_DRV_RAID_MAP) +
18274799d485SKashyap D Desai 		(sizeof(MR_LD_SPAN_MAP) * (sc->drv_supported_vd_count-1));
18284799d485SKashyap D Desai 
18294799d485SKashyap D Desai 	for (i = 0; i < 2; i++) {
18304799d485SKashyap D Desai 		sc->ld_drv_map[i] =
18314799d485SKashyap D Desai 			(void*) malloc(sc->drv_map_sz, M_MRSAS, M_NOWAIT);
18324799d485SKashyap D Desai 		/* Do Error handling */
18334799d485SKashyap D Desai 		if (!sc->ld_drv_map[i]) {
18344799d485SKashyap D Desai 			device_printf(sc->mrsas_dev, "Could not allocate memory for local map");
18354799d485SKashyap D Desai 
18364799d485SKashyap D Desai 			if (i == 1)
18374799d485SKashyap D Desai 				free (sc->ld_drv_map[0], M_MRSAS);
18384799d485SKashyap D Desai 			//ABORT driver initialization
18394799d485SKashyap D Desai 			goto ABORT;
18404799d485SKashyap D Desai 		}
18414799d485SKashyap D Desai 	}
18424799d485SKashyap D Desai 
18434799d485SKashyap D Desai     sc->max_map_sz = max(sc->old_map_sz, sc->new_map_sz);
18444799d485SKashyap D Desai 
18454799d485SKashyap D Desai     if(sc->max256vdSupport)
18464799d485SKashyap D Desai 	sc->current_map_sz = sc->new_map_sz;
18474799d485SKashyap D Desai     else
18484799d485SKashyap D Desai 	sc->current_map_sz = sc->old_map_sz;
18494799d485SKashyap D Desai 
1850665484d8SDoug Ambrisko 
1851665484d8SDoug Ambrisko     for (int i=0; i < 2; i++)
1852665484d8SDoug Ambrisko     {
1853665484d8SDoug Ambrisko         if (bus_dma_tag_create(sc->mrsas_parent_tag,    // parent
1854665484d8SDoug Ambrisko                             4, 0,                   // algnmnt, boundary
1855665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR_32BIT,// lowaddr
1856665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR,      // highaddr
1857665484d8SDoug Ambrisko                             NULL, NULL,             // filter, filterarg
18584799d485SKashyap D Desai                             sc->max_map_sz,         // maxsize
1859665484d8SDoug Ambrisko                             1,                      // nsegments
18604799d485SKashyap D Desai                             sc->max_map_sz,         // maxsegsize
1861665484d8SDoug Ambrisko                             BUS_DMA_ALLOCNOW,       // flags
1862665484d8SDoug Ambrisko                             NULL, NULL,             // lockfunc, lockarg
1863665484d8SDoug Ambrisko                             &sc->raidmap_tag[i])) {
18644799d485SKashyap D Desai 		device_printf(sc->mrsas_dev,
18654799d485SKashyap D Desai 				"Cannot allocate raid map tag.\n");
1866665484d8SDoug Ambrisko 		return (ENOMEM);
1867665484d8SDoug Ambrisko         }
18684799d485SKashyap D Desai         if (bus_dmamem_alloc(sc->raidmap_tag[i],
18694799d485SKashyap D Desai 				(void **)&sc->raidmap_mem[i],
1870665484d8SDoug Ambrisko 				BUS_DMA_NOWAIT, &sc->raidmap_dmamap[i])) {
18714799d485SKashyap D Desai 		device_printf(sc->mrsas_dev,
18724799d485SKashyap D Desai 				"Cannot allocate raidmap memory.\n");
1873665484d8SDoug Ambrisko 		return (ENOMEM);
1874665484d8SDoug Ambrisko         }
18754799d485SKashyap D Desai 
18764799d485SKashyap D Desai 	bzero (sc->raidmap_mem[i], sc->max_map_sz);
18774799d485SKashyap D Desai 
1878665484d8SDoug Ambrisko         if (bus_dmamap_load(sc->raidmap_tag[i], sc->raidmap_dmamap[i],
18794799d485SKashyap D Desai 				sc->raidmap_mem[i], sc->max_map_sz,
18804799d485SKashyap D Desai 				mrsas_addr_cb, &sc->raidmap_phys_addr[i],
1881665484d8SDoug Ambrisko 				BUS_DMA_NOWAIT)){
1882665484d8SDoug Ambrisko             device_printf(sc->mrsas_dev, "Cannot load raidmap memory.\n");
1883665484d8SDoug Ambrisko             return (ENOMEM);
1884665484d8SDoug Ambrisko         }
1885665484d8SDoug Ambrisko         if (!sc->raidmap_mem[i]) {
18864799d485SKashyap D Desai             device_printf(sc->mrsas_dev,
18874799d485SKashyap D Desai 			    "Cannot allocate memory for raid map.\n");
1888665484d8SDoug Ambrisko             return (ENOMEM);
1889665484d8SDoug Ambrisko         }
1890665484d8SDoug Ambrisko     }
1891665484d8SDoug Ambrisko 
1892665484d8SDoug Ambrisko     if (!mrsas_get_map_info(sc))
1893665484d8SDoug Ambrisko         mrsas_sync_map_info(sc);
1894665484d8SDoug Ambrisko 
1895665484d8SDoug Ambrisko     return (0);
18964799d485SKashyap D Desai 
18974799d485SKashyap D Desai ABORT:
18984799d485SKashyap D Desai     return (1);
1899665484d8SDoug Ambrisko }
1900665484d8SDoug Ambrisko 
1901665484d8SDoug Ambrisko /**
1902665484d8SDoug Ambrisko  * mrsas_init_fw:      Initialize Firmware
1903665484d8SDoug Ambrisko  * input:              Adapter soft state
1904665484d8SDoug Ambrisko  *
1905665484d8SDoug Ambrisko  * Calls transition_to_ready() to make sure Firmware is in operational
1906665484d8SDoug Ambrisko  * state and calls mrsas_init_adapter() to send IOC_INIT command to
1907665484d8SDoug Ambrisko  * Firmware.  It issues internal commands to get the controller info
1908665484d8SDoug Ambrisko  * after the IOC_INIT command response is received by Firmware.
1909665484d8SDoug Ambrisko  * Note:  code relating to get_pdlist, get_ld_list and max_sectors
1910665484d8SDoug Ambrisko  * are currently not being used, it is left here as placeholder.
1911665484d8SDoug Ambrisko  */
1912665484d8SDoug Ambrisko static int mrsas_init_fw(struct mrsas_softc *sc)
1913665484d8SDoug Ambrisko {
1914*d18d1b47SKashyap D Desai 
1915*d18d1b47SKashyap D Desai     int ret, loop, ocr = 0;
1916665484d8SDoug Ambrisko     u_int32_t max_sectors_1;
1917665484d8SDoug Ambrisko     u_int32_t max_sectors_2;
1918665484d8SDoug Ambrisko     u_int32_t tmp_sectors;
1919665484d8SDoug Ambrisko     struct mrsas_ctrl_info *ctrl_info;
1920*d18d1b47SKashyap D Desai     u_int32_t scratch_pad_2;
1921*d18d1b47SKashyap D Desai     int msix_enable = 0;
1922*d18d1b47SKashyap D Desai     int fw_msix_count = 0;
1923665484d8SDoug Ambrisko 
1924665484d8SDoug Ambrisko     /* Make sure Firmware is ready */
1925665484d8SDoug Ambrisko     ret = mrsas_transition_to_ready(sc, ocr);
1926665484d8SDoug Ambrisko     if (ret != SUCCESS) {
1927665484d8SDoug Ambrisko         return(ret);
1928665484d8SDoug Ambrisko 	}
1929665484d8SDoug Ambrisko 
1930*d18d1b47SKashyap D Desai 
1931*d18d1b47SKashyap D Desai     /* MSI-x index 0- reply post host index register */
1932*d18d1b47SKashyap D Desai 	sc->msix_reg_offset[0] = MPI2_REPLY_POST_HOST_INDEX_OFFSET;
1933*d18d1b47SKashyap D Desai 	/* Check if MSI-X is supported while in ready state */
1934*d18d1b47SKashyap D Desai 	msix_enable = (mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad)) & 0x4000000) >> 0x1a;
1935*d18d1b47SKashyap D Desai 
1936*d18d1b47SKashyap D Desai 	if (msix_enable) {
1937*d18d1b47SKashyap D Desai 		scratch_pad_2 = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
1938*d18d1b47SKashyap D Desai 			outbound_scratch_pad_2));
1939*d18d1b47SKashyap D Desai 
1940*d18d1b47SKashyap D Desai 		/* Check max MSI-X vectors */
1941*d18d1b47SKashyap D Desai 		if (sc->device_id == MRSAS_TBOLT) {
1942*d18d1b47SKashyap D Desai 			sc->msix_vectors = (scratch_pad_2
1943*d18d1b47SKashyap D Desai 				& MR_MAX_REPLY_QUEUES_OFFSET) + 1;
1944*d18d1b47SKashyap D Desai 			fw_msix_count = sc->msix_vectors;
1945*d18d1b47SKashyap D Desai 		} else {
1946*d18d1b47SKashyap D Desai 			/* Invader/Fury supports 96 MSI-X vectors */
1947*d18d1b47SKashyap D Desai 			sc->msix_vectors = ((scratch_pad_2
1948*d18d1b47SKashyap D Desai 				& MR_MAX_REPLY_QUEUES_EXT_OFFSET)
1949*d18d1b47SKashyap D Desai 				>> MR_MAX_REPLY_QUEUES_EXT_OFFSET_SHIFT) +				1;
1950*d18d1b47SKashyap D Desai 			fw_msix_count = sc->msix_vectors;
1951*d18d1b47SKashyap D Desai 
1952*d18d1b47SKashyap D Desai 			/* Save 1-15 reply post index address to local
1953*d18d1b47SKashyap D Desai 			 * memory
1954*d18d1b47SKashyap D Desai 			 * Index 0 is already saved from reg offset
1955*d18d1b47SKashyap D Desai 			 * MPI2_REPLY_POST_HOST_INDEX_OFFSET
1956*d18d1b47SKashyap D Desai 			 */
1957*d18d1b47SKashyap D Desai 			for (loop = 1; loop < MR_MAX_MSIX_REG_ARRAY;
1958*d18d1b47SKashyap D Desai 				loop++) {
1959*d18d1b47SKashyap D Desai 				sc->msix_reg_offset[loop] =
1960*d18d1b47SKashyap D Desai 				MPI2_SUP_REPLY_POST_HOST_INDEX_OFFSET +
1961*d18d1b47SKashyap D Desai 				(loop * 0x10);
1962*d18d1b47SKashyap D Desai 			}
1963*d18d1b47SKashyap D Desai 		}
1964*d18d1b47SKashyap D Desai 
1965*d18d1b47SKashyap D Desai 		/* Don't bother allocating more MSI-X vectors than cpus */
1966*d18d1b47SKashyap D Desai 		sc->msix_vectors = min(sc->msix_vectors,
1967*d18d1b47SKashyap D Desai 					 mp_ncpus);
1968*d18d1b47SKashyap D Desai 
1969*d18d1b47SKashyap D Desai 		/* Allocate MSI-x vectors */
1970*d18d1b47SKashyap D Desai 		if (mrsas_allocate_msix(sc) == SUCCESS)
1971*d18d1b47SKashyap D Desai 			sc->msix_enable = 1;
1972*d18d1b47SKashyap D Desai 		else
1973*d18d1b47SKashyap D Desai 			sc->msix_enable = 0;
1974*d18d1b47SKashyap D Desai 
1975*d18d1b47SKashyap D Desai 		device_printf(sc->mrsas_dev, "FW supports <%d> MSIX vector,"
1976*d18d1b47SKashyap D Desai 						"Online CPU %d Current MSIX <%d>\n",
1977*d18d1b47SKashyap D Desai 						fw_msix_count, mp_ncpus, sc->msix_vectors);
1978*d18d1b47SKashyap D Desai 	}
1979*d18d1b47SKashyap D Desai 
1980665484d8SDoug Ambrisko 	/* Get operational params, sge flags, send init cmd to ctlr */
1981665484d8SDoug Ambrisko     if (mrsas_init_adapter(sc) != SUCCESS){
1982665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Adapter initialize Fail.\n");
1983665484d8SDoug Ambrisko         return(1);
1984665484d8SDoug Ambrisko     }
1985665484d8SDoug Ambrisko 
1986665484d8SDoug Ambrisko     /* Allocate internal commands for pass-thru */
1987665484d8SDoug Ambrisko     if (mrsas_alloc_mfi_cmds(sc) != SUCCESS){
1988665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Allocate MFI cmd failed.\n");
1989665484d8SDoug Ambrisko         return(1);
1990665484d8SDoug Ambrisko     }
1991665484d8SDoug Ambrisko 
19924799d485SKashyap D Desai     /*
19934799d485SKashyap D Desai      * Get the controller info from FW, so that
19944799d485SKashyap D Desai      * the MAX VD support availability can be decided.
19954799d485SKashyap D Desai      */
19964799d485SKashyap D Desai     ctrl_info = malloc(sizeof(struct mrsas_ctrl_info), M_MRSAS, M_NOWAIT);
19974799d485SKashyap D Desai     if (!ctrl_info)
19984799d485SKashyap D Desai         device_printf(sc->mrsas_dev, "Malloc for ctrl_info failed.\n");
19994799d485SKashyap D Desai 
20004799d485SKashyap D Desai     if (mrsas_get_ctrl_info(sc, ctrl_info)) {
20014799d485SKashyap D Desai         device_printf(sc->mrsas_dev, "Unable to get FW ctrl_info.\n");
20024799d485SKashyap D Desai     }
20034799d485SKashyap D Desai 
20044799d485SKashyap D Desai     sc->max256vdSupport =
20054799d485SKashyap D Desai 	    (u_int8_t) ctrl_info->adapterOperations3.supportMaxExtLDs;
20064799d485SKashyap D Desai 
20074799d485SKashyap D Desai     if (ctrl_info->max_lds > 64){
20084799d485SKashyap D Desai 	sc->max256vdSupport = 1;
20094799d485SKashyap D Desai 	}
20104799d485SKashyap D Desai 
2011665484d8SDoug Ambrisko     if (mrsas_setup_raidmap(sc) != SUCCESS) {
2012665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Set up RAID map failed.\n");
2013665484d8SDoug Ambrisko         return(1);
2014665484d8SDoug Ambrisko 	}
2015665484d8SDoug Ambrisko 
2016665484d8SDoug Ambrisko     /* For pass-thru, get PD/LD list and controller info */
20174799d485SKashyap D Desai     memset(sc->pd_list, 0,
20184799d485SKashyap D Desai 		    MRSAS_MAX_PD * sizeof(struct mrsas_pd_list));
2019665484d8SDoug Ambrisko     mrsas_get_pd_list(sc);
2020665484d8SDoug Ambrisko 
20214799d485SKashyap D Desai     memset(sc->ld_ids, 0xff, MRSAS_MAX_LD_IDS);
2022665484d8SDoug Ambrisko     mrsas_get_ld_list(sc);
2023665484d8SDoug Ambrisko 
2024665484d8SDoug Ambrisko     /*
2025665484d8SDoug Ambrisko      * Compute the max allowed sectors per IO: The controller info has two
2026665484d8SDoug Ambrisko      * limits on max sectors. Driver should use the minimum of these two.
2027665484d8SDoug Ambrisko      *
2028665484d8SDoug Ambrisko      * 1 << stripe_sz_ops.min = max sectors per strip
2029665484d8SDoug Ambrisko      *
2030665484d8SDoug Ambrisko      * Note that older firmwares ( < FW ver 30) didn't report information
2031665484d8SDoug Ambrisko      * to calculate max_sectors_1. So the number ended up as zero always.
2032665484d8SDoug Ambrisko      */
2033665484d8SDoug Ambrisko     tmp_sectors = 0;
2034665484d8SDoug Ambrisko     max_sectors_1 = (1 << ctrl_info->stripe_sz_ops.min) *
2035665484d8SDoug Ambrisko 	    ctrl_info->max_strips_per_io;
2036665484d8SDoug Ambrisko     max_sectors_2 = ctrl_info->max_request_size;
2037665484d8SDoug Ambrisko     tmp_sectors = min(max_sectors_1 , max_sectors_2);
20384799d485SKashyap D Desai     sc->max_sectors_per_req = sc->max_num_sge * MRSAS_PAGE_SIZE / 512;
20394799d485SKashyap D Desai 
20404799d485SKashyap D Desai     if (tmp_sectors && (sc->max_sectors_per_req > tmp_sectors))
20414799d485SKashyap D Desai         sc->max_sectors_per_req = tmp_sectors;
20424799d485SKashyap D Desai 
2043665484d8SDoug Ambrisko     sc->disableOnlineCtrlReset =
2044665484d8SDoug Ambrisko             ctrl_info->properties.OnOffProperties.disableOnlineCtrlReset;
2045665484d8SDoug Ambrisko     sc->UnevenSpanSupport =
2046665484d8SDoug Ambrisko             ctrl_info->adapterOperations2.supportUnevenSpans;
2047665484d8SDoug Ambrisko     if(sc->UnevenSpanSupport) {
20484799d485SKashyap D Desai         printf("FW supports: UnevenSpanSupport=%x\n\n",
2049665484d8SDoug Ambrisko 			sc->UnevenSpanSupport);
20504799d485SKashyap D Desai 
2051665484d8SDoug Ambrisko        if (MR_ValidateMapInfo(sc))
2052665484d8SDoug Ambrisko 	    sc->fast_path_io = 1;
2053665484d8SDoug Ambrisko         else
2054665484d8SDoug Ambrisko             sc->fast_path_io = 0;
2055665484d8SDoug Ambrisko     }
2056665484d8SDoug Ambrisko 
2057665484d8SDoug Ambrisko     if (ctrl_info)
2058665484d8SDoug Ambrisko         free(ctrl_info, M_MRSAS);
2059665484d8SDoug Ambrisko 
2060665484d8SDoug Ambrisko     return(0);
2061665484d8SDoug Ambrisko }
2062665484d8SDoug Ambrisko 
2063665484d8SDoug Ambrisko /**
2064665484d8SDoug Ambrisko  * mrsas_init_adapter:     Initializes the adapter/controller
2065665484d8SDoug Ambrisko  * input:                  Adapter soft state
2066665484d8SDoug Ambrisko  *
2067665484d8SDoug Ambrisko  * Prepares for the issuing of the IOC Init cmd to FW for initializing the
2068665484d8SDoug Ambrisko  * ROC/controller.  The FW register is read to determined the number of
2069665484d8SDoug Ambrisko  * commands that is supported.  All memory allocations for IO is based on
2070665484d8SDoug Ambrisko  * max_cmd.  Appropriate calculations are performed in this function.
2071665484d8SDoug Ambrisko  */
2072665484d8SDoug Ambrisko int mrsas_init_adapter(struct mrsas_softc *sc)
2073665484d8SDoug Ambrisko {
2074665484d8SDoug Ambrisko     uint32_t status;
2075665484d8SDoug Ambrisko     u_int32_t max_cmd;
2076665484d8SDoug Ambrisko     int ret;
2077*d18d1b47SKashyap D Desai     int i = 0;
2078665484d8SDoug Ambrisko 
2079665484d8SDoug Ambrisko     /* Read FW status register */
2080665484d8SDoug Ambrisko     status = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad));
2081665484d8SDoug Ambrisko 
2082665484d8SDoug Ambrisko     /* Get operational params from status register */
2083665484d8SDoug Ambrisko     sc->max_fw_cmds = status & MRSAS_FWSTATE_MAXCMD_MASK;
2084665484d8SDoug Ambrisko 
2085665484d8SDoug Ambrisko     /* Decrement the max supported by 1, to correlate with FW */
2086665484d8SDoug Ambrisko     sc->max_fw_cmds = sc->max_fw_cmds-1;
2087665484d8SDoug Ambrisko     max_cmd = sc->max_fw_cmds;
2088665484d8SDoug Ambrisko 
2089665484d8SDoug Ambrisko     /* Determine allocation size of command frames */
2090*d18d1b47SKashyap D Desai     sc->reply_q_depth = ((max_cmd +1 +15)/16*16);
2091665484d8SDoug Ambrisko     sc->request_alloc_sz = sizeof(MRSAS_REQUEST_DESCRIPTOR_UNION) * max_cmd;
2092665484d8SDoug Ambrisko     sc->reply_alloc_sz = sizeof(MPI2_REPLY_DESCRIPTORS_UNION) * (sc->reply_q_depth);
2093665484d8SDoug Ambrisko     sc->io_frames_alloc_sz = MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE + (MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE * (max_cmd + 1));
2094665484d8SDoug Ambrisko     sc->chain_frames_alloc_sz = 1024 * max_cmd;
2095665484d8SDoug Ambrisko     sc->max_sge_in_main_msg = (MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE -
2096665484d8SDoug Ambrisko         offsetof(MRSAS_RAID_SCSI_IO_REQUEST, SGL))/16;
2097665484d8SDoug Ambrisko 
2098665484d8SDoug Ambrisko     sc->max_sge_in_chain = MRSAS_MAX_SZ_CHAIN_FRAME / sizeof(MPI2_SGE_IO_UNION);
2099665484d8SDoug Ambrisko     sc->max_num_sge = sc->max_sge_in_main_msg + sc->max_sge_in_chain - 2;
2100665484d8SDoug Ambrisko 
2101665484d8SDoug Ambrisko     /* Used for pass thru MFI frame (DCMD) */
2102665484d8SDoug Ambrisko     sc->chain_offset_mfi_pthru = offsetof(MRSAS_RAID_SCSI_IO_REQUEST, SGL)/16;
2103665484d8SDoug Ambrisko 
2104665484d8SDoug Ambrisko     sc->chain_offset_io_request = (MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE -
2105665484d8SDoug Ambrisko         sizeof(MPI2_SGE_IO_UNION))/16;
2106665484d8SDoug Ambrisko 
2107*d18d1b47SKashyap D Desai     int count = sc->msix_vectors > 0 ? sc->msix_vectors : 1;
2108*d18d1b47SKashyap D Desai     for (i = 0 ; i < count; i++)
2109*d18d1b47SKashyap D Desai 		sc->last_reply_idx[i] = 0;
2110665484d8SDoug Ambrisko 
2111665484d8SDoug Ambrisko     ret = mrsas_alloc_mem(sc);
2112665484d8SDoug Ambrisko     if (ret != SUCCESS)
2113665484d8SDoug Ambrisko         return(ret);
2114665484d8SDoug Ambrisko 
2115665484d8SDoug Ambrisko     ret = mrsas_alloc_mpt_cmds(sc);
2116665484d8SDoug Ambrisko     if (ret != SUCCESS)
2117665484d8SDoug Ambrisko         return(ret);
2118665484d8SDoug Ambrisko 
2119665484d8SDoug Ambrisko     ret = mrsas_ioc_init(sc);
2120665484d8SDoug Ambrisko     if (ret != SUCCESS)
2121665484d8SDoug Ambrisko         return(ret);
2122665484d8SDoug Ambrisko 
2123665484d8SDoug Ambrisko 
2124*d18d1b47SKashyap D Desai 
2125665484d8SDoug Ambrisko     return(0);
2126665484d8SDoug Ambrisko }
2127665484d8SDoug Ambrisko 
2128665484d8SDoug Ambrisko /**
2129665484d8SDoug Ambrisko  * mrsas_alloc_ioc_cmd:   Allocates memory for IOC Init command
2130665484d8SDoug Ambrisko  * input:                 Adapter soft state
2131665484d8SDoug Ambrisko  *
2132665484d8SDoug Ambrisko  * Allocates for the IOC Init cmd to FW to initialize the ROC/controller.
2133665484d8SDoug Ambrisko  */
2134665484d8SDoug Ambrisko int mrsas_alloc_ioc_cmd(struct mrsas_softc *sc)
2135665484d8SDoug Ambrisko {
2136665484d8SDoug Ambrisko     int ioc_init_size;
2137665484d8SDoug Ambrisko 
2138665484d8SDoug Ambrisko     /* Allocate IOC INIT command */
2139665484d8SDoug Ambrisko     ioc_init_size = 1024 + sizeof(MPI2_IOC_INIT_REQUEST);
2140665484d8SDoug Ambrisko     if (bus_dma_tag_create( sc->mrsas_parent_tag,   // parent
2141665484d8SDoug Ambrisko                             1, 0,                   // algnmnt, boundary
2142665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR_32BIT,// lowaddr
2143665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR,      // highaddr
2144665484d8SDoug Ambrisko                             NULL, NULL,             // filter, filterarg
2145665484d8SDoug Ambrisko                             ioc_init_size,          // maxsize
2146665484d8SDoug Ambrisko                             1,                      // msegments
2147665484d8SDoug Ambrisko                             ioc_init_size,          // maxsegsize
2148665484d8SDoug Ambrisko                             BUS_DMA_ALLOCNOW,       // flags
2149665484d8SDoug Ambrisko                             NULL, NULL,             // lockfunc, lockarg
2150665484d8SDoug Ambrisko                             &sc->ioc_init_tag)) {
2151665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot allocate ioc init tag\n");
2152665484d8SDoug Ambrisko         return (ENOMEM);
2153665484d8SDoug Ambrisko     }
2154665484d8SDoug Ambrisko     if (bus_dmamem_alloc(sc->ioc_init_tag, (void **)&sc->ioc_init_mem,
2155665484d8SDoug Ambrisko             BUS_DMA_NOWAIT, &sc->ioc_init_dmamap)) {
2156665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot allocate ioc init cmd mem\n");
2157665484d8SDoug Ambrisko         return (ENOMEM);
2158665484d8SDoug Ambrisko     }
2159665484d8SDoug Ambrisko     bzero(sc->ioc_init_mem, ioc_init_size);
2160665484d8SDoug Ambrisko     if (bus_dmamap_load(sc->ioc_init_tag, sc->ioc_init_dmamap,
2161665484d8SDoug Ambrisko             sc->ioc_init_mem, ioc_init_size, mrsas_addr_cb,
2162665484d8SDoug Ambrisko             &sc->ioc_init_phys_mem, BUS_DMA_NOWAIT)) {
2163665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot load ioc init cmd mem\n");
2164665484d8SDoug Ambrisko         return (ENOMEM);
2165665484d8SDoug Ambrisko     }
2166665484d8SDoug Ambrisko 
2167665484d8SDoug Ambrisko     return (0);
2168665484d8SDoug Ambrisko }
2169665484d8SDoug Ambrisko 
2170665484d8SDoug Ambrisko /**
2171665484d8SDoug Ambrisko  * mrsas_free_ioc_cmd:   Allocates memory for IOC Init command
2172665484d8SDoug Ambrisko  * input:                Adapter soft state
2173665484d8SDoug Ambrisko  *
2174665484d8SDoug Ambrisko  * Deallocates memory of the IOC Init cmd.
2175665484d8SDoug Ambrisko  */
2176665484d8SDoug Ambrisko void mrsas_free_ioc_cmd(struct mrsas_softc *sc)
2177665484d8SDoug Ambrisko {
2178665484d8SDoug Ambrisko     if (sc->ioc_init_phys_mem)
2179665484d8SDoug Ambrisko         bus_dmamap_unload(sc->ioc_init_tag, sc->ioc_init_dmamap);
2180665484d8SDoug Ambrisko     if (sc->ioc_init_mem != NULL)
2181665484d8SDoug Ambrisko         bus_dmamem_free(sc->ioc_init_tag, sc->ioc_init_mem, sc->ioc_init_dmamap);
2182665484d8SDoug Ambrisko     if (sc->ioc_init_tag != NULL)
2183665484d8SDoug Ambrisko         bus_dma_tag_destroy(sc->ioc_init_tag);
2184665484d8SDoug Ambrisko }
2185665484d8SDoug Ambrisko 
2186665484d8SDoug Ambrisko /**
2187665484d8SDoug Ambrisko  * mrsas_ioc_init:     Sends IOC Init command to FW
2188665484d8SDoug Ambrisko  * input:              Adapter soft state
2189665484d8SDoug Ambrisko  *
2190665484d8SDoug Ambrisko  * Issues the IOC Init cmd to FW to initialize the ROC/controller.
2191665484d8SDoug Ambrisko  */
2192665484d8SDoug Ambrisko int mrsas_ioc_init(struct mrsas_softc *sc)
2193665484d8SDoug Ambrisko {
2194665484d8SDoug Ambrisko     struct mrsas_init_frame *init_frame;
2195665484d8SDoug Ambrisko     pMpi2IOCInitRequest_t   IOCInitMsg;
2196665484d8SDoug Ambrisko     MRSAS_REQUEST_DESCRIPTOR_UNION req_desc;
2197665484d8SDoug Ambrisko     u_int8_t max_wait = MRSAS_IOC_INIT_WAIT_TIME;
2198665484d8SDoug Ambrisko     bus_addr_t phys_addr;
2199665484d8SDoug Ambrisko     int i, retcode = 0;
2200665484d8SDoug Ambrisko 
2201665484d8SDoug Ambrisko     /* Allocate memory for the IOC INIT command */
2202665484d8SDoug Ambrisko     if (mrsas_alloc_ioc_cmd(sc)) {
2203665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot allocate IOC command.\n");
2204665484d8SDoug Ambrisko         return(1);
2205665484d8SDoug Ambrisko     }
2206665484d8SDoug Ambrisko 
2207665484d8SDoug Ambrisko     IOCInitMsg = (pMpi2IOCInitRequest_t)(((char *)sc->ioc_init_mem) +1024);
2208665484d8SDoug Ambrisko     IOCInitMsg->Function = MPI2_FUNCTION_IOC_INIT;
2209665484d8SDoug Ambrisko     IOCInitMsg->WhoInit = MPI2_WHOINIT_HOST_DRIVER;
2210665484d8SDoug Ambrisko     IOCInitMsg->MsgVersion = MPI2_VERSION;
2211665484d8SDoug Ambrisko     IOCInitMsg->HeaderVersion = MPI2_HEADER_VERSION;
2212665484d8SDoug Ambrisko     IOCInitMsg->SystemRequestFrameSize = MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE / 4;
2213665484d8SDoug Ambrisko     IOCInitMsg->ReplyDescriptorPostQueueDepth = sc->reply_q_depth;
2214665484d8SDoug Ambrisko     IOCInitMsg->ReplyDescriptorPostQueueAddress = sc->reply_desc_phys_addr;
2215665484d8SDoug Ambrisko     IOCInitMsg->SystemRequestFrameBaseAddress = sc->io_request_phys_addr;
2216*d18d1b47SKashyap D Desai     IOCInitMsg->HostMSIxVectors = (sc->msix_vectors > 0 ? sc->msix_vectors : 0);
2217665484d8SDoug Ambrisko 
2218665484d8SDoug Ambrisko     init_frame = (struct mrsas_init_frame *)sc->ioc_init_mem;
2219665484d8SDoug Ambrisko     init_frame->cmd = MFI_CMD_INIT;
2220665484d8SDoug Ambrisko     init_frame->cmd_status = 0xFF;
2221665484d8SDoug Ambrisko     init_frame->flags |= MFI_FRAME_DONT_POST_IN_REPLY_QUEUE;
2222665484d8SDoug Ambrisko 
2223*d18d1b47SKashyap D Desai     /* driver support Extended MSIX */
2224*d18d1b47SKashyap D Desai     if ((sc->device_id == MRSAS_INVADER) ||
2225*d18d1b47SKashyap D Desai          (sc->device_id == MRSAS_FURY)) {
2226*d18d1b47SKashyap D Desai              init_frame->driver_operations.
2227*d18d1b47SKashyap D Desai                 mfi_capabilities.support_additional_msix = 1;
2228*d18d1b47SKashyap D Desai     }
2229*d18d1b47SKashyap D Desai 
2230*d18d1b47SKashyap D Desai 
2231665484d8SDoug Ambrisko     if (sc->verbuf_mem) {
2232665484d8SDoug Ambrisko         snprintf((char *)sc->verbuf_mem, strlen(MRSAS_VERSION)+2,"%s\n",
2233665484d8SDoug Ambrisko                 MRSAS_VERSION);
2234665484d8SDoug Ambrisko         init_frame->driver_ver_lo = (bus_addr_t)sc->verbuf_phys_addr;
2235665484d8SDoug Ambrisko         init_frame->driver_ver_hi = 0;
2236665484d8SDoug Ambrisko     }
2237665484d8SDoug Ambrisko 
22384799d485SKashyap D Desai     init_frame->driver_operations.mfi_capabilities.support_max_255lds = 1;
2239665484d8SDoug Ambrisko     phys_addr = (bus_addr_t)sc->ioc_init_phys_mem + 1024;
2240665484d8SDoug Ambrisko     init_frame->queue_info_new_phys_addr_lo = phys_addr;
2241665484d8SDoug Ambrisko     init_frame->data_xfer_len = sizeof(Mpi2IOCInitRequest_t);
2242665484d8SDoug Ambrisko 
2243665484d8SDoug Ambrisko     req_desc.addr.Words = (bus_addr_t)sc->ioc_init_phys_mem;
2244665484d8SDoug Ambrisko     req_desc.MFAIo.RequestFlags =
2245665484d8SDoug Ambrisko         (MRSAS_REQ_DESCRIPT_FLAGS_MFA << MRSAS_REQ_DESCRIPT_FLAGS_TYPE_SHIFT);
2246665484d8SDoug Ambrisko 
2247665484d8SDoug Ambrisko     mrsas_disable_intr(sc);
2248665484d8SDoug Ambrisko     mrsas_dprint(sc, MRSAS_OCR, "Issuing IOC INIT command to FW.\n");
2249665484d8SDoug Ambrisko     //device_printf(sc->mrsas_dev, "Issuing IOC INIT command to FW.\n");del?
2250665484d8SDoug Ambrisko     mrsas_fire_cmd(sc, req_desc.addr.u.low, req_desc.addr.u.high);
2251665484d8SDoug Ambrisko 
2252665484d8SDoug Ambrisko     /*
2253665484d8SDoug Ambrisko      * Poll response timer to wait for Firmware response.  While this
2254665484d8SDoug Ambrisko      * timer with the DELAY call could block CPU, the time interval for
2255665484d8SDoug Ambrisko      * this is only 1 millisecond.
2256665484d8SDoug Ambrisko      */
2257665484d8SDoug Ambrisko     if (init_frame->cmd_status == 0xFF) {
2258665484d8SDoug Ambrisko         for (i=0; i < (max_wait * 1000); i++){
2259665484d8SDoug Ambrisko             if (init_frame->cmd_status == 0xFF)
2260665484d8SDoug Ambrisko                 DELAY(1000);
2261665484d8SDoug Ambrisko             else
2262665484d8SDoug Ambrisko                 break;
2263665484d8SDoug Ambrisko         }
2264665484d8SDoug Ambrisko     }
2265665484d8SDoug Ambrisko 
2266665484d8SDoug Ambrisko     if (init_frame->cmd_status == 0)
2267665484d8SDoug Ambrisko          mrsas_dprint(sc, MRSAS_OCR,
2268665484d8SDoug Ambrisko                "IOC INIT response received from FW.\n");
2269665484d8SDoug Ambrisko          //device_printf(sc->mrsas_dev, "IOC INIT response received from FW.\n");del?
2270665484d8SDoug Ambrisko     else
2271665484d8SDoug Ambrisko     {
2272665484d8SDoug Ambrisko         if (init_frame->cmd_status == 0xFF)
2273665484d8SDoug Ambrisko             device_printf(sc->mrsas_dev, "IOC Init timed out after %d seconds.\n", max_wait);
2274665484d8SDoug Ambrisko         else
2275665484d8SDoug Ambrisko             device_printf(sc->mrsas_dev, "IOC Init failed, status = 0x%x\n", init_frame->cmd_status);
2276665484d8SDoug Ambrisko         retcode = 1;
2277665484d8SDoug Ambrisko     }
2278665484d8SDoug Ambrisko 
2279665484d8SDoug Ambrisko     mrsas_free_ioc_cmd(sc);
2280665484d8SDoug Ambrisko     return (retcode);
2281665484d8SDoug Ambrisko }
2282665484d8SDoug Ambrisko 
2283665484d8SDoug Ambrisko /**
2284665484d8SDoug Ambrisko  * mrsas_alloc_mpt_cmds:  Allocates the command packets
2285665484d8SDoug Ambrisko  * input:                 Adapter instance soft state
2286665484d8SDoug Ambrisko  *
2287665484d8SDoug Ambrisko  * This function allocates the internal commands for IOs. Each command that is
2288665484d8SDoug Ambrisko  * issued to FW is wrapped in a local data structure called mrsas_mpt_cmd.
2289665484d8SDoug Ambrisko  * An array is allocated with mrsas_mpt_cmd context.  The free commands are
2290665484d8SDoug Ambrisko  * maintained in a linked list (cmd pool). SMID value range is from 1 to
2291665484d8SDoug Ambrisko  * max_fw_cmds.
2292665484d8SDoug Ambrisko  */
2293665484d8SDoug Ambrisko int mrsas_alloc_mpt_cmds(struct mrsas_softc *sc)
2294665484d8SDoug Ambrisko {
2295665484d8SDoug Ambrisko     int i, j;
2296*d18d1b47SKashyap D Desai     u_int32_t max_cmd, count;
2297665484d8SDoug Ambrisko     struct mrsas_mpt_cmd *cmd;
2298665484d8SDoug Ambrisko     pMpi2ReplyDescriptorsUnion_t reply_desc;
2299665484d8SDoug Ambrisko     u_int32_t offset, chain_offset, sense_offset;
2300665484d8SDoug Ambrisko     bus_addr_t io_req_base_phys, chain_frame_base_phys, sense_base_phys;
2301665484d8SDoug Ambrisko     u_int8_t *io_req_base, *chain_frame_base, *sense_base;
2302665484d8SDoug Ambrisko 
2303665484d8SDoug Ambrisko     max_cmd = sc->max_fw_cmds;
2304665484d8SDoug Ambrisko 
2305665484d8SDoug Ambrisko     sc->req_desc = malloc(sc->request_alloc_sz, M_MRSAS, M_NOWAIT);
2306665484d8SDoug Ambrisko     if (!sc->req_desc) {
2307665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Out of memory, cannot alloc req desc\n");
2308665484d8SDoug Ambrisko         return(ENOMEM);
2309665484d8SDoug Ambrisko     }
2310665484d8SDoug Ambrisko     memset(sc->req_desc, 0, sc->request_alloc_sz);
2311665484d8SDoug Ambrisko 
2312665484d8SDoug Ambrisko     /*
2313665484d8SDoug Ambrisko      * sc->mpt_cmd_list is an array of struct mrsas_mpt_cmd pointers. Allocate the
2314665484d8SDoug Ambrisko      * dynamic array first and then allocate individual commands.
2315665484d8SDoug Ambrisko      */
2316665484d8SDoug Ambrisko     sc->mpt_cmd_list = malloc(sizeof(struct mrsas_mpt_cmd*)*max_cmd, M_MRSAS, M_NOWAIT);
2317665484d8SDoug Ambrisko     if (!sc->mpt_cmd_list) {
2318665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot alloc memory for mpt_cmd_list.\n");
2319665484d8SDoug Ambrisko         return(ENOMEM);
2320665484d8SDoug Ambrisko     }
2321665484d8SDoug Ambrisko     memset(sc->mpt_cmd_list, 0, sizeof(struct mrsas_mpt_cmd *)*max_cmd);
2322665484d8SDoug Ambrisko     for (i = 0; i < max_cmd; i++) {
2323665484d8SDoug Ambrisko         sc->mpt_cmd_list[i] = malloc(sizeof(struct mrsas_mpt_cmd),
2324665484d8SDoug Ambrisko                                  M_MRSAS, M_NOWAIT);
2325665484d8SDoug Ambrisko         if (!sc->mpt_cmd_list[i]) {
2326665484d8SDoug Ambrisko             for (j = 0; j < i; j++)
2327665484d8SDoug Ambrisko                 free(sc->mpt_cmd_list[j],M_MRSAS);
2328665484d8SDoug Ambrisko             free(sc->mpt_cmd_list, M_MRSAS);
2329665484d8SDoug Ambrisko             sc->mpt_cmd_list = NULL;
2330665484d8SDoug Ambrisko             return(ENOMEM);
2331665484d8SDoug Ambrisko         }
2332665484d8SDoug Ambrisko     }
2333665484d8SDoug Ambrisko 
2334665484d8SDoug Ambrisko     io_req_base = (u_int8_t*)sc->io_request_mem + MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE;
2335665484d8SDoug Ambrisko     io_req_base_phys = (bus_addr_t)sc->io_request_phys_addr + MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE;
2336665484d8SDoug Ambrisko     chain_frame_base = (u_int8_t*)sc->chain_frame_mem;
2337665484d8SDoug Ambrisko     chain_frame_base_phys = (bus_addr_t)sc->chain_frame_phys_addr;
2338665484d8SDoug Ambrisko     sense_base = (u_int8_t*)sc->sense_mem;
2339665484d8SDoug Ambrisko     sense_base_phys = (bus_addr_t)sc->sense_phys_addr;
2340665484d8SDoug Ambrisko     for (i = 0; i < max_cmd; i++) {
2341665484d8SDoug Ambrisko         cmd = sc->mpt_cmd_list[i];
2342665484d8SDoug Ambrisko         offset = MRSAS_MPI2_RAID_DEFAULT_IO_FRAME_SIZE * i;
2343665484d8SDoug Ambrisko 	chain_offset = 1024 * i;
2344665484d8SDoug Ambrisko         sense_offset = MRSAS_SENSE_LEN * i;
2345665484d8SDoug Ambrisko         memset(cmd, 0, sizeof(struct mrsas_mpt_cmd));
2346665484d8SDoug Ambrisko         cmd->index = i + 1;
2347665484d8SDoug Ambrisko         cmd->ccb_ptr = NULL;
2348665484d8SDoug Ambrisko         callout_init(&cmd->cm_callout, 0);
2349665484d8SDoug Ambrisko         cmd->sync_cmd_idx = (u_int32_t)MRSAS_ULONG_MAX;
2350665484d8SDoug Ambrisko         cmd->sc = sc;
2351665484d8SDoug Ambrisko         cmd->io_request = (MRSAS_RAID_SCSI_IO_REQUEST *) (io_req_base + offset);
2352665484d8SDoug Ambrisko         memset(cmd->io_request, 0, sizeof(MRSAS_RAID_SCSI_IO_REQUEST));
2353665484d8SDoug Ambrisko         cmd->io_request_phys_addr = io_req_base_phys + offset;
2354665484d8SDoug Ambrisko 	cmd->chain_frame = (MPI2_SGE_IO_UNION *) (chain_frame_base + chain_offset);
2355665484d8SDoug Ambrisko 	cmd->chain_frame_phys_addr = chain_frame_base_phys + chain_offset;
2356665484d8SDoug Ambrisko         cmd->sense = sense_base + sense_offset;
2357665484d8SDoug Ambrisko         cmd->sense_phys_addr = sense_base_phys + sense_offset;
2358665484d8SDoug Ambrisko         if (bus_dmamap_create(sc->data_tag, 0, &cmd->data_dmamap)) {
2359665484d8SDoug Ambrisko             return(FAIL);
2360665484d8SDoug Ambrisko         }
2361665484d8SDoug Ambrisko         TAILQ_INSERT_TAIL(&(sc->mrsas_mpt_cmd_list_head), cmd, next);
2362665484d8SDoug Ambrisko     }
2363665484d8SDoug Ambrisko 
2364665484d8SDoug Ambrisko     /* Initialize reply descriptor array to 0xFFFFFFFF */
2365665484d8SDoug Ambrisko     reply_desc = sc->reply_desc_mem;
2366*d18d1b47SKashyap D Desai     count = sc->msix_vectors > 0 ? sc->msix_vectors : 1;
2367*d18d1b47SKashyap D Desai     for (i = 0; i < sc->reply_q_depth * count ; i++, reply_desc++) {
2368665484d8SDoug Ambrisko         reply_desc->Words = MRSAS_ULONG_MAX;
2369665484d8SDoug Ambrisko     }
2370665484d8SDoug Ambrisko     return(0);
2371665484d8SDoug Ambrisko }
2372665484d8SDoug Ambrisko 
2373665484d8SDoug Ambrisko /**
2374665484d8SDoug Ambrisko  * mrsas_fire_cmd:     Sends command to FW
2375665484d8SDoug Ambrisko  * input:              Adapter soft state
2376665484d8SDoug Ambrisko  *                     request descriptor address low
2377665484d8SDoug Ambrisko  *                     request descriptor address high
2378665484d8SDoug Ambrisko  *
2379665484d8SDoug Ambrisko  * This functions fires the command to Firmware by writing to the
2380665484d8SDoug Ambrisko  * inbound_low_queue_port and inbound_high_queue_port.
2381665484d8SDoug Ambrisko  */
2382665484d8SDoug Ambrisko void mrsas_fire_cmd(struct mrsas_softc *sc, u_int32_t req_desc_lo,
2383665484d8SDoug Ambrisko                    u_int32_t req_desc_hi)
2384665484d8SDoug Ambrisko {
2385665484d8SDoug Ambrisko     mtx_lock(&sc->pci_lock);
2386665484d8SDoug Ambrisko     mrsas_write_reg(sc, offsetof(mrsas_reg_set, inbound_low_queue_port),
2387665484d8SDoug Ambrisko                     req_desc_lo);
2388665484d8SDoug Ambrisko     mrsas_write_reg(sc, offsetof(mrsas_reg_set, inbound_high_queue_port),
2389665484d8SDoug Ambrisko                     req_desc_hi);
2390665484d8SDoug Ambrisko     mtx_unlock(&sc->pci_lock);
2391665484d8SDoug Ambrisko }
2392665484d8SDoug Ambrisko 
2393665484d8SDoug Ambrisko /**
2394665484d8SDoug Ambrisko  * mrsas_transition_to_ready:  Move FW to Ready state
2395665484d8SDoug Ambrisko  * input:                      Adapter instance soft state
2396665484d8SDoug Ambrisko  *
2397665484d8SDoug Ambrisko  * During the initialization, FW passes can potentially be in any one of
2398665484d8SDoug Ambrisko  * several possible states. If the FW in operational, waiting-for-handshake
2399665484d8SDoug Ambrisko  * states, driver must take steps to bring it to ready state. Otherwise, it
2400665484d8SDoug Ambrisko  * has to wait for the ready state.
2401665484d8SDoug Ambrisko  */
2402665484d8SDoug Ambrisko int mrsas_transition_to_ready(struct mrsas_softc *sc, int ocr)
2403665484d8SDoug Ambrisko {
2404665484d8SDoug Ambrisko     int i;
2405665484d8SDoug Ambrisko     u_int8_t max_wait;
2406665484d8SDoug Ambrisko     u_int32_t val, fw_state;
2407665484d8SDoug Ambrisko     u_int32_t cur_state;
2408665484d8SDoug Ambrisko     u_int32_t abs_state, curr_abs_state;
2409665484d8SDoug Ambrisko 
2410665484d8SDoug Ambrisko     val = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad));
2411665484d8SDoug Ambrisko     fw_state = val & MFI_STATE_MASK;
2412665484d8SDoug Ambrisko     max_wait = MRSAS_RESET_WAIT_TIME;
2413665484d8SDoug Ambrisko 
2414665484d8SDoug Ambrisko     if (fw_state != MFI_STATE_READY)
2415665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Waiting for FW to come to ready state\n");
2416665484d8SDoug Ambrisko 
2417665484d8SDoug Ambrisko     while (fw_state != MFI_STATE_READY) {
2418665484d8SDoug Ambrisko 	abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set, outbound_scratch_pad));
2419665484d8SDoug Ambrisko 	switch (fw_state) {
2420665484d8SDoug Ambrisko 	    case MFI_STATE_FAULT:
2421665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "FW is in FAULT state!!\n");
2422665484d8SDoug Ambrisko 		if (ocr) {
2423665484d8SDoug Ambrisko 		    cur_state = MFI_STATE_FAULT;
2424665484d8SDoug Ambrisko 	  	    break;
2425665484d8SDoug Ambrisko 		}
2426665484d8SDoug Ambrisko         else
2427665484d8SDoug Ambrisko 			return -ENODEV;
2428665484d8SDoug Ambrisko 	    case MFI_STATE_WAIT_HANDSHAKE:
2429665484d8SDoug Ambrisko 		/* Set the CLR bit in inbound doorbell */
2430665484d8SDoug Ambrisko                 mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell),
2431665484d8SDoug Ambrisko 		        MFI_INIT_CLEAR_HANDSHAKE|MFI_INIT_HOTPLUG);
2432665484d8SDoug Ambrisko 		cur_state = MFI_STATE_WAIT_HANDSHAKE;
2433665484d8SDoug Ambrisko 		break;
2434665484d8SDoug Ambrisko 	    case MFI_STATE_BOOT_MESSAGE_PENDING:
2435665484d8SDoug Ambrisko                 mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell),
2436665484d8SDoug Ambrisko 		        MFI_INIT_HOTPLUG);
2437665484d8SDoug Ambrisko 		cur_state = MFI_STATE_BOOT_MESSAGE_PENDING;
2438665484d8SDoug Ambrisko 		break;
2439665484d8SDoug Ambrisko 	    case MFI_STATE_OPERATIONAL:
2440665484d8SDoug Ambrisko 		/* Bring it to READY state; assuming max wait 10 secs */
2441665484d8SDoug Ambrisko 		mrsas_disable_intr(sc);
2442665484d8SDoug Ambrisko                 mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell), MFI_RESET_FLAGS);
2443665484d8SDoug Ambrisko                 for (i=0; i < max_wait * 1000; i++) {
2444665484d8SDoug Ambrisko 	            if (mrsas_read_reg(sc, offsetof(mrsas_reg_set, doorbell)) & 1)
2445665484d8SDoug Ambrisko                         DELAY(1000);
2446665484d8SDoug Ambrisko 		    else
2447665484d8SDoug Ambrisko 		        break;
2448665484d8SDoug Ambrisko                 }
2449665484d8SDoug Ambrisko 		cur_state = MFI_STATE_OPERATIONAL;
2450665484d8SDoug Ambrisko 	        break;
2451665484d8SDoug Ambrisko 	    case MFI_STATE_UNDEFINED:
2452665484d8SDoug Ambrisko 	        /* This state should not last for more than 2 seconds */
2453665484d8SDoug Ambrisko 	        cur_state = MFI_STATE_UNDEFINED;
2454665484d8SDoug Ambrisko 	        break;
2455665484d8SDoug Ambrisko 	    case MFI_STATE_BB_INIT:
2456665484d8SDoug Ambrisko 		cur_state = MFI_STATE_BB_INIT;
2457665484d8SDoug Ambrisko 		break;
2458665484d8SDoug Ambrisko 	    case MFI_STATE_FW_INIT:
2459665484d8SDoug Ambrisko 		cur_state = MFI_STATE_FW_INIT;
2460665484d8SDoug Ambrisko 		break;
2461665484d8SDoug Ambrisko 	    case MFI_STATE_FW_INIT_2:
2462665484d8SDoug Ambrisko 		cur_state = MFI_STATE_FW_INIT_2;
2463665484d8SDoug Ambrisko 		break;
2464665484d8SDoug Ambrisko 	    case MFI_STATE_DEVICE_SCAN:
2465665484d8SDoug Ambrisko 		cur_state = MFI_STATE_DEVICE_SCAN;
2466665484d8SDoug Ambrisko 		break;
2467665484d8SDoug Ambrisko 	    case MFI_STATE_FLUSH_CACHE:
2468665484d8SDoug Ambrisko 		cur_state = MFI_STATE_FLUSH_CACHE;
2469665484d8SDoug Ambrisko 		break;
2470665484d8SDoug Ambrisko 	    default:
2471665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "Unknown state 0x%x\n", fw_state);
2472665484d8SDoug Ambrisko 		return -ENODEV;
2473665484d8SDoug Ambrisko 	}
2474665484d8SDoug Ambrisko 
2475665484d8SDoug Ambrisko 	/*
2476665484d8SDoug Ambrisko 	 * The cur_state should not last for more than max_wait secs
2477665484d8SDoug Ambrisko 	 */
2478665484d8SDoug Ambrisko 	for (i = 0; i < (max_wait * 1000); i++) {
2479665484d8SDoug Ambrisko             fw_state = (mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2480665484d8SDoug Ambrisko                     outbound_scratch_pad))& MFI_STATE_MASK);
2481665484d8SDoug Ambrisko 	    curr_abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2482665484d8SDoug Ambrisko                     outbound_scratch_pad));
2483665484d8SDoug Ambrisko             if (abs_state == curr_abs_state)
2484665484d8SDoug Ambrisko                 DELAY(1000);
2485665484d8SDoug Ambrisko 	    else
2486665484d8SDoug Ambrisko 		break;
2487665484d8SDoug Ambrisko 	}
2488665484d8SDoug Ambrisko 
2489665484d8SDoug Ambrisko 	/*
2490665484d8SDoug Ambrisko 	 * Return error if fw_state hasn't changed after max_wait
2491665484d8SDoug Ambrisko 	 */
2492665484d8SDoug Ambrisko 	if (curr_abs_state == abs_state) {
2493665484d8SDoug Ambrisko             device_printf(sc->mrsas_dev, "FW state [%d] hasn't changed "
2494665484d8SDoug Ambrisko 		       "in %d secs\n", fw_state, max_wait);
2495665484d8SDoug Ambrisko 	    return -ENODEV;
2496665484d8SDoug Ambrisko 	}
2497665484d8SDoug Ambrisko     }
2498665484d8SDoug Ambrisko     mrsas_dprint(sc, MRSAS_OCR, "FW now in Ready state\n");
2499665484d8SDoug Ambrisko     //device_printf(sc->mrsas_dev, "FW now in Ready state\n");del?
2500665484d8SDoug Ambrisko     return 0;
2501665484d8SDoug Ambrisko }
2502665484d8SDoug Ambrisko 
2503665484d8SDoug Ambrisko /**
2504665484d8SDoug Ambrisko  * mrsas_get_mfi_cmd:      Get a cmd from free command pool
2505665484d8SDoug Ambrisko  * input:                  Adapter soft state
2506665484d8SDoug Ambrisko  *
2507665484d8SDoug Ambrisko  * This function removes an MFI command from the command list.
2508665484d8SDoug Ambrisko  */
2509665484d8SDoug Ambrisko struct mrsas_mfi_cmd* mrsas_get_mfi_cmd(struct mrsas_softc *sc)
2510665484d8SDoug Ambrisko {
2511665484d8SDoug Ambrisko     struct mrsas_mfi_cmd *cmd = NULL;
2512665484d8SDoug Ambrisko 
2513665484d8SDoug Ambrisko     mtx_lock(&sc->mfi_cmd_pool_lock);
2514665484d8SDoug Ambrisko     if (!TAILQ_EMPTY(&sc->mrsas_mfi_cmd_list_head)){
2515665484d8SDoug Ambrisko         cmd = TAILQ_FIRST(&sc->mrsas_mfi_cmd_list_head);
2516665484d8SDoug Ambrisko         TAILQ_REMOVE(&sc->mrsas_mfi_cmd_list_head, cmd, next);
2517665484d8SDoug Ambrisko     }
2518665484d8SDoug Ambrisko     mtx_unlock(&sc->mfi_cmd_pool_lock);
2519665484d8SDoug Ambrisko 
2520665484d8SDoug Ambrisko     return cmd;
2521665484d8SDoug Ambrisko }
2522665484d8SDoug Ambrisko 
2523665484d8SDoug Ambrisko /**
2524665484d8SDoug Ambrisko  * mrsas_ocr_thread             Thread to handle OCR/Kill Adapter.
2525665484d8SDoug Ambrisko  * input:               Adapter Context.
2526665484d8SDoug Ambrisko  *
2527665484d8SDoug Ambrisko  * This function will check FW status register and flag
2528665484d8SDoug Ambrisko  * do_timeout_reset flag. It will do OCR/Kill adapter if
2529665484d8SDoug Ambrisko  * FW is in fault state or IO timed out has trigger reset.
2530665484d8SDoug Ambrisko  */
2531665484d8SDoug Ambrisko static void
2532665484d8SDoug Ambrisko mrsas_ocr_thread(void *arg)
2533665484d8SDoug Ambrisko {
2534665484d8SDoug Ambrisko     struct mrsas_softc *sc;
2535665484d8SDoug Ambrisko     u_int32_t  fw_status, fw_state;
2536665484d8SDoug Ambrisko 
2537665484d8SDoug Ambrisko     sc = (struct mrsas_softc *)arg;
2538665484d8SDoug Ambrisko 
2539665484d8SDoug Ambrisko     mrsas_dprint(sc, MRSAS_TRACE, "%s\n", __func__);
2540665484d8SDoug Ambrisko 
2541665484d8SDoug Ambrisko     sc->ocr_thread_active = 1;
2542665484d8SDoug Ambrisko     mtx_lock(&sc->sim_lock);
2543665484d8SDoug Ambrisko     for (;;) {
2544665484d8SDoug Ambrisko         /* Sleep for 1 second and check the queue status*/
2545665484d8SDoug Ambrisko         msleep(&sc->ocr_chan, &sc->sim_lock, PRIBIO,
2546665484d8SDoug Ambrisko                "mrsas_ocr", sc->mrsas_fw_fault_check_delay * hz);
2547665484d8SDoug Ambrisko         if (sc->remove_in_progress) {
2548665484d8SDoug Ambrisko             mrsas_dprint(sc, MRSAS_OCR,
2549665484d8SDoug Ambrisko 				"Exit due to shutdown from %s\n", __func__);
2550665484d8SDoug Ambrisko             break;
2551665484d8SDoug Ambrisko         }
2552665484d8SDoug Ambrisko         fw_status = mrsas_read_reg(sc,
2553665484d8SDoug Ambrisko 				offsetof(mrsas_reg_set, outbound_scratch_pad));
2554665484d8SDoug Ambrisko         fw_state = fw_status & MFI_STATE_MASK;
2555665484d8SDoug Ambrisko         if (fw_state == MFI_STATE_FAULT || sc->do_timedout_reset) {
2556665484d8SDoug Ambrisko             device_printf(sc->mrsas_dev, "OCR started due to %s!\n",
2557665484d8SDoug Ambrisko                  sc->do_timedout_reset?"IO Timeout":
2558665484d8SDoug Ambrisko                  "FW fault detected");
2559665484d8SDoug Ambrisko             mtx_lock_spin(&sc->ioctl_lock);
2560665484d8SDoug Ambrisko             sc->reset_in_progress = 1;
2561665484d8SDoug Ambrisko             sc->reset_count++;
2562665484d8SDoug Ambrisko             mtx_unlock_spin(&sc->ioctl_lock);
2563665484d8SDoug Ambrisko             mrsas_xpt_freeze(sc);
2564665484d8SDoug Ambrisko             mrsas_reset_ctrl(sc);
2565665484d8SDoug Ambrisko             mrsas_xpt_release(sc);
2566665484d8SDoug Ambrisko             sc->reset_in_progress = 0;
2567665484d8SDoug Ambrisko             sc->do_timedout_reset = 0;
2568665484d8SDoug Ambrisko         }
2569665484d8SDoug Ambrisko     }
2570665484d8SDoug Ambrisko     mtx_unlock(&sc->sim_lock);
2571665484d8SDoug Ambrisko     sc->ocr_thread_active = 0;
2572665484d8SDoug Ambrisko     mrsas_kproc_exit(0);
2573665484d8SDoug Ambrisko }
2574665484d8SDoug Ambrisko 
2575665484d8SDoug Ambrisko /**
2576665484d8SDoug Ambrisko  * mrsas_reset_reply_desc       Reset Reply descriptor as part of OCR.
2577665484d8SDoug Ambrisko  * input:                       Adapter Context.
2578665484d8SDoug Ambrisko  *
2579665484d8SDoug Ambrisko  * This function will clear reply descriptor so that post OCR
2580665484d8SDoug Ambrisko  * driver and FW will lost old history.
2581665484d8SDoug Ambrisko  */
2582665484d8SDoug Ambrisko void  mrsas_reset_reply_desc(struct mrsas_softc *sc)
2583665484d8SDoug Ambrisko {
2584*d18d1b47SKashyap D Desai     int i, count;
2585665484d8SDoug Ambrisko     pMpi2ReplyDescriptorsUnion_t reply_desc;
2586665484d8SDoug Ambrisko 
2587*d18d1b47SKashyap D Desai     count = sc->msix_vectors > 0 ? sc->msix_vectors : 1;
2588*d18d1b47SKashyap D Desai     for (i = 0 ; i < count ; i++)
2589*d18d1b47SKashyap D Desai 		sc->last_reply_idx[i] = 0;
2590*d18d1b47SKashyap D Desai 
2591665484d8SDoug Ambrisko 	reply_desc = sc->reply_desc_mem;
2592665484d8SDoug Ambrisko     for (i = 0; i < sc->reply_q_depth; i++, reply_desc++) {
2593665484d8SDoug Ambrisko         reply_desc->Words = MRSAS_ULONG_MAX;
2594665484d8SDoug Ambrisko     }
2595665484d8SDoug Ambrisko }
2596665484d8SDoug Ambrisko 
2597665484d8SDoug Ambrisko /**
2598665484d8SDoug Ambrisko  * mrsas_reset_ctrl     Core function to OCR/Kill adapter.
2599665484d8SDoug Ambrisko  * input:               Adapter Context.
2600665484d8SDoug Ambrisko  *
2601665484d8SDoug Ambrisko  * This function will run from thread context so that it can sleep.
2602665484d8SDoug Ambrisko  * 1. Do not handle OCR if FW is in HW critical error.
2603665484d8SDoug Ambrisko  * 2. Wait for outstanding command to complete for 180 seconds.
2604665484d8SDoug Ambrisko  * 3. If #2 does not find any outstanding command Controller is in working
2605665484d8SDoug Ambrisko  * state, so skip OCR.
2606665484d8SDoug Ambrisko  * Otherwise, do OCR/kill Adapter based on flag disableOnlineCtrlReset.
2607665484d8SDoug Ambrisko  * 4. Start of the OCR, return all SCSI command back to CAM layer which has
2608665484d8SDoug Ambrisko  * ccb_ptr.
2609665484d8SDoug Ambrisko  * 5. Post OCR, Re-fire Managment command and move Controller to Operation
2610665484d8SDoug Ambrisko  * state.
2611665484d8SDoug Ambrisko  */
2612665484d8SDoug Ambrisko int mrsas_reset_ctrl(struct mrsas_softc *sc)
2613665484d8SDoug Ambrisko {
2614665484d8SDoug Ambrisko     int retval = SUCCESS, i, j, retry = 0;
2615665484d8SDoug Ambrisko     u_int32_t       host_diag, abs_state, status_reg, reset_adapter;
2616665484d8SDoug Ambrisko     union ccb   *ccb;
2617665484d8SDoug Ambrisko     struct mrsas_mfi_cmd *mfi_cmd;
2618665484d8SDoug Ambrisko     struct mrsas_mpt_cmd *mpt_cmd;
2619665484d8SDoug Ambrisko     MRSAS_REQUEST_DESCRIPTOR_UNION *req_desc;
2620665484d8SDoug Ambrisko 
2621665484d8SDoug Ambrisko     if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR) {
2622665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev,
2623665484d8SDoug Ambrisko                         "mrsas: Hardware critical error, returning FAIL.\n");
2624665484d8SDoug Ambrisko         return FAIL;
2625665484d8SDoug Ambrisko     }
2626665484d8SDoug Ambrisko 
2627665484d8SDoug Ambrisko     set_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags);
2628665484d8SDoug Ambrisko     sc->adprecovery = MRSAS_ADPRESET_SM_INFAULT;
2629665484d8SDoug Ambrisko     mrsas_disable_intr(sc);
2630665484d8SDoug Ambrisko     DELAY(1000 * 1000);
2631665484d8SDoug Ambrisko 
2632665484d8SDoug Ambrisko     /* First try waiting for commands to complete */
2633665484d8SDoug Ambrisko     if (mrsas_wait_for_outstanding(sc)) {
2634665484d8SDoug Ambrisko         mrsas_dprint(sc, MRSAS_OCR,
2635665484d8SDoug Ambrisko                      "resetting adapter from %s.\n",
2636665484d8SDoug Ambrisko                       __func__);
2637665484d8SDoug Ambrisko         /* Now return commands back to the CAM layer */
2638665484d8SDoug Ambrisko         for (i = 0 ; i < sc->max_fw_cmds; i++) {
2639665484d8SDoug Ambrisko             mpt_cmd = sc->mpt_cmd_list[i];
2640665484d8SDoug Ambrisko             if (mpt_cmd->ccb_ptr) {
2641665484d8SDoug Ambrisko                 ccb = (union ccb *)(mpt_cmd->ccb_ptr);
2642665484d8SDoug Ambrisko                 ccb->ccb_h.status = CAM_SCSI_BUS_RESET;
2643665484d8SDoug Ambrisko                 mrsas_cmd_done(sc, mpt_cmd);
2644665484d8SDoug Ambrisko                 atomic_dec(&sc->fw_outstanding);
2645665484d8SDoug Ambrisko             }
2646665484d8SDoug Ambrisko         }
2647665484d8SDoug Ambrisko 
2648665484d8SDoug Ambrisko         status_reg = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2649665484d8SDoug Ambrisko                                                            outbound_scratch_pad));
2650665484d8SDoug Ambrisko         abs_state = status_reg & MFI_STATE_MASK;
2651665484d8SDoug Ambrisko         reset_adapter = status_reg & MFI_RESET_ADAPTER;
2652665484d8SDoug Ambrisko         if (sc->disableOnlineCtrlReset ||
2653665484d8SDoug Ambrisko                         (abs_state == MFI_STATE_FAULT && !reset_adapter)) {
2654665484d8SDoug Ambrisko             /* Reset not supported, kill adapter */
2655665484d8SDoug Ambrisko             mrsas_dprint(sc, MRSAS_OCR,"Reset not supported, killing adapter.\n");
2656665484d8SDoug Ambrisko             mrsas_kill_hba(sc);
2657665484d8SDoug Ambrisko             sc->adprecovery = MRSAS_HW_CRITICAL_ERROR;
2658665484d8SDoug Ambrisko             retval = FAIL;
2659665484d8SDoug Ambrisko              goto out;
2660665484d8SDoug Ambrisko         }
2661665484d8SDoug Ambrisko 
2662665484d8SDoug Ambrisko         /* Now try to reset the chip */
2663665484d8SDoug Ambrisko         for (i = 0; i < MRSAS_FUSION_MAX_RESET_TRIES; i++) {
2664665484d8SDoug Ambrisko             mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset),
2665665484d8SDoug Ambrisko                 MPI2_WRSEQ_FLUSH_KEY_VALUE);
2666665484d8SDoug Ambrisko             mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset),
2667665484d8SDoug Ambrisko                 MPI2_WRSEQ_1ST_KEY_VALUE);
2668665484d8SDoug Ambrisko             mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset),
2669665484d8SDoug Ambrisko                 MPI2_WRSEQ_2ND_KEY_VALUE);
2670665484d8SDoug Ambrisko             mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset),
2671665484d8SDoug Ambrisko                 MPI2_WRSEQ_3RD_KEY_VALUE);
2672665484d8SDoug Ambrisko             mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset),
2673665484d8SDoug Ambrisko                 MPI2_WRSEQ_4TH_KEY_VALUE);
2674665484d8SDoug Ambrisko             mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset),
2675665484d8SDoug Ambrisko                 MPI2_WRSEQ_5TH_KEY_VALUE);
2676665484d8SDoug Ambrisko             mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_seq_offset),
2677665484d8SDoug Ambrisko                 MPI2_WRSEQ_6TH_KEY_VALUE);
2678665484d8SDoug Ambrisko 
2679665484d8SDoug Ambrisko             /* Check that the diag write enable (DRWE) bit is on */
2680665484d8SDoug Ambrisko             host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2681665484d8SDoug Ambrisko                                                         fusion_host_diag));
2682665484d8SDoug Ambrisko             retry = 0;
2683665484d8SDoug Ambrisko             while (!(host_diag & HOST_DIAG_WRITE_ENABLE)) {
2684665484d8SDoug Ambrisko                 DELAY(100 * 1000);
2685665484d8SDoug Ambrisko                 host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2686665484d8SDoug Ambrisko                                                         fusion_host_diag));
2687665484d8SDoug Ambrisko                 if (retry++ == 100) {
2688665484d8SDoug Ambrisko                     mrsas_dprint(sc, MRSAS_OCR,
2689665484d8SDoug Ambrisko                     "Host diag unlock failed!\n");
2690665484d8SDoug Ambrisko                     break;
2691665484d8SDoug Ambrisko                 }
2692665484d8SDoug Ambrisko             }
2693665484d8SDoug Ambrisko             if (!(host_diag & HOST_DIAG_WRITE_ENABLE))
2694665484d8SDoug Ambrisko                 continue;
2695665484d8SDoug Ambrisko 
2696665484d8SDoug Ambrisko             /* Send chip reset command */
2697665484d8SDoug Ambrisko             mrsas_write_reg(sc, offsetof(mrsas_reg_set, fusion_host_diag),
2698665484d8SDoug Ambrisko                host_diag | HOST_DIAG_RESET_ADAPTER);
2699665484d8SDoug Ambrisko             DELAY(3000 * 1000);
2700665484d8SDoug Ambrisko 
2701665484d8SDoug Ambrisko             /* Make sure reset adapter bit is cleared */
2702665484d8SDoug Ambrisko             host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2703665484d8SDoug Ambrisko                                                         fusion_host_diag));
2704665484d8SDoug Ambrisko             retry = 0;
2705665484d8SDoug Ambrisko             while (host_diag & HOST_DIAG_RESET_ADAPTER) {
2706665484d8SDoug Ambrisko                 DELAY(100 * 1000);
2707665484d8SDoug Ambrisko                 host_diag = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2708665484d8SDoug Ambrisko                                                         fusion_host_diag));
2709665484d8SDoug Ambrisko                 if (retry++ == 1000) {
2710665484d8SDoug Ambrisko                     mrsas_dprint(sc, MRSAS_OCR,
2711665484d8SDoug Ambrisko                                         "Diag reset adapter never cleared!\n");
2712665484d8SDoug Ambrisko                     break;
2713665484d8SDoug Ambrisko                 }
2714665484d8SDoug Ambrisko             }
2715665484d8SDoug Ambrisko             if (host_diag & HOST_DIAG_RESET_ADAPTER)
2716665484d8SDoug Ambrisko                 continue;
2717665484d8SDoug Ambrisko 
2718665484d8SDoug Ambrisko             abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2719665484d8SDoug Ambrisko                                     outbound_scratch_pad)) & MFI_STATE_MASK;
2720665484d8SDoug Ambrisko             retry = 0;
2721665484d8SDoug Ambrisko 
2722665484d8SDoug Ambrisko             while ((abs_state <= MFI_STATE_FW_INIT) && (retry++ < 1000)) {
2723665484d8SDoug Ambrisko                 DELAY(100 * 1000);
2724665484d8SDoug Ambrisko                 abs_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2725665484d8SDoug Ambrisko                                      outbound_scratch_pad)) & MFI_STATE_MASK;
2726665484d8SDoug Ambrisko             }
2727665484d8SDoug Ambrisko             if (abs_state <= MFI_STATE_FW_INIT) {
2728665484d8SDoug Ambrisko                 mrsas_dprint(sc, MRSAS_OCR, "firmware state < MFI_STATE_FW_INIT,"
2729665484d8SDoug Ambrisko                                 " state = 0x%x\n", abs_state);
2730665484d8SDoug Ambrisko                 continue;
2731665484d8SDoug Ambrisko             }
2732665484d8SDoug Ambrisko 
2733665484d8SDoug Ambrisko             /* Wait for FW to become ready */
2734665484d8SDoug Ambrisko             if (mrsas_transition_to_ready(sc, 1)) {
2735665484d8SDoug Ambrisko                 mrsas_dprint(sc, MRSAS_OCR,
2736665484d8SDoug Ambrisko                            "mrsas: Failed to transition controller to ready.\n");
2737665484d8SDoug Ambrisko                 continue;
2738665484d8SDoug Ambrisko             }
2739665484d8SDoug Ambrisko 
2740665484d8SDoug Ambrisko             mrsas_reset_reply_desc(sc);
2741665484d8SDoug Ambrisko             if (mrsas_ioc_init(sc)) {
2742665484d8SDoug Ambrisko                 mrsas_dprint(sc, MRSAS_OCR, "mrsas_ioc_init() failed!\n");
2743665484d8SDoug Ambrisko                 continue;
2744665484d8SDoug Ambrisko             }
2745665484d8SDoug Ambrisko 
2746665484d8SDoug Ambrisko             clear_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags);
2747665484d8SDoug Ambrisko             mrsas_enable_intr(sc);
2748665484d8SDoug Ambrisko             sc->adprecovery = MRSAS_HBA_OPERATIONAL;
2749665484d8SDoug Ambrisko 
2750665484d8SDoug Ambrisko             /* Re-fire management commands */
2751665484d8SDoug Ambrisko             for (j = 0 ; j < sc->max_fw_cmds; j++) {
2752665484d8SDoug Ambrisko                 mpt_cmd = sc->mpt_cmd_list[j];
2753665484d8SDoug Ambrisko                 if (mpt_cmd->sync_cmd_idx != (u_int32_t)MRSAS_ULONG_MAX) {
2754665484d8SDoug Ambrisko                     mfi_cmd = sc->mfi_cmd_list[mpt_cmd->sync_cmd_idx];
2755665484d8SDoug Ambrisko                     if (mfi_cmd->frame->dcmd.opcode ==
2756665484d8SDoug Ambrisko                                           MR_DCMD_LD_MAP_GET_INFO) {
2757665484d8SDoug Ambrisko                         mrsas_release_mfi_cmd(mfi_cmd);
2758665484d8SDoug Ambrisko                         mrsas_release_mpt_cmd(mpt_cmd);
2759665484d8SDoug Ambrisko                     } else  {
2760665484d8SDoug Ambrisko                         req_desc = mrsas_get_request_desc(sc,
2761665484d8SDoug Ambrisko                             mfi_cmd->cmd_id.context.smid - 1);
2762665484d8SDoug Ambrisko                         mrsas_dprint(sc, MRSAS_OCR,
2763665484d8SDoug Ambrisko                             "Re-fire command DCMD opcode 0x%x index %d\n ",
2764665484d8SDoug Ambrisko                              mfi_cmd->frame->dcmd.opcode, j);
2765665484d8SDoug Ambrisko                         if (!req_desc)
2766665484d8SDoug Ambrisko                             device_printf(sc->mrsas_dev,
2767665484d8SDoug Ambrisko                                           "Cannot build MPT cmd.\n");
2768665484d8SDoug Ambrisko                         else
2769665484d8SDoug Ambrisko                             mrsas_fire_cmd(sc, req_desc->addr.u.low,
2770665484d8SDoug Ambrisko                                                      req_desc->addr.u.high);
2771665484d8SDoug Ambrisko                     }
2772665484d8SDoug Ambrisko                 }
2773665484d8SDoug Ambrisko             }
2774665484d8SDoug Ambrisko 
2775665484d8SDoug Ambrisko             /* Reset load balance info */
2776665484d8SDoug Ambrisko             memset(sc->load_balance_info, 0,
27774799d485SKashyap D Desai                    sizeof(LD_LOAD_BALANCE_INFO) * MAX_LOGICAL_DRIVES_EXT);
2778665484d8SDoug Ambrisko 
2779665484d8SDoug Ambrisko             if (!mrsas_get_map_info(sc))
2780665484d8SDoug Ambrisko                 mrsas_sync_map_info(sc);
2781665484d8SDoug Ambrisko 
2782665484d8SDoug Ambrisko             /* Adapter reset completed successfully */
2783665484d8SDoug Ambrisko             device_printf(sc->mrsas_dev, "Reset successful\n");
2784665484d8SDoug Ambrisko             retval = SUCCESS;
2785665484d8SDoug Ambrisko             goto out;
2786665484d8SDoug Ambrisko         }
2787665484d8SDoug Ambrisko         /* Reset failed, kill the adapter */
2788665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Reset failed, killing adapter.\n");
2789665484d8SDoug Ambrisko         mrsas_kill_hba(sc);
2790665484d8SDoug Ambrisko         retval = FAIL;
2791665484d8SDoug Ambrisko     } else {
2792665484d8SDoug Ambrisko         clear_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags);
2793665484d8SDoug Ambrisko         mrsas_enable_intr(sc);
2794665484d8SDoug Ambrisko         sc->adprecovery = MRSAS_HBA_OPERATIONAL;
2795665484d8SDoug Ambrisko     }
2796665484d8SDoug Ambrisko out:
2797665484d8SDoug Ambrisko     clear_bit(MRSAS_FUSION_IN_RESET, &sc->reset_flags);
2798665484d8SDoug Ambrisko     mrsas_dprint(sc, MRSAS_OCR,
2799665484d8SDoug Ambrisko             "Reset Exit with %d.\n", retval);
2800665484d8SDoug Ambrisko     return retval;
2801665484d8SDoug Ambrisko }
2802665484d8SDoug Ambrisko 
2803665484d8SDoug Ambrisko /**
2804665484d8SDoug Ambrisko  * mrsas_kill_hba       Kill HBA when OCR is not supported.
2805665484d8SDoug Ambrisko  * input:               Adapter Context.
2806665484d8SDoug Ambrisko  *
2807665484d8SDoug Ambrisko  * This function will kill HBA when OCR is not supported.
2808665484d8SDoug Ambrisko  */
2809665484d8SDoug Ambrisko void mrsas_kill_hba (struct mrsas_softc *sc)
2810665484d8SDoug Ambrisko {
2811665484d8SDoug Ambrisko     mrsas_dprint(sc, MRSAS_OCR, "%s\n", __func__);
2812665484d8SDoug Ambrisko     mrsas_write_reg(sc, offsetof(mrsas_reg_set, doorbell),
2813665484d8SDoug Ambrisko                         MFI_STOP_ADP);
2814665484d8SDoug Ambrisko     /* Flush */
2815665484d8SDoug Ambrisko     mrsas_read_reg(sc, offsetof(mrsas_reg_set, doorbell));
2816665484d8SDoug Ambrisko }
2817665484d8SDoug Ambrisko 
2818665484d8SDoug Ambrisko /**
2819665484d8SDoug Ambrisko  * mrsas_wait_for_outstanding           Wait for outstanding commands
2820665484d8SDoug Ambrisko  * input:                               Adapter Context.
2821665484d8SDoug Ambrisko  *
2822665484d8SDoug Ambrisko  * This function will wait for 180 seconds for outstanding
2823665484d8SDoug Ambrisko  * commands to be completed.
2824665484d8SDoug Ambrisko  */
2825665484d8SDoug Ambrisko int mrsas_wait_for_outstanding(struct mrsas_softc *sc)
2826665484d8SDoug Ambrisko {
2827665484d8SDoug Ambrisko     int i, outstanding, retval = 0;
2828*d18d1b47SKashyap D Desai     u_int32_t fw_state, count, MSIxIndex;
2829*d18d1b47SKashyap D Desai 
2830665484d8SDoug Ambrisko 
2831665484d8SDoug Ambrisko     for (i = 0; i < MRSAS_RESET_WAIT_TIME; i++) {
2832665484d8SDoug Ambrisko         if (sc->remove_in_progress) {
2833665484d8SDoug Ambrisko             mrsas_dprint(sc, MRSAS_OCR,
2834665484d8SDoug Ambrisko                 "Driver remove or shutdown called.\n");
2835665484d8SDoug Ambrisko             retval = 1;
2836665484d8SDoug Ambrisko             goto out;
2837665484d8SDoug Ambrisko         }
2838665484d8SDoug Ambrisko         /* Check if firmware is in fault state */
2839665484d8SDoug Ambrisko         fw_state = mrsas_read_reg(sc, offsetof(mrsas_reg_set,
2840665484d8SDoug Ambrisko                                   outbound_scratch_pad)) & MFI_STATE_MASK;
2841665484d8SDoug Ambrisko         if (fw_state == MFI_STATE_FAULT) {
2842665484d8SDoug Ambrisko             mrsas_dprint(sc, MRSAS_OCR,
2843665484d8SDoug Ambrisko                          "Found FW in FAULT state, will reset adapter.\n");
2844665484d8SDoug Ambrisko             retval = 1;
2845665484d8SDoug Ambrisko             goto out;
2846665484d8SDoug Ambrisko         }
2847665484d8SDoug Ambrisko         outstanding = atomic_read(&sc->fw_outstanding);
2848665484d8SDoug Ambrisko         if (!outstanding)
2849665484d8SDoug Ambrisko             goto out;
2850665484d8SDoug Ambrisko 
2851665484d8SDoug Ambrisko         if (!(i % MRSAS_RESET_NOTICE_INTERVAL)) {
2852665484d8SDoug Ambrisko             mrsas_dprint(sc, MRSAS_OCR, "[%2d]waiting for %d "
2853665484d8SDoug Ambrisko                                 "commands to complete\n",i,outstanding);
2854*d18d1b47SKashyap D Desai             count = sc->msix_vectors > 0 ? sc->msix_vectors : 1;
2855*d18d1b47SKashyap D Desai             for (MSIxIndex = 0 ; MSIxIndex < count; MSIxIndex++)
2856*d18d1b47SKashyap D Desai                   mrsas_complete_cmd(sc, MSIxIndex);
2857665484d8SDoug Ambrisko         }
2858665484d8SDoug Ambrisko         DELAY(1000 * 1000);
2859665484d8SDoug Ambrisko     }
2860665484d8SDoug Ambrisko 
2861665484d8SDoug Ambrisko     if (atomic_read(&sc->fw_outstanding)) {
2862665484d8SDoug Ambrisko         mrsas_dprint(sc, MRSAS_OCR,
2863665484d8SDoug Ambrisko                         " pending commands remain after waiting,"
2864665484d8SDoug Ambrisko                         " will reset adapter.\n");
2865665484d8SDoug Ambrisko         retval = 1;
2866665484d8SDoug Ambrisko     }
2867665484d8SDoug Ambrisko out:
2868665484d8SDoug Ambrisko     return retval;
2869665484d8SDoug Ambrisko }
2870665484d8SDoug Ambrisko 
2871665484d8SDoug Ambrisko /**
2872665484d8SDoug Ambrisko  * mrsas_release_mfi_cmd: Return a cmd to free command pool
2873665484d8SDoug Ambrisko  * input:                 Command packet for return to free cmd pool
2874665484d8SDoug Ambrisko  *
2875665484d8SDoug Ambrisko  * This function returns the MFI command to the command list.
2876665484d8SDoug Ambrisko  */
2877665484d8SDoug Ambrisko void mrsas_release_mfi_cmd(struct mrsas_mfi_cmd *cmd)
2878665484d8SDoug Ambrisko {
2879665484d8SDoug Ambrisko     struct mrsas_softc *sc = cmd->sc;
2880665484d8SDoug Ambrisko 
2881665484d8SDoug Ambrisko     mtx_lock(&sc->mfi_cmd_pool_lock);
2882665484d8SDoug Ambrisko     cmd->ccb_ptr = NULL;
2883665484d8SDoug Ambrisko 	cmd->cmd_id.frame_count = 0;
2884665484d8SDoug Ambrisko     TAILQ_INSERT_TAIL(&(sc->mrsas_mfi_cmd_list_head), cmd, next);
2885665484d8SDoug Ambrisko     mtx_unlock(&sc->mfi_cmd_pool_lock);
2886665484d8SDoug Ambrisko 
2887665484d8SDoug Ambrisko     return;
2888665484d8SDoug Ambrisko }
2889665484d8SDoug Ambrisko 
2890665484d8SDoug Ambrisko /**
2891665484d8SDoug Ambrisko  * mrsas_get_controller_info -        Returns FW's controller structure
2892665484d8SDoug Ambrisko  * input:                             Adapter soft state
2893665484d8SDoug Ambrisko  *                                    Controller information structure
2894665484d8SDoug Ambrisko  *
2895665484d8SDoug Ambrisko  * Issues an internal command (DCMD) to get the FW's controller structure.
2896665484d8SDoug Ambrisko  * This information is mainly used to find out the maximum IO transfer per
2897665484d8SDoug Ambrisko  * command supported by the FW.
2898665484d8SDoug Ambrisko  */
2899665484d8SDoug Ambrisko static int mrsas_get_ctrl_info(struct mrsas_softc *sc,
2900665484d8SDoug Ambrisko                       struct mrsas_ctrl_info *ctrl_info)
2901665484d8SDoug Ambrisko {
2902665484d8SDoug Ambrisko     int retcode = 0;
2903665484d8SDoug Ambrisko     struct mrsas_mfi_cmd *cmd;
2904665484d8SDoug Ambrisko     struct mrsas_dcmd_frame *dcmd;
2905665484d8SDoug Ambrisko 
2906665484d8SDoug Ambrisko     cmd = mrsas_get_mfi_cmd(sc);
2907665484d8SDoug Ambrisko 
2908665484d8SDoug Ambrisko     if (!cmd) {
2909665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Failed to get a free cmd\n");
2910665484d8SDoug Ambrisko         return -ENOMEM;
2911665484d8SDoug Ambrisko     }
2912665484d8SDoug Ambrisko     dcmd = &cmd->frame->dcmd;
2913665484d8SDoug Ambrisko 
2914665484d8SDoug Ambrisko     if (mrsas_alloc_ctlr_info_cmd(sc) != SUCCESS) {
2915665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot allocate get ctlr info cmd\n");
2916665484d8SDoug Ambrisko         mrsas_release_mfi_cmd(cmd);
2917665484d8SDoug Ambrisko         return -ENOMEM;
2918665484d8SDoug Ambrisko     }
2919665484d8SDoug Ambrisko     memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE);
2920665484d8SDoug Ambrisko 
2921665484d8SDoug Ambrisko     dcmd->cmd = MFI_CMD_DCMD;
2922665484d8SDoug Ambrisko     dcmd->cmd_status = 0xFF;
2923665484d8SDoug Ambrisko     dcmd->sge_count = 1;
2924665484d8SDoug Ambrisko     dcmd->flags = MFI_FRAME_DIR_READ;
2925665484d8SDoug Ambrisko     dcmd->timeout = 0;
2926665484d8SDoug Ambrisko     dcmd->pad_0 = 0;
2927665484d8SDoug Ambrisko     dcmd->data_xfer_len = sizeof(struct mrsas_ctrl_info);
2928665484d8SDoug Ambrisko     dcmd->opcode = MR_DCMD_CTRL_GET_INFO;
2929665484d8SDoug Ambrisko     dcmd->sgl.sge32[0].phys_addr = sc->ctlr_info_phys_addr;
2930665484d8SDoug Ambrisko     dcmd->sgl.sge32[0].length = sizeof(struct mrsas_ctrl_info);
2931665484d8SDoug Ambrisko 
2932665484d8SDoug Ambrisko     if (!mrsas_issue_polled(sc, cmd))
2933665484d8SDoug Ambrisko         memcpy(ctrl_info, sc->ctlr_info_mem, sizeof(struct mrsas_ctrl_info));
2934665484d8SDoug Ambrisko     else
2935665484d8SDoug Ambrisko         retcode = 1;
2936665484d8SDoug Ambrisko 
2937665484d8SDoug Ambrisko     mrsas_free_ctlr_info_cmd(sc);
2938665484d8SDoug Ambrisko     mrsas_release_mfi_cmd(cmd);
2939665484d8SDoug Ambrisko     return(retcode);
2940665484d8SDoug Ambrisko }
2941665484d8SDoug Ambrisko 
2942665484d8SDoug Ambrisko /**
2943665484d8SDoug Ambrisko  * mrsas_alloc_ctlr_info_cmd:  Allocates memory for controller info command
2944665484d8SDoug Ambrisko  * input:                      Adapter soft state
2945665484d8SDoug Ambrisko  *
2946665484d8SDoug Ambrisko  * Allocates DMAable memory for the controller info internal command.
2947665484d8SDoug Ambrisko  */
2948665484d8SDoug Ambrisko int mrsas_alloc_ctlr_info_cmd(struct mrsas_softc *sc)
2949665484d8SDoug Ambrisko {
2950665484d8SDoug Ambrisko     int ctlr_info_size;
2951665484d8SDoug Ambrisko 
2952665484d8SDoug Ambrisko     /* Allocate get controller info command */
2953665484d8SDoug Ambrisko     ctlr_info_size = sizeof(struct mrsas_ctrl_info);
2954665484d8SDoug Ambrisko     if (bus_dma_tag_create( sc->mrsas_parent_tag,   // parent
2955665484d8SDoug Ambrisko                             1, 0,                   // algnmnt, boundary
2956665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR_32BIT,// lowaddr
2957665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR,      // highaddr
2958665484d8SDoug Ambrisko                             NULL, NULL,             // filter, filterarg
2959665484d8SDoug Ambrisko                             ctlr_info_size,          // maxsize
2960665484d8SDoug Ambrisko                             1,                      // msegments
2961665484d8SDoug Ambrisko                             ctlr_info_size,          // maxsegsize
2962665484d8SDoug Ambrisko                             BUS_DMA_ALLOCNOW,       // flags
2963665484d8SDoug Ambrisko                             NULL, NULL,             // lockfunc, lockarg
2964665484d8SDoug Ambrisko                             &sc->ctlr_info_tag)) {
2965665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot allocate ctlr info tag\n");
2966665484d8SDoug Ambrisko         return (ENOMEM);
2967665484d8SDoug Ambrisko     }
2968665484d8SDoug Ambrisko     if (bus_dmamem_alloc(sc->ctlr_info_tag, (void **)&sc->ctlr_info_mem,
2969665484d8SDoug Ambrisko             BUS_DMA_NOWAIT, &sc->ctlr_info_dmamap)) {
2970665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot allocate ctlr info cmd mem\n");
2971665484d8SDoug Ambrisko         return (ENOMEM);
2972665484d8SDoug Ambrisko     }
2973665484d8SDoug Ambrisko     if (bus_dmamap_load(sc->ctlr_info_tag, sc->ctlr_info_dmamap,
2974665484d8SDoug Ambrisko             sc->ctlr_info_mem, ctlr_info_size, mrsas_addr_cb,
2975665484d8SDoug Ambrisko             &sc->ctlr_info_phys_addr, BUS_DMA_NOWAIT)) {
2976665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot load ctlr info cmd mem\n");
2977665484d8SDoug Ambrisko         return (ENOMEM);
2978665484d8SDoug Ambrisko     }
2979665484d8SDoug Ambrisko 
2980665484d8SDoug Ambrisko     memset(sc->ctlr_info_mem, 0, ctlr_info_size);
2981665484d8SDoug Ambrisko     return (0);
2982665484d8SDoug Ambrisko }
2983665484d8SDoug Ambrisko 
2984665484d8SDoug Ambrisko /**
2985665484d8SDoug Ambrisko  * mrsas_free_ctlr_info_cmd: Free memory for controller info command
2986665484d8SDoug Ambrisko  * input:                    Adapter soft state
2987665484d8SDoug Ambrisko  *
2988665484d8SDoug Ambrisko  * Deallocates memory of the get controller info cmd.
2989665484d8SDoug Ambrisko  */
2990665484d8SDoug Ambrisko void mrsas_free_ctlr_info_cmd(struct mrsas_softc *sc)
2991665484d8SDoug Ambrisko {
2992665484d8SDoug Ambrisko     if (sc->ctlr_info_phys_addr)
2993665484d8SDoug Ambrisko         bus_dmamap_unload(sc->ctlr_info_tag, sc->ctlr_info_dmamap);
2994665484d8SDoug Ambrisko     if (sc->ctlr_info_mem != NULL)
2995665484d8SDoug Ambrisko         bus_dmamem_free(sc->ctlr_info_tag, sc->ctlr_info_mem, sc->ctlr_info_dmamap);
2996665484d8SDoug Ambrisko     if (sc->ctlr_info_tag != NULL)
2997665484d8SDoug Ambrisko         bus_dma_tag_destroy(sc->ctlr_info_tag);
2998665484d8SDoug Ambrisko }
2999665484d8SDoug Ambrisko 
3000665484d8SDoug Ambrisko /**
3001665484d8SDoug Ambrisko  * mrsas_issue_polled:        Issues a polling command
3002665484d8SDoug Ambrisko  * inputs:                    Adapter soft state
3003665484d8SDoug Ambrisko  *                            Command packet to be issued
3004665484d8SDoug Ambrisko  *
3005665484d8SDoug Ambrisko  * This function is for posting of internal commands to Firmware.  MFI
3006665484d8SDoug Ambrisko  * requires the cmd_status to be set to 0xFF before posting.  The maximun
3007665484d8SDoug Ambrisko  * wait time of the poll response timer is 180 seconds.
3008665484d8SDoug Ambrisko  */
3009665484d8SDoug Ambrisko int mrsas_issue_polled(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd)
3010665484d8SDoug Ambrisko {
3011665484d8SDoug Ambrisko     struct mrsas_header *frame_hdr = &cmd->frame->hdr;
3012665484d8SDoug Ambrisko     u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME;
3013665484d8SDoug Ambrisko     int i, retcode = 0;
3014665484d8SDoug Ambrisko 
3015665484d8SDoug Ambrisko     frame_hdr->cmd_status = 0xFF;
3016665484d8SDoug Ambrisko     frame_hdr->flags |= MFI_FRAME_DONT_POST_IN_REPLY_QUEUE;
3017665484d8SDoug Ambrisko 
3018665484d8SDoug Ambrisko     /* Issue the frame using inbound queue port */
3019665484d8SDoug Ambrisko     if (mrsas_issue_dcmd(sc, cmd)) {
3020665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot issue DCMD internal command.\n");
3021665484d8SDoug Ambrisko         return(1);
3022665484d8SDoug Ambrisko     }
3023665484d8SDoug Ambrisko 
3024665484d8SDoug Ambrisko     /*
3025665484d8SDoug Ambrisko      * Poll response timer to wait for Firmware response.  While this
3026665484d8SDoug Ambrisko      * timer with the DELAY call could block CPU, the time interval for
3027665484d8SDoug Ambrisko      * this is only 1 millisecond.
3028665484d8SDoug Ambrisko      */
3029665484d8SDoug Ambrisko     if (frame_hdr->cmd_status == 0xFF) {
3030665484d8SDoug Ambrisko         for (i=0; i < (max_wait * 1000); i++){
3031665484d8SDoug Ambrisko             if (frame_hdr->cmd_status == 0xFF)
3032665484d8SDoug Ambrisko                 DELAY(1000);
3033665484d8SDoug Ambrisko             else
3034665484d8SDoug Ambrisko                 break;
3035665484d8SDoug Ambrisko         }
3036665484d8SDoug Ambrisko     }
3037665484d8SDoug Ambrisko     if (frame_hdr->cmd_status != 0)
3038665484d8SDoug Ambrisko     {
3039665484d8SDoug Ambrisko         if (frame_hdr->cmd_status == 0xFF)
3040665484d8SDoug Ambrisko             device_printf(sc->mrsas_dev, "DCMD timed out after %d seconds.\n", max_wait);
3041665484d8SDoug Ambrisko         else
3042665484d8SDoug Ambrisko             device_printf(sc->mrsas_dev, "DCMD failed, status = 0x%x\n", frame_hdr->cmd_status);
3043665484d8SDoug Ambrisko         retcode = 1;
3044665484d8SDoug Ambrisko     }
3045665484d8SDoug Ambrisko     return(retcode);
3046665484d8SDoug Ambrisko }
3047665484d8SDoug Ambrisko 
3048665484d8SDoug Ambrisko /**
3049665484d8SDoug Ambrisko  * mrsas_issue_dcmd -     Issues a MFI Pass thru cmd
3050665484d8SDoug Ambrisko  * input:                 Adapter soft state
3051665484d8SDoug Ambrisko  *                        mfi cmd pointer
3052665484d8SDoug Ambrisko  *
3053665484d8SDoug Ambrisko  * This function is called by mrsas_issued_blocked_cmd() and
3054665484d8SDoug Ambrisko  * mrsas_issued_polled(), to build the MPT command and then fire the
3055665484d8SDoug Ambrisko  * command to Firmware.
3056665484d8SDoug Ambrisko  */
3057665484d8SDoug Ambrisko int
3058665484d8SDoug Ambrisko mrsas_issue_dcmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd)
3059665484d8SDoug Ambrisko {
3060665484d8SDoug Ambrisko     MRSAS_REQUEST_DESCRIPTOR_UNION *req_desc;
3061665484d8SDoug Ambrisko 
3062665484d8SDoug Ambrisko     req_desc = mrsas_build_mpt_cmd(sc, cmd);
3063665484d8SDoug Ambrisko     if (!req_desc) {
3064665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot build MPT cmd.\n");
3065665484d8SDoug Ambrisko         return(1);
3066665484d8SDoug Ambrisko     }
3067665484d8SDoug Ambrisko 
3068665484d8SDoug Ambrisko     mrsas_fire_cmd(sc, req_desc->addr.u.low, req_desc->addr.u.high);
3069665484d8SDoug Ambrisko 
3070665484d8SDoug Ambrisko     return(0);
3071665484d8SDoug Ambrisko }
3072665484d8SDoug Ambrisko 
3073665484d8SDoug Ambrisko /**
3074665484d8SDoug Ambrisko  * mrsas_build_mpt_cmd - Calls helper function to build Passthru cmd
3075665484d8SDoug Ambrisko  * input:                Adapter soft state
3076665484d8SDoug Ambrisko  *                       mfi cmd to build
3077665484d8SDoug Ambrisko  *
3078665484d8SDoug Ambrisko  * This function is called by mrsas_issue_cmd() to build the MPT-MFI
3079665484d8SDoug Ambrisko  * passthru command and prepares the MPT command to send to Firmware.
3080665484d8SDoug Ambrisko  */
3081665484d8SDoug Ambrisko MRSAS_REQUEST_DESCRIPTOR_UNION *
3082665484d8SDoug Ambrisko mrsas_build_mpt_cmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd)
3083665484d8SDoug Ambrisko {
3084665484d8SDoug Ambrisko     MRSAS_REQUEST_DESCRIPTOR_UNION *req_desc;
3085665484d8SDoug Ambrisko     u_int16_t index;
3086665484d8SDoug Ambrisko 
3087665484d8SDoug Ambrisko     if (mrsas_build_mptmfi_passthru(sc, cmd)) {
3088665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot build MPT-MFI passthru cmd.\n");
3089665484d8SDoug Ambrisko         return NULL;
3090665484d8SDoug Ambrisko     }
3091665484d8SDoug Ambrisko 
3092665484d8SDoug Ambrisko     index = cmd->cmd_id.context.smid;
3093665484d8SDoug Ambrisko 
3094665484d8SDoug Ambrisko     req_desc = mrsas_get_request_desc(sc, index-1);
3095665484d8SDoug Ambrisko     if(!req_desc)
3096665484d8SDoug Ambrisko         return NULL;
3097665484d8SDoug Ambrisko 
3098665484d8SDoug Ambrisko     req_desc->addr.Words = 0;
3099665484d8SDoug Ambrisko     req_desc->SCSIIO.RequestFlags = (MPI2_REQ_DESCRIPT_FLAGS_SCSI_IO << MRSAS_REQ_DESCRIPT_FLAGS_TYPE_SHIFT);
3100665484d8SDoug Ambrisko 
3101665484d8SDoug Ambrisko     req_desc->SCSIIO.SMID = index;
3102665484d8SDoug Ambrisko 
3103665484d8SDoug Ambrisko     return(req_desc);
3104665484d8SDoug Ambrisko }
3105665484d8SDoug Ambrisko 
3106665484d8SDoug Ambrisko /**
3107665484d8SDoug Ambrisko  * mrsas_build_mptmfi_passthru - Builds a MPT MFI Passthru command
3108665484d8SDoug Ambrisko  * input:                        Adapter soft state
3109665484d8SDoug Ambrisko  *                               mfi cmd pointer
3110665484d8SDoug Ambrisko  *
3111665484d8SDoug Ambrisko  * The MPT command and the io_request are setup as a passthru command.
3112665484d8SDoug Ambrisko  * The SGE chain address is set to frame_phys_addr of the MFI command.
3113665484d8SDoug Ambrisko  */
3114665484d8SDoug Ambrisko u_int8_t
3115665484d8SDoug Ambrisko mrsas_build_mptmfi_passthru(struct mrsas_softc *sc, struct mrsas_mfi_cmd *mfi_cmd)
3116665484d8SDoug Ambrisko {
3117665484d8SDoug Ambrisko     MPI25_IEEE_SGE_CHAIN64 *mpi25_ieee_chain;
3118665484d8SDoug Ambrisko     PTR_MRSAS_RAID_SCSI_IO_REQUEST io_req;
3119665484d8SDoug Ambrisko     struct mrsas_mpt_cmd *mpt_cmd;
3120665484d8SDoug Ambrisko     struct mrsas_header *frame_hdr = &mfi_cmd->frame->hdr;
3121665484d8SDoug Ambrisko 
3122665484d8SDoug Ambrisko     mpt_cmd = mrsas_get_mpt_cmd(sc);
3123665484d8SDoug Ambrisko     if (!mpt_cmd)
3124665484d8SDoug Ambrisko         return(1);
3125665484d8SDoug Ambrisko 
3126665484d8SDoug Ambrisko     /* Save the smid. To be used for returning the cmd */
3127665484d8SDoug Ambrisko     mfi_cmd->cmd_id.context.smid = mpt_cmd->index;
3128665484d8SDoug Ambrisko 
3129665484d8SDoug Ambrisko     mpt_cmd->sync_cmd_idx = mfi_cmd->index;
3130665484d8SDoug Ambrisko 
3131665484d8SDoug Ambrisko     /*
3132665484d8SDoug Ambrisko      * For cmds where the flag is set, store the flag and check
3133665484d8SDoug Ambrisko      * on completion. For cmds with this flag, don't call
3134665484d8SDoug Ambrisko      * mrsas_complete_cmd.
3135665484d8SDoug Ambrisko      */
3136665484d8SDoug Ambrisko 
3137665484d8SDoug Ambrisko     if (frame_hdr->flags & MFI_FRAME_DONT_POST_IN_REPLY_QUEUE)
3138665484d8SDoug Ambrisko         mpt_cmd->flags = MFI_FRAME_DONT_POST_IN_REPLY_QUEUE;
3139665484d8SDoug Ambrisko 
3140665484d8SDoug Ambrisko     io_req = mpt_cmd->io_request;
3141665484d8SDoug Ambrisko 
3142665484d8SDoug Ambrisko     if ((sc->device_id == MRSAS_INVADER) || (sc->device_id == MRSAS_FURY)) {
3143665484d8SDoug Ambrisko 		pMpi25IeeeSgeChain64_t sgl_ptr_end = (pMpi25IeeeSgeChain64_t) &io_req->SGL;
3144665484d8SDoug Ambrisko                 sgl_ptr_end += sc->max_sge_in_main_msg - 1;
3145665484d8SDoug Ambrisko                 sgl_ptr_end->Flags = 0;
3146665484d8SDoug Ambrisko     }
3147665484d8SDoug Ambrisko 
3148665484d8SDoug Ambrisko     mpi25_ieee_chain = (MPI25_IEEE_SGE_CHAIN64 *)&io_req->SGL.IeeeChain;
3149665484d8SDoug Ambrisko 
3150665484d8SDoug Ambrisko     io_req->Function    = MRSAS_MPI2_FUNCTION_PASSTHRU_IO_REQUEST;
3151665484d8SDoug Ambrisko     io_req->SGLOffset0  = offsetof(MRSAS_RAID_SCSI_IO_REQUEST, SGL) / 4;
3152665484d8SDoug Ambrisko     io_req->ChainOffset = sc->chain_offset_mfi_pthru;
3153665484d8SDoug Ambrisko 
3154665484d8SDoug Ambrisko     mpi25_ieee_chain->Address = mfi_cmd->frame_phys_addr;
3155665484d8SDoug Ambrisko 
3156665484d8SDoug Ambrisko     mpi25_ieee_chain->Flags= IEEE_SGE_FLAGS_CHAIN_ELEMENT |
3157665484d8SDoug Ambrisko               MPI2_IEEE_SGE_FLAGS_IOCPLBNTA_ADDR;
3158665484d8SDoug Ambrisko 
3159665484d8SDoug Ambrisko     mpi25_ieee_chain->Length = MRSAS_MAX_SZ_CHAIN_FRAME;
3160665484d8SDoug Ambrisko 
3161665484d8SDoug Ambrisko     return(0);
3162665484d8SDoug Ambrisko }
3163665484d8SDoug Ambrisko 
3164665484d8SDoug Ambrisko /**
3165665484d8SDoug Ambrisko  * mrsas_issue_blocked_cmd - Synchronous wrapper around regular FW cmds
3166665484d8SDoug Ambrisko  * input:                    Adapter soft state
3167665484d8SDoug Ambrisko  *                           Command to be issued
3168665484d8SDoug Ambrisko  *
3169665484d8SDoug Ambrisko  * This function waits on an event for the command to be returned
3170665484d8SDoug Ambrisko  * from the ISR. Max wait time is MRSAS_INTERNAL_CMD_WAIT_TIME secs.
3171665484d8SDoug Ambrisko  * Used for issuing internal and ioctl commands.
3172665484d8SDoug Ambrisko  */
3173665484d8SDoug Ambrisko int mrsas_issue_blocked_cmd(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd)
3174665484d8SDoug Ambrisko {
3175665484d8SDoug Ambrisko     u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME;
3176665484d8SDoug Ambrisko     unsigned long total_time = 0;
3177665484d8SDoug Ambrisko     int retcode = 0;
3178665484d8SDoug Ambrisko 
3179665484d8SDoug Ambrisko     /* Initialize cmd_status */
3180665484d8SDoug Ambrisko     cmd->cmd_status = ECONNREFUSED;
3181665484d8SDoug Ambrisko 
3182665484d8SDoug Ambrisko     /* Build MPT-MFI command for issue to FW */
3183665484d8SDoug Ambrisko     if (mrsas_issue_dcmd(sc, cmd)){
3184665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot issue DCMD internal command.\n");
3185665484d8SDoug Ambrisko         return(1);
3186665484d8SDoug Ambrisko     }
3187665484d8SDoug Ambrisko 
3188665484d8SDoug Ambrisko     sc->chan = (void*)&cmd;
3189665484d8SDoug Ambrisko 
3190665484d8SDoug Ambrisko     /* The following is for debug only... */
3191665484d8SDoug Ambrisko     //device_printf(sc->mrsas_dev,"DCMD issued to FW, about to sleep-wait...\n");
3192665484d8SDoug Ambrisko     //device_printf(sc->mrsas_dev,"sc->chan = %p\n", sc->chan);
3193665484d8SDoug Ambrisko 
3194665484d8SDoug Ambrisko     while (1) {
3195665484d8SDoug Ambrisko        if (cmd->cmd_status == ECONNREFUSED){
3196665484d8SDoug Ambrisko            tsleep((void *)&sc->chan, 0, "mrsas_sleep", hz);
3197665484d8SDoug Ambrisko        }
3198665484d8SDoug Ambrisko        else
3199665484d8SDoug Ambrisko            break;
3200665484d8SDoug Ambrisko        total_time++;
3201665484d8SDoug Ambrisko        if (total_time >= max_wait) {
3202665484d8SDoug Ambrisko            device_printf(sc->mrsas_dev, "Internal command timed out after %d seconds.\n", max_wait);
3203665484d8SDoug Ambrisko            retcode = 1;
3204665484d8SDoug Ambrisko            break;
3205665484d8SDoug Ambrisko        }
3206665484d8SDoug Ambrisko     }
3207665484d8SDoug Ambrisko     return(retcode);
3208665484d8SDoug Ambrisko }
3209665484d8SDoug Ambrisko 
3210665484d8SDoug Ambrisko /**
3211665484d8SDoug Ambrisko  * mrsas_complete_mptmfi_passthru - Completes a command
3212665484d8SDoug Ambrisko  * input:                           sc: Adapter soft state
3213665484d8SDoug Ambrisko  *                                  cmd: Command to be completed
3214665484d8SDoug Ambrisko  *                                  status: cmd completion status
3215665484d8SDoug Ambrisko  *
3216665484d8SDoug Ambrisko  * This function is called from mrsas_complete_cmd() after an interrupt
3217665484d8SDoug Ambrisko  * is received from Firmware, and io_request->Function is
3218665484d8SDoug Ambrisko  * MRSAS_MPI2_FUNCTION_PASSTHRU_IO_REQUEST.
3219665484d8SDoug Ambrisko  */
3220665484d8SDoug Ambrisko void
3221665484d8SDoug Ambrisko mrsas_complete_mptmfi_passthru(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd,
3222665484d8SDoug Ambrisko                      u_int8_t status)
3223665484d8SDoug Ambrisko {
3224665484d8SDoug Ambrisko     struct mrsas_header *hdr = &cmd->frame->hdr;
3225665484d8SDoug Ambrisko     u_int8_t cmd_status = cmd->frame->hdr.cmd_status;
3226665484d8SDoug Ambrisko 
3227665484d8SDoug Ambrisko     /* Reset the retry counter for future re-tries */
3228665484d8SDoug Ambrisko     cmd->retry_for_fw_reset = 0;
3229665484d8SDoug Ambrisko 
3230665484d8SDoug Ambrisko     if (cmd->ccb_ptr)
3231665484d8SDoug Ambrisko         cmd->ccb_ptr = NULL;
3232665484d8SDoug Ambrisko 
3233665484d8SDoug Ambrisko     switch (hdr->cmd) {
3234665484d8SDoug Ambrisko         case MFI_CMD_INVALID:
3235665484d8SDoug Ambrisko             device_printf(sc->mrsas_dev, "MFI_CMD_INVALID command.\n");
3236665484d8SDoug Ambrisko             break;
3237665484d8SDoug Ambrisko         case MFI_CMD_PD_SCSI_IO:
3238665484d8SDoug Ambrisko         case MFI_CMD_LD_SCSI_IO:
3239665484d8SDoug Ambrisko             /*
3240665484d8SDoug Ambrisko              * MFI_CMD_PD_SCSI_IO and MFI_CMD_LD_SCSI_IO could have been
3241665484d8SDoug Ambrisko              * issued either through an IO path or an IOCTL path. If it
3242665484d8SDoug Ambrisko              * was via IOCTL, we will send it to internal completion.
3243665484d8SDoug Ambrisko              */
3244665484d8SDoug Ambrisko             if (cmd->sync_cmd) {
3245665484d8SDoug Ambrisko                 cmd->sync_cmd = 0;
3246665484d8SDoug Ambrisko                 mrsas_wakeup(sc, cmd);
3247665484d8SDoug Ambrisko                 break;
3248665484d8SDoug Ambrisko             }
3249665484d8SDoug Ambrisko         case MFI_CMD_SMP:
3250665484d8SDoug Ambrisko         case MFI_CMD_STP:
3251665484d8SDoug Ambrisko         case MFI_CMD_DCMD:
3252665484d8SDoug Ambrisko             /* Check for LD map update */
3253665484d8SDoug Ambrisko             if ((cmd->frame->dcmd.opcode == MR_DCMD_LD_MAP_GET_INFO) &&
3254665484d8SDoug Ambrisko                 (cmd->frame->dcmd.mbox.b[1] == 1)) {
3255665484d8SDoug Ambrisko                 sc->fast_path_io = 0;
3256665484d8SDoug Ambrisko 		        mtx_lock(&sc->raidmap_lock);
3257665484d8SDoug Ambrisko                 if (cmd_status != 0) {
3258665484d8SDoug Ambrisko                     if (cmd_status != MFI_STAT_NOT_FOUND)
3259665484d8SDoug Ambrisko                         device_printf(sc->mrsas_dev, "map sync failed, status=%x\n",cmd_status);
3260665484d8SDoug Ambrisko                     else {
3261665484d8SDoug Ambrisko                         mrsas_release_mfi_cmd(cmd);
3262665484d8SDoug Ambrisko 		        mtx_unlock(&sc->raidmap_lock);
3263665484d8SDoug Ambrisko                         break;
3264665484d8SDoug Ambrisko                     }
3265665484d8SDoug Ambrisko                 }
3266665484d8SDoug Ambrisko                 else
3267665484d8SDoug Ambrisko                     sc->map_id++;
3268665484d8SDoug Ambrisko                 mrsas_release_mfi_cmd(cmd);
3269665484d8SDoug Ambrisko                 if (MR_ValidateMapInfo(sc))
3270665484d8SDoug Ambrisko                     sc->fast_path_io = 0;
3271665484d8SDoug Ambrisko                 else
3272665484d8SDoug Ambrisko                     sc->fast_path_io = 1;
3273665484d8SDoug Ambrisko                 mrsas_sync_map_info(sc);
3274665484d8SDoug Ambrisko                 mtx_unlock(&sc->raidmap_lock);
3275665484d8SDoug Ambrisko                 break;
3276665484d8SDoug Ambrisko             }
3277665484d8SDoug Ambrisko #if 0 //currently not supporting event handling, so commenting out
3278665484d8SDoug Ambrisko             if (cmd->frame->dcmd.opcode == MR_DCMD_CTRL_EVENT_GET_INFO ||
3279665484d8SDoug Ambrisko                     cmd->frame->dcmd.opcode == MR_DCMD_CTRL_EVENT_GET) {
3280665484d8SDoug Ambrisko                 mrsas_poll_wait_aen = 0;
3281665484d8SDoug Ambrisko             }
3282665484d8SDoug Ambrisko #endif
3283665484d8SDoug Ambrisko             /* See if got an event notification */
3284665484d8SDoug Ambrisko             if (cmd->frame->dcmd.opcode == MR_DCMD_CTRL_EVENT_WAIT)
3285665484d8SDoug Ambrisko                 mrsas_complete_aen(sc, cmd);
3286665484d8SDoug Ambrisko             else
3287665484d8SDoug Ambrisko                 mrsas_wakeup(sc, cmd);
3288665484d8SDoug Ambrisko             break;
3289665484d8SDoug Ambrisko         case MFI_CMD_ABORT:
3290665484d8SDoug Ambrisko             /* Command issued to abort another cmd return */
3291665484d8SDoug Ambrisko             mrsas_complete_abort(sc, cmd);
3292665484d8SDoug Ambrisko             break;
3293665484d8SDoug Ambrisko         default:
3294665484d8SDoug Ambrisko             device_printf(sc->mrsas_dev,"Unknown command completed! [0x%X]\n", hdr->cmd);
3295665484d8SDoug Ambrisko             break;
3296665484d8SDoug Ambrisko     }
3297665484d8SDoug Ambrisko }
3298665484d8SDoug Ambrisko 
3299665484d8SDoug Ambrisko /**
3300665484d8SDoug Ambrisko  * mrsas_wakeup -         Completes an internal command
3301665484d8SDoug Ambrisko  * input:                 Adapter soft state
3302665484d8SDoug Ambrisko  *                        Command to be completed
3303665484d8SDoug Ambrisko  *
3304665484d8SDoug Ambrisko  * In mrsas_issue_blocked_cmd(), after a command is issued to Firmware,
3305665484d8SDoug Ambrisko  * a wait timer is started.  This function is called from
3306665484d8SDoug Ambrisko  * mrsas_complete_mptmfi_passthru() as it completes the command,
3307665484d8SDoug Ambrisko  * to wake up from the command wait.
3308665484d8SDoug Ambrisko  */
3309665484d8SDoug Ambrisko void mrsas_wakeup(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd)
3310665484d8SDoug Ambrisko {
3311665484d8SDoug Ambrisko     cmd->cmd_status = cmd->frame->io.cmd_status;
3312665484d8SDoug Ambrisko 
3313665484d8SDoug Ambrisko     if (cmd->cmd_status == ECONNREFUSED)
3314665484d8SDoug Ambrisko         cmd->cmd_status = 0;
3315665484d8SDoug Ambrisko 
3316665484d8SDoug Ambrisko     /* For debug only ... */
3317665484d8SDoug Ambrisko     //device_printf(sc->mrsas_dev,"DCMD rec'd for wakeup, sc->chan=%p\n", sc->chan);
3318665484d8SDoug Ambrisko 
3319665484d8SDoug Ambrisko     sc->chan = (void*)&cmd;
3320665484d8SDoug Ambrisko     wakeup_one((void *)&sc->chan);
3321665484d8SDoug Ambrisko     return;
3322665484d8SDoug Ambrisko }
3323665484d8SDoug Ambrisko 
3324665484d8SDoug Ambrisko /**
3325665484d8SDoug Ambrisko  * mrsas_shutdown_ctlr:       Instructs FW to shutdown the controller
3326665484d8SDoug Ambrisko  * input:                     Adapter soft state
3327665484d8SDoug Ambrisko  *                            Shutdown/Hibernate
3328665484d8SDoug Ambrisko  *
3329665484d8SDoug Ambrisko  * This function issues a DCMD internal command to Firmware to initiate
3330665484d8SDoug Ambrisko  * shutdown of the controller.
3331665484d8SDoug Ambrisko  */
3332665484d8SDoug Ambrisko static void mrsas_shutdown_ctlr(struct mrsas_softc *sc, u_int32_t opcode)
3333665484d8SDoug Ambrisko {
3334665484d8SDoug Ambrisko     struct mrsas_mfi_cmd *cmd;
3335665484d8SDoug Ambrisko     struct mrsas_dcmd_frame *dcmd;
3336665484d8SDoug Ambrisko 
3337665484d8SDoug Ambrisko     if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR)
3338665484d8SDoug Ambrisko         return;
3339665484d8SDoug Ambrisko 
3340665484d8SDoug Ambrisko     cmd = mrsas_get_mfi_cmd(sc);
3341665484d8SDoug Ambrisko     if (!cmd) {
3342665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev,"Cannot allocate for shutdown cmd.\n");
3343665484d8SDoug Ambrisko         return;
3344665484d8SDoug Ambrisko     }
3345665484d8SDoug Ambrisko 
3346665484d8SDoug Ambrisko 	if (sc->aen_cmd)
3347665484d8SDoug Ambrisko         mrsas_issue_blocked_abort_cmd(sc, sc->aen_cmd);
3348665484d8SDoug Ambrisko 
3349665484d8SDoug Ambrisko 	if (sc->map_update_cmd)
3350665484d8SDoug Ambrisko         mrsas_issue_blocked_abort_cmd(sc, sc->map_update_cmd);
3351665484d8SDoug Ambrisko 
3352665484d8SDoug Ambrisko     dcmd = &cmd->frame->dcmd;
3353665484d8SDoug Ambrisko     memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE);
3354665484d8SDoug Ambrisko 
3355665484d8SDoug Ambrisko     dcmd->cmd = MFI_CMD_DCMD;
3356665484d8SDoug Ambrisko     dcmd->cmd_status = 0x0;
3357665484d8SDoug Ambrisko     dcmd->sge_count = 0;
3358665484d8SDoug Ambrisko     dcmd->flags = MFI_FRAME_DIR_NONE;
3359665484d8SDoug Ambrisko     dcmd->timeout = 0;
3360665484d8SDoug Ambrisko     dcmd->pad_0 = 0;
3361665484d8SDoug Ambrisko     dcmd->data_xfer_len = 0;
3362665484d8SDoug Ambrisko     dcmd->opcode = opcode;
3363665484d8SDoug Ambrisko 
3364665484d8SDoug Ambrisko     device_printf(sc->mrsas_dev,"Preparing to shut down controller.\n");
3365665484d8SDoug Ambrisko 
3366665484d8SDoug Ambrisko     mrsas_issue_blocked_cmd(sc, cmd);
3367665484d8SDoug Ambrisko     mrsas_release_mfi_cmd(cmd);
3368665484d8SDoug Ambrisko 
3369665484d8SDoug Ambrisko     return;
3370665484d8SDoug Ambrisko }
3371665484d8SDoug Ambrisko 
3372665484d8SDoug Ambrisko /**
3373665484d8SDoug Ambrisko  * mrsas_flush_cache:         Requests FW to flush all its caches
3374665484d8SDoug Ambrisko  * input:                     Adapter soft state
3375665484d8SDoug Ambrisko  *
3376665484d8SDoug Ambrisko  * This function is issues a DCMD internal command to Firmware to initiate
3377665484d8SDoug Ambrisko  * flushing of all caches.
3378665484d8SDoug Ambrisko  */
3379665484d8SDoug Ambrisko static void mrsas_flush_cache(struct mrsas_softc *sc)
3380665484d8SDoug Ambrisko {
3381665484d8SDoug Ambrisko     struct mrsas_mfi_cmd *cmd;
3382665484d8SDoug Ambrisko     struct mrsas_dcmd_frame *dcmd;
3383665484d8SDoug Ambrisko 
3384665484d8SDoug Ambrisko     if (sc->adprecovery == MRSAS_HW_CRITICAL_ERROR)
3385665484d8SDoug Ambrisko         return;
3386665484d8SDoug Ambrisko 
3387665484d8SDoug Ambrisko     cmd = mrsas_get_mfi_cmd(sc);
3388665484d8SDoug Ambrisko     if (!cmd) {
3389665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev,"Cannot allocate for flush cache cmd.\n");
3390665484d8SDoug Ambrisko         return;
3391665484d8SDoug Ambrisko     }
3392665484d8SDoug Ambrisko 
3393665484d8SDoug Ambrisko     dcmd = &cmd->frame->dcmd;
3394665484d8SDoug Ambrisko     memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE);
3395665484d8SDoug Ambrisko 
3396665484d8SDoug Ambrisko     dcmd->cmd = MFI_CMD_DCMD;
3397665484d8SDoug Ambrisko     dcmd->cmd_status = 0x0;
3398665484d8SDoug Ambrisko     dcmd->sge_count = 0;
3399665484d8SDoug Ambrisko     dcmd->flags = MFI_FRAME_DIR_NONE;
3400665484d8SDoug Ambrisko     dcmd->timeout = 0;
3401665484d8SDoug Ambrisko     dcmd->pad_0 = 0;
3402665484d8SDoug Ambrisko     dcmd->data_xfer_len = 0;
3403665484d8SDoug Ambrisko     dcmd->opcode = MR_DCMD_CTRL_CACHE_FLUSH;
3404665484d8SDoug Ambrisko     dcmd->mbox.b[0] = MR_FLUSH_CTRL_CACHE | MR_FLUSH_DISK_CACHE;
3405665484d8SDoug Ambrisko 
3406665484d8SDoug Ambrisko     mrsas_issue_blocked_cmd(sc, cmd);
3407665484d8SDoug Ambrisko     mrsas_release_mfi_cmd(cmd);
3408665484d8SDoug Ambrisko 
3409665484d8SDoug Ambrisko     return;
3410665484d8SDoug Ambrisko }
3411665484d8SDoug Ambrisko 
3412665484d8SDoug Ambrisko /**
3413665484d8SDoug Ambrisko  * mrsas_get_map_info:        Load and validate RAID map
3414665484d8SDoug Ambrisko  * input:                     Adapter instance soft state
3415665484d8SDoug Ambrisko  *
3416665484d8SDoug Ambrisko  * This function calls mrsas_get_ld_map_info() and MR_ValidateMapInfo()
3417665484d8SDoug Ambrisko  * to load and validate RAID map.  It returns 0 if successful, 1 other-
3418665484d8SDoug Ambrisko  * wise.
3419665484d8SDoug Ambrisko  */
3420665484d8SDoug Ambrisko static int mrsas_get_map_info(struct mrsas_softc *sc)
3421665484d8SDoug Ambrisko {
3422665484d8SDoug Ambrisko    uint8_t  retcode = 0;
3423665484d8SDoug Ambrisko 
3424665484d8SDoug Ambrisko     sc->fast_path_io = 0;
3425665484d8SDoug Ambrisko     if (!mrsas_get_ld_map_info(sc)) {
3426665484d8SDoug Ambrisko         retcode = MR_ValidateMapInfo(sc);
3427665484d8SDoug Ambrisko         if (retcode == 0) {
3428665484d8SDoug Ambrisko             sc->fast_path_io = 1;
3429665484d8SDoug Ambrisko             return 0;
3430665484d8SDoug Ambrisko         }
3431665484d8SDoug Ambrisko     }
3432665484d8SDoug Ambrisko     return 1;
3433665484d8SDoug Ambrisko }
3434665484d8SDoug Ambrisko 
3435665484d8SDoug Ambrisko /**
3436665484d8SDoug Ambrisko  * mrsas_get_ld_map_info:      Get FW's ld_map structure
3437665484d8SDoug Ambrisko  * input:                      Adapter instance soft state
3438665484d8SDoug Ambrisko  *
3439665484d8SDoug Ambrisko  * Issues an internal command (DCMD) to get the FW's controller PD
3440665484d8SDoug Ambrisko  * list structure.
3441665484d8SDoug Ambrisko  */
3442665484d8SDoug Ambrisko static int mrsas_get_ld_map_info(struct mrsas_softc *sc)
3443665484d8SDoug Ambrisko {
3444665484d8SDoug Ambrisko     int retcode = 0;
3445665484d8SDoug Ambrisko     struct mrsas_mfi_cmd *cmd;
3446665484d8SDoug Ambrisko     struct mrsas_dcmd_frame *dcmd;
34474799d485SKashyap D Desai     void *map;
3448665484d8SDoug Ambrisko     bus_addr_t map_phys_addr = 0;
3449665484d8SDoug Ambrisko 
3450665484d8SDoug Ambrisko     cmd = mrsas_get_mfi_cmd(sc);
3451665484d8SDoug Ambrisko     if (!cmd) {
34524799d485SKashyap D Desai         device_printf(sc->mrsas_dev,
34534799d485SKashyap D Desai 			"Cannot alloc for ld map info cmd.\n");
3454665484d8SDoug Ambrisko         return 1;
3455665484d8SDoug Ambrisko     }
3456665484d8SDoug Ambrisko 
3457665484d8SDoug Ambrisko     dcmd = &cmd->frame->dcmd;
3458665484d8SDoug Ambrisko 
34594799d485SKashyap D Desai     map = (void *)sc->raidmap_mem[(sc->map_id & 1)];
3460665484d8SDoug Ambrisko     map_phys_addr = sc->raidmap_phys_addr[(sc->map_id & 1)];
3461665484d8SDoug Ambrisko     if (!map) {
34624799d485SKashyap D Desai         device_printf(sc->mrsas_dev,
34634799d485SKashyap D Desai 			"Failed to alloc mem for ld map info.\n");
3464665484d8SDoug Ambrisko         mrsas_release_mfi_cmd(cmd);
3465665484d8SDoug Ambrisko         return (ENOMEM);
3466665484d8SDoug Ambrisko     }
34674799d485SKashyap D Desai     memset(map, 0, sizeof(sc->max_map_sz));
3468665484d8SDoug Ambrisko     memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE);
3469665484d8SDoug Ambrisko 
3470665484d8SDoug Ambrisko     dcmd->cmd = MFI_CMD_DCMD;
3471665484d8SDoug Ambrisko     dcmd->cmd_status = 0xFF;
3472665484d8SDoug Ambrisko     dcmd->sge_count = 1;
3473665484d8SDoug Ambrisko     dcmd->flags = MFI_FRAME_DIR_READ;
3474665484d8SDoug Ambrisko     dcmd->timeout = 0;
3475665484d8SDoug Ambrisko     dcmd->pad_0 = 0;
34764799d485SKashyap D Desai     dcmd->data_xfer_len = sc->current_map_sz;
3477665484d8SDoug Ambrisko     dcmd->opcode = MR_DCMD_LD_MAP_GET_INFO;
3478665484d8SDoug Ambrisko     dcmd->sgl.sge32[0].phys_addr = map_phys_addr;
34794799d485SKashyap D Desai     dcmd->sgl.sge32[0].length = sc->current_map_sz;
34804799d485SKashyap D Desai 
3481665484d8SDoug Ambrisko     if (!mrsas_issue_polled(sc, cmd))
3482665484d8SDoug Ambrisko         retcode = 0;
3483665484d8SDoug Ambrisko     else
3484665484d8SDoug Ambrisko     {
34854799d485SKashyap D Desai         device_printf(sc->mrsas_dev,
34864799d485SKashyap D Desai 			"Fail to send get LD map info cmd.\n");
3487665484d8SDoug Ambrisko         retcode = 1;
3488665484d8SDoug Ambrisko     }
3489665484d8SDoug Ambrisko     mrsas_release_mfi_cmd(cmd);
34904799d485SKashyap D Desai 
3491665484d8SDoug Ambrisko     return(retcode);
3492665484d8SDoug Ambrisko }
3493665484d8SDoug Ambrisko 
3494665484d8SDoug Ambrisko /**
3495665484d8SDoug Ambrisko  * mrsas_sync_map_info:        Get FW's ld_map structure
3496665484d8SDoug Ambrisko  * input:                      Adapter instance soft state
3497665484d8SDoug Ambrisko  *
3498665484d8SDoug Ambrisko  * Issues an internal command (DCMD) to get the FW's controller PD
3499665484d8SDoug Ambrisko  * list structure.
3500665484d8SDoug Ambrisko  */
3501665484d8SDoug Ambrisko static int mrsas_sync_map_info(struct mrsas_softc *sc)
3502665484d8SDoug Ambrisko {
3503665484d8SDoug Ambrisko     int retcode = 0, i;
3504665484d8SDoug Ambrisko     struct mrsas_mfi_cmd *cmd;
3505665484d8SDoug Ambrisko     struct mrsas_dcmd_frame *dcmd;
3506665484d8SDoug Ambrisko     uint32_t size_sync_info, num_lds;
3507665484d8SDoug Ambrisko     MR_LD_TARGET_SYNC *target_map = NULL;
35084799d485SKashyap D Desai     MR_DRV_RAID_MAP_ALL *map;
3509665484d8SDoug Ambrisko     MR_LD_RAID  *raid;
3510665484d8SDoug Ambrisko     MR_LD_TARGET_SYNC *ld_sync;
3511665484d8SDoug Ambrisko     bus_addr_t map_phys_addr = 0;
3512665484d8SDoug Ambrisko 
3513665484d8SDoug Ambrisko     cmd = mrsas_get_mfi_cmd(sc);
3514665484d8SDoug Ambrisko     if (!cmd) {
35154799d485SKashyap D Desai         device_printf(sc->mrsas_dev,
35164799d485SKashyap D Desai 			"Cannot alloc for sync map info cmd\n");
3517665484d8SDoug Ambrisko         return 1;
3518665484d8SDoug Ambrisko     }
3519665484d8SDoug Ambrisko 
35204799d485SKashyap D Desai     map = sc->ld_drv_map[sc->map_id & 1];
3521665484d8SDoug Ambrisko     num_lds = map->raidMap.ldCount;
3522665484d8SDoug Ambrisko 
3523665484d8SDoug Ambrisko     dcmd = &cmd->frame->dcmd;
3524665484d8SDoug Ambrisko     size_sync_info = sizeof(MR_LD_TARGET_SYNC) * num_lds;
3525665484d8SDoug Ambrisko     memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE);
3526665484d8SDoug Ambrisko 
35274799d485SKashyap D Desai     target_map =
35284799d485SKashyap D Desai 	    (MR_LD_TARGET_SYNC *)sc->raidmap_mem[(sc->map_id - 1) & 1];
35294799d485SKashyap D Desai     memset(target_map, 0, sc->max_map_sz);
3530665484d8SDoug Ambrisko 
3531665484d8SDoug Ambrisko     map_phys_addr = sc->raidmap_phys_addr[(sc->map_id - 1) & 1];
3532665484d8SDoug Ambrisko 
3533665484d8SDoug Ambrisko     ld_sync = (MR_LD_TARGET_SYNC *)target_map;
3534665484d8SDoug Ambrisko 
3535665484d8SDoug Ambrisko     for (i = 0; i < num_lds; i++, ld_sync++) {
3536665484d8SDoug Ambrisko         raid = MR_LdRaidGet(i, map);
3537665484d8SDoug Ambrisko         ld_sync->targetId = MR_GetLDTgtId(i, map);
3538665484d8SDoug Ambrisko         ld_sync->seqNum = raid->seqNum;
3539665484d8SDoug Ambrisko     }
3540665484d8SDoug Ambrisko 
3541665484d8SDoug Ambrisko     dcmd->cmd = MFI_CMD_DCMD;
3542665484d8SDoug Ambrisko     dcmd->cmd_status = 0xFF;
3543665484d8SDoug Ambrisko     dcmd->sge_count = 1;
3544665484d8SDoug Ambrisko     dcmd->flags = MFI_FRAME_DIR_WRITE;
3545665484d8SDoug Ambrisko     dcmd->timeout = 0;
3546665484d8SDoug Ambrisko     dcmd->pad_0 = 0;
35474799d485SKashyap D Desai     dcmd->data_xfer_len = sc->current_map_sz;
3548665484d8SDoug Ambrisko     dcmd->mbox.b[0] = num_lds;
3549665484d8SDoug Ambrisko     dcmd->mbox.b[1] = MRSAS_DCMD_MBOX_PEND_FLAG;
3550665484d8SDoug Ambrisko     dcmd->opcode = MR_DCMD_LD_MAP_GET_INFO;
3551665484d8SDoug Ambrisko     dcmd->sgl.sge32[0].phys_addr = map_phys_addr;
35524799d485SKashyap D Desai     dcmd->sgl.sge32[0].length = sc->current_map_sz;
3553665484d8SDoug Ambrisko 
3554665484d8SDoug Ambrisko     sc->map_update_cmd = cmd;
3555665484d8SDoug Ambrisko     if (mrsas_issue_dcmd(sc, cmd)) {
35564799d485SKashyap D Desai         device_printf(sc->mrsas_dev,
35574799d485SKashyap D Desai 			"Fail to send sync map info command.\n");
3558665484d8SDoug Ambrisko         return(1);
3559665484d8SDoug Ambrisko     }
3560665484d8SDoug Ambrisko     return(retcode);
3561665484d8SDoug Ambrisko }
3562665484d8SDoug Ambrisko 
3563665484d8SDoug Ambrisko /**
3564665484d8SDoug Ambrisko  * mrsas_get_pd_list:           Returns FW's PD list structure
3565665484d8SDoug Ambrisko  * input:                       Adapter soft state
3566665484d8SDoug Ambrisko  *
3567665484d8SDoug Ambrisko  * Issues an internal command (DCMD) to get the FW's controller PD
3568665484d8SDoug Ambrisko  * list structure.  This information is mainly used to find out about
3569665484d8SDoug Ambrisko  * system supported by Firmware.
3570665484d8SDoug Ambrisko  */
3571665484d8SDoug Ambrisko static int mrsas_get_pd_list(struct mrsas_softc *sc)
3572665484d8SDoug Ambrisko {
3573665484d8SDoug Ambrisko     int retcode = 0, pd_index = 0, pd_count=0, pd_list_size;
3574665484d8SDoug Ambrisko     struct mrsas_mfi_cmd *cmd;
3575665484d8SDoug Ambrisko     struct mrsas_dcmd_frame *dcmd;
3576665484d8SDoug Ambrisko     struct MR_PD_LIST *pd_list_mem;
3577665484d8SDoug Ambrisko     struct MR_PD_ADDRESS *pd_addr;
3578665484d8SDoug Ambrisko     bus_addr_t pd_list_phys_addr = 0;
3579665484d8SDoug Ambrisko     struct mrsas_tmp_dcmd *tcmd;
3580665484d8SDoug Ambrisko 
3581665484d8SDoug Ambrisko     cmd = mrsas_get_mfi_cmd(sc);
3582665484d8SDoug Ambrisko     if (!cmd) {
35834799d485SKashyap D Desai         device_printf(sc->mrsas_dev,
35844799d485SKashyap D Desai 			"Cannot alloc for get PD list cmd\n");
3585665484d8SDoug Ambrisko         return 1;
3586665484d8SDoug Ambrisko     }
3587665484d8SDoug Ambrisko 
3588665484d8SDoug Ambrisko     dcmd = &cmd->frame->dcmd;
3589665484d8SDoug Ambrisko 
3590665484d8SDoug Ambrisko     tcmd = malloc(sizeof(struct mrsas_tmp_dcmd), M_MRSAS, M_NOWAIT);
3591665484d8SDoug Ambrisko     pd_list_size = MRSAS_MAX_PD * sizeof(struct MR_PD_LIST);
3592665484d8SDoug Ambrisko     if (mrsas_alloc_tmp_dcmd(sc, tcmd, pd_list_size) != SUCCESS) {
35934799d485SKashyap D Desai         device_printf(sc->mrsas_dev,
35944799d485SKashyap D Desai 			"Cannot alloc dmamap for get PD list cmd\n");
3595665484d8SDoug Ambrisko         mrsas_release_mfi_cmd(cmd);
3596665484d8SDoug Ambrisko         return(ENOMEM);
3597665484d8SDoug Ambrisko     }
3598665484d8SDoug Ambrisko     else {
3599665484d8SDoug Ambrisko         pd_list_mem = tcmd->tmp_dcmd_mem;
3600665484d8SDoug Ambrisko         pd_list_phys_addr = tcmd->tmp_dcmd_phys_addr;
3601665484d8SDoug Ambrisko     }
3602665484d8SDoug Ambrisko     memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE);
3603665484d8SDoug Ambrisko 
3604665484d8SDoug Ambrisko     dcmd->mbox.b[0] = MR_PD_QUERY_TYPE_EXPOSED_TO_HOST;
3605665484d8SDoug Ambrisko     dcmd->mbox.b[1] = 0;
3606665484d8SDoug Ambrisko     dcmd->cmd = MFI_CMD_DCMD;
3607665484d8SDoug Ambrisko     dcmd->cmd_status = 0xFF;
3608665484d8SDoug Ambrisko     dcmd->sge_count = 1;
3609665484d8SDoug Ambrisko     dcmd->flags = MFI_FRAME_DIR_READ;
3610665484d8SDoug Ambrisko     dcmd->timeout = 0;
3611665484d8SDoug Ambrisko     dcmd->pad_0 = 0;
3612665484d8SDoug Ambrisko     dcmd->data_xfer_len = MRSAS_MAX_PD * sizeof(struct MR_PD_LIST);
3613665484d8SDoug Ambrisko     dcmd->opcode = MR_DCMD_PD_LIST_QUERY;
3614665484d8SDoug Ambrisko     dcmd->sgl.sge32[0].phys_addr = pd_list_phys_addr;
3615665484d8SDoug Ambrisko     dcmd->sgl.sge32[0].length = MRSAS_MAX_PD * sizeof(struct MR_PD_LIST);
3616665484d8SDoug Ambrisko 
3617665484d8SDoug Ambrisko     if (!mrsas_issue_polled(sc, cmd))
3618665484d8SDoug Ambrisko         retcode = 0;
3619665484d8SDoug Ambrisko     else
3620665484d8SDoug Ambrisko         retcode = 1;
3621665484d8SDoug Ambrisko 
3622665484d8SDoug Ambrisko     /* Get the instance PD list */
3623665484d8SDoug Ambrisko     pd_count = MRSAS_MAX_PD;
3624665484d8SDoug Ambrisko     pd_addr = pd_list_mem->addr;
3625665484d8SDoug Ambrisko     if (retcode == 0 && pd_list_mem->count < pd_count) {
36264799d485SKashyap D Desai         memset(sc->local_pd_list, 0,
36274799d485SKashyap D Desai 			MRSAS_MAX_PD * sizeof(struct mrsas_pd_list));
3628665484d8SDoug Ambrisko         for (pd_index = 0; pd_index < pd_list_mem->count; pd_index++) {
3629665484d8SDoug Ambrisko             sc->local_pd_list[pd_addr->deviceId].tid = pd_addr->deviceId;
36304799d485SKashyap D Desai             sc->local_pd_list[pd_addr->deviceId].driveType =
36314799d485SKashyap D Desai 		    pd_addr->scsiDevType;
36324799d485SKashyap D Desai             sc->local_pd_list[pd_addr->deviceId].driveState =
36334799d485SKashyap D Desai 		    MR_PD_STATE_SYSTEM;
3634665484d8SDoug Ambrisko             pd_addr++;
3635665484d8SDoug Ambrisko         }
3636665484d8SDoug Ambrisko     }
3637665484d8SDoug Ambrisko 
3638665484d8SDoug Ambrisko     /* Use mutext/spinlock if pd_list component size increase more than 32 bit. */
3639665484d8SDoug Ambrisko     memcpy(sc->pd_list, sc->local_pd_list, sizeof(sc->local_pd_list));
3640665484d8SDoug Ambrisko     mrsas_free_tmp_dcmd(tcmd);
3641665484d8SDoug Ambrisko     mrsas_release_mfi_cmd(cmd);
3642665484d8SDoug Ambrisko     free(tcmd, M_MRSAS);
3643665484d8SDoug Ambrisko     return(retcode);
3644665484d8SDoug Ambrisko }
3645665484d8SDoug Ambrisko 
3646665484d8SDoug Ambrisko /**
3647665484d8SDoug Ambrisko  * mrsas_get_ld_list:           Returns FW's LD list structure
3648665484d8SDoug Ambrisko  * input:                       Adapter soft state
3649665484d8SDoug Ambrisko  *
3650665484d8SDoug Ambrisko  * Issues an internal command (DCMD) to get the FW's controller PD
3651665484d8SDoug Ambrisko  * list structure.  This information is mainly used to find out about
3652665484d8SDoug Ambrisko  * supported by the FW.
3653665484d8SDoug Ambrisko  */
3654665484d8SDoug Ambrisko static int mrsas_get_ld_list(struct mrsas_softc *sc)
3655665484d8SDoug Ambrisko {
3656665484d8SDoug Ambrisko     int ld_list_size, retcode = 0, ld_index = 0, ids = 0;
3657665484d8SDoug Ambrisko     struct mrsas_mfi_cmd *cmd;
3658665484d8SDoug Ambrisko     struct mrsas_dcmd_frame *dcmd;
3659665484d8SDoug Ambrisko     struct MR_LD_LIST *ld_list_mem;
3660665484d8SDoug Ambrisko     bus_addr_t ld_list_phys_addr = 0;
3661665484d8SDoug Ambrisko     struct mrsas_tmp_dcmd *tcmd;
3662665484d8SDoug Ambrisko 
3663665484d8SDoug Ambrisko     cmd = mrsas_get_mfi_cmd(sc);
3664665484d8SDoug Ambrisko     if (!cmd) {
36654799d485SKashyap D Desai         device_printf(sc->mrsas_dev,
36664799d485SKashyap D Desai 			"Cannot alloc for get LD list cmd\n");
3667665484d8SDoug Ambrisko         return 1;
3668665484d8SDoug Ambrisko     }
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     ld_list_size = sizeof(struct MR_LD_LIST);
3674665484d8SDoug Ambrisko     if (mrsas_alloc_tmp_dcmd(sc, tcmd, ld_list_size) != SUCCESS) {
36754799d485SKashyap D Desai         device_printf(sc->mrsas_dev,
36764799d485SKashyap D Desai 			"Cannot alloc dmamap for get LD list cmd\n");
3677665484d8SDoug Ambrisko         mrsas_release_mfi_cmd(cmd);
3678665484d8SDoug Ambrisko         return(ENOMEM);
3679665484d8SDoug Ambrisko     }
3680665484d8SDoug Ambrisko     else {
3681665484d8SDoug Ambrisko         ld_list_mem = tcmd->tmp_dcmd_mem;
3682665484d8SDoug Ambrisko         ld_list_phys_addr = tcmd->tmp_dcmd_phys_addr;
3683665484d8SDoug Ambrisko     }
3684665484d8SDoug Ambrisko     memset(dcmd->mbox.b, 0, MFI_MBOX_SIZE);
3685665484d8SDoug Ambrisko 
36864799d485SKashyap D Desai     if (sc->max256vdSupport)
36874799d485SKashyap D Desai         dcmd->mbox.b[0]=1;
36884799d485SKashyap D Desai 
3689665484d8SDoug Ambrisko     dcmd->cmd = MFI_CMD_DCMD;
3690665484d8SDoug Ambrisko     dcmd->cmd_status = 0xFF;
3691665484d8SDoug Ambrisko     dcmd->sge_count = 1;
3692665484d8SDoug Ambrisko     dcmd->flags = MFI_FRAME_DIR_READ;
3693665484d8SDoug Ambrisko     dcmd->timeout = 0;
3694665484d8SDoug Ambrisko     dcmd->data_xfer_len = sizeof(struct MR_LD_LIST);
3695665484d8SDoug Ambrisko     dcmd->opcode = MR_DCMD_LD_GET_LIST;
3696665484d8SDoug Ambrisko     dcmd->sgl.sge32[0].phys_addr = ld_list_phys_addr;
3697665484d8SDoug Ambrisko     dcmd->sgl.sge32[0].length = sizeof(struct MR_LD_LIST);
3698665484d8SDoug Ambrisko     dcmd->pad_0  = 0;
3699665484d8SDoug Ambrisko 
3700665484d8SDoug Ambrisko     if (!mrsas_issue_polled(sc, cmd))
3701665484d8SDoug Ambrisko         retcode = 0;
3702665484d8SDoug Ambrisko     else
3703665484d8SDoug Ambrisko         retcode = 1;
3704665484d8SDoug Ambrisko 
37054799d485SKashyap D Desai #if VD_EXT_DEBUG
37064799d485SKashyap D Desai     printf ("Number of LDs %d\n", ld_list_mem->ldCount);
37074799d485SKashyap D Desai #endif
37084799d485SKashyap D Desai 
3709665484d8SDoug Ambrisko      /* Get the instance LD list */
37104799d485SKashyap D Desai      if ((retcode == 0) &&
37114799d485SKashyap D Desai 		     (ld_list_mem->ldCount <= sc->fw_supported_vd_count)){
3712665484d8SDoug Ambrisko         sc->CurLdCount = ld_list_mem->ldCount;
37134799d485SKashyap D Desai         memset(sc->ld_ids, 0xff, MAX_LOGICAL_DRIVES_EXT);
3714665484d8SDoug Ambrisko         for (ld_index = 0; ld_index < ld_list_mem->ldCount; ld_index++) {
3715665484d8SDoug Ambrisko             if (ld_list_mem->ldList[ld_index].state != 0) {
3716665484d8SDoug Ambrisko                 ids = ld_list_mem->ldList[ld_index].ref.ld_context.targetId;
3717665484d8SDoug Ambrisko                 sc->ld_ids[ids] = ld_list_mem->ldList[ld_index].ref.ld_context.targetId;
3718665484d8SDoug Ambrisko             }
3719665484d8SDoug Ambrisko         }
3720665484d8SDoug Ambrisko     }
3721665484d8SDoug Ambrisko 
3722665484d8SDoug Ambrisko     mrsas_free_tmp_dcmd(tcmd);
3723665484d8SDoug Ambrisko     mrsas_release_mfi_cmd(cmd);
3724665484d8SDoug Ambrisko     free(tcmd, M_MRSAS);
3725665484d8SDoug Ambrisko     return(retcode);
3726665484d8SDoug Ambrisko }
3727665484d8SDoug Ambrisko 
3728665484d8SDoug Ambrisko /**
3729665484d8SDoug Ambrisko  * mrsas_alloc_tmp_dcmd:       Allocates memory for temporary command
3730665484d8SDoug Ambrisko  * input:                      Adapter soft state
3731665484d8SDoug Ambrisko  *                             Temp command
3732665484d8SDoug Ambrisko  *                             Size of alloction
3733665484d8SDoug Ambrisko  *
3734665484d8SDoug Ambrisko  * Allocates DMAable memory for a temporary internal command. The allocated
3735665484d8SDoug Ambrisko  * memory is initialized to all zeros upon successful loading of the dma
3736665484d8SDoug Ambrisko  * mapped memory.
3737665484d8SDoug Ambrisko  */
3738665484d8SDoug Ambrisko int mrsas_alloc_tmp_dcmd(struct mrsas_softc *sc, struct mrsas_tmp_dcmd *tcmd,
3739665484d8SDoug Ambrisko           int size)
3740665484d8SDoug Ambrisko {
3741665484d8SDoug Ambrisko     if (bus_dma_tag_create( sc->mrsas_parent_tag,   // parent
3742665484d8SDoug Ambrisko                             1, 0,                   // algnmnt, boundary
3743665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR_32BIT,// lowaddr
3744665484d8SDoug Ambrisko                             BUS_SPACE_MAXADDR,      // highaddr
3745665484d8SDoug Ambrisko                             NULL, NULL,             // filter, filterarg
3746665484d8SDoug Ambrisko                             size,                   // maxsize
3747665484d8SDoug Ambrisko                             1,                      // msegments
3748665484d8SDoug Ambrisko                             size,                   // maxsegsize
3749665484d8SDoug Ambrisko                             BUS_DMA_ALLOCNOW,       // flags
3750665484d8SDoug Ambrisko                             NULL, NULL,             // lockfunc, lockarg
3751665484d8SDoug Ambrisko                             &tcmd->tmp_dcmd_tag)) {
3752665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot allocate tmp dcmd tag\n");
3753665484d8SDoug Ambrisko         return (ENOMEM);
3754665484d8SDoug Ambrisko     }
3755665484d8SDoug Ambrisko     if (bus_dmamem_alloc(tcmd->tmp_dcmd_tag, (void **)&tcmd->tmp_dcmd_mem,
3756665484d8SDoug Ambrisko             BUS_DMA_NOWAIT, &tcmd->tmp_dcmd_dmamap)) {
3757665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot allocate tmp dcmd mem\n");
3758665484d8SDoug Ambrisko         return (ENOMEM);
3759665484d8SDoug Ambrisko     }
3760665484d8SDoug Ambrisko     if (bus_dmamap_load(tcmd->tmp_dcmd_tag, tcmd->tmp_dcmd_dmamap,
3761665484d8SDoug Ambrisko             tcmd->tmp_dcmd_mem, size, mrsas_addr_cb,
3762665484d8SDoug Ambrisko             &tcmd->tmp_dcmd_phys_addr, BUS_DMA_NOWAIT)) {
3763665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot load tmp dcmd mem\n");
3764665484d8SDoug Ambrisko         return (ENOMEM);
3765665484d8SDoug Ambrisko     }
3766665484d8SDoug Ambrisko 
3767665484d8SDoug Ambrisko     memset(tcmd->tmp_dcmd_mem, 0, size);
3768665484d8SDoug Ambrisko     return (0);
3769665484d8SDoug Ambrisko }
3770665484d8SDoug Ambrisko 
3771665484d8SDoug Ambrisko /**
3772665484d8SDoug Ambrisko  * mrsas_free_tmp_dcmd:      Free memory for temporary command
3773665484d8SDoug Ambrisko  * input:                    temporary dcmd pointer
3774665484d8SDoug Ambrisko  *
3775665484d8SDoug Ambrisko  * Deallocates memory of the temporary command for use in the construction
3776665484d8SDoug Ambrisko  * of the internal DCMD.
3777665484d8SDoug Ambrisko  */
3778665484d8SDoug Ambrisko void mrsas_free_tmp_dcmd(struct mrsas_tmp_dcmd *tmp)
3779665484d8SDoug Ambrisko {
3780665484d8SDoug Ambrisko     if (tmp->tmp_dcmd_phys_addr)
3781665484d8SDoug Ambrisko         bus_dmamap_unload(tmp->tmp_dcmd_tag, tmp->tmp_dcmd_dmamap);
3782665484d8SDoug Ambrisko     if (tmp->tmp_dcmd_mem != NULL)
3783665484d8SDoug Ambrisko         bus_dmamem_free(tmp->tmp_dcmd_tag, tmp->tmp_dcmd_mem, tmp->tmp_dcmd_dmamap);
3784665484d8SDoug Ambrisko     if (tmp->tmp_dcmd_tag != NULL)
3785665484d8SDoug Ambrisko         bus_dma_tag_destroy(tmp->tmp_dcmd_tag);
3786665484d8SDoug Ambrisko }
3787665484d8SDoug Ambrisko 
3788665484d8SDoug Ambrisko /**
3789665484d8SDoug Ambrisko  * mrsas_issue_blocked_abort_cmd:       Aborts previously issued cmd
3790665484d8SDoug Ambrisko  * input:                               Adapter soft state
3791665484d8SDoug Ambrisko  *                                      Previously issued cmd to be aborted
3792665484d8SDoug Ambrisko  *
3793665484d8SDoug Ambrisko  * This function is used to abort previously issued commands, such as AEN and
3794665484d8SDoug Ambrisko  * RAID map sync map commands.  The abort command is sent as a DCMD internal
3795665484d8SDoug Ambrisko  * command and subsequently the driver will wait for a return status.  The
3796665484d8SDoug Ambrisko  * max wait time is MRSAS_INTERNAL_CMD_WAIT_TIME seconds.
3797665484d8SDoug Ambrisko  */
3798665484d8SDoug Ambrisko static int mrsas_issue_blocked_abort_cmd(struct mrsas_softc *sc,
3799665484d8SDoug Ambrisko                                          struct mrsas_mfi_cmd *cmd_to_abort)
3800665484d8SDoug Ambrisko {
3801665484d8SDoug Ambrisko     struct mrsas_mfi_cmd *cmd;
3802665484d8SDoug Ambrisko     struct mrsas_abort_frame *abort_fr;
3803665484d8SDoug Ambrisko     u_int8_t retcode = 0;
3804665484d8SDoug Ambrisko     unsigned long total_time = 0;
3805665484d8SDoug Ambrisko     u_int8_t max_wait = MRSAS_INTERNAL_CMD_WAIT_TIME;
3806665484d8SDoug Ambrisko 
3807665484d8SDoug Ambrisko     cmd = mrsas_get_mfi_cmd(sc);
3808665484d8SDoug Ambrisko     if (!cmd) {
3809665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Cannot alloc for abort cmd\n");
3810665484d8SDoug Ambrisko         return(1);
3811665484d8SDoug Ambrisko     }
3812665484d8SDoug Ambrisko 
3813665484d8SDoug Ambrisko     abort_fr = &cmd->frame->abort;
3814665484d8SDoug Ambrisko 
3815665484d8SDoug Ambrisko     /* Prepare and issue the abort frame */
3816665484d8SDoug Ambrisko     abort_fr->cmd = MFI_CMD_ABORT;
3817665484d8SDoug Ambrisko     abort_fr->cmd_status = 0xFF;
3818665484d8SDoug Ambrisko     abort_fr->flags = 0;
3819665484d8SDoug Ambrisko     abort_fr->abort_context = cmd_to_abort->index;
3820665484d8SDoug Ambrisko     abort_fr->abort_mfi_phys_addr_lo = cmd_to_abort->frame_phys_addr;
3821665484d8SDoug Ambrisko     abort_fr->abort_mfi_phys_addr_hi = 0;
3822665484d8SDoug Ambrisko 
3823665484d8SDoug Ambrisko     cmd->sync_cmd = 1;
3824665484d8SDoug Ambrisko     cmd->cmd_status = 0xFF;
3825665484d8SDoug Ambrisko 
3826665484d8SDoug Ambrisko     if (mrsas_issue_dcmd(sc, cmd)) {
3827665484d8SDoug Ambrisko         device_printf(sc->mrsas_dev, "Fail to send abort command.\n");
3828665484d8SDoug Ambrisko         return(1);
3829665484d8SDoug Ambrisko     }
3830665484d8SDoug Ambrisko 
3831665484d8SDoug Ambrisko     /* Wait for this cmd to complete */
3832665484d8SDoug Ambrisko     sc->chan = (void*)&cmd;
3833665484d8SDoug Ambrisko     while (1) {
3834665484d8SDoug Ambrisko        if (cmd->cmd_status == 0xFF){
3835665484d8SDoug Ambrisko            tsleep((void *)&sc->chan, 0, "mrsas_sleep", hz);
3836665484d8SDoug Ambrisko        }
3837665484d8SDoug Ambrisko        else
3838665484d8SDoug Ambrisko            break;
3839665484d8SDoug Ambrisko        total_time++;
3840665484d8SDoug Ambrisko        if (total_time >= max_wait) {
3841665484d8SDoug Ambrisko            device_printf(sc->mrsas_dev, "Abort cmd timed out after %d sec.\n", max_wait);
3842665484d8SDoug Ambrisko            retcode = 1;
3843665484d8SDoug Ambrisko            break;
3844665484d8SDoug Ambrisko        }
3845665484d8SDoug Ambrisko     }
3846665484d8SDoug Ambrisko 
3847665484d8SDoug Ambrisko     cmd->sync_cmd = 0;
3848665484d8SDoug Ambrisko     mrsas_release_mfi_cmd(cmd);
3849665484d8SDoug Ambrisko     return(retcode);
3850665484d8SDoug Ambrisko }
3851665484d8SDoug Ambrisko 
3852665484d8SDoug Ambrisko /**
3853665484d8SDoug Ambrisko  * mrsas_complete_abort:      Completes aborting a command
3854665484d8SDoug Ambrisko  * input:                     Adapter soft state
3855665484d8SDoug Ambrisko  *                            Cmd that was issued to abort another cmd
3856665484d8SDoug Ambrisko  *
3857665484d8SDoug Ambrisko  * The mrsas_issue_blocked_abort_cmd() function waits for the command status
3858665484d8SDoug Ambrisko  * to change after sending the command.  This function is called from
3859665484d8SDoug Ambrisko  * mrsas_complete_mptmfi_passthru() to wake up the sleep thread associated.
3860665484d8SDoug Ambrisko  */
3861665484d8SDoug Ambrisko void mrsas_complete_abort(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd)
3862665484d8SDoug Ambrisko {
3863665484d8SDoug Ambrisko     if (cmd->sync_cmd) {
3864665484d8SDoug Ambrisko         cmd->sync_cmd = 0;
3865665484d8SDoug Ambrisko         cmd->cmd_status = 0;
3866665484d8SDoug Ambrisko         sc->chan = (void*)&cmd;
3867665484d8SDoug Ambrisko         wakeup_one((void *)&sc->chan);
3868665484d8SDoug Ambrisko     }
3869665484d8SDoug Ambrisko     return;
3870665484d8SDoug Ambrisko }
3871665484d8SDoug Ambrisko 
3872665484d8SDoug Ambrisko /**
3873665484d8SDoug Ambrisko  * mrsas_aen_handler:		Callback function for AEN processing from thread context.
3874665484d8SDoug Ambrisko  * input:					Adapter soft state
3875665484d8SDoug Ambrisko  *
3876665484d8SDoug Ambrisko  */
3877665484d8SDoug Ambrisko void mrsas_aen_handler(struct mrsas_softc *sc)
3878665484d8SDoug Ambrisko {
3879665484d8SDoug Ambrisko 	union mrsas_evt_class_locale class_locale;
3880665484d8SDoug Ambrisko 	int     doscan = 0;
3881665484d8SDoug Ambrisko 	u_int32_t seq_num;
3882665484d8SDoug Ambrisko 	int error;
3883665484d8SDoug Ambrisko 
3884665484d8SDoug Ambrisko 	if (!sc) {
3885665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "invalid instance!\n");
3886665484d8SDoug Ambrisko 		return;
3887665484d8SDoug Ambrisko 	}
3888665484d8SDoug Ambrisko 
3889665484d8SDoug Ambrisko 	if (sc->evt_detail_mem) {
3890665484d8SDoug Ambrisko 		switch (sc->evt_detail_mem->code) {
3891665484d8SDoug Ambrisko 			case MR_EVT_PD_INSERTED:
3892665484d8SDoug Ambrisko 				mrsas_get_pd_list(sc);
3893665484d8SDoug Ambrisko 				mrsas_bus_scan_sim(sc, sc->sim_1);
3894665484d8SDoug Ambrisko 				doscan = 0;
3895665484d8SDoug Ambrisko 				break;
3896665484d8SDoug Ambrisko 			case MR_EVT_PD_REMOVED:
3897665484d8SDoug Ambrisko 				mrsas_get_pd_list(sc);
3898665484d8SDoug Ambrisko 				mrsas_bus_scan_sim(sc, sc->sim_1);
3899665484d8SDoug Ambrisko 				doscan = 0;
3900665484d8SDoug Ambrisko 				break;
3901665484d8SDoug Ambrisko 			case MR_EVT_LD_OFFLINE:
3902665484d8SDoug Ambrisko 			case MR_EVT_CFG_CLEARED:
3903665484d8SDoug Ambrisko 			case MR_EVT_LD_DELETED:
3904665484d8SDoug Ambrisko 				mrsas_bus_scan_sim(sc, sc->sim_0);
3905665484d8SDoug Ambrisko 				doscan = 0;
3906665484d8SDoug Ambrisko 				break;
3907665484d8SDoug Ambrisko 			case MR_EVT_LD_CREATED:
3908665484d8SDoug Ambrisko 				mrsas_get_ld_list(sc);
3909665484d8SDoug Ambrisko 				mrsas_bus_scan_sim(sc, sc->sim_0);
3910665484d8SDoug Ambrisko 				doscan = 0;
3911665484d8SDoug Ambrisko 				break;
3912665484d8SDoug Ambrisko 			case MR_EVT_CTRL_HOST_BUS_SCAN_REQUESTED:
3913665484d8SDoug Ambrisko 			case MR_EVT_FOREIGN_CFG_IMPORTED:
3914665484d8SDoug Ambrisko 			case MR_EVT_LD_STATE_CHANGE:
3915665484d8SDoug Ambrisko 				doscan = 1;
3916665484d8SDoug Ambrisko 				break;
3917665484d8SDoug Ambrisko 			default:
3918665484d8SDoug Ambrisko 				doscan = 0;
3919665484d8SDoug Ambrisko 				break;
3920665484d8SDoug Ambrisko 		}
3921665484d8SDoug Ambrisko 	} else {
3922665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "invalid evt_detail\n");
3923665484d8SDoug Ambrisko 		return;
3924665484d8SDoug Ambrisko 	}
3925665484d8SDoug Ambrisko 	if (doscan) {
3926665484d8SDoug Ambrisko 		mrsas_get_pd_list(sc);
3927665484d8SDoug Ambrisko 		mrsas_dprint(sc, MRSAS_AEN, "scanning ...sim 1\n");
3928665484d8SDoug Ambrisko 		mrsas_bus_scan_sim(sc, sc->sim_1);
3929665484d8SDoug Ambrisko 		mrsas_get_ld_list(sc);
3930665484d8SDoug Ambrisko 		mrsas_dprint(sc, MRSAS_AEN, "scanning ...sim 0\n");
3931665484d8SDoug Ambrisko 		mrsas_bus_scan_sim(sc, sc->sim_0);
3932665484d8SDoug Ambrisko 	}
3933665484d8SDoug Ambrisko 
3934665484d8SDoug Ambrisko 	seq_num = sc->evt_detail_mem->seq_num + 1;
3935665484d8SDoug Ambrisko 
3936665484d8SDoug Ambrisko 	// Register AEN with FW for latest sequence number plus 1
3937665484d8SDoug Ambrisko 	class_locale.members.reserved = 0;
3938665484d8SDoug Ambrisko 	class_locale.members.locale = MR_EVT_LOCALE_ALL;
3939665484d8SDoug Ambrisko 	class_locale.members.class = MR_EVT_CLASS_DEBUG;
3940665484d8SDoug Ambrisko 
3941665484d8SDoug Ambrisko 	if (sc->aen_cmd != NULL )
3942665484d8SDoug Ambrisko 		return ;
3943665484d8SDoug Ambrisko 
3944665484d8SDoug Ambrisko 	mtx_lock(&sc->aen_lock);
3945665484d8SDoug Ambrisko 	error = mrsas_register_aen(sc, seq_num,
3946665484d8SDoug Ambrisko 					class_locale.word);
3947665484d8SDoug Ambrisko 	mtx_unlock(&sc->aen_lock);
3948665484d8SDoug Ambrisko 
3949665484d8SDoug Ambrisko 	if (error)
3950665484d8SDoug Ambrisko 		device_printf(sc->mrsas_dev, "register aen failed error %x\n", error);
3951665484d8SDoug Ambrisko 
3952665484d8SDoug Ambrisko }
3953665484d8SDoug Ambrisko 
3954665484d8SDoug Ambrisko 
3955665484d8SDoug Ambrisko /**
3956665484d8SDoug Ambrisko  * mrsas_complete_aen:        	Completes AEN command
3957665484d8SDoug Ambrisko  * input:                     	Adapter soft state
3958665484d8SDoug Ambrisko  *                            	Cmd that was issued to abort another cmd
3959665484d8SDoug Ambrisko  *
3960665484d8SDoug Ambrisko  * 								This function will be called from ISR and will continue
3961665484d8SDoug Ambrisko  * 								event processing from thread context by enqueuing task
3962665484d8SDoug Ambrisko  * 								in ev_tq (callback function "mrsas_aen_handler").
3963665484d8SDoug Ambrisko  */
3964665484d8SDoug Ambrisko void mrsas_complete_aen(struct mrsas_softc *sc, struct mrsas_mfi_cmd *cmd)
3965665484d8SDoug Ambrisko {
3966665484d8SDoug Ambrisko 	/*
3967665484d8SDoug Ambrisko 	* Don't signal app if it is just an aborted previously registered aen
3968665484d8SDoug Ambrisko 	*/
3969665484d8SDoug Ambrisko 	if ((!cmd->abort_aen) && (sc->remove_in_progress == 0)) {
3970665484d8SDoug Ambrisko 		/* TO DO (?) */
3971665484d8SDoug Ambrisko 	}
3972665484d8SDoug Ambrisko 	else
3973665484d8SDoug Ambrisko 		cmd->abort_aen = 0;
3974665484d8SDoug Ambrisko 
3975665484d8SDoug Ambrisko 	sc->aen_cmd = NULL;
3976665484d8SDoug Ambrisko 	mrsas_release_mfi_cmd(cmd);
3977665484d8SDoug Ambrisko 
3978665484d8SDoug Ambrisko 	if (!sc->remove_in_progress)
3979665484d8SDoug Ambrisko 		taskqueue_enqueue(sc->ev_tq, &sc->ev_task);
3980665484d8SDoug Ambrisko 
3981665484d8SDoug Ambrisko 	return;
3982665484d8SDoug Ambrisko }
3983665484d8SDoug Ambrisko 
3984665484d8SDoug Ambrisko static device_method_t mrsas_methods[] = {
3985665484d8SDoug Ambrisko     DEVMETHOD(device_probe,     mrsas_probe),
3986665484d8SDoug Ambrisko     DEVMETHOD(device_attach,    mrsas_attach),
3987665484d8SDoug Ambrisko     DEVMETHOD(device_detach,    mrsas_detach),
3988665484d8SDoug Ambrisko     DEVMETHOD(device_suspend,   mrsas_suspend),
3989665484d8SDoug Ambrisko     DEVMETHOD(device_resume,    mrsas_resume),
3990665484d8SDoug Ambrisko     DEVMETHOD(bus_print_child,  bus_generic_print_child),
3991665484d8SDoug Ambrisko     DEVMETHOD(bus_driver_added, bus_generic_driver_added),
3992665484d8SDoug Ambrisko     { 0, 0 }
3993665484d8SDoug Ambrisko };
3994665484d8SDoug Ambrisko 
3995665484d8SDoug Ambrisko static driver_t mrsas_driver = {
3996665484d8SDoug Ambrisko     "mrsas",
3997665484d8SDoug Ambrisko     mrsas_methods,
3998665484d8SDoug Ambrisko     sizeof(struct mrsas_softc)
3999665484d8SDoug Ambrisko };
4000665484d8SDoug Ambrisko 
4001665484d8SDoug Ambrisko static devclass_t       mrsas_devclass;
4002665484d8SDoug Ambrisko DRIVER_MODULE(mrsas, pci, mrsas_driver, mrsas_devclass, 0, 0);
4003665484d8SDoug Ambrisko MODULE_DEPEND(mrsas, cam, 1,1,1);
4004665484d8SDoug Ambrisko 
4005