1 /* SPDX-License-Identifier: BSD-3-Clause
3 * Copyright (c) 2007-2018 Solarflare Communications Inc.
12 static const efx_phy_ops_t __efx_phy_siena_ops = {
13 siena_phy_power, /* epo_power */
15 siena_phy_reconfigure, /* epo_reconfigure */
16 siena_phy_verify, /* epo_verify */
17 siena_phy_oui_get, /* epo_oui_get */
18 NULL, /* epo_fec_type_get */
19 #if EFSYS_OPT_PHY_STATS
20 siena_phy_stats_update, /* epo_stats_update */
21 #endif /* EFSYS_OPT_PHY_STATS */
23 NULL, /* epo_bist_enable_offline */
24 siena_phy_bist_start, /* epo_bist_start */
25 siena_phy_bist_poll, /* epo_bist_poll */
26 siena_phy_bist_stop, /* epo_bist_stop */
27 #endif /* EFSYS_OPT_BIST */
29 #endif /* EFSYS_OPT_SIENA */
31 #if EFSYS_OPT_HUNTINGTON || EFSYS_OPT_MEDFORD || EFSYS_OPT_MEDFORD2
32 static const efx_phy_ops_t __efx_phy_ef10_ops = {
33 ef10_phy_power, /* epo_power */
35 ef10_phy_reconfigure, /* epo_reconfigure */
36 ef10_phy_verify, /* epo_verify */
37 ef10_phy_oui_get, /* epo_oui_get */
38 ef10_phy_fec_type_get, /* epo_fec_type_get */
39 #if EFSYS_OPT_PHY_STATS
40 ef10_phy_stats_update, /* epo_stats_update */
41 #endif /* EFSYS_OPT_PHY_STATS */
43 ef10_bist_enable_offline, /* epo_bist_enable_offline */
44 ef10_bist_start, /* epo_bist_start */
45 ef10_bist_poll, /* epo_bist_poll */
46 ef10_bist_stop, /* epo_bist_stop */
47 #endif /* EFSYS_OPT_BIST */
49 #endif /* EFSYS_OPT_HUNTINGTON || EFSYS_OPT_MEDFORD || EFSYS_OPT_MEDFORD2 */
51 __checkReturn efx_rc_t
55 efx_port_t *epp = &(enp->en_port);
56 efx_nic_cfg_t *encp = &(enp->en_nic_cfg);
57 const efx_phy_ops_t *epop;
60 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
62 epp->ep_port = encp->enc_port;
63 epp->ep_phy_type = encp->enc_phy_type;
65 /* Hook in operations structure */
66 switch (enp->en_family) {
68 case EFX_FAMILY_SIENA:
69 epop = &__efx_phy_siena_ops;
71 #endif /* EFSYS_OPT_SIENA */
73 #if EFSYS_OPT_HUNTINGTON
74 case EFX_FAMILY_HUNTINGTON:
75 epop = &__efx_phy_ef10_ops;
77 #endif /* EFSYS_OPT_HUNTINGTON */
80 case EFX_FAMILY_MEDFORD:
81 epop = &__efx_phy_ef10_ops;
83 #endif /* EFSYS_OPT_MEDFORD */
85 #if EFSYS_OPT_MEDFORD2
86 case EFX_FAMILY_MEDFORD2:
87 epop = &__efx_phy_ef10_ops;
89 #endif /* EFSYS_OPT_MEDFORD2 */
101 EFSYS_PROBE1(fail1, efx_rc_t, rc);
104 epp->ep_phy_type = 0;
109 __checkReturn efx_rc_t
113 efx_port_t *epp = &(enp->en_port);
114 const efx_phy_ops_t *epop = epp->ep_epop;
116 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
117 EFSYS_ASSERT3U(enp->en_mod_flags, &, EFX_MOD_PORT);
119 return (epop->epo_verify(enp));
122 #if EFSYS_OPT_PHY_LED_CONTROL
124 __checkReturn efx_rc_t
127 __in efx_phy_led_mode_t mode)
129 efx_nic_cfg_t *encp = (&enp->en_nic_cfg);
130 efx_port_t *epp = &(enp->en_port);
131 const efx_phy_ops_t *epop = epp->ep_epop;
135 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
136 EFSYS_ASSERT3U(enp->en_mod_flags, &, EFX_MOD_PORT);
138 if (epp->ep_phy_led_mode == mode)
141 mask = (1 << EFX_PHY_LED_DEFAULT);
142 mask |= encp->enc_led_mask;
144 if (!((1 << mode) & mask)) {
149 EFSYS_ASSERT3U(mode, <, EFX_PHY_LED_NMODES);
150 epp->ep_phy_led_mode = mode;
152 if ((rc = epop->epo_reconfigure(enp)) != 0)
161 EFSYS_PROBE1(fail1, efx_rc_t, rc);
165 #endif /* EFSYS_OPT_PHY_LED_CONTROL */
171 __out uint32_t *maskp)
173 efx_port_t *epp = &(enp->en_port);
175 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
176 EFSYS_ASSERT3U(enp->en_mod_flags, &, EFX_MOD_PROBE);
179 case EFX_PHY_CAP_CURRENT:
180 *maskp = epp->ep_adv_cap_mask;
182 case EFX_PHY_CAP_DEFAULT:
183 *maskp = epp->ep_default_adv_cap_mask;
185 case EFX_PHY_CAP_PERM:
186 *maskp = epp->ep_phy_cap_mask;
189 EFSYS_ASSERT(B_FALSE);
195 __checkReturn efx_rc_t
200 efx_port_t *epp = &(enp->en_port);
201 const efx_phy_ops_t *epop = epp->ep_epop;
205 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
206 EFSYS_ASSERT3U(enp->en_mod_flags, &, EFX_MOD_PORT);
208 if ((mask & ~epp->ep_phy_cap_mask) != 0) {
213 if (epp->ep_adv_cap_mask == mask)
216 old_mask = epp->ep_adv_cap_mask;
217 epp->ep_adv_cap_mask = mask;
219 if ((rc = epop->epo_reconfigure(enp)) != 0)
228 epp->ep_adv_cap_mask = old_mask;
229 /* Reconfigure for robustness */
230 if (epop->epo_reconfigure(enp) != 0) {
232 * We may have an inconsistent view of our advertised speed
239 EFSYS_PROBE1(fail1, efx_rc_t, rc);
247 __out uint32_t *maskp)
249 efx_port_t *epp = &(enp->en_port);
251 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
252 EFSYS_ASSERT3U(enp->en_mod_flags, &, EFX_MOD_PORT);
254 *maskp = epp->ep_lp_cap_mask;
257 __checkReturn efx_rc_t
260 __out uint32_t *ouip)
262 efx_port_t *epp = &(enp->en_port);
263 const efx_phy_ops_t *epop = epp->ep_epop;
265 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
266 EFSYS_ASSERT3U(enp->en_mod_flags, &, EFX_MOD_PORT);
268 return (epop->epo_oui_get(enp, ouip));
272 efx_phy_media_type_get(
274 __out efx_phy_media_type_t *typep)
276 efx_port_t *epp = &(enp->en_port);
278 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
279 EFSYS_ASSERT3U(enp->en_mod_flags, &, EFX_MOD_PORT);
281 if (epp->ep_module_type != EFX_PHY_MEDIA_INVALID)
282 *typep = epp->ep_module_type;
284 *typep = epp->ep_fixed_port_type;
287 __checkReturn efx_rc_t
288 efx_phy_module_get_info(
290 __in uint8_t dev_addr,
293 __out_bcount(len) uint8_t *data)
297 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
298 EFSYS_ASSERT(data != NULL);
300 if ((uint32_t)offset + len > 0xff) {
305 if ((rc = efx_mcdi_phy_module_get_info(enp, dev_addr,
306 offset, len, data)) != 0)
314 EFSYS_PROBE1(fail1, efx_rc_t, rc);
319 __checkReturn efx_rc_t
320 efx_phy_fec_type_get(
322 __out efx_phy_fec_type_t *typep)
324 efx_port_t *epp = &(enp->en_port);
325 const efx_phy_ops_t *epop = epp->ep_epop;
328 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
330 if (epop->epo_fec_type_get == NULL) {
335 if ((rc = epop->epo_fec_type_get(enp, typep)) != 0)
343 EFSYS_PROBE1(fail1, efx_rc_t, rc);
348 #if EFSYS_OPT_PHY_STATS
352 /* START MKCONFIG GENERATED PhyStatNamesBlock af9ffa24da3bc100 */
353 static const char * const __efx_phy_stat_name[] = {
402 /* END MKCONFIG GENERATED PhyStatNamesBlock */
407 __in efx_phy_stat_t type)
409 _NOTE(ARGUNUSED(enp))
410 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
411 EFSYS_ASSERT3U(type, <, EFX_PHY_NSTATS);
413 return (__efx_phy_stat_name[type]);
416 #endif /* EFSYS_OPT_NAMES */
418 __checkReturn efx_rc_t
419 efx_phy_stats_update(
421 __in efsys_mem_t *esmp,
422 __inout_ecount(EFX_PHY_NSTATS) uint32_t *stat)
424 efx_port_t *epp = &(enp->en_port);
425 const efx_phy_ops_t *epop = epp->ep_epop;
427 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
428 EFSYS_ASSERT3U(enp->en_mod_flags, &, EFX_MOD_PORT);
430 return (epop->epo_stats_update(enp, esmp, stat));
433 #endif /* EFSYS_OPT_PHY_STATS */
438 __checkReturn efx_rc_t
439 efx_bist_enable_offline(
442 efx_port_t *epp = &(enp->en_port);
443 const efx_phy_ops_t *epop = epp->ep_epop;
446 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
448 if (epop->epo_bist_enable_offline == NULL) {
453 if ((rc = epop->epo_bist_enable_offline(enp)) != 0)
461 EFSYS_PROBE1(fail1, efx_rc_t, rc);
467 __checkReturn efx_rc_t
470 __in efx_bist_type_t type)
472 efx_port_t *epp = &(enp->en_port);
473 const efx_phy_ops_t *epop = epp->ep_epop;
476 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
478 EFSYS_ASSERT3U(type, !=, EFX_BIST_TYPE_UNKNOWN);
479 EFSYS_ASSERT3U(type, <, EFX_BIST_TYPE_NTYPES);
480 EFSYS_ASSERT3U(epp->ep_current_bist, ==, EFX_BIST_TYPE_UNKNOWN);
482 if (epop->epo_bist_start == NULL) {
487 if ((rc = epop->epo_bist_start(enp, type)) != 0)
490 epp->ep_current_bist = type;
497 EFSYS_PROBE1(fail1, efx_rc_t, rc);
502 __checkReturn efx_rc_t
505 __in efx_bist_type_t type,
506 __out efx_bist_result_t *resultp,
507 __out_opt uint32_t *value_maskp,
508 __out_ecount_opt(count) unsigned long *valuesp,
511 efx_port_t *epp = &(enp->en_port);
512 const efx_phy_ops_t *epop = epp->ep_epop;
515 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
517 EFSYS_ASSERT3U(type, !=, EFX_BIST_TYPE_UNKNOWN);
518 EFSYS_ASSERT3U(type, <, EFX_BIST_TYPE_NTYPES);
519 EFSYS_ASSERT3U(epp->ep_current_bist, ==, type);
521 EFSYS_ASSERT(epop->epo_bist_poll != NULL);
522 if (epop->epo_bist_poll == NULL) {
527 if ((rc = epop->epo_bist_poll(enp, type, resultp, value_maskp,
528 valuesp, count)) != 0)
536 EFSYS_PROBE1(fail1, efx_rc_t, rc);
544 __in efx_bist_type_t type)
546 efx_port_t *epp = &(enp->en_port);
547 const efx_phy_ops_t *epop = epp->ep_epop;
549 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
551 EFSYS_ASSERT3U(type, !=, EFX_BIST_TYPE_UNKNOWN);
552 EFSYS_ASSERT3U(type, <, EFX_BIST_TYPE_NTYPES);
553 EFSYS_ASSERT3U(epp->ep_current_bist, ==, type);
555 EFSYS_ASSERT(epop->epo_bist_stop != NULL);
557 if (epop->epo_bist_stop != NULL)
558 epop->epo_bist_stop(enp, type);
560 epp->ep_current_bist = EFX_BIST_TYPE_UNKNOWN;
563 #endif /* EFSYS_OPT_BIST */
568 efx_port_t *epp = &(enp->en_port);
570 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
574 epp->ep_adv_cap_mask = 0;
577 epp->ep_phy_type = 0;