2 * Copyright (c) 2011, 2012 LSI Corp.
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
14 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
26 * LSI MPT-Fusion Host Adapter FreeBSD
29 #include <sys/cdefs.h>
30 __FBSDID("$FreeBSD$");
32 /* TODO Move headers to mpsvar */
33 #include <sys/types.h>
34 #include <sys/param.h>
36 #include <sys/mutex.h>
37 #include <sys/systm.h>
38 #include <sys/kernel.h>
39 #include <sys/malloc.h>
40 #include <sys/kthread.h>
41 #include <sys/taskqueue.h>
43 #include <sys/endian.h>
44 #include <sys/sysctl.h>
45 #include <sys/eventhandler.h>
47 #include <machine/bus.h>
48 #include <machine/resource.h>
49 #include <dev/mps/mpi/mpi2_type.h>
50 #include <dev/mps/mpi/mpi2.h>
51 #include <dev/mps/mpi/mpi2_ioc.h>
52 #include <dev/mps/mpi/mpi2_sas.h>
53 #include <dev/mps/mpi/mpi2_cnfg.h>
54 #include <dev/mps/mpi/mpi2_init.h>
55 #include <dev/mps/mpi/mpi2_tool.h>
56 #include <dev/mps/mps_ioctl.h>
57 #include <dev/mps/mpsvar.h>
58 #include <dev/mps/mps_mapping.h>
61 * _mapping_clear_entry - Clear a particular mapping entry.
62 * @map_entry: map table entry
67 _mapping_clear_map_entry(struct dev_mapping_table *map_entry)
69 map_entry->physical_id = 0;
70 map_entry->device_info = 0;
71 map_entry->phy_bits = 0;
72 map_entry->dpm_entry_num = MPS_DPM_BAD_IDX;
73 map_entry->dev_handle = 0;
74 map_entry->channel = -1;
76 map_entry->missing_count = 0;
77 map_entry->init_complete = 0;
78 map_entry->TLR_bits = (u8)MPI2_SCSIIO_CONTROL_NO_TLR;
82 * _mapping_clear_enc_entry - Clear a particular enclosure table entry.
83 * @enc_entry: enclosure table entry
88 _mapping_clear_enc_entry(struct enc_mapping_table *enc_entry)
90 enc_entry->enclosure_id = 0;
91 enc_entry->start_index = MPS_MAPTABLE_BAD_IDX;
92 enc_entry->phy_bits = 0;
93 enc_entry->dpm_entry_num = MPS_DPM_BAD_IDX;
94 enc_entry->enc_handle = 0;
95 enc_entry->num_slots = 0;
96 enc_entry->start_slot = 0;
97 enc_entry->missing_count = 0;
98 enc_entry->removal_flag = 0;
99 enc_entry->skip_search = 0;
100 enc_entry->init_complete = 0;
104 * _mapping_commit_enc_entry - write a particular enc entry in DPM page0.
105 * @sc: per adapter object
106 * @enc_entry: enclosure table entry
108 * Returns 0 for success, non-zero for failure.
111 _mapping_commit_enc_entry(struct mps_softc *sc,
112 struct enc_mapping_table *et_entry)
114 Mpi2DriverMap0Entry_t *dpm_entry;
115 struct dev_mapping_table *mt_entry;
116 Mpi2ConfigReply_t mpi_reply;
117 Mpi2DriverMappingPage0_t config_page;
119 if (!sc->is_dpm_enable)
122 memset(&config_page, 0, sizeof(Mpi2DriverMappingPage0_t));
123 memcpy(&config_page.Header, (u8 *) sc->dpm_pg0,
124 sizeof(MPI2_CONFIG_EXTENDED_PAGE_HEADER));
125 dpm_entry = (Mpi2DriverMap0Entry_t *)((u8 *)sc->dpm_pg0 +
126 sizeof(MPI2_CONFIG_EXTENDED_PAGE_HEADER));
127 dpm_entry += et_entry->dpm_entry_num;
128 dpm_entry->PhysicalIdentifier.Low =
129 ( 0xFFFFFFFF & et_entry->enclosure_id);
130 dpm_entry->PhysicalIdentifier.High =
131 ( et_entry->enclosure_id >> 32);
132 mt_entry = &sc->mapping_table[et_entry->start_index];
133 dpm_entry->DeviceIndex = htole16(mt_entry->id);
134 dpm_entry->MappingInformation = et_entry->num_slots;
135 dpm_entry->MappingInformation <<= MPI2_DRVMAP0_MAPINFO_SLOT_SHIFT;
136 dpm_entry->MappingInformation |= et_entry->missing_count;
137 dpm_entry->MappingInformation = htole16(dpm_entry->MappingInformation);
138 dpm_entry->PhysicalBitsMapping = htole32(et_entry->phy_bits);
139 dpm_entry->Reserved1 = 0;
141 memcpy(&config_page.Entry, (u8 *)dpm_entry,
142 sizeof(Mpi2DriverMap0Entry_t));
143 if (mps_config_set_dpm_pg0(sc, &mpi_reply, &config_page,
144 et_entry->dpm_entry_num)) {
145 printf("%s: write of dpm entry %d for enclosure failed\n",
146 __func__, et_entry->dpm_entry_num);
147 dpm_entry->MappingInformation = le16toh(dpm_entry->
149 dpm_entry->DeviceIndex = le16toh(dpm_entry->DeviceIndex);
150 dpm_entry->PhysicalBitsMapping =
151 le32toh(dpm_entry->PhysicalBitsMapping);
154 dpm_entry->MappingInformation = le16toh(dpm_entry->
156 dpm_entry->DeviceIndex = le16toh(dpm_entry->DeviceIndex);
157 dpm_entry->PhysicalBitsMapping =
158 le32toh(dpm_entry->PhysicalBitsMapping);
163 * _mapping_commit_map_entry - write a particular map table entry in DPM page0.
164 * @sc: per adapter object
165 * @enc_entry: enclosure table entry
167 * Returns 0 for success, non-zero for failure.
171 _mapping_commit_map_entry(struct mps_softc *sc,
172 struct dev_mapping_table *mt_entry)
174 Mpi2DriverMap0Entry_t *dpm_entry;
175 Mpi2ConfigReply_t mpi_reply;
176 Mpi2DriverMappingPage0_t config_page;
178 if (!sc->is_dpm_enable)
181 memset(&config_page, 0, sizeof(Mpi2DriverMappingPage0_t));
182 memcpy(&config_page.Header, (u8 *)sc->dpm_pg0,
183 sizeof(MPI2_CONFIG_EXTENDED_PAGE_HEADER));
184 dpm_entry = (Mpi2DriverMap0Entry_t *)((u8 *) sc->dpm_pg0 +
185 sizeof(MPI2_CONFIG_EXTENDED_PAGE_HEADER));
186 dpm_entry = dpm_entry + mt_entry->dpm_entry_num;
187 dpm_entry->PhysicalIdentifier.Low = (0xFFFFFFFF &
188 mt_entry->physical_id);
189 dpm_entry->PhysicalIdentifier.High = (mt_entry->physical_id >> 32);
190 dpm_entry->DeviceIndex = htole16(mt_entry->id);
191 dpm_entry->MappingInformation = htole16(mt_entry->missing_count);
192 dpm_entry->PhysicalBitsMapping = 0;
193 dpm_entry->Reserved1 = 0;
194 dpm_entry->MappingInformation = htole16(dpm_entry->MappingInformation);
195 memcpy(&config_page.Entry, (u8 *)dpm_entry,
196 sizeof(Mpi2DriverMap0Entry_t));
197 if (mps_config_set_dpm_pg0(sc, &mpi_reply, &config_page,
198 mt_entry->dpm_entry_num)) {
199 printf("%s: write of dpm entry %d for device failed\n",
200 __func__, mt_entry->dpm_entry_num);
201 dpm_entry->MappingInformation = le16toh(dpm_entry->
203 dpm_entry->DeviceIndex = le16toh(dpm_entry->DeviceIndex);
207 dpm_entry->MappingInformation = le16toh(dpm_entry->MappingInformation);
208 dpm_entry->DeviceIndex = le16toh(dpm_entry->DeviceIndex);
213 * _mapping_get_ir_maprange - get start and end index for IR map range.
214 * @sc: per adapter object
215 * @start_idx: place holder for start index
216 * @end_idx: place holder for end index
218 * The IR volumes can be mapped either at start or end of the mapping table
219 * this function gets the detail of where IR volume mapping starts and ends
220 * in the device mapping table
225 _mapping_get_ir_maprange(struct mps_softc *sc, u32 *start_idx, u32 *end_idx)
227 u16 volume_mapping_flags;
228 u16 ioc_pg8_flags = le16toh(sc->ioc_pg8.Flags);
230 volume_mapping_flags = le16toh(sc->ioc_pg8.IRVolumeMappingFlags) &
231 MPI2_IOCPAGE8_IRFLAGS_MASK_VOLUME_MAPPING_MODE;
232 if (volume_mapping_flags == MPI2_IOCPAGE8_IRFLAGS_LOW_VOLUME_MAPPING) {
234 if (ioc_pg8_flags & MPI2_IOCPAGE8_FLAGS_RESERVED_TARGETID_0)
237 *start_idx = sc->max_devices - sc->max_volumes;
238 *end_idx = *start_idx + sc->max_volumes - 1;
242 * _mapping_get_enc_idx_from_id - get enclosure index from enclosure ID
243 * @sc: per adapter object
244 * @enc_id: enclosure logical identifier
246 * Returns the index of enclosure entry on success or bad index.
249 _mapping_get_enc_idx_from_id(struct mps_softc *sc, u64 enc_id,
252 struct enc_mapping_table *et_entry;
255 for (enc_idx = 0; enc_idx < sc->num_enc_table_entries; enc_idx++) {
256 et_entry = &sc->enclosure_table[enc_idx];
257 if ((et_entry->enclosure_id == le64toh(enc_id)) &&
258 (!et_entry->phy_bits || (et_entry->phy_bits &
262 return MPS_ENCTABLE_BAD_IDX;
266 * _mapping_get_enc_idx_from_handle - get enclosure index from handle
267 * @sc: per adapter object
268 * @enc_id: enclosure handle
270 * Returns the index of enclosure entry on success or bad index.
273 _mapping_get_enc_idx_from_handle(struct mps_softc *sc, u16 handle)
275 struct enc_mapping_table *et_entry;
278 for (enc_idx = 0; enc_idx < sc->num_enc_table_entries; enc_idx++) {
279 et_entry = &sc->enclosure_table[enc_idx];
280 if (et_entry->missing_count)
282 if (et_entry->enc_handle == handle)
285 return MPS_ENCTABLE_BAD_IDX;
289 * _mapping_get_high_missing_et_idx - get missing enclosure index
290 * @sc: per adapter object
292 * Search through the enclosure table and identifies the enclosure entry
293 * with high missing count and returns it's index
295 * Returns the index of enclosure entry on success or bad index.
298 _mapping_get_high_missing_et_idx(struct mps_softc *sc)
300 struct enc_mapping_table *et_entry;
301 u8 high_missing_count = 0;
302 u8 enc_idx, high_idx = MPS_ENCTABLE_BAD_IDX;
304 for (enc_idx = 0; enc_idx < sc->num_enc_table_entries; enc_idx++) {
305 et_entry = &sc->enclosure_table[enc_idx];
306 if ((et_entry->missing_count > high_missing_count) &&
307 !et_entry->skip_search) {
308 high_missing_count = et_entry->missing_count;
316 * _mapping_get_high_missing_mt_idx - get missing map table index
317 * @sc: per adapter object
319 * Search through the map table and identifies the device entry
320 * with high missing count and returns it's index
322 * Returns the index of map table entry on success or bad index.
325 _mapping_get_high_missing_mt_idx(struct mps_softc *sc)
327 u32 map_idx, high_idx = MPS_ENCTABLE_BAD_IDX;
328 u8 high_missing_count = 0;
329 u32 start_idx, end_idx, start_idx_ir, end_idx_ir;
330 struct dev_mapping_table *mt_entry;
331 u16 ioc_pg8_flags = le16toh(sc->ioc_pg8.Flags);
336 end_idx = sc->max_devices;
337 if (ioc_pg8_flags & MPI2_IOCPAGE8_FLAGS_RESERVED_TARGETID_0)
340 _mapping_get_ir_maprange(sc, &start_idx_ir, &end_idx_ir);
341 if (start_idx == start_idx_ir)
342 start_idx = end_idx_ir + 1;
344 end_idx = start_idx_ir;
345 mt_entry = &sc->mapping_table[start_idx];
346 for (map_idx = start_idx; map_idx < end_idx; map_idx++, mt_entry++) {
347 if (mt_entry->missing_count > high_missing_count) {
348 high_missing_count = mt_entry->missing_count;
356 * _mapping_get_ir_mt_idx_from_wwid - get map table index from volume WWID
357 * @sc: per adapter object
358 * @wwid: world wide unique ID of the volume
360 * Returns the index of map table entry on success or bad index.
363 _mapping_get_ir_mt_idx_from_wwid(struct mps_softc *sc, u64 wwid)
365 u32 start_idx, end_idx, map_idx;
366 struct dev_mapping_table *mt_entry;
368 _mapping_get_ir_maprange(sc, &start_idx, &end_idx);
369 mt_entry = &sc->mapping_table[start_idx];
370 for (map_idx = start_idx; map_idx <= end_idx; map_idx++, mt_entry++)
371 if (mt_entry->physical_id == wwid)
374 return MPS_MAPTABLE_BAD_IDX;
378 * _mapping_get_mt_idx_from_id - get map table index from a device ID
379 * @sc: per adapter object
380 * @dev_id: device identifer (SAS Address)
382 * Returns the index of map table entry on success or bad index.
385 _mapping_get_mt_idx_from_id(struct mps_softc *sc, u64 dev_id)
388 struct dev_mapping_table *mt_entry;
390 for (map_idx = 0; map_idx < sc->max_devices; map_idx++) {
391 mt_entry = &sc->mapping_table[map_idx];
392 if (mt_entry->physical_id == dev_id)
395 return MPS_MAPTABLE_BAD_IDX;
399 * _mapping_get_ir_mt_idx_from_handle - get map table index from volume handle
400 * @sc: per adapter object
401 * @wwid: volume device handle
403 * Returns the index of map table entry on success or bad index.
406 _mapping_get_ir_mt_idx_from_handle(struct mps_softc *sc, u16 volHandle)
408 u32 start_idx, end_idx, map_idx;
409 struct dev_mapping_table *mt_entry;
411 _mapping_get_ir_maprange(sc, &start_idx, &end_idx);
412 mt_entry = &sc->mapping_table[start_idx];
413 for (map_idx = start_idx; map_idx <= end_idx; map_idx++, mt_entry++)
414 if (mt_entry->dev_handle == volHandle)
417 return MPS_MAPTABLE_BAD_IDX;
421 * _mapping_get_mt_idx_from_handle - get map table index from handle
422 * @sc: per adapter object
423 * @dev_id: device handle
425 * Returns the index of map table entry on success or bad index.
428 _mapping_get_mt_idx_from_handle(struct mps_softc *sc, u16 handle)
431 struct dev_mapping_table *mt_entry;
433 for (map_idx = 0; map_idx < sc->max_devices; map_idx++) {
434 mt_entry = &sc->mapping_table[map_idx];
435 if (mt_entry->dev_handle == handle)
438 return MPS_MAPTABLE_BAD_IDX;
442 * _mapping_get_free_ir_mt_idx - get first free index for a volume
443 * @sc: per adapter object
445 * Search through mapping table for free index for a volume and if no free
446 * index then looks for a volume with high mapping index
448 * Returns the index of map table entry on success or bad index.
451 _mapping_get_free_ir_mt_idx(struct mps_softc *sc)
453 u8 high_missing_count = 0;
454 u32 start_idx, end_idx, map_idx;
455 u32 high_idx = MPS_MAPTABLE_BAD_IDX;
456 struct dev_mapping_table *mt_entry;
458 _mapping_get_ir_maprange(sc, &start_idx, &end_idx);
460 mt_entry = &sc->mapping_table[start_idx];
461 for (map_idx = start_idx; map_idx <= end_idx; map_idx++, mt_entry++)
462 if (!(mt_entry->device_info & MPS_MAP_IN_USE))
465 mt_entry = &sc->mapping_table[start_idx];
466 for (map_idx = start_idx; map_idx <= end_idx; map_idx++, mt_entry++) {
467 if (mt_entry->missing_count > high_missing_count) {
468 high_missing_count = mt_entry->missing_count;
476 * _mapping_get_free_mt_idx - get first free index for a device
477 * @sc: per adapter object
478 * @start_idx: offset in the table to start search
480 * Returns the index of map table entry on success or bad index.
483 _mapping_get_free_mt_idx(struct mps_softc *sc, u32 start_idx)
485 u32 map_idx, max_idx = sc->max_devices;
486 struct dev_mapping_table *mt_entry = &sc->mapping_table[start_idx];
487 u16 volume_mapping_flags;
489 volume_mapping_flags = le16toh(sc->ioc_pg8.IRVolumeMappingFlags) &
490 MPI2_IOCPAGE8_IRFLAGS_MASK_VOLUME_MAPPING_MODE;
491 if (sc->ir_firmware && (volume_mapping_flags ==
492 MPI2_IOCPAGE8_IRFLAGS_HIGH_VOLUME_MAPPING))
493 max_idx -= sc->max_volumes;
494 for (map_idx = start_idx; map_idx < max_idx; map_idx++, mt_entry++)
495 if (!(mt_entry->device_info & (MPS_MAP_IN_USE |
499 return MPS_MAPTABLE_BAD_IDX;
503 * _mapping_get_dpm_idx_from_id - get DPM index from ID
504 * @sc: per adapter object
505 * @id: volume WWID or enclosure ID or device ID
507 * Returns the index of DPM entry on success or bad index.
510 _mapping_get_dpm_idx_from_id(struct mps_softc *sc, u64 id, u32 phy_bits)
513 uint64_t PhysicalIdentifier;
514 Mpi2DriverMap0Entry_t *dpm_entry;
516 dpm_entry = (Mpi2DriverMap0Entry_t *)((u8 *)sc->dpm_pg0 +
517 sizeof(MPI2_CONFIG_EXTENDED_PAGE_HEADER));
518 PhysicalIdentifier = dpm_entry->PhysicalIdentifier.High;
519 PhysicalIdentifier = (PhysicalIdentifier << 32) |
520 dpm_entry->PhysicalIdentifier.Low;
521 for (entry_num = 0; entry_num < sc->max_dpm_entries; entry_num++,
523 if ((id == PhysicalIdentifier) &&
524 (!phy_bits || !dpm_entry->PhysicalBitsMapping ||
525 (phy_bits & dpm_entry->PhysicalBitsMapping)))
528 return MPS_DPM_BAD_IDX;
533 * _mapping_get_free_dpm_idx - get first available DPM index
534 * @sc: per adapter object
536 * Returns the index of DPM entry on success or bad index.
539 _mapping_get_free_dpm_idx(struct mps_softc *sc)
543 for (entry_num = 0; entry_num < sc->max_dpm_entries; entry_num++) {
544 if (!sc->dpm_entry_used[entry_num])
547 return MPS_DPM_BAD_IDX;
551 * _mapping_update_ir_missing_cnt - Updates missing count for a volume
552 * @sc: per adapter object
553 * @map_idx: map table index of the volume
554 * @element: IR configuration change element
555 * @wwid: IR volume ID.
557 * Updates the missing count in the map table and in the DPM entry for a volume
562 _mapping_update_ir_missing_cnt(struct mps_softc *sc, u32 map_idx,
563 Mpi2EventIrConfigElement_t *element, u64 wwid)
565 struct dev_mapping_table *mt_entry;
566 u8 missing_cnt, reason = element->ReasonCode;
568 Mpi2DriverMap0Entry_t *dpm_entry;
570 if (!sc->is_dpm_enable)
572 mt_entry = &sc->mapping_table[map_idx];
573 if (reason == MPI2_EVENT_IR_CHANGE_RC_ADDED) {
574 mt_entry->missing_count = 0;
575 } else if (reason == MPI2_EVENT_IR_CHANGE_RC_VOLUME_CREATED) {
576 mt_entry->missing_count = 0;
577 mt_entry->init_complete = 0;
578 } else if ((reason == MPI2_EVENT_IR_CHANGE_RC_REMOVED) ||
579 (reason == MPI2_EVENT_IR_CHANGE_RC_VOLUME_DELETED)) {
580 if (!mt_entry->init_complete) {
581 if (mt_entry->missing_count < MPS_MAX_MISSING_COUNT)
582 mt_entry->missing_count++;
584 mt_entry->init_complete = 1;
586 if (!mt_entry->missing_count)
587 mt_entry->missing_count++;
588 mt_entry->dev_handle = 0;
591 dpm_idx = mt_entry->dpm_entry_num;
592 if (dpm_idx == MPS_DPM_BAD_IDX) {
593 if ((reason == MPI2_EVENT_IR_CHANGE_RC_ADDED) ||
594 (reason == MPI2_EVENT_IR_CHANGE_RC_REMOVED))
595 dpm_idx = _mapping_get_dpm_idx_from_id(sc,
596 mt_entry->physical_id, 0);
597 else if (reason == MPI2_EVENT_IR_CHANGE_RC_VOLUME_DELETED)
600 if (dpm_idx != MPS_DPM_BAD_IDX) {
601 dpm_entry = (Mpi2DriverMap0Entry_t *)((u8 *)sc->dpm_pg0 +
602 sizeof(MPI2_CONFIG_EXTENDED_PAGE_HEADER));
603 dpm_entry += dpm_idx;
604 missing_cnt = dpm_entry->MappingInformation &
605 MPI2_DRVMAP0_MAPINFO_MISSING_MASK;
606 if ((mt_entry->physical_id ==
607 le64toh((u64)dpm_entry->PhysicalIdentifier.High |
608 dpm_entry->PhysicalIdentifier.Low)) && (missing_cnt ==
609 mt_entry->missing_count))
610 mt_entry->init_complete = 1;
612 dpm_idx = _mapping_get_free_dpm_idx(sc);
613 mt_entry->init_complete = 0;
616 if ((dpm_idx != MPS_DPM_BAD_IDX) && !mt_entry->init_complete) {
617 mt_entry->init_complete = 1;
618 mt_entry->dpm_entry_num = dpm_idx;
619 dpm_entry = (Mpi2DriverMap0Entry_t *)((u8 *)sc->dpm_pg0 +
620 sizeof(MPI2_CONFIG_EXTENDED_PAGE_HEADER));
621 dpm_entry += dpm_idx;
622 dpm_entry->PhysicalIdentifier.Low =
623 (0xFFFFFFFF & mt_entry->physical_id);
624 dpm_entry->PhysicalIdentifier.High =
625 (mt_entry->physical_id >> 32);
626 dpm_entry->DeviceIndex = map_idx;
627 dpm_entry->MappingInformation = mt_entry->missing_count;
628 dpm_entry->PhysicalBitsMapping = 0;
629 dpm_entry->Reserved1 = 0;
630 sc->dpm_flush_entry[dpm_idx] = 1;
631 sc->dpm_entry_used[dpm_idx] = 1;
632 } else if (dpm_idx == MPS_DPM_BAD_IDX) {
633 printf("%s: no space to add entry in DPM table\n", __func__);
634 mt_entry->init_complete = 1;
639 * _mapping_add_to_removal_table - mark an entry for removal
640 * @sc: per adapter object
641 * @handle: Handle of enclosures/device/volume
643 * Adds the handle or DPM entry number in removal table.
648 _mapping_add_to_removal_table(struct mps_softc *sc, u16 handle,
651 struct map_removal_table *remove_entry;
653 u16 ioc_pg8_flags = le16toh(sc->ioc_pg8.Flags);
655 remove_entry = sc->removal_table;
657 for (i = 0; i < sc->max_devices; i++, remove_entry++) {
658 if (remove_entry->dev_handle || remove_entry->dpm_entry_num !=
661 if ((ioc_pg8_flags & MPI2_IOCPAGE8_FLAGS_MASK_MAPPING_MODE) ==
662 MPI2_IOCPAGE8_FLAGS_ENCLOSURE_SLOT_MAPPING) {
664 remove_entry->dpm_entry_num = dpm_idx;
665 if (remove_entry->dpm_entry_num == MPS_DPM_BAD_IDX)
666 remove_entry->dev_handle = handle;
667 } else if ((ioc_pg8_flags &
668 MPI2_IOCPAGE8_FLAGS_MASK_MAPPING_MODE) ==
669 MPI2_IOCPAGE8_FLAGS_DEVICE_PERSISTENCE_MAPPING)
670 remove_entry->dev_handle = handle;
677 * _mapping_update_missing_count - Update missing count for a device
678 * @sc: per adapter object
679 * @topo_change: Topology change event entry
681 * Search through the topology change list and if any device is found not
682 * responding it's associated map table entry and DPM entry is updated
687 _mapping_update_missing_count(struct mps_softc *sc,
688 struct _map_topology_change *topo_change)
690 u16 ioc_pg8_flags = le16toh(sc->ioc_pg8.Flags);
692 struct _map_phy_change *phy_change;
694 struct dev_mapping_table *mt_entry;
695 Mpi2DriverMap0Entry_t *dpm_entry;
697 for (entry = 0; entry < topo_change->num_entries; entry++) {
698 phy_change = &topo_change->phy_details[entry];
699 if (!phy_change->dev_handle || (phy_change->reason !=
700 MPI2_EVENT_SAS_TOPO_RC_TARG_NOT_RESPONDING))
702 map_idx = _mapping_get_mt_idx_from_handle(sc, phy_change->
704 phy_change->is_processed = 1;
705 if (map_idx == MPS_MAPTABLE_BAD_IDX) {
706 printf("%s: device is already removed from mapping "
707 "table\n", __func__);
710 mt_entry = &sc->mapping_table[map_idx];
711 if (!mt_entry->init_complete) {
712 if (mt_entry->missing_count < MPS_MAX_MISSING_COUNT)
713 mt_entry->missing_count++;
715 mt_entry->init_complete = 1;
717 if (!mt_entry->missing_count)
718 mt_entry->missing_count++;
719 _mapping_add_to_removal_table(sc, mt_entry->dev_handle, 0);
720 mt_entry->dev_handle = 0;
722 if (((ioc_pg8_flags & MPI2_IOCPAGE8_FLAGS_MASK_MAPPING_MODE) ==
723 MPI2_IOCPAGE8_FLAGS_DEVICE_PERSISTENCE_MAPPING) &&
724 sc->is_dpm_enable && !mt_entry->init_complete &&
725 mt_entry->dpm_entry_num != MPS_DPM_BAD_IDX) {
727 (Mpi2DriverMap0Entry_t *) ((u8 *)sc->dpm_pg0 +
728 sizeof(MPI2_CONFIG_EXTENDED_PAGE_HEADER));
729 dpm_entry += mt_entry->dpm_entry_num;
730 dpm_entry->MappingInformation = mt_entry->missing_count;
731 sc->dpm_flush_entry[mt_entry->dpm_entry_num] = 1;
733 mt_entry->init_complete = 1;
738 * _mapping_find_enc_map_space -find map table entries for enclosure
739 * @sc: per adapter object
740 * @et_entry: enclosure entry
742 * Search through the mapping table defragment it and provide contiguous
743 * space in map table for a particular enclosure entry
745 * Returns start index in map table or bad index.
748 _mapping_find_enc_map_space(struct mps_softc *sc,
749 struct enc_mapping_table *et_entry)
751 u16 vol_mapping_flags;
752 u32 skip_count, end_of_table, map_idx, enc_idx;
754 u32 start_idx = MPS_MAPTABLE_BAD_IDX;
755 struct dev_mapping_table *mt_entry;
756 struct enc_mapping_table *enc_entry;
757 unsigned char done_flag = 0, found_space;
758 u16 max_num_phy_ids = le16toh(sc->ioc_pg8.MaxNumPhysicalMappedIDs);
760 skip_count = sc->num_rsvd_entries;
763 vol_mapping_flags = le16toh(sc->ioc_pg8.IRVolumeMappingFlags) &
764 MPI2_IOCPAGE8_IRFLAGS_MASK_VOLUME_MAPPING_MODE;
766 if (!sc->ir_firmware)
767 end_of_table = sc->max_devices;
768 else if (vol_mapping_flags == MPI2_IOCPAGE8_IRFLAGS_LOW_VOLUME_MAPPING)
769 end_of_table = sc->max_devices;
771 end_of_table = sc->max_devices - sc->max_volumes;
773 for (map_idx = (max_num_phy_ids + skip_count);
774 map_idx < end_of_table; map_idx++) {
775 mt_entry = &sc->mapping_table[map_idx];
776 if ((et_entry->enclosure_id == mt_entry->physical_id) &&
777 (!mt_entry->phy_bits || (mt_entry->phy_bits &
778 et_entry->phy_bits))) {
780 if (num_found == et_entry->num_slots) {
781 start_idx = (map_idx - num_found) + 1;
787 for (map_idx = (max_num_phy_ids + skip_count);
788 map_idx < end_of_table; map_idx++) {
789 mt_entry = &sc->mapping_table[map_idx];
790 if (!(mt_entry->device_info & MPS_DEV_RESERVED)) {
792 if (num_found == et_entry->num_slots) {
793 start_idx = (map_idx - num_found) + 1;
801 enc_idx = _mapping_get_high_missing_et_idx(sc);
802 if (enc_idx == MPS_ENCTABLE_BAD_IDX)
803 return MPS_MAPTABLE_BAD_IDX;
804 enc_entry = &sc->enclosure_table[enc_idx];
806 enc_entry->skip_search = 1;
807 mt_entry = &sc->mapping_table[enc_entry->start_index];
808 for (map_idx = enc_entry->start_index; map_idx <
809 (enc_entry->start_index + enc_entry->num_slots); map_idx++,
811 mt_entry->device_info &= ~MPS_DEV_RESERVED;
813 for (map_idx = (max_num_phy_ids +
814 skip_count); map_idx < end_of_table; map_idx++) {
815 mt_entry = &sc->mapping_table[map_idx];
816 if (!(mt_entry->device_info & MPS_DEV_RESERVED)) {
818 if (num_found == et_entry->num_slots) {
819 start_idx = (map_idx - num_found) + 1;
828 for (map_idx = start_idx; map_idx < (start_idx + num_found);
830 enc_entry = sc->enclosure_table;
831 for (enc_idx = 0; enc_idx < sc->num_enc_table_entries;
832 enc_idx++, enc_entry++) {
833 if (map_idx < enc_entry->start_index ||
834 map_idx > (enc_entry->start_index +
835 enc_entry->num_slots))
837 if (!enc_entry->removal_flag) {
838 enc_entry->removal_flag = 1;
839 _mapping_add_to_removal_table(sc, 0,
840 enc_entry->dpm_entry_num);
842 mt_entry = &sc->mapping_table[map_idx];
843 if (mt_entry->device_info &
845 _mapping_add_to_removal_table(sc,
846 mt_entry->dev_handle, 0);
847 _mapping_clear_map_entry(mt_entry);
849 if (map_idx == (enc_entry->start_index +
850 enc_entry->num_slots - 1))
851 _mapping_clear_enc_entry(et_entry);
854 enc_entry = sc->enclosure_table;
855 for (enc_idx = 0; enc_idx < sc->num_enc_table_entries;
856 enc_idx++, enc_entry++) {
857 if (!enc_entry->removal_flag) {
858 mt_entry = &sc->mapping_table[enc_entry->
860 for (map_idx = enc_entry->start_index; map_idx <
861 (enc_entry->start_index +
862 enc_entry->num_slots); map_idx++,
864 mt_entry->device_info |=
866 et_entry->skip_search = 0;
875 * _mapping_get_dev_info -get information about newly added devices
876 * @sc: per adapter object
877 * @topo_change: Topology change event entry
879 * Search through the topology change event list and issues sas device pg0
880 * requests for the newly added device and reserved entries in tables
885 _mapping_get_dev_info(struct mps_softc *sc,
886 struct _map_topology_change *topo_change)
888 u16 ioc_pg8_flags = le16toh(sc->ioc_pg8.Flags);
889 Mpi2ConfigReply_t mpi_reply;
890 Mpi2SasDevicePage0_t sas_device_pg0;
891 u8 entry, enc_idx, phy_idx;
892 u32 map_idx, index, device_info;
893 struct _map_phy_change *phy_change, *tmp_phy_change;
894 uint64_t sas_address;
895 struct enc_mapping_table *et_entry;
896 struct dev_mapping_table *mt_entry;
897 u8 add_code = MPI2_EVENT_SAS_TOPO_RC_TARG_ADDED;
900 for (entry = 0; entry < topo_change->num_entries; entry++) {
901 phy_change = &topo_change->phy_details[entry];
902 if (phy_change->is_processed || !phy_change->dev_handle ||
903 phy_change->reason != MPI2_EVENT_SAS_TOPO_RC_TARG_ADDED)
905 if (mps_config_get_sas_device_pg0(sc, &mpi_reply,
906 &sas_device_pg0, MPI2_SAS_DEVICE_PGAD_FORM_HANDLE,
907 phy_change->dev_handle)) {
908 phy_change->is_processed = 1;
912 device_info = le32toh(sas_device_pg0.DeviceInfo);
913 if ((ioc_pg8_flags & MPI2_IOCPAGE8_FLAGS_MASK_MAPPING_MODE) ==
914 MPI2_IOCPAGE8_FLAGS_DEVICE_PERSISTENCE_MAPPING) {
915 if ((device_info & MPI2_SAS_DEVICE_INFO_END_DEVICE) &&
916 (device_info & MPI2_SAS_DEVICE_INFO_SATA_DEVICE)) {
917 rc = mpssas_get_sas_address_for_sata_disk(sc,
918 &sas_address, phy_change->dev_handle,
921 printf("%s: failed to compute the "
922 "hashed SAS Address for SATA "
923 "device with handle 0x%04x\n",
924 __func__, phy_change->dev_handle);
926 sas_device_pg0.SASAddress.High;
927 sas_address = (sas_address << 32) |
928 sas_device_pg0.SASAddress.Low;
930 mps_dprint(sc, MPS_INFO, "SAS Address for SATA "
931 "device = %jx\n", sas_address);
934 sas_device_pg0.SASAddress.High;
935 sas_address = (sas_address << 32) |
936 sas_device_pg0.SASAddress.Low;
939 sas_address = sas_device_pg0.SASAddress.High;
940 sas_address = (sas_address << 32) |
941 sas_device_pg0.SASAddress.Low;
943 phy_change->physical_id = sas_address;
944 phy_change->slot = le16toh(sas_device_pg0.Slot);
945 phy_change->device_info =
946 le32toh(sas_device_pg0.DeviceInfo);
948 if ((ioc_pg8_flags & MPI2_IOCPAGE8_FLAGS_MASK_MAPPING_MODE) ==
949 MPI2_IOCPAGE8_FLAGS_ENCLOSURE_SLOT_MAPPING) {
950 enc_idx = _mapping_get_enc_idx_from_handle(sc,
951 topo_change->enc_handle);
952 if (enc_idx == MPS_ENCTABLE_BAD_IDX) {
953 phy_change->is_processed = 1;
954 printf("%s: failed to add the device with "
955 "handle 0x%04x because the enclosure is "
956 "not in the mapping table\n", __func__,
957 phy_change->dev_handle);
960 if (!((phy_change->device_info &
961 MPI2_SAS_DEVICE_INFO_END_DEVICE) &&
962 (phy_change->device_info &
963 (MPI2_SAS_DEVICE_INFO_SSP_TARGET |
964 MPI2_SAS_DEVICE_INFO_STP_TARGET |
965 MPI2_SAS_DEVICE_INFO_SATA_DEVICE)))) {
966 phy_change->is_processed = 1;
969 et_entry = &sc->enclosure_table[enc_idx];
970 if (et_entry->start_index != MPS_MAPTABLE_BAD_IDX)
972 if (!topo_change->exp_handle) {
973 map_idx = sc->num_rsvd_entries;
974 et_entry->start_index = map_idx;
976 map_idx = _mapping_find_enc_map_space(sc,
978 et_entry->start_index = map_idx;
979 if (et_entry->start_index ==
980 MPS_MAPTABLE_BAD_IDX) {
981 phy_change->is_processed = 1;
982 for (phy_idx = 0; phy_idx <
983 topo_change->num_entries;
986 &topo_change->phy_details
988 if (tmp_phy_change->reason ==
996 mt_entry = &sc->mapping_table[map_idx];
997 for (index = map_idx; index < (et_entry->num_slots
998 + map_idx); index++, mt_entry++) {
999 mt_entry->device_info = MPS_DEV_RESERVED;
1000 mt_entry->physical_id = et_entry->enclosure_id;
1001 mt_entry->phy_bits = et_entry->phy_bits;
1008 * _mapping_set_mid_to_eid -set map table data from enclosure table
1009 * @sc: per adapter object
1010 * @et_entry: enclosure entry
1015 _mapping_set_mid_to_eid(struct mps_softc *sc,
1016 struct enc_mapping_table *et_entry)
1018 struct dev_mapping_table *mt_entry;
1019 u16 slots = et_entry->num_slots, map_idx;
1020 u32 start_idx = et_entry->start_index;
1021 if (start_idx != MPS_MAPTABLE_BAD_IDX) {
1022 mt_entry = &sc->mapping_table[start_idx];
1023 for (map_idx = 0; map_idx < slots; map_idx++, mt_entry++)
1024 mt_entry->physical_id = et_entry->enclosure_id;
1029 * _mapping_clear_removed_entries - mark the entries to be cleared
1030 * @sc: per adapter object
1032 * Search through the removal table and mark the entries which needs to be
1033 * flushed to DPM and also updates the map table and enclosure table by
1034 * clearing the corresponding entries.
1039 _mapping_clear_removed_entries(struct mps_softc *sc)
1042 struct map_removal_table *remove_entry;
1043 Mpi2DriverMap0Entry_t *dpm_entry;
1044 u8 done_flag = 0, num_entries, m, i;
1045 struct enc_mapping_table *et_entry, *from, *to;
1046 u16 ioc_pg8_flags = le16toh(sc->ioc_pg8.Flags);
1048 if (sc->is_dpm_enable) {
1049 remove_entry = sc->removal_table;
1050 for (remove_idx = 0; remove_idx < sc->max_devices;
1051 remove_idx++, remove_entry++) {
1052 if (remove_entry->dpm_entry_num != MPS_DPM_BAD_IDX) {
1053 dpm_entry = (Mpi2DriverMap0Entry_t *)
1054 ((u8 *) sc->dpm_pg0 +
1055 sizeof(MPI2_CONFIG_EXTENDED_PAGE_HEADER));
1056 dpm_entry += remove_entry->dpm_entry_num;
1057 dpm_entry->PhysicalIdentifier.Low = 0;
1058 dpm_entry->PhysicalIdentifier.High = 0;
1059 dpm_entry->DeviceIndex = 0;
1060 dpm_entry->MappingInformation = 0;
1061 dpm_entry->PhysicalBitsMapping = 0;
1062 sc->dpm_flush_entry[remove_entry->
1064 sc->dpm_entry_used[remove_entry->dpm_entry_num]
1066 remove_entry->dpm_entry_num = MPS_DPM_BAD_IDX;
1070 if ((ioc_pg8_flags & MPI2_IOCPAGE8_FLAGS_MASK_MAPPING_MODE) ==
1071 MPI2_IOCPAGE8_FLAGS_ENCLOSURE_SLOT_MAPPING) {
1072 num_entries = sc->num_enc_table_entries;
1073 while (!done_flag) {
1075 et_entry = sc->enclosure_table;
1076 for (i = 0; i < num_entries; i++, et_entry++) {
1077 if (!et_entry->enc_handle && et_entry->
1080 if (i != (num_entries - 1)) {
1081 from = &sc->enclosure_table
1083 to = &sc->enclosure_table[i];
1084 for (m = i; m < (num_entries -
1085 1); m++, from++, to++) {
1086 _mapping_set_mid_to_eid
1090 _mapping_clear_enc_entry(to);
1091 sc->num_enc_table_entries--;
1093 sc->num_enc_table_entries;
1095 _mapping_clear_enc_entry
1097 sc->num_enc_table_entries--;
1099 sc->num_enc_table_entries;
1108 * _mapping_add_new_device -Add the new device into mapping table
1109 * @sc: per adapter object
1110 * @topo_change: Topology change event entry
1112 * Search through the topology change event list and updates map table,
1113 * enclosure table and DPM pages for for the newly added devices.
1118 _mapping_add_new_device(struct mps_softc *sc,
1119 struct _map_topology_change *topo_change)
1121 u8 enc_idx, missing_cnt, is_removed = 0;
1123 u32 search_idx, map_idx;
1125 struct dev_mapping_table *mt_entry;
1126 struct enc_mapping_table *et_entry;
1127 struct _map_phy_change *phy_change;
1128 u16 ioc_pg8_flags = le16toh(sc->ioc_pg8.Flags);
1129 Mpi2DriverMap0Entry_t *dpm_entry;
1130 uint64_t temp64_var;
1131 u8 map_shift = MPI2_DRVMAP0_MAPINFO_SLOT_SHIFT;
1132 u8 hdr_sz = sizeof(MPI2_CONFIG_EXTENDED_PAGE_HEADER);
1133 u16 max_num_phy_ids = le16toh(sc->ioc_pg8.MaxNumPhysicalMappedIDs);
1135 for (entry = 0; entry < topo_change->num_entries; entry++) {
1136 phy_change = &topo_change->phy_details[entry];
1137 if (phy_change->is_processed)
1139 if (phy_change->reason != MPI2_EVENT_SAS_TOPO_RC_TARG_ADDED ||
1140 !phy_change->dev_handle) {
1141 phy_change->is_processed = 1;
1144 if ((ioc_pg8_flags & MPI2_IOCPAGE8_FLAGS_MASK_MAPPING_MODE) ==
1145 MPI2_IOCPAGE8_FLAGS_ENCLOSURE_SLOT_MAPPING) {
1146 enc_idx = _mapping_get_enc_idx_from_handle
1147 (sc, topo_change->enc_handle);
1148 if (enc_idx == MPS_ENCTABLE_BAD_IDX) {
1149 phy_change->is_processed = 1;
1150 printf("%s: failed to add the device with "
1151 "handle 0x%04x because the enclosure is "
1152 "not in the mapping table\n", __func__,
1153 phy_change->dev_handle);
1156 et_entry = &sc->enclosure_table[enc_idx];
1157 if (et_entry->start_index == MPS_MAPTABLE_BAD_IDX) {
1158 phy_change->is_processed = 1;
1159 if (!sc->mt_full_retry) {
1160 sc->mt_add_device_failed = 1;
1163 printf("%s: failed to add the device with "
1164 "handle 0x%04x because there is no free "
1165 "space available in the mapping table\n",
1166 __func__, phy_change->dev_handle);
1169 map_idx = et_entry->start_index + phy_change->slot -
1170 et_entry->start_slot;
1171 mt_entry = &sc->mapping_table[map_idx];
1172 mt_entry->physical_id = phy_change->physical_id;
1173 mt_entry->channel = 0;
1174 mt_entry->id = map_idx;
1175 mt_entry->dev_handle = phy_change->dev_handle;
1176 mt_entry->missing_count = 0;
1177 mt_entry->dpm_entry_num = et_entry->dpm_entry_num;
1178 mt_entry->device_info = phy_change->device_info |
1179 (MPS_DEV_RESERVED | MPS_MAP_IN_USE);
1180 if (sc->is_dpm_enable) {
1181 dpm_idx = et_entry->dpm_entry_num;
1182 if (dpm_idx == MPS_DPM_BAD_IDX)
1183 dpm_idx = _mapping_get_dpm_idx_from_id
1184 (sc, et_entry->enclosure_id,
1185 et_entry->phy_bits);
1186 if (dpm_idx == MPS_DPM_BAD_IDX) {
1187 dpm_idx = _mapping_get_free_dpm_idx(sc);
1188 if (dpm_idx != MPS_DPM_BAD_IDX) {
1190 (Mpi2DriverMap0Entry_t *)
1191 ((u8 *) sc->dpm_pg0 +
1193 dpm_entry += dpm_idx;
1195 PhysicalIdentifier.Low =
1197 et_entry->enclosure_id);
1199 PhysicalIdentifier.High =
1200 ( et_entry->enclosure_id
1202 dpm_entry->DeviceIndex =
1203 (U16)et_entry->start_index;
1204 dpm_entry->MappingInformation =
1205 et_entry->num_slots;
1206 dpm_entry->MappingInformation
1208 dpm_entry->PhysicalBitsMapping
1209 = et_entry->phy_bits;
1210 et_entry->dpm_entry_num =
1212 /* FIXME Do I need to set the dpm_idxin mt_entry too */
1213 sc->dpm_entry_used[dpm_idx] = 1;
1214 sc->dpm_flush_entry[dpm_idx] =
1216 phy_change->is_processed = 1;
1218 phy_change->is_processed = 1;
1219 printf("%s: failed to add the "
1220 "device with handle 0x%04x "
1221 "to persistent table "
1222 "because there is no free "
1223 "space available\n",
1225 phy_change->dev_handle);
1228 et_entry->dpm_entry_num = dpm_idx;
1229 mt_entry->dpm_entry_num = dpm_idx;
1232 /* FIXME Why not mt_entry too? */
1233 et_entry->init_complete = 1;
1234 } else if ((ioc_pg8_flags &
1235 MPI2_IOCPAGE8_FLAGS_MASK_MAPPING_MODE) ==
1236 MPI2_IOCPAGE8_FLAGS_DEVICE_PERSISTENCE_MAPPING) {
1237 map_idx = _mapping_get_mt_idx_from_id
1238 (sc, phy_change->physical_id);
1239 if (map_idx == MPS_MAPTABLE_BAD_IDX) {
1240 search_idx = sc->num_rsvd_entries;
1241 if (topo_change->exp_handle)
1242 search_idx += max_num_phy_ids;
1243 map_idx = _mapping_get_free_mt_idx(sc,
1246 if (map_idx == MPS_MAPTABLE_BAD_IDX) {
1247 map_idx = _mapping_get_high_missing_mt_idx(sc);
1248 if (map_idx != MPS_MAPTABLE_BAD_IDX) {
1249 mt_entry = &sc->mapping_table[map_idx];
1250 if (mt_entry->dev_handle) {
1251 _mapping_add_to_removal_table
1252 (sc, mt_entry->dev_handle,
1256 mt_entry->init_complete = 0;
1259 if (map_idx != MPS_MAPTABLE_BAD_IDX) {
1260 mt_entry = &sc->mapping_table[map_idx];
1261 mt_entry->physical_id = phy_change->physical_id;
1262 mt_entry->channel = 0;
1263 mt_entry->id = map_idx;
1264 mt_entry->dev_handle = phy_change->dev_handle;
1265 mt_entry->missing_count = 0;
1266 mt_entry->device_info = phy_change->device_info
1267 | (MPS_DEV_RESERVED | MPS_MAP_IN_USE);
1269 phy_change->is_processed = 1;
1270 if (!sc->mt_full_retry) {
1271 sc->mt_add_device_failed = 1;
1274 printf("%s: failed to add the device with "
1275 "handle 0x%04x because there is no free "
1276 "space available in the mapping table\n",
1277 __func__, phy_change->dev_handle);
1280 if (sc->is_dpm_enable) {
1281 if (mt_entry->dpm_entry_num !=
1283 dpm_idx = mt_entry->dpm_entry_num;
1284 dpm_entry = (Mpi2DriverMap0Entry_t *)
1285 ((u8 *)sc->dpm_pg0 + hdr_sz);
1286 dpm_entry += dpm_idx;
1287 missing_cnt = dpm_entry->
1288 MappingInformation &
1289 MPI2_DRVMAP0_MAPINFO_MISSING_MASK;
1290 temp64_var = dpm_entry->
1291 PhysicalIdentifier.High;
1292 temp64_var = (temp64_var << 32) |
1293 dpm_entry->PhysicalIdentifier.Low;
1294 if ((mt_entry->physical_id ==
1295 temp64_var) && !missing_cnt)
1296 mt_entry->init_complete = 1;
1298 dpm_idx = _mapping_get_free_dpm_idx(sc);
1299 mt_entry->init_complete = 0;
1301 if (dpm_idx != MPS_DPM_BAD_IDX &&
1302 !mt_entry->init_complete) {
1303 mt_entry->init_complete = 1;
1304 mt_entry->dpm_entry_num = dpm_idx;
1305 dpm_entry = (Mpi2DriverMap0Entry_t *)
1306 ((u8 *)sc->dpm_pg0 + hdr_sz);
1307 dpm_entry += dpm_idx;
1308 dpm_entry->PhysicalIdentifier.Low =
1310 mt_entry->physical_id);
1311 dpm_entry->PhysicalIdentifier.High =
1312 (mt_entry->physical_id >> 32);
1313 dpm_entry->DeviceIndex = (U16) map_idx;
1314 dpm_entry->MappingInformation = 0;
1315 dpm_entry->PhysicalBitsMapping = 0;
1316 sc->dpm_entry_used[dpm_idx] = 1;
1317 sc->dpm_flush_entry[dpm_idx] = 1;
1318 phy_change->is_processed = 1;
1319 } else if (dpm_idx == MPS_DPM_BAD_IDX) {
1320 phy_change->is_processed = 1;
1321 printf("%s: failed to add the "
1322 "device with handle 0x%04x "
1323 "to persistent table "
1324 "because there is no free "
1325 "space available\n",
1327 phy_change->dev_handle);
1330 mt_entry->init_complete = 1;
1333 phy_change->is_processed = 1;
1336 _mapping_clear_removed_entries(sc);
1340 * _mapping_flush_dpm_pages -Flush the DPM pages to NVRAM
1341 * @sc: per adapter object
1346 _mapping_flush_dpm_pages(struct mps_softc *sc)
1348 Mpi2DriverMap0Entry_t *dpm_entry;
1349 Mpi2ConfigReply_t mpi_reply;
1350 Mpi2DriverMappingPage0_t config_page;
1353 for (entry_num = 0; entry_num < sc->max_dpm_entries; entry_num++) {
1354 if (!sc->dpm_flush_entry[entry_num])
1356 memset(&config_page, 0, sizeof(Mpi2DriverMappingPage0_t));
1357 memcpy(&config_page.Header, (u8 *)sc->dpm_pg0,
1358 sizeof(MPI2_CONFIG_EXTENDED_PAGE_HEADER));
1359 dpm_entry = (Mpi2DriverMap0Entry_t *) ((u8 *)sc->dpm_pg0 +
1360 sizeof(MPI2_CONFIG_EXTENDED_PAGE_HEADER));
1361 dpm_entry += entry_num;
1362 dpm_entry->MappingInformation = htole16(dpm_entry->
1363 MappingInformation);
1364 dpm_entry->DeviceIndex = htole16(dpm_entry->DeviceIndex);
1365 dpm_entry->PhysicalBitsMapping = htole32(dpm_entry->
1366 PhysicalBitsMapping);
1367 memcpy(&config_page.Entry, (u8 *)dpm_entry,
1368 sizeof(Mpi2DriverMap0Entry_t));
1369 /* TODO-How to handle failed writes? */
1370 if (mps_config_set_dpm_pg0(sc, &mpi_reply, &config_page,
1372 printf("%s: write of dpm entry %d for device failed\n",
1373 __func__, entry_num);
1375 sc->dpm_flush_entry[entry_num] = 0;
1376 dpm_entry->MappingInformation = le16toh(dpm_entry->
1377 MappingInformation);
1378 dpm_entry->DeviceIndex = le16toh(dpm_entry->DeviceIndex);
1379 dpm_entry->PhysicalBitsMapping = le32toh(dpm_entry->
1380 PhysicalBitsMapping);
1385 * _mapping_allocate_memory- allocates the memory required for mapping tables
1386 * @sc: per adapter object
1388 * Allocates the memory for all the tables required for host mapping
1390 * Return 0 on success or non-zero on failure.
1393 mps_mapping_allocate_memory(struct mps_softc *sc)
1395 uint32_t dpm_pg0_sz;
1397 sc->mapping_table = malloc((sizeof(struct dev_mapping_table) *
1398 sc->max_devices), M_MPT2, M_ZERO|M_NOWAIT);
1399 if (!sc->mapping_table)
1400 goto free_resources;
1402 sc->removal_table = malloc((sizeof(struct map_removal_table) *
1403 sc->max_devices), M_MPT2, M_ZERO|M_NOWAIT);
1404 if (!sc->removal_table)
1405 goto free_resources;
1407 sc->enclosure_table = malloc((sizeof(struct enc_mapping_table) *
1408 sc->max_enclosures), M_MPT2, M_ZERO|M_NOWAIT);
1409 if (!sc->enclosure_table)
1410 goto free_resources;
1412 sc->dpm_entry_used = malloc((sizeof(u8) * sc->max_dpm_entries),
1413 M_MPT2, M_ZERO|M_NOWAIT);
1414 if (!sc->dpm_entry_used)
1415 goto free_resources;
1417 sc->dpm_flush_entry = malloc((sizeof(u8) * sc->max_dpm_entries),
1418 M_MPT2, M_ZERO|M_NOWAIT);
1419 if (!sc->dpm_flush_entry)
1420 goto free_resources;
1422 dpm_pg0_sz = sizeof(MPI2_CONFIG_EXTENDED_PAGE_HEADER) +
1423 (sc->max_dpm_entries * sizeof(MPI2_CONFIG_PAGE_DRIVER_MAP0_ENTRY));
1425 sc->dpm_pg0 = malloc(dpm_pg0_sz, M_MPT2, M_ZERO|M_NOWAIT);
1427 printf("%s: memory alloc failed for dpm page; disabling dpm\n",
1429 sc->is_dpm_enable = 0;
1435 free(sc->mapping_table, M_MPT2);
1436 free(sc->removal_table, M_MPT2);
1437 free(sc->enclosure_table, M_MPT2);
1438 free(sc->dpm_entry_used, M_MPT2);
1439 free(sc->dpm_flush_entry, M_MPT2);
1440 free(sc->dpm_pg0, M_MPT2);
1441 printf("%s: device initialization failed due to failure in mapping "
1442 "table memory allocation\n", __func__);
1447 * mps_mapping_free_memory- frees the memory allocated for mapping tables
1448 * @sc: per adapter object
1453 mps_mapping_free_memory(struct mps_softc *sc)
1455 free(sc->mapping_table, M_MPT2);
1456 free(sc->removal_table, M_MPT2);
1457 free(sc->enclosure_table, M_MPT2);
1458 free(sc->dpm_entry_used, M_MPT2);
1459 free(sc->dpm_flush_entry, M_MPT2);
1460 free(sc->dpm_pg0, M_MPT2);
1465 _mapping_process_dpm_pg0(struct mps_softc *sc)
1467 u8 missing_cnt, enc_idx;
1468 u16 slot_id, entry_num, num_slots;
1469 u32 map_idx, dev_idx, start_idx, end_idx;
1470 struct dev_mapping_table *mt_entry;
1471 Mpi2DriverMap0Entry_t *dpm_entry;
1472 u16 ioc_pg8_flags = le16toh(sc->ioc_pg8.Flags);
1473 u16 max_num_phy_ids = le16toh(sc->ioc_pg8.MaxNumPhysicalMappedIDs);
1474 struct enc_mapping_table *et_entry;
1478 if (sc->ir_firmware)
1479 _mapping_get_ir_maprange(sc, &start_idx, &end_idx);
1481 dpm_entry = (Mpi2DriverMap0Entry_t *) ((uint8_t *) sc->dpm_pg0 +
1482 sizeof(MPI2_CONFIG_EXTENDED_PAGE_HEADER));
1483 for (entry_num = 0; entry_num < sc->max_dpm_entries; entry_num++,
1485 physical_id = dpm_entry->PhysicalIdentifier.High;
1486 physical_id = (physical_id << 32) |
1487 dpm_entry->PhysicalIdentifier.Low;
1489 sc->dpm_entry_used[entry_num] = 0;
1492 sc->dpm_entry_used[entry_num] = 1;
1493 dpm_entry->MappingInformation = le16toh(dpm_entry->
1494 MappingInformation);
1495 missing_cnt = dpm_entry->MappingInformation &
1496 MPI2_DRVMAP0_MAPINFO_MISSING_MASK;
1497 dev_idx = le16toh(dpm_entry->DeviceIndex);
1498 phy_bits = le32toh(dpm_entry->PhysicalBitsMapping);
1499 if (sc->ir_firmware && (dev_idx >= start_idx) &&
1500 (dev_idx <= end_idx)) {
1501 mt_entry = &sc->mapping_table[dev_idx];
1502 mt_entry->physical_id = dpm_entry->PhysicalIdentifier.High;
1503 mt_entry->physical_id = (mt_entry->physical_id << 32) |
1504 dpm_entry->PhysicalIdentifier.Low;
1505 mt_entry->channel = MPS_RAID_CHANNEL;
1506 mt_entry->id = dev_idx;
1507 mt_entry->missing_count = missing_cnt;
1508 mt_entry->dpm_entry_num = entry_num;
1509 mt_entry->device_info = MPS_DEV_RESERVED;
1512 if ((ioc_pg8_flags & MPI2_IOCPAGE8_FLAGS_MASK_MAPPING_MODE) ==
1513 MPI2_IOCPAGE8_FLAGS_ENCLOSURE_SLOT_MAPPING) {
1514 if (dev_idx < (sc->num_rsvd_entries +
1518 MPI2_IOCPAGE8_FLAGS_DA_START_SLOT_1)
1520 num_slots = max_num_phy_ids;
1523 num_slots = dpm_entry->MappingInformation &
1524 MPI2_DRVMAP0_MAPINFO_SLOT_MASK;
1525 num_slots >>= MPI2_DRVMAP0_MAPINFO_SLOT_SHIFT;
1527 enc_idx = sc->num_enc_table_entries;
1528 if (enc_idx >= sc->max_enclosures) {
1529 printf("%s: enclosure entries exceed max "
1530 "enclosures of %d\n", __func__,
1531 sc->max_enclosures);
1534 sc->num_enc_table_entries++;
1535 et_entry = &sc->enclosure_table[enc_idx];
1536 physical_id = dpm_entry->PhysicalIdentifier.High;
1537 et_entry->enclosure_id = (physical_id << 32) |
1538 dpm_entry->PhysicalIdentifier.Low;
1539 et_entry->start_index = dev_idx;
1540 et_entry->dpm_entry_num = entry_num;
1541 et_entry->num_slots = num_slots;
1542 et_entry->start_slot = slot_id;
1543 et_entry->missing_count = missing_cnt;
1544 et_entry->phy_bits = phy_bits;
1546 mt_entry = &sc->mapping_table[dev_idx];
1547 for (map_idx = dev_idx; map_idx < (dev_idx + num_slots);
1548 map_idx++, mt_entry++) {
1549 if (mt_entry->dpm_entry_num !=
1551 printf("%s: conflict in mapping table "
1552 "for enclosure %d\n", __func__,
1556 physical_id = dpm_entry->PhysicalIdentifier.High;
1557 mt_entry->physical_id = (physical_id << 32) |
1558 dpm_entry->PhysicalIdentifier.Low;
1559 mt_entry->phy_bits = phy_bits;
1560 mt_entry->channel = 0;
1561 mt_entry->id = dev_idx;
1562 mt_entry->dpm_entry_num = entry_num;
1563 mt_entry->missing_count = missing_cnt;
1564 mt_entry->device_info = MPS_DEV_RESERVED;
1566 } else if ((ioc_pg8_flags &
1567 MPI2_IOCPAGE8_FLAGS_MASK_MAPPING_MODE) ==
1568 MPI2_IOCPAGE8_FLAGS_DEVICE_PERSISTENCE_MAPPING) {
1570 mt_entry = &sc->mapping_table[map_idx];
1571 if (mt_entry->dpm_entry_num != MPS_DPM_BAD_IDX) {
1572 printf("%s: conflict in mapping table for "
1573 "device %d\n", __func__, map_idx);
1576 physical_id = dpm_entry->PhysicalIdentifier.High;
1577 mt_entry->physical_id = (physical_id << 32) |
1578 dpm_entry->PhysicalIdentifier.Low;
1579 mt_entry->phy_bits = phy_bits;
1580 mt_entry->channel = 0;
1581 mt_entry->id = dev_idx;
1582 mt_entry->missing_count = missing_cnt;
1583 mt_entry->dpm_entry_num = entry_num;
1584 mt_entry->device_info = MPS_DEV_RESERVED;
1586 } /*close the loop for DPM table */
1590 * mps_mapping_check_devices - start of the day check for device availabilty
1591 * @sc: per adapter object
1592 * @sleep_flag: Flag indicating whether this function can sleep or not
1597 mps_mapping_check_devices(struct mps_softc *sc, int sleep_flag)
1602 struct dev_mapping_table *mt_entry;
1603 u16 ioc_pg8_flags = le16toh(sc->ioc_pg8.Flags);
1604 struct enc_mapping_table *et_entry;
1605 u32 start_idx, end_idx;
1607 /* We need to ucomment this when this function is called
1608 * from the port enable complete */
1610 sc->track_mapping_events = 0;
1611 cntdn = (sleep_flag == CAN_SLEEP) ? 1000*timeout : 2000*timeout;
1613 if (!sc->pending_map_events)
1615 if (sleep_flag == CAN_SLEEP)
1616 pause("mps_pause", (hz/1000));/* 1msec sleep */
1618 DELAY(500); /* 500 useconds delay */
1623 printf("%s: there are %d"
1624 " pending events after %d seconds of delay\n",
1625 __func__, sc->pending_map_events, timeout);
1627 sc->pending_map_events = 0;
1629 if ((ioc_pg8_flags & MPI2_IOCPAGE8_FLAGS_MASK_MAPPING_MODE) ==
1630 MPI2_IOCPAGE8_FLAGS_ENCLOSURE_SLOT_MAPPING) {
1631 et_entry = sc->enclosure_table;
1632 for (i = 0; i < sc->num_enc_table_entries; i++, et_entry++) {
1633 if (!et_entry->init_complete) {
1634 if (et_entry->missing_count <
1635 MPS_MAX_MISSING_COUNT) {
1636 et_entry->missing_count++;
1637 if (et_entry->dpm_entry_num !=
1639 _mapping_commit_enc_entry(sc,
1642 et_entry->init_complete = 1;
1645 if (!sc->ir_firmware)
1647 _mapping_get_ir_maprange(sc, &start_idx, &end_idx);
1648 mt_entry = &sc->mapping_table[start_idx];
1649 for (i = start_idx; i < (end_idx + 1); i++, mt_entry++) {
1650 if (mt_entry->device_info & MPS_DEV_RESERVED
1651 && !mt_entry->physical_id)
1652 mt_entry->init_complete = 1;
1653 else if (mt_entry->device_info & MPS_DEV_RESERVED) {
1654 if (!mt_entry->init_complete) {
1655 if (mt_entry->missing_count <
1656 MPS_MAX_MISSING_COUNT) {
1657 mt_entry->missing_count++;
1658 if (mt_entry->dpm_entry_num !=
1660 _mapping_commit_map_entry(sc,
1663 mt_entry->init_complete = 1;
1667 } else if ((ioc_pg8_flags & MPI2_IOCPAGE8_FLAGS_MASK_MAPPING_MODE) ==
1668 MPI2_IOCPAGE8_FLAGS_DEVICE_PERSISTENCE_MAPPING) {
1669 mt_entry = sc->mapping_table;
1670 for (i = 0; i < sc->max_devices; i++, mt_entry++) {
1671 if (mt_entry->device_info & MPS_DEV_RESERVED
1672 && !mt_entry->physical_id)
1673 mt_entry->init_complete = 1;
1674 else if (mt_entry->device_info & MPS_DEV_RESERVED) {
1675 if (!mt_entry->init_complete) {
1676 if (mt_entry->missing_count <
1677 MPS_MAX_MISSING_COUNT) {
1678 mt_entry->missing_count++;
1679 if (mt_entry->dpm_entry_num !=
1681 _mapping_commit_map_entry(sc,
1684 mt_entry->init_complete = 1;
1693 * mps_mapping_is_reinit_required - check whether event replay required
1694 * @sc: per adapter object
1696 * Checks the per ioc flags and decide whether reinit of events required
1698 * Returns 1 for reinit of ioc 0 for not.
1700 int mps_mapping_is_reinit_required(struct mps_softc *sc)
1702 if (!sc->mt_full_retry && sc->mt_add_device_failed) {
1703 sc->mt_full_retry = 1;
1704 sc->mt_add_device_failed = 0;
1705 _mapping_flush_dpm_pages(sc);
1708 sc->mt_full_retry = 1;
1713 * mps_mapping_initialize - initialize mapping tables
1714 * @sc: per adapter object
1716 * Read controller persitant mapping tables into internal data area.
1718 * Return 0 for success or non-zero for failure.
1721 mps_mapping_initialize(struct mps_softc *sc)
1723 uint16_t volume_mapping_flags, dpm_pg0_sz;
1725 Mpi2ConfigReply_t mpi_reply;
1727 uint8_t retry_count;
1728 uint16_t ioc_pg8_flags = le16toh(sc->ioc_pg8.Flags);
1730 /* The additional 1 accounts for the virtual enclosure
1731 * created for the controller
1733 sc->max_enclosures = sc->facts->MaxEnclosures + 1;
1734 sc->max_expanders = sc->facts->MaxSasExpanders;
1735 sc->max_volumes = sc->facts->MaxVolumes;
1736 sc->max_devices = sc->facts->MaxTargets + sc->max_volumes;
1737 sc->pending_map_events = 0;
1738 sc->num_enc_table_entries = 0;
1739 sc->num_rsvd_entries = 0;
1740 sc->num_channels = 1;
1741 sc->max_dpm_entries = sc->ioc_pg8.MaxPersistentEntries;
1742 sc->is_dpm_enable = (sc->max_dpm_entries) ? 1 : 0;
1743 sc->track_mapping_events = 0;
1745 if (ioc_pg8_flags & MPI2_IOCPAGE8_FLAGS_DISABLE_PERSISTENT_MAPPING)
1746 sc->is_dpm_enable = 0;
1748 if (ioc_pg8_flags & MPI2_IOCPAGE8_FLAGS_RESERVED_TARGETID_0)
1749 sc->num_rsvd_entries = 1;
1751 volume_mapping_flags = sc->ioc_pg8.IRVolumeMappingFlags &
1752 MPI2_IOCPAGE8_IRFLAGS_MASK_VOLUME_MAPPING_MODE;
1753 if (sc->ir_firmware && (volume_mapping_flags ==
1754 MPI2_IOCPAGE8_IRFLAGS_LOW_VOLUME_MAPPING))
1755 sc->num_rsvd_entries += sc->max_volumes;
1757 error = mps_mapping_allocate_memory(sc);
1761 for (i = 0; i < sc->max_devices; i++)
1762 _mapping_clear_map_entry(sc->mapping_table + i);
1764 for (i = 0; i < sc->max_enclosures; i++)
1765 _mapping_clear_enc_entry(sc->enclosure_table + i);
1767 for (i = 0; i < sc->max_devices; i++) {
1768 sc->removal_table[i].dev_handle = 0;
1769 sc->removal_table[i].dpm_entry_num = MPS_DPM_BAD_IDX;
1772 memset(sc->dpm_entry_used, 0, sc->max_dpm_entries);
1773 memset(sc->dpm_flush_entry, 0, sc->max_dpm_entries);
1775 if (sc->is_dpm_enable) {
1776 dpm_pg0_sz = sizeof(MPI2_CONFIG_EXTENDED_PAGE_HEADER) +
1777 (sc->max_dpm_entries *
1778 sizeof(MPI2_CONFIG_PAGE_DRIVER_MAP0_ENTRY));
1782 if (mps_config_get_dpm_pg0(sc, &mpi_reply, sc->dpm_pg0,
1784 printf("%s: dpm page read failed; disabling dpm\n",
1786 if (retry_count < 3) {
1788 goto retry_read_dpm;
1790 sc->is_dpm_enable = 0;
1794 if (sc->is_dpm_enable)
1795 _mapping_process_dpm_pg0(sc);
1797 sc->track_mapping_events = 1;
1802 * mps_mapping_exit - clear mapping table and associated memory
1803 * @sc: per adapter object
1808 mps_mapping_exit(struct mps_softc *sc)
1810 _mapping_flush_dpm_pages(sc);
1811 mps_mapping_free_memory(sc);
1815 * mps_mapping_get_sas_id - assign a target id for sas device
1816 * @sc: per adapter object
1817 * @sas_address: sas address of the device
1818 * @handle: device handle
1820 * Returns valid ID on success or BAD_ID.
1823 mps_mapping_get_sas_id(struct mps_softc *sc, uint64_t sas_address, u16 handle)
1826 struct dev_mapping_table *mt_entry;
1828 for (map_idx = 0; map_idx < sc->max_devices; map_idx++) {
1829 mt_entry = &sc->mapping_table[map_idx];
1830 if (mt_entry->dev_handle == handle && mt_entry->physical_id ==
1832 return mt_entry->id;
1835 return MPS_MAP_BAD_ID;
1839 * mps_mapping_get_sas_id_from_handle - find a target id in mapping table using
1840 * only the dev handle. This is just a wrapper function for the local function
1841 * _mapping_get_mt_idx_from_handle.
1842 * @sc: per adapter object
1843 * @handle: device handle
1845 * Returns valid ID on success or BAD_ID.
1848 mps_mapping_get_sas_id_from_handle(struct mps_softc *sc, u16 handle)
1850 return (_mapping_get_mt_idx_from_handle(sc, handle));
1854 * mps_mapping_get_raid_id - assign a target id for raid device
1855 * @sc: per adapter object
1856 * @wwid: world wide identifier for raid volume
1857 * @handle: device handle
1859 * Returns valid ID on success or BAD_ID.
1862 mps_mapping_get_raid_id(struct mps_softc *sc, u64 wwid, u16 handle)
1865 struct dev_mapping_table *mt_entry;
1867 for (map_idx = 0; map_idx < sc->max_devices; map_idx++) {
1868 mt_entry = &sc->mapping_table[map_idx];
1869 if (mt_entry->dev_handle == handle && mt_entry->physical_id ==
1871 return mt_entry->id;
1874 return MPS_MAP_BAD_ID;
1878 * mps_mapping_get_raid_id_from_handle - find raid device in mapping table
1879 * using only the volume dev handle. This is just a wrapper function for the
1880 * local function _mapping_get_ir_mt_idx_from_handle.
1881 * @sc: per adapter object
1882 * @volHandle: volume device handle
1884 * Returns valid ID on success or BAD_ID.
1887 mps_mapping_get_raid_id_from_handle(struct mps_softc *sc, u16 volHandle)
1889 return (_mapping_get_ir_mt_idx_from_handle(sc, volHandle));
1893 * mps_mapping_enclosure_dev_status_change_event - handle enclosure events
1894 * @sc: per adapter object
1895 * @event_data: event data payload
1900 mps_mapping_enclosure_dev_status_change_event(struct mps_softc *sc,
1901 Mpi2EventDataSasEnclDevStatusChange_t *event_data)
1903 u8 enc_idx, missing_count;
1904 struct enc_mapping_table *et_entry;
1905 Mpi2DriverMap0Entry_t *dpm_entry;
1906 u16 ioc_pg8_flags = le16toh(sc->ioc_pg8.Flags);
1907 u8 map_shift = MPI2_DRVMAP0_MAPINFO_SLOT_SHIFT;
1908 u8 update_phy_bits = 0;
1910 uint64_t temp64_var;
1912 if ((ioc_pg8_flags & MPI2_IOCPAGE8_FLAGS_MASK_MAPPING_MODE) !=
1913 MPI2_IOCPAGE8_FLAGS_ENCLOSURE_SLOT_MAPPING)
1916 dpm_entry = (Mpi2DriverMap0Entry_t *)((u8 *)sc->dpm_pg0 +
1917 sizeof(MPI2_CONFIG_EXTENDED_PAGE_HEADER));
1919 if (event_data->ReasonCode == MPI2_EVENT_SAS_ENCL_RC_ADDED) {
1920 if (!event_data->NumSlots) {
1921 printf("%s: enclosure with handle = 0x%x reported 0 "
1922 "slots\n", __func__,
1923 le16toh(event_data->EnclosureHandle));
1926 temp64_var = event_data->EnclosureLogicalID.High;
1927 temp64_var = (temp64_var << 32) |
1928 event_data->EnclosureLogicalID.Low;
1929 enc_idx = _mapping_get_enc_idx_from_id(sc, temp64_var,
1930 event_data->PhyBits);
1931 if (enc_idx != MPS_ENCTABLE_BAD_IDX) {
1932 et_entry = &sc->enclosure_table[enc_idx];
1933 if (et_entry->init_complete &&
1934 !et_entry->missing_count) {
1935 printf("%s: enclosure %d is already present "
1936 "with handle = 0x%x\n",__func__, enc_idx,
1937 et_entry->enc_handle);
1940 et_entry->enc_handle = le16toh(event_data->
1942 et_entry->start_slot = le16toh(event_data->StartSlot);
1943 saved_phy_bits = et_entry->phy_bits;
1944 et_entry->phy_bits |= le32toh(event_data->PhyBits);
1945 if (saved_phy_bits != et_entry->phy_bits)
1946 update_phy_bits = 1;
1947 if (et_entry->missing_count || update_phy_bits) {
1948 et_entry->missing_count = 0;
1949 if (sc->is_dpm_enable &&
1950 et_entry->dpm_entry_num !=
1952 dpm_entry += et_entry->dpm_entry_num;
1954 (u8)(dpm_entry->MappingInformation &
1955 MPI2_DRVMAP0_MAPINFO_MISSING_MASK);
1956 if (!et_entry->init_complete && (
1957 missing_count || update_phy_bits)) {
1958 dpm_entry->MappingInformation
1959 = et_entry->num_slots;
1960 dpm_entry->MappingInformation
1962 dpm_entry->PhysicalBitsMapping
1963 = et_entry->phy_bits;
1964 sc->dpm_flush_entry[et_entry->
1970 enc_idx = sc->num_enc_table_entries;
1971 if (enc_idx >= sc->max_enclosures) {
1972 printf("%s: enclosure can not be added; "
1973 "mapping table is full\n", __func__);
1976 sc->num_enc_table_entries++;
1977 et_entry = &sc->enclosure_table[enc_idx];
1978 et_entry->enc_handle = le16toh(event_data->
1980 et_entry->enclosure_id = event_data->
1981 EnclosureLogicalID.High;
1982 et_entry->enclosure_id = ( et_entry->enclosure_id <<
1983 32) | event_data->EnclosureLogicalID.Low;
1984 et_entry->start_index = MPS_MAPTABLE_BAD_IDX;
1985 et_entry->dpm_entry_num = MPS_DPM_BAD_IDX;
1986 et_entry->num_slots = le16toh(event_data->NumSlots);
1987 et_entry->start_slot = le16toh(event_data->StartSlot);
1988 et_entry->phy_bits = le32toh(event_data->PhyBits);
1990 et_entry->init_complete = 1;
1991 } else if (event_data->ReasonCode ==
1992 MPI2_EVENT_SAS_ENCL_RC_NOT_RESPONDING) {
1993 enc_idx = _mapping_get_enc_idx_from_handle(sc,
1994 le16toh(event_data->EnclosureHandle));
1995 if (enc_idx == MPS_ENCTABLE_BAD_IDX) {
1996 printf("%s: cannot unmap enclosure %d because it has "
1997 "already been deleted", __func__, enc_idx);
2000 et_entry = &sc->enclosure_table[enc_idx];
2001 if (!et_entry->init_complete) {
2002 if (et_entry->missing_count < MPS_MAX_MISSING_COUNT)
2003 et_entry->missing_count++;
2005 et_entry->init_complete = 1;
2007 if (!et_entry->missing_count)
2008 et_entry->missing_count++;
2009 if (sc->is_dpm_enable && !et_entry->init_complete &&
2010 et_entry->dpm_entry_num != MPS_DPM_BAD_IDX) {
2011 dpm_entry += et_entry->dpm_entry_num;
2012 dpm_entry->MappingInformation = et_entry->num_slots;
2013 dpm_entry->MappingInformation <<= map_shift;
2014 dpm_entry->MappingInformation |=
2015 et_entry->missing_count;
2016 sc->dpm_flush_entry[et_entry->dpm_entry_num] = 1;
2018 et_entry->init_complete = 1;
2022 _mapping_flush_dpm_pages(sc);
2023 if (sc->pending_map_events)
2024 sc->pending_map_events--;
2028 * mps_mapping_topology_change_event - handle topology change events
2029 * @sc: per adapter object
2030 * @event_data: event data payload
2035 mps_mapping_topology_change_event(struct mps_softc *sc,
2036 Mpi2EventDataSasTopologyChangeList_t *event_data)
2038 struct _map_topology_change topo_change;
2039 struct _map_phy_change *phy_change;
2040 Mpi2EventSasTopoPhyEntry_t *event_phy_change;
2043 topo_change.enc_handle = le16toh(event_data->EnclosureHandle);
2044 topo_change.exp_handle = le16toh(event_data->ExpanderDevHandle);
2045 num_entries = event_data->NumEntries;
2046 topo_change.num_entries = num_entries;
2047 topo_change.start_phy_num = event_data->StartPhyNum;
2048 topo_change.num_phys = event_data->NumPhys;
2049 topo_change.exp_status = event_data->ExpStatus;
2050 event_phy_change = event_data->PHY;
2051 topo_change.phy_details = NULL;
2055 phy_change = malloc(sizeof(struct _map_phy_change) * num_entries,
2056 M_MPT2, M_NOWAIT|M_ZERO);
2057 topo_change.phy_details = phy_change;
2060 for (i = 0; i < num_entries; i++, event_phy_change++, phy_change++) {
2061 phy_change->dev_handle = le16toh(event_phy_change->
2063 phy_change->reason = event_phy_change->PhyStatus &
2064 MPI2_EVENT_SAS_TOPO_RC_MASK;
2066 _mapping_update_missing_count(sc, &topo_change);
2067 _mapping_get_dev_info(sc, &topo_change);
2068 _mapping_clear_removed_entries(sc);
2069 _mapping_add_new_device(sc, &topo_change);
2072 free(topo_change.phy_details, M_MPT2);
2073 _mapping_flush_dpm_pages(sc);
2074 if (sc->pending_map_events)
2075 sc->pending_map_events--;
2079 * _mapping_check_update_ir_mt_idx - Check and update IR map table index
2080 * @sc: per adapter object
2081 * @event_data: event data payload
2082 * @evt_idx: current event index
2083 * @map_idx: current index and the place holder for new map table index
2084 * @wwid_table: world wide name for volumes in the element table
2086 * pass through IR events and find whether any events matches and if so
2087 * tries to find new index if not returns failure
2089 * Returns 0 on success and 1 on failure
2092 _mapping_check_update_ir_mt_idx(struct mps_softc *sc,
2093 Mpi2EventDataIrConfigChangeList_t *event_data, int evt_idx, u32 *map_idx,
2096 struct dev_mapping_table *mt_entry;
2097 u32 st_idx, end_idx, mt_idx = *map_idx;
2099 Mpi2EventIrConfigElement_t *element;
2103 mt_entry = &sc->mapping_table[mt_idx];
2104 _mapping_get_ir_maprange(sc, &st_idx, &end_idx);
2107 for (i = evt_idx + 1; i < event_data->NumElements; i++) {
2108 element = (Mpi2EventIrConfigElement_t *)
2109 &event_data->ConfigElement[i];
2110 element_flags = le16toh(element->ElementFlags);
2111 if ((element_flags &
2112 MPI2_EVENT_IR_CHANGE_EFLAGS_ELEMENT_TYPE_MASK) !=
2113 MPI2_EVENT_IR_CHANGE_EFLAGS_VOLUME_ELEMENT)
2115 if (element->ReasonCode == MPI2_EVENT_IR_CHANGE_RC_ADDED ||
2116 element->ReasonCode ==
2117 MPI2_EVENT_IR_CHANGE_RC_VOLUME_CREATED) {
2118 if (mt_entry->physical_id == wwid_table[i]) {
2128 if (mt_idx > end_idx)
2130 mt_entry = &sc->mapping_table[mt_idx];
2131 } while (mt_entry->device_info & MPS_MAP_IN_USE);
2139 * mps_mapping_ir_config_change_event - handle IR config change list events
2140 * @sc: per adapter object
2141 * @event_data: event data payload
2146 mps_mapping_ir_config_change_event(struct mps_softc *sc,
2147 Mpi2EventDataIrConfigChangeList_t *event_data)
2149 Mpi2EventIrConfigElement_t *element;
2153 struct dev_mapping_table *mt_entry;
2155 u8 log_full_error = 0;
2157 wwid_table = malloc(sizeof(u64) * event_data->NumElements, M_MPT2,
2161 element = (Mpi2EventIrConfigElement_t *)&event_data->ConfigElement[0];
2162 flags = le32toh(event_data->Flags);
2163 for (i = 0; i < event_data->NumElements; i++, element++) {
2164 element_flags = le16toh(element->ElementFlags);
2165 if ((element->ReasonCode != MPI2_EVENT_IR_CHANGE_RC_ADDED) &&
2166 (element->ReasonCode != MPI2_EVENT_IR_CHANGE_RC_REMOVED) &&
2167 (element->ReasonCode != MPI2_EVENT_IR_CHANGE_RC_NO_CHANGE)
2168 && (element->ReasonCode !=
2169 MPI2_EVENT_IR_CHANGE_RC_VOLUME_CREATED))
2171 if ((element_flags &
2172 MPI2_EVENT_IR_CHANGE_EFLAGS_ELEMENT_TYPE_MASK) ==
2173 MPI2_EVENT_IR_CHANGE_EFLAGS_VOLUME_ELEMENT) {
2174 mps_config_get_volume_wwid(sc,
2175 le16toh(element->VolDevHandle), &wwid_table[i]);
2176 map_idx = _mapping_get_ir_mt_idx_from_wwid(sc,
2178 if (map_idx != MPS_MAPTABLE_BAD_IDX) {
2179 mt_entry = &sc->mapping_table[map_idx];
2180 mt_entry->device_info |= MPS_MAP_IN_USE;
2184 if (flags == MPI2_EVENT_IR_CHANGE_FLAGS_FOREIGN_CONFIG)
2187 element = (Mpi2EventIrConfigElement_t *)&event_data->
2189 for (i = 0; i < event_data->NumElements; i++, element++) {
2190 if (element->ReasonCode ==
2191 MPI2_EVENT_IR_CHANGE_RC_ADDED ||
2192 element->ReasonCode ==
2193 MPI2_EVENT_IR_CHANGE_RC_VOLUME_CREATED) {
2194 map_idx = _mapping_get_ir_mt_idx_from_wwid
2195 (sc, wwid_table[i]);
2196 if (map_idx != MPS_MAPTABLE_BAD_IDX) {
2197 mt_entry = &sc->mapping_table[map_idx];
2198 mt_entry->channel = MPS_RAID_CHANNEL;
2199 mt_entry->id = map_idx;
2200 mt_entry->dev_handle = le16toh
2201 (element->VolDevHandle);
2202 mt_entry->device_info =
2203 MPS_DEV_RESERVED | MPS_MAP_IN_USE;
2204 _mapping_update_ir_missing_cnt(sc,
2205 map_idx, element, wwid_table[i]);
2208 map_idx = _mapping_get_free_ir_mt_idx(sc);
2209 if (map_idx == MPS_MAPTABLE_BAD_IDX)
2211 else if (i < (event_data->NumElements - 1)) {
2213 _mapping_check_update_ir_mt_idx
2214 (sc, event_data, i, &map_idx,
2217 if (log_full_error) {
2218 printf("%s: no space to add the RAID "
2219 "volume with handle 0x%04x in "
2220 "mapping table\n", __func__, le16toh
2221 (element->VolDevHandle));
2224 mt_entry = &sc->mapping_table[map_idx];
2225 mt_entry->physical_id = wwid_table[i];
2226 mt_entry->channel = MPS_RAID_CHANNEL;
2227 mt_entry->id = map_idx;
2228 mt_entry->dev_handle = le16toh(element->
2230 mt_entry->device_info = MPS_DEV_RESERVED |
2232 mt_entry->init_complete = 0;
2233 _mapping_update_ir_missing_cnt(sc, map_idx,
2234 element, wwid_table[i]);
2235 } else if (element->ReasonCode ==
2236 MPI2_EVENT_IR_CHANGE_RC_REMOVED) {
2237 map_idx = _mapping_get_ir_mt_idx_from_wwid(sc,
2239 if (map_idx == MPS_MAPTABLE_BAD_IDX) {
2240 printf("%s: failed to remove a volume "
2241 "because it has already been "
2242 "removed\n", __func__);
2245 _mapping_update_ir_missing_cnt(sc, map_idx,
2246 element, wwid_table[i]);
2247 } else if (element->ReasonCode ==
2248 MPI2_EVENT_IR_CHANGE_RC_VOLUME_DELETED) {
2249 map_idx = _mapping_get_mt_idx_from_handle(sc,
2250 le16toh(element->VolDevHandle));
2251 if (map_idx == MPS_MAPTABLE_BAD_IDX) {
2252 printf("%s: failed to remove volume "
2253 "with handle 0x%04x because it has "
2254 "already been removed\n", __func__,
2255 le16toh(element->VolDevHandle));
2258 mt_entry = &sc->mapping_table[map_idx];
2259 _mapping_update_ir_missing_cnt(sc, map_idx,
2260 element, mt_entry->physical_id);
2266 _mapping_flush_dpm_pages(sc);
2267 free(wwid_table, M_MPT2);
2268 if (sc->pending_map_events)
2269 sc->pending_map_events--;