xref: /dflybsd-src/sys/dev/raid/mps/mps_sas_lsi.c (revision 43f7c5533b94bf18b84d4924b43e9288b4dcdedb)
1c12c399aSSascha Wildner /*-
2c12c399aSSascha Wildner  * Copyright (c) 2011 LSI Corp.
3c12c399aSSascha Wildner  * All rights reserved.
4c12c399aSSascha Wildner  *
5c12c399aSSascha Wildner  * Redistribution and use in source and binary forms, with or without
6c12c399aSSascha Wildner  * modification, are permitted provided that the following conditions
7c12c399aSSascha Wildner  * are met:
8c12c399aSSascha Wildner  * 1. Redistributions of source code must retain the above copyright
9c12c399aSSascha Wildner  *    notice, this list of conditions and the following disclaimer.
10c12c399aSSascha Wildner  * 2. Redistributions in binary form must reproduce the above copyright
11c12c399aSSascha Wildner  *    notice, this list of conditions and the following disclaimer in the
12c12c399aSSascha Wildner  *    documentation and/or other materials provided with the distribution.
13c12c399aSSascha Wildner  *
14c12c399aSSascha Wildner  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15c12c399aSSascha Wildner  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16c12c399aSSascha Wildner  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17c12c399aSSascha Wildner  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18c12c399aSSascha Wildner  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19c12c399aSSascha Wildner  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20c12c399aSSascha Wildner  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21c12c399aSSascha Wildner  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22c12c399aSSascha Wildner  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23c12c399aSSascha Wildner  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24c12c399aSSascha Wildner  * SUCH DAMAGE.
25c12c399aSSascha Wildner  *
26c12c399aSSascha Wildner  * LSI MPT-Fusion Host Adapter FreeBSD
27c12c399aSSascha Wildner  *
28c12c399aSSascha Wildner  * $FreeBSD: src/sys/dev/mps/mps_sas_lsi.c,v 1.1 2012/01/26 18:17:21 ken Exp $
29c12c399aSSascha Wildner  */
30c12c399aSSascha Wildner 
31c12c399aSSascha Wildner /* Communications core for LSI MPT2 */
32c12c399aSSascha Wildner 
33c12c399aSSascha Wildner /* TODO Move headers to mpsvar */
34c12c399aSSascha Wildner #include <sys/types.h>
35c12c399aSSascha Wildner #include <sys/param.h>
36c12c399aSSascha Wildner #include <sys/systm.h>
37c12c399aSSascha Wildner #include <sys/kernel.h>
38c12c399aSSascha Wildner #include <sys/module.h>
39c12c399aSSascha Wildner #include <sys/bus.h>
40c12c399aSSascha Wildner #include <sys/conf.h>
41c12c399aSSascha Wildner #include <sys/eventhandler.h>
42c12c399aSSascha Wildner #include <sys/bio.h>
43c12c399aSSascha Wildner #include <sys/malloc.h>
44c12c399aSSascha Wildner #include <sys/uio.h>
45c12c399aSSascha Wildner #include <sys/sysctl.h>
46c12c399aSSascha Wildner #include <sys/endian.h>
47c12c399aSSascha Wildner #include <sys/queue.h>
48c12c399aSSascha Wildner #include <sys/kthread.h>
49c12c399aSSascha Wildner #include <sys/taskqueue.h>
50c12c399aSSascha Wildner #include <sys/sbuf.h>
51c12c399aSSascha Wildner 
52c12c399aSSascha Wildner #include <sys/rman.h>
53c12c399aSSascha Wildner 
54c12c399aSSascha Wildner #include <machine/stdarg.h>
55c12c399aSSascha Wildner 
56c12c399aSSascha Wildner #include <bus/cam/cam.h>
57c12c399aSSascha Wildner #include <bus/cam/cam_ccb.h>
58c12c399aSSascha Wildner #include <bus/cam/cam_debug.h>
59c12c399aSSascha Wildner #include <bus/cam/cam_sim.h>
60c12c399aSSascha Wildner #include <bus/cam/cam_xpt_sim.h>
61c12c399aSSascha Wildner #include <bus/cam/cam_xpt_periph.h>
62c12c399aSSascha Wildner #include <bus/cam/cam_periph.h>
63c12c399aSSascha Wildner #include <bus/cam/scsi/scsi_all.h>
64c12c399aSSascha Wildner #include <bus/cam/scsi/scsi_message.h>
65c12c399aSSascha Wildner 
66c12c399aSSascha Wildner #include <dev/raid/mps/mpi/mpi2_type.h>
67c12c399aSSascha Wildner #include <dev/raid/mps/mpi/mpi2.h>
68c12c399aSSascha Wildner #include <dev/raid/mps/mpi/mpi2_ioc.h>
69c12c399aSSascha Wildner #include <dev/raid/mps/mpi/mpi2_sas.h>
70c12c399aSSascha Wildner #include <dev/raid/mps/mpi/mpi2_cnfg.h>
71c12c399aSSascha Wildner #include <dev/raid/mps/mpi/mpi2_init.h>
72c12c399aSSascha Wildner #include <dev/raid/mps/mpi/mpi2_raid.h>
73c12c399aSSascha Wildner #include <dev/raid/mps/mpi/mpi2_tool.h>
74c12c399aSSascha Wildner #include <dev/raid/mps/mps_ioctl.h>
75c12c399aSSascha Wildner #include <dev/raid/mps/mpsvar.h>
76c12c399aSSascha Wildner #include <dev/raid/mps/mps_table.h>
77c12c399aSSascha Wildner #include <dev/raid/mps/mps_sas.h>
78c12c399aSSascha Wildner 
79c12c399aSSascha Wildner /* For Hashed SAS Address creation for SATA Drives */
80c12c399aSSascha Wildner #define MPT2SAS_SN_LEN 20
81c12c399aSSascha Wildner #define MPT2SAS_MN_LEN 40
82c12c399aSSascha Wildner 
83c12c399aSSascha Wildner struct mps_fw_event_work {
84c12c399aSSascha Wildner 	u16			event;
85c12c399aSSascha Wildner 	void			*event_data;
86c12c399aSSascha Wildner 	TAILQ_ENTRY(mps_fw_event_work)	ev_link;
87c12c399aSSascha Wildner };
88c12c399aSSascha Wildner 
89c12c399aSSascha Wildner union _sata_sas_address {
90c12c399aSSascha Wildner 	u8 wwid[8];
91c12c399aSSascha Wildner 	struct {
92c12c399aSSascha Wildner 		u32 high;
93c12c399aSSascha Wildner 		u32 low;
94c12c399aSSascha Wildner 	} word;
95c12c399aSSascha Wildner };
96c12c399aSSascha Wildner 
97c12c399aSSascha Wildner /*
98c12c399aSSascha Wildner  * define the IDENTIFY DEVICE structure
99c12c399aSSascha Wildner  */
100c12c399aSSascha Wildner struct _ata_identify_device_data {
101c12c399aSSascha Wildner 	u16 reserved1[10];	/* 0-9 */
102c12c399aSSascha Wildner 	u16 serial_number[10];	/* 10-19 */
103c12c399aSSascha Wildner 	u16 reserved2[7];	/* 20-26 */
104c12c399aSSascha Wildner 	u16 model_number[20];	/* 27-46*/
105c12c399aSSascha Wildner 	u16 reserved3[209];	/* 47-255*/
106c12c399aSSascha Wildner };
107c12c399aSSascha Wildner 
108*43f7c553SMatthew Dillon static u32 event_count;
109*43f7c553SMatthew Dillon 
110c12c399aSSascha Wildner static void mpssas_fw_work(struct mps_softc *sc,
111c12c399aSSascha Wildner     struct mps_fw_event_work *fw_event);
112c12c399aSSascha Wildner static void mpssas_fw_event_free(struct mps_softc *,
113c12c399aSSascha Wildner     struct mps_fw_event_work *);
114c12c399aSSascha Wildner static int mpssas_add_device(struct mps_softc *sc, u16 handle, u8 linkrate);
115c12c399aSSascha Wildner static int mpssas_get_sata_identify(struct mps_softc *sc, u16 handle,
116c12c399aSSascha Wildner     Mpi2SataPassthroughReply_t *mpi_reply, char *id_buffer, int sz,
117c12c399aSSascha Wildner     u32 devinfo);
118c12c399aSSascha Wildner int mpssas_get_sas_address_for_sata_disk(struct mps_softc *sc,
119c12c399aSSascha Wildner     u64 *sas_address, u16 handle, u32 device_info);
120c12c399aSSascha Wildner static int mpssas_volume_add(struct mps_softc *sc,
121*43f7c553SMatthew Dillon     u16 handle);
122c12c399aSSascha Wildner 
123c12c399aSSascha Wildner void
mpssas_evt_handler(struct mps_softc * sc,uintptr_t data,MPI2_EVENT_NOTIFICATION_REPLY * event)124c12c399aSSascha Wildner mpssas_evt_handler(struct mps_softc *sc, uintptr_t data,
125c12c399aSSascha Wildner     MPI2_EVENT_NOTIFICATION_REPLY *event)
126c12c399aSSascha Wildner {
127c12c399aSSascha Wildner 	struct mps_fw_event_work *fw_event;
128c12c399aSSascha Wildner 	u16 sz;
129c12c399aSSascha Wildner 
130c12c399aSSascha Wildner 	mps_dprint(sc, MPS_TRACE, "%s\n", __func__);
131c12c399aSSascha Wildner 	mps_print_evt_sas(sc, event);
132c12c399aSSascha Wildner 	mpssas_record_event(sc, event);
133c12c399aSSascha Wildner 
134c12c399aSSascha Wildner 	fw_event = kmalloc(sizeof(struct mps_fw_event_work), M_MPT2,
1353eb4d33cSMatthew Dillon 	     M_ZERO|M_INTWAIT);
136c12c399aSSascha Wildner 	if (!fw_event) {
137c12c399aSSascha Wildner 		kprintf("%s: allocate failed for fw_event\n", __func__);
138c12c399aSSascha Wildner 		return;
139c12c399aSSascha Wildner 	}
140c12c399aSSascha Wildner 	sz = le16toh(event->EventDataLength) * 4;
1413eb4d33cSMatthew Dillon 	fw_event->event_data = kmalloc(sz, M_MPT2, M_ZERO|M_INTWAIT);
142c12c399aSSascha Wildner 	if (!fw_event->event_data) {
143c12c399aSSascha Wildner 		kprintf("%s: allocate failed for event_data\n", __func__);
144c12c399aSSascha Wildner 		kfree(fw_event, M_MPT2);
145c12c399aSSascha Wildner 		return;
146c12c399aSSascha Wildner 	}
147c12c399aSSascha Wildner 
148c12c399aSSascha Wildner 	bcopy(event->EventData, fw_event->event_data, sz);
149c12c399aSSascha Wildner 	fw_event->event = event->Event;
150c12c399aSSascha Wildner 	if ((event->Event == MPI2_EVENT_SAS_TOPOLOGY_CHANGE_LIST ||
151c12c399aSSascha Wildner 	    event->Event == MPI2_EVENT_SAS_ENCL_DEVICE_STATUS_CHANGE ||
152c12c399aSSascha Wildner 	    event->Event == MPI2_EVENT_IR_CONFIGURATION_CHANGE_LIST) &&
153c12c399aSSascha Wildner 	    sc->track_mapping_events)
154c12c399aSSascha Wildner 		sc->pending_map_events++;
155c12c399aSSascha Wildner 
156c12c399aSSascha Wildner 	/*
157c12c399aSSascha Wildner 	 * When wait_for_port_enable flag is set, make sure that all the events
158c12c399aSSascha Wildner 	 * are processed. Increment the startup_refcount and decrement it after
159c12c399aSSascha Wildner 	 * events are processed.
160c12c399aSSascha Wildner 	 */
161c12c399aSSascha Wildner 	if ((event->Event == MPI2_EVENT_SAS_TOPOLOGY_CHANGE_LIST ||
162c12c399aSSascha Wildner 	    event->Event == MPI2_EVENT_IR_CONFIGURATION_CHANGE_LIST) &&
163c12c399aSSascha Wildner 	    sc->wait_for_port_enable)
164c12c399aSSascha Wildner 		mpssas_startup_increment(sc->sassc);
165c12c399aSSascha Wildner 
166c12c399aSSascha Wildner 	TAILQ_INSERT_TAIL(&sc->sassc->ev_queue, fw_event, ev_link);
167c12c399aSSascha Wildner 	taskqueue_enqueue(sc->sassc->ev_tq, &sc->sassc->ev_task);
168c12c399aSSascha Wildner 
169c12c399aSSascha Wildner }
170c12c399aSSascha Wildner 
171c12c399aSSascha Wildner static void
mpssas_fw_event_free(struct mps_softc * sc,struct mps_fw_event_work * fw_event)172c12c399aSSascha Wildner mpssas_fw_event_free(struct mps_softc *sc, struct mps_fw_event_work *fw_event)
173c12c399aSSascha Wildner {
174c12c399aSSascha Wildner 
175c12c399aSSascha Wildner 	kfree(fw_event->event_data, M_MPT2);
176c12c399aSSascha Wildner 	kfree(fw_event, M_MPT2);
177c12c399aSSascha Wildner }
178c12c399aSSascha Wildner 
179c12c399aSSascha Wildner /**
180c12c399aSSascha Wildner  * _mps_fw_work - delayed task for processing firmware events
181c12c399aSSascha Wildner  * @sc: per adapter object
182c12c399aSSascha Wildner  * @fw_event: The fw_event_work object
183c12c399aSSascha Wildner  * Context: user.
184c12c399aSSascha Wildner  *
185c12c399aSSascha Wildner  * Return nothing.
186c12c399aSSascha Wildner  */
187c12c399aSSascha Wildner static void
mpssas_fw_work(struct mps_softc * sc,struct mps_fw_event_work * fw_event)188c12c399aSSascha Wildner mpssas_fw_work(struct mps_softc *sc, struct mps_fw_event_work *fw_event)
189c12c399aSSascha Wildner {
190c12c399aSSascha Wildner 	struct mpssas_softc *sassc;
191c12c399aSSascha Wildner 	sassc = sc->sassc;
192c12c399aSSascha Wildner 
193*43f7c553SMatthew Dillon 	mps_dprint(sc, MPS_INFO, "(%d)->(%s) Working on  Event: [%x]\n",
194*43f7c553SMatthew Dillon 		   event_count++,__func__,fw_event->event);
195*43f7c553SMatthew Dillon 
196c12c399aSSascha Wildner 	switch (fw_event->event) {
197c12c399aSSascha Wildner 	case MPI2_EVENT_SAS_TOPOLOGY_CHANGE_LIST:
198c12c399aSSascha Wildner 	{
199c12c399aSSascha Wildner 		MPI2_EVENT_DATA_SAS_TOPOLOGY_CHANGE_LIST *data;
200c12c399aSSascha Wildner 		MPI2_EVENT_SAS_TOPO_PHY_ENTRY *phy;
201c12c399aSSascha Wildner 		int i;
202c12c399aSSascha Wildner 
203c12c399aSSascha Wildner 		data = (MPI2_EVENT_DATA_SAS_TOPOLOGY_CHANGE_LIST *)
204c12c399aSSascha Wildner 		    fw_event->event_data;
205c12c399aSSascha Wildner 
206c12c399aSSascha Wildner 		mps_mapping_topology_change_event(sc, fw_event->event_data);
207c12c399aSSascha Wildner 
208c12c399aSSascha Wildner 		for (i = 0; i < data->NumEntries; i++) {
209c12c399aSSascha Wildner 			phy = &data->PHY[i];
210c12c399aSSascha Wildner 			switch (phy->PhyStatus & MPI2_EVENT_SAS_TOPO_RC_MASK) {
211c12c399aSSascha Wildner 			case MPI2_EVENT_SAS_TOPO_RC_TARG_ADDED:
212c12c399aSSascha Wildner 				if (mpssas_add_device(sc,
213c12c399aSSascha Wildner 				    phy->AttachedDevHandle, phy->LinkRate)){
214c12c399aSSascha Wildner 					kprintf("%s: failed to add device with "
215c12c399aSSascha Wildner 					    "handle 0x%x\n", __func__,
216c12c399aSSascha Wildner 					    phy->AttachedDevHandle);
217c12c399aSSascha Wildner 					mpssas_prepare_remove(sassc, phy->
218c12c399aSSascha Wildner 					    AttachedDevHandle);
219c12c399aSSascha Wildner 				}
220c12c399aSSascha Wildner 				break;
221c12c399aSSascha Wildner 			case MPI2_EVENT_SAS_TOPO_RC_TARG_NOT_RESPONDING:
222c12c399aSSascha Wildner 				mpssas_prepare_remove(sassc, phy->
223c12c399aSSascha Wildner 				    AttachedDevHandle);
224c12c399aSSascha Wildner 				break;
225c12c399aSSascha Wildner 			case MPI2_EVENT_SAS_TOPO_RC_PHY_CHANGED:
226c12c399aSSascha Wildner 			case MPI2_EVENT_SAS_TOPO_RC_NO_CHANGE:
227c12c399aSSascha Wildner 			case MPI2_EVENT_SAS_TOPO_RC_DELAY_NOT_RESPONDING:
228c12c399aSSascha Wildner 			default:
229c12c399aSSascha Wildner 				break;
230c12c399aSSascha Wildner 			}
231c12c399aSSascha Wildner 		}
232c12c399aSSascha Wildner 		/*
233c12c399aSSascha Wildner 		 * refcount was incremented for this event in
234c12c399aSSascha Wildner 		 * mpssas_evt_handler.  Decrement it here because the event has
235c12c399aSSascha Wildner 		 * been processed.
236c12c399aSSascha Wildner 		 */
237c12c399aSSascha Wildner 		mpssas_startup_decrement(sassc);
238c12c399aSSascha Wildner 		break;
239c12c399aSSascha Wildner 	}
240c12c399aSSascha Wildner 	case MPI2_EVENT_SAS_DISCOVERY:
241c12c399aSSascha Wildner 	{
242c12c399aSSascha Wildner 		MPI2_EVENT_DATA_SAS_DISCOVERY *data;
243c12c399aSSascha Wildner 
244c12c399aSSascha Wildner 		data = (MPI2_EVENT_DATA_SAS_DISCOVERY *)fw_event->event_data;
245c12c399aSSascha Wildner 
246c12c399aSSascha Wildner 		if (data->ReasonCode & MPI2_EVENT_SAS_DISC_RC_STARTED)
247c12c399aSSascha Wildner 			mps_dprint(sc, MPS_TRACE,"SAS discovery start event\n");
248c12c399aSSascha Wildner 		if (data->ReasonCode & MPI2_EVENT_SAS_DISC_RC_COMPLETED) {
249c12c399aSSascha Wildner 			mps_dprint(sc, MPS_TRACE,"SAS discovery stop event\n");
250c12c399aSSascha Wildner 			sassc->flags &= ~MPSSAS_IN_DISCOVERY;
251c12c399aSSascha Wildner 			mpssas_discovery_end(sassc);
252c12c399aSSascha Wildner 		}
253c12c399aSSascha Wildner 		break;
254c12c399aSSascha Wildner 	}
255c12c399aSSascha Wildner 	case MPI2_EVENT_SAS_ENCL_DEVICE_STATUS_CHANGE:
256c12c399aSSascha Wildner 	{
257c12c399aSSascha Wildner 		Mpi2EventDataSasEnclDevStatusChange_t *data;
258c12c399aSSascha Wildner 		data = (Mpi2EventDataSasEnclDevStatusChange_t *)
259c12c399aSSascha Wildner 		    fw_event->event_data;
260f8fc9ba5SSascha Wildner 		mps_mapping_enclosure_dev_status_change_event(sc, data);
261c12c399aSSascha Wildner 		break;
262c12c399aSSascha Wildner 	}
263c12c399aSSascha Wildner 	case MPI2_EVENT_IR_CONFIGURATION_CHANGE_LIST:
264c12c399aSSascha Wildner 	{
265c12c399aSSascha Wildner 		Mpi2EventIrConfigElement_t *element;
266c12c399aSSascha Wildner 		int i;
267c12c399aSSascha Wildner 		u8 foreign_config;
268c12c399aSSascha Wildner 		Mpi2EventDataIrConfigChangeList_t *event_data;
269c12c399aSSascha Wildner 		struct mpssas_target *targ;
270c12c399aSSascha Wildner 		unsigned int id;
271c12c399aSSascha Wildner 
272c12c399aSSascha Wildner 		event_data = fw_event->event_data;
273c12c399aSSascha Wildner 		foreign_config = (le32toh(event_data->Flags) &
274c12c399aSSascha Wildner 		    MPI2_EVENT_IR_CHANGE_FLAGS_FOREIGN_CONFIG) ? 1 : 0;
275c12c399aSSascha Wildner 
276c12c399aSSascha Wildner 		element =
277c12c399aSSascha Wildner 		    (Mpi2EventIrConfigElement_t *)&event_data->ConfigElement[0];
278c12c399aSSascha Wildner 		id = mps_mapping_get_raid_id_from_handle
279c12c399aSSascha Wildner 		    (sc, element->VolDevHandle);
280c12c399aSSascha Wildner 
281c12c399aSSascha Wildner 		mps_mapping_ir_config_change_event(sc, event_data);
282c12c399aSSascha Wildner 
283c12c399aSSascha Wildner 		for (i = 0; i < event_data->NumElements; i++, element++) {
284c12c399aSSascha Wildner 			switch (element->ReasonCode) {
285c12c399aSSascha Wildner 			case MPI2_EVENT_IR_CHANGE_RC_VOLUME_CREATED:
286c12c399aSSascha Wildner 			case MPI2_EVENT_IR_CHANGE_RC_ADDED:
287c12c399aSSascha Wildner 				if (!foreign_config) {
288*43f7c553SMatthew Dillon 					if (mpssas_volume_add(sc, le16toh(element->VolDevHandle))) {
289c12c399aSSascha Wildner 						kprintf("%s: failed to add RAID "
290c12c399aSSascha Wildner 						    "volume with handle 0x%x\n",
291c12c399aSSascha Wildner 						    __func__, le16toh(element->
292c12c399aSSascha Wildner 						    VolDevHandle));
293c12c399aSSascha Wildner 					}
294c12c399aSSascha Wildner 				}
295c12c399aSSascha Wildner 				break;
296c12c399aSSascha Wildner 			case MPI2_EVENT_IR_CHANGE_RC_VOLUME_DELETED:
297c12c399aSSascha Wildner 			case MPI2_EVENT_IR_CHANGE_RC_REMOVED:
298c12c399aSSascha Wildner 				/*
299c12c399aSSascha Wildner 				 * Rescan after volume is deleted or removed.
300c12c399aSSascha Wildner 				 */
301c12c399aSSascha Wildner 				if (!foreign_config) {
302c12c399aSSascha Wildner 					if (id == MPS_MAP_BAD_ID) {
303c12c399aSSascha Wildner 						kprintf("%s: could not get ID "
304c12c399aSSascha Wildner 						    "for volume with handle "
305c12c399aSSascha Wildner 						    "0x%04x\n", __func__,
306c12c399aSSascha Wildner 						    element->VolDevHandle);
307c12c399aSSascha Wildner 						break;
308c12c399aSSascha Wildner 					}
309c12c399aSSascha Wildner 
310c12c399aSSascha Wildner 					targ = &sassc->targets[id];
311c12c399aSSascha Wildner 					targ->handle = 0x0;
312c12c399aSSascha Wildner 					targ->encl_slot = 0x0;
313c12c399aSSascha Wildner 					targ->encl_handle = 0x0;
314c12c399aSSascha Wildner 					targ->exp_dev_handle = 0x0;
315c12c399aSSascha Wildner 					targ->phy_num = 0x0;
316c12c399aSSascha Wildner 					targ->linkrate = 0x0;
317c12c399aSSascha Wildner 					mpssas_rescan_target(sc, targ);
318c12c399aSSascha Wildner 					kprintf("RAID target id 0x%x removed\n",
319c12c399aSSascha Wildner 					    targ->tid);
320c12c399aSSascha Wildner 				}
321c12c399aSSascha Wildner 				break;
322c12c399aSSascha Wildner 			case MPI2_EVENT_IR_CHANGE_RC_PD_CREATED:
323*43f7c553SMatthew Dillon 			case MPI2_EVENT_IR_CHANGE_RC_HIDE:
324c12c399aSSascha Wildner 				/*
325c12c399aSSascha Wildner 				 * Phys Disk of a volume has been created.  Hide
326c12c399aSSascha Wildner 				 * it from the OS.
327c12c399aSSascha Wildner 				 */
328*43f7c553SMatthew Dillon 				targ = mpssas_find_target_by_handle(sassc, 0, element->PhysDiskDevHandle);
329*43f7c553SMatthew Dillon 				if (targ == NULL)
330*43f7c553SMatthew Dillon 					break;
331*43f7c553SMatthew Dillon 				targ->flags |= MPS_TARGET_FLAGS_RAID_COMPONENT;
332*43f7c553SMatthew Dillon 				mpssas_rescan_target(sc, targ);
333c12c399aSSascha Wildner 				break;
334c12c399aSSascha Wildner 			case MPI2_EVENT_IR_CHANGE_RC_PD_DELETED:
335c12c399aSSascha Wildner 				/*
336c12c399aSSascha Wildner 				 * Phys Disk of a volume has been deleted.
337c12c399aSSascha Wildner 				 * Expose it to the OS.
338c12c399aSSascha Wildner 				 */
339c12c399aSSascha Wildner 				if (mpssas_add_device(sc,
340c12c399aSSascha Wildner 				    element->PhysDiskDevHandle, 0)){
341c12c399aSSascha Wildner 					kprintf("%s: failed to add device with "
342c12c399aSSascha Wildner 					    "handle 0x%x\n", __func__,
343c12c399aSSascha Wildner 					    element->PhysDiskDevHandle);
344c12c399aSSascha Wildner 					mpssas_prepare_remove(sassc, element->
345c12c399aSSascha Wildner 					    PhysDiskDevHandle);
346c12c399aSSascha Wildner 				}
347c12c399aSSascha Wildner 				break;
348c12c399aSSascha Wildner 			}
349c12c399aSSascha Wildner 		}
350c12c399aSSascha Wildner 		/*
351c12c399aSSascha Wildner 		 * refcount was incremented for this event in
352c12c399aSSascha Wildner 		 * mpssas_evt_handler.  Decrement it here because the event has
353c12c399aSSascha Wildner 		 * been processed.
354c12c399aSSascha Wildner 		 */
355c12c399aSSascha Wildner 		mpssas_startup_decrement(sassc);
356c12c399aSSascha Wildner 		break;
357c12c399aSSascha Wildner 	}
358c12c399aSSascha Wildner 	case MPI2_EVENT_IR_VOLUME:
359c12c399aSSascha Wildner 	{
360c12c399aSSascha Wildner 		Mpi2EventDataIrVolume_t *event_data = fw_event->event_data;
361c12c399aSSascha Wildner 
362c12c399aSSascha Wildner 		/*
363c12c399aSSascha Wildner 		 * Informational only.
364c12c399aSSascha Wildner 		 */
365c12c399aSSascha Wildner 		mps_dprint(sc, MPS_INFO, "Received IR Volume event:\n");
366c12c399aSSascha Wildner 		switch (event_data->ReasonCode) {
367c12c399aSSascha Wildner 		case MPI2_EVENT_IR_VOLUME_RC_SETTINGS_CHANGED:
368c12c399aSSascha Wildner 			mps_dprint(sc, MPS_INFO, "   Volume Settings "
369c12c399aSSascha Wildner 			    "changed from 0x%x to 0x%x for Volome with "
37090ff74f1SSascha Wildner 			    "handle 0x%x", event_data->PreviousValue,
37190ff74f1SSascha Wildner 			    event_data->NewValue,
372c12c399aSSascha Wildner 			    event_data->VolDevHandle);
373c12c399aSSascha Wildner 			break;
374c12c399aSSascha Wildner 		case MPI2_EVENT_IR_VOLUME_RC_STATUS_FLAGS_CHANGED:
375c12c399aSSascha Wildner 			mps_dprint(sc, MPS_INFO, "   Volume Status "
376c12c399aSSascha Wildner 			    "changed from 0x%x to 0x%x for Volome with "
37790ff74f1SSascha Wildner 			    "handle 0x%x", event_data->PreviousValue,
37890ff74f1SSascha Wildner 			    event_data->NewValue,
379c12c399aSSascha Wildner 			    event_data->VolDevHandle);
380c12c399aSSascha Wildner 			break;
381c12c399aSSascha Wildner 		case MPI2_EVENT_IR_VOLUME_RC_STATE_CHANGED:
382c12c399aSSascha Wildner 			mps_dprint(sc, MPS_INFO, "   Volume State "
383c12c399aSSascha Wildner 			    "changed from 0x%x to 0x%x for Volome with "
38490ff74f1SSascha Wildner 			    "handle 0x%x", event_data->PreviousValue,
38590ff74f1SSascha Wildner 			    event_data->NewValue,
386c12c399aSSascha Wildner 			    event_data->VolDevHandle);
387*43f7c553SMatthew Dillon 			u32 state;
388*43f7c553SMatthew Dillon 			struct mpssas_target *targ;
389*43f7c553SMatthew Dillon 
390*43f7c553SMatthew Dillon 			state = le32toh(event_data->NewValue);
391*43f7c553SMatthew Dillon 			switch (state) {
392*43f7c553SMatthew Dillon 			case MPI2_RAID_VOL_STATE_MISSING:
393*43f7c553SMatthew Dillon 			case MPI2_RAID_VOL_STATE_FAILED:
394*43f7c553SMatthew Dillon 				mpssas_prepare_volume_remove(sassc,
395*43f7c553SMatthew Dillon 							     event_data->VolDevHandle);
396*43f7c553SMatthew Dillon 				break;
397*43f7c553SMatthew Dillon 			case MPI2_RAID_VOL_STATE_ONLINE:
398*43f7c553SMatthew Dillon 			case MPI2_RAID_VOL_STATE_DEGRADED:
399*43f7c553SMatthew Dillon 			case MPI2_RAID_VOL_STATE_OPTIMAL:
400*43f7c553SMatthew Dillon 				targ = mpssas_find_target_by_handle(sassc, 0, event_data->VolDevHandle);
401*43f7c553SMatthew Dillon 				if (targ) {
402*43f7c553SMatthew Dillon 					kprintf("%s %d: Volume handle 0x%x is already added \n",
403*43f7c553SMatthew Dillon 						__func__, __LINE__ , event_data->VolDevHandle);
404*43f7c553SMatthew Dillon 					break;
405*43f7c553SMatthew Dillon 				}
406*43f7c553SMatthew Dillon 				if (mpssas_volume_add(sc, le16toh(event_data->VolDevHandle))) {
407*43f7c553SMatthew Dillon 					kprintf("%s: failed to add RAID "
408*43f7c553SMatthew Dillon 						"volume with handle 0x%x\n",
409*43f7c553SMatthew Dillon 						__func__, le16toh(event_data->
410*43f7c553SMatthew Dillon 						VolDevHandle));
411*43f7c553SMatthew Dillon 				}
412*43f7c553SMatthew Dillon 				break;
413*43f7c553SMatthew Dillon 			default:
414*43f7c553SMatthew Dillon 				break;
415*43f7c553SMatthew Dillon 			}
416c12c399aSSascha Wildner 			break;
417c12c399aSSascha Wildner 		default:
418c12c399aSSascha Wildner 			break;
419c12c399aSSascha Wildner 		}
420c12c399aSSascha Wildner 		break;
421c12c399aSSascha Wildner 	}
422c12c399aSSascha Wildner 	case MPI2_EVENT_IR_PHYSICAL_DISK:
423c12c399aSSascha Wildner 	{
424c12c399aSSascha Wildner 		Mpi2EventDataIrPhysicalDisk_t *event_data =
425c12c399aSSascha Wildner 		    fw_event->event_data;
426*43f7c553SMatthew Dillon 		struct mpssas_target *targ;
427c12c399aSSascha Wildner 
428c12c399aSSascha Wildner 		/*
429c12c399aSSascha Wildner 		 * Informational only.
430c12c399aSSascha Wildner 		 */
431c12c399aSSascha Wildner 		mps_dprint(sc, MPS_INFO, "Received IR Phys Disk event:\n");
432c12c399aSSascha Wildner 		switch (event_data->ReasonCode) {
433c12c399aSSascha Wildner 		case MPI2_EVENT_IR_PHYSDISK_RC_SETTINGS_CHANGED:
434c12c399aSSascha Wildner 			mps_dprint(sc, MPS_INFO, "   Phys Disk Settings "
435c12c399aSSascha Wildner 			    "changed from 0x%x to 0x%x for Phys Disk Number "
436c12c399aSSascha Wildner 			    "%d and handle 0x%x at Enclosure handle 0x%x, Slot "
437*43f7c553SMatthew Dillon 			    "%d\n", event_data->PreviousValue,
43890ff74f1SSascha Wildner 			    event_data->NewValue, event_data->PhysDiskNum,
439c12c399aSSascha Wildner 			    event_data->PhysDiskDevHandle,
440c12c399aSSascha Wildner 			    event_data->EnclosureHandle, event_data->Slot);
441c12c399aSSascha Wildner 			break;
442c12c399aSSascha Wildner 		case MPI2_EVENT_IR_PHYSDISK_RC_STATUS_FLAGS_CHANGED:
443c12c399aSSascha Wildner 			mps_dprint(sc, MPS_INFO, "   Phys Disk Status changed "
444c12c399aSSascha Wildner 			    "from 0x%x to 0x%x for Phys Disk Number %d and "
445*43f7c553SMatthew Dillon 			    "handle 0x%x at Enclosure handle 0x%x, Slot %d\n",
44690ff74f1SSascha Wildner 			    event_data->PreviousValue, event_data->NewValue,
447c12c399aSSascha Wildner 			    event_data->PhysDiskNum,
448c12c399aSSascha Wildner 			    event_data->PhysDiskDevHandle,
449c12c399aSSascha Wildner 			    event_data->EnclosureHandle, event_data->Slot);
450c12c399aSSascha Wildner 			break;
451c12c399aSSascha Wildner 		case MPI2_EVENT_IR_PHYSDISK_RC_STATE_CHANGED:
452c12c399aSSascha Wildner 			mps_dprint(sc, MPS_INFO, "   Phys Disk State changed "
453c12c399aSSascha Wildner 			    "from 0x%x to 0x%x for Phys Disk Number %d and "
454*43f7c553SMatthew Dillon 			    "handle 0x%x at Enclosure handle 0x%x, Slot %d\n",
45590ff74f1SSascha Wildner 			    event_data->PreviousValue, event_data->NewValue,
456c12c399aSSascha Wildner 			    event_data->PhysDiskNum,
457c12c399aSSascha Wildner 			    event_data->PhysDiskDevHandle,
458c12c399aSSascha Wildner 			    event_data->EnclosureHandle, event_data->Slot);
459*43f7c553SMatthew Dillon 			switch (event_data->NewValue) {
460*43f7c553SMatthew Dillon 			case MPI2_RAID_PD_STATE_ONLINE:
461*43f7c553SMatthew Dillon 			case MPI2_RAID_PD_STATE_DEGRADED:
462*43f7c553SMatthew Dillon 			case MPI2_RAID_PD_STATE_REBUILDING:
463*43f7c553SMatthew Dillon 			case MPI2_RAID_PD_STATE_OPTIMAL:
464*43f7c553SMatthew Dillon 			case MPI2_RAID_PD_STATE_HOT_SPARE:
465*43f7c553SMatthew Dillon 				targ = mpssas_find_target_by_handle(sassc, 0,
466*43f7c553SMatthew Dillon 						event_data->PhysDiskDevHandle);
467*43f7c553SMatthew Dillon 				if (targ) {
468*43f7c553SMatthew Dillon 					targ->flags |= MPS_TARGET_FLAGS_RAID_COMPONENT;
469*43f7c553SMatthew Dillon 					kprintf("%s %d: Found Target for handle 0x%x.  \n",
470*43f7c553SMatthew Dillon 						__func__, __LINE__,
471*43f7c553SMatthew Dillon 						event_data->PhysDiskDevHandle);
472*43f7c553SMatthew Dillon 				}
473c12c399aSSascha Wildner 				break;
474*43f7c553SMatthew Dillon 			case MPI2_RAID_PD_STATE_OFFLINE:
475*43f7c553SMatthew Dillon 			case MPI2_RAID_PD_STATE_NOT_CONFIGURED:
476*43f7c553SMatthew Dillon 			case MPI2_RAID_PD_STATE_NOT_COMPATIBLE:
477*43f7c553SMatthew Dillon 			default:
478*43f7c553SMatthew Dillon 				targ = mpssas_find_target_by_handle(sassc, 0,
479*43f7c553SMatthew Dillon 						event_data->PhysDiskDevHandle);
480*43f7c553SMatthew Dillon 				if (targ) {
481*43f7c553SMatthew Dillon 					targ->flags |= ~MPS_TARGET_FLAGS_RAID_COMPONENT;
482*43f7c553SMatthew Dillon 					kprintf("%s %d: Found Target for handle 0x%x.  \n",
483*43f7c553SMatthew Dillon 						__func__, __LINE__,
484*43f7c553SMatthew Dillon 						event_data->PhysDiskDevHandle);
485*43f7c553SMatthew Dillon 				}
486*43f7c553SMatthew Dillon 				break;
487*43f7c553SMatthew Dillon 			}
488c12c399aSSascha Wildner 		default:
489c12c399aSSascha Wildner 			break;
490c12c399aSSascha Wildner 		}
491c12c399aSSascha Wildner 		break;
492c12c399aSSascha Wildner 	}
493c12c399aSSascha Wildner 	case MPI2_EVENT_IR_OPERATION_STATUS:
494c12c399aSSascha Wildner 	{
495c12c399aSSascha Wildner 		Mpi2EventDataIrOperationStatus_t *event_data =
496c12c399aSSascha Wildner 		    fw_event->event_data;
497c12c399aSSascha Wildner 
498c12c399aSSascha Wildner 		/*
499c12c399aSSascha Wildner 		 * Informational only.
500c12c399aSSascha Wildner 		 */
501c12c399aSSascha Wildner 		mps_dprint(sc, MPS_INFO, "Received IR Op Status event:\n");
502c12c399aSSascha Wildner 		mps_dprint(sc, MPS_INFO, "   RAID Operation of %d is %d "
503c12c399aSSascha Wildner 		    "percent complete for Volume with handle 0x%x",
504c12c399aSSascha Wildner 		    event_data->RAIDOperation, event_data->PercentComplete,
505c12c399aSSascha Wildner 		    event_data->VolDevHandle);
506c12c399aSSascha Wildner 		break;
507c12c399aSSascha Wildner 	}
508c12c399aSSascha Wildner 	case MPI2_EVENT_LOG_ENTRY_ADDED:
509c12c399aSSascha Wildner 	{
510c12c399aSSascha Wildner 		pMpi2EventDataLogEntryAdded_t	logEntry;
511c12c399aSSascha Wildner 		uint16_t			logQualifier;
512c12c399aSSascha Wildner 		uint8_t				logCode;
513c12c399aSSascha Wildner 
514c12c399aSSascha Wildner 		logEntry = (pMpi2EventDataLogEntryAdded_t)fw_event->event_data;
515c12c399aSSascha Wildner 		logQualifier = logEntry->LogEntryQualifier;
516c12c399aSSascha Wildner 
517c12c399aSSascha Wildner 		if (logQualifier == MPI2_WD_LOG_ENTRY) {
518c12c399aSSascha Wildner 			logCode = logEntry->LogData[0];
519c12c399aSSascha Wildner 
520c12c399aSSascha Wildner 			switch (logCode) {
521c12c399aSSascha Wildner 			case MPI2_WD_SSD_THROTTLING:
522c12c399aSSascha Wildner 				kprintf("WarpDrive Warning: IO Throttling has "
523c12c399aSSascha Wildner 				    "occurred in the WarpDrive subsystem. "
524c12c399aSSascha Wildner 				    "Check WarpDrive documentation for "
525c12c399aSSascha Wildner 				    "additional details\n");
526c12c399aSSascha Wildner 				break;
527c12c399aSSascha Wildner 			case MPI2_WD_DRIVE_LIFE_WARN:
528c12c399aSSascha Wildner 				kprintf("WarpDrive Warning: Program/Erase "
529c12c399aSSascha Wildner 				    "Cycles for the WarpDrive subsystem in "
530c12c399aSSascha Wildner 				    "degraded range. Check WarpDrive "
531c12c399aSSascha Wildner 				    "documentation for additional details\n");
532c12c399aSSascha Wildner 				break;
533c12c399aSSascha Wildner 			case MPI2_WD_DRIVE_LIFE_DEAD:
534c12c399aSSascha Wildner 				kprintf("WarpDrive Fatal Error: There are no "
535c12c399aSSascha Wildner 				    "Program/Erase Cycles for the WarpDrive "
536c12c399aSSascha Wildner 				    "subsystem. The storage device will be in "
537c12c399aSSascha Wildner 				    "read-only mode. Check WarpDrive "
538c12c399aSSascha Wildner 				    "documentation for additional details\n");
539c12c399aSSascha Wildner 				break;
540c12c399aSSascha Wildner 			case MPI2_WD_RAIL_MON_FAIL:
541c12c399aSSascha Wildner 				kprintf("WarpDrive Fatal Error: The Backup Rail "
542c12c399aSSascha Wildner 				    "Monitor has failed on the WarpDrive "
543c12c399aSSascha Wildner 				    "subsystem. Check WarpDrive documentation "
544c12c399aSSascha Wildner 				    "for additional details\n");
545c12c399aSSascha Wildner 				break;
546c12c399aSSascha Wildner 			default:
547c12c399aSSascha Wildner 				break;
548c12c399aSSascha Wildner 			}
549c12c399aSSascha Wildner 		}
550c12c399aSSascha Wildner 		break;
551c12c399aSSascha Wildner 	}
552c12c399aSSascha Wildner 	case MPI2_EVENT_SAS_DEVICE_STATUS_CHANGE:
553c12c399aSSascha Wildner 	case MPI2_EVENT_SAS_BROADCAST_PRIMITIVE:
554c12c399aSSascha Wildner 	default:
555c12c399aSSascha Wildner 		mps_dprint(sc, MPS_TRACE,"Unhandled event 0x%0X\n",
556c12c399aSSascha Wildner 		    fw_event->event);
557c12c399aSSascha Wildner 		break;
558c12c399aSSascha Wildner 
559c12c399aSSascha Wildner 	}
560*43f7c553SMatthew Dillon 	mps_dprint(sc, MPS_INFO, "(%d)->(%s) Event Free: [%x]\n",event_count,__func__, fw_event->event);
561c12c399aSSascha Wildner 	mpssas_fw_event_free(sc, fw_event);
562c12c399aSSascha Wildner }
563c12c399aSSascha Wildner 
564c12c399aSSascha Wildner void
mpssas_firmware_event_work(void * arg,int pending)565c12c399aSSascha Wildner mpssas_firmware_event_work(void *arg, int pending)
566c12c399aSSascha Wildner {
567c12c399aSSascha Wildner 	struct mps_fw_event_work *fw_event;
568c12c399aSSascha Wildner 	struct mps_softc *sc;
569c12c399aSSascha Wildner 
570c12c399aSSascha Wildner 	sc = (struct mps_softc *)arg;
571c12c399aSSascha Wildner 	mps_lock(sc);
572c12c399aSSascha Wildner 	while ((fw_event = TAILQ_FIRST(&sc->sassc->ev_queue)) != NULL) {
573c12c399aSSascha Wildner 		TAILQ_REMOVE(&sc->sassc->ev_queue, fw_event, ev_link);
574c12c399aSSascha Wildner 		mpssas_fw_work(sc, fw_event);
575c12c399aSSascha Wildner 	}
576c12c399aSSascha Wildner 	mps_unlock(sc);
577c12c399aSSascha Wildner }
578c12c399aSSascha Wildner 
579c12c399aSSascha Wildner static int
mpssas_add_device(struct mps_softc * sc,u16 handle,u8 linkrate)580c12c399aSSascha Wildner mpssas_add_device(struct mps_softc *sc, u16 handle, u8 linkrate){
581c12c399aSSascha Wildner 	char devstring[80];
582c12c399aSSascha Wildner 	struct mpssas_softc *sassc;
583c12c399aSSascha Wildner 	struct mpssas_target *targ;
584c12c399aSSascha Wildner 	Mpi2ConfigReply_t mpi_reply;
585c12c399aSSascha Wildner 	Mpi2SasDevicePage0_t config_page;
586c12c399aSSascha Wildner 	uint64_t sas_address, sata_sas_address;
587c12c399aSSascha Wildner 	uint64_t parent_sas_address = 0;
588c12c399aSSascha Wildner 	u16 ioc_pg8_flags = le16toh(sc->ioc_pg8.Flags);
589c12c399aSSascha Wildner 	u32 device_info, parent_devinfo = 0;
590c12c399aSSascha Wildner 	unsigned int id;
591c12c399aSSascha Wildner 	int ret;
592c12c399aSSascha Wildner 	int error = 0;
593c12c399aSSascha Wildner 
594c12c399aSSascha Wildner 	sassc = sc->sassc;
595c12c399aSSascha Wildner 	mpssas_startup_increment(sassc);
596c12c399aSSascha Wildner 	if ((mps_config_get_sas_device_pg0(sc, &mpi_reply, &config_page,
597c12c399aSSascha Wildner 	     MPI2_SAS_DEVICE_PGAD_FORM_HANDLE, handle))) {
598c12c399aSSascha Wildner 		kprintf("%s: error reading SAS device page0\n", __func__);
599c12c399aSSascha Wildner 		error = ENXIO;
600c12c399aSSascha Wildner 		goto out;
601c12c399aSSascha Wildner 	}
602c12c399aSSascha Wildner 
603c12c399aSSascha Wildner 	device_info = le32toh(config_page.DeviceInfo);
604c12c399aSSascha Wildner 
605c12c399aSSascha Wildner 	if (((device_info & MPI2_SAS_DEVICE_INFO_SMP_TARGET) == 0)
606c12c399aSSascha Wildner 	 && (config_page.ParentDevHandle != 0)) {
607c12c399aSSascha Wildner 		Mpi2ConfigReply_t tmp_mpi_reply;
608c12c399aSSascha Wildner 		Mpi2SasDevicePage0_t parent_config_page;
609c12c399aSSascha Wildner 
610c12c399aSSascha Wildner 		if ((mps_config_get_sas_device_pg0(sc, &tmp_mpi_reply,
611c12c399aSSascha Wildner 		     &parent_config_page, MPI2_SAS_DEVICE_PGAD_FORM_HANDLE,
612c12c399aSSascha Wildner 		     le16toh(config_page.ParentDevHandle)))) {
613c12c399aSSascha Wildner 			kprintf("%s: error reading SAS device %#x page0\n",
614c12c399aSSascha Wildner 			       __func__, le16toh(config_page.ParentDevHandle));
615c12c399aSSascha Wildner 		} else {
616c12c399aSSascha Wildner 			parent_sas_address = parent_config_page.SASAddress.High;
617c12c399aSSascha Wildner 			parent_sas_address = (parent_sas_address << 32) |
618c12c399aSSascha Wildner 				parent_config_page.SASAddress.Low;
619c12c399aSSascha Wildner 			parent_devinfo = le32toh(parent_config_page.DeviceInfo);
620c12c399aSSascha Wildner 		}
621c12c399aSSascha Wildner 	}
622c12c399aSSascha Wildner 	/* TODO Check proper endianess */
623c12c399aSSascha Wildner 	sas_address = config_page.SASAddress.High;
624c12c399aSSascha Wildner 	sas_address = (sas_address << 32) |
625c12c399aSSascha Wildner 	    config_page.SASAddress.Low;
626c12c399aSSascha Wildner 
627c12c399aSSascha Wildner 	if ((ioc_pg8_flags & MPI2_IOCPAGE8_FLAGS_MASK_MAPPING_MODE)
628c12c399aSSascha Wildner 		    == MPI2_IOCPAGE8_FLAGS_DEVICE_PERSISTENCE_MAPPING) {
629c12c399aSSascha Wildner 		if (device_info & MPI2_SAS_DEVICE_INFO_SATA_DEVICE) {
630c12c399aSSascha Wildner 			ret = mpssas_get_sas_address_for_sata_disk(sc,
631c12c399aSSascha Wildner 			    &sata_sas_address, handle, device_info);
632c12c399aSSascha Wildner 			if (!ret)
633c12c399aSSascha Wildner 				id = mps_mapping_get_sas_id(sc,
634c12c399aSSascha Wildner 				    sata_sas_address, handle);
635c12c399aSSascha Wildner 			else
636c12c399aSSascha Wildner 				id = mps_mapping_get_sas_id(sc,
637c12c399aSSascha Wildner 				    sas_address, handle);
638c12c399aSSascha Wildner 		} else
639c12c399aSSascha Wildner 			id = mps_mapping_get_sas_id(sc, sas_address,
640c12c399aSSascha Wildner 			    handle);
641c12c399aSSascha Wildner 	} else
642c12c399aSSascha Wildner 		id = mps_mapping_get_sas_id(sc, sas_address, handle);
643c12c399aSSascha Wildner 
644c12c399aSSascha Wildner 	if (id == MPS_MAP_BAD_ID) {
645c12c399aSSascha Wildner 		kprintf("failure at %s:%d/%s()! Could not get ID for device "
646c12c399aSSascha Wildner 		    "with handle 0x%04x\n", __FILE__, __LINE__, __func__,
647c12c399aSSascha Wildner 		    handle);
648c12c399aSSascha Wildner 		error = ENXIO;
649c12c399aSSascha Wildner 		goto out;
650c12c399aSSascha Wildner 	}
651*43f7c553SMatthew Dillon 	mps_dprint(sc, MPS_INFO, "SAS Address from SAS device page0 = %jx\n",
652c12c399aSSascha Wildner 		   sas_address);
653c12c399aSSascha Wildner 	targ = &sassc->targets[id];
654c12c399aSSascha Wildner 	targ->devinfo = device_info;
655c12c399aSSascha Wildner 	targ->devname = le32toh(config_page.DeviceName.High);
656c12c399aSSascha Wildner 	targ->devname = (targ->devname << 32) |
657c12c399aSSascha Wildner 	    le32toh(config_page.DeviceName.Low);
658c12c399aSSascha Wildner 	targ->encl_handle = le16toh(config_page.EnclosureHandle);
659c12c399aSSascha Wildner 	targ->encl_slot = le16toh(config_page.Slot);
660c12c399aSSascha Wildner 	targ->handle = handle;
661c12c399aSSascha Wildner 	targ->parent_handle = le16toh(config_page.ParentDevHandle);
662c12c399aSSascha Wildner 	targ->sasaddr = mps_to_u64(&config_page.SASAddress);
663c12c399aSSascha Wildner 	targ->parent_sasaddr = le64toh(parent_sas_address);
664c12c399aSSascha Wildner 	targ->parent_devinfo = parent_devinfo;
665c12c399aSSascha Wildner 	targ->tid = id;
666c12c399aSSascha Wildner 	targ->linkrate = (linkrate>>4);
667c12c399aSSascha Wildner 	targ->flags = 0;
668c12c399aSSascha Wildner 	TAILQ_INIT(&targ->commands);
669c12c399aSSascha Wildner 	TAILQ_INIT(&targ->timedout_commands);
670c12c399aSSascha Wildner 	SLIST_INIT(&targ->luns);
671c12c399aSSascha Wildner 	mps_describe_devinfo(targ->devinfo, devstring, 80);
672*43f7c553SMatthew Dillon 	mps_dprint(sc, MPS_INFO, "Found device <%s> <%s> <0x%04x> <%d/%d>\n", devstring,
673c12c399aSSascha Wildner 	    mps_describe_table(mps_linkrate_names, targ->linkrate),
674c12c399aSSascha Wildner 	    targ->handle, targ->encl_handle, targ->encl_slot);
675c12c399aSSascha Wildner 	if ((sassc->flags & MPSSAS_IN_STARTUP) == 0)
676c12c399aSSascha Wildner 		mpssas_rescan_target(sc, targ);
677*43f7c553SMatthew Dillon 	mps_dprint(sc, MPS_INFO, "Target id 0x%x added\n", targ->tid);
678c12c399aSSascha Wildner out:
679c12c399aSSascha Wildner 	mpssas_startup_decrement(sassc);
680c12c399aSSascha Wildner 	return (error);
681c12c399aSSascha Wildner 
682c12c399aSSascha Wildner }
683c12c399aSSascha Wildner 
684c12c399aSSascha Wildner int
mpssas_get_sas_address_for_sata_disk(struct mps_softc * sc,u64 * sas_address,u16 handle,u32 device_info)685c12c399aSSascha Wildner mpssas_get_sas_address_for_sata_disk(struct mps_softc *sc,
686c12c399aSSascha Wildner     u64 *sas_address, u16 handle, u32 device_info)
687c12c399aSSascha Wildner {
688c12c399aSSascha Wildner 	Mpi2SataPassthroughReply_t mpi_reply;
689c12c399aSSascha Wildner 	int i, rc, try_count;
690c12c399aSSascha Wildner 	u32 *bufferptr;
691c12c399aSSascha Wildner 	union _sata_sas_address hash_address;
692c12c399aSSascha Wildner 	struct _ata_identify_device_data ata_identify;
693c12c399aSSascha Wildner 	u8 buffer[MPT2SAS_MN_LEN + MPT2SAS_SN_LEN];
694c12c399aSSascha Wildner 	u32 ioc_status;
695c12c399aSSascha Wildner 	u8 sas_status;
696c12c399aSSascha Wildner 
697c12c399aSSascha Wildner 	memset(&ata_identify, 0, sizeof(ata_identify));
698c12c399aSSascha Wildner 	try_count = 0;
699c12c399aSSascha Wildner 	do {
700c12c399aSSascha Wildner 		rc = mpssas_get_sata_identify(sc, handle, &mpi_reply,
701c12c399aSSascha Wildner 		    (char *)&ata_identify, sizeof(ata_identify), device_info);
702c12c399aSSascha Wildner 		try_count++;
703c12c399aSSascha Wildner 		ioc_status = le16toh(mpi_reply.IOCStatus)
704c12c399aSSascha Wildner 		    & MPI2_IOCSTATUS_MASK;
705c12c399aSSascha Wildner 		sas_status = mpi_reply.SASStatus;
706c12c399aSSascha Wildner 	} while ((rc == -EAGAIN || ioc_status || sas_status) &&
707c12c399aSSascha Wildner 	    (try_count < 5));
708c12c399aSSascha Wildner 
709c12c399aSSascha Wildner 	if (rc == 0 && !ioc_status && !sas_status) {
710c12c399aSSascha Wildner 		mps_dprint(sc, MPS_INFO, "%s: got SATA identify successfully "
711c12c399aSSascha Wildner 			   "for handle = 0x%x with try_count = %d\n",
712c12c399aSSascha Wildner 			   __func__, handle, try_count);
713c12c399aSSascha Wildner 	} else {
714c12c399aSSascha Wildner 		mps_dprint(sc, MPS_INFO, "%s: handle = 0x%x failed\n",
715c12c399aSSascha Wildner 			   __func__, handle);
716c12c399aSSascha Wildner 		return -1;
717c12c399aSSascha Wildner 	}
718c12c399aSSascha Wildner 	/* Copy & byteswap the 40 byte model number to a buffer */
719c12c399aSSascha Wildner 	for (i = 0; i < MPT2SAS_MN_LEN; i += 2) {
720c12c399aSSascha Wildner 		buffer[i] = ((u8 *)ata_identify.model_number)[i + 1];
721c12c399aSSascha Wildner 		buffer[i + 1] = ((u8 *)ata_identify.model_number)[i];
722c12c399aSSascha Wildner 	}
723c12c399aSSascha Wildner 	/* Copy & byteswap the 20 byte serial number to a buffer */
724c12c399aSSascha Wildner 	for (i = 0; i < MPT2SAS_SN_LEN; i += 2) {
725c12c399aSSascha Wildner 		buffer[MPT2SAS_MN_LEN + i] =
726c12c399aSSascha Wildner 			((u8 *)ata_identify.serial_number)[i + 1];
727c12c399aSSascha Wildner 		buffer[MPT2SAS_MN_LEN + i + 1] =
728c12c399aSSascha Wildner 			((u8 *)ata_identify.serial_number)[i];
729c12c399aSSascha Wildner 	}
730c12c399aSSascha Wildner 	bufferptr = (u32 *)buffer;
731c12c399aSSascha Wildner 	/* There are 60 bytes to hash down to 8. 60 isn't divisible by 8,
732c12c399aSSascha Wildner 	 * so loop through the first 56 bytes (7*8),
733c12c399aSSascha Wildner 	 * and then add in the last dword.
734c12c399aSSascha Wildner 	 */
735c12c399aSSascha Wildner 	hash_address.word.low  = 0;
736c12c399aSSascha Wildner 	hash_address.word.high = 0;
737c12c399aSSascha Wildner 	for (i = 0; (i < ((MPT2SAS_MN_LEN+MPT2SAS_SN_LEN)/8)); i++) {
738c12c399aSSascha Wildner 		hash_address.word.low += *bufferptr;
739c12c399aSSascha Wildner 		bufferptr++;
740c12c399aSSascha Wildner 		hash_address.word.high += *bufferptr;
741c12c399aSSascha Wildner 		bufferptr++;
742c12c399aSSascha Wildner 	}
743c12c399aSSascha Wildner 	/* Add the last dword */
744c12c399aSSascha Wildner 	hash_address.word.low += *bufferptr;
745c12c399aSSascha Wildner 	/* Make sure the hash doesn't start with 5, because it could clash
746c12c399aSSascha Wildner 	 * with a SAS address. Change 5 to a D.
747c12c399aSSascha Wildner 	 */
748c12c399aSSascha Wildner 	if ((hash_address.word.high & 0x000000F0) == (0x00000050))
749c12c399aSSascha Wildner 		hash_address.word.high |= 0x00000080;
750c12c399aSSascha Wildner 	*sas_address = (u64)hash_address.wwid[0] << 56 |
751c12c399aSSascha Wildner 	    (u64)hash_address.wwid[1] << 48 | (u64)hash_address.wwid[2] << 40 |
752c12c399aSSascha Wildner 	    (u64)hash_address.wwid[3] << 32 | (u64)hash_address.wwid[4] << 24 |
753c12c399aSSascha Wildner 	    (u64)hash_address.wwid[5] << 16 | (u64)hash_address.wwid[6] <<  8 |
754c12c399aSSascha Wildner 	    (u64)hash_address.wwid[7];
755c12c399aSSascha Wildner 	return 0;
756c12c399aSSascha Wildner }
757c12c399aSSascha Wildner 
758c12c399aSSascha Wildner static int
mpssas_get_sata_identify(struct mps_softc * sc,u16 handle,Mpi2SataPassthroughReply_t * mpi_reply,char * id_buffer,int sz,u32 devinfo)759c12c399aSSascha Wildner mpssas_get_sata_identify(struct mps_softc *sc, u16 handle,
760c12c399aSSascha Wildner     Mpi2SataPassthroughReply_t *mpi_reply, char *id_buffer, int sz, u32 devinfo)
761c12c399aSSascha Wildner {
762c12c399aSSascha Wildner 	Mpi2SataPassthroughRequest_t *mpi_request;
763c12c399aSSascha Wildner 	Mpi2SataPassthroughReply_t *reply;
764c12c399aSSascha Wildner 	struct mps_command *cm;
765c12c399aSSascha Wildner 	char *buffer;
766c12c399aSSascha Wildner 	int error = 0;
767c12c399aSSascha Wildner 
7683eb4d33cSMatthew Dillon 	buffer = kmalloc( sz, M_MPT2, M_INTWAIT | M_ZERO);
769c12c399aSSascha Wildner 	if (!buffer)
770c12c399aSSascha Wildner 		return ENOMEM;
771c12c399aSSascha Wildner 
772df8658e0SSascha Wildner 	if ((cm = mps_alloc_command(sc)) == NULL) {
773df8658e0SSascha Wildner 		kfree(buffer, M_MPT2);
774c12c399aSSascha Wildner 		return (EBUSY);
775df8658e0SSascha Wildner 	}
776c12c399aSSascha Wildner 	mpi_request = (MPI2_SATA_PASSTHROUGH_REQUEST *)cm->cm_req;
777c12c399aSSascha Wildner 	bzero(mpi_request,sizeof(MPI2_SATA_PASSTHROUGH_REQUEST));
778c12c399aSSascha Wildner 	mpi_request->Function = MPI2_FUNCTION_SATA_PASSTHROUGH;
779c12c399aSSascha Wildner 	mpi_request->VF_ID = 0;
780c12c399aSSascha Wildner 	mpi_request->DevHandle = htole16(handle);
781c12c399aSSascha Wildner 	mpi_request->PassthroughFlags = (MPI2_SATA_PT_REQ_PT_FLAGS_PIO |
782c12c399aSSascha Wildner 	    MPI2_SATA_PT_REQ_PT_FLAGS_READ);
783c12c399aSSascha Wildner 	mpi_request->DataLength = htole32(sz);
784c12c399aSSascha Wildner 	mpi_request->CommandFIS[0] = 0x27;
785c12c399aSSascha Wildner 	mpi_request->CommandFIS[1] = 0x80;
786c12c399aSSascha Wildner 	mpi_request->CommandFIS[2] =  (devinfo &
787c12c399aSSascha Wildner 	    MPI2_SAS_DEVICE_INFO_ATAPI_DEVICE) ? 0xA1 : 0xEC;
788c12c399aSSascha Wildner 	cm->cm_sge = &mpi_request->SGL;
789c12c399aSSascha Wildner 	cm->cm_sglsize = sizeof(MPI2_SGE_IO_UNION);
790c12c399aSSascha Wildner 	cm->cm_flags = MPS_CM_FLAGS_SGE_SIMPLE | MPS_CM_FLAGS_DATAIN;
791c12c399aSSascha Wildner 	cm->cm_desc.Default.RequestFlags = MPI2_REQ_DESCRIPT_FLAGS_DEFAULT_TYPE;
792c12c399aSSascha Wildner 	cm->cm_data = buffer;
793c12c399aSSascha Wildner 	cm->cm_length = htole32(sz);
794c12c399aSSascha Wildner 	error = mps_request_polled(sc, cm);
795c12c399aSSascha Wildner 	reply = (Mpi2SataPassthroughReply_t *)cm->cm_reply;
796c12c399aSSascha Wildner 	if (error || (reply == NULL)) {
797c12c399aSSascha Wildner 		/* FIXME */
798c12c399aSSascha Wildner 		/* If the poll returns error then we need to do diag reset */
799c12c399aSSascha Wildner 		kprintf("%s: poll for page completed with error %d",
800c12c399aSSascha Wildner 		    __func__, error);
801c12c399aSSascha Wildner 		error = ENXIO;
802c12c399aSSascha Wildner 		goto out;
803c12c399aSSascha Wildner 	}
804c12c399aSSascha Wildner 	bcopy(buffer, id_buffer, sz);
805c12c399aSSascha Wildner 	bcopy(reply, mpi_reply, sizeof(Mpi2SataPassthroughReply_t));
806c12c399aSSascha Wildner 	if ((reply->IOCStatus & MPI2_IOCSTATUS_MASK) !=
807c12c399aSSascha Wildner 	    MPI2_IOCSTATUS_SUCCESS) {
808c12c399aSSascha Wildner 		kprintf("%s: error reading SATA PASSTHRU; iocstatus = 0x%x\n",
809c12c399aSSascha Wildner 		    __func__, reply->IOCStatus);
810c12c399aSSascha Wildner 		error = ENXIO;
811c12c399aSSascha Wildner 		goto out;
812c12c399aSSascha Wildner 	}
813c12c399aSSascha Wildner out:
814c12c399aSSascha Wildner 	mps_free_command(sc, cm);
815c12c399aSSascha Wildner 	kfree(buffer, M_MPT2);
816c12c399aSSascha Wildner 	return (error);
817c12c399aSSascha Wildner }
818c12c399aSSascha Wildner 
819c12c399aSSascha Wildner static int
mpssas_volume_add(struct mps_softc * sc,u16 handle)820*43f7c553SMatthew Dillon mpssas_volume_add(struct mps_softc *sc, u16 handle)
821c12c399aSSascha Wildner {
822c12c399aSSascha Wildner 	struct mpssas_softc *sassc;
823c12c399aSSascha Wildner 	struct mpssas_target *targ;
824c12c399aSSascha Wildner 	u64 wwid;
825c12c399aSSascha Wildner 	unsigned int id;
826c12c399aSSascha Wildner 	int error = 0;
827c12c399aSSascha Wildner 
828c12c399aSSascha Wildner 	sassc = sc->sassc;
829c12c399aSSascha Wildner 	mpssas_startup_increment(sassc);
830c12c399aSSascha Wildner 	mps_config_get_volume_wwid(sc, handle, &wwid);
831c12c399aSSascha Wildner 	if (!wwid) {
832c12c399aSSascha Wildner 		kprintf("%s: invalid WWID; cannot add volume to mapping table\n",
833c12c399aSSascha Wildner 		    __func__);
834c12c399aSSascha Wildner 		error = ENXIO;
835c12c399aSSascha Wildner 		goto out;
836c12c399aSSascha Wildner 	}
837c12c399aSSascha Wildner 
838c12c399aSSascha Wildner 	id = mps_mapping_get_raid_id(sc, wwid, handle);
839c12c399aSSascha Wildner 	if (id == MPS_MAP_BAD_ID) {
840c12c399aSSascha Wildner 		kprintf("%s: could not get ID for volume with handle 0x%04x and "
841c12c399aSSascha Wildner 		    "WWID 0x%016llx\n", __func__, handle,
842c12c399aSSascha Wildner 		    (unsigned long long)wwid);
843c12c399aSSascha Wildner 		error = ENXIO;
844c12c399aSSascha Wildner 		goto out;
845c12c399aSSascha Wildner 	}
846c12c399aSSascha Wildner 
847c12c399aSSascha Wildner 	targ = &sassc->targets[id];
848c12c399aSSascha Wildner 	targ->tid = id;
849c12c399aSSascha Wildner 	targ->handle = handle;
850c12c399aSSascha Wildner 	targ->devname = wwid;
851c12c399aSSascha Wildner 	TAILQ_INIT(&targ->commands);
852c12c399aSSascha Wildner 	TAILQ_INIT(&targ->timedout_commands);
853c12c399aSSascha Wildner 	SLIST_INIT(&targ->luns);
854c12c399aSSascha Wildner 	if ((sassc->flags & MPSSAS_IN_STARTUP) == 0)
855c12c399aSSascha Wildner 		mpssas_rescan_target(sc, targ);
856c12c399aSSascha Wildner 	mps_dprint(sc, MPS_INFO, "RAID target id %d added (WWID = 0x%jx)\n",
857c12c399aSSascha Wildner 	    targ->tid, wwid);
858c12c399aSSascha Wildner out:
859c12c399aSSascha Wildner 	mpssas_startup_decrement(sassc);
860c12c399aSSascha Wildner 	return (error);
861c12c399aSSascha Wildner }
862c12c399aSSascha Wildner 
863c12c399aSSascha Wildner /**
864c12c399aSSascha Wildner  * mpssas_ir_shutdown - IR shutdown notification
865c12c399aSSascha Wildner  * @sc: per adapter object
866c12c399aSSascha Wildner  *
867c12c399aSSascha Wildner  * Sending RAID Action to alert the Integrated RAID subsystem of the IOC that
868c12c399aSSascha Wildner  * the host system is shutting down.
869c12c399aSSascha Wildner  *
870c12c399aSSascha Wildner  * Return nothing.
871c12c399aSSascha Wildner  */
872c12c399aSSascha Wildner void
mpssas_ir_shutdown(struct mps_softc * sc)873c12c399aSSascha Wildner mpssas_ir_shutdown(struct mps_softc *sc)
874c12c399aSSascha Wildner {
875c12c399aSSascha Wildner 	u16 volume_mapping_flags;
876c12c399aSSascha Wildner 	u16 ioc_pg8_flags = le16toh(sc->ioc_pg8.Flags);
877c12c399aSSascha Wildner 	struct dev_mapping_table *mt_entry;
878c12c399aSSascha Wildner 	u32 start_idx, end_idx;
879c12c399aSSascha Wildner 	unsigned int id, found_volume = 0;
880c12c399aSSascha Wildner 	struct mps_command *cm;
881c12c399aSSascha Wildner 	Mpi2RaidActionRequest_t	*action;
882c12c399aSSascha Wildner 
883cb7593b0SSepherosa Ziehau 	mps_lock(sc);
884cb7593b0SSepherosa Ziehau 
885c12c399aSSascha Wildner 	mps_dprint(sc, MPS_TRACE, "%s\n", __func__);
886c12c399aSSascha Wildner 
887c12c399aSSascha Wildner 	/* is IR firmware build loaded? */
888c12c399aSSascha Wildner 	if (!sc->ir_firmware)
889cb7593b0SSepherosa Ziehau 		goto back;
890c12c399aSSascha Wildner 
891c12c399aSSascha Wildner 	/* are there any volumes?  Look at IR target IDs. */
892c12c399aSSascha Wildner 	// TODO-later, this should be looked up in the RAID config structure
893c12c399aSSascha Wildner 	// when it is implemented.
894c12c399aSSascha Wildner 	volume_mapping_flags = le16toh(sc->ioc_pg8.IRVolumeMappingFlags) &
895c12c399aSSascha Wildner 	    MPI2_IOCPAGE8_IRFLAGS_MASK_VOLUME_MAPPING_MODE;
896c12c399aSSascha Wildner 	if (volume_mapping_flags == MPI2_IOCPAGE8_IRFLAGS_LOW_VOLUME_MAPPING) {
897c12c399aSSascha Wildner 		start_idx = 0;
898c12c399aSSascha Wildner 		if (ioc_pg8_flags & MPI2_IOCPAGE8_FLAGS_RESERVED_TARGETID_0)
899c12c399aSSascha Wildner 			start_idx = 1;
900c12c399aSSascha Wildner 	} else
901c12c399aSSascha Wildner 		start_idx = sc->max_devices - sc->max_volumes;
902c12c399aSSascha Wildner 	end_idx = start_idx + sc->max_volumes - 1;
903c12c399aSSascha Wildner 
904c12c399aSSascha Wildner 	for (id = start_idx; id < end_idx; id++) {
905c12c399aSSascha Wildner 		mt_entry = &sc->mapping_table[id];
906c12c399aSSascha Wildner 		if ((mt_entry->physical_id != 0) &&
907c12c399aSSascha Wildner 		    (mt_entry->missing_count == 0)) {
908c12c399aSSascha Wildner 			found_volume = 1;
909c12c399aSSascha Wildner 			break;
910c12c399aSSascha Wildner 		}
911c12c399aSSascha Wildner 	}
912c12c399aSSascha Wildner 
913c12c399aSSascha Wildner 	if (!found_volume)
914cb7593b0SSepherosa Ziehau 		goto back;
915c12c399aSSascha Wildner 
916c12c399aSSascha Wildner 	if ((cm = mps_alloc_command(sc)) == NULL) {
917c12c399aSSascha Wildner 		kprintf("%s: command alloc failed\n", __func__);
918cb7593b0SSepherosa Ziehau 		goto back;
919c12c399aSSascha Wildner 	}
920c12c399aSSascha Wildner 
921c12c399aSSascha Wildner 	action = (MPI2_RAID_ACTION_REQUEST *)cm->cm_req;
922c12c399aSSascha Wildner 	action->Function = MPI2_FUNCTION_RAID_ACTION;
923c12c399aSSascha Wildner 	action->Action = MPI2_RAID_ACTION_SYSTEM_SHUTDOWN_INITIATED;
924c12c399aSSascha Wildner 	cm->cm_desc.Default.RequestFlags = MPI2_REQ_DESCRIPT_FLAGS_DEFAULT_TYPE;
925*43f7c553SMatthew Dillon 	mps_lock(sc);
926c12c399aSSascha Wildner 	mps_request_polled(sc, cm);
927*43f7c553SMatthew Dillon 	mps_unlock(sc);
928c12c399aSSascha Wildner 
929c12c399aSSascha Wildner 	/*
930c12c399aSSascha Wildner 	 * Don't check for reply, just leave.
931c12c399aSSascha Wildner 	 */
932c12c399aSSascha Wildner 	if (cm)
933c12c399aSSascha Wildner 		mps_free_command(sc, cm);
934cb7593b0SSepherosa Ziehau 
935cb7593b0SSepherosa Ziehau back:
936cb7593b0SSepherosa Ziehau 	mps_unlock(sc);
937c12c399aSSascha Wildner }
938