1 /*- 2 * Copyright (c) 2015,2016 Annapurna Labs Ltd. and affiliates 3 * All rights reserved. 4 * 5 * Developed by Semihalf. 6 * 7 * Redistribution and use in source and binary forms, with or without 8 * modification, are permitted provided that the following conditions 9 * are met: 10 * 1. Redistributions of source code must retain the above copyright 11 * notice, this list of conditions and the following disclaimer. 12 * 2. Redistributions in binary form must reproduce the above copyright 13 * notice, this list of conditions and the following disclaimer in the 14 * documentation and/or other materials provided with the distribution. 15 * 16 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND 17 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 18 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 19 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE 20 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 21 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 22 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 23 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 24 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 25 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 26 * SUCH DAMAGE. 27 */ 28 29 #include <sys/cdefs.h> 30 __FBSDID("$FreeBSD$"); 31 32 #include "al_init_eth_lm.h" 33 #include "al_serdes.h" 34 #include "al_hal_eth.h" 35 #include "al_init_eth_kr.h" 36 37 /** 38 * @{ 39 * @file al_init_eth_lm.c 40 * 41 * @brief ethernet link management common utilities 42 * 43 */ 44 45 /* delay before checking link status with new serdes parameters (uSec) */ 46 #define AL_ETH_LM_LINK_STATUS_DELAY 1000 47 /* delay before checking link status after reconfiguring the retimer (uSec) */ 48 #define AL_ETH_LM_RETIMER_LINK_STATUS_DELAY 50000 49 50 #define AL_ETH_LM_EQ_ITERATIONS 15 51 #define AL_ETH_LM_MAX_DCGAIN 8 52 53 /* num of link training failures till serdes reset */ 54 #define AL_ETH_LT_FAILURES_TO_RESET 10 55 56 #define MODULE_IDENTIFIER_IDX 0 57 #define MODULE_IDENTIFIER_SFP 0x3 58 #define MODULE_IDENTIFIER_QSFP 0xd 59 60 #define SFP_PRESENT 0 61 #define SFP_NOT_PRESENT 1 62 63 /* SFP+ module */ 64 #define SFP_I2C_HEADER_10G_IDX 3 65 #define SFP_I2C_HEADER_10G_DA_IDX 8 66 #define SFP_I2C_HEADER_10G_DA_LEN_IDX 18 67 #define SFP_I2C_HEADER_1G_IDX 6 68 #define SFP_I2C_HEADER_SIGNAL_RATE 12 /* Nominal signaling rate, units of 100MBd. */ 69 70 #define SFP_MIN_SIGNAL_RATE_25G 250 71 #define SFP_MIN_SIGNAL_RATE_10G 100 72 73 /* QSFP+ module */ 74 #define QSFP_COMPLIANCE_CODE_IDX 131 75 /* 40GBASE-LR4 and 40GBASE-SR4 are optic modules */ 76 #define QSFP_COMPLIANCE_CODE_OPTIC ((1 << 1) | (1 << 2)) 77 #define QSFP_COMPLIANCE_CODE_DAC (1 << 3) 78 #define QSFP_CABLE_LEN_IDX 146 79 80 /* TODO: need to check the necessary delay */ 81 #define AL_ETH_LM_RETIMER_WAIT_FOR_LOCK 500 /* delay after retimer reset to lock (mSec) */ 82 #define AL_ETH_LM_SERDES_WAIT_FOR_LOCK 50 /* delay after signal detect to lock (mSec) */ 83 84 #define AL_ETH_LM_GEARBOX_RESET_DELAY 1000 /* (uSec) */ 85 86 static const uint32_t 87 al_eth_retimer_boost_addr[AL_ETH_RETIMER_CHANNEL_MAX][AL_ETH_RETIMER_TYPE_MAX] = { 88 /* BR_210 | BR_410 */ 89 /* AL_ETH_RETIMER_CHANNEL_A */ {0xf, 0x1a}, 90 /* AL_ETH_RETIMER_CHANNEL_B */ {0x16, 0x18}, 91 /* AL_ETH_RETIMER_CHANNEL_C */ {0x0, 0x16}, 92 /* AL_ETH_RETIMER_CHANNEL_D */ {0x0, 0x14}, 93 }; 94 95 #define RETIMER_LENS_MAX 5 96 static const uint32_t 97 al_eth_retimer_boost_lens[RETIMER_LENS_MAX] = {0, 1, 2, 3, 5}; 98 99 static const uint32_t 100 al_eth_retimer_boost_value[RETIMER_LENS_MAX + 1][AL_ETH_RETIMER_TYPE_MAX] = { 101 /* BR_210 | BR_410 */ 102 /* 0 */ {0x0, 0x0}, 103 /* 1 */ {0x1, 0x1}, 104 /* 2 */ {0x2, 0x1}, 105 /* 3 */ {0x3, 0x3}, 106 /* 5 */ {0x7, 0x3}, 107 /* 5+ */{0xb, 0x7}, 108 }; 109 110 struct retimer_config_reg { 111 uint8_t addr; 112 uint8_t value; 113 uint8_t mask; 114 }; 115 116 static struct retimer_config_reg retimer_ds25_25g_mode_tx_ch[] = { 117 {.addr = 0x0A, .value = 0x0C, .mask = 0xff }, 118 {.addr = 0x2F, .value = 0x54, .mask = 0xff }, 119 {.addr = 0x31, .value = 0x20, .mask = 0xff }, 120 {.addr = 0x1E, .value = 0xE9, .mask = 0xff }, 121 {.addr = 0x1F, .value = 0x0B, .mask = 0xff }, 122 {.addr = 0xA6, .value = 0x43, .mask = 0xff }, 123 {.addr = 0x2A, .value = 0x5A, .mask = 0xff }, 124 {.addr = 0x2B, .value = 0x0A, .mask = 0xff }, 125 {.addr = 0x2C, .value = 0xF6, .mask = 0xff }, 126 {.addr = 0x70, .value = 0x05, .mask = 0xff }, 127 {.addr = 0x6A, .value = 0x21, .mask = 0xff }, 128 {.addr = 0x35, .value = 0x0F, .mask = 0xff }, 129 {.addr = 0x12, .value = 0x83, .mask = 0xff }, 130 {.addr = 0x9C, .value = 0x24, .mask = 0xff }, 131 {.addr = 0x98, .value = 0x00, .mask = 0xff }, 132 {.addr = 0x42, .value = 0x50, .mask = 0xff }, 133 {.addr = 0x44, .value = 0x90, .mask = 0xff }, 134 {.addr = 0x45, .value = 0xC0, .mask = 0xff }, 135 {.addr = 0x46, .value = 0xD0, .mask = 0xff }, 136 {.addr = 0x47, .value = 0xD1, .mask = 0xff }, 137 {.addr = 0x48, .value = 0xD5, .mask = 0xff }, 138 {.addr = 0x49, .value = 0xD8, .mask = 0xff }, 139 {.addr = 0x4A, .value = 0xEA, .mask = 0xff }, 140 {.addr = 0x4B, .value = 0xF7, .mask = 0xff }, 141 {.addr = 0x4C, .value = 0xFD, .mask = 0xff }, 142 {.addr = 0x8E, .value = 0x00, .mask = 0xff }, 143 {.addr = 0x3D, .value = 0x94, .mask = 0xff }, 144 {.addr = 0x3F, .value = 0x40, .mask = 0xff }, 145 {.addr = 0x3E, .value = 0x43, .mask = 0xff }, 146 {.addr = 0x0A, .value = 0x00, .mask = 0xff }, 147 }; 148 149 static struct retimer_config_reg retimer_ds25_25g_mode_rx_ch[] = { 150 {.addr = 0x0A, .value = 0x0C, .mask = 0xff}, 151 {.addr = 0x2F, .value = 0x54, .mask = 0xff}, 152 {.addr = 0x31, .value = 0x40, .mask = 0xff}, 153 {.addr = 0x1E, .value = 0xE3, .mask = 0xff}, 154 {.addr = 0x1F, .value = 0x0B, .mask = 0xff}, 155 {.addr = 0xA6, .value = 0x43, .mask = 0xff}, 156 {.addr = 0x2A, .value = 0x5A, .mask = 0xff}, 157 {.addr = 0x2B, .value = 0x0A, .mask = 0xff}, 158 {.addr = 0x2C, .value = 0xF6, .mask = 0xff}, 159 {.addr = 0x70, .value = 0x05, .mask = 0xff}, 160 {.addr = 0x6A, .value = 0x21, .mask = 0xff}, 161 {.addr = 0x35, .value = 0x0F, .mask = 0xff}, 162 {.addr = 0x12, .value = 0x83, .mask = 0xff}, 163 {.addr = 0x9C, .value = 0x24, .mask = 0xff}, 164 {.addr = 0x98, .value = 0x00, .mask = 0xff}, 165 {.addr = 0x42, .value = 0x50, .mask = 0xff}, 166 {.addr = 0x44, .value = 0x90, .mask = 0xff}, 167 {.addr = 0x45, .value = 0xC0, .mask = 0xff}, 168 {.addr = 0x46, .value = 0xD0, .mask = 0xff}, 169 {.addr = 0x47, .value = 0xD1, .mask = 0xff}, 170 {.addr = 0x48, .value = 0xD5, .mask = 0xff}, 171 {.addr = 0x49, .value = 0xD8, .mask = 0xff}, 172 {.addr = 0x4A, .value = 0xEA, .mask = 0xff}, 173 {.addr = 0x4B, .value = 0xF7, .mask = 0xff}, 174 {.addr = 0x4C, .value = 0xFD, .mask = 0xff}, 175 {.addr = 0x8E, .value = 0x00, .mask = 0xff}, 176 {.addr = 0x3D, .value = 0x94, .mask = 0xff}, 177 {.addr = 0x3F, .value = 0x40, .mask = 0xff}, 178 {.addr = 0x3E, .value = 0x43, .mask = 0xff}, 179 {.addr = 0x0A, .value = 0x00, .mask = 0xff}, 180 }; 181 182 static struct retimer_config_reg retimer_ds25_10g_mode[] = { 183 /* Assert CDR reset (6.3) */ 184 {.addr = 0x0A, .value = 0x0C, .mask = 0x0C}, 185 /* Select 10.3125Gbps standard rate mode (6.6) */ 186 {.addr = 0x2F, .value = 0x00, .mask = 0xF0}, 187 /* Enable loop filter auto-adjust */ 188 {.addr = 0x1F, .value = 0x08, .mask = 0x08}, 189 /* Set Adapt Mode 1 (6.13) */ 190 {.addr = 0x31, .value = 0x20, .mask = 0x60}, 191 /* Disable the DFE since most applications do not need it (6.18) */ 192 {.addr = 0x1E, .value = 0x08, .mask = 0x08}, 193 /* Release CDR reset (6.4) */ 194 {.addr = 0x0A, .value = 0x00, .mask = 0x0C}, 195 /* Enable FIR (6.12) */ 196 {.addr = 0x3D, .value = 0x80, .mask = 0x80}, 197 /* Set Main-cursor tap sign to positive (6.12) */ 198 {.addr = 0x3D, .value = 0x00, .mask = 0x40}, 199 /* Set Post-cursor tap sign to negative (6.12) */ 200 {.addr = 0x3F, .value = 0x40, .mask = 0x40}, 201 /* Set Pre-cursor tap sign to negative (6.12) */ 202 {.addr = 0x3E, .value = 0x40, .mask = 0x40}, 203 /* Set Main-cursor tap magnitude to 13 (6.12) */ 204 {.addr = 0x3D, .value = 0x0D, .mask = 0x1F}, 205 }; 206 207 static int al_eth_lm_retimer_boost_config(struct al_eth_lm_context *lm_context); 208 static int al_eth_lm_retimer_ds25_full_config(struct al_eth_lm_context *lm_context); 209 static al_bool al_eth_lm_retimer_ds25_signal_detect( 210 struct al_eth_lm_context *lm_context, uint32_t channel); 211 static int al_eth_lm_retimer_ds25_cdr_reset(struct al_eth_lm_context *lm_context, uint32_t channel); 212 static al_bool al_eth_lm_retimer_ds25_cdr_lock( 213 struct al_eth_lm_context *lm_context, uint32_t channel); 214 static int al_eth_lm_retimer_25g_rx_adaptation(struct al_eth_lm_context *lm_context); 215 216 struct al_eth_lm_retimer { 217 int (*config)(struct al_eth_lm_context *lm_context); 218 int (*reset)(struct al_eth_lm_context *lm_context, uint32_t channel); 219 int (*signal_detect)(struct al_eth_lm_context *lm_context, uint32_t channel); 220 int (*cdr_lock)(struct al_eth_lm_context *lm_context, uint32_t channel); 221 int (*rx_adaptation)(struct al_eth_lm_context *lm_context); 222 }; 223 224 static struct al_eth_lm_retimer retimer[] = { 225 {.config = al_eth_lm_retimer_boost_config, .signal_detect = NULL, 226 .reset = NULL, .cdr_lock = NULL, .rx_adaptation = NULL}, 227 {.config = al_eth_lm_retimer_boost_config, .signal_detect = NULL, 228 .reset = NULL, .cdr_lock = NULL, .rx_adaptation = NULL}, 229 {.config = al_eth_lm_retimer_ds25_full_config, 230 .signal_detect = al_eth_lm_retimer_ds25_signal_detect, 231 .reset = al_eth_lm_retimer_ds25_cdr_reset, 232 .cdr_lock = al_eth_lm_retimer_ds25_cdr_lock, 233 .rx_adaptation = al_eth_lm_retimer_25g_rx_adaptation}, 234 }; 235 236 #define SFP_10G_DA_ACTIVE 0x8 237 #define SFP_10G_DA_PASSIVE 0x4 238 239 #define lm_debug(...) \ 240 do { \ 241 if (lm_context->debug) \ 242 al_warn(__VA_ARGS__); \ 243 else \ 244 al_dbg(__VA_ARGS__); \ 245 } while (0) 246 247 static int 248 al_eth_sfp_detect(struct al_eth_lm_context *lm_context, 249 enum al_eth_lm_link_mode *new_mode) 250 { 251 int rc = 0; 252 uint8_t sfp_10g; 253 uint8_t sfp_1g; 254 uint8_t sfp_cable_tech; 255 uint8_t sfp_da_len; 256 uint8_t signal_rate; 257 258 do { 259 rc = lm_context->i2c_read(lm_context->i2c_context, 260 lm_context->sfp_bus_id, lm_context->sfp_i2c_addr, 261 SFP_I2C_HEADER_10G_IDX, &sfp_10g); 262 if (rc != 0) 263 break; 264 265 rc = lm_context->i2c_read(lm_context->i2c_context, 266 lm_context->sfp_bus_id, lm_context->sfp_i2c_addr, 267 SFP_I2C_HEADER_1G_IDX, &sfp_1g); 268 if (rc != 0) 269 break; 270 271 rc = lm_context->i2c_read(lm_context->i2c_context, 272 lm_context->sfp_bus_id, lm_context->sfp_i2c_addr, 273 SFP_I2C_HEADER_10G_DA_IDX, &sfp_cable_tech); 274 if (rc != 0) 275 break; 276 277 rc = lm_context->i2c_read(lm_context->i2c_context, 278 lm_context->sfp_bus_id, lm_context->sfp_i2c_addr, 279 SFP_I2C_HEADER_10G_DA_LEN_IDX, &sfp_da_len); 280 if (rc != 0) 281 break; 282 283 rc = lm_context->i2c_read(lm_context->i2c_context, 284 lm_context->sfp_bus_id, 285 lm_context->sfp_i2c_addr, 286 SFP_I2C_HEADER_SIGNAL_RATE, 287 &signal_rate); 288 } while (0); 289 290 if (rc != 0) { 291 if (rc == ETIMEDOUT) { 292 /* ETIMEDOUT is returned when no SFP is connected */ 293 if (lm_context->mode != AL_ETH_LM_MODE_DISCONNECTED) 294 lm_debug("%s: SFP Disconnected\n", __func__); 295 *new_mode = AL_ETH_LM_MODE_DISCONNECTED; 296 } else { 297 return (rc); 298 } 299 } else if ((sfp_cable_tech & (SFP_10G_DA_PASSIVE | SFP_10G_DA_ACTIVE)) != 0) { 300 if ((signal_rate >= SFP_MIN_SIGNAL_RATE_25G) && 301 ((lm_context->max_speed == AL_ETH_LM_MAX_SPEED_25G) || 302 (lm_context->max_speed == AL_ETH_LM_MAX_SPEED_MAX))) 303 *new_mode = AL_ETH_LM_MODE_25G; 304 else if ((signal_rate >= SFP_MIN_SIGNAL_RATE_10G) && 305 ((lm_context->max_speed == AL_ETH_LM_MAX_SPEED_10G) || 306 (lm_context->max_speed == AL_ETH_LM_MAX_SPEED_MAX))) 307 *new_mode = AL_ETH_LM_MODE_10G_DA; 308 else 309 *new_mode = AL_ETH_LM_MODE_1G; 310 311 lm_debug("%s: %s DAC (%d M) detected (max signal rate %d)\n", 312 __func__, 313 (sfp_cable_tech & SFP_10G_DA_PASSIVE) ? "Passive" : "Active", 314 sfp_da_len, 315 signal_rate); 316 317 /* for active direct attached need to use len 0 in the retimer configuration */ 318 lm_context->da_len = (sfp_cable_tech & SFP_10G_DA_PASSIVE) ? sfp_da_len : 0; 319 } else if (sfp_10g != 0) { 320 lm_debug("%s: 10 SFP detected\n", __func__); 321 *new_mode = AL_ETH_LM_MODE_10G_OPTIC; 322 } else if (sfp_1g != 0) { 323 lm_debug("%s: 1G SFP detected\n", __func__); 324 *new_mode = AL_ETH_LM_MODE_1G; 325 } else { 326 al_warn("%s: unknown SFP inserted. eeprom content: 10G compliance 0x%x," 327 " 1G compliance 0x%x, sfp+cable 0x%x. default to %s\n", 328 __func__, sfp_10g, sfp_1g, sfp_cable_tech, 329 al_eth_lm_mode_convert_to_str(lm_context->default_mode)); 330 *new_mode = lm_context->default_mode; 331 lm_context->da_len = lm_context->default_dac_len; 332 } 333 334 if ((lm_context->sfp_detect_force_mode) && (*new_mode != AL_ETH_LM_MODE_DISCONNECTED) && 335 (*new_mode != lm_context->default_mode)) { 336 al_warn("%s: Force mode to default (%s). mode based of the SFP EEPROM %s\n", 337 __func__, al_eth_lm_mode_convert_to_str(lm_context->default_mode), 338 al_eth_lm_mode_convert_to_str(*new_mode)); 339 340 *new_mode = lm_context->default_mode; 341 } 342 343 lm_context->mode = *new_mode; 344 345 return (0); 346 } 347 348 static int 349 al_eth_qsfp_detect(struct al_eth_lm_context *lm_context, 350 enum al_eth_lm_link_mode *new_mode) 351 { 352 int rc = 0; 353 uint8_t qsfp_comp_code; 354 uint8_t qsfp_da_len; 355 356 do { 357 rc = lm_context->i2c_read(lm_context->i2c_context, 358 lm_context->sfp_bus_id, lm_context->sfp_i2c_addr, 359 QSFP_COMPLIANCE_CODE_IDX, &qsfp_comp_code); 360 if (rc != 0) 361 break; 362 363 rc = lm_context->i2c_read(lm_context->i2c_context, 364 lm_context->sfp_bus_id, lm_context->sfp_i2c_addr, 365 QSFP_CABLE_LEN_IDX, &qsfp_da_len); 366 if (rc != 0) 367 break; 368 } while (0); 369 370 if (rc != 0) { 371 if (rc == ETIMEDOUT) { 372 /* ETIMEDOUT is returned when no SFP is connected */ 373 lm_debug("%s: SFP Disconnected\n", __func__); 374 *new_mode = AL_ETH_LM_MODE_DISCONNECTED; 375 } else { 376 return (rc); 377 } 378 } else if ((qsfp_comp_code & QSFP_COMPLIANCE_CODE_DAC) != 0) { 379 lm_debug("%s: 10G passive DAC (%d M) detected\n", 380 __func__, qsfp_da_len); 381 *new_mode = AL_ETH_LM_MODE_10G_DA; 382 lm_context->da_len = qsfp_da_len; 383 } else if ((qsfp_comp_code & QSFP_COMPLIANCE_CODE_OPTIC) != 0) { 384 lm_debug("%s: 10G optic module detected\n", __func__); 385 *new_mode = AL_ETH_LM_MODE_10G_OPTIC; 386 } else { 387 al_warn("%s: unknown QSFP inserted. eeprom content: 10G " 388 "compliance 0x%x default to %s\n", __func__, qsfp_comp_code, 389 al_eth_lm_mode_convert_to_str(lm_context->default_mode)); 390 *new_mode = lm_context->default_mode; 391 lm_context->da_len = lm_context->default_dac_len; 392 } 393 394 lm_context->mode = *new_mode; 395 396 return (0); 397 } 398 399 static int 400 al_eth_module_detect(struct al_eth_lm_context *lm_context, 401 enum al_eth_lm_link_mode *new_mode) 402 { 403 int rc = 0; 404 uint8_t module_idx; 405 int sfp_present = SFP_PRESENT; 406 407 if ((lm_context->gpio_get) && (lm_context->gpio_present != 0)) 408 sfp_present = lm_context->gpio_get(lm_context->gpio_present); 409 410 if (sfp_present == SFP_NOT_PRESENT) { 411 lm_debug("%s: SFP not exist\n", __func__); 412 *new_mode = AL_ETH_LM_MODE_DISCONNECTED; 413 414 return 0; 415 } 416 417 rc = lm_context->i2c_read(lm_context->i2c_context, 418 lm_context->sfp_bus_id, lm_context->sfp_i2c_addr, 419 MODULE_IDENTIFIER_IDX, &module_idx); 420 if (rc != 0) { 421 if (rc == ETIMEDOUT) { 422 /* ETIMEDOUT is returned when no SFP is connected */ 423 if (lm_context->mode != AL_ETH_LM_MODE_DISCONNECTED) 424 lm_debug("%s: SFP Disconnected\n", __func__); 425 *new_mode = AL_ETH_LM_MODE_DISCONNECTED; 426 return (0); 427 } else { 428 return (rc); 429 } 430 } 431 432 if (module_idx == MODULE_IDENTIFIER_QSFP) 433 return (al_eth_qsfp_detect(lm_context, new_mode)); 434 else 435 return (al_eth_sfp_detect(lm_context, new_mode)); 436 437 return (0); 438 } 439 440 static struct al_serdes_adv_tx_params da_tx_params = { 441 .override = TRUE, 442 .amp = 0x1, 443 .total_driver_units = 0x13, 444 .c_plus_1 = 0x2, 445 .c_plus_2 = 0, 446 .c_minus_1 = 0x2, 447 .slew_rate = 0, 448 }; 449 450 static struct al_serdes_adv_rx_params da_rx_params = { 451 .override = TRUE, 452 .dcgain = 0x4, 453 .dfe_3db_freq = 0x4, 454 .dfe_gain = 0x3, 455 .dfe_first_tap_ctrl = 0x5, 456 .dfe_secound_tap_ctrl = 0x1, 457 .dfe_third_tap_ctrl = 0x8, 458 .dfe_fourth_tap_ctrl = 0x1, 459 .low_freq_agc_gain = 0x7, 460 .precal_code_sel = 0, 461 .high_freq_agc_boost = 0x1d, 462 }; 463 464 static struct al_serdes_adv_tx_params optic_tx_params = { 465 .override = TRUE, 466 .amp = 0x1, 467 .total_driver_units = 0x13, 468 .c_plus_1 = 0x2, 469 .c_plus_2 = 0, 470 .c_minus_1 = 0, 471 .slew_rate = 0, 472 }; 473 474 static struct al_serdes_adv_rx_params optic_rx_params = { 475 .override = TRUE, 476 .dcgain = 0x0, 477 .dfe_3db_freq = 0x7, 478 .dfe_gain = 0x0, 479 .dfe_first_tap_ctrl = 0x0, 480 .dfe_secound_tap_ctrl = 0x8, 481 .dfe_third_tap_ctrl = 0x0, 482 .dfe_fourth_tap_ctrl = 0x8, 483 .low_freq_agc_gain = 0x7, 484 .precal_code_sel = 0, 485 .high_freq_agc_boost = 0x4, 486 }; 487 488 static void 489 al_eth_serdes_static_tx_params_set(struct al_eth_lm_context *lm_context) 490 { 491 492 if (lm_context->tx_param_dirty == 0) 493 return; 494 495 if (lm_context->serdes_tx_params_valid != 0) { 496 lm_context->tx_param_dirty = 0; 497 498 lm_context->tx_params_override.override = TRUE; 499 500 if ((lm_context->serdes_obj->tx_advanced_params_set) == 0) { 501 al_err("tx_advanced_params_set is not supported for this serdes group\n"); 502 return; 503 } 504 505 lm_context->serdes_obj->tx_advanced_params_set( 506 lm_context->serdes_obj, 507 lm_context->lane, 508 &lm_context->tx_params_override); 509 510 } else if (lm_context->static_values != 0) { 511 lm_context->tx_param_dirty = 0; 512 513 if ((lm_context->serdes_obj->tx_advanced_params_set) == 0) { 514 al_err("tx_advanced_params_set is not supported for this serdes group\n"); 515 return; 516 } 517 518 if ((lm_context->retimer_exist == 0) && 519 (lm_context->mode == AL_ETH_LM_MODE_10G_DA)) 520 lm_context->serdes_obj->tx_advanced_params_set( 521 lm_context->serdes_obj, 522 lm_context->lane, 523 &da_tx_params); 524 else 525 lm_context->serdes_obj->tx_advanced_params_set( 526 lm_context->serdes_obj, 527 lm_context->lane, 528 &optic_tx_params); 529 } 530 } 531 532 static void 533 al_eth_serdes_static_rx_params_set(struct al_eth_lm_context *lm_context) 534 { 535 536 if (lm_context->rx_param_dirty == 0) 537 return; 538 539 if (lm_context->serdes_rx_params_valid != 0) { 540 lm_context->rx_param_dirty = 0; 541 542 lm_context->rx_params_override.override = TRUE; 543 544 if ((lm_context->serdes_obj->rx_advanced_params_set) == 0) { 545 al_err("rx_advanced_params_set is not supported for this serdes group\n"); 546 return; 547 } 548 549 lm_context->serdes_obj->rx_advanced_params_set( 550 lm_context->serdes_obj, 551 lm_context->lane, 552 &lm_context->rx_params_override); 553 554 } else if (lm_context->static_values != 0) { 555 lm_context->rx_param_dirty = 0; 556 557 if ((lm_context->serdes_obj->rx_advanced_params_set) == 0) { 558 al_err("rx_advanced_params_set is not supported for this serdes group\n"); 559 return; 560 } 561 562 if ((lm_context->retimer_exist == 0) && 563 (lm_context->mode == AL_ETH_LM_MODE_10G_DA)) 564 lm_context->serdes_obj->rx_advanced_params_set( 565 lm_context->serdes_obj, 566 lm_context->lane, 567 &da_rx_params); 568 else 569 lm_context->serdes_obj->rx_advanced_params_set( 570 lm_context->serdes_obj, 571 lm_context->lane, 572 &optic_rx_params); 573 } 574 } 575 576 static int 577 al_eth_rx_equal_run(struct al_eth_lm_context *lm_context) 578 { 579 struct al_serdes_adv_rx_params rx_params; 580 int dcgain; 581 int best_dcgain = -1; 582 int i; 583 int best_score = -1; 584 int test_score = -1; 585 586 rx_params.override = FALSE; 587 lm_context->serdes_obj->rx_advanced_params_set(lm_context->serdes_obj, 588 lm_context->lane, &rx_params); 589 590 lm_debug("score | dcgain | dfe3db | dfegain | tap1 | tap2 | tap3 | " 591 "tap4 | low freq | high freq\n"); 592 593 for (dcgain = 0; dcgain < AL_ETH_LM_MAX_DCGAIN; dcgain++) { 594 lm_context->serdes_obj->dcgain_set( 595 lm_context->serdes_obj, 596 dcgain); 597 598 test_score = lm_context->serdes_obj->rx_equalization( 599 lm_context->serdes_obj, 600 lm_context->lane); 601 602 if (test_score < 0) { 603 al_warn("serdes rx equalization failed on error\n"); 604 return (test_score); 605 } 606 607 if (test_score > best_score) { 608 best_score = test_score; 609 best_dcgain = dcgain; 610 } 611 612 lm_context->serdes_obj->rx_advanced_params_get( 613 lm_context->serdes_obj, 614 lm_context->lane, 615 &rx_params); 616 617 lm_debug("%6d|%8x|%8x|%9x|%6x|%6x|%6x|%6x|%10x|%10x|\n", 618 test_score, rx_params.dcgain, rx_params.dfe_3db_freq, 619 rx_params.dfe_gain, rx_params.dfe_first_tap_ctrl, 620 rx_params.dfe_secound_tap_ctrl, rx_params.dfe_third_tap_ctrl, 621 rx_params.dfe_fourth_tap_ctrl, rx_params.low_freq_agc_gain, 622 rx_params.high_freq_agc_boost); 623 } 624 625 lm_context->serdes_obj->dcgain_set( 626 lm_context->serdes_obj, 627 best_dcgain); 628 629 best_score = -1; 630 for(i = 0; i < AL_ETH_LM_EQ_ITERATIONS; i++) { 631 test_score = lm_context->serdes_obj->rx_equalization( 632 lm_context->serdes_obj, 633 lm_context->lane); 634 635 if (test_score < 0) { 636 al_warn("serdes rx equalization failed on error\n"); 637 return (test_score); 638 } 639 640 if (test_score > best_score) { 641 best_score = test_score; 642 lm_context->serdes_obj->rx_advanced_params_get( 643 lm_context->serdes_obj, 644 lm_context->lane, 645 &rx_params); 646 } 647 } 648 649 rx_params.precal_code_sel = 0; 650 rx_params.override = TRUE; 651 lm_context->serdes_obj->rx_advanced_params_set( 652 lm_context->serdes_obj, 653 lm_context->lane, 654 &rx_params); 655 656 lm_debug("-------------------- best dcgain %d ------------------------------------\n", best_dcgain); 657 lm_debug("%6d|%8x|%8x|%9x|%6x|%6x|%6x|%6x|%10x|%10x|\n", 658 best_score, rx_params.dcgain, rx_params.dfe_3db_freq, 659 rx_params.dfe_gain, rx_params.dfe_first_tap_ctrl, 660 rx_params.dfe_secound_tap_ctrl, rx_params.dfe_third_tap_ctrl, 661 rx_params.dfe_fourth_tap_ctrl, rx_params.low_freq_agc_gain, 662 rx_params.high_freq_agc_boost); 663 664 return (0); 665 } 666 667 static int al_eth_lm_retimer_boost_config(struct al_eth_lm_context *lm_context) 668 { 669 int i; 670 int rc = 0; 671 uint8_t boost = 0; 672 uint32_t boost_addr = 673 al_eth_retimer_boost_addr[lm_context->retimer_channel][lm_context->retimer_type]; 674 675 if (lm_context->mode != AL_ETH_LM_MODE_10G_DA) { 676 boost = al_eth_retimer_boost_value[0][lm_context->retimer_type]; 677 } else { 678 for (i = 0; i < RETIMER_LENS_MAX; i++) { 679 if (lm_context->da_len <= al_eth_retimer_boost_lens[i]) { 680 boost = al_eth_retimer_boost_value[i][lm_context->retimer_type]; 681 break; 682 } 683 } 684 685 if (i == RETIMER_LENS_MAX) 686 boost = al_eth_retimer_boost_value[RETIMER_LENS_MAX][lm_context->retimer_type]; 687 } 688 689 lm_debug("config retimer boost in channel %d (addr %x) to 0x%x\n", 690 lm_context->retimer_channel, boost_addr, boost); 691 692 rc = lm_context->i2c_write(lm_context->i2c_context, 693 lm_context->retimer_bus_id, lm_context->retimer_i2c_addr, 694 boost_addr, boost); 695 696 if (rc != 0) { 697 al_err("%s: Error occurred (%d) while writing retimer " 698 "configuration (bus-id %x i2c-addr %x)\n", 699 __func__, rc, lm_context->retimer_bus_id, 700 lm_context->retimer_i2c_addr); 701 return (rc); 702 } 703 704 return (0); 705 } 706 707 /******************************************************************************* 708 ************************** retimer DS25 *************************************** 709 ******************************************************************************/ 710 #define LM_DS25_CHANNEL_EN_REG 0xff 711 #define LM_DS25_CHANNEL_EN_MASK 0x03 712 #define LM_DS25_CHANNEL_EN_VAL 0x01 713 714 #define LM_DS25_CHANNEL_SEL_REG 0xfc 715 #define LM_DS25_CHANNEL_SEL_MASK 0xff 716 717 #define LM_DS25_CDR_RESET_REG 0x0a 718 #define LM_DS25_CDR_RESET_MASK 0x0c 719 #define LM_DS25_CDR_RESET_ASSERT 0x0c 720 #define LM_DS25_CDR_RESET_RELEASE 0x00 721 722 #define LM_DS25_SIGNAL_DETECT_REG 0x78 723 #define LM_DS25_SIGNAL_DETECT_MASK 0x20 724 725 #define LM_DS25_CDR_LOCK_REG 0x78 726 #define LM_DS25_CDR_LOCK_MASK 0x10 727 728 #define LM_DS25_DRV_PD_REG 0x15 729 #define LM_DS25_DRV_PD_MASK 0x08 730 731 static int al_eth_lm_retimer_ds25_write_reg(struct al_eth_lm_context *lm_context, 732 uint8_t reg_addr, 733 uint8_t reg_mask, 734 uint8_t reg_value) 735 { 736 uint8_t reg; 737 int rc; 738 739 rc = lm_context->i2c_read(lm_context->i2c_context, 740 lm_context->retimer_bus_id, 741 lm_context->retimer_i2c_addr, 742 reg_addr, 743 ®); 744 745 if (rc != 0) 746 return (EIO); 747 748 reg &= ~(reg_mask); 749 reg |= reg_value; 750 751 rc = lm_context->i2c_write(lm_context->i2c_context, 752 lm_context->retimer_bus_id, 753 lm_context->retimer_i2c_addr, 754 reg_addr, 755 reg); 756 757 if (rc != 0) 758 return (EIO); 759 760 return (0); 761 } 762 763 static int al_eth_lm_retimer_ds25_channel_select(struct al_eth_lm_context *lm_context, 764 uint8_t channel) 765 { 766 int rc = 0; 767 768 /* Write to specific channel */ 769 rc = al_eth_lm_retimer_ds25_write_reg(lm_context, 770 LM_DS25_CHANNEL_EN_REG, 771 LM_DS25_CHANNEL_EN_MASK, 772 LM_DS25_CHANNEL_EN_VAL); 773 774 if (rc != 0) 775 return (rc); 776 777 rc = al_eth_lm_retimer_ds25_write_reg(lm_context, 778 LM_DS25_CHANNEL_SEL_REG, 779 LM_DS25_CHANNEL_SEL_MASK, 780 (1 << channel)); 781 782 return (rc); 783 } 784 785 static int al_eth_lm_retimer_ds25_channel_config(struct al_eth_lm_context *lm_context, 786 uint8_t channel, 787 struct retimer_config_reg *config, 788 uint8_t config_size) 789 { 790 uint8_t i; 791 int rc; 792 793 rc = al_eth_lm_retimer_ds25_channel_select(lm_context, channel); 794 if (rc != 0) 795 goto config_error; 796 797 for (i = 0; i < config_size; i++) { 798 rc = al_eth_lm_retimer_ds25_write_reg(lm_context, 799 config[i].addr, 800 config[i].mask, 801 config[i].value); 802 803 if (rc != 0) 804 goto config_error; 805 } 806 807 lm_debug("%s: retimer channel config done for channel %d\n", __func__, channel); 808 809 return (0); 810 811 config_error: 812 al_err("%s: failed to access to the retimer\n", __func__); 813 814 return (rc); 815 } 816 817 static int al_eth_lm_retimer_ds25_cdr_reset(struct al_eth_lm_context *lm_context, uint32_t channel) 818 { 819 int rc; 820 821 lm_debug("Perform CDR reset to channel %d\n", channel); 822 823 rc = al_eth_lm_retimer_ds25_channel_select(lm_context, channel); 824 if (rc) 825 goto config_error; 826 827 rc = al_eth_lm_retimer_ds25_write_reg(lm_context, 828 LM_DS25_CDR_RESET_REG, 829 LM_DS25_CDR_RESET_MASK, 830 LM_DS25_CDR_RESET_ASSERT); 831 832 if (rc) 833 goto config_error; 834 835 rc = al_eth_lm_retimer_ds25_write_reg(lm_context, 836 LM_DS25_CDR_RESET_REG, 837 LM_DS25_CDR_RESET_MASK, 838 LM_DS25_CDR_RESET_RELEASE); 839 840 if (rc) 841 goto config_error; 842 843 return 0; 844 845 config_error: 846 al_err("%s: failed to access to the retimer\n", __func__); 847 848 return rc; 849 } 850 851 static al_bool al_eth_lm_retimer_ds25_signal_detect(struct al_eth_lm_context *lm_context, 852 uint32_t channel) 853 { 854 int rc = 0; 855 uint8_t reg; 856 857 rc = al_eth_lm_retimer_ds25_channel_select(lm_context, channel); 858 if (rc) 859 goto config_error; 860 861 rc = lm_context->i2c_read(lm_context->i2c_context, 862 lm_context->retimer_bus_id, 863 lm_context->retimer_i2c_addr, 864 LM_DS25_SIGNAL_DETECT_REG, 865 ®); 866 867 if (rc) 868 goto config_error; 869 870 if (reg & LM_DS25_SIGNAL_DETECT_MASK) 871 return TRUE; 872 873 return FALSE; 874 875 config_error: 876 al_err("%s: failed to access to the retimer\n", __func__); 877 878 return FALSE; 879 } 880 881 static al_bool al_eth_lm_retimer_ds25_cdr_lock(struct al_eth_lm_context *lm_context, 882 uint32_t channel) 883 { 884 int rc = 0; 885 uint8_t reg; 886 887 rc = al_eth_lm_retimer_ds25_channel_select(lm_context, channel); 888 if (rc) 889 goto config_error; 890 891 rc = lm_context->i2c_read(lm_context->i2c_context, 892 lm_context->retimer_bus_id, 893 lm_context->retimer_i2c_addr, 894 LM_DS25_CDR_LOCK_REG, 895 ®); 896 897 if (rc) 898 goto config_error; 899 900 if (reg & LM_DS25_CDR_LOCK_MASK) 901 return TRUE; 902 903 return FALSE; 904 905 config_error: 906 al_err("%s: failed to access to the retimer\n", __func__); 907 908 return FALSE; 909 } 910 911 static boolean_t al_eth_lm_wait_for_lock(struct al_eth_lm_context *lm_context, 912 uint32_t channel) 913 { 914 uint32_t timeout = AL_ETH_LM_RETIMER_WAIT_FOR_LOCK; 915 al_bool lock = AL_FALSE; 916 917 while ((timeout > 0) && (lock == FALSE)) { 918 al_msleep(10); 919 timeout -= 10; 920 921 lock = retimer[lm_context->retimer_type].cdr_lock(lm_context, channel); 922 } 923 924 lm_debug("%s: %s to achieve CDR lock in %d msec\n", 925 __func__, (lock) ? "succeed" : "FAILED", 926 (AL_ETH_LM_RETIMER_WAIT_FOR_LOCK - timeout)); 927 928 return lock; 929 } 930 931 static void al_eth_lm_retimer_signal_lock_check(struct al_eth_lm_context *lm_context, 932 uint32_t channel, 933 boolean_t *ready) 934 { 935 al_bool signal_detect = TRUE; 936 al_bool cdr_lock = TRUE; 937 938 if (retimer[lm_context->retimer_type].signal_detect) { 939 if (!retimer[lm_context->retimer_type].signal_detect(lm_context, channel)) { 940 lm_debug("no signal detected on retimer channel %d\n", channel); 941 942 signal_detect = AL_FALSE; 943 } else { 944 if (retimer[lm_context->retimer_type].cdr_lock) { 945 cdr_lock = retimer[lm_context->retimer_type].cdr_lock( 946 lm_context, 947 channel); 948 if (!cdr_lock) { 949 if (retimer[lm_context->retimer_type].reset) { 950 retimer[lm_context->retimer_type].reset(lm_context, 951 channel); 952 953 cdr_lock = al_eth_lm_wait_for_lock(lm_context, 954 channel); 955 } 956 } 957 } 958 } 959 } 960 961 al_info("%s: (channel %d) signal %d cdr lock %d\n", 962 __func__, channel, signal_detect, (signal_detect) ? cdr_lock : 0); 963 964 *ready = ((cdr_lock == TRUE) && (signal_detect == TRUE)); 965 } 966 967 static int al_eth_lm_retimer_ds25_full_config(struct al_eth_lm_context *lm_context) 968 { 969 int rc = 0; 970 al_bool ready; 971 struct retimer_config_reg *config_tx; 972 uint32_t config_tx_size; 973 struct retimer_config_reg *config_rx; 974 uint32_t config_rx_size; 975 976 if (lm_context->mode == AL_ETH_LM_MODE_25G) { 977 config_tx = retimer_ds25_25g_mode_tx_ch; 978 config_tx_size = AL_ARR_SIZE(retimer_ds25_25g_mode_tx_ch); 979 980 config_rx = retimer_ds25_25g_mode_rx_ch; 981 config_rx_size = AL_ARR_SIZE(retimer_ds25_25g_mode_rx_ch); 982 983 } else { 984 config_tx = retimer_ds25_10g_mode; 985 config_tx_size = AL_ARR_SIZE(retimer_ds25_10g_mode); 986 987 config_rx = retimer_ds25_10g_mode; 988 config_rx_size = AL_ARR_SIZE(retimer_ds25_10g_mode); 989 } 990 991 rc = al_eth_lm_retimer_ds25_channel_config(lm_context, 992 lm_context->retimer_channel, 993 config_rx, 994 config_rx_size); 995 996 if (rc) 997 return rc; 998 999 rc = al_eth_lm_retimer_ds25_channel_config(lm_context, 1000 lm_context->retimer_tx_channel, 1001 config_tx, 1002 config_tx_size); 1003 1004 if (rc) 1005 return rc; 1006 1007 if (lm_context->serdes_obj->type_get() == AL_SRDS_TYPE_25G) { 1008 lm_debug("%s: serdes 25G - perform tx and rx gearbox reset\n", __func__); 1009 al_eth_gearbox_reset(lm_context->adapter, TRUE, TRUE); 1010 DELAY(AL_ETH_LM_GEARBOX_RESET_DELAY); 1011 } 1012 1013 al_eth_lm_retimer_signal_lock_check(lm_context, lm_context->retimer_tx_channel, &ready); 1014 1015 if (!ready) { 1016 lm_debug("%s: Failed to lock tx channel!\n", __func__); 1017 return (1); 1018 } 1019 1020 lm_debug("%s: retimer full configuration done\n", __func__); 1021 1022 return rc; 1023 } 1024 1025 static int al_eth_lm_retimer_25g_rx_adaptation(struct al_eth_lm_context *lm_context) 1026 { 1027 int rc = 0; 1028 al_bool ready; 1029 1030 al_eth_lm_retimer_signal_lock_check(lm_context, lm_context->retimer_channel, &ready); 1031 1032 if (!ready) { 1033 lm_debug("%s: no signal detected on retimer Rx channel (%d)\n", 1034 __func__, lm_context->retimer_channel); 1035 1036 return rc; 1037 } 1038 1039 al_msleep(AL_ETH_LM_SERDES_WAIT_FOR_LOCK); 1040 1041 return 0; 1042 } 1043 1044 static int al_eth_lm_check_for_link(struct al_eth_lm_context *lm_context, boolean_t *link_up) 1045 { 1046 struct al_eth_link_status status; 1047 int ret = 0; 1048 1049 al_eth_link_status_clear(lm_context->adapter); 1050 al_eth_link_status_get(lm_context->adapter, &status); 1051 1052 if (status.link_up == AL_TRUE) { 1053 lm_debug("%s: >>>> Link state DOWN ==> UP\n", __func__); 1054 al_eth_led_set(lm_context->adapter, AL_TRUE); 1055 lm_context->link_state = AL_ETH_LM_LINK_UP; 1056 *link_up = AL_TRUE; 1057 1058 return 0; 1059 } else if (status.local_fault) { 1060 lm_context->link_state = AL_ETH_LM_LINK_DOWN; 1061 al_eth_led_set(lm_context->adapter, AL_FALSE); 1062 1063 al_err("%s: Failed to establish link\n", __func__); 1064 ret = 1; 1065 } else { 1066 lm_debug("%s: >>>> Link state DOWN ==> DOWN_RF\n", __func__); 1067 lm_context->link_state = AL_ETH_LM_LINK_DOWN_RF; 1068 al_eth_led_set(lm_context->adapter, AL_FALSE); 1069 1070 ret = 0; 1071 } 1072 1073 *link_up = AL_FALSE; 1074 return ret; 1075 } 1076 1077 /*****************************************************************************/ 1078 /***************************** API functions *********************************/ 1079 /*****************************************************************************/ 1080 int 1081 al_eth_lm_init(struct al_eth_lm_context *lm_context, 1082 struct al_eth_lm_init_params *params) 1083 { 1084 1085 lm_context->adapter = params->adapter; 1086 lm_context->serdes_obj = params->serdes_obj; 1087 lm_context->lane = params->lane; 1088 lm_context->sfp_detection = params->sfp_detection; 1089 lm_context->sfp_bus_id = params->sfp_bus_id; 1090 lm_context->sfp_i2c_addr = params->sfp_i2c_addr; 1091 1092 lm_context->retimer_exist = params->retimer_exist; 1093 lm_context->retimer_type = params->retimer_type; 1094 lm_context->retimer_bus_id = params->retimer_bus_id; 1095 lm_context->retimer_i2c_addr = params->retimer_i2c_addr; 1096 lm_context->retimer_channel = params->retimer_channel; 1097 lm_context->retimer_tx_channel = params->retimer_tx_channel; 1098 1099 lm_context->default_mode = params->default_mode; 1100 lm_context->default_dac_len = params->default_dac_len; 1101 lm_context->link_training = params->link_training; 1102 lm_context->rx_equal = params->rx_equal; 1103 lm_context->static_values = params->static_values; 1104 lm_context->i2c_read = params->i2c_read; 1105 lm_context->i2c_write = params->i2c_write; 1106 lm_context->i2c_context = params->i2c_context; 1107 lm_context->get_random_byte = params->get_random_byte; 1108 1109 /* eeprom_read must be provided if sfp_detection is true */ 1110 al_assert((lm_context->sfp_detection == FALSE) || 1111 (lm_context->i2c_read != NULL)); 1112 1113 al_assert((lm_context->retimer_exist == FALSE) || 1114 (lm_context->i2c_write != NULL)); 1115 1116 lm_context->local_adv.selector_field = 1; 1117 lm_context->local_adv.capability = 0; 1118 lm_context->local_adv.remote_fault = 0; 1119 lm_context->local_adv.acknowledge = 0; 1120 lm_context->local_adv.next_page = 0; 1121 lm_context->local_adv.technology = AL_ETH_AN_TECH_10GBASE_KR; 1122 lm_context->local_adv.fec_capability = params->kr_fec_enable; 1123 1124 lm_context->mode = AL_ETH_LM_MODE_DISCONNECTED; 1125 lm_context->serdes_tx_params_valid = FALSE; 1126 lm_context->serdes_rx_params_valid = FALSE; 1127 1128 lm_context->rx_param_dirty = 1; 1129 lm_context->tx_param_dirty = 1; 1130 1131 lm_context->gpio_get = params->gpio_get; 1132 lm_context->gpio_present = params->gpio_present; 1133 1134 lm_context->max_speed = params->max_speed; 1135 lm_context->sfp_detect_force_mode = params->sfp_detect_force_mode; 1136 1137 lm_context->lm_pause = params->lm_pause; 1138 1139 lm_context->led_config = params->led_config; 1140 1141 lm_context->retimer_configured = FALSE; 1142 1143 lm_context->link_state = AL_ETH_LM_LINK_DOWN; 1144 1145 return (0); 1146 } 1147 1148 int 1149 al_eth_lm_link_detection(struct al_eth_lm_context *lm_context, 1150 boolean_t *link_fault, enum al_eth_lm_link_mode *old_mode, 1151 enum al_eth_lm_link_mode *new_mode) 1152 { 1153 int err; 1154 struct al_eth_link_status status; 1155 1156 al_assert(lm_context != NULL); 1157 al_assert(old_mode != NULL); 1158 al_assert(new_mode != NULL); 1159 1160 /** 1161 * if Link management is disabled, report no link fault in case the link was up 1162 * before and set new mode to disconnected to avoid calling to link establish 1163 * if the link wasn't up. 1164 */ 1165 if (lm_context->lm_pause != NULL) { 1166 boolean_t lm_pause = lm_context->lm_pause(lm_context->i2c_context); 1167 if (lm_pause == TRUE) { 1168 *new_mode = AL_ETH_LM_MODE_DISCONNECTED; 1169 if (link_fault != NULL) { 1170 if (lm_context->link_state == AL_ETH_LM_LINK_UP) 1171 *link_fault = FALSE; 1172 else 1173 *link_fault = TRUE; 1174 } 1175 1176 return 0; 1177 } 1178 } 1179 1180 *old_mode = lm_context->mode; 1181 *new_mode = lm_context->mode; 1182 1183 if (link_fault != NULL) 1184 *link_fault = TRUE; 1185 1186 switch (lm_context->link_state) { 1187 case AL_ETH_LM_LINK_UP: 1188 al_eth_link_status_get(lm_context->adapter, &status); 1189 1190 if (status.link_up) { 1191 if (link_fault != NULL) 1192 *link_fault = FALSE; 1193 1194 al_eth_led_set(lm_context->adapter, TRUE); 1195 1196 return (0); 1197 } else if (status.local_fault) { 1198 lm_debug("%s: >>>> Link state UP ==> DOWN\n", __func__); 1199 lm_context->link_state = AL_ETH_LM_LINK_DOWN; 1200 } else { 1201 lm_debug("%s: >>>> Link state UP ==> DOWN_RF\n", __func__); 1202 lm_context->link_state = AL_ETH_LM_LINK_DOWN_RF; 1203 } 1204 1205 break; 1206 case AL_ETH_LM_LINK_DOWN_RF: 1207 al_eth_link_status_get(lm_context->adapter, &status); 1208 1209 if (status.local_fault) { 1210 lm_debug("%s: >>>> Link state DOWN_RF ==> DOWN\n", __func__); 1211 lm_context->link_state = AL_ETH_LM_LINK_DOWN; 1212 1213 break; 1214 } else if (status.remote_fault == FALSE) { 1215 lm_debug("%s: >>>> Link state DOWN_RF ==> UP\n", __func__); 1216 lm_context->link_state = AL_ETH_LM_LINK_UP; 1217 } 1218 /* in case of remote fault only no need to check SFP again */ 1219 return (0); 1220 case AL_ETH_LM_LINK_DOWN: 1221 break; 1222 }; 1223 1224 al_eth_led_set(lm_context->adapter, FALSE); 1225 1226 if (lm_context->sfp_detection) { 1227 err = al_eth_module_detect(lm_context, new_mode); 1228 if (err != 0) { 1229 al_err("module_detection failed!\n"); 1230 return (err); 1231 } 1232 1233 lm_context->mode = *new_mode; 1234 } else { 1235 lm_context->mode = lm_context->default_mode; 1236 *new_mode = lm_context->mode; 1237 } 1238 1239 if (*old_mode != *new_mode) { 1240 al_info("%s: New SFP mode detected %s -> %s\n", 1241 __func__, al_eth_lm_mode_convert_to_str(*old_mode), 1242 al_eth_lm_mode_convert_to_str(*new_mode)); 1243 1244 lm_context->rx_param_dirty = 1; 1245 lm_context->tx_param_dirty = 1; 1246 1247 lm_context->new_port = TRUE; 1248 1249 if ((*new_mode != AL_ETH_LM_MODE_DISCONNECTED) && (lm_context->led_config)) { 1250 struct al_eth_lm_led_config_data data = {0}; 1251 1252 switch (*new_mode) { 1253 case AL_ETH_LM_MODE_10G_OPTIC: 1254 case AL_ETH_LM_MODE_10G_DA: 1255 data.speed = AL_ETH_LM_LED_CONFIG_10G; 1256 break; 1257 case AL_ETH_LM_MODE_1G: 1258 data.speed = AL_ETH_LM_LED_CONFIG_1G; 1259 break; 1260 case AL_ETH_LM_MODE_25G: 1261 data.speed = AL_ETH_LM_LED_CONFIG_25G; 1262 break; 1263 default: 1264 al_err("%s: unknown LM mode!\n", __func__); 1265 }; 1266 1267 lm_context->led_config(lm_context->i2c_context, &data); 1268 } 1269 } 1270 1271 return (0); 1272 } 1273 1274 int 1275 al_eth_lm_link_establish(struct al_eth_lm_context *lm_context, boolean_t *link_up) 1276 { 1277 boolean_t signal_detected; 1278 int ret = 0; 1279 1280 switch (lm_context->link_state) { 1281 case AL_ETH_LM_LINK_UP: 1282 *link_up = TRUE; 1283 lm_debug("%s: return link up\n", __func__); 1284 1285 return (0); 1286 case AL_ETH_LM_LINK_DOWN_RF: 1287 *link_up = FALSE; 1288 lm_debug("%s: return link down (DOWN_RF)\n", __func__); 1289 1290 return (0); 1291 case AL_ETH_LM_LINK_DOWN: 1292 break; 1293 }; 1294 1295 /** 1296 * At this point we will get LM disable only if changed to disable after link detection 1297 * finished. in this case link will not be established until LM will be enable again. 1298 */ 1299 if (lm_context->lm_pause) { 1300 boolean_t lm_pause = lm_context->lm_pause(lm_context->i2c_context); 1301 if (lm_pause == TRUE) { 1302 *link_up = FALSE; 1303 1304 return (0); 1305 } 1306 } 1307 1308 if ((lm_context->new_port) && (lm_context->retimer_exist)) { 1309 al_eth_serdes_static_rx_params_set(lm_context); 1310 al_eth_serdes_static_tx_params_set(lm_context); 1311 #if 0 1312 al_eth_lm_retimer_config(lm_context); 1313 DELAY(AL_ETH_LM_RETIMER_LINK_STATUS_DELAY); 1314 #endif 1315 1316 if (retimer[lm_context->retimer_type].config(lm_context)) { 1317 al_info("%s: failed to configure the retimer\n", __func__); 1318 1319 *link_up = FALSE; 1320 return (1); 1321 } 1322 1323 lm_context->new_port = FALSE; 1324 1325 DELAY(1000); 1326 } 1327 1328 if (lm_context->retimer_exist) { 1329 if (retimer[lm_context->retimer_type].rx_adaptation) { 1330 ret = retimer[lm_context->retimer_type].rx_adaptation(lm_context); 1331 1332 if (ret != 0) { 1333 lm_debug("retimer rx is not ready\n"); 1334 *link_up = FALSE; 1335 1336 return (0); 1337 } 1338 } 1339 } 1340 1341 signal_detected = lm_context->serdes_obj->signal_is_detected( 1342 lm_context->serdes_obj, 1343 lm_context->lane); 1344 1345 if (signal_detected == FALSE) { 1346 /* if no signal detected there is nothing to do */ 1347 lm_debug("serdes signal is down\n"); 1348 *link_up = AL_FALSE; 1349 return 0; 1350 } 1351 1352 if (lm_context->serdes_obj->type_get() == AL_SRDS_TYPE_25G) { 1353 lm_debug("%s: serdes 25G - perform rx gearbox reset\n", __func__); 1354 al_eth_gearbox_reset(lm_context->adapter, FALSE, TRUE); 1355 DELAY(AL_ETH_LM_GEARBOX_RESET_DELAY); 1356 } 1357 1358 if (lm_context->retimer_exist) { 1359 DELAY(AL_ETH_LM_RETIMER_LINK_STATUS_DELAY); 1360 1361 ret = al_eth_lm_check_for_link(lm_context, link_up); 1362 1363 if (ret == 0) { 1364 lm_debug("%s: link is up with retimer\n", __func__); 1365 return 0; 1366 } 1367 1368 return ret; 1369 } 1370 1371 if ((lm_context->mode == AL_ETH_LM_MODE_10G_DA) && (lm_context->link_training)) { 1372 lm_context->local_adv.transmitted_nonce = lm_context->get_random_byte(); 1373 lm_context->local_adv.transmitted_nonce &= 0x1f; 1374 1375 ret = al_eth_an_lt_execute(lm_context->adapter, 1376 lm_context->serdes_obj, 1377 lm_context->lane, 1378 &lm_context->local_adv, 1379 &lm_context->partner_adv); 1380 1381 lm_context->rx_param_dirty = 1; 1382 lm_context->tx_param_dirty = 1; 1383 1384 if (ret == 0) { 1385 al_info("%s: link training finished successfully\n", __func__); 1386 lm_context->link_training_failures = 0; 1387 ret = al_eth_lm_check_for_link(lm_context, link_up); 1388 1389 if (ret == 0) { 1390 lm_debug("%s: link is up with LT\n", __func__); 1391 return (0); 1392 } 1393 } 1394 1395 lm_context->link_training_failures++; 1396 if (lm_context->link_training_failures > AL_ETH_LT_FAILURES_TO_RESET) { 1397 lm_debug("%s: failed to establish LT %d times. reset serdes\n", 1398 __func__, AL_ETH_LT_FAILURES_TO_RESET); 1399 1400 lm_context->serdes_obj->pma_hard_reset_lane( 1401 lm_context->serdes_obj, 1402 lm_context->lane, 1403 TRUE); 1404 lm_context->serdes_obj->pma_hard_reset_lane( 1405 lm_context->serdes_obj, 1406 lm_context->lane, 1407 FALSE); 1408 lm_context->link_training_failures = 0; 1409 } 1410 } 1411 1412 al_eth_serdes_static_tx_params_set(lm_context); 1413 1414 if ((lm_context->mode == AL_ETH_LM_MODE_10G_DA) && 1415 (lm_context->rx_equal)) { 1416 ret = al_eth_rx_equal_run(lm_context); 1417 1418 if (ret == 0) { 1419 DELAY(AL_ETH_LM_LINK_STATUS_DELAY); 1420 ret = al_eth_lm_check_for_link(lm_context, link_up); 1421 1422 if (ret == 0) { 1423 lm_debug("%s: link is up with Rx Equalization\n", __func__); 1424 return (0); 1425 } 1426 } 1427 } 1428 1429 al_eth_serdes_static_rx_params_set(lm_context); 1430 1431 DELAY(AL_ETH_LM_LINK_STATUS_DELAY); 1432 1433 ret = al_eth_lm_check_for_link(lm_context, link_up); 1434 1435 if (ret == 0) { 1436 lm_debug("%s: link is up with static parameters\n", __func__); 1437 return (0); 1438 } 1439 1440 *link_up = FALSE; 1441 return (1); 1442 } 1443 1444 int 1445 al_eth_lm_static_parameters_override(struct al_eth_lm_context *lm_context, 1446 struct al_serdes_adv_tx_params *tx_params, 1447 struct al_serdes_adv_rx_params *rx_params) 1448 { 1449 1450 if (tx_params != NULL) { 1451 lm_context->tx_params_override = *tx_params; 1452 lm_context->tx_param_dirty = 1; 1453 lm_context->serdes_tx_params_valid = TRUE; 1454 } 1455 1456 if (rx_params != NULL) { 1457 lm_context->rx_params_override = *rx_params; 1458 lm_context->rx_param_dirty = 1; 1459 lm_context->serdes_rx_params_valid = TRUE; 1460 } 1461 1462 return (0); 1463 } 1464 1465 int 1466 al_eth_lm_static_parameters_override_disable(struct al_eth_lm_context *lm_context, 1467 boolean_t tx_params, boolean_t rx_params) 1468 { 1469 1470 if (tx_params != 0) 1471 lm_context->serdes_tx_params_valid = FALSE; 1472 if (rx_params != 0) 1473 lm_context->serdes_tx_params_valid = FALSE; 1474 1475 return (0); 1476 } 1477 1478 int 1479 al_eth_lm_static_parameters_get(struct al_eth_lm_context *lm_context, 1480 struct al_serdes_adv_tx_params *tx_params, 1481 struct al_serdes_adv_rx_params *rx_params) 1482 { 1483 1484 if (tx_params != NULL) { 1485 if (lm_context->serdes_tx_params_valid) 1486 *tx_params = lm_context->tx_params_override; 1487 else 1488 lm_context->serdes_obj->tx_advanced_params_get( 1489 lm_context->serdes_obj, 1490 lm_context->lane, 1491 tx_params); 1492 } 1493 1494 if (rx_params != NULL) { 1495 if (lm_context->serdes_rx_params_valid) 1496 *rx_params = lm_context->rx_params_override; 1497 else 1498 lm_context->serdes_obj->rx_advanced_params_get( 1499 lm_context->serdes_obj, 1500 lm_context->lane, 1501 rx_params); 1502 } 1503 1504 return (0); 1505 } 1506 1507 const char * 1508 al_eth_lm_mode_convert_to_str(enum al_eth_lm_link_mode val) 1509 { 1510 1511 switch (val) { 1512 case AL_ETH_LM_MODE_DISCONNECTED: 1513 return ("AL_ETH_LM_MODE_DISCONNECTED"); 1514 case AL_ETH_LM_MODE_10G_OPTIC: 1515 return ("AL_ETH_LM_MODE_10G_OPTIC"); 1516 case AL_ETH_LM_MODE_10G_DA: 1517 return ("AL_ETH_LM_MODE_10G_DA"); 1518 case AL_ETH_LM_MODE_1G: 1519 return ("AL_ETH_LM_MODE_1G"); 1520 case AL_ETH_LM_MODE_25G: 1521 return ("AL_ETH_LM_MODE_25G"); 1522 } 1523 1524 return ("N/A"); 1525 } 1526 1527 void 1528 al_eth_lm_debug_mode_set(struct al_eth_lm_context *lm_context, 1529 boolean_t enable) 1530 { 1531 1532 lm_context->debug = enable; 1533 } 1534