xref: /dpdk/drivers/common/sfc_efx/base/ef10_phy.c (revision 80e33a289e4430e4d7b1356dbfda99de88db5013)
15e111ed8SAndrew Rybchenko /* SPDX-License-Identifier: BSD-3-Clause
25e111ed8SAndrew Rybchenko  *
3672386c1SAndrew Rybchenko  * Copyright(c) 2019-2021 Xilinx, Inc.
45e111ed8SAndrew Rybchenko  * Copyright(c) 2012-2019 Solarflare Communications Inc.
55e111ed8SAndrew Rybchenko  */
65e111ed8SAndrew Rybchenko 
75e111ed8SAndrew Rybchenko #include "efx.h"
85e111ed8SAndrew Rybchenko #include "efx_impl.h"
95e111ed8SAndrew Rybchenko 
103ac211cfSAndrew Rybchenko #if EFSYS_OPT_RIVERHEAD || EFX_OPTS_EF10()
115e111ed8SAndrew Rybchenko 
125e111ed8SAndrew Rybchenko static			void
mcdi_phy_decode_cap(__in uint32_t mcdi_cap,__out uint32_t * maskp)135e111ed8SAndrew Rybchenko mcdi_phy_decode_cap(
145e111ed8SAndrew Rybchenko 	__in		uint32_t mcdi_cap,
155e111ed8SAndrew Rybchenko 	__out		uint32_t *maskp)
165e111ed8SAndrew Rybchenko {
175e111ed8SAndrew Rybchenko 	uint32_t mask;
185e111ed8SAndrew Rybchenko 
195e111ed8SAndrew Rybchenko #define	CHECK_CAP(_cap) \
205e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(EFX_PHY_CAP_##_cap == MC_CMD_PHY_CAP_##_cap##_LBN)
215e111ed8SAndrew Rybchenko 
225e111ed8SAndrew Rybchenko 	CHECK_CAP(10HDX);
235e111ed8SAndrew Rybchenko 	CHECK_CAP(10FDX);
245e111ed8SAndrew Rybchenko 	CHECK_CAP(100HDX);
255e111ed8SAndrew Rybchenko 	CHECK_CAP(100FDX);
265e111ed8SAndrew Rybchenko 	CHECK_CAP(1000HDX);
275e111ed8SAndrew Rybchenko 	CHECK_CAP(1000FDX);
285e111ed8SAndrew Rybchenko 	CHECK_CAP(10000FDX);
295e111ed8SAndrew Rybchenko 	CHECK_CAP(25000FDX);
305e111ed8SAndrew Rybchenko 	CHECK_CAP(40000FDX);
315e111ed8SAndrew Rybchenko 	CHECK_CAP(50000FDX);
325e111ed8SAndrew Rybchenko 	CHECK_CAP(100000FDX);
335e111ed8SAndrew Rybchenko 	CHECK_CAP(PAUSE);
345e111ed8SAndrew Rybchenko 	CHECK_CAP(ASYM);
355e111ed8SAndrew Rybchenko 	CHECK_CAP(AN);
365e111ed8SAndrew Rybchenko 	CHECK_CAP(DDM);
375e111ed8SAndrew Rybchenko 	CHECK_CAP(BASER_FEC);
385e111ed8SAndrew Rybchenko 	CHECK_CAP(BASER_FEC_REQUESTED);
395e111ed8SAndrew Rybchenko 	CHECK_CAP(RS_FEC);
405e111ed8SAndrew Rybchenko 	CHECK_CAP(RS_FEC_REQUESTED);
415e111ed8SAndrew Rybchenko 	CHECK_CAP(25G_BASER_FEC);
425e111ed8SAndrew Rybchenko 	CHECK_CAP(25G_BASER_FEC_REQUESTED);
435e111ed8SAndrew Rybchenko #undef CHECK_CAP
445e111ed8SAndrew Rybchenko 
455e111ed8SAndrew Rybchenko 	mask = 0;
465e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_10HDX_LBN))
475e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_10HDX);
485e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_10FDX_LBN))
495e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_10FDX);
505e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_100HDX_LBN))
515e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_100HDX);
525e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_100FDX_LBN))
535e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_100FDX);
545e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_1000HDX_LBN))
555e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_1000HDX);
565e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_1000FDX_LBN))
575e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_1000FDX);
585e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_10000FDX_LBN))
595e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_10000FDX);
605e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_25000FDX_LBN))
615e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_25000FDX);
625e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_40000FDX_LBN))
635e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_40000FDX);
645e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_50000FDX_LBN))
655e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_50000FDX);
665e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_100000FDX_LBN))
675e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_100000FDX);
685e111ed8SAndrew Rybchenko 
695e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_PAUSE_LBN))
705e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_PAUSE);
715e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_ASYM_LBN))
725e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_ASYM);
735e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_AN_LBN))
745e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_AN);
755e111ed8SAndrew Rybchenko 
765e111ed8SAndrew Rybchenko 	/* FEC caps (supported on Medford2 and later) */
775e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_BASER_FEC_LBN))
785e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_BASER_FEC);
795e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_BASER_FEC_REQUESTED_LBN))
805e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_BASER_FEC_REQUESTED);
815e111ed8SAndrew Rybchenko 
825e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_RS_FEC_LBN))
835e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_RS_FEC);
845e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_RS_FEC_REQUESTED_LBN))
855e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_RS_FEC_REQUESTED);
865e111ed8SAndrew Rybchenko 
875e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_25G_BASER_FEC_LBN))
885e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_25G_BASER_FEC);
895e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_25G_BASER_FEC_REQUESTED_LBN))
905e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_25G_BASER_FEC_REQUESTED);
915e111ed8SAndrew Rybchenko 
925e111ed8SAndrew Rybchenko 	*maskp = mask;
935e111ed8SAndrew Rybchenko }
945e111ed8SAndrew Rybchenko 
955e111ed8SAndrew Rybchenko static			void
mcdi_phy_decode_link_mode(__in efx_nic_t * enp,__in uint32_t link_flags,__in unsigned int speed,__in unsigned int fcntl,__in uint32_t fec,__out efx_link_mode_t * link_modep,__out unsigned int * fcntlp,__out efx_phy_fec_type_t * fecp)965e111ed8SAndrew Rybchenko mcdi_phy_decode_link_mode(
975e111ed8SAndrew Rybchenko 	__in		efx_nic_t *enp,
985e111ed8SAndrew Rybchenko 	__in		uint32_t link_flags,
995e111ed8SAndrew Rybchenko 	__in		unsigned int speed,
1005e111ed8SAndrew Rybchenko 	__in		unsigned int fcntl,
1015e111ed8SAndrew Rybchenko 	__in		uint32_t fec,
1025e111ed8SAndrew Rybchenko 	__out		efx_link_mode_t *link_modep,
1035e111ed8SAndrew Rybchenko 	__out		unsigned int *fcntlp,
1045e111ed8SAndrew Rybchenko 	__out		efx_phy_fec_type_t *fecp)
1055e111ed8SAndrew Rybchenko {
1065e111ed8SAndrew Rybchenko 	boolean_t fd = !!(link_flags &
1075e111ed8SAndrew Rybchenko 		    (1 << MC_CMD_GET_LINK_OUT_FULL_DUPLEX_LBN));
1085e111ed8SAndrew Rybchenko 	boolean_t up = !!(link_flags &
1095e111ed8SAndrew Rybchenko 		    (1 << MC_CMD_GET_LINK_OUT_LINK_UP_LBN));
1105e111ed8SAndrew Rybchenko 
1115e111ed8SAndrew Rybchenko 	_NOTE(ARGUNUSED(enp))
1125e111ed8SAndrew Rybchenko 
1135e111ed8SAndrew Rybchenko 	if (!up)
1145e111ed8SAndrew Rybchenko 		*link_modep = EFX_LINK_DOWN;
1155e111ed8SAndrew Rybchenko 	else if (speed == 100000 && fd)
1165e111ed8SAndrew Rybchenko 		*link_modep = EFX_LINK_100000FDX;
1175e111ed8SAndrew Rybchenko 	else if (speed == 50000 && fd)
1185e111ed8SAndrew Rybchenko 		*link_modep = EFX_LINK_50000FDX;
1195e111ed8SAndrew Rybchenko 	else if (speed == 40000 && fd)
1205e111ed8SAndrew Rybchenko 		*link_modep = EFX_LINK_40000FDX;
1215e111ed8SAndrew Rybchenko 	else if (speed == 25000 && fd)
1225e111ed8SAndrew Rybchenko 		*link_modep = EFX_LINK_25000FDX;
1235e111ed8SAndrew Rybchenko 	else if (speed == 10000 && fd)
1245e111ed8SAndrew Rybchenko 		*link_modep = EFX_LINK_10000FDX;
1255e111ed8SAndrew Rybchenko 	else if (speed == 1000)
1265e111ed8SAndrew Rybchenko 		*link_modep = fd ? EFX_LINK_1000FDX : EFX_LINK_1000HDX;
1275e111ed8SAndrew Rybchenko 	else if (speed == 100)
1285e111ed8SAndrew Rybchenko 		*link_modep = fd ? EFX_LINK_100FDX : EFX_LINK_100HDX;
1295e111ed8SAndrew Rybchenko 	else if (speed == 10)
1305e111ed8SAndrew Rybchenko 		*link_modep = fd ? EFX_LINK_10FDX : EFX_LINK_10HDX;
1315e111ed8SAndrew Rybchenko 	else
1325e111ed8SAndrew Rybchenko 		*link_modep = EFX_LINK_UNKNOWN;
1335e111ed8SAndrew Rybchenko 
1345e111ed8SAndrew Rybchenko 	if (fcntl == MC_CMD_FCNTL_OFF)
1355e111ed8SAndrew Rybchenko 		*fcntlp = 0;
1365e111ed8SAndrew Rybchenko 	else if (fcntl == MC_CMD_FCNTL_RESPOND)
1375e111ed8SAndrew Rybchenko 		*fcntlp = EFX_FCNTL_RESPOND;
1385e111ed8SAndrew Rybchenko 	else if (fcntl == MC_CMD_FCNTL_GENERATE)
1395e111ed8SAndrew Rybchenko 		*fcntlp = EFX_FCNTL_GENERATE;
1405e111ed8SAndrew Rybchenko 	else if (fcntl == MC_CMD_FCNTL_BIDIR)
1415e111ed8SAndrew Rybchenko 		*fcntlp = EFX_FCNTL_RESPOND | EFX_FCNTL_GENERATE;
1425e111ed8SAndrew Rybchenko 	else {
1435e111ed8SAndrew Rybchenko 		EFSYS_PROBE1(mc_pcol_error, int, fcntl);
1445e111ed8SAndrew Rybchenko 		*fcntlp = 0;
1455e111ed8SAndrew Rybchenko 	}
1465e111ed8SAndrew Rybchenko 
1475e111ed8SAndrew Rybchenko 	switch (fec) {
1485e111ed8SAndrew Rybchenko 	case MC_CMD_FEC_NONE:
1495e111ed8SAndrew Rybchenko 		*fecp = EFX_PHY_FEC_NONE;
1505e111ed8SAndrew Rybchenko 		break;
1515e111ed8SAndrew Rybchenko 	case MC_CMD_FEC_BASER:
1525e111ed8SAndrew Rybchenko 		*fecp = EFX_PHY_FEC_BASER;
1535e111ed8SAndrew Rybchenko 		break;
1545e111ed8SAndrew Rybchenko 	case MC_CMD_FEC_RS:
1555e111ed8SAndrew Rybchenko 		*fecp = EFX_PHY_FEC_RS;
1565e111ed8SAndrew Rybchenko 		break;
1575e111ed8SAndrew Rybchenko 	default:
1585e111ed8SAndrew Rybchenko 		EFSYS_PROBE1(mc_pcol_error, int, fec);
1595e111ed8SAndrew Rybchenko 		*fecp = EFX_PHY_FEC_NONE;
1605e111ed8SAndrew Rybchenko 		break;
1615e111ed8SAndrew Rybchenko 	}
1625e111ed8SAndrew Rybchenko }
1635e111ed8SAndrew Rybchenko 
1645e111ed8SAndrew Rybchenko 
1655e111ed8SAndrew Rybchenko 			void
ef10_phy_link_ev(__in efx_nic_t * enp,__in efx_qword_t * eqp,__in boolean_t ev_is_v2,__out efx_link_mode_t * link_modep)1665e111ed8SAndrew Rybchenko ef10_phy_link_ev(
1675e111ed8SAndrew Rybchenko 	__in		efx_nic_t *enp,
1685e111ed8SAndrew Rybchenko 	__in		efx_qword_t *eqp,
169*80e33a28SIvan Malov 	__in		boolean_t ev_is_v2,
1705e111ed8SAndrew Rybchenko 	__out		efx_link_mode_t *link_modep)
1715e111ed8SAndrew Rybchenko {
1725e111ed8SAndrew Rybchenko 	efx_port_t *epp = &(enp->en_port);
1735e111ed8SAndrew Rybchenko 	unsigned int link_flags;
1745e111ed8SAndrew Rybchenko 	unsigned int speed;
1755e111ed8SAndrew Rybchenko 	unsigned int fcntl;
1765e111ed8SAndrew Rybchenko 	efx_phy_fec_type_t fec = MC_CMD_FEC_NONE;
1775e111ed8SAndrew Rybchenko 	efx_link_mode_t link_mode;
178*80e33a28SIvan Malov 	unsigned int ev_lp_cap;
179*80e33a28SIvan Malov 	unsigned int ev_fcntl;
180*80e33a28SIvan Malov 	unsigned int ev_speed;
1815e111ed8SAndrew Rybchenko 	uint32_t lp_cap_mask;
1825e111ed8SAndrew Rybchenko 
183*80e33a28SIvan Malov 	if (ev_is_v2) {
184*80e33a28SIvan Malov 		link_flags = (1 << MC_CMD_GET_LINK_OUT_FULL_DUPLEX_LBN);
185*80e33a28SIvan Malov 		if (MCDI_EV_FIELD(eqp, LINKCHANGE_V2_FLAGS_LINK_UP))
186*80e33a28SIvan Malov 			link_flags |= (1 << MC_CMD_GET_LINK_OUT_LINK_UP_LBN);
187*80e33a28SIvan Malov 
188*80e33a28SIvan Malov 		ev_lp_cap = MCDI_EV_FIELD(eqp, LINKCHANGE_V2_LP_CAP);
189*80e33a28SIvan Malov 		ev_fcntl = MCDI_EV_FIELD(eqp, LINKCHANGE_V2_FCNTL);
190*80e33a28SIvan Malov 		ev_speed = MCDI_EV_FIELD(eqp, LINKCHANGE_V2_SPEED);
191*80e33a28SIvan Malov 	} else {
192*80e33a28SIvan Malov 		link_flags = MCDI_EV_FIELD(eqp, LINKCHANGE_LINK_FLAGS);
193*80e33a28SIvan Malov 		ev_lp_cap = MCDI_EV_FIELD(eqp, LINKCHANGE_LP_CAP);
194*80e33a28SIvan Malov 		ev_fcntl = MCDI_EV_FIELD(eqp, LINKCHANGE_FCNTL);
195*80e33a28SIvan Malov 		ev_speed = MCDI_EV_FIELD(eqp, LINKCHANGE_SPEED);
196*80e33a28SIvan Malov 	}
197*80e33a28SIvan Malov 
1985e111ed8SAndrew Rybchenko 	/*
1995e111ed8SAndrew Rybchenko 	 * Convert the LINKCHANGE speed enumeration into mbit/s, in the
2005e111ed8SAndrew Rybchenko 	 * same way as GET_LINK encodes the speed
2015e111ed8SAndrew Rybchenko 	 */
202*80e33a28SIvan Malov 	switch (ev_speed) {
2035e111ed8SAndrew Rybchenko 	case MCDI_EVENT_LINKCHANGE_SPEED_100M:
2045e111ed8SAndrew Rybchenko 		speed = 100;
2055e111ed8SAndrew Rybchenko 		break;
2065e111ed8SAndrew Rybchenko 	case MCDI_EVENT_LINKCHANGE_SPEED_1G:
2075e111ed8SAndrew Rybchenko 		speed = 1000;
2085e111ed8SAndrew Rybchenko 		break;
2095e111ed8SAndrew Rybchenko 	case MCDI_EVENT_LINKCHANGE_SPEED_10G:
2105e111ed8SAndrew Rybchenko 		speed = 10000;
2115e111ed8SAndrew Rybchenko 		break;
2125e111ed8SAndrew Rybchenko 	case MCDI_EVENT_LINKCHANGE_SPEED_25G:
2135e111ed8SAndrew Rybchenko 		speed = 25000;
2145e111ed8SAndrew Rybchenko 		break;
2155e111ed8SAndrew Rybchenko 	case MCDI_EVENT_LINKCHANGE_SPEED_40G:
2165e111ed8SAndrew Rybchenko 		speed = 40000;
2175e111ed8SAndrew Rybchenko 		break;
2185e111ed8SAndrew Rybchenko 	case MCDI_EVENT_LINKCHANGE_SPEED_50G:
2195e111ed8SAndrew Rybchenko 		speed = 50000;
2205e111ed8SAndrew Rybchenko 		break;
2215e111ed8SAndrew Rybchenko 	case MCDI_EVENT_LINKCHANGE_SPEED_100G:
2225e111ed8SAndrew Rybchenko 		speed = 100000;
2235e111ed8SAndrew Rybchenko 		break;
2245e111ed8SAndrew Rybchenko 	default:
2255e111ed8SAndrew Rybchenko 		speed = 0;
2265e111ed8SAndrew Rybchenko 		break;
2275e111ed8SAndrew Rybchenko 	}
2285e111ed8SAndrew Rybchenko 
229*80e33a28SIvan Malov 	mcdi_phy_decode_link_mode(enp, link_flags, speed, ev_fcntl,
2305e111ed8SAndrew Rybchenko 				    MC_CMD_FEC_NONE, &link_mode,
2315e111ed8SAndrew Rybchenko 				    &fcntl, &fec);
232*80e33a28SIvan Malov 	mcdi_phy_decode_cap(ev_lp_cap, &lp_cap_mask);
2335e111ed8SAndrew Rybchenko 
2345e111ed8SAndrew Rybchenko 	/*
2355e111ed8SAndrew Rybchenko 	 * It's safe to update ep_lp_cap_mask without the driver's port lock
2365e111ed8SAndrew Rybchenko 	 * because presumably any concurrently running efx_port_poll() is
2375e111ed8SAndrew Rybchenko 	 * only going to arrive at the same value.
2385e111ed8SAndrew Rybchenko 	 *
2395e111ed8SAndrew Rybchenko 	 * ep_fcntl has two meanings. It's either the link common fcntl
2405e111ed8SAndrew Rybchenko 	 * (if the PHY supports AN), or it's the forced link state. If
2415e111ed8SAndrew Rybchenko 	 * the former, it's safe to update the value for the same reason as
2425e111ed8SAndrew Rybchenko 	 * for ep_lp_cap_mask. If the latter, then just ignore the value,
2435e111ed8SAndrew Rybchenko 	 * because we can race with efx_mac_fcntl_set().
2445e111ed8SAndrew Rybchenko 	 */
2455e111ed8SAndrew Rybchenko 	epp->ep_lp_cap_mask = lp_cap_mask;
2465e111ed8SAndrew Rybchenko 	epp->ep_fcntl = fcntl;
2475e111ed8SAndrew Rybchenko 
2485e111ed8SAndrew Rybchenko 	*link_modep = link_mode;
2495e111ed8SAndrew Rybchenko }
2505e111ed8SAndrew Rybchenko 
2515e111ed8SAndrew Rybchenko 	__checkReturn	efx_rc_t
ef10_phy_power(__in efx_nic_t * enp,__in boolean_t power)2525e111ed8SAndrew Rybchenko ef10_phy_power(
2535e111ed8SAndrew Rybchenko 	__in		efx_nic_t *enp,
2545e111ed8SAndrew Rybchenko 	__in		boolean_t power)
2555e111ed8SAndrew Rybchenko {
2565e111ed8SAndrew Rybchenko 	efx_rc_t rc;
2575e111ed8SAndrew Rybchenko 
2585e111ed8SAndrew Rybchenko 	if (!power)
2595e111ed8SAndrew Rybchenko 		return (0);
2605e111ed8SAndrew Rybchenko 
2615e111ed8SAndrew Rybchenko 	/* Check if the PHY is a zombie */
2625e111ed8SAndrew Rybchenko 	if ((rc = ef10_phy_verify(enp)) != 0)
2635e111ed8SAndrew Rybchenko 		goto fail1;
2645e111ed8SAndrew Rybchenko 
2655e111ed8SAndrew Rybchenko 	enp->en_reset_flags |= EFX_RESET_PHY;
2665e111ed8SAndrew Rybchenko 
2675e111ed8SAndrew Rybchenko 	return (0);
2685e111ed8SAndrew Rybchenko 
2695e111ed8SAndrew Rybchenko fail1:
2705e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
2715e111ed8SAndrew Rybchenko 
2725e111ed8SAndrew Rybchenko 	return (rc);
2735e111ed8SAndrew Rybchenko }
2745e111ed8SAndrew Rybchenko 
2755e111ed8SAndrew Rybchenko 	__checkReturn	efx_rc_t
ef10_phy_get_link(__in efx_nic_t * enp,__out ef10_link_state_t * elsp)2765e111ed8SAndrew Rybchenko ef10_phy_get_link(
2775e111ed8SAndrew Rybchenko 	__in		efx_nic_t *enp,
2785e111ed8SAndrew Rybchenko 	__out		ef10_link_state_t *elsp)
2795e111ed8SAndrew Rybchenko {
2805e111ed8SAndrew Rybchenko 	efx_mcdi_req_t req;
2815e111ed8SAndrew Rybchenko 	uint32_t fec;
2825e111ed8SAndrew Rybchenko 	EFX_MCDI_DECLARE_BUF(payload, MC_CMD_GET_LINK_IN_LEN,
2835e111ed8SAndrew Rybchenko 		MC_CMD_GET_LINK_OUT_V2_LEN);
2845e111ed8SAndrew Rybchenko 	efx_rc_t rc;
2855e111ed8SAndrew Rybchenko 
2865e111ed8SAndrew Rybchenko 	req.emr_cmd = MC_CMD_GET_LINK;
2875e111ed8SAndrew Rybchenko 	req.emr_in_buf = payload;
2885e111ed8SAndrew Rybchenko 	req.emr_in_length = MC_CMD_GET_LINK_IN_LEN;
2895e111ed8SAndrew Rybchenko 	req.emr_out_buf = payload;
2905e111ed8SAndrew Rybchenko 	req.emr_out_length = MC_CMD_GET_LINK_OUT_V2_LEN;
2915e111ed8SAndrew Rybchenko 
2925e111ed8SAndrew Rybchenko 	efx_mcdi_execute(enp, &req);
2935e111ed8SAndrew Rybchenko 
2945e111ed8SAndrew Rybchenko 	if (req.emr_rc != 0) {
2955e111ed8SAndrew Rybchenko 		rc = req.emr_rc;
2965e111ed8SAndrew Rybchenko 		goto fail1;
2975e111ed8SAndrew Rybchenko 	}
2985e111ed8SAndrew Rybchenko 
2995e111ed8SAndrew Rybchenko 	if (req.emr_out_length_used < MC_CMD_GET_LINK_OUT_LEN) {
3005e111ed8SAndrew Rybchenko 		rc = EMSGSIZE;
3015e111ed8SAndrew Rybchenko 		goto fail2;
3025e111ed8SAndrew Rybchenko 	}
3035e111ed8SAndrew Rybchenko 
3045e111ed8SAndrew Rybchenko 	mcdi_phy_decode_cap(MCDI_OUT_DWORD(req, GET_LINK_OUT_CAP),
3055e111ed8SAndrew Rybchenko 			    &elsp->epls.epls_adv_cap_mask);
3065e111ed8SAndrew Rybchenko 	mcdi_phy_decode_cap(MCDI_OUT_DWORD(req, GET_LINK_OUT_LP_CAP),
3075e111ed8SAndrew Rybchenko 			    &elsp->epls.epls_lp_cap_mask);
3085e111ed8SAndrew Rybchenko 
3095e111ed8SAndrew Rybchenko 	if (req.emr_out_length_used < MC_CMD_GET_LINK_OUT_V2_LEN)
3105e111ed8SAndrew Rybchenko 		fec = MC_CMD_FEC_NONE;
3115e111ed8SAndrew Rybchenko 	else
3125e111ed8SAndrew Rybchenko 		fec = MCDI_OUT_DWORD(req, GET_LINK_OUT_V2_FEC_TYPE);
3135e111ed8SAndrew Rybchenko 
3145e111ed8SAndrew Rybchenko 	mcdi_phy_decode_link_mode(enp, MCDI_OUT_DWORD(req, GET_LINK_OUT_FLAGS),
3155e111ed8SAndrew Rybchenko 			    MCDI_OUT_DWORD(req, GET_LINK_OUT_LINK_SPEED),
3165e111ed8SAndrew Rybchenko 			    MCDI_OUT_DWORD(req, GET_LINK_OUT_FCNTL),
3175e111ed8SAndrew Rybchenko 			    fec, &elsp->epls.epls_link_mode,
3185e111ed8SAndrew Rybchenko 			    &elsp->epls.epls_fcntl, &elsp->epls.epls_fec);
3195e111ed8SAndrew Rybchenko 
3205e111ed8SAndrew Rybchenko 	if (req.emr_out_length_used < MC_CMD_GET_LINK_OUT_V2_LEN) {
3215e111ed8SAndrew Rybchenko 		elsp->epls.epls_ld_cap_mask = 0;
3225e111ed8SAndrew Rybchenko 	} else {
3235e111ed8SAndrew Rybchenko 		mcdi_phy_decode_cap(MCDI_OUT_DWORD(req, GET_LINK_OUT_V2_LD_CAP),
3245e111ed8SAndrew Rybchenko 				    &elsp->epls.epls_ld_cap_mask);
3255e111ed8SAndrew Rybchenko 	}
3265e111ed8SAndrew Rybchenko 
3275e111ed8SAndrew Rybchenko 
3285e111ed8SAndrew Rybchenko #if EFSYS_OPT_LOOPBACK
3295e111ed8SAndrew Rybchenko 	/*
3305e111ed8SAndrew Rybchenko 	 * MC_CMD_LOOPBACK and EFX_LOOPBACK names are equivalent, so use the
3315e111ed8SAndrew Rybchenko 	 * MCDI value directly. Agreement is checked in efx_loopback_mask().
3325e111ed8SAndrew Rybchenko 	 */
3335e111ed8SAndrew Rybchenko 	elsp->els_loopback = MCDI_OUT_DWORD(req, GET_LINK_OUT_LOOPBACK_MODE);
3345e111ed8SAndrew Rybchenko #endif	/* EFSYS_OPT_LOOPBACK */
3355e111ed8SAndrew Rybchenko 
3365e111ed8SAndrew Rybchenko 	elsp->els_mac_up = MCDI_OUT_DWORD(req, GET_LINK_OUT_MAC_FAULT) == 0;
3375e111ed8SAndrew Rybchenko 
3385e111ed8SAndrew Rybchenko 	return (0);
3395e111ed8SAndrew Rybchenko 
3405e111ed8SAndrew Rybchenko fail2:
3415e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail2);
3425e111ed8SAndrew Rybchenko fail1:
3435e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
3445e111ed8SAndrew Rybchenko 
3455e111ed8SAndrew Rybchenko 	return (rc);
3465e111ed8SAndrew Rybchenko }
3475e111ed8SAndrew Rybchenko 
348252b2aecSAndrew Rybchenko static	__checkReturn	efx_rc_t
efx_mcdi_phy_set_link(__in efx_nic_t * enp,__in uint32_t cap_mask,__in efx_loopback_type_t loopback_type,__in efx_link_mode_t loopback_link_mode,__in uint32_t phy_flags)349252b2aecSAndrew Rybchenko efx_mcdi_phy_set_link(
350252b2aecSAndrew Rybchenko 	__in		efx_nic_t *enp,
351252b2aecSAndrew Rybchenko 	__in		uint32_t cap_mask,
352252b2aecSAndrew Rybchenko 	__in		efx_loopback_type_t loopback_type,
353252b2aecSAndrew Rybchenko 	__in		efx_link_mode_t loopback_link_mode,
354252b2aecSAndrew Rybchenko 	__in		uint32_t phy_flags)
3555e111ed8SAndrew Rybchenko {
3565e111ed8SAndrew Rybchenko 	efx_mcdi_req_t req;
3575e111ed8SAndrew Rybchenko 	EFX_MCDI_DECLARE_BUF(payload, MC_CMD_SET_LINK_IN_LEN,
3585e111ed8SAndrew Rybchenko 		MC_CMD_SET_LINK_OUT_LEN);
3595e111ed8SAndrew Rybchenko 	unsigned int speed;
3605e111ed8SAndrew Rybchenko 	efx_rc_t rc;
3615e111ed8SAndrew Rybchenko 
3625e111ed8SAndrew Rybchenko 	req.emr_cmd = MC_CMD_SET_LINK;
3635e111ed8SAndrew Rybchenko 	req.emr_in_buf = payload;
3645e111ed8SAndrew Rybchenko 	req.emr_in_length = MC_CMD_SET_LINK_IN_LEN;
3655e111ed8SAndrew Rybchenko 	req.emr_out_buf = payload;
3665e111ed8SAndrew Rybchenko 	req.emr_out_length = MC_CMD_SET_LINK_OUT_LEN;
3675e111ed8SAndrew Rybchenko 
3685e111ed8SAndrew Rybchenko 	MCDI_IN_POPULATE_DWORD_10(req, SET_LINK_IN_CAP,
3695e111ed8SAndrew Rybchenko 		PHY_CAP_10HDX, (cap_mask >> EFX_PHY_CAP_10HDX) & 0x1,
3705e111ed8SAndrew Rybchenko 		PHY_CAP_10FDX, (cap_mask >> EFX_PHY_CAP_10FDX) & 0x1,
3715e111ed8SAndrew Rybchenko 		PHY_CAP_100HDX, (cap_mask >> EFX_PHY_CAP_100HDX) & 0x1,
3725e111ed8SAndrew Rybchenko 		PHY_CAP_100FDX, (cap_mask >> EFX_PHY_CAP_100FDX) & 0x1,
3735e111ed8SAndrew Rybchenko 		PHY_CAP_1000HDX, (cap_mask >> EFX_PHY_CAP_1000HDX) & 0x1,
3745e111ed8SAndrew Rybchenko 		PHY_CAP_1000FDX, (cap_mask >> EFX_PHY_CAP_1000FDX) & 0x1,
3755e111ed8SAndrew Rybchenko 		PHY_CAP_10000FDX, (cap_mask >> EFX_PHY_CAP_10000FDX) & 0x1,
3765e111ed8SAndrew Rybchenko 		PHY_CAP_PAUSE, (cap_mask >> EFX_PHY_CAP_PAUSE) & 0x1,
3775e111ed8SAndrew Rybchenko 		PHY_CAP_ASYM, (cap_mask >> EFX_PHY_CAP_ASYM) & 0x1,
3785e111ed8SAndrew Rybchenko 		PHY_CAP_AN, (cap_mask >> EFX_PHY_CAP_AN) & 0x1);
3795e111ed8SAndrew Rybchenko 	/* Too many fields for for POPULATE macros, so insert this afterwards */
3805e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD_FIELD(req, SET_LINK_IN_CAP,
3815e111ed8SAndrew Rybchenko 	    PHY_CAP_25000FDX, (cap_mask >> EFX_PHY_CAP_25000FDX) & 0x1);
3825e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD_FIELD(req, SET_LINK_IN_CAP,
3835e111ed8SAndrew Rybchenko 	    PHY_CAP_40000FDX, (cap_mask >> EFX_PHY_CAP_40000FDX) & 0x1);
3845e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD_FIELD(req, SET_LINK_IN_CAP,
3855e111ed8SAndrew Rybchenko 	    PHY_CAP_50000FDX, (cap_mask >> EFX_PHY_CAP_50000FDX) & 0x1);
3865e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD_FIELD(req, SET_LINK_IN_CAP,
3875e111ed8SAndrew Rybchenko 	    PHY_CAP_100000FDX, (cap_mask >> EFX_PHY_CAP_100000FDX) & 0x1);
3885e111ed8SAndrew Rybchenko 
3895e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD_FIELD(req, SET_LINK_IN_CAP,
3905e111ed8SAndrew Rybchenko 	    PHY_CAP_BASER_FEC, (cap_mask >> EFX_PHY_CAP_BASER_FEC) & 0x1);
3915e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD_FIELD(req, SET_LINK_IN_CAP,
3925e111ed8SAndrew Rybchenko 	    PHY_CAP_BASER_FEC_REQUESTED,
3935e111ed8SAndrew Rybchenko 	    (cap_mask >> EFX_PHY_CAP_BASER_FEC_REQUESTED) & 0x1);
3945e111ed8SAndrew Rybchenko 
3955e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD_FIELD(req, SET_LINK_IN_CAP,
3965e111ed8SAndrew Rybchenko 	    PHY_CAP_RS_FEC, (cap_mask >> EFX_PHY_CAP_RS_FEC) & 0x1);
3975e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD_FIELD(req, SET_LINK_IN_CAP,
3985e111ed8SAndrew Rybchenko 	    PHY_CAP_RS_FEC_REQUESTED,
3995e111ed8SAndrew Rybchenko 	    (cap_mask >> EFX_PHY_CAP_RS_FEC_REQUESTED) & 0x1);
4005e111ed8SAndrew Rybchenko 
4015e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD_FIELD(req, SET_LINK_IN_CAP,
4025e111ed8SAndrew Rybchenko 	    PHY_CAP_25G_BASER_FEC,
4035e111ed8SAndrew Rybchenko 	    (cap_mask >> EFX_PHY_CAP_25G_BASER_FEC) & 0x1);
4045e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD_FIELD(req, SET_LINK_IN_CAP,
4055e111ed8SAndrew Rybchenko 	    PHY_CAP_25G_BASER_FEC_REQUESTED,
4065e111ed8SAndrew Rybchenko 	    (cap_mask >> EFX_PHY_CAP_25G_BASER_FEC_REQUESTED) & 0x1);
4075e111ed8SAndrew Rybchenko 
408252b2aecSAndrew Rybchenko 	MCDI_IN_SET_DWORD(req, SET_LINK_IN_LOOPBACK_MODE, loopback_type);
409252b2aecSAndrew Rybchenko 
410252b2aecSAndrew Rybchenko 	switch (loopback_link_mode) {
4115e111ed8SAndrew Rybchenko 	case EFX_LINK_100FDX:
4125e111ed8SAndrew Rybchenko 		speed = 100;
4135e111ed8SAndrew Rybchenko 		break;
4145e111ed8SAndrew Rybchenko 	case EFX_LINK_1000FDX:
4155e111ed8SAndrew Rybchenko 		speed = 1000;
4165e111ed8SAndrew Rybchenko 		break;
4175e111ed8SAndrew Rybchenko 	case EFX_LINK_10000FDX:
4185e111ed8SAndrew Rybchenko 		speed = 10000;
4195e111ed8SAndrew Rybchenko 		break;
4205e111ed8SAndrew Rybchenko 	case EFX_LINK_25000FDX:
4215e111ed8SAndrew Rybchenko 		speed = 25000;
4225e111ed8SAndrew Rybchenko 		break;
4235e111ed8SAndrew Rybchenko 	case EFX_LINK_40000FDX:
4245e111ed8SAndrew Rybchenko 		speed = 40000;
4255e111ed8SAndrew Rybchenko 		break;
4265e111ed8SAndrew Rybchenko 	case EFX_LINK_50000FDX:
4275e111ed8SAndrew Rybchenko 		speed = 50000;
4285e111ed8SAndrew Rybchenko 		break;
4295e111ed8SAndrew Rybchenko 	case EFX_LINK_100000FDX:
4305e111ed8SAndrew Rybchenko 		speed = 100000;
4315e111ed8SAndrew Rybchenko 		break;
4325e111ed8SAndrew Rybchenko 	default:
4335e111ed8SAndrew Rybchenko 		speed = 0;
434252b2aecSAndrew Rybchenko 		break;
4355e111ed8SAndrew Rybchenko 	}
4365e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD(req, SET_LINK_IN_LOOPBACK_SPEED, speed);
4375e111ed8SAndrew Rybchenko 
438252b2aecSAndrew Rybchenko 	MCDI_IN_SET_DWORD(req, SET_LINK_IN_FLAGS, phy_flags);
4395e111ed8SAndrew Rybchenko 
4405e111ed8SAndrew Rybchenko 	efx_mcdi_execute(enp, &req);
4415e111ed8SAndrew Rybchenko 
4425e111ed8SAndrew Rybchenko 	if (req.emr_rc != 0) {
4435e111ed8SAndrew Rybchenko 		rc = req.emr_rc;
444252b2aecSAndrew Rybchenko 		goto fail1;
4455e111ed8SAndrew Rybchenko 	}
4465e111ed8SAndrew Rybchenko 
447252b2aecSAndrew Rybchenko 	return (0);
448252b2aecSAndrew Rybchenko 
449252b2aecSAndrew Rybchenko fail1:
450252b2aecSAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
451252b2aecSAndrew Rybchenko 
452252b2aecSAndrew Rybchenko 	return (rc);
453252b2aecSAndrew Rybchenko }
454252b2aecSAndrew Rybchenko 
4556194d9dcSAndrew Rybchenko static	__checkReturn	efx_rc_t
efx_mcdi_phy_set_led(__in efx_nic_t * enp,__in efx_phy_led_mode_t phy_led_mode)4566194d9dcSAndrew Rybchenko efx_mcdi_phy_set_led(
4576194d9dcSAndrew Rybchenko 	__in		efx_nic_t *enp,
4586194d9dcSAndrew Rybchenko 	__in		efx_phy_led_mode_t phy_led_mode)
4596194d9dcSAndrew Rybchenko {
4606194d9dcSAndrew Rybchenko 	efx_mcdi_req_t req;
4616194d9dcSAndrew Rybchenko 	EFX_MCDI_DECLARE_BUF(payload, MC_CMD_SET_ID_LED_IN_LEN,
4626194d9dcSAndrew Rybchenko 		MC_CMD_SET_ID_LED_OUT_LEN);
4636194d9dcSAndrew Rybchenko 	unsigned int led_mode;
4646194d9dcSAndrew Rybchenko 	efx_rc_t rc;
4656194d9dcSAndrew Rybchenko 
4666194d9dcSAndrew Rybchenko 	req.emr_cmd = MC_CMD_SET_ID_LED;
4676194d9dcSAndrew Rybchenko 	req.emr_in_buf = payload;
4686194d9dcSAndrew Rybchenko 	req.emr_in_length = MC_CMD_SET_ID_LED_IN_LEN;
4696194d9dcSAndrew Rybchenko 	req.emr_out_buf = payload;
4706194d9dcSAndrew Rybchenko 	req.emr_out_length = MC_CMD_SET_ID_LED_OUT_LEN;
4716194d9dcSAndrew Rybchenko 
4726194d9dcSAndrew Rybchenko 	switch (phy_led_mode) {
4736194d9dcSAndrew Rybchenko 	case EFX_PHY_LED_DEFAULT:
4746194d9dcSAndrew Rybchenko 		led_mode = MC_CMD_LED_DEFAULT;
4756194d9dcSAndrew Rybchenko 		break;
4766194d9dcSAndrew Rybchenko 	case EFX_PHY_LED_OFF:
4776194d9dcSAndrew Rybchenko 		led_mode = MC_CMD_LED_OFF;
4786194d9dcSAndrew Rybchenko 		break;
4796194d9dcSAndrew Rybchenko 	case EFX_PHY_LED_ON:
4806194d9dcSAndrew Rybchenko 		led_mode = MC_CMD_LED_ON;
4816194d9dcSAndrew Rybchenko 		break;
4826194d9dcSAndrew Rybchenko 	default:
4836194d9dcSAndrew Rybchenko 		EFSYS_ASSERT(0);
4846194d9dcSAndrew Rybchenko 		led_mode = MC_CMD_LED_DEFAULT;
4856194d9dcSAndrew Rybchenko 		break;
4866194d9dcSAndrew Rybchenko 	}
4876194d9dcSAndrew Rybchenko 
4886194d9dcSAndrew Rybchenko 	MCDI_IN_SET_DWORD(req, SET_ID_LED_IN_STATE, led_mode);
4896194d9dcSAndrew Rybchenko 
4906194d9dcSAndrew Rybchenko 	efx_mcdi_execute(enp, &req);
4916194d9dcSAndrew Rybchenko 
4926194d9dcSAndrew Rybchenko 	if (req.emr_rc != 0) {
4936194d9dcSAndrew Rybchenko 		rc = req.emr_rc;
4946194d9dcSAndrew Rybchenko 		goto fail1;
4956194d9dcSAndrew Rybchenko 	}
4966194d9dcSAndrew Rybchenko 
4976194d9dcSAndrew Rybchenko 	return (0);
4986194d9dcSAndrew Rybchenko 
4996194d9dcSAndrew Rybchenko fail1:
5006194d9dcSAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
5016194d9dcSAndrew Rybchenko 
5026194d9dcSAndrew Rybchenko 	return (rc);
5036194d9dcSAndrew Rybchenko }
5046194d9dcSAndrew Rybchenko 
505252b2aecSAndrew Rybchenko 	__checkReturn	efx_rc_t
ef10_phy_reconfigure(__in efx_nic_t * enp)506252b2aecSAndrew Rybchenko ef10_phy_reconfigure(
507252b2aecSAndrew Rybchenko 	__in		efx_nic_t *enp)
508252b2aecSAndrew Rybchenko {
509252b2aecSAndrew Rybchenko 	efx_port_t *epp = &(enp->en_port);
510252b2aecSAndrew Rybchenko 	efx_loopback_type_t loopback_type;
511252b2aecSAndrew Rybchenko 	efx_link_mode_t loopback_link_mode;
512252b2aecSAndrew Rybchenko 	uint32_t phy_flags;
5136194d9dcSAndrew Rybchenko 	efx_phy_led_mode_t phy_led_mode;
514252b2aecSAndrew Rybchenko 	boolean_t supported;
515252b2aecSAndrew Rybchenko 	efx_rc_t rc;
516252b2aecSAndrew Rybchenko 
517252b2aecSAndrew Rybchenko 	if ((rc = efx_mcdi_link_control_supported(enp, &supported)) != 0)
518252b2aecSAndrew Rybchenko 		goto fail1;
519252b2aecSAndrew Rybchenko 	if (supported == B_FALSE)
520252b2aecSAndrew Rybchenko 		goto out;
521252b2aecSAndrew Rybchenko 
522252b2aecSAndrew Rybchenko #if EFSYS_OPT_LOOPBACK
523252b2aecSAndrew Rybchenko 	loopback_type = epp->ep_loopback_type;
524252b2aecSAndrew Rybchenko 	loopback_link_mode = epp->ep_loopback_link_mode;
525252b2aecSAndrew Rybchenko #else
526252b2aecSAndrew Rybchenko 	loopback_type = EFX_LOOPBACK_OFF;
527252b2aecSAndrew Rybchenko 	loopback_link_mode = EFX_LINK_UNKNOWN;
528252b2aecSAndrew Rybchenko #endif
529252b2aecSAndrew Rybchenko #if EFSYS_OPT_PHY_FLAGS
530252b2aecSAndrew Rybchenko 	phy_flags = epp->ep_phy_flags;
531252b2aecSAndrew Rybchenko #else
532252b2aecSAndrew Rybchenko 	phy_flags = 0;
533252b2aecSAndrew Rybchenko #endif
534252b2aecSAndrew Rybchenko 
535252b2aecSAndrew Rybchenko 	rc = efx_mcdi_phy_set_link(enp, epp->ep_adv_cap_mask,
536252b2aecSAndrew Rybchenko 	    loopback_type, loopback_link_mode, phy_flags);
537252b2aecSAndrew Rybchenko 	if (rc != 0)
538252b2aecSAndrew Rybchenko 		goto fail2;
539252b2aecSAndrew Rybchenko 
5405e111ed8SAndrew Rybchenko 	/* And set the blink mode */
5415e111ed8SAndrew Rybchenko 
5425e111ed8SAndrew Rybchenko #if EFSYS_OPT_PHY_LED_CONTROL
5436194d9dcSAndrew Rybchenko 	phy_led_mode = epp->ep_phy_led_mode;
5445e111ed8SAndrew Rybchenko #else
5456194d9dcSAndrew Rybchenko 	phy_led_mode = EFX_PHY_LED_DEFAULT;
5466194d9dcSAndrew Rybchenko #endif
5475e111ed8SAndrew Rybchenko 
5486194d9dcSAndrew Rybchenko 	rc = efx_mcdi_phy_set_led(enp, phy_led_mode);
54936d79278SAndrew Rybchenko 	if (rc != 0) {
55036d79278SAndrew Rybchenko 		/*
55136d79278SAndrew Rybchenko 		 * If LED control is not supported by firmware, we can
55236d79278SAndrew Rybchenko 		 * silently ignore default mode set failure
55336d79278SAndrew Rybchenko 		 * (see FWRIVERHD-198).
55436d79278SAndrew Rybchenko 		 */
55536d79278SAndrew Rybchenko 		if (rc == EOPNOTSUPP && phy_led_mode == EFX_PHY_LED_DEFAULT)
55636d79278SAndrew Rybchenko 			goto out;
5575e111ed8SAndrew Rybchenko 		goto fail3;
55836d79278SAndrew Rybchenko 	}
5596194d9dcSAndrew Rybchenko 
5605e111ed8SAndrew Rybchenko out:
5615e111ed8SAndrew Rybchenko 	return (0);
5625e111ed8SAndrew Rybchenko 
5635e111ed8SAndrew Rybchenko fail3:
5645e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail3);
5655e111ed8SAndrew Rybchenko fail2:
5665e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail2);
5675e111ed8SAndrew Rybchenko fail1:
5685e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
5695e111ed8SAndrew Rybchenko 
5705e111ed8SAndrew Rybchenko 	return (rc);
5715e111ed8SAndrew Rybchenko }
5725e111ed8SAndrew Rybchenko 
5735e111ed8SAndrew Rybchenko 	__checkReturn	efx_rc_t
ef10_phy_verify(__in efx_nic_t * enp)5745e111ed8SAndrew Rybchenko ef10_phy_verify(
5755e111ed8SAndrew Rybchenko 	__in		efx_nic_t *enp)
5765e111ed8SAndrew Rybchenko {
5775e111ed8SAndrew Rybchenko 	efx_mcdi_req_t req;
5785e111ed8SAndrew Rybchenko 	EFX_MCDI_DECLARE_BUF(payload, MC_CMD_GET_PHY_STATE_IN_LEN,
5795e111ed8SAndrew Rybchenko 		MC_CMD_GET_PHY_STATE_OUT_LEN);
5805e111ed8SAndrew Rybchenko 	uint32_t state;
5815e111ed8SAndrew Rybchenko 	efx_rc_t rc;
5825e111ed8SAndrew Rybchenko 
5835e111ed8SAndrew Rybchenko 	req.emr_cmd = MC_CMD_GET_PHY_STATE;
5845e111ed8SAndrew Rybchenko 	req.emr_in_buf = payload;
5855e111ed8SAndrew Rybchenko 	req.emr_in_length = MC_CMD_GET_PHY_STATE_IN_LEN;
5865e111ed8SAndrew Rybchenko 	req.emr_out_buf = payload;
5875e111ed8SAndrew Rybchenko 	req.emr_out_length = MC_CMD_GET_PHY_STATE_OUT_LEN;
5885e111ed8SAndrew Rybchenko 
5895e111ed8SAndrew Rybchenko 	efx_mcdi_execute(enp, &req);
5905e111ed8SAndrew Rybchenko 
5915e111ed8SAndrew Rybchenko 	if (req.emr_rc != 0) {
5925e111ed8SAndrew Rybchenko 		rc = req.emr_rc;
5935e111ed8SAndrew Rybchenko 		goto fail1;
5945e111ed8SAndrew Rybchenko 	}
5955e111ed8SAndrew Rybchenko 
5965e111ed8SAndrew Rybchenko 	if (req.emr_out_length_used < MC_CMD_GET_PHY_STATE_OUT_LEN) {
5975e111ed8SAndrew Rybchenko 		rc = EMSGSIZE;
5985e111ed8SAndrew Rybchenko 		goto fail2;
5995e111ed8SAndrew Rybchenko 	}
6005e111ed8SAndrew Rybchenko 
6015e111ed8SAndrew Rybchenko 	state = MCDI_OUT_DWORD(req, GET_PHY_STATE_OUT_STATE);
6025e111ed8SAndrew Rybchenko 	if (state != MC_CMD_PHY_STATE_OK) {
6035e111ed8SAndrew Rybchenko 		if (state != MC_CMD_PHY_STATE_ZOMBIE)
6045e111ed8SAndrew Rybchenko 			EFSYS_PROBE1(mc_pcol_error, int, state);
6055e111ed8SAndrew Rybchenko 		rc = ENOTACTIVE;
6065e111ed8SAndrew Rybchenko 		goto fail3;
6075e111ed8SAndrew Rybchenko 	}
6085e111ed8SAndrew Rybchenko 
6095e111ed8SAndrew Rybchenko 	return (0);
6105e111ed8SAndrew Rybchenko 
6115e111ed8SAndrew Rybchenko fail3:
6125e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail3);
6135e111ed8SAndrew Rybchenko fail2:
6145e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail2);
6155e111ed8SAndrew Rybchenko fail1:
6165e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
6175e111ed8SAndrew Rybchenko 
6185e111ed8SAndrew Rybchenko 	return (rc);
6195e111ed8SAndrew Rybchenko }
6205e111ed8SAndrew Rybchenko 
6215e111ed8SAndrew Rybchenko 	__checkReturn	efx_rc_t
ef10_phy_oui_get(__in efx_nic_t * enp,__out uint32_t * ouip)6225e111ed8SAndrew Rybchenko ef10_phy_oui_get(
6235e111ed8SAndrew Rybchenko 	__in		efx_nic_t *enp,
6245e111ed8SAndrew Rybchenko 	__out		uint32_t *ouip)
6255e111ed8SAndrew Rybchenko {
6265e111ed8SAndrew Rybchenko 	_NOTE(ARGUNUSED(enp, ouip))
6275e111ed8SAndrew Rybchenko 
6285e111ed8SAndrew Rybchenko 	return (ENOTSUP);
6295e111ed8SAndrew Rybchenko }
6305e111ed8SAndrew Rybchenko 
6315e111ed8SAndrew Rybchenko 	__checkReturn	efx_rc_t
ef10_phy_link_state_get(__in efx_nic_t * enp,__out efx_phy_link_state_t * eplsp)6325e111ed8SAndrew Rybchenko ef10_phy_link_state_get(
6335e111ed8SAndrew Rybchenko 	__in		efx_nic_t *enp,
6345e111ed8SAndrew Rybchenko 	__out		efx_phy_link_state_t  *eplsp)
6355e111ed8SAndrew Rybchenko {
6365e111ed8SAndrew Rybchenko 	efx_rc_t rc;
6375e111ed8SAndrew Rybchenko 	ef10_link_state_t els;
6385e111ed8SAndrew Rybchenko 
6395e111ed8SAndrew Rybchenko 	/* Obtain the active link state */
6405e111ed8SAndrew Rybchenko 	if ((rc = ef10_phy_get_link(enp, &els)) != 0)
6415e111ed8SAndrew Rybchenko 		goto fail1;
6425e111ed8SAndrew Rybchenko 
6435e111ed8SAndrew Rybchenko 	*eplsp = els.epls;
6445e111ed8SAndrew Rybchenko 
6455e111ed8SAndrew Rybchenko 	return (0);
6465e111ed8SAndrew Rybchenko 
6475e111ed8SAndrew Rybchenko fail1:
6485e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
6495e111ed8SAndrew Rybchenko 
6505e111ed8SAndrew Rybchenko 	return (rc);
6515e111ed8SAndrew Rybchenko }
6525e111ed8SAndrew Rybchenko 
6535e111ed8SAndrew Rybchenko 
6545e111ed8SAndrew Rybchenko #if EFSYS_OPT_PHY_STATS
6555e111ed8SAndrew Rybchenko 
6565e111ed8SAndrew Rybchenko 	__checkReturn				efx_rc_t
ef10_phy_stats_update(__in efx_nic_t * enp,__in efsys_mem_t * esmp,__inout_ecount (EFX_PHY_NSTATS)uint32_t * stat)6575e111ed8SAndrew Rybchenko ef10_phy_stats_update(
6585e111ed8SAndrew Rybchenko 	__in					efx_nic_t *enp,
6595e111ed8SAndrew Rybchenko 	__in					efsys_mem_t *esmp,
6605e111ed8SAndrew Rybchenko 	__inout_ecount(EFX_PHY_NSTATS)		uint32_t *stat)
6615e111ed8SAndrew Rybchenko {
6625e111ed8SAndrew Rybchenko 	/* TBD: no stats support in firmware yet */
6635e111ed8SAndrew Rybchenko 	_NOTE(ARGUNUSED(enp, esmp))
6645e111ed8SAndrew Rybchenko 	memset(stat, 0, EFX_PHY_NSTATS * sizeof (*stat));
6655e111ed8SAndrew Rybchenko 
6665e111ed8SAndrew Rybchenko 	return (0);
6675e111ed8SAndrew Rybchenko }
6685e111ed8SAndrew Rybchenko 
6695e111ed8SAndrew Rybchenko #endif	/* EFSYS_OPT_PHY_STATS */
6705e111ed8SAndrew Rybchenko 
6715e111ed8SAndrew Rybchenko #if EFSYS_OPT_BIST
6725e111ed8SAndrew Rybchenko 
6735e111ed8SAndrew Rybchenko 	__checkReturn		efx_rc_t
ef10_bist_enable_offline(__in efx_nic_t * enp)6745e111ed8SAndrew Rybchenko ef10_bist_enable_offline(
6755e111ed8SAndrew Rybchenko 	__in			efx_nic_t *enp)
6765e111ed8SAndrew Rybchenko {
6775e111ed8SAndrew Rybchenko 	efx_rc_t rc;
6785e111ed8SAndrew Rybchenko 
6795e111ed8SAndrew Rybchenko 	if ((rc = efx_mcdi_bist_enable_offline(enp)) != 0)
6805e111ed8SAndrew Rybchenko 		goto fail1;
6815e111ed8SAndrew Rybchenko 
6825e111ed8SAndrew Rybchenko 	return (0);
6835e111ed8SAndrew Rybchenko 
6845e111ed8SAndrew Rybchenko fail1:
6855e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
6865e111ed8SAndrew Rybchenko 
6875e111ed8SAndrew Rybchenko 	return (rc);
6885e111ed8SAndrew Rybchenko }
6895e111ed8SAndrew Rybchenko 
6905e111ed8SAndrew Rybchenko 	__checkReturn		efx_rc_t
ef10_bist_start(__in efx_nic_t * enp,__in efx_bist_type_t type)6915e111ed8SAndrew Rybchenko ef10_bist_start(
6925e111ed8SAndrew Rybchenko 	__in			efx_nic_t *enp,
6935e111ed8SAndrew Rybchenko 	__in			efx_bist_type_t type)
6945e111ed8SAndrew Rybchenko {
6955e111ed8SAndrew Rybchenko 	efx_rc_t rc;
6965e111ed8SAndrew Rybchenko 
6975e111ed8SAndrew Rybchenko 	if ((rc = efx_mcdi_bist_start(enp, type)) != 0)
6985e111ed8SAndrew Rybchenko 		goto fail1;
6995e111ed8SAndrew Rybchenko 
7005e111ed8SAndrew Rybchenko 	return (0);
7015e111ed8SAndrew Rybchenko 
7025e111ed8SAndrew Rybchenko fail1:
7035e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
7045e111ed8SAndrew Rybchenko 
7055e111ed8SAndrew Rybchenko 	return (rc);
7065e111ed8SAndrew Rybchenko }
7075e111ed8SAndrew Rybchenko 
7085e111ed8SAndrew Rybchenko 	__checkReturn		efx_rc_t
7095e111ed8SAndrew Rybchenko ef10_bist_poll(
7105e111ed8SAndrew Rybchenko 	__in			efx_nic_t *enp,
7115e111ed8SAndrew Rybchenko 	__in			efx_bist_type_t type,
7125e111ed8SAndrew Rybchenko 	__out			efx_bist_result_t *resultp,
7135e111ed8SAndrew Rybchenko 	__out_opt __drv_when(count > 0, __notnull)
7145e111ed8SAndrew Rybchenko 	uint32_t *value_maskp,
7155e111ed8SAndrew Rybchenko 	__out_ecount_opt(count)	__drv_when(count > 0, __notnull)
7165e111ed8SAndrew Rybchenko 	unsigned long *valuesp,
7175e111ed8SAndrew Rybchenko 	__in			size_t count)
7185e111ed8SAndrew Rybchenko {
7195e111ed8SAndrew Rybchenko 	/*
7205e111ed8SAndrew Rybchenko 	 * MCDI_CTL_SDU_LEN_MAX_V1 is large enough cover all BIST results,
7215e111ed8SAndrew Rybchenko 	 * whilst not wasting stack.
7225e111ed8SAndrew Rybchenko 	 */
7235e111ed8SAndrew Rybchenko 	EFX_MCDI_DECLARE_BUF(payload, MC_CMD_POLL_BIST_IN_LEN,
7245e111ed8SAndrew Rybchenko 		MCDI_CTL_SDU_LEN_MAX_V1);
7255e111ed8SAndrew Rybchenko 	efx_nic_cfg_t *encp = &(enp->en_nic_cfg);
7265e111ed8SAndrew Rybchenko 	efx_mcdi_req_t req;
7275e111ed8SAndrew Rybchenko 	uint32_t value_mask = 0;
7285e111ed8SAndrew Rybchenko 	uint32_t result;
7295e111ed8SAndrew Rybchenko 	efx_rc_t rc;
7305e111ed8SAndrew Rybchenko 
7315e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_POLL_BIST_OUT_LEN <=
7325e111ed8SAndrew Rybchenko 	    MCDI_CTL_SDU_LEN_MAX_V1);
7335e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_POLL_BIST_OUT_SFT9001_LEN <=
7345e111ed8SAndrew Rybchenko 	    MCDI_CTL_SDU_LEN_MAX_V1);
7355e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_POLL_BIST_OUT_MRSFP_LEN <=
7365e111ed8SAndrew Rybchenko 	    MCDI_CTL_SDU_LEN_MAX_V1);
7375e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_POLL_BIST_OUT_MEM_LEN <=
7385e111ed8SAndrew Rybchenko 	    MCDI_CTL_SDU_LEN_MAX_V1);
7395e111ed8SAndrew Rybchenko 
7405e111ed8SAndrew Rybchenko 	_NOTE(ARGUNUSED(type))
7415e111ed8SAndrew Rybchenko 
7425e111ed8SAndrew Rybchenko 	req.emr_cmd = MC_CMD_POLL_BIST;
7435e111ed8SAndrew Rybchenko 	req.emr_in_buf = payload;
7445e111ed8SAndrew Rybchenko 	req.emr_in_length = MC_CMD_POLL_BIST_IN_LEN;
7455e111ed8SAndrew Rybchenko 	req.emr_out_buf = payload;
7465e111ed8SAndrew Rybchenko 	req.emr_out_length = MCDI_CTL_SDU_LEN_MAX_V1;
7475e111ed8SAndrew Rybchenko 
7485e111ed8SAndrew Rybchenko 	efx_mcdi_execute(enp, &req);
7495e111ed8SAndrew Rybchenko 
7505e111ed8SAndrew Rybchenko 	if (req.emr_rc != 0) {
7515e111ed8SAndrew Rybchenko 		rc = req.emr_rc;
7525e111ed8SAndrew Rybchenko 		goto fail1;
7535e111ed8SAndrew Rybchenko 	}
7545e111ed8SAndrew Rybchenko 
7555e111ed8SAndrew Rybchenko 	if (req.emr_out_length_used < MC_CMD_POLL_BIST_OUT_RESULT_OFST + 4) {
7565e111ed8SAndrew Rybchenko 		rc = EMSGSIZE;
7575e111ed8SAndrew Rybchenko 		goto fail2;
7585e111ed8SAndrew Rybchenko 	}
7595e111ed8SAndrew Rybchenko 
7605e111ed8SAndrew Rybchenko 	if (count > 0)
7615e111ed8SAndrew Rybchenko 		(void) memset(valuesp, '\0', count * sizeof (unsigned long));
7625e111ed8SAndrew Rybchenko 
7635e111ed8SAndrew Rybchenko 	result = MCDI_OUT_DWORD(req, POLL_BIST_OUT_RESULT);
7645e111ed8SAndrew Rybchenko 
7655e111ed8SAndrew Rybchenko 	if (result == MC_CMD_POLL_BIST_FAILED &&
7665e111ed8SAndrew Rybchenko 	    req.emr_out_length >= MC_CMD_POLL_BIST_OUT_MEM_LEN &&
7675e111ed8SAndrew Rybchenko 	    count > EFX_BIST_MEM_ECC_FATAL) {
7685e111ed8SAndrew Rybchenko 		if (valuesp != NULL) {
7695e111ed8SAndrew Rybchenko 			valuesp[EFX_BIST_MEM_TEST] =
7705e111ed8SAndrew Rybchenko 			    MCDI_OUT_DWORD(req, POLL_BIST_OUT_MEM_TEST);
7715e111ed8SAndrew Rybchenko 			valuesp[EFX_BIST_MEM_ADDR] =
7725e111ed8SAndrew Rybchenko 			    MCDI_OUT_DWORD(req, POLL_BIST_OUT_MEM_ADDR);
7735e111ed8SAndrew Rybchenko 			valuesp[EFX_BIST_MEM_BUS] =
7745e111ed8SAndrew Rybchenko 			    MCDI_OUT_DWORD(req, POLL_BIST_OUT_MEM_BUS);
7755e111ed8SAndrew Rybchenko 			valuesp[EFX_BIST_MEM_EXPECT] =
7765e111ed8SAndrew Rybchenko 			    MCDI_OUT_DWORD(req, POLL_BIST_OUT_MEM_EXPECT);
7775e111ed8SAndrew Rybchenko 			valuesp[EFX_BIST_MEM_ACTUAL] =
7785e111ed8SAndrew Rybchenko 			    MCDI_OUT_DWORD(req, POLL_BIST_OUT_MEM_ACTUAL);
7795e111ed8SAndrew Rybchenko 			valuesp[EFX_BIST_MEM_ECC] =
7805e111ed8SAndrew Rybchenko 			    MCDI_OUT_DWORD(req, POLL_BIST_OUT_MEM_ECC);
7815e111ed8SAndrew Rybchenko 			valuesp[EFX_BIST_MEM_ECC_PARITY] =
7825e111ed8SAndrew Rybchenko 			    MCDI_OUT_DWORD(req, POLL_BIST_OUT_MEM_ECC_PARITY);
7835e111ed8SAndrew Rybchenko 			valuesp[EFX_BIST_MEM_ECC_FATAL] =
7845e111ed8SAndrew Rybchenko 			    MCDI_OUT_DWORD(req, POLL_BIST_OUT_MEM_ECC_FATAL);
7855e111ed8SAndrew Rybchenko 		}
7865e111ed8SAndrew Rybchenko 		value_mask |= (1 << EFX_BIST_MEM_TEST) |
7875e111ed8SAndrew Rybchenko 		    (1 << EFX_BIST_MEM_ADDR) |
7885e111ed8SAndrew Rybchenko 		    (1 << EFX_BIST_MEM_BUS) |
7895e111ed8SAndrew Rybchenko 		    (1 << EFX_BIST_MEM_EXPECT) |
7905e111ed8SAndrew Rybchenko 		    (1 << EFX_BIST_MEM_ACTUAL) |
7915e111ed8SAndrew Rybchenko 		    (1 << EFX_BIST_MEM_ECC) |
7925e111ed8SAndrew Rybchenko 		    (1 << EFX_BIST_MEM_ECC_PARITY) |
7935e111ed8SAndrew Rybchenko 		    (1 << EFX_BIST_MEM_ECC_FATAL);
7945e111ed8SAndrew Rybchenko 	} else if (result == MC_CMD_POLL_BIST_FAILED &&
7955e111ed8SAndrew Rybchenko 	    encp->enc_phy_type == EFX_PHY_XFI_FARMI &&
7965e111ed8SAndrew Rybchenko 	    req.emr_out_length >= MC_CMD_POLL_BIST_OUT_MRSFP_LEN &&
7975e111ed8SAndrew Rybchenko 	    count > EFX_BIST_FAULT_CODE) {
7985e111ed8SAndrew Rybchenko 		if (valuesp != NULL)
7995e111ed8SAndrew Rybchenko 			valuesp[EFX_BIST_FAULT_CODE] =
8005e111ed8SAndrew Rybchenko 			    MCDI_OUT_DWORD(req, POLL_BIST_OUT_MRSFP_TEST);
8015e111ed8SAndrew Rybchenko 		value_mask |= 1 << EFX_BIST_FAULT_CODE;
8025e111ed8SAndrew Rybchenko 	}
8035e111ed8SAndrew Rybchenko 
8045e111ed8SAndrew Rybchenko 	if (value_maskp != NULL)
8055e111ed8SAndrew Rybchenko 		*value_maskp = value_mask;
8065e111ed8SAndrew Rybchenko 
8075e111ed8SAndrew Rybchenko 	EFSYS_ASSERT(resultp != NULL);
8085e111ed8SAndrew Rybchenko 	if (result == MC_CMD_POLL_BIST_RUNNING)
8095e111ed8SAndrew Rybchenko 		*resultp = EFX_BIST_RESULT_RUNNING;
8105e111ed8SAndrew Rybchenko 	else if (result == MC_CMD_POLL_BIST_PASSED)
8115e111ed8SAndrew Rybchenko 		*resultp = EFX_BIST_RESULT_PASSED;
8125e111ed8SAndrew Rybchenko 	else
8135e111ed8SAndrew Rybchenko 		*resultp = EFX_BIST_RESULT_FAILED;
8145e111ed8SAndrew Rybchenko 
8155e111ed8SAndrew Rybchenko 	return (0);
8165e111ed8SAndrew Rybchenko 
8175e111ed8SAndrew Rybchenko fail2:
8185e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail2);
8195e111ed8SAndrew Rybchenko fail1:
8205e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
8215e111ed8SAndrew Rybchenko 
8225e111ed8SAndrew Rybchenko 	return (rc);
8235e111ed8SAndrew Rybchenko }
8245e111ed8SAndrew Rybchenko 
8255e111ed8SAndrew Rybchenko 			void
ef10_bist_stop(__in efx_nic_t * enp,__in efx_bist_type_t type)8265e111ed8SAndrew Rybchenko ef10_bist_stop(
8275e111ed8SAndrew Rybchenko 	__in		efx_nic_t *enp,
8285e111ed8SAndrew Rybchenko 	__in		efx_bist_type_t type)
8295e111ed8SAndrew Rybchenko {
8305e111ed8SAndrew Rybchenko 	/* There is no way to stop BIST on EF10. */
8315e111ed8SAndrew Rybchenko 	_NOTE(ARGUNUSED(enp, type))
8325e111ed8SAndrew Rybchenko }
8335e111ed8SAndrew Rybchenko 
8345e111ed8SAndrew Rybchenko #endif	/* EFSYS_OPT_BIST */
8355e111ed8SAndrew Rybchenko 
8363ac211cfSAndrew Rybchenko #endif	/* EFSYS_OPT_RIVERHEAD || EFX_OPTS_EF10() */
837