[SCSI] mptsas: add support for enclosure and bay identifier attributes
Adds support to retrieve the enclosure and bay identifiers. This patch is from Eric with minor modifications from me, rewritten from a buggy patch of mine, based on the earlier CSMI implementation from Eric.. Signed-off-by: Christoph Hellwig <hch@lst.de> Signed-off-by: James Bottomley <James.Bottomley@SteelEye.com>
This commit is contained in:
parent
ae198df377
commit
e3094447e0
1 changed files with 161 additions and 35 deletions
|
@ -117,6 +117,8 @@ struct mptsas_hotplug_event {
|
|||
struct mptsas_devinfo {
|
||||
u16 handle; /* unique id to address this device */
|
||||
u16 handle_parent; /* unique id to address parent device */
|
||||
u16 handle_enclosure; /* enclosure identifier of the enclosure */
|
||||
u16 slot; /* physical slot in enclosure */
|
||||
u8 phy_id; /* phy number of parent device */
|
||||
u8 port_id; /* sas physical port this device
|
||||
is assoc'd with */
|
||||
|
@ -146,6 +148,18 @@ struct mptsas_portinfo {
|
|||
struct mptsas_phyinfo *phy_info;
|
||||
};
|
||||
|
||||
struct mptsas_enclosure {
|
||||
u64 enclosure_logical_id; /* The WWN for the enclosure */
|
||||
u16 enclosure_handle; /* unique id to address this */
|
||||
u16 flags; /* details enclosure management */
|
||||
u16 num_slot; /* num slots */
|
||||
u16 start_slot; /* first slot */
|
||||
u8 start_id; /* starting logical target id */
|
||||
u8 start_channel; /* starting logical channel id */
|
||||
u8 sep_id; /* SEP device logical target id */
|
||||
u8 sep_channel; /* SEP channel logical channel id */
|
||||
};
|
||||
|
||||
|
||||
#ifdef SASDEBUG
|
||||
static void mptsas_print_phy_data(MPI_SAS_IO_UNIT0_PHY_DATA *phy_data)
|
||||
|
@ -205,6 +219,7 @@ static void mptsas_print_device_pg0(SasDevicePage0_t *pg0)
|
|||
|
||||
printk("---- SAS DEVICE PAGE 0 ---------\n");
|
||||
printk("Handle=0x%X\n" ,le16_to_cpu(pg0->DevHandle));
|
||||
printk("Parent Handle=0x%X\n" ,le16_to_cpu(pg0->ParentDevHandle));
|
||||
printk("Enclosure Handle=0x%X\n", le16_to_cpu(pg0->EnclosureHandle));
|
||||
printk("Slot=0x%X\n", le16_to_cpu(pg0->Slot));
|
||||
printk("SAS Address=0x%llX\n", le64_to_cpu(sas_address));
|
||||
|
@ -243,6 +258,82 @@ static void mptsas_print_expander_pg1(SasExpanderPage1_t *pg1)
|
|||
#define mptsas_print_expander_pg1(pg1) do { } while (0)
|
||||
#endif
|
||||
|
||||
static inline MPT_ADAPTER *phy_to_ioc(struct sas_phy *phy)
|
||||
{
|
||||
struct Scsi_Host *shost = dev_to_shost(phy->dev.parent);
|
||||
return ((MPT_SCSI_HOST *)shost->hostdata)->ioc;
|
||||
}
|
||||
|
||||
static inline MPT_ADAPTER *rphy_to_ioc(struct sas_rphy *rphy)
|
||||
{
|
||||
struct Scsi_Host *shost = dev_to_shost(rphy->dev.parent->parent);
|
||||
return ((MPT_SCSI_HOST *)shost->hostdata)->ioc;
|
||||
}
|
||||
|
||||
static int
|
||||
mptsas_sas_exclosure_pg0(MPT_ADAPTER *ioc, struct mptsas_enclosure *enclosure,
|
||||
u32 form, u32 form_specific)
|
||||
{
|
||||
ConfigExtendedPageHeader_t hdr;
|
||||
CONFIGPARMS cfg;
|
||||
SasEnclosurePage0_t *buffer;
|
||||
dma_addr_t dma_handle;
|
||||
int error;
|
||||
__le64 le_identifier;
|
||||
|
||||
memset(&hdr, 0, sizeof(hdr));
|
||||
hdr.PageVersion = MPI_SASENCLOSURE0_PAGEVERSION;
|
||||
hdr.PageNumber = 0;
|
||||
hdr.PageType = MPI_CONFIG_PAGETYPE_EXTENDED;
|
||||
hdr.ExtPageType = MPI_CONFIG_EXTPAGETYPE_ENCLOSURE;
|
||||
|
||||
cfg.cfghdr.ehdr = &hdr;
|
||||
cfg.physAddr = -1;
|
||||
cfg.pageAddr = form + form_specific;
|
||||
cfg.action = MPI_CONFIG_ACTION_PAGE_HEADER;
|
||||
cfg.dir = 0; /* read */
|
||||
cfg.timeout = 10;
|
||||
|
||||
error = mpt_config(ioc, &cfg);
|
||||
if (error)
|
||||
goto out;
|
||||
if (!hdr.ExtPageLength) {
|
||||
error = -ENXIO;
|
||||
goto out;
|
||||
}
|
||||
|
||||
buffer = pci_alloc_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
|
||||
&dma_handle);
|
||||
if (!buffer) {
|
||||
error = -ENOMEM;
|
||||
goto out;
|
||||
}
|
||||
|
||||
cfg.physAddr = dma_handle;
|
||||
cfg.action = MPI_CONFIG_ACTION_PAGE_READ_CURRENT;
|
||||
|
||||
error = mpt_config(ioc, &cfg);
|
||||
if (error)
|
||||
goto out_free_consistent;
|
||||
|
||||
/* save config data */
|
||||
memcpy(&le_identifier, &buffer->EnclosureLogicalID, sizeof(__le64));
|
||||
enclosure->enclosure_logical_id = le64_to_cpu(le_identifier);
|
||||
enclosure->enclosure_handle = le16_to_cpu(buffer->EnclosureHandle);
|
||||
enclosure->flags = le16_to_cpu(buffer->Flags);
|
||||
enclosure->num_slot = le16_to_cpu(buffer->NumSlots);
|
||||
enclosure->start_slot = le16_to_cpu(buffer->StartSlot);
|
||||
enclosure->start_id = buffer->StartTargetID;
|
||||
enclosure->start_channel = buffer->StartBus;
|
||||
enclosure->sep_id = buffer->SEPTargetID;
|
||||
enclosure->sep_channel = buffer->SEPBus;
|
||||
|
||||
out_free_consistent:
|
||||
pci_free_consistent(ioc->pcidev, hdr.ExtPageLength * 4,
|
||||
buffer, dma_handle);
|
||||
out:
|
||||
return error;
|
||||
}
|
||||
|
||||
/*
|
||||
* This is pretty ugly. We will be able to seriously clean it up
|
||||
|
@ -399,12 +490,6 @@ static struct scsi_host_template mptsas_driver_template = {
|
|||
.use_clustering = ENABLE_CLUSTERING,
|
||||
};
|
||||
|
||||
static inline MPT_ADAPTER *phy_to_ioc(struct sas_phy *phy)
|
||||
{
|
||||
struct Scsi_Host *shost = dev_to_shost(phy->dev.parent);
|
||||
return ((MPT_SCSI_HOST *)shost->hostdata)->ioc;
|
||||
}
|
||||
|
||||
static int mptsas_get_linkerrors(struct sas_phy *phy)
|
||||
{
|
||||
MPT_ADAPTER *ioc = phy_to_ioc(phy);
|
||||
|
@ -546,8 +631,67 @@ static int mptsas_phy_reset(struct sas_phy *phy, int hard_reset)
|
|||
return error;
|
||||
}
|
||||
|
||||
static int
|
||||
mptsas_get_enclosure_identifier(struct sas_rphy *rphy, u64 *identifier)
|
||||
{
|
||||
MPT_ADAPTER *ioc = rphy_to_ioc(rphy);
|
||||
int i, error;
|
||||
struct mptsas_portinfo *p;
|
||||
struct mptsas_enclosure enclosure_info;
|
||||
u64 enclosure_handle;
|
||||
|
||||
mutex_lock(&ioc->sas_topology_mutex);
|
||||
list_for_each_entry(p, &ioc->sas_topology, list) {
|
||||
for (i = 0; i < p->num_phys; i++) {
|
||||
if (p->phy_info[i].attached.sas_address ==
|
||||
rphy->identify.sas_address) {
|
||||
enclosure_handle = p->phy_info[i].
|
||||
attached.handle_enclosure;
|
||||
goto found_info;
|
||||
}
|
||||
}
|
||||
}
|
||||
mutex_unlock(&ioc->sas_topology_mutex);
|
||||
return -ENXIO;
|
||||
|
||||
found_info:
|
||||
mutex_unlock(&ioc->sas_topology_mutex);
|
||||
memset(&enclosure_info, 0, sizeof(struct mptsas_enclosure));
|
||||
error = mptsas_sas_exclosure_pg0(ioc, &enclosure_info,
|
||||
(MPI_SAS_ENCLOS_PGAD_FORM_HANDLE <<
|
||||
MPI_SAS_ENCLOS_PGAD_FORM_SHIFT), enclosure_handle);
|
||||
if (!error)
|
||||
*identifier = enclosure_info.enclosure_logical_id;
|
||||
return error;
|
||||
}
|
||||
|
||||
static int
|
||||
mptsas_get_bay_identifier(struct sas_rphy *rphy)
|
||||
{
|
||||
MPT_ADAPTER *ioc = rphy_to_ioc(rphy);
|
||||
struct mptsas_portinfo *p;
|
||||
int i, rc;
|
||||
|
||||
mutex_lock(&ioc->sas_topology_mutex);
|
||||
list_for_each_entry(p, &ioc->sas_topology, list) {
|
||||
for (i = 0; i < p->num_phys; i++) {
|
||||
if (p->phy_info[i].attached.sas_address ==
|
||||
rphy->identify.sas_address) {
|
||||
rc = p->phy_info[i].attached.slot;
|
||||
goto out;
|
||||
}
|
||||
}
|
||||
}
|
||||
rc = -ENXIO;
|
||||
out:
|
||||
mutex_unlock(&ioc->sas_topology_mutex);
|
||||
return rc;
|
||||
}
|
||||
|
||||
static struct sas_function_template mptsas_transport_functions = {
|
||||
.get_linkerrors = mptsas_get_linkerrors,
|
||||
.get_enclosure_identifier = mptsas_get_enclosure_identifier,
|
||||
.get_bay_identifier = mptsas_get_bay_identifier,
|
||||
.phy_reset = mptsas_phy_reset,
|
||||
};
|
||||
|
||||
|
@ -739,6 +883,9 @@ mptsas_sas_device_pg0(MPT_ADAPTER *ioc, struct mptsas_devinfo *device_info,
|
|||
|
||||
device_info->handle = le16_to_cpu(buffer->DevHandle);
|
||||
device_info->handle_parent = le16_to_cpu(buffer->ParentDevHandle);
|
||||
device_info->handle_enclosure =
|
||||
le16_to_cpu(buffer->EnclosureHandle);
|
||||
device_info->slot = le16_to_cpu(buffer->Slot);
|
||||
device_info->phy_id = buffer->PhyNum;
|
||||
device_info->port_id = buffer->PhysicalPort;
|
||||
device_info->id = buffer->TargetID;
|
||||
|
@ -1335,29 +1482,15 @@ mptsas_hotplug_work(void *arg)
|
|||
case MPTSAS_ADD_DEVICE:
|
||||
|
||||
/*
|
||||
* When there is no sas address,
|
||||
* RAID volumes are being deleted,
|
||||
* and hidden phy disk are being added.
|
||||
* We don't know the SAS data yet,
|
||||
* so lookup sas device page to get
|
||||
* pertaining info
|
||||
* Refresh sas device pg0 data
|
||||
*/
|
||||
if (!ev->sas_address) {
|
||||
if (mptsas_sas_device_pg0(ioc,
|
||||
&sas_device, ev->id,
|
||||
(MPI_SAS_DEVICE_PGAD_FORM_BUS_TARGET_ID <<
|
||||
MPI_SAS_DEVICE_PGAD_FORM_SHIFT)))
|
||||
break;
|
||||
ev->handle = sas_device.handle;
|
||||
ev->parent_handle = sas_device.handle_parent;
|
||||
ev->channel = sas_device.channel;
|
||||
ev->phy_id = sas_device.phy_id;
|
||||
ev->sas_address = sas_device.sas_address;
|
||||
ev->device_info = sas_device.device_info;
|
||||
}
|
||||
if (mptsas_sas_device_pg0(ioc, &sas_device,
|
||||
(MPI_SAS_DEVICE_PGAD_FORM_BUS_TARGET_ID <<
|
||||
MPI_SAS_DEVICE_PGAD_FORM_SHIFT), ev->id))
|
||||
break;
|
||||
|
||||
phy_info = mptsas_find_phyinfo_by_parent(ioc,
|
||||
ev->parent_handle, ev->phy_id);
|
||||
sas_device.handle_parent, sas_device.phy_id);
|
||||
if (!phy_info) {
|
||||
printk("mptsas: add event for non-existant PHY.\n");
|
||||
break;
|
||||
|
@ -1368,14 +1501,8 @@ mptsas_hotplug_work(void *arg)
|
|||
break;
|
||||
}
|
||||
|
||||
/* fill attached info */
|
||||
phy_info->attached.handle = ev->handle;
|
||||
phy_info->attached.phy_id = ev->phy_id;
|
||||
phy_info->attached.port_id = phy_info->identify.port_id;
|
||||
phy_info->attached.id = ev->id;
|
||||
phy_info->attached.channel = ev->channel;
|
||||
phy_info->attached.sas_address = ev->sas_address;
|
||||
phy_info->attached.device_info = ev->device_info;
|
||||
memcpy(&phy_info->attached, &sas_device,
|
||||
sizeof(struct mptsas_devinfo));
|
||||
|
||||
if (phy_info->attached.device_info & MPI_SAS_DEVICE_INFO_SSP_TARGET)
|
||||
ds = "ssp";
|
||||
|
@ -1393,7 +1520,6 @@ mptsas_hotplug_work(void *arg)
|
|||
if (!rphy)
|
||||
break; /* non-fatal: an rphy can be added later */
|
||||
|
||||
rphy->scsi_target_id = phy_info->attached.id;
|
||||
mptsas_parse_device_info(&rphy->identify, &phy_info->attached);
|
||||
if (sas_rphy_add(rphy)) {
|
||||
sas_rphy_free(rphy);
|
||||
|
|
Loading…
Reference in a new issue