diff options
author | Jeff Skirvin <jeffrey.d.skirvin@intel.com> | 2012-01-04 04:32:39 -0500 |
---|---|---|
committer | James Bottomley <JBottomley@Parallels.com> | 2012-01-16 02:40:29 -0500 |
commit | afd13a1f2b05157c7621d87dfe89ea6ea9061bd8 (patch) | |
tree | 1dc7a3c2816de4598ada6871f1381336bf1b4192 | |
parent | 2e5da889d44a3a9629f895de3488306e7f5ddf16 (diff) |
[SCSI] isci: update afe (analog-front-end) recipe for C1
C1 silicon requires updates to the phy tuning recipe and also support
for user provided cable selects (per-phy) for short, medium, and long
cables. Default to 'short' awaiting support for selecting the cable via
oem parameters.
Reviewed-by: Jiangbi Liu <jiangbi.liu@intel.com>
Signed-off-by: Marcin Tomczak <marcin.tomczak@intel.com>
Signed-off-by: Jeff Skirvin <jeffrey.d.skirvin@intel.com>
Signed-off-by: Dan Williams <dan.j.williams@intel.com>
Signed-off-by: James Bottomley <JBottomley@Parallels.com>
-rw-r--r-- | drivers/scsi/isci/host.c | 111 | ||||
-rw-r--r-- | drivers/scsi/isci/host.h | 9 | ||||
-rw-r--r-- | drivers/scsi/isci/phy.c | 7 | ||||
-rw-r--r-- | drivers/scsi/isci/probe_roms.c | 2 |
4 files changed, 103 insertions, 26 deletions
diff --git a/drivers/scsi/isci/host.c b/drivers/scsi/isci/host.c index 8e7de192cf6d..383bb6913087 100644 --- a/drivers/scsi/isci/host.c +++ b/drivers/scsi/isci/host.c | |||
@@ -1908,12 +1908,23 @@ void sci_controller_power_control_queue_remove(struct isci_host *ihost, | |||
1908 | ihost->power_control.requesters[iphy->phy_index] = NULL; | 1908 | ihost->power_control.requesters[iphy->phy_index] = NULL; |
1909 | } | 1909 | } |
1910 | 1910 | ||
1911 | static int is_long_cable(int phy, unsigned char selection_byte) | ||
1912 | { | ||
1913 | return 0; | ||
1914 | } | ||
1915 | |||
1916 | static int is_medium_cable(int phy, unsigned char selection_byte) | ||
1917 | { | ||
1918 | return 0; | ||
1919 | } | ||
1920 | |||
1911 | #define AFE_REGISTER_WRITE_DELAY 10 | 1921 | #define AFE_REGISTER_WRITE_DELAY 10 |
1912 | 1922 | ||
1913 | static void sci_controller_afe_initialization(struct isci_host *ihost) | 1923 | static void sci_controller_afe_initialization(struct isci_host *ihost) |
1914 | { | 1924 | { |
1915 | struct scu_afe_registers __iomem *afe = &ihost->scu_registers->afe; | 1925 | struct scu_afe_registers __iomem *afe = &ihost->scu_registers->afe; |
1916 | const struct sci_oem_params *oem = &ihost->oem_parameters; | 1926 | const struct sci_oem_params *oem = &ihost->oem_parameters; |
1927 | unsigned char cable_selection_mask = 0; | ||
1917 | struct pci_dev *pdev = ihost->pdev; | 1928 | struct pci_dev *pdev = ihost->pdev; |
1918 | u32 afe_status; | 1929 | u32 afe_status; |
1919 | u32 phy_id; | 1930 | u32 phy_id; |
@@ -1922,11 +1933,11 @@ static void sci_controller_afe_initialization(struct isci_host *ihost) | |||
1922 | writel(0x0081000f, &afe->afe_dfx_master_control0); | 1933 | writel(0x0081000f, &afe->afe_dfx_master_control0); |
1923 | udelay(AFE_REGISTER_WRITE_DELAY); | 1934 | udelay(AFE_REGISTER_WRITE_DELAY); |
1924 | 1935 | ||
1925 | if (is_b0(pdev)) { | 1936 | if (is_b0(pdev) || is_c0(pdev) || is_c1(pdev)) { |
1926 | /* PM Rx Equalization Save, PM SPhy Rx Acknowledgement | 1937 | /* PM Rx Equalization Save, PM SPhy Rx Acknowledgement |
1927 | * Timer, PM Stagger Timer | 1938 | * Timer, PM Stagger Timer |
1928 | */ | 1939 | */ |
1929 | writel(0x0007BFFF, &afe->afe_pmsn_master_control2); | 1940 | writel(0x0007FFFF, &afe->afe_pmsn_master_control2); |
1930 | udelay(AFE_REGISTER_WRITE_DELAY); | 1941 | udelay(AFE_REGISTER_WRITE_DELAY); |
1931 | } | 1942 | } |
1932 | 1943 | ||
@@ -1935,14 +1946,23 @@ static void sci_controller_afe_initialization(struct isci_host *ihost) | |||
1935 | writel(0x00005A00, &afe->afe_bias_control); | 1946 | writel(0x00005A00, &afe->afe_bias_control); |
1936 | else if (is_b0(pdev) || is_c0(pdev)) | 1947 | else if (is_b0(pdev) || is_c0(pdev)) |
1937 | writel(0x00005F00, &afe->afe_bias_control); | 1948 | writel(0x00005F00, &afe->afe_bias_control); |
1949 | else if (is_c1(pdev)) | ||
1950 | writel(0x00005500, &afe->afe_bias_control); | ||
1938 | 1951 | ||
1939 | udelay(AFE_REGISTER_WRITE_DELAY); | 1952 | udelay(AFE_REGISTER_WRITE_DELAY); |
1940 | 1953 | ||
1941 | /* Enable PLL */ | 1954 | /* Enable PLL */ |
1942 | if (is_b0(pdev) || is_c0(pdev)) | 1955 | if (is_a2(pdev)) |
1943 | writel(0x80040A08, &afe->afe_pll_control0); | ||
1944 | else | ||
1945 | writel(0x80040908, &afe->afe_pll_control0); | 1956 | writel(0x80040908, &afe->afe_pll_control0); |
1957 | else if (is_b0(pdev) || is_c0(pdev)) | ||
1958 | writel(0x80040A08, &afe->afe_pll_control0); | ||
1959 | else if (is_c1(pdev)) { | ||
1960 | writel(0x80000B08, &afe->afe_pll_control0); | ||
1961 | udelay(AFE_REGISTER_WRITE_DELAY); | ||
1962 | writel(0x00000B08, &afe->afe_pll_control0); | ||
1963 | udelay(AFE_REGISTER_WRITE_DELAY); | ||
1964 | writel(0x80000B08, &afe->afe_pll_control0); | ||
1965 | } | ||
1946 | 1966 | ||
1947 | udelay(AFE_REGISTER_WRITE_DELAY); | 1967 | udelay(AFE_REGISTER_WRITE_DELAY); |
1948 | 1968 | ||
@@ -1963,46 +1983,68 @@ static void sci_controller_afe_initialization(struct isci_host *ihost) | |||
1963 | for (phy_id = 0; phy_id < SCI_MAX_PHYS; phy_id++) { | 1983 | for (phy_id = 0; phy_id < SCI_MAX_PHYS; phy_id++) { |
1964 | struct scu_afe_transceiver *xcvr = &afe->scu_afe_xcvr[phy_id]; | 1984 | struct scu_afe_transceiver *xcvr = &afe->scu_afe_xcvr[phy_id]; |
1965 | const struct sci_phy_oem_params *oem_phy = &oem->phys[phy_id]; | 1985 | const struct sci_phy_oem_params *oem_phy = &oem->phys[phy_id]; |
1986 | int cable_length_long = | ||
1987 | is_long_cable(phy_id, cable_selection_mask); | ||
1988 | int cable_length_medium = | ||
1989 | is_medium_cable(phy_id, cable_selection_mask); | ||
1990 | |||
1991 | if (is_a2(pdev)) { | ||
1992 | /* All defaults, except the Receive Word | ||
1993 | * Alignament/Comma Detect Enable....(0xe800) | ||
1994 | */ | ||
1995 | writel(0x00004512, &xcvr->afe_xcvr_control0); | ||
1996 | udelay(AFE_REGISTER_WRITE_DELAY); | ||
1966 | 1997 | ||
1967 | if (is_b0(pdev)) { | 1998 | writel(0x0050100F, &xcvr->afe_xcvr_control1); |
1968 | /* Configure transmitter SSC parameters */ | 1999 | udelay(AFE_REGISTER_WRITE_DELAY); |
2000 | } else if (is_b0(pdev)) { | ||
2001 | /* Configure transmitter SSC parameters */ | ||
1969 | writel(0x00030000, &xcvr->afe_tx_ssc_control); | 2002 | writel(0x00030000, &xcvr->afe_tx_ssc_control); |
1970 | udelay(AFE_REGISTER_WRITE_DELAY); | 2003 | udelay(AFE_REGISTER_WRITE_DELAY); |
1971 | } else if (is_c0(pdev)) { | 2004 | } else if (is_c0(pdev)) { |
1972 | /* Configure transmitter SSC parameters */ | 2005 | /* Configure transmitter SSC parameters */ |
1973 | writel(0x0003000, &xcvr->afe_tx_ssc_control); | 2006 | writel(0x00010202, &xcvr->afe_tx_ssc_control); |
1974 | udelay(AFE_REGISTER_WRITE_DELAY); | 2007 | udelay(AFE_REGISTER_WRITE_DELAY); |
1975 | 2008 | ||
1976 | /* All defaults, except the Receive Word | 2009 | /* All defaults, except the Receive Word |
1977 | * Alignament/Comma Detect Enable....(0xe800) | 2010 | * Alignament/Comma Detect Enable....(0xe800) |
1978 | */ | 2011 | */ |
1979 | writel(0x00004500, &xcvr->afe_xcvr_control0); | 2012 | writel(0x00014500, &xcvr->afe_xcvr_control0); |
1980 | udelay(AFE_REGISTER_WRITE_DELAY); | 2013 | udelay(AFE_REGISTER_WRITE_DELAY); |
1981 | } else { | 2014 | } else if (is_c1(pdev)) { |
2015 | /* Configure transmitter SSC parameters */ | ||
2016 | writel(0x00010202, &xcvr->afe_tx_ssc_control); | ||
2017 | udelay(AFE_REGISTER_WRITE_DELAY); | ||
2018 | |||
1982 | /* All defaults, except the Receive Word | 2019 | /* All defaults, except the Receive Word |
1983 | * Alignament/Comma Detect Enable....(0xe800) | 2020 | * Alignament/Comma Detect Enable....(0xe800) |
1984 | */ | 2021 | */ |
1985 | writel(0x00004512, &xcvr->afe_xcvr_control0); | 2022 | writel(0x0001C500, &xcvr->afe_xcvr_control0); |
1986 | udelay(AFE_REGISTER_WRITE_DELAY); | ||
1987 | |||
1988 | writel(0x0050100F, &xcvr->afe_xcvr_control1); | ||
1989 | udelay(AFE_REGISTER_WRITE_DELAY); | 2023 | udelay(AFE_REGISTER_WRITE_DELAY); |
1990 | } | 2024 | } |
1991 | 2025 | ||
1992 | /* Power up TX and RX out from power down (PWRDNTX and PWRDNRX) | 2026 | /* Power up TX and RX out from power down (PWRDNTX and |
1993 | * & increase TX int & ext bias 20%....(0xe85c) | 2027 | * PWRDNRX) & increase TX int & ext bias 20%....(0xe85c) |
1994 | */ | 2028 | */ |
1995 | if (is_a2(pdev)) | 2029 | if (is_a2(pdev)) |
1996 | writel(0x000003F0, &xcvr->afe_channel_control); | 2030 | writel(0x000003F0, &xcvr->afe_channel_control); |
1997 | else if (is_b0(pdev)) { | 2031 | else if (is_b0(pdev)) { |
1998 | /* Power down TX and RX (PWRDNTX and PWRDNRX) */ | ||
1999 | writel(0x000003D7, &xcvr->afe_channel_control); | 2032 | writel(0x000003D7, &xcvr->afe_channel_control); |
2000 | udelay(AFE_REGISTER_WRITE_DELAY); | 2033 | udelay(AFE_REGISTER_WRITE_DELAY); |
2034 | |||
2001 | writel(0x000003D4, &xcvr->afe_channel_control); | 2035 | writel(0x000003D4, &xcvr->afe_channel_control); |
2002 | } else { | 2036 | } else if (is_c0(pdev)) { |
2003 | writel(0x000001E7, &xcvr->afe_channel_control); | 2037 | writel(0x000001E7, &xcvr->afe_channel_control); |
2004 | udelay(AFE_REGISTER_WRITE_DELAY); | 2038 | udelay(AFE_REGISTER_WRITE_DELAY); |
2039 | |||
2005 | writel(0x000001E4, &xcvr->afe_channel_control); | 2040 | writel(0x000001E4, &xcvr->afe_channel_control); |
2041 | } else if (is_c1(pdev)) { | ||
2042 | writel(cable_length_long ? 0x000002F7 : 0x000001F7, | ||
2043 | &xcvr->afe_channel_control); | ||
2044 | udelay(AFE_REGISTER_WRITE_DELAY); | ||
2045 | |||
2046 | writel(cable_length_long ? 0x000002F4 : 0x000001F4, | ||
2047 | &xcvr->afe_channel_control); | ||
2006 | } | 2048 | } |
2007 | udelay(AFE_REGISTER_WRITE_DELAY); | 2049 | udelay(AFE_REGISTER_WRITE_DELAY); |
2008 | 2050 | ||
@@ -2012,7 +2054,16 @@ static void sci_controller_afe_initialization(struct isci_host *ihost) | |||
2012 | udelay(AFE_REGISTER_WRITE_DELAY); | 2054 | udelay(AFE_REGISTER_WRITE_DELAY); |
2013 | } | 2055 | } |
2014 | 2056 | ||
2015 | writel(0x00004100, &xcvr->afe_xcvr_control0); | 2057 | if (is_a2(pdev) || is_b0(pdev)) |
2058 | /* RDPI=0x0(RX Power On), RXOOBDETPDNC=0x0, | ||
2059 | * TPD=0x0(TX Power On), RDD=0x0(RX Detect | ||
2060 | * Enabled) ....(0xe800) | ||
2061 | */ | ||
2062 | writel(0x00004100, &xcvr->afe_xcvr_control0); | ||
2063 | else if (is_c0(pdev)) | ||
2064 | writel(0x00014100, &xcvr->afe_xcvr_control0); | ||
2065 | else if (is_c1(pdev)) | ||
2066 | writel(0x0001C100, &xcvr->afe_xcvr_control0); | ||
2016 | udelay(AFE_REGISTER_WRITE_DELAY); | 2067 | udelay(AFE_REGISTER_WRITE_DELAY); |
2017 | 2068 | ||
2018 | /* Leave DFE/FFE on */ | 2069 | /* Leave DFE/FFE on */ |
@@ -2023,8 +2074,8 @@ static void sci_controller_afe_initialization(struct isci_host *ihost) | |||
2023 | udelay(AFE_REGISTER_WRITE_DELAY); | 2074 | udelay(AFE_REGISTER_WRITE_DELAY); |
2024 | /* Enable TX equalization (0xe824) */ | 2075 | /* Enable TX equalization (0xe824) */ |
2025 | writel(0x00040000, &xcvr->afe_tx_control); | 2076 | writel(0x00040000, &xcvr->afe_tx_control); |
2026 | } else { | 2077 | } else if (is_c0(pdev)) { |
2027 | writel(0x0140DF0F, &xcvr->afe_rx_ssc_control1); | 2078 | writel(0x01400C0F, &xcvr->afe_rx_ssc_control1); |
2028 | udelay(AFE_REGISTER_WRITE_DELAY); | 2079 | udelay(AFE_REGISTER_WRITE_DELAY); |
2029 | 2080 | ||
2030 | writel(0x3F6F103F, &xcvr->afe_rx_ssc_control0); | 2081 | writel(0x3F6F103F, &xcvr->afe_rx_ssc_control0); |
@@ -2032,6 +2083,22 @@ static void sci_controller_afe_initialization(struct isci_host *ihost) | |||
2032 | 2083 | ||
2033 | /* Enable TX equalization (0xe824) */ | 2084 | /* Enable TX equalization (0xe824) */ |
2034 | writel(0x00040000, &xcvr->afe_tx_control); | 2085 | writel(0x00040000, &xcvr->afe_tx_control); |
2086 | } else if (is_c1(pdev)) { | ||
2087 | writel(cable_length_long ? 0x01500C0C : | ||
2088 | cable_length_medium ? 0x01400C0D : 0x02400C0D, | ||
2089 | &xcvr->afe_xcvr_control1); | ||
2090 | udelay(AFE_REGISTER_WRITE_DELAY); | ||
2091 | |||
2092 | writel(0x000003E0, &xcvr->afe_dfx_rx_control1); | ||
2093 | udelay(AFE_REGISTER_WRITE_DELAY); | ||
2094 | |||
2095 | writel(cable_length_long ? 0x33091C1F : | ||
2096 | cable_length_medium ? 0x3315181F : 0x2B17161F, | ||
2097 | &xcvr->afe_rx_ssc_control0); | ||
2098 | udelay(AFE_REGISTER_WRITE_DELAY); | ||
2099 | |||
2100 | /* Enable TX equalization (0xe824) */ | ||
2101 | writel(0x00040000, &xcvr->afe_tx_control); | ||
2035 | } | 2102 | } |
2036 | 2103 | ||
2037 | udelay(AFE_REGISTER_WRITE_DELAY); | 2104 | udelay(AFE_REGISTER_WRITE_DELAY); |
diff --git a/drivers/scsi/isci/host.h b/drivers/scsi/isci/host.h index 646051afd3cb..4573075a6b97 100644 --- a/drivers/scsi/isci/host.h +++ b/drivers/scsi/isci/host.h | |||
@@ -435,7 +435,14 @@ static inline bool is_b0(struct pci_dev *pdev) | |||
435 | 435 | ||
436 | static inline bool is_c0(struct pci_dev *pdev) | 436 | static inline bool is_c0(struct pci_dev *pdev) |
437 | { | 437 | { |
438 | if (pdev->revision >= 5) | 438 | if (pdev->revision == 5) |
439 | return true; | ||
440 | return false; | ||
441 | } | ||
442 | |||
443 | static inline bool is_c1(struct pci_dev *pdev) | ||
444 | { | ||
445 | if (pdev->revision >= 6) | ||
439 | return true; | 446 | return true; |
440 | return false; | 447 | return false; |
441 | } | 448 | } |
diff --git a/drivers/scsi/isci/phy.c b/drivers/scsi/isci/phy.c index 0ae0990b39dd..c650d3003c22 100644 --- a/drivers/scsi/isci/phy.c +++ b/drivers/scsi/isci/phy.c | |||
@@ -186,8 +186,11 @@ sci_phy_link_layer_initialization(struct isci_phy *iphy, | |||
186 | 186 | ||
187 | writel(clksm_value, &llr->clock_skew_management); | 187 | writel(clksm_value, &llr->clock_skew_management); |
188 | 188 | ||
189 | /* @todo Provide a way to write this register correctly */ | 189 | if (is_c0(ihost->pdev) || is_c1(ihost->pdev)) { |
190 | writel(0x02108421, &llr->afe_lookup_table_control); | 190 | writel(0x04210400, &llr->afe_lookup_table_control); |
191 | writel(0x020A7C05, &llr->sas_primitive_timeout); | ||
192 | } else | ||
193 | writel(0x02108421, &llr->afe_lookup_table_control); | ||
191 | 194 | ||
192 | llctl = SCU_SAS_LLCTL_GEN_VAL(NO_OUTBOUND_TASK_TIMEOUT, | 195 | llctl = SCU_SAS_LLCTL_GEN_VAL(NO_OUTBOUND_TASK_TIMEOUT, |
193 | (u8)ihost->user_parameters.no_outbound_task_timeout); | 196 | (u8)ihost->user_parameters.no_outbound_task_timeout); |
diff --git a/drivers/scsi/isci/probe_roms.c b/drivers/scsi/isci/probe_roms.c index b5f4341de243..9b8117b9d756 100644 --- a/drivers/scsi/isci/probe_roms.c +++ b/drivers/scsi/isci/probe_roms.c | |||
@@ -147,7 +147,7 @@ struct isci_orom *isci_request_firmware(struct pci_dev *pdev, const struct firmw | |||
147 | 147 | ||
148 | memcpy(orom, fw->data, fw->size); | 148 | memcpy(orom, fw->data, fw->size); |
149 | 149 | ||
150 | if (is_c0(pdev)) | 150 | if (is_c0(pdev) || is_c1(pdev)) |
151 | goto out; | 151 | goto out; |
152 | 152 | ||
153 | /* | 153 | /* |