[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
|
@ -117,6 +117,8 @@ struct mptsas_hotplug_event {
|
||||||
struct mptsas_devinfo {
|
struct mptsas_devinfo {
|
||||||
u16 handle; /* unique id to address this device */
|
u16 handle; /* unique id to address this device */
|
||||||
u16 handle_parent; /* unique id to address parent 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 phy_id; /* phy number of parent device */
|
||||||
u8 port_id; /* sas physical port this device
|
u8 port_id; /* sas physical port this device
|
||||||
is assoc'd with */
|
is assoc'd with */
|
||||||
|
@ -146,6 +148,18 @@ struct mptsas_portinfo {
|
||||||
struct mptsas_phyinfo *phy_info;
|
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
|
#ifdef SASDEBUG
|
||||||
static void mptsas_print_phy_data(MPI_SAS_IO_UNIT0_PHY_DATA *phy_data)
|
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("---- SAS DEVICE PAGE 0 ---------\n");
|
||||||
printk("Handle=0x%X\n" ,le16_to_cpu(pg0->DevHandle));
|
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("Enclosure Handle=0x%X\n", le16_to_cpu(pg0->EnclosureHandle));
|
||||||
printk("Slot=0x%X\n", le16_to_cpu(pg0->Slot));
|
printk("Slot=0x%X\n", le16_to_cpu(pg0->Slot));
|
||||||
printk("SAS Address=0x%llX\n", le64_to_cpu(sas_address));
|
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)
|
#define mptsas_print_expander_pg1(pg1) do { } while (0)
|
||||||
#endif
|
#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
|
* 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,
|
.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)
|
static int mptsas_get_linkerrors(struct sas_phy *phy)
|
||||||
{
|
{
|
||||||
MPT_ADAPTER *ioc = phy_to_ioc(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;
|
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 = {
|
static struct sas_function_template mptsas_transport_functions = {
|
||||||
.get_linkerrors = mptsas_get_linkerrors,
|
.get_linkerrors = mptsas_get_linkerrors,
|
||||||
|
.get_enclosure_identifier = mptsas_get_enclosure_identifier,
|
||||||
|
.get_bay_identifier = mptsas_get_bay_identifier,
|
||||||
.phy_reset = mptsas_phy_reset,
|
.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 = le16_to_cpu(buffer->DevHandle);
|
||||||
device_info->handle_parent = le16_to_cpu(buffer->ParentDevHandle);
|
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->phy_id = buffer->PhyNum;
|
||||||
device_info->port_id = buffer->PhysicalPort;
|
device_info->port_id = buffer->PhysicalPort;
|
||||||
device_info->id = buffer->TargetID;
|
device_info->id = buffer->TargetID;
|
||||||
|
@ -1335,29 +1482,15 @@ mptsas_hotplug_work(void *arg)
|
||||||
case MPTSAS_ADD_DEVICE:
|
case MPTSAS_ADD_DEVICE:
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* When there is no sas address,
|
* Refresh sas device pg0 data
|
||||||
* 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
|
|
||||||
*/
|
*/
|
||||||
if (!ev->sas_address) {
|
if (mptsas_sas_device_pg0(ioc, &sas_device,
|
||||||
if (mptsas_sas_device_pg0(ioc,
|
|
||||||
&sas_device, ev->id,
|
|
||||||
(MPI_SAS_DEVICE_PGAD_FORM_BUS_TARGET_ID <<
|
(MPI_SAS_DEVICE_PGAD_FORM_BUS_TARGET_ID <<
|
||||||
MPI_SAS_DEVICE_PGAD_FORM_SHIFT)))
|
MPI_SAS_DEVICE_PGAD_FORM_SHIFT), ev->id))
|
||||||
break;
|
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;
|
|
||||||
}
|
|
||||||
|
|
||||||
phy_info = mptsas_find_phyinfo_by_parent(ioc,
|
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) {
|
if (!phy_info) {
|
||||||
printk("mptsas: add event for non-existant PHY.\n");
|
printk("mptsas: add event for non-existant PHY.\n");
|
||||||
break;
|
break;
|
||||||
|
@ -1368,14 +1501,8 @@ mptsas_hotplug_work(void *arg)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* fill attached info */
|
memcpy(&phy_info->attached, &sas_device,
|
||||||
phy_info->attached.handle = ev->handle;
|
sizeof(struct mptsas_devinfo));
|
||||||
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;
|
|
||||||
|
|
||||||
if (phy_info->attached.device_info & MPI_SAS_DEVICE_INFO_SSP_TARGET)
|
if (phy_info->attached.device_info & MPI_SAS_DEVICE_INFO_SSP_TARGET)
|
||||||
ds = "ssp";
|
ds = "ssp";
|
||||||
|
@ -1393,7 +1520,6 @@ mptsas_hotplug_work(void *arg)
|
||||||
if (!rphy)
|
if (!rphy)
|
||||||
break; /* non-fatal: an rphy can be added later */
|
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);
|
mptsas_parse_device_info(&rphy->identify, &phy_info->attached);
|
||||||
if (sas_rphy_add(rphy)) {
|
if (sas_rphy_add(rphy)) {
|
||||||
sas_rphy_free(rphy);
|
sas_rphy_free(rphy);
|
||||||
|
|
Loading…
Reference in New Issue