[PATCH v2 7/9] pm80xx: wait a minimum of 500ms before issuing commands to SPCv

2015-11-02 Thread Benjamin Rood
The documentation for the 8070 and 8072 SPCv chip explicitly states that
a minimum of 500ms must elapse before issuing commands, otherwise the
SPCv may not process them and the firmware may get into an unrecoverable
state requiring a reboot.  While the Linux guys will probably think this
is 'racy', it is called out in the chip documentation and inserting this
delay makes power management function properly.

Signed-off-by: Benjamin Rood <br...@attotech.com>
---
Changes from v1 -> v2:
-Used mdelay(500) instead of a loop to facilitate required delay
-Added comment as to why this behavior exists

 drivers/scsi/pm8001/pm8001_init.c | 13 +
 1 file changed, 13 insertions(+)

diff --git a/drivers/scsi/pm8001/pm8001_init.c 
b/drivers/scsi/pm8001/pm8001_init.c
index 7ce7ea3..d147c41 100644
--- a/drivers/scsi/pm8001/pm8001_init.c
+++ b/drivers/scsi/pm8001/pm8001_init.c
@@ -1243,6 +1243,19 @@ static int pm8001_pci_resume(struct pci_dev *pdev)
for (i = 1; i < pm8001_ha->number_of_intr; i++)
PM8001_CHIP_DISP->interrupt_enable(pm8001_ha, i);
}
+
+   /* Chip documentation for the 8070 and 8072 SPCv*/
+   /* states that a 500ms minimum delay is required*/
+   /* before issuing commands.  Otherwise, the firmare */
+   /* will enter an unrecoverable state.   */
+
+   if (pm8001_ha->chip_id == chip_8070 ||
+   pm8001_ha->chip_id == chip_8072) {
+   mdelay(500);
+   }
+
+   /* Spin up the PHYs */
+
pm8001_ha->flags = PM8001F_RUN_TIME;
for (i = 0; i < pm8001_ha->chip->n_phy; i++) {
pm8001_ha->phy[i].enable_completion = 
-- 
2.4.3

--
To unsubscribe from this list: send the line "unsubscribe linux-scsi" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[PATCH v2 4/9] pm80xx: add support for ATTO devices during SAS address initiailization

2015-11-02 Thread Benjamin Rood
ATTO SAS controllers retrieve the SAS address from the NVRAM in a location
different from non-ATTO PMC Sierra SAS controllers.  This patch makes the
necessary adjustments in order to retrieve the SAS address on these types
of adapters.

Signed-off-by: Benjamin Rood <br...@attotech.com>
---
Changes from v1 -> v2:
-Fixed indentation issues noted by Hannes and Jack

 drivers/scsi/pm8001/pm8001_init.c | 10 ++
 1 file changed, 10 insertions(+)

diff --git a/drivers/scsi/pm8001/pm8001_init.c 
b/drivers/scsi/pm8001/pm8001_init.c
index feaf504..861416f 100644
--- a/drivers/scsi/pm8001/pm8001_init.c
+++ b/drivers/scsi/pm8001/pm8001_init.c
@@ -636,6 +636,11 @@ static void pm8001_init_sas_add(struct pm8001_hba_info 
*pm8001_ha)
payload.minor_function = 0;
payload.length = 128;
}
+   } else if ((pm8001_ha->chip_id == chip_8070 ||
+   pm8001_ha->chip_id == chip_8072) &&
+   pm8001_ha->pdev->subsystem_vendor == 
PCI_VENDOR_ID_ATTO) {
+   payload.minor_function = 4;
+   payload.length = 4096;
} else {
payload.minor_function = 1;
payload.length = 4096;
@@ -662,6 +667,11 @@ static void pm8001_init_sas_add(struct pm8001_hba_info 
*pm8001_ha)
else if (deviceid == 0x0042)
pm8001_ha->sas_addr[j] =
payload.func_specific[0x010 + i];
+   } else if ((pm8001_ha->chip_id == chip_8070 ||
+   pm8001_ha->chip_id == chip_8072) &&
+   pm8001_ha->pdev->subsystem_vendor == 
PCI_VENDOR_ID_ATTO) {
+   pm8001_ha->sas_addr[j] =
+   payload.func_specific[0x010 + i];
} else
pm8001_ha->sas_addr[j] =
payload.func_specific[0x804 + i];
-- 
2.4.3

--
To unsubscribe from this list: send the line "unsubscribe linux-scsi" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[PATCH 0/8] pm80xx: Add ATTO 12Gb HBA support and fix various issues

2015-10-30 Thread Benjamin Rood
The following series of patches are primarily aimed at adding support for
ATTO's 12Gb SAS adapters, which are based off of the PMC-Sierra 8070/8072
series chips.  I have also addressed other various issues that were
discovered during our testing and verification phase, mainly with
resume-from-sleep, configuring PHY profiles, and using legacy interrupts.

Diffstat:
drivers/scsi/pm8001/pm8001_defs.h |   2 ++
drivers/scsi/pm8001/pm8001_init.c | 212 
+++
drivers/scsi/pm8001/pm8001_sas.h  |   6 +++-
drivers/scsi/pm8001/pm80xx_hwi.c  |  34 ++
4 files changed, 241 insertions(+), 13 deletions(-)

--
To unsubscribe from this list: send the line "unsubscribe linux-scsi" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[PATCH 7/8] pm80xx: wait a minimum of 500ms before issuing commands to SPCv

2015-10-30 Thread Benjamin Rood
The documentation for the 8070 and 8072 SPCv chip explicitly states that
a minimum of 500ms must elapse before issuing commands, otherwise the
SPCv may not process them and the firmware may get into an unrecoverable
state requiring a reboot.  While the Linux guys will probably think this
is 'racy', it is called out in the chip documentation and inserting this
delay makes power management function properly.

Signed-off-by: Benjamin Rood <br...@attotech.com>
---
 drivers/scsi/pm8001/pm8001_init.c | 12 
 1 file changed, 12 insertions(+)

diff --git a/drivers/scsi/pm8001/pm8001_init.c 
b/drivers/scsi/pm8001/pm8001_init.c
index a0e55d4..ab99984 100644
--- a/drivers/scsi/pm8001/pm8001_init.c
+++ b/drivers/scsi/pm8001/pm8001_init.c
@@ -1190,6 +1190,7 @@ static int pm8001_pci_resume(struct pci_dev *pdev)
int rc;
u8 i = 0, j;
u32 device_state;
+   u32 wait_count;
DECLARE_COMPLETION_ONSTACK(completion);
pm8001_ha = sha->lldd_ha;
device_state = pdev->current_state;
@@ -1243,6 +1244,17 @@ static int pm8001_pci_resume(struct pci_dev *pdev)
for (i = 1; i < pm8001_ha->number_of_intr; i++)
PM8001_CHIP_DISP->interrupt_enable(pm8001_ha, i);
}
+
+   if (pm8001_ha->chip_id == chip_8070 ||
+   pm8001_ha->chip_id == chip_8072) {
+   wait_count = 500;
+   do {
+   mdelay(1);
+   } while (--wait_count);
+   }
+
+   /* Spin up the PHYs */
+
pm8001_ha->flags = PM8001F_RUN_TIME;
for (i = 0; i < pm8001_ha->chip->n_phy; i++) {
pm8001_ha->phy[i].enable_completion = 
-- 
2.4.3

--
To unsubscribe from this list: send the line "unsubscribe linux-scsi" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[PATCH 3/8] pm80xx: add ATTO PCI IDs to pm8001_pci_table

2015-10-30 Thread Benjamin Rood
These PCI IDs allow the pm8001 driver to load against ATTO 12Gb SAS
controllers that use PMC Sierra 8070 and PMC Sierra 8072 SAS chips.

Signed-off-by: Benjamin Rood <br...@attotech.com>
---
 drivers/scsi/pm8001/pm8001_init.c | 14 ++
 1 file changed, 14 insertions(+)

diff --git a/drivers/scsi/pm8001/pm8001_init.c 
b/drivers/scsi/pm8001/pm8001_init.c
index 2106ac3..feaf504 100644
--- a/drivers/scsi/pm8001/pm8001_init.c
+++ b/drivers/scsi/pm8001/pm8001_init.c
@@ -1181,6 +1181,20 @@ static struct pci_device_id pm8001_pci_table[] = {
PCI_VENDOR_ID_ADAPTEC2, 0x0808, 0, 0, chip_8077 },
{ PCI_VENDOR_ID_ADAPTEC2, 0x8074,
PCI_VENDOR_ID_ADAPTEC2, 0x0404, 0, 0, chip_8074 },
+   { PCI_VENDOR_ID_ATTO, 0x8070,
+   PCI_VENDOR_ID_ATTO, 0x0070, 0, 0, chip_8070 },
+   { PCI_VENDOR_ID_ATTO, 0x8070,
+   PCI_VENDOR_ID_ATTO, 0x0071, 0, 0, chip_8070 },
+   { PCI_VENDOR_ID_ATTO, 0x8072,
+   PCI_VENDOR_ID_ATTO, 0x0072, 0, 0, chip_8072 },
+   { PCI_VENDOR_ID_ATTO, 0x8072,
+   PCI_VENDOR_ID_ATTO, 0x0073, 0, 0, chip_8072 },
+   { PCI_VENDOR_ID_ATTO, 0x8070,
+   PCI_VENDOR_ID_ATTO, 0x0080, 0, 0, chip_8070 },
+   { PCI_VENDOR_ID_ATTO, 0x8072,
+   PCI_VENDOR_ID_ATTO, 0x0081, 0, 0, chip_8072 },
+   { PCI_VENDOR_ID_ATTO, 0x8072,
+   PCI_VENDOR_ID_ATTO, 0x0082, 0, 0, chip_8072 },
{} /* terminate list */
 };
 
-- 
2.4.3

--
To unsubscribe from this list: send the line "unsubscribe linux-scsi" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[PATCH 1/8] pm80xx: configure PHY settings based on subsystem vendor ID

2015-10-30 Thread Benjamin Rood
Previuosly, all PMC Sierra 80xx controllers are assumed to be a
motherboard controller, except if the subsystem vendor ID was equal to
PCI_VENDOR_ID_ADAPTEC.  The driver then attempts to load PHY settings
from NVRAM.  While this may be correct behavior for most controllers, it
does not work with Adaptec and ATTO controllers since they do not store
PHY settings in NVRAM and choose to use either custom PHY settings or chip
defaults.  Loading random values from NVRAM may cause the controllers to
malfunction in this edge case.

Signed-off-by: Benjamin Rood <br...@attotech.com>
---
 drivers/scsi/pm8001/pm8001_init.c | 26 --
 1 file changed, 20 insertions(+), 6 deletions(-)

diff --git a/drivers/scsi/pm8001/pm8001_init.c 
b/drivers/scsi/pm8001/pm8001_init.c
index 5c0356f..8c094fd 100644
--- a/drivers/scsi/pm8001/pm8001_init.c
+++ b/drivers/scsi/pm8001/pm8001_init.c
@@ -720,6 +720,23 @@ static int pm8001_get_phy_settings_info(struct 
pm8001_hba_info *pm8001_ha)
return 0;
 }
 
+/**
+ * pm8001_configure_phy_settings : Configures PHY settings based on vendor ID.
+ * @pm8001_ha : our hba.
+ */
+static int pm8001_configure_phy_settings(struct pm8001_hba_info *pm8001_ha)
+{
+   switch (pm8001_ha->pdev->subsystem_vendor) {
+   case PCI_VENDOR_ID_ATTO:
+   case PCI_VENDOR_ID_ADAPTEC2:
+   case 0:
+   return 0;
+
+   default:
+   return pm8001_get_phy_settings_info(pm8001_ha);
+   }
+}
+
 #ifdef PM8001_USE_MSIX
 /**
  * pm8001_setup_msix - enable MSI-X interrupt
@@ -902,12 +919,9 @@ static int pm8001_pci_probe(struct pci_dev *pdev,
 
pm8001_init_sas_add(pm8001_ha);
/* phy setting support for motherboard controller */
-   if (pdev->subsystem_vendor != PCI_VENDOR_ID_ADAPTEC2 &&
-   pdev->subsystem_vendor != 0) {
-   rc = pm8001_get_phy_settings_info(pm8001_ha);
-   if (rc)
-   goto err_out_shost;
-   }
+   if (pm8001_configure_phy_settings(pm8001_ha))
+   goto err_out_shost;
+
pm8001_post_sas_ha_init(shost, chip);
rc = sas_register_ha(SHOST_TO_SAS_HA(shost));
if (rc)
-- 
2.4.3

--
To unsubscribe from this list: send the line "unsubscribe linux-scsi" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[PATCH 8/8] pm80xx: avoid a panic if MSI(X) interrupts are disabled

2015-10-30 Thread Benjamin Rood
If MSI(X) interrupts are disabled via the kernel command line
(pci=nomsi), the pm8001 driver will kernel panic because it does not
detect that MSI interrupts are disabled and will soldier on and attempt to
configure MSI interrupts anyways.  This leads to a kernel panic, most
likely because a required data structure is not available down the
line.  Using the pci_msi_enabled() function in order to detect if MSI
interrupts are enabled before configuring them resolves this issue and
avoids a kernel panic when the module is loaded.  Additionally, the
irq_vector structure must be initialized when legacy interrupts are
being used otherwise legacy interrupts will simply not function and
result in another panic.

Signed-off-by: Benjamin Rood <br...@attotech.com>
---
 drivers/scsi/pm8001/pm8001_init.c | 16 +++-
 1 file changed, 11 insertions(+), 5 deletions(-)

diff --git a/drivers/scsi/pm8001/pm8001_init.c 
b/drivers/scsi/pm8001/pm8001_init.c
index ab99984..bdc624f 100644
--- a/drivers/scsi/pm8001/pm8001_init.c
+++ b/drivers/scsi/pm8001/pm8001_init.c
@@ -482,7 +482,8 @@ static struct pm8001_hba_info *pm8001_pci_alloc(struct 
pci_dev *pdev,
 
 #ifdef PM8001_USE_TASKLET
/* Tasklet for non msi-x interrupt handler */
-   if ((!pdev->msix_cap) || (pm8001_ha->chip_id == chip_8001))
+   if ((!pdev->msix_cap || !pci_msi_enabled())
+   || (pm8001_ha->chip_id == chip_8001))
tasklet_init(_ha->tasklet[0], pm8001_tasklet,
(unsigned long)&(pm8001_ha->irq_vector[0]));
else
@@ -951,7 +952,7 @@ static u32 pm8001_request_irq(struct pm8001_hba_info 
*pm8001_ha)
pdev = pm8001_ha->pdev;
 
 #ifdef PM8001_USE_MSIX
-   if (pdev->msix_cap)
+   if (pdev->msix_cap && pci_msi_enabled())
return pm8001_setup_msix(pm8001_ha);
else {
PM8001_INIT_DBG(pm8001_ha,
@@ -962,6 +963,8 @@ static u32 pm8001_request_irq(struct pm8001_hba_info 
*pm8001_ha)
 
 intx:
/* initialize the INT-X interrupt */
+   pm8001_ha->irq_vector[0].irq_id = 0;
+   pm8001_ha->irq_vector[0].drv_inst = pm8001_ha;
rc = request_irq(pdev->irq, pm8001_interrupt_handler_intx, IRQF_SHARED,
DRV_NAME, SHOST_TO_SAS_HA(pm8001_ha->shost));
return rc;
@@ -1112,7 +1115,8 @@ static void pm8001_pci_remove(struct pci_dev *pdev)
 #endif
 #ifdef PM8001_USE_TASKLET
/* For non-msix and msix interrupts */
-   if ((!pdev->msix_cap) || (pm8001_ha->chip_id == chip_8001))
+   if ((!pdev->msix_cap || !pci_msi_enabled()) ||
+   (pm8001_ha->chip_id == chip_8001))
tasklet_kill(_ha->tasklet[0]);
else
for (j = 0; j < PM8001_MAX_MSIX_VEC; j++)
@@ -1161,7 +1165,8 @@ static int pm8001_pci_suspend(struct pci_dev *pdev, 
pm_message_t state)
 #endif
 #ifdef PM8001_USE_TASKLET
/* For non-msix and msix interrupts */
-   if ((!pdev->msix_cap) || (pm8001_ha->chip_id == chip_8001))
+   if ((!pdev->msix_cap || !pci_msi_enabled()) ||
+   (pm8001_ha->chip_id == chip_8001))
tasklet_kill(_ha->tasklet[0]);
else
for (j = 0; j < PM8001_MAX_MSIX_VEC; j++)
@@ -1231,7 +1236,8 @@ static int pm8001_pci_resume(struct pci_dev *pdev)
goto err_out_disable;
 #ifdef PM8001_USE_TASKLET
/*  Tasklet for non msi-x interrupt handler */
-   if ((!pdev->msix_cap) || (pm8001_ha->chip_id == chip_8001))
+   if ((!pdev->msix_cap || !pci_msi_enabled()) ||
+   (pm8001_ha->chip_id == chip_8001))
tasklet_init(_ha->tasklet[0], pm8001_tasklet,
(unsigned long)&(pm8001_ha->irq_vector[0]));
else
-- 
2.4.3

--
To unsubscribe from this list: send the line "unsubscribe linux-scsi" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[PATCH 4/8] pm80xx: add support for ATTO devices during SAS address initiailization

2015-10-30 Thread Benjamin Rood
ATTO SAS controllers retrieve the SAS address from the NVRAM in a location
different from non-ATTO PMC Sierra SAS controllers.  This patch makes the
necessary adjustments in order to retrieve the SAS address on these types
of adapters.

Signed-off-by: Benjamin Rood <br...@attotech.com>
---
 drivers/scsi/pm8001/pm8001_init.c | 10 ++
 1 file changed, 10 insertions(+)

diff --git a/drivers/scsi/pm8001/pm8001_init.c 
b/drivers/scsi/pm8001/pm8001_init.c
index feaf504..fdbfab6 100644
--- a/drivers/scsi/pm8001/pm8001_init.c
+++ b/drivers/scsi/pm8001/pm8001_init.c
@@ -636,6 +636,11 @@ static void pm8001_init_sas_add(struct pm8001_hba_info 
*pm8001_ha)
payload.minor_function = 0;
payload.length = 128;
}
+   } else if ((pm8001_ha->chip_id == chip_8070 ||
+   pm8001_ha->chip_id == chip_8072) &&
+   pm8001_ha->pdev->subsystem_vendor == PCI_VENDOR_ID_ATTO) {
+   payload.minor_function = 4;
+   payload.length = 4096;
} else {
payload.minor_function = 1;
payload.length = 4096;
@@ -662,6 +667,11 @@ static void pm8001_init_sas_add(struct pm8001_hba_info 
*pm8001_ha)
else if (deviceid == 0x0042)
pm8001_ha->sas_addr[j] =
payload.func_specific[0x010 + i];
+   } else if ((pm8001_ha->chip_id == chip_8070 ||
+   pm8001_ha->chip_id == chip_8072) &&
+   pm8001_ha->pdev->subsystem_vendor == PCI_VENDOR_ID_ATTO) {
+   pm8001_ha->sas_addr[j] =
+   payload.func_specific[0x010 + i];
} else
pm8001_ha->sas_addr[j] =
payload.func_specific[0x804 + i];
-- 
2.4.3

--
To unsubscribe from this list: send the line "unsubscribe linux-scsi" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[PATCH 5/8] pm80xx: set PHY profiles for ATTO 12Gb SAS controllers

2015-10-30 Thread Benjamin Rood
PHY profiles are not saved in NVRAM on ATTO 12Gb SAS controllers.
Therefore, in order for the controller to function in a wide range of
configurations, the PHY profiles must be statically set.  This patch
provides the necessary functionality to do so.

Signed-off-by: Benjamin Rood <br...@attotech.com>
---
 drivers/scsi/pm8001/pm8001_init.c | 130 ++
 drivers/scsi/pm8001/pm8001_sas.h  |   2 +
 drivers/scsi/pm8001/pm80xx_hwi.c  |  32 ++
 3 files changed, 164 insertions(+)

diff --git a/drivers/scsi/pm8001/pm8001_init.c 
b/drivers/scsi/pm8001/pm8001_init.c
index fdbfab6..a0e55d4 100644
--- a/drivers/scsi/pm8001/pm8001_init.c
+++ b/drivers/scsi/pm8001/pm8001_init.c
@@ -732,6 +732,131 @@ static int pm8001_get_phy_settings_info(struct 
pm8001_hba_info *pm8001_ha)
return 0;
 }
 
+struct pm8001_mpi3_phy_pg_trx_config {
+   u32 LaneLosCfg;
+   u32 LanePgaCfg1;
+   u32 LanePisoCfg1;
+   u32 LanePisoCfg2;
+   u32 LanePisoCfg3;
+   u32 LanePisoCfg4;
+   u32 LanePisoCfg5;
+   u32 LanePisoCfg6;
+   u32 LaneBctCtrl;
+};
+
+/**
+ * pm8001_get_internal_phy_settings : Retrieves the internal PHY settings
+ * @pm8001_ha : our adapter
+ * @phycfg : PHY config page to populate
+ */
+static
+void pm8001_get_internal_phy_settings(struct pm8001_hba_info *pm8001_ha,
+   struct pm8001_mpi3_phy_pg_trx_config *phycfg)
+{
+   phycfg->LaneLosCfg   = 0x0132;
+   phycfg->LanePgaCfg1  = 0x00203949;
+   phycfg->LanePisoCfg1 = 0x00FF;
+   phycfg->LanePisoCfg2 = 0xFF01;
+   phycfg->LanePisoCfg3 = 0xE7011300;
+   phycfg->LanePisoCfg4 = 0x631C40C0;
+   phycfg->LanePisoCfg5 = 0xF8102036;
+   phycfg->LanePisoCfg6 = 0xF74A1000;
+   phycfg->LaneBctCtrl  = 0x00FB33F8;
+}
+
+/**
+ * pm8001_get_external_phy_settings : Retrieves the external PHY settings
+ * @pm8001_ha : our adapter
+ * @phycfg : PHY config page to populate
+ */
+static
+void pm8001_get_external_phy_settings(struct pm8001_hba_info *pm8001_ha,
+   struct pm8001_mpi3_phy_pg_trx_config *phycfg)
+{
+   phycfg->LaneLosCfg   = 0x0132;
+   phycfg->LanePgaCfg1  = 0x00203949;
+   phycfg->LanePisoCfg1 = 0x00FF;
+   phycfg->LanePisoCfg2 = 0xFF01;
+   phycfg->LanePisoCfg3 = 0xE7011300;
+   phycfg->LanePisoCfg4 = 0x63349140;
+   phycfg->LanePisoCfg5 = 0xF8102036;
+   phycfg->LanePisoCfg6 = 0xF80D9300;
+   phycfg->LaneBctCtrl  = 0x00FB33F8;
+}
+
+/**
+ * pm8001_get_phy_mask : Retrieves the mask that denotes if a PHY is int/ext
+ * @pm8001_ha : our adapter
+ * @phymask : The PHY mask
+ */
+static
+void pm8001_get_phy_mask(struct pm8001_hba_info *pm8001_ha, int *phymask)
+{
+   switch (pm8001_ha->pdev->subsystem_device) {
+   case 0x0070: /* H1280 - 8 external 0 internal */
+   case 0x0072: /* H12F0 - 16 external 0 internal */
+   *phymask = 0x;
+   break;
+
+   case 0x0071: /* H1208 - 0 external 8 internal */
+   case 0x0073: /* H120F - 0 external 16 internal */
+   *phymask = 0x;
+   break;
+
+   case 0x0080: /* H1244 - 4 external 4 internal */
+   *phymask = 0x00F0;
+   break;
+
+   case 0x0081: /* H1248 - 4 external 8 internal */
+   *phymask = 0x0FF0;
+   break;
+
+   case 0x0082: /* H1288 - 8 external 8 internal */
+   *phymask = 0xFF00;
+   break;
+
+   default:
+   PM8001_INIT_DBG(pm8001_ha,
+   pm8001_printk("Unknown subsystem device=0x%.04x",
+   pm8001_ha->pdev->subsystem_device));
+   }
+}
+
+/**
+ * pm8001_set_phy_settings_ven_117c_12Gb : Configure ATTO 12Gb PHY settings
+ * @pm8001_ha : our adapter
+ */
+static
+int pm8001_set_phy_settings_ven_117c_12G(struct pm8001_hba_info *pm8001_ha)
+{
+   struct pm8001_mpi3_phy_pg_trx_config phycfg_int;
+   struct pm8001_mpi3_phy_pg_trx_config phycfg_ext;
+   int phymask = 0;
+   int i = 0;
+
+   memset(_int, 0, sizeof(phycfg_int));
+   memset(_ext, 0, sizeof(phycfg_ext));
+
+   pm8001_get_internal_phy_settings(pm8001_ha, _int);
+   pm8001_get_external_phy_settings(pm8001_ha, _ext);
+   pm8001_get_phy_mask(pm8001_ha, );
+
+   for (i = 0; i < pm8001_ha->chip->n_phy; i++) {
+   if (phymask & (1 << i)) {/* Internal PHY */
+   pm8001_set_phy_profile_single(pm8001_ha, i,
+   sizeof(phycfg_int) / sizeof(u32),
+   (u32 *)_int);
+
+   } else { /* External PHY */
+   pm8001_set_phy_profile_single(pm8001_ha, i,
+   sizeof(phycfg_ext) / sizeof(u32),
+   (u32 *)_ext);
+   }
+   

[PATCH 6/8] pm80xx: do not examine registers for iButton feature if ATTO adapter

2015-10-30 Thread Benjamin Rood
ATTO adapters do not support this feature.  If the firmware fails to be
ready, it should not check the examined registers in order to examine
the state of the feature in order to prevent undefined behavior.

Signed-off-by: Benjamin Rood <br...@attotech.com>
---
 drivers/scsi/pm8001/pm80xx_hwi.c | 2 ++
 1 file changed, 2 insertions(+)

diff --git a/drivers/scsi/pm8001/pm80xx_hwi.c b/drivers/scsi/pm8001/pm80xx_hwi.c
index 29c548b..eb4fee6 100644
--- a/drivers/scsi/pm8001/pm80xx_hwi.c
+++ b/drivers/scsi/pm8001/pm80xx_hwi.c
@@ -1267,6 +1267,8 @@ pm80xx_chip_soft_rst(struct pm8001_hba_info *pm8001_ha)
/* check iButton feature support for motherboard controller */
if (pm8001_ha->pdev->subsystem_vendor !=
PCI_VENDOR_ID_ADAPTEC2 &&
+   pm8001_ha->pdev->subsystem_vendor !=
+   PCI_VENDOR_ID_ATTO &&
pm8001_ha->pdev->subsystem_vendor != 0) {
ibutton0 = pm8001_cr32(pm8001_ha, 0,
MSGU_HOST_SCRATCH_PAD_6);
-- 
2.4.3

--
To unsubscribe from this list: send the line "unsubscribe linux-scsi" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[PATCH 2/8] pm80xx: add support for PMC Sierra 8070 and PMC Sierra 8072 SAS controllers

2015-10-30 Thread Benjamin Rood
These SAS controllers support speeds up to 12Gb.

Signed-off-by: Benjamin Rood <br...@attotech.com>
---
 drivers/scsi/pm8001/pm8001_defs.h | 2 ++
 drivers/scsi/pm8001/pm8001_init.c | 4 +++-
 drivers/scsi/pm8001/pm8001_sas.h  | 4 +++-
 3 files changed, 8 insertions(+), 2 deletions(-)

diff --git a/drivers/scsi/pm8001/pm8001_defs.h 
b/drivers/scsi/pm8001/pm8001_defs.h
index f14ec6e..199527d 100644
--- a/drivers/scsi/pm8001/pm8001_defs.h
+++ b/drivers/scsi/pm8001/pm8001_defs.h
@@ -51,6 +51,8 @@ enum chip_flavors {
chip_8076,
chip_8077,
chip_8006,
+   chip_8070,
+   chip_8072
 };
 
 enum phy_speed {
diff --git a/drivers/scsi/pm8001/pm8001_init.c 
b/drivers/scsi/pm8001/pm8001_init.c
index 8c094fd..2106ac3 100644
--- a/drivers/scsi/pm8001/pm8001_init.c
+++ b/drivers/scsi/pm8001/pm8001_init.c
@@ -58,6 +58,8 @@ static const struct pm8001_chip_info pm8001_chips[] = {
[chip_8076] = {0,  16, _80xx_dispatch,},
[chip_8077] = {0,  16, _80xx_dispatch,},
[chip_8006] = {0,  16, _80xx_dispatch,},
+   [chip_8070] = {0,  8, _80xx_dispatch,},
+   [chip_8072] = {0,  16, _80xx_dispatch,},
 };
 static int pm8001_id;
 
@@ -1234,7 +1236,7 @@ MODULE_AUTHOR("Anand Kumar Santhanam 
<anandkumar.santha...@pmcs.com>");
 MODULE_AUTHOR("Sangeetha Gnanasekaran <sangeetha.gnanaseka...@pmcs.com>");
 MODULE_AUTHOR("Nikith Ganigarakoppal <nikith.ganigarakop...@pmcs.com>");
 MODULE_DESCRIPTION(
-   "PMC-Sierra PM8001/8006/8081/8088/8089/8074/8076/8077 "
+   "PMC-Sierra PM8001/8006/8081/8088/8089/8074/8076/8077/8070/8072 
"
"SAS/SATA controller driver");
 MODULE_VERSION(DRV_VERSION);
 MODULE_LICENSE("GPL");
diff --git a/drivers/scsi/pm8001/pm8001_sas.h b/drivers/scsi/pm8001/pm8001_sas.h
index e2e97db..9fa9705 100644
--- a/drivers/scsi/pm8001/pm8001_sas.h
+++ b/drivers/scsi/pm8001/pm8001_sas.h
@@ -106,7 +106,9 @@ do {\
 #define DEV_IS_EXPANDER(type)  ((type == SAS_EDGE_EXPANDER_DEVICE) || (type == 
SAS_FANOUT_EXPANDER_DEVICE))
 #define IS_SPCV_12G(dev)   ((dev->device == 0X8074)\
|| (dev->device == 0X8076)  \
-   || (dev->device == 0X8077))
+   || (dev->device == 0X8077)  \
+   || (dev->device == 0X8070)  \
+   || (dev->device == 0X8072))
 
 #define PM8001_NAME_LENGTH 32/* generic length of strings */
 extern struct list_head hba_list;
-- 
2.4.3

--
To unsubscribe from this list: send the line "unsubscribe linux-scsi" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html


[PATCH] pm80xx: remove the SCSI host before detaching from SAS transport

2015-10-30 Thread Benjamin Rood
Previously, when this module was unloaded via 'rmmod' with at least one
drive attached, the SCSI error handler thread would become stuck in an
infinite recovery loop and lockup the system, necessitating a reboot.

Once the SAS layer is detached, the driver will fail any subsequent
commands since the target devices are removed.  However, removing the
SCSI host generates a SYNCHRONIZE CACHE (10) command, which was failed
and left the error handler no method of recovery.

This patch simply removes the SCSI host first so that no more commands
can come down, prior to cleaning up the SAS layer.  Note that the stack
is built up with the SCSI host first, and then the SAS layer.  Perhaps
it should be reversed for symmetry, so that commands cannot be sent to
the pm80xx driver prior to attaching the SAS layer?

What was really strange about this bug was that it was introduced at
commit cff549e4860f ("[SCSI]: proper state checking and module refcount
handling in scsi_device_get").  This commit appears to tinker with how
the reference counting is performed for SCSI device objects.  My theory
is that prior to this commit, the refcount for a device object was
blindly incremented at some point during the teardown process which
coincidentially made the device stick around during the procedure, which
also coincidentially made any commands sent to the driver not fail
(since the device was technically still "there").  After this commit was
applied, my theory is the refcount for the device object is not being
incremented at a specific point anymore, which makes the device go away,
and thus made the pm80xx driver fail any subsequent commands.

You may also want to see the following for more details:

[1] http://www.spinics.net/lists/linux-scsi/msg37208.html
[2] http://marc.info/?l=linux-scsi=144416476406993=2

Signed-off-by: Benjamin Rood <br...@attotech.com>
---
 drivers/scsi/pm8001/pm8001_init.c | 2 +-
 1 file changed, 1 insertion(+), 1 deletion(-)

diff --git a/drivers/scsi/pm8001/pm8001_init.c 
b/drivers/scsi/pm8001/pm8001_init.c
index bdc624f..b1b2dd7 100644
--- a/drivers/scsi/pm8001/pm8001_init.c
+++ b/drivers/scsi/pm8001/pm8001_init.c
@@ -1096,10 +1096,10 @@ static void pm8001_pci_remove(struct pci_dev *pdev)
struct pm8001_hba_info *pm8001_ha;
int i, j;
pm8001_ha = sha->lldd_ha;
+   scsi_remove_host(pm8001_ha->shost);
sas_unregister_ha(sha);
sas_remove_host(pm8001_ha->shost);
list_del(_ha->list);
-   scsi_remove_host(pm8001_ha->shost);
PM8001_CHIP_DISP->interrupt_disable(pm8001_ha, 0xFF);
PM8001_CHIP_DISP->chip_soft_rst(pm8001_ha);
 
-- 
2.4.3

--
To unsubscribe from this list: send the line "unsubscribe linux-scsi" in
the body of a message to majord...@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html