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 ((offset > EFX_PHY_MEDIA_INFO_MAX_OFFSET) ||
301 ((offset + len) > EFX_PHY_MEDIA_INFO_MAX_OFFSET)) {
306 if ((rc = efx_mcdi_phy_module_get_info(enp, dev_addr,
307 offset, len, data)) != 0)
315 EFSYS_PROBE1(fail1, efx_rc_t, rc);
320 __checkReturn efx_rc_t
321 efx_phy_fec_type_get(
323 __out efx_phy_fec_type_t *typep)
325 efx_port_t *epp = &(enp->en_port);
326 const efx_phy_ops_t *epop = epp->ep_epop;
329 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
331 if (epop->epo_fec_type_get == NULL) {
336 if ((rc = epop->epo_fec_type_get(enp, typep)) != 0)
344 EFSYS_PROBE1(fail1, efx_rc_t, rc);
349 #if EFSYS_OPT_PHY_STATS
353 /* START MKCONFIG GENERATED PhyStatNamesBlock af9ffa24da3bc100 */
354 static const char * const __efx_phy_stat_name[] = {
403 /* END MKCONFIG GENERATED PhyStatNamesBlock */
408 __in efx_phy_stat_t type)
410 _NOTE(ARGUNUSED(enp))
411 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
412 EFSYS_ASSERT3U(type, <, EFX_PHY_NSTATS);
414 return (__efx_phy_stat_name[type]);
417 #endif /* EFSYS_OPT_NAMES */
419 __checkReturn efx_rc_t
420 efx_phy_stats_update(
422 __in efsys_mem_t *esmp,
423 __inout_ecount(EFX_PHY_NSTATS) uint32_t *stat)
425 efx_port_t *epp = &(enp->en_port);
426 const efx_phy_ops_t *epop = epp->ep_epop;
428 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
429 EFSYS_ASSERT3U(enp->en_mod_flags, &, EFX_MOD_PORT);
431 return (epop->epo_stats_update(enp, esmp, stat));
434 #endif /* EFSYS_OPT_PHY_STATS */
439 __checkReturn efx_rc_t
440 efx_bist_enable_offline(
443 efx_port_t *epp = &(enp->en_port);
444 const efx_phy_ops_t *epop = epp->ep_epop;
447 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
449 if (epop->epo_bist_enable_offline == NULL) {
454 if ((rc = epop->epo_bist_enable_offline(enp)) != 0)
462 EFSYS_PROBE1(fail1, efx_rc_t, rc);
468 __checkReturn efx_rc_t
471 __in efx_bist_type_t type)
473 efx_port_t *epp = &(enp->en_port);
474 const efx_phy_ops_t *epop = epp->ep_epop;
477 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
479 EFSYS_ASSERT3U(type, !=, EFX_BIST_TYPE_UNKNOWN);
480 EFSYS_ASSERT3U(type, <, EFX_BIST_TYPE_NTYPES);
481 EFSYS_ASSERT3U(epp->ep_current_bist, ==, EFX_BIST_TYPE_UNKNOWN);
483 if (epop->epo_bist_start == NULL) {
488 if ((rc = epop->epo_bist_start(enp, type)) != 0)
491 epp->ep_current_bist = type;
498 EFSYS_PROBE1(fail1, efx_rc_t, rc);
503 __checkReturn efx_rc_t
506 __in efx_bist_type_t type,
507 __out efx_bist_result_t *resultp,
508 __out_opt uint32_t *value_maskp,
509 __out_ecount_opt(count) unsigned long *valuesp,
512 efx_port_t *epp = &(enp->en_port);
513 const efx_phy_ops_t *epop = epp->ep_epop;
516 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
518 EFSYS_ASSERT3U(type, !=, EFX_BIST_TYPE_UNKNOWN);
519 EFSYS_ASSERT3U(type, <, EFX_BIST_TYPE_NTYPES);
520 EFSYS_ASSERT3U(epp->ep_current_bist, ==, type);
522 EFSYS_ASSERT(epop->epo_bist_poll != NULL);
523 if (epop->epo_bist_poll == NULL) {
528 if ((rc = epop->epo_bist_poll(enp, type, resultp, value_maskp,
529 valuesp, count)) != 0)
537 EFSYS_PROBE1(fail1, efx_rc_t, rc);
545 __in efx_bist_type_t type)
547 efx_port_t *epp = &(enp->en_port);
548 const efx_phy_ops_t *epop = epp->ep_epop;
550 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
552 EFSYS_ASSERT3U(type, !=, EFX_BIST_TYPE_UNKNOWN);
553 EFSYS_ASSERT3U(type, <, EFX_BIST_TYPE_NTYPES);
554 EFSYS_ASSERT3U(epp->ep_current_bist, ==, type);
556 EFSYS_ASSERT(epop->epo_bist_stop != NULL);
558 if (epop->epo_bist_stop != NULL)
559 epop->epo_bist_stop(enp, type);
561 epp->ep_current_bist = EFX_BIST_TYPE_UNKNOWN;
564 #endif /* EFSYS_OPT_BIST */
569 efx_port_t *epp = &(enp->en_port);
571 EFSYS_ASSERT3U(enp->en_magic, ==, EFX_NIC_MAGIC);
575 epp->ep_adv_cap_mask = 0;
578 epp->ep_phy_type = 0;