diff options
Diffstat (limited to 'drivers/scsi/lpfc/lpfc_sli.c')
-rw-r--r-- | drivers/scsi/lpfc/lpfc_sli.c | 99 |
1 files changed, 93 insertions, 6 deletions
diff --git a/drivers/scsi/lpfc/lpfc_sli.c b/drivers/scsi/lpfc/lpfc_sli.c index 764aadbec71b..bb69a7a1ec59 100644 --- a/drivers/scsi/lpfc/lpfc_sli.c +++ b/drivers/scsi/lpfc/lpfc_sli.c | |||
@@ -1566,6 +1566,79 @@ lpfc_sli_brdready(struct lpfc_hba * phba, uint32_t mask) | |||
1566 | return retval; | 1566 | return retval; |
1567 | } | 1567 | } |
1568 | 1568 | ||
1569 | #define BARRIER_TEST_PATTERN (0xdeadbeef) | ||
1570 | |||
1571 | void lpfc_reset_barrier(struct lpfc_hba * phba) | ||
1572 | { | ||
1573 | uint32_t * resp_buf; | ||
1574 | uint32_t * mbox_buf; | ||
1575 | volatile uint32_t mbox; | ||
1576 | uint32_t hc_copy; | ||
1577 | int i; | ||
1578 | uint8_t hdrtype; | ||
1579 | |||
1580 | pci_read_config_byte(phba->pcidev, PCI_HEADER_TYPE, &hdrtype); | ||
1581 | if (hdrtype != 0x80 || | ||
1582 | (FC_JEDEC_ID(phba->vpd.rev.biuRev) != HELIOS_JEDEC_ID && | ||
1583 | FC_JEDEC_ID(phba->vpd.rev.biuRev) != THOR_JEDEC_ID)) | ||
1584 | return; | ||
1585 | |||
1586 | /* | ||
1587 | * Tell the other part of the chip to suspend temporarily all | ||
1588 | * its DMA activity. | ||
1589 | */ | ||
1590 | resp_buf = (uint32_t *)phba->MBslimaddr; | ||
1591 | |||
1592 | /* Disable the error attention */ | ||
1593 | hc_copy = readl(phba->HCregaddr); | ||
1594 | writel((hc_copy & ~HC_ERINT_ENA), phba->HCregaddr); | ||
1595 | readl(phba->HCregaddr); /* flush */ | ||
1596 | |||
1597 | if (readl(phba->HAregaddr) & HA_ERATT) { | ||
1598 | /* Clear Chip error bit */ | ||
1599 | writel(HA_ERATT, phba->HAregaddr); | ||
1600 | phba->stopped = 1; | ||
1601 | } | ||
1602 | |||
1603 | mbox = 0; | ||
1604 | ((MAILBOX_t *)&mbox)->mbxCommand = MBX_KILL_BOARD; | ||
1605 | ((MAILBOX_t *)&mbox)->mbxOwner = OWN_CHIP; | ||
1606 | |||
1607 | writel(BARRIER_TEST_PATTERN, (resp_buf + 1)); | ||
1608 | mbox_buf = (uint32_t *)phba->MBslimaddr; | ||
1609 | writel(mbox, mbox_buf); | ||
1610 | |||
1611 | for (i = 0; | ||
1612 | readl(resp_buf + 1) != ~(BARRIER_TEST_PATTERN) && i < 50; i++) | ||
1613 | mdelay(1); | ||
1614 | |||
1615 | if (readl(resp_buf + 1) != ~(BARRIER_TEST_PATTERN)) { | ||
1616 | if (phba->sli.sli_flag & LPFC_SLI2_ACTIVE || | ||
1617 | phba->stopped) | ||
1618 | goto restore_hc; | ||
1619 | else | ||
1620 | goto clear_errat; | ||
1621 | } | ||
1622 | |||
1623 | ((MAILBOX_t *)&mbox)->mbxOwner = OWN_HOST; | ||
1624 | for (i = 0; readl(resp_buf) != mbox && i < 500; i++) | ||
1625 | mdelay(1); | ||
1626 | |||
1627 | clear_errat: | ||
1628 | |||
1629 | while (!(readl(phba->HAregaddr) & HA_ERATT) && ++i < 500) | ||
1630 | mdelay(1); | ||
1631 | |||
1632 | if (readl(phba->HAregaddr) & HA_ERATT) { | ||
1633 | writel(HA_ERATT, phba->HAregaddr); | ||
1634 | phba->stopped = 1; | ||
1635 | } | ||
1636 | |||
1637 | restore_hc: | ||
1638 | writel(hc_copy, phba->HCregaddr); | ||
1639 | readl(phba->HCregaddr); /* flush */ | ||
1640 | } | ||
1641 | |||
1569 | int | 1642 | int |
1570 | lpfc_sli_brdkill(struct lpfc_hba * phba) | 1643 | lpfc_sli_brdkill(struct lpfc_hba * phba) |
1571 | { | 1644 | { |
@@ -1588,9 +1661,8 @@ lpfc_sli_brdkill(struct lpfc_hba * phba) | |||
1588 | psli->sli_flag); | 1661 | psli->sli_flag); |
1589 | 1662 | ||
1590 | if ((pmb = (LPFC_MBOXQ_t *) mempool_alloc(phba->mbox_mem_pool, | 1663 | if ((pmb = (LPFC_MBOXQ_t *) mempool_alloc(phba->mbox_mem_pool, |
1591 | GFP_ATOMIC)) == 0) { | 1664 | GFP_KERNEL)) == 0) |
1592 | return 1; | 1665 | return 1; |
1593 | } | ||
1594 | 1666 | ||
1595 | /* Disable the error attention */ | 1667 | /* Disable the error attention */ |
1596 | spin_lock_irq(phba->host->host_lock); | 1668 | spin_lock_irq(phba->host->host_lock); |
@@ -1610,6 +1682,8 @@ lpfc_sli_brdkill(struct lpfc_hba * phba) | |||
1610 | return 1; | 1682 | return 1; |
1611 | } | 1683 | } |
1612 | 1684 | ||
1685 | psli->sli_flag &= ~LPFC_SLI2_ACTIVE; | ||
1686 | |||
1613 | mempool_free(pmb, phba->mbox_mem_pool); | 1687 | mempool_free(pmb, phba->mbox_mem_pool); |
1614 | 1688 | ||
1615 | /* There is no completion for a KILL_BOARD mbox cmd. Check for an error | 1689 | /* There is no completion for a KILL_BOARD mbox cmd. Check for an error |
@@ -1625,7 +1699,10 @@ lpfc_sli_brdkill(struct lpfc_hba * phba) | |||
1625 | } | 1699 | } |
1626 | 1700 | ||
1627 | del_timer_sync(&psli->mbox_tmo); | 1701 | del_timer_sync(&psli->mbox_tmo); |
1628 | 1702 | if (ha_copy & HA_ERATT) { | |
1703 | writel(HA_ERATT, phba->HAregaddr); | ||
1704 | phba->stopped = 1; | ||
1705 | } | ||
1629 | spin_lock_irq(phba->host->host_lock); | 1706 | spin_lock_irq(phba->host->host_lock); |
1630 | psli->sli_flag &= ~LPFC_SLI_MBOX_ACTIVE; | 1707 | psli->sli_flag &= ~LPFC_SLI_MBOX_ACTIVE; |
1631 | spin_unlock_irq(phba->host->host_lock); | 1708 | spin_unlock_irq(phba->host->host_lock); |
@@ -1665,6 +1742,7 @@ lpfc_sli_brdreset(struct lpfc_hba * phba) | |||
1665 | (cfg_value & | 1742 | (cfg_value & |
1666 | ~(PCI_COMMAND_PARITY | PCI_COMMAND_SERR))); | 1743 | ~(PCI_COMMAND_PARITY | PCI_COMMAND_SERR))); |
1667 | 1744 | ||
1745 | psli->sli_flag &= ~LPFC_SLI2_ACTIVE; | ||
1668 | /* Now toggle INITFF bit in the Host Control Register */ | 1746 | /* Now toggle INITFF bit in the Host Control Register */ |
1669 | writel(HC_INITFF, phba->HCregaddr); | 1747 | writel(HC_INITFF, phba->HCregaddr); |
1670 | mdelay(1); | 1748 | mdelay(1); |
@@ -1713,6 +1791,8 @@ lpfc_sli_brdrestart(struct lpfc_hba * phba) | |||
1713 | mb->mbxCommand = MBX_RESTART; | 1791 | mb->mbxCommand = MBX_RESTART; |
1714 | mb->mbxHc = 1; | 1792 | mb->mbxHc = 1; |
1715 | 1793 | ||
1794 | lpfc_reset_barrier(phba); | ||
1795 | |||
1716 | to_slim = phba->MBslimaddr; | 1796 | to_slim = phba->MBslimaddr; |
1717 | writel(*(uint32_t *) mb, to_slim); | 1797 | writel(*(uint32_t *) mb, to_slim); |
1718 | readl(to_slim); /* flush */ | 1798 | readl(to_slim); /* flush */ |
@@ -1730,7 +1810,7 @@ lpfc_sli_brdrestart(struct lpfc_hba * phba) | |||
1730 | readl(to_slim); /* flush */ | 1810 | readl(to_slim); /* flush */ |
1731 | 1811 | ||
1732 | lpfc_sli_brdreset(phba); | 1812 | lpfc_sli_brdreset(phba); |
1733 | 1813 | phba->stopped = 0; | |
1734 | phba->hba_state = LPFC_INIT_START; | 1814 | phba->hba_state = LPFC_INIT_START; |
1735 | 1815 | ||
1736 | spin_unlock_irq(phba->host->host_lock); | 1816 | spin_unlock_irq(phba->host->host_lock); |
@@ -2038,6 +2118,13 @@ lpfc_sli_issue_mbox(struct lpfc_hba * phba, LPFC_MBOXQ_t * pmbox, uint32_t flag) | |||
2038 | return (MBX_NOT_FINISHED); | 2118 | return (MBX_NOT_FINISHED); |
2039 | } | 2119 | } |
2040 | 2120 | ||
2121 | if (mb->mbxCommand != MBX_KILL_BOARD && flag & MBX_NOWAIT && | ||
2122 | !(readl(phba->HCregaddr) & HC_MBINT_ENA)) { | ||
2123 | spin_unlock_irqrestore(phba->host->host_lock, drvr_flag); | ||
2124 | LOG_MBOX_CANNOT_ISSUE_DATA( phba, mb, psli, flag) | ||
2125 | return (MBX_NOT_FINISHED); | ||
2126 | } | ||
2127 | |||
2041 | if (psli->sli_flag & LPFC_SLI_MBOX_ACTIVE) { | 2128 | if (psli->sli_flag & LPFC_SLI_MBOX_ACTIVE) { |
2042 | /* Polling for a mbox command when another one is already active | 2129 | /* Polling for a mbox command when another one is already active |
2043 | * is not allowed in SLI. Also, the driver must have established | 2130 | * is not allowed in SLI. Also, the driver must have established |
@@ -2154,8 +2241,7 @@ lpfc_sli_issue_mbox(struct lpfc_hba * phba, LPFC_MBOXQ_t * pmbox, uint32_t flag) | |||
2154 | /* First copy command data to host SLIM area */ | 2241 | /* First copy command data to host SLIM area */ |
2155 | lpfc_sli_pcimem_bcopy(mb, &phba->slim2p->mbx, MAILBOX_CMD_SIZE); | 2242 | lpfc_sli_pcimem_bcopy(mb, &phba->slim2p->mbx, MAILBOX_CMD_SIZE); |
2156 | } else { | 2243 | } else { |
2157 | if (mb->mbxCommand == MBX_CONFIG_PORT || | 2244 | if (mb->mbxCommand == MBX_CONFIG_PORT) { |
2158 | mb->mbxCommand == MBX_KILL_BOARD) { | ||
2159 | /* copy command data into host mbox for cmpl */ | 2245 | /* copy command data into host mbox for cmpl */ |
2160 | lpfc_sli_pcimem_bcopy(mb, &phba->slim2p->mbx, | 2246 | lpfc_sli_pcimem_bcopy(mb, &phba->slim2p->mbx, |
2161 | MAILBOX_CMD_SIZE); | 2247 | MAILBOX_CMD_SIZE); |
@@ -3121,6 +3207,7 @@ lpfc_intr_handler(int irq, void *dev_id, struct pt_regs * regs) | |||
3121 | /* Clear Chip error bit */ | 3207 | /* Clear Chip error bit */ |
3122 | writel(HA_ERATT, phba->HAregaddr); | 3208 | writel(HA_ERATT, phba->HAregaddr); |
3123 | readl(phba->HAregaddr); /* flush */ | 3209 | readl(phba->HAregaddr); /* flush */ |
3210 | phba->stopped = 1; | ||
3124 | } | 3211 | } |
3125 | 3212 | ||
3126 | spin_lock(phba->host->host_lock); | 3213 | spin_lock(phba->host->host_lock); |