aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--drivers/firewire/ohci.c74
1 files changed, 23 insertions, 51 deletions
diff --git a/drivers/firewire/ohci.c b/drivers/firewire/ohci.c
index b983581cfe35..862fdf3400cf 100644
--- a/drivers/firewire/ohci.c
+++ b/drivers/firewire/ohci.c
@@ -325,7 +325,7 @@ MODULE_PARM_DESC(quirks, "Chip quirks (default = 0"
325 ", AR/selfID endianess = " __stringify(QUIRK_BE_HEADERS) 325 ", AR/selfID endianess = " __stringify(QUIRK_BE_HEADERS)
326 ", no 1394a enhancements = " __stringify(QUIRK_NO_1394A) 326 ", no 1394a enhancements = " __stringify(QUIRK_NO_1394A)
327 ", disable MSI = " __stringify(QUIRK_NO_MSI) 327 ", disable MSI = " __stringify(QUIRK_NO_MSI)
328 ", workaround for TI SLLZ059 errata = " __stringify(QUIRK_TI_SLLZ059) 328 ", TI SLLZ059 erratum = " __stringify(QUIRK_TI_SLLZ059)
329 ")"); 329 ")");
330 330
331#define OHCI_PARAM_DEBUG_AT_AR 1 331#define OHCI_PARAM_DEBUG_AT_AR 1
@@ -1730,12 +1730,8 @@ static int get_status_for_port(struct fw_ohci *ohci, int port_index)
1730 1730
1731 mutex_lock(&ohci->phy_reg_mutex); 1731 mutex_lock(&ohci->phy_reg_mutex);
1732 reg = write_phy_reg(ohci, 7, port_index); 1732 reg = write_phy_reg(ohci, 7, port_index);
1733 mutex_unlock(&ohci->phy_reg_mutex); 1733 if (reg >= 0)
1734 if (reg < 0) 1734 reg = read_phy_reg(ohci, 8);
1735 return reg;
1736
1737 mutex_lock(&ohci->phy_reg_mutex);
1738 reg = read_phy_reg(ohci, 8);
1739 mutex_unlock(&ohci->phy_reg_mutex); 1735 mutex_unlock(&ohci->phy_reg_mutex);
1740 if (reg < 0) 1736 if (reg < 0)
1741 return reg; 1737 return reg;
@@ -1754,6 +1750,7 @@ static int get_self_id_pos(struct fw_ohci *ohci, u32 self_id,
1754{ 1750{
1755 int i; 1751 int i;
1756 u32 entry; 1752 u32 entry;
1753
1757 for (i = 0; i < self_id_count; i++) { 1754 for (i = 0; i < self_id_count; i++) {
1758 entry = ohci->self_id_buffer[i]; 1755 entry = ohci->self_id_buffer[i];
1759 if ((self_id & 0xff000000) == (entry & 0xff000000)) 1756 if ((self_id & 0xff000000) == (entry & 0xff000000))
@@ -1765,33 +1762,16 @@ static int get_self_id_pos(struct fw_ohci *ohci, u32 self_id,
1765} 1762}
1766 1763
1767/* 1764/*
1768 * This function implements a work around for the Texas Instruments PHY 1765 * TI TSB82AA2B and TSB12LV26 do not receive the selfID of a locally
1769 * TSB41BA3D. This phy has a bug at least in combination with the TI 1766 * attached TSB41BA3D phy; see http://www.ti.com/litv/pdf/sllz059.
1770 * LLCs TSB82AA2B and TSB12LV26. The selfid coming from the locally 1767 * Construct the selfID from phy register contents.
1771 * connected phy is not propagated into the selfid buffer of the OHCI 1768 * FIXME: How to determine the selfID.i flag?
1772 * (see http://www.ti.com/litv/pdf/sllz059 for details).
1773 * The main idea is to construct the selfid ourselves.
1774 */ 1769 */
1775
1776static int find_and_insert_self_id(struct fw_ohci *ohci, int self_id_count) 1770static int find_and_insert_self_id(struct fw_ohci *ohci, int self_id_count)
1777{ 1771{
1778 int reg; 1772 int reg, i, pos, status;
1779 int i; 1773 /* link active 1, speed 3, bridge 0, contender 1, more packets 0 */
1780 int pos; 1774 u32 self_id = 0x8040c800;
1781 int status;
1782 u32 self_id;
1783
1784/*
1785 * preset bits in self_id
1786 *
1787 * link active: 0b1
1788 * speed: 0b11
1789 * bridge: 0b00
1790 * contender: 0b1
1791 * initiated reset: 0b0
1792 * more packets: 0b0
1793 */
1794 self_id = 0x8040C800;
1795 1775
1796 reg = reg_read(ohci, OHCI1394_NodeID); 1776 reg = reg_read(ohci, OHCI1394_NodeID);
1797 if (!(reg & OHCI1394_NodeID_idValid)) { 1777 if (!(reg & OHCI1394_NodeID_idValid)) {
@@ -1800,16 +1780,12 @@ static int find_and_insert_self_id(struct fw_ohci *ohci, int self_id_count)
1800 } 1780 }
1801 self_id |= ((reg & 0x3f) << 24); /* phy ID */ 1781 self_id |= ((reg & 0x3f) << 24); /* phy ID */
1802 1782
1803 mutex_lock(&ohci->phy_reg_mutex); 1783 reg = ohci_read_phy_reg(&ohci->card, 4);
1804 reg = read_phy_reg(ohci, 4);
1805 mutex_unlock(&ohci->phy_reg_mutex);
1806 if (reg < 0) 1784 if (reg < 0)
1807 return reg; 1785 return reg;
1808 self_id |= ((reg & 0x07) << 8); /* power class */ 1786 self_id |= ((reg & 0x07) << 8); /* power class */
1809 1787
1810 mutex_lock(&ohci->phy_reg_mutex); 1788 reg = ohci_read_phy_reg(&ohci->card, 1);
1811 reg = read_phy_reg(ohci, 1);
1812 mutex_unlock(&ohci->phy_reg_mutex);
1813 if (reg < 0) 1789 if (reg < 0)
1814 return reg; 1790 return reg;
1815 self_id |= ((reg & 0x3f) << 16); /* gap count */ 1791 self_id |= ((reg & 0x3f) << 16); /* gap count */
@@ -1894,7 +1870,7 @@ static void bus_reset_work(struct work_struct *work)
1894 if (ohci->quirks & QUIRK_TI_SLLZ059) { 1870 if (ohci->quirks & QUIRK_TI_SLLZ059) {
1895 self_id_count = find_and_insert_self_id(ohci, self_id_count); 1871 self_id_count = find_and_insert_self_id(ohci, self_id_count);
1896 if (self_id_count < 0) { 1872 if (self_id_count < 0) {
1897 fw_notify("could not construct local self IDs\n"); 1873 fw_notify("could not construct local self ID\n");
1898 return; 1874 return;
1899 } 1875 }
1900 } 1876 }
@@ -2188,10 +2164,7 @@ static int configure_1394a_enhancements(struct fw_ohci *ohci)
2188 2164
2189static int probe_tsb41ba3d(struct fw_ohci *ohci) 2165static int probe_tsb41ba3d(struct fw_ohci *ohci)
2190{ 2166{
2191 int reg; 2167 int reg, i, vendor_id, product_id;
2192 int i;
2193 int vendor_id;
2194 int product_id;
2195 2168
2196 reg = read_phy_reg(ohci, 2); 2169 reg = read_phy_reg(ohci, 2);
2197 if (reg < 0) 2170 if (reg < 0)
@@ -2214,7 +2187,7 @@ static int probe_tsb41ba3d(struct fw_ohci *ohci)
2214 } 2187 }
2215 2188
2216 if ((vendor_id == TSB41BA3D_VID) && 2189 if ((vendor_id == TSB41BA3D_VID) &&
2217 (product_id == TSB41BA3D_PID)) 2190 (product_id == TSB41BA3D_PID))
2218 return 1; 2191 return 1;
2219 } 2192 }
2220 return 0; 2193 return 0;
@@ -2226,7 +2199,7 @@ static int ohci_enable(struct fw_card *card,
2226 struct fw_ohci *ohci = fw_ohci(card); 2199 struct fw_ohci *ohci = fw_ohci(card);
2227 struct pci_dev *dev = to_pci_dev(card->device); 2200 struct pci_dev *dev = to_pci_dev(card->device);
2228 u32 lps, seconds, version, irqs; 2201 u32 lps, seconds, version, irqs;
2229 int i, ret, tsb41ba3d_found; 2202 int i, ret;
2230 2203
2231 if (software_reset(ohci)) { 2204 if (software_reset(ohci)) {
2232 fw_error("Failed to reset ohci card.\n"); 2205 fw_error("Failed to reset ohci card.\n");
@@ -2258,14 +2231,13 @@ static int ohci_enable(struct fw_card *card,
2258 } 2231 }
2259 2232
2260 if (ohci->quirks & QUIRK_TI_SLLZ059) { 2233 if (ohci->quirks & QUIRK_TI_SLLZ059) {
2261 tsb41ba3d_found = probe_tsb41ba3d(ohci); 2234 ret = probe_tsb41ba3d(ohci);
2262 if (tsb41ba3d_found < 0) 2235 if (ret < 0)
2263 return tsb41ba3d_found; 2236 return ret;
2264 if (!tsb41ba3d_found) { 2237 if (ret)
2265 fw_notify("No TSB41BA3D found, " 2238 fw_notify("local TSB41BA3D phy\n");
2266 "resetting QUIRK_TI_SLLZ059\n"); 2239 else
2267 ohci->quirks &= ~QUIRK_TI_SLLZ059; 2240 ohci->quirks &= ~QUIRK_TI_SLLZ059;
2268 }
2269 } 2241 }
2270 2242
2271 reg_write(ohci, OHCI1394_HCControlClear, 2243 reg_write(ohci, OHCI1394_HCControlClear,