xref: /dpdk/drivers/common/sfc_efx/base/siena_phy.c (revision 672386c1e9e1f64f7aa3b1360ad22dc737ea8d72)
15e111ed8SAndrew Rybchenko /* SPDX-License-Identifier: BSD-3-Clause
25e111ed8SAndrew Rybchenko  *
3*672386c1SAndrew Rybchenko  * Copyright(c) 2019-2021 Xilinx, Inc.
45e111ed8SAndrew Rybchenko  * Copyright(c) 2009-2019 Solarflare Communications Inc.
55e111ed8SAndrew Rybchenko  */
65e111ed8SAndrew Rybchenko 
75e111ed8SAndrew Rybchenko #include "efx.h"
85e111ed8SAndrew Rybchenko #include "efx_impl.h"
95e111ed8SAndrew Rybchenko 
105e111ed8SAndrew Rybchenko #if EFSYS_OPT_SIENA
115e111ed8SAndrew Rybchenko 
125e111ed8SAndrew Rybchenko static			void
siena_phy_decode_cap(__in uint32_t mcdi_cap,__out uint32_t * maskp)135e111ed8SAndrew Rybchenko siena_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 	mask = 0;
205e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_10HDX_LBN))
215e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_10HDX);
225e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_10FDX_LBN))
235e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_10FDX);
245e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_100HDX_LBN))
255e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_100HDX);
265e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_100FDX_LBN))
275e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_100FDX);
285e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_1000HDX_LBN))
295e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_1000HDX);
305e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_1000FDX_LBN))
315e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_1000FDX);
325e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_10000FDX_LBN))
335e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_10000FDX);
345e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_PAUSE_LBN))
355e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_PAUSE);
365e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_ASYM_LBN))
375e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_ASYM);
385e111ed8SAndrew Rybchenko 	if (mcdi_cap & (1 << MC_CMD_PHY_CAP_AN_LBN))
395e111ed8SAndrew Rybchenko 		mask |= (1 << EFX_PHY_CAP_AN);
405e111ed8SAndrew Rybchenko 
415e111ed8SAndrew Rybchenko 	*maskp = mask;
425e111ed8SAndrew Rybchenko }
435e111ed8SAndrew Rybchenko 
445e111ed8SAndrew Rybchenko static			void
siena_phy_decode_link_mode(__in efx_nic_t * enp,__in uint32_t link_flags,__in unsigned int speed,__in unsigned int fcntl,__out efx_link_mode_t * link_modep,__out unsigned int * fcntlp)455e111ed8SAndrew Rybchenko siena_phy_decode_link_mode(
465e111ed8SAndrew Rybchenko 	__in		efx_nic_t *enp,
475e111ed8SAndrew Rybchenko 	__in		uint32_t link_flags,
485e111ed8SAndrew Rybchenko 	__in		unsigned int speed,
495e111ed8SAndrew Rybchenko 	__in		unsigned int fcntl,
505e111ed8SAndrew Rybchenko 	__out		efx_link_mode_t *link_modep,
515e111ed8SAndrew Rybchenko 	__out		unsigned int *fcntlp)
525e111ed8SAndrew Rybchenko {
535e111ed8SAndrew Rybchenko 	boolean_t fd = !!(link_flags &
545e111ed8SAndrew Rybchenko 		    (1 << MC_CMD_GET_LINK_OUT_FULL_DUPLEX_LBN));
555e111ed8SAndrew Rybchenko 	boolean_t up = !!(link_flags &
565e111ed8SAndrew Rybchenko 		    (1 << MC_CMD_GET_LINK_OUT_LINK_UP_LBN));
575e111ed8SAndrew Rybchenko 
585e111ed8SAndrew Rybchenko 	_NOTE(ARGUNUSED(enp))
595e111ed8SAndrew Rybchenko 
605e111ed8SAndrew Rybchenko 	if (!up)
615e111ed8SAndrew Rybchenko 		*link_modep = EFX_LINK_DOWN;
625e111ed8SAndrew Rybchenko 	else if (speed == 10000 && fd)
635e111ed8SAndrew Rybchenko 		*link_modep = EFX_LINK_10000FDX;
645e111ed8SAndrew Rybchenko 	else if (speed == 1000)
655e111ed8SAndrew Rybchenko 		*link_modep = fd ? EFX_LINK_1000FDX : EFX_LINK_1000HDX;
665e111ed8SAndrew Rybchenko 	else if (speed == 100)
675e111ed8SAndrew Rybchenko 		*link_modep = fd ? EFX_LINK_100FDX : EFX_LINK_100HDX;
685e111ed8SAndrew Rybchenko 	else if (speed == 10)
695e111ed8SAndrew Rybchenko 		*link_modep = fd ? EFX_LINK_10FDX : EFX_LINK_10HDX;
705e111ed8SAndrew Rybchenko 	else
715e111ed8SAndrew Rybchenko 		*link_modep = EFX_LINK_UNKNOWN;
725e111ed8SAndrew Rybchenko 
735e111ed8SAndrew Rybchenko 	if (fcntl == MC_CMD_FCNTL_OFF)
745e111ed8SAndrew Rybchenko 		*fcntlp = 0;
755e111ed8SAndrew Rybchenko 	else if (fcntl == MC_CMD_FCNTL_RESPOND)
765e111ed8SAndrew Rybchenko 		*fcntlp = EFX_FCNTL_RESPOND;
775e111ed8SAndrew Rybchenko 	else if (fcntl == MC_CMD_FCNTL_BIDIR)
785e111ed8SAndrew Rybchenko 		*fcntlp = EFX_FCNTL_RESPOND | EFX_FCNTL_GENERATE;
795e111ed8SAndrew Rybchenko 	else {
805e111ed8SAndrew Rybchenko 		EFSYS_PROBE1(mc_pcol_error, int, fcntl);
815e111ed8SAndrew Rybchenko 		*fcntlp = 0;
825e111ed8SAndrew Rybchenko 	}
835e111ed8SAndrew Rybchenko }
845e111ed8SAndrew Rybchenko 
855e111ed8SAndrew Rybchenko 			void
siena_phy_link_ev(__in efx_nic_t * enp,__in efx_qword_t * eqp,__out efx_link_mode_t * link_modep)865e111ed8SAndrew Rybchenko siena_phy_link_ev(
875e111ed8SAndrew Rybchenko 	__in		efx_nic_t *enp,
885e111ed8SAndrew Rybchenko 	__in		efx_qword_t *eqp,
895e111ed8SAndrew Rybchenko 	__out		efx_link_mode_t *link_modep)
905e111ed8SAndrew Rybchenko {
915e111ed8SAndrew Rybchenko 	efx_port_t *epp = &(enp->en_port);
925e111ed8SAndrew Rybchenko 	unsigned int link_flags;
935e111ed8SAndrew Rybchenko 	unsigned int speed;
945e111ed8SAndrew Rybchenko 	unsigned int fcntl;
955e111ed8SAndrew Rybchenko 	efx_link_mode_t link_mode;
965e111ed8SAndrew Rybchenko 	uint32_t lp_cap_mask;
975e111ed8SAndrew Rybchenko 
985e111ed8SAndrew Rybchenko 	/*
995e111ed8SAndrew Rybchenko 	 * Convert the LINKCHANGE speed enumeration into mbit/s, in the
1005e111ed8SAndrew Rybchenko 	 * same way as GET_LINK encodes the speed
1015e111ed8SAndrew Rybchenko 	 */
1025e111ed8SAndrew Rybchenko 	switch (MCDI_EV_FIELD(eqp, LINKCHANGE_SPEED)) {
1035e111ed8SAndrew Rybchenko 	case MCDI_EVENT_LINKCHANGE_SPEED_100M:
1045e111ed8SAndrew Rybchenko 		speed = 100;
1055e111ed8SAndrew Rybchenko 		break;
1065e111ed8SAndrew Rybchenko 	case MCDI_EVENT_LINKCHANGE_SPEED_1G:
1075e111ed8SAndrew Rybchenko 		speed = 1000;
1085e111ed8SAndrew Rybchenko 		break;
1095e111ed8SAndrew Rybchenko 	case MCDI_EVENT_LINKCHANGE_SPEED_10G:
1105e111ed8SAndrew Rybchenko 		speed = 10000;
1115e111ed8SAndrew Rybchenko 		break;
1125e111ed8SAndrew Rybchenko 	default:
1135e111ed8SAndrew Rybchenko 		speed = 0;
1145e111ed8SAndrew Rybchenko 		break;
1155e111ed8SAndrew Rybchenko 	}
1165e111ed8SAndrew Rybchenko 
1175e111ed8SAndrew Rybchenko 	link_flags = MCDI_EV_FIELD(eqp, LINKCHANGE_LINK_FLAGS);
1185e111ed8SAndrew Rybchenko 	siena_phy_decode_link_mode(enp, link_flags, speed,
1195e111ed8SAndrew Rybchenko 				    MCDI_EV_FIELD(eqp, LINKCHANGE_FCNTL),
1205e111ed8SAndrew Rybchenko 				    &link_mode, &fcntl);
1215e111ed8SAndrew Rybchenko 	siena_phy_decode_cap(MCDI_EV_FIELD(eqp, LINKCHANGE_LP_CAP),
1225e111ed8SAndrew Rybchenko 			    &lp_cap_mask);
1235e111ed8SAndrew Rybchenko 
1245e111ed8SAndrew Rybchenko 	/*
1255e111ed8SAndrew Rybchenko 	 * It's safe to update ep_lp_cap_mask without the driver's port lock
1265e111ed8SAndrew Rybchenko 	 * because presumably any concurrently running efx_port_poll() is
1275e111ed8SAndrew Rybchenko 	 * only going to arrive at the same value.
1285e111ed8SAndrew Rybchenko 	 *
1295e111ed8SAndrew Rybchenko 	 * ep_fcntl has two meanings. It's either the link common fcntl
1305e111ed8SAndrew Rybchenko 	 * (if the PHY supports AN), or it's the forced link state. If
1315e111ed8SAndrew Rybchenko 	 * the former, it's safe to update the value for the same reason as
1325e111ed8SAndrew Rybchenko 	 * for ep_lp_cap_mask. If the latter, then just ignore the value,
1335e111ed8SAndrew Rybchenko 	 * because we can race with efx_mac_fcntl_set().
1345e111ed8SAndrew Rybchenko 	 */
1355e111ed8SAndrew Rybchenko 	epp->ep_lp_cap_mask = lp_cap_mask;
1365e111ed8SAndrew Rybchenko 	if (epp->ep_phy_cap_mask & (1 << EFX_PHY_CAP_AN))
1375e111ed8SAndrew Rybchenko 		epp->ep_fcntl = fcntl;
1385e111ed8SAndrew Rybchenko 
1395e111ed8SAndrew Rybchenko 	*link_modep = link_mode;
1405e111ed8SAndrew Rybchenko }
1415e111ed8SAndrew Rybchenko 
1425e111ed8SAndrew Rybchenko 	__checkReturn	efx_rc_t
siena_phy_power(__in efx_nic_t * enp,__in boolean_t power)1435e111ed8SAndrew Rybchenko siena_phy_power(
1445e111ed8SAndrew Rybchenko 	__in		efx_nic_t *enp,
1455e111ed8SAndrew Rybchenko 	__in		boolean_t power)
1465e111ed8SAndrew Rybchenko {
1475e111ed8SAndrew Rybchenko 	efx_rc_t rc;
1485e111ed8SAndrew Rybchenko 
1495e111ed8SAndrew Rybchenko 	if (!power)
1505e111ed8SAndrew Rybchenko 		return (0);
1515e111ed8SAndrew Rybchenko 
1525e111ed8SAndrew Rybchenko 	/* Check if the PHY is a zombie */
1535e111ed8SAndrew Rybchenko 	if ((rc = siena_phy_verify(enp)) != 0)
1545e111ed8SAndrew Rybchenko 		goto fail1;
1555e111ed8SAndrew Rybchenko 
1565e111ed8SAndrew Rybchenko 	enp->en_reset_flags |= EFX_RESET_PHY;
1575e111ed8SAndrew Rybchenko 
1585e111ed8SAndrew Rybchenko 	return (0);
1595e111ed8SAndrew Rybchenko 
1605e111ed8SAndrew Rybchenko fail1:
1615e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
1625e111ed8SAndrew Rybchenko 
1635e111ed8SAndrew Rybchenko 	return (rc);
1645e111ed8SAndrew Rybchenko }
1655e111ed8SAndrew Rybchenko 
1665e111ed8SAndrew Rybchenko 	__checkReturn	efx_rc_t
siena_phy_get_link(__in efx_nic_t * enp,__out siena_link_state_t * slsp)1675e111ed8SAndrew Rybchenko siena_phy_get_link(
1685e111ed8SAndrew Rybchenko 	__in		efx_nic_t *enp,
1695e111ed8SAndrew Rybchenko 	__out		siena_link_state_t *slsp)
1705e111ed8SAndrew Rybchenko {
1715e111ed8SAndrew Rybchenko 	efx_mcdi_req_t req;
1725e111ed8SAndrew Rybchenko 	EFX_MCDI_DECLARE_BUF(payload, MC_CMD_GET_LINK_IN_LEN,
1735e111ed8SAndrew Rybchenko 		MC_CMD_GET_LINK_OUT_LEN);
1745e111ed8SAndrew Rybchenko 	efx_rc_t rc;
1755e111ed8SAndrew Rybchenko 
1765e111ed8SAndrew Rybchenko 	req.emr_cmd = MC_CMD_GET_LINK;
1775e111ed8SAndrew Rybchenko 	req.emr_in_buf = payload;
1785e111ed8SAndrew Rybchenko 	req.emr_in_length = MC_CMD_GET_LINK_IN_LEN;
1795e111ed8SAndrew Rybchenko 	req.emr_out_buf = payload;
1805e111ed8SAndrew Rybchenko 	req.emr_out_length = MC_CMD_GET_LINK_OUT_LEN;
1815e111ed8SAndrew Rybchenko 
1825e111ed8SAndrew Rybchenko 	efx_mcdi_execute(enp, &req);
1835e111ed8SAndrew Rybchenko 
1845e111ed8SAndrew Rybchenko 	if (req.emr_rc != 0) {
1855e111ed8SAndrew Rybchenko 		rc = req.emr_rc;
1865e111ed8SAndrew Rybchenko 		goto fail1;
1875e111ed8SAndrew Rybchenko 	}
1885e111ed8SAndrew Rybchenko 
1895e111ed8SAndrew Rybchenko 	if (req.emr_out_length_used < MC_CMD_GET_LINK_OUT_LEN) {
1905e111ed8SAndrew Rybchenko 		rc = EMSGSIZE;
1915e111ed8SAndrew Rybchenko 		goto fail2;
1925e111ed8SAndrew Rybchenko 	}
1935e111ed8SAndrew Rybchenko 
1945e111ed8SAndrew Rybchenko 	siena_phy_decode_cap(MCDI_OUT_DWORD(req, GET_LINK_OUT_CAP),
1955e111ed8SAndrew Rybchenko 			    &slsp->sls_adv_cap_mask);
1965e111ed8SAndrew Rybchenko 	siena_phy_decode_cap(MCDI_OUT_DWORD(req, GET_LINK_OUT_LP_CAP),
1975e111ed8SAndrew Rybchenko 			    &slsp->sls_lp_cap_mask);
1985e111ed8SAndrew Rybchenko 
1995e111ed8SAndrew Rybchenko 	siena_phy_decode_link_mode(enp, MCDI_OUT_DWORD(req, GET_LINK_OUT_FLAGS),
2005e111ed8SAndrew Rybchenko 			    MCDI_OUT_DWORD(req, GET_LINK_OUT_LINK_SPEED),
2015e111ed8SAndrew Rybchenko 			    MCDI_OUT_DWORD(req, GET_LINK_OUT_FCNTL),
2025e111ed8SAndrew Rybchenko 			    &slsp->sls_link_mode, &slsp->sls_fcntl);
2035e111ed8SAndrew Rybchenko 
2045e111ed8SAndrew Rybchenko #if EFSYS_OPT_LOOPBACK
2055e111ed8SAndrew Rybchenko 	/* Assert the MC_CMD_LOOPBACK and EFX_LOOPBACK namespace agree */
2065e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_LOOPBACK_NONE == EFX_LOOPBACK_OFF);
2075e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_LOOPBACK_DATA == EFX_LOOPBACK_DATA);
2085e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_LOOPBACK_GMAC == EFX_LOOPBACK_GMAC);
2095e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_LOOPBACK_XGMII == EFX_LOOPBACK_XGMII);
2105e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_LOOPBACK_XGXS == EFX_LOOPBACK_XGXS);
2115e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_LOOPBACK_XAUI == EFX_LOOPBACK_XAUI);
2125e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_LOOPBACK_GMII == EFX_LOOPBACK_GMII);
2135e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_LOOPBACK_SGMII == EFX_LOOPBACK_SGMII);
2145e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_LOOPBACK_XGBR == EFX_LOOPBACK_XGBR);
2155e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_LOOPBACK_XFI == EFX_LOOPBACK_XFI);
2165e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_LOOPBACK_XAUI_FAR == EFX_LOOPBACK_XAUI_FAR);
2175e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_LOOPBACK_GMII_FAR == EFX_LOOPBACK_GMII_FAR);
2185e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_LOOPBACK_SGMII_FAR == EFX_LOOPBACK_SGMII_FAR);
2195e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_LOOPBACK_XFI_FAR == EFX_LOOPBACK_XFI_FAR);
2205e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_LOOPBACK_GPHY == EFX_LOOPBACK_GPHY);
2215e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_LOOPBACK_PHYXS == EFX_LOOPBACK_PHY_XS);
2225e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_LOOPBACK_PCS == EFX_LOOPBACK_PCS);
2235e111ed8SAndrew Rybchenko 	EFX_STATIC_ASSERT(MC_CMD_LOOPBACK_PMAPMD == EFX_LOOPBACK_PMA_PMD);
2245e111ed8SAndrew Rybchenko 
2255e111ed8SAndrew Rybchenko 	slsp->sls_loopback = MCDI_OUT_DWORD(req, GET_LINK_OUT_LOOPBACK_MODE);
2265e111ed8SAndrew Rybchenko #endif	/* EFSYS_OPT_LOOPBACK */
2275e111ed8SAndrew Rybchenko 
2285e111ed8SAndrew Rybchenko 	slsp->sls_mac_up = MCDI_OUT_DWORD(req, GET_LINK_OUT_MAC_FAULT) == 0;
2295e111ed8SAndrew Rybchenko 
2305e111ed8SAndrew Rybchenko 	return (0);
2315e111ed8SAndrew Rybchenko 
2325e111ed8SAndrew Rybchenko fail2:
2335e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail2);
2345e111ed8SAndrew Rybchenko fail1:
2355e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
2365e111ed8SAndrew Rybchenko 
2375e111ed8SAndrew Rybchenko 	return (rc);
2385e111ed8SAndrew Rybchenko }
2395e111ed8SAndrew Rybchenko 
2405e111ed8SAndrew Rybchenko 	__checkReturn	efx_rc_t
siena_phy_reconfigure(__in efx_nic_t * enp)2415e111ed8SAndrew Rybchenko siena_phy_reconfigure(
2425e111ed8SAndrew Rybchenko 	__in		efx_nic_t *enp)
2435e111ed8SAndrew Rybchenko {
2445e111ed8SAndrew Rybchenko 	efx_port_t *epp = &(enp->en_port);
2455e111ed8SAndrew Rybchenko 	efx_mcdi_req_t req;
2465e111ed8SAndrew Rybchenko 	EFX_MCDI_DECLARE_BUF(payload,
2475e111ed8SAndrew Rybchenko 		MAX(MC_CMD_SET_ID_LED_IN_LEN, MC_CMD_SET_LINK_IN_LEN),
2485e111ed8SAndrew Rybchenko 		MAX(MC_CMD_SET_ID_LED_OUT_LEN, MC_CMD_SET_LINK_OUT_LEN));
2495e111ed8SAndrew Rybchenko 	uint32_t cap_mask;
2505e111ed8SAndrew Rybchenko #if EFSYS_OPT_PHY_LED_CONTROL
2515e111ed8SAndrew Rybchenko 	unsigned int led_mode;
2525e111ed8SAndrew Rybchenko #endif
2535e111ed8SAndrew Rybchenko 	unsigned int speed;
2545e111ed8SAndrew Rybchenko 	efx_rc_t rc;
2555e111ed8SAndrew Rybchenko 
2565e111ed8SAndrew Rybchenko 	req.emr_cmd = MC_CMD_SET_LINK;
2575e111ed8SAndrew Rybchenko 	req.emr_in_buf = payload;
2585e111ed8SAndrew Rybchenko 	req.emr_in_length = MC_CMD_SET_LINK_IN_LEN;
2595e111ed8SAndrew Rybchenko 	req.emr_out_buf = payload;
2605e111ed8SAndrew Rybchenko 	req.emr_out_length = MC_CMD_SET_LINK_OUT_LEN;
2615e111ed8SAndrew Rybchenko 
2625e111ed8SAndrew Rybchenko 	cap_mask = epp->ep_adv_cap_mask;
2635e111ed8SAndrew Rybchenko 	MCDI_IN_POPULATE_DWORD_10(req, SET_LINK_IN_CAP,
2645e111ed8SAndrew Rybchenko 		PHY_CAP_10HDX, (cap_mask >> EFX_PHY_CAP_10HDX) & 0x1,
2655e111ed8SAndrew Rybchenko 		PHY_CAP_10FDX, (cap_mask >> EFX_PHY_CAP_10FDX) & 0x1,
2665e111ed8SAndrew Rybchenko 		PHY_CAP_100HDX, (cap_mask >> EFX_PHY_CAP_100HDX) & 0x1,
2675e111ed8SAndrew Rybchenko 		PHY_CAP_100FDX, (cap_mask >> EFX_PHY_CAP_100FDX) & 0x1,
2685e111ed8SAndrew Rybchenko 		PHY_CAP_1000HDX, (cap_mask >> EFX_PHY_CAP_1000HDX) & 0x1,
2695e111ed8SAndrew Rybchenko 		PHY_CAP_1000FDX, (cap_mask >> EFX_PHY_CAP_1000FDX) & 0x1,
2705e111ed8SAndrew Rybchenko 		PHY_CAP_10000FDX, (cap_mask >> EFX_PHY_CAP_10000FDX) & 0x1,
2715e111ed8SAndrew Rybchenko 		PHY_CAP_PAUSE, (cap_mask >> EFX_PHY_CAP_PAUSE) & 0x1,
2725e111ed8SAndrew Rybchenko 		PHY_CAP_ASYM, (cap_mask >> EFX_PHY_CAP_ASYM) & 0x1,
2735e111ed8SAndrew Rybchenko 		PHY_CAP_AN, (cap_mask >> EFX_PHY_CAP_AN) & 0x1);
2745e111ed8SAndrew Rybchenko 
2755e111ed8SAndrew Rybchenko #if EFSYS_OPT_LOOPBACK
2765e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD(req, SET_LINK_IN_LOOPBACK_MODE,
2775e111ed8SAndrew Rybchenko 		    epp->ep_loopback_type);
2785e111ed8SAndrew Rybchenko 	switch (epp->ep_loopback_link_mode) {
2795e111ed8SAndrew Rybchenko 	case EFX_LINK_100FDX:
2805e111ed8SAndrew Rybchenko 		speed = 100;
2815e111ed8SAndrew Rybchenko 		break;
2825e111ed8SAndrew Rybchenko 	case EFX_LINK_1000FDX:
2835e111ed8SAndrew Rybchenko 		speed = 1000;
2845e111ed8SAndrew Rybchenko 		break;
2855e111ed8SAndrew Rybchenko 	case EFX_LINK_10000FDX:
2865e111ed8SAndrew Rybchenko 		speed = 10000;
2875e111ed8SAndrew Rybchenko 		break;
2885e111ed8SAndrew Rybchenko 	default:
2895e111ed8SAndrew Rybchenko 		speed = 0;
2905e111ed8SAndrew Rybchenko 	}
2915e111ed8SAndrew Rybchenko #else
2925e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD(req, SET_LINK_IN_LOOPBACK_MODE, MC_CMD_LOOPBACK_NONE);
2935e111ed8SAndrew Rybchenko 	speed = 0;
2945e111ed8SAndrew Rybchenko #endif	/* EFSYS_OPT_LOOPBACK */
2955e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD(req, SET_LINK_IN_LOOPBACK_SPEED, speed);
2965e111ed8SAndrew Rybchenko 
2975e111ed8SAndrew Rybchenko #if EFSYS_OPT_PHY_FLAGS
2985e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD(req, SET_LINK_IN_FLAGS, epp->ep_phy_flags);
2995e111ed8SAndrew Rybchenko #else
3005e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD(req, SET_LINK_IN_FLAGS, 0);
3015e111ed8SAndrew Rybchenko #endif	/* EFSYS_OPT_PHY_FLAGS */
3025e111ed8SAndrew Rybchenko 
3035e111ed8SAndrew Rybchenko 	efx_mcdi_execute(enp, &req);
3045e111ed8SAndrew Rybchenko 
3055e111ed8SAndrew Rybchenko 	if (req.emr_rc != 0) {
3065e111ed8SAndrew Rybchenko 		rc = req.emr_rc;
3075e111ed8SAndrew Rybchenko 		goto fail1;
3085e111ed8SAndrew Rybchenko 	}
3095e111ed8SAndrew Rybchenko 
3105e111ed8SAndrew Rybchenko 	/* And set the blink mode */
3115e111ed8SAndrew Rybchenko 	(void) memset(payload, 0, sizeof (payload));
3125e111ed8SAndrew Rybchenko 	req.emr_cmd = MC_CMD_SET_ID_LED;
3135e111ed8SAndrew Rybchenko 	req.emr_in_buf = payload;
3145e111ed8SAndrew Rybchenko 	req.emr_in_length = MC_CMD_SET_ID_LED_IN_LEN;
3155e111ed8SAndrew Rybchenko 	req.emr_out_buf = payload;
3165e111ed8SAndrew Rybchenko 	req.emr_out_length = MC_CMD_SET_ID_LED_OUT_LEN;
3175e111ed8SAndrew Rybchenko 
3185e111ed8SAndrew Rybchenko #if EFSYS_OPT_PHY_LED_CONTROL
3195e111ed8SAndrew Rybchenko 	switch (epp->ep_phy_led_mode) {
3205e111ed8SAndrew Rybchenko 	case EFX_PHY_LED_DEFAULT:
3215e111ed8SAndrew Rybchenko 		led_mode = MC_CMD_LED_DEFAULT;
3225e111ed8SAndrew Rybchenko 		break;
3235e111ed8SAndrew Rybchenko 	case EFX_PHY_LED_OFF:
3245e111ed8SAndrew Rybchenko 		led_mode = MC_CMD_LED_OFF;
3255e111ed8SAndrew Rybchenko 		break;
3265e111ed8SAndrew Rybchenko 	case EFX_PHY_LED_ON:
3275e111ed8SAndrew Rybchenko 		led_mode = MC_CMD_LED_ON;
3285e111ed8SAndrew Rybchenko 		break;
3295e111ed8SAndrew Rybchenko 	default:
3305e111ed8SAndrew Rybchenko 		EFSYS_ASSERT(0);
3315e111ed8SAndrew Rybchenko 		led_mode = MC_CMD_LED_DEFAULT;
3325e111ed8SAndrew Rybchenko 	}
3335e111ed8SAndrew Rybchenko 
3345e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD(req, SET_ID_LED_IN_STATE, led_mode);
3355e111ed8SAndrew Rybchenko #else
3365e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD(req, SET_ID_LED_IN_STATE, MC_CMD_LED_DEFAULT);
3375e111ed8SAndrew Rybchenko #endif	/* EFSYS_OPT_PHY_LED_CONTROL */
3385e111ed8SAndrew Rybchenko 
3395e111ed8SAndrew Rybchenko 	efx_mcdi_execute(enp, &req);
3405e111ed8SAndrew Rybchenko 
3415e111ed8SAndrew Rybchenko 	if (req.emr_rc != 0) {
3425e111ed8SAndrew Rybchenko 		rc = req.emr_rc;
3435e111ed8SAndrew Rybchenko 		goto fail2;
3445e111ed8SAndrew Rybchenko 	}
3455e111ed8SAndrew Rybchenko 
3465e111ed8SAndrew Rybchenko 	return (0);
3475e111ed8SAndrew Rybchenko 
3485e111ed8SAndrew Rybchenko fail2:
3495e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail2);
3505e111ed8SAndrew Rybchenko fail1:
3515e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
3525e111ed8SAndrew Rybchenko 
3535e111ed8SAndrew Rybchenko 	return (rc);
3545e111ed8SAndrew Rybchenko }
3555e111ed8SAndrew Rybchenko 
3565e111ed8SAndrew Rybchenko 	__checkReturn	efx_rc_t
siena_phy_verify(__in efx_nic_t * enp)3575e111ed8SAndrew Rybchenko siena_phy_verify(
3585e111ed8SAndrew Rybchenko 	__in		efx_nic_t *enp)
3595e111ed8SAndrew Rybchenko {
3605e111ed8SAndrew Rybchenko 	efx_mcdi_req_t req;
3615e111ed8SAndrew Rybchenko 	EFX_MCDI_DECLARE_BUF(payload, MC_CMD_GET_PHY_STATE_IN_LEN,
3625e111ed8SAndrew Rybchenko 		MC_CMD_GET_PHY_STATE_OUT_LEN);
3635e111ed8SAndrew Rybchenko 	uint32_t state;
3645e111ed8SAndrew Rybchenko 	efx_rc_t rc;
3655e111ed8SAndrew Rybchenko 
3665e111ed8SAndrew Rybchenko 	req.emr_cmd = MC_CMD_GET_PHY_STATE;
3675e111ed8SAndrew Rybchenko 	req.emr_in_buf = payload;
3685e111ed8SAndrew Rybchenko 	req.emr_in_length = MC_CMD_GET_PHY_STATE_IN_LEN;
3695e111ed8SAndrew Rybchenko 	req.emr_out_buf = payload;
3705e111ed8SAndrew Rybchenko 	req.emr_out_length = MC_CMD_GET_PHY_STATE_OUT_LEN;
3715e111ed8SAndrew Rybchenko 
3725e111ed8SAndrew Rybchenko 	efx_mcdi_execute(enp, &req);
3735e111ed8SAndrew Rybchenko 
3745e111ed8SAndrew Rybchenko 	if (req.emr_rc != 0) {
3755e111ed8SAndrew Rybchenko 		rc = req.emr_rc;
3765e111ed8SAndrew Rybchenko 		goto fail1;
3775e111ed8SAndrew Rybchenko 	}
3785e111ed8SAndrew Rybchenko 
3795e111ed8SAndrew Rybchenko 	if (req.emr_out_length_used < MC_CMD_GET_PHY_STATE_OUT_LEN) {
3805e111ed8SAndrew Rybchenko 		rc = EMSGSIZE;
3815e111ed8SAndrew Rybchenko 		goto fail2;
3825e111ed8SAndrew Rybchenko 	}
3835e111ed8SAndrew Rybchenko 
3845e111ed8SAndrew Rybchenko 	state = MCDI_OUT_DWORD(req, GET_PHY_STATE_OUT_STATE);
3855e111ed8SAndrew Rybchenko 	if (state != MC_CMD_PHY_STATE_OK) {
3865e111ed8SAndrew Rybchenko 		if (state != MC_CMD_PHY_STATE_ZOMBIE)
3875e111ed8SAndrew Rybchenko 			EFSYS_PROBE1(mc_pcol_error, int, state);
3885e111ed8SAndrew Rybchenko 		rc = ENOTACTIVE;
3895e111ed8SAndrew Rybchenko 		goto fail3;
3905e111ed8SAndrew Rybchenko 	}
3915e111ed8SAndrew Rybchenko 
3925e111ed8SAndrew Rybchenko 	return (0);
3935e111ed8SAndrew Rybchenko 
3945e111ed8SAndrew Rybchenko fail3:
3955e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail3);
3965e111ed8SAndrew Rybchenko fail2:
3975e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail2);
3985e111ed8SAndrew Rybchenko fail1:
3995e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
4005e111ed8SAndrew Rybchenko 
4015e111ed8SAndrew Rybchenko 	return (rc);
4025e111ed8SAndrew Rybchenko }
4035e111ed8SAndrew Rybchenko 
4045e111ed8SAndrew Rybchenko 	__checkReturn	efx_rc_t
siena_phy_oui_get(__in efx_nic_t * enp,__out uint32_t * ouip)4055e111ed8SAndrew Rybchenko siena_phy_oui_get(
4065e111ed8SAndrew Rybchenko 	__in		efx_nic_t *enp,
4075e111ed8SAndrew Rybchenko 	__out		uint32_t *ouip)
4085e111ed8SAndrew Rybchenko {
4095e111ed8SAndrew Rybchenko 	_NOTE(ARGUNUSED(enp, ouip))
4105e111ed8SAndrew Rybchenko 
4115e111ed8SAndrew Rybchenko 	return (ENOTSUP);
4125e111ed8SAndrew Rybchenko }
4135e111ed8SAndrew Rybchenko 
4145e111ed8SAndrew Rybchenko #if EFSYS_OPT_PHY_STATS
4155e111ed8SAndrew Rybchenko 
4165e111ed8SAndrew Rybchenko #define	SIENA_SIMPLE_STAT_SET(_vmask, _esmp, _smask, _stat,		\
4175e111ed8SAndrew Rybchenko 			    _mc_record, _efx_record)			\
4185e111ed8SAndrew Rybchenko 	if ((_vmask) & (1ULL << (_mc_record))) {			\
4195e111ed8SAndrew Rybchenko 		(_smask) |= (1ULL << (_efx_record));			\
4205e111ed8SAndrew Rybchenko 		if ((_stat) != NULL && !EFSYS_MEM_IS_NULL(_esmp)) {	\
4215e111ed8SAndrew Rybchenko 			efx_dword_t dword;				\
4225e111ed8SAndrew Rybchenko 			EFSYS_MEM_READD(_esmp, (_mc_record) * 4, &dword);\
4235e111ed8SAndrew Rybchenko 			(_stat)[_efx_record] =				\
4245e111ed8SAndrew Rybchenko 				EFX_DWORD_FIELD(dword, EFX_DWORD_0);	\
4255e111ed8SAndrew Rybchenko 		}							\
4265e111ed8SAndrew Rybchenko 	}
4275e111ed8SAndrew Rybchenko 
4285e111ed8SAndrew Rybchenko #define	SIENA_SIMPLE_STAT_SET2(_vmask, _esmp, _smask, _stat, _record)	\
4295e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET(_vmask, _esmp, _smask, _stat,		\
4305e111ed8SAndrew Rybchenko 			    MC_CMD_ ## _record,				\
4315e111ed8SAndrew Rybchenko 			    EFX_PHY_STAT_ ## _record)
4325e111ed8SAndrew Rybchenko 
4335e111ed8SAndrew Rybchenko 						void
siena_phy_decode_stats(__in efx_nic_t * enp,__in uint32_t vmask,__in_opt efsys_mem_t * esmp,__out_opt uint64_t * smaskp,__inout_ecount_opt (EFX_PHY_NSTATS)uint32_t * stat)4345e111ed8SAndrew Rybchenko siena_phy_decode_stats(
4355e111ed8SAndrew Rybchenko 	__in					efx_nic_t *enp,
4365e111ed8SAndrew Rybchenko 	__in					uint32_t vmask,
4375e111ed8SAndrew Rybchenko 	__in_opt				efsys_mem_t *esmp,
4385e111ed8SAndrew Rybchenko 	__out_opt				uint64_t *smaskp,
4395e111ed8SAndrew Rybchenko 	__inout_ecount_opt(EFX_PHY_NSTATS)	uint32_t *stat)
4405e111ed8SAndrew Rybchenko {
4415e111ed8SAndrew Rybchenko 	uint64_t smask = 0;
4425e111ed8SAndrew Rybchenko 
4435e111ed8SAndrew Rybchenko 	_NOTE(ARGUNUSED(enp))
4445e111ed8SAndrew Rybchenko 
4455e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET2(vmask, esmp, smask, stat, OUI);
4465e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET2(vmask, esmp, smask, stat, PMA_PMD_LINK_UP);
4475e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET2(vmask, esmp, smask, stat, PMA_PMD_RX_FAULT);
4485e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET2(vmask, esmp, smask, stat, PMA_PMD_TX_FAULT);
4495e111ed8SAndrew Rybchenko 
4505e111ed8SAndrew Rybchenko 	if (vmask & (1 << MC_CMD_PMA_PMD_SIGNAL)) {
4515e111ed8SAndrew Rybchenko 		smask |=   ((1ULL << EFX_PHY_STAT_PMA_PMD_SIGNAL_A) |
4525e111ed8SAndrew Rybchenko 			    (1ULL << EFX_PHY_STAT_PMA_PMD_SIGNAL_B) |
4535e111ed8SAndrew Rybchenko 			    (1ULL << EFX_PHY_STAT_PMA_PMD_SIGNAL_C) |
4545e111ed8SAndrew Rybchenko 			    (1ULL << EFX_PHY_STAT_PMA_PMD_SIGNAL_D));
4555e111ed8SAndrew Rybchenko 		if (stat != NULL && esmp != NULL && !EFSYS_MEM_IS_NULL(esmp)) {
4565e111ed8SAndrew Rybchenko 			efx_dword_t dword;
4575e111ed8SAndrew Rybchenko 			uint32_t sig;
4585e111ed8SAndrew Rybchenko 			EFSYS_MEM_READD(esmp, 4 * MC_CMD_PMA_PMD_SIGNAL,
4595e111ed8SAndrew Rybchenko 					&dword);
4605e111ed8SAndrew Rybchenko 			sig = EFX_DWORD_FIELD(dword, EFX_DWORD_0);
4615e111ed8SAndrew Rybchenko 			stat[EFX_PHY_STAT_PMA_PMD_SIGNAL_A] = (sig >> 1) & 1;
4625e111ed8SAndrew Rybchenko 			stat[EFX_PHY_STAT_PMA_PMD_SIGNAL_B] = (sig >> 2) & 1;
4635e111ed8SAndrew Rybchenko 			stat[EFX_PHY_STAT_PMA_PMD_SIGNAL_C] = (sig >> 3) & 1;
4645e111ed8SAndrew Rybchenko 			stat[EFX_PHY_STAT_PMA_PMD_SIGNAL_D] = (sig >> 4) & 1;
4655e111ed8SAndrew Rybchenko 		}
4665e111ed8SAndrew Rybchenko 	}
4675e111ed8SAndrew Rybchenko 
4685e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET(vmask, esmp, smask, stat, MC_CMD_PMA_PMD_SNR_A,
4695e111ed8SAndrew Rybchenko 			    EFX_PHY_STAT_SNR_A);
4705e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET(vmask, esmp, smask, stat, MC_CMD_PMA_PMD_SNR_B,
4715e111ed8SAndrew Rybchenko 			    EFX_PHY_STAT_SNR_B);
4725e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET(vmask, esmp, smask, stat, MC_CMD_PMA_PMD_SNR_C,
4735e111ed8SAndrew Rybchenko 			    EFX_PHY_STAT_SNR_C);
4745e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET(vmask, esmp, smask, stat, MC_CMD_PMA_PMD_SNR_D,
4755e111ed8SAndrew Rybchenko 			    EFX_PHY_STAT_SNR_D);
4765e111ed8SAndrew Rybchenko 
4775e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET2(vmask, esmp, smask, stat, PCS_LINK_UP);
4785e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET2(vmask, esmp, smask, stat, PCS_RX_FAULT);
4795e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET2(vmask, esmp, smask, stat, PCS_TX_FAULT);
4805e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET2(vmask, esmp, smask, stat, PCS_BER);
4815e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET2(vmask, esmp, smask, stat, PCS_BLOCK_ERRORS);
4825e111ed8SAndrew Rybchenko 
4835e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET(vmask, esmp, smask, stat, MC_CMD_PHYXS_LINK_UP,
4845e111ed8SAndrew Rybchenko 			    EFX_PHY_STAT_PHY_XS_LINK_UP);
4855e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET(vmask, esmp, smask, stat, MC_CMD_PHYXS_RX_FAULT,
4865e111ed8SAndrew Rybchenko 			    EFX_PHY_STAT_PHY_XS_RX_FAULT);
4875e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET(vmask, esmp, smask, stat, MC_CMD_PHYXS_TX_FAULT,
4885e111ed8SAndrew Rybchenko 			    EFX_PHY_STAT_PHY_XS_TX_FAULT);
4895e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET(vmask, esmp, smask, stat, MC_CMD_PHYXS_ALIGN,
4905e111ed8SAndrew Rybchenko 			    EFX_PHY_STAT_PHY_XS_ALIGN);
4915e111ed8SAndrew Rybchenko 
4925e111ed8SAndrew Rybchenko 	if (vmask & (1 << MC_CMD_PHYXS_SYNC)) {
4935e111ed8SAndrew Rybchenko 		smask |=   ((1 << EFX_PHY_STAT_PHY_XS_SYNC_A) |
4945e111ed8SAndrew Rybchenko 			    (1 << EFX_PHY_STAT_PHY_XS_SYNC_B) |
4955e111ed8SAndrew Rybchenko 			    (1 << EFX_PHY_STAT_PHY_XS_SYNC_C) |
4965e111ed8SAndrew Rybchenko 			    (1 << EFX_PHY_STAT_PHY_XS_SYNC_D));
4975e111ed8SAndrew Rybchenko 		if (stat != NULL && !EFSYS_MEM_IS_NULL(esmp)) {
4985e111ed8SAndrew Rybchenko 			efx_dword_t dword;
4995e111ed8SAndrew Rybchenko 			uint32_t sync;
5005e111ed8SAndrew Rybchenko 			EFSYS_MEM_READD(esmp, 4 * MC_CMD_PHYXS_SYNC, &dword);
5015e111ed8SAndrew Rybchenko 			sync = EFX_DWORD_FIELD(dword, EFX_DWORD_0);
5025e111ed8SAndrew Rybchenko 			stat[EFX_PHY_STAT_PHY_XS_SYNC_A] = (sync >> 0) & 1;
5035e111ed8SAndrew Rybchenko 			stat[EFX_PHY_STAT_PHY_XS_SYNC_B] = (sync >> 1) & 1;
5045e111ed8SAndrew Rybchenko 			stat[EFX_PHY_STAT_PHY_XS_SYNC_C] = (sync >> 2) & 1;
5055e111ed8SAndrew Rybchenko 			stat[EFX_PHY_STAT_PHY_XS_SYNC_D] = (sync >> 3) & 1;
5065e111ed8SAndrew Rybchenko 		}
5075e111ed8SAndrew Rybchenko 	}
5085e111ed8SAndrew Rybchenko 
5095e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET2(vmask, esmp, smask, stat, AN_LINK_UP);
5105e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET2(vmask, esmp, smask, stat, AN_COMPLETE);
5115e111ed8SAndrew Rybchenko 
5125e111ed8SAndrew Rybchenko 	SIENA_SIMPLE_STAT_SET(vmask, esmp, smask, stat, MC_CMD_CL22_LINK_UP,
5135e111ed8SAndrew Rybchenko 			    EFX_PHY_STAT_CL22EXT_LINK_UP);
5145e111ed8SAndrew Rybchenko 
5155e111ed8SAndrew Rybchenko 	if (smaskp != NULL)
5165e111ed8SAndrew Rybchenko 		*smaskp = smask;
5175e111ed8SAndrew Rybchenko }
5185e111ed8SAndrew Rybchenko 
5195e111ed8SAndrew Rybchenko 	__checkReturn				efx_rc_t
siena_phy_stats_update(__in efx_nic_t * enp,__in efsys_mem_t * esmp,__inout_ecount (EFX_PHY_NSTATS)uint32_t * stat)5205e111ed8SAndrew Rybchenko siena_phy_stats_update(
5215e111ed8SAndrew Rybchenko 	__in					efx_nic_t *enp,
5225e111ed8SAndrew Rybchenko 	__in					efsys_mem_t *esmp,
5235e111ed8SAndrew Rybchenko 	__inout_ecount(EFX_PHY_NSTATS)		uint32_t *stat)
5245e111ed8SAndrew Rybchenko {
5255e111ed8SAndrew Rybchenko 	efx_nic_cfg_t *encp = &(enp->en_nic_cfg);
5265e111ed8SAndrew Rybchenko 	uint32_t vmask = encp->enc_mcdi_phy_stat_mask;
5275e111ed8SAndrew Rybchenko 	uint64_t smask;
5285e111ed8SAndrew Rybchenko 	efx_mcdi_req_t req;
5295e111ed8SAndrew Rybchenko 	EFX_MCDI_DECLARE_BUF(payload, MC_CMD_PHY_STATS_IN_LEN,
5305e111ed8SAndrew Rybchenko 		MC_CMD_PHY_STATS_OUT_DMA_LEN);
5315e111ed8SAndrew Rybchenko 	efx_rc_t rc;
5325e111ed8SAndrew Rybchenko 
5335e111ed8SAndrew Rybchenko 	if ((esmp == NULL) || (EFSYS_MEM_SIZE(esmp) < EFX_PHY_STATS_SIZE)) {
5345e111ed8SAndrew Rybchenko 		rc = EINVAL;
5355e111ed8SAndrew Rybchenko 		goto fail1;
5365e111ed8SAndrew Rybchenko 	}
5375e111ed8SAndrew Rybchenko 
5385e111ed8SAndrew Rybchenko 	req.emr_cmd = MC_CMD_PHY_STATS;
5395e111ed8SAndrew Rybchenko 	req.emr_in_buf = payload;
5405e111ed8SAndrew Rybchenko 	req.emr_in_length = MC_CMD_PHY_STATS_IN_LEN;
5415e111ed8SAndrew Rybchenko 	req.emr_out_buf = payload;
5425e111ed8SAndrew Rybchenko 	req.emr_out_length = MC_CMD_PHY_STATS_OUT_DMA_LEN;
5435e111ed8SAndrew Rybchenko 
5445e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD(req, PHY_STATS_IN_DMA_ADDR_LO,
5455e111ed8SAndrew Rybchenko 			    EFSYS_MEM_ADDR(esmp) & 0xffffffff);
5465e111ed8SAndrew Rybchenko 	MCDI_IN_SET_DWORD(req, PHY_STATS_IN_DMA_ADDR_HI,
5475e111ed8SAndrew Rybchenko 			    EFSYS_MEM_ADDR(esmp) >> 32);
5485e111ed8SAndrew Rybchenko 
5495e111ed8SAndrew Rybchenko 	efx_mcdi_execute(enp, &req);
5505e111ed8SAndrew Rybchenko 
5515e111ed8SAndrew Rybchenko 	if (req.emr_rc != 0) {
5525e111ed8SAndrew Rybchenko 		rc = req.emr_rc;
5535e111ed8SAndrew Rybchenko 		goto fail2;
5545e111ed8SAndrew Rybchenko 	}
5555e111ed8SAndrew Rybchenko 	EFSYS_ASSERT3U(req.emr_out_length, ==, MC_CMD_PHY_STATS_OUT_DMA_LEN);
5565e111ed8SAndrew Rybchenko 
5575e111ed8SAndrew Rybchenko 	siena_phy_decode_stats(enp, vmask, esmp, &smask, stat);
5585e111ed8SAndrew Rybchenko 	EFSYS_ASSERT(smask == encp->enc_phy_stat_mask);
5595e111ed8SAndrew Rybchenko 
5605e111ed8SAndrew Rybchenko 	return (0);
5615e111ed8SAndrew Rybchenko 
5625e111ed8SAndrew Rybchenko fail2:
5635e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail2);
5645e111ed8SAndrew Rybchenko fail1:
5655e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
5665e111ed8SAndrew Rybchenko 
5675e111ed8SAndrew Rybchenko 	return (0);
5685e111ed8SAndrew Rybchenko }
5695e111ed8SAndrew Rybchenko 
5705e111ed8SAndrew Rybchenko #endif	/* EFSYS_OPT_PHY_STATS */
5715e111ed8SAndrew Rybchenko 
5725e111ed8SAndrew Rybchenko #if EFSYS_OPT_BIST
5735e111ed8SAndrew Rybchenko 
5745e111ed8SAndrew Rybchenko 	__checkReturn		efx_rc_t
siena_phy_bist_start(__in efx_nic_t * enp,__in efx_bist_type_t type)5755e111ed8SAndrew Rybchenko siena_phy_bist_start(
5765e111ed8SAndrew Rybchenko 	__in			efx_nic_t *enp,
5775e111ed8SAndrew Rybchenko 	__in			efx_bist_type_t type)
5785e111ed8SAndrew Rybchenko {
5795e111ed8SAndrew Rybchenko 	efx_rc_t rc;
5805e111ed8SAndrew Rybchenko 
5815e111ed8SAndrew Rybchenko 	if ((rc = efx_mcdi_bist_start(enp, type)) != 0)
5825e111ed8SAndrew Rybchenko 		goto fail1;
5835e111ed8SAndrew Rybchenko 
5845e111ed8SAndrew Rybchenko 	return (0);
5855e111ed8SAndrew Rybchenko 
5865e111ed8SAndrew Rybchenko fail1:
5875e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
5885e111ed8SAndrew Rybchenko 
5895e111ed8SAndrew Rybchenko 	return (rc);
5905e111ed8SAndrew Rybchenko }
5915e111ed8SAndrew Rybchenko 
5925e111ed8SAndrew Rybchenko static	__checkReturn		unsigned long
siena_phy_sft9001_bist_status(__in uint16_t code)5935e111ed8SAndrew Rybchenko siena_phy_sft9001_bist_status(
5945e111ed8SAndrew Rybchenko 	__in			uint16_t code)
5955e111ed8SAndrew Rybchenko {
5965e111ed8SAndrew Rybchenko 	switch (code) {
5975e111ed8SAndrew Rybchenko 	case MC_CMD_POLL_BIST_SFT9001_PAIR_BUSY:
5985e111ed8SAndrew Rybchenko 		return (EFX_PHY_CABLE_STATUS_BUSY);
5995e111ed8SAndrew Rybchenko 	case MC_CMD_POLL_BIST_SFT9001_INTER_PAIR_SHORT:
6005e111ed8SAndrew Rybchenko 		return (EFX_PHY_CABLE_STATUS_INTERPAIRSHORT);
6015e111ed8SAndrew Rybchenko 	case MC_CMD_POLL_BIST_SFT9001_INTRA_PAIR_SHORT:
6025e111ed8SAndrew Rybchenko 		return (EFX_PHY_CABLE_STATUS_INTRAPAIRSHORT);
6035e111ed8SAndrew Rybchenko 	case MC_CMD_POLL_BIST_SFT9001_PAIR_OPEN:
6045e111ed8SAndrew Rybchenko 		return (EFX_PHY_CABLE_STATUS_OPEN);
6055e111ed8SAndrew Rybchenko 	case MC_CMD_POLL_BIST_SFT9001_PAIR_OK:
6065e111ed8SAndrew Rybchenko 		return (EFX_PHY_CABLE_STATUS_OK);
6075e111ed8SAndrew Rybchenko 	default:
6085e111ed8SAndrew Rybchenko 		return (EFX_PHY_CABLE_STATUS_INVALID);
6095e111ed8SAndrew Rybchenko 	}
6105e111ed8SAndrew Rybchenko }
6115e111ed8SAndrew Rybchenko 
6125e111ed8SAndrew Rybchenko 	__checkReturn		efx_rc_t
6135e111ed8SAndrew Rybchenko siena_phy_bist_poll(
6145e111ed8SAndrew Rybchenko 	__in			efx_nic_t *enp,
6155e111ed8SAndrew Rybchenko 	__in			efx_bist_type_t type,
6165e111ed8SAndrew Rybchenko 	__out			efx_bist_result_t *resultp,
6175e111ed8SAndrew Rybchenko 	__out_opt __drv_when(count > 0, __notnull)
6185e111ed8SAndrew Rybchenko 	uint32_t *value_maskp,
6195e111ed8SAndrew Rybchenko 	__out_ecount_opt(count)	__drv_when(count > 0, __notnull)
6205e111ed8SAndrew Rybchenko 	unsigned long *valuesp,
6215e111ed8SAndrew Rybchenko 	__in			size_t count)
6225e111ed8SAndrew Rybchenko {
6235e111ed8SAndrew Rybchenko 	efx_nic_cfg_t *encp = &(enp->en_nic_cfg);
6245e111ed8SAndrew Rybchenko 	EFX_MCDI_DECLARE_BUF(payload, MC_CMD_POLL_BIST_IN_LEN,
6255e111ed8SAndrew Rybchenko 		MCDI_CTL_SDU_LEN_MAX);
6265e111ed8SAndrew Rybchenko 	uint32_t value_mask = 0;
6275e111ed8SAndrew Rybchenko 	efx_mcdi_req_t req;
6285e111ed8SAndrew Rybchenko 	uint32_t result;
6295e111ed8SAndrew Rybchenko 	efx_rc_t rc;
6305e111ed8SAndrew Rybchenko 
6315e111ed8SAndrew Rybchenko 	req.emr_cmd = MC_CMD_POLL_BIST;
6325e111ed8SAndrew Rybchenko 	req.emr_in_buf = payload;
6335e111ed8SAndrew Rybchenko 	req.emr_in_length = MC_CMD_POLL_BIST_IN_LEN;
6345e111ed8SAndrew Rybchenko 	req.emr_out_buf = payload;
6355e111ed8SAndrew Rybchenko 	req.emr_out_length = MCDI_CTL_SDU_LEN_MAX;
6365e111ed8SAndrew Rybchenko 
6375e111ed8SAndrew Rybchenko 	efx_mcdi_execute(enp, &req);
6385e111ed8SAndrew Rybchenko 
6395e111ed8SAndrew Rybchenko 	if (req.emr_rc != 0) {
6405e111ed8SAndrew Rybchenko 		rc = req.emr_rc;
6415e111ed8SAndrew Rybchenko 		goto fail1;
6425e111ed8SAndrew Rybchenko 	}
6435e111ed8SAndrew Rybchenko 
6445e111ed8SAndrew Rybchenko 	if (req.emr_out_length_used < MC_CMD_POLL_BIST_OUT_RESULT_OFST + 4) {
6455e111ed8SAndrew Rybchenko 		rc = EMSGSIZE;
6465e111ed8SAndrew Rybchenko 		goto fail2;
6475e111ed8SAndrew Rybchenko 	}
6485e111ed8SAndrew Rybchenko 
6495e111ed8SAndrew Rybchenko 	if (count > 0)
6505e111ed8SAndrew Rybchenko 		(void) memset(valuesp, '\0', count * sizeof (unsigned long));
6515e111ed8SAndrew Rybchenko 
6525e111ed8SAndrew Rybchenko 	result = MCDI_OUT_DWORD(req, POLL_BIST_OUT_RESULT);
6535e111ed8SAndrew Rybchenko 
6545e111ed8SAndrew Rybchenko 	/* Extract PHY specific results */
6555e111ed8SAndrew Rybchenko 	if (result == MC_CMD_POLL_BIST_PASSED &&
6565e111ed8SAndrew Rybchenko 	    encp->enc_phy_type == EFX_PHY_SFT9001B &&
6575e111ed8SAndrew Rybchenko 	    req.emr_out_length_used >= MC_CMD_POLL_BIST_OUT_SFT9001_LEN &&
6585e111ed8SAndrew Rybchenko 	    (type == EFX_BIST_TYPE_PHY_CABLE_SHORT ||
6595e111ed8SAndrew Rybchenko 	    type == EFX_BIST_TYPE_PHY_CABLE_LONG)) {
6605e111ed8SAndrew Rybchenko 		uint16_t word;
6615e111ed8SAndrew Rybchenko 
6625e111ed8SAndrew Rybchenko 		if (count > EFX_BIST_PHY_CABLE_LENGTH_A) {
6635e111ed8SAndrew Rybchenko 			if (valuesp != NULL)
6645e111ed8SAndrew Rybchenko 				valuesp[EFX_BIST_PHY_CABLE_LENGTH_A] =
6655e111ed8SAndrew Rybchenko 				    MCDI_OUT_DWORD(req,
6665e111ed8SAndrew Rybchenko 				    POLL_BIST_OUT_SFT9001_CABLE_LENGTH_A);
6675e111ed8SAndrew Rybchenko 			value_mask |= (1 << EFX_BIST_PHY_CABLE_LENGTH_A);
6685e111ed8SAndrew Rybchenko 		}
6695e111ed8SAndrew Rybchenko 
6705e111ed8SAndrew Rybchenko 		if (count > EFX_BIST_PHY_CABLE_LENGTH_B) {
6715e111ed8SAndrew Rybchenko 			if (valuesp != NULL)
6725e111ed8SAndrew Rybchenko 				valuesp[EFX_BIST_PHY_CABLE_LENGTH_B] =
6735e111ed8SAndrew Rybchenko 				    MCDI_OUT_DWORD(req,
6745e111ed8SAndrew Rybchenko 				    POLL_BIST_OUT_SFT9001_CABLE_LENGTH_B);
6755e111ed8SAndrew Rybchenko 			value_mask |= (1 << EFX_BIST_PHY_CABLE_LENGTH_B);
6765e111ed8SAndrew Rybchenko 		}
6775e111ed8SAndrew Rybchenko 
6785e111ed8SAndrew Rybchenko 		if (count > EFX_BIST_PHY_CABLE_LENGTH_C) {
6795e111ed8SAndrew Rybchenko 			if (valuesp != NULL)
6805e111ed8SAndrew Rybchenko 				valuesp[EFX_BIST_PHY_CABLE_LENGTH_C] =
6815e111ed8SAndrew Rybchenko 				    MCDI_OUT_DWORD(req,
6825e111ed8SAndrew Rybchenko 				    POLL_BIST_OUT_SFT9001_CABLE_LENGTH_C);
6835e111ed8SAndrew Rybchenko 			value_mask |= (1 << EFX_BIST_PHY_CABLE_LENGTH_C);
6845e111ed8SAndrew Rybchenko 		}
6855e111ed8SAndrew Rybchenko 
6865e111ed8SAndrew Rybchenko 		if (count > EFX_BIST_PHY_CABLE_LENGTH_D) {
6875e111ed8SAndrew Rybchenko 			if (valuesp != NULL)
6885e111ed8SAndrew Rybchenko 				valuesp[EFX_BIST_PHY_CABLE_LENGTH_D] =
6895e111ed8SAndrew Rybchenko 				    MCDI_OUT_DWORD(req,
6905e111ed8SAndrew Rybchenko 				    POLL_BIST_OUT_SFT9001_CABLE_LENGTH_D);
6915e111ed8SAndrew Rybchenko 			value_mask |= (1 << EFX_BIST_PHY_CABLE_LENGTH_D);
6925e111ed8SAndrew Rybchenko 		}
6935e111ed8SAndrew Rybchenko 
6945e111ed8SAndrew Rybchenko 		if (count > EFX_BIST_PHY_CABLE_STATUS_A) {
6955e111ed8SAndrew Rybchenko 			if (valuesp != NULL) {
6965e111ed8SAndrew Rybchenko 				word = MCDI_OUT_WORD(req,
6975e111ed8SAndrew Rybchenko 				    POLL_BIST_OUT_SFT9001_CABLE_STATUS_A);
6985e111ed8SAndrew Rybchenko 				valuesp[EFX_BIST_PHY_CABLE_STATUS_A] =
6995e111ed8SAndrew Rybchenko 				    siena_phy_sft9001_bist_status(word);
7005e111ed8SAndrew Rybchenko 			}
7015e111ed8SAndrew Rybchenko 			value_mask |= (1 << EFX_BIST_PHY_CABLE_STATUS_A);
7025e111ed8SAndrew Rybchenko 		}
7035e111ed8SAndrew Rybchenko 
7045e111ed8SAndrew Rybchenko 		if (count > EFX_BIST_PHY_CABLE_STATUS_B) {
7055e111ed8SAndrew Rybchenko 			if (valuesp != NULL) {
7065e111ed8SAndrew Rybchenko 				word = MCDI_OUT_WORD(req,
7075e111ed8SAndrew Rybchenko 				    POLL_BIST_OUT_SFT9001_CABLE_STATUS_B);
7085e111ed8SAndrew Rybchenko 				valuesp[EFX_BIST_PHY_CABLE_STATUS_B] =
7095e111ed8SAndrew Rybchenko 				    siena_phy_sft9001_bist_status(word);
7105e111ed8SAndrew Rybchenko 			}
7115e111ed8SAndrew Rybchenko 			value_mask |= (1 << EFX_BIST_PHY_CABLE_STATUS_B);
7125e111ed8SAndrew Rybchenko 		}
7135e111ed8SAndrew Rybchenko 
7145e111ed8SAndrew Rybchenko 		if (count > EFX_BIST_PHY_CABLE_STATUS_C) {
7155e111ed8SAndrew Rybchenko 			if (valuesp != NULL) {
7165e111ed8SAndrew Rybchenko 				word = MCDI_OUT_WORD(req,
7175e111ed8SAndrew Rybchenko 				    POLL_BIST_OUT_SFT9001_CABLE_STATUS_C);
7185e111ed8SAndrew Rybchenko 				valuesp[EFX_BIST_PHY_CABLE_STATUS_C] =
7195e111ed8SAndrew Rybchenko 				    siena_phy_sft9001_bist_status(word);
7205e111ed8SAndrew Rybchenko 			}
7215e111ed8SAndrew Rybchenko 			value_mask |= (1 << EFX_BIST_PHY_CABLE_STATUS_C);
7225e111ed8SAndrew Rybchenko 		}
7235e111ed8SAndrew Rybchenko 
7245e111ed8SAndrew Rybchenko 		if (count > EFX_BIST_PHY_CABLE_STATUS_D) {
7255e111ed8SAndrew Rybchenko 			if (valuesp != NULL) {
7265e111ed8SAndrew Rybchenko 				word = MCDI_OUT_WORD(req,
7275e111ed8SAndrew Rybchenko 				    POLL_BIST_OUT_SFT9001_CABLE_STATUS_D);
7285e111ed8SAndrew Rybchenko 				valuesp[EFX_BIST_PHY_CABLE_STATUS_D] =
7295e111ed8SAndrew Rybchenko 				    siena_phy_sft9001_bist_status(word);
7305e111ed8SAndrew Rybchenko 			}
7315e111ed8SAndrew Rybchenko 			value_mask |= (1 << EFX_BIST_PHY_CABLE_STATUS_D);
7325e111ed8SAndrew Rybchenko 		}
7335e111ed8SAndrew Rybchenko 
7345e111ed8SAndrew Rybchenko 	} else if (result == MC_CMD_POLL_BIST_FAILED &&
7355e111ed8SAndrew Rybchenko 		    encp->enc_phy_type == EFX_PHY_QLX111V &&
7365e111ed8SAndrew Rybchenko 		    req.emr_out_length >= MC_CMD_POLL_BIST_OUT_MRSFP_LEN &&
7375e111ed8SAndrew Rybchenko 		    count > EFX_BIST_FAULT_CODE) {
7385e111ed8SAndrew Rybchenko 		if (valuesp != NULL)
7395e111ed8SAndrew Rybchenko 			valuesp[EFX_BIST_FAULT_CODE] =
7405e111ed8SAndrew Rybchenko 			    MCDI_OUT_DWORD(req, POLL_BIST_OUT_MRSFP_TEST);
7415e111ed8SAndrew Rybchenko 		value_mask |= 1 << EFX_BIST_FAULT_CODE;
7425e111ed8SAndrew Rybchenko 	}
7435e111ed8SAndrew Rybchenko 
7445e111ed8SAndrew Rybchenko 	if (value_maskp != NULL)
7455e111ed8SAndrew Rybchenko 		*value_maskp = value_mask;
7465e111ed8SAndrew Rybchenko 
7475e111ed8SAndrew Rybchenko 	EFSYS_ASSERT(resultp != NULL);
7485e111ed8SAndrew Rybchenko 	if (result == MC_CMD_POLL_BIST_RUNNING)
7495e111ed8SAndrew Rybchenko 		*resultp = EFX_BIST_RESULT_RUNNING;
7505e111ed8SAndrew Rybchenko 	else if (result == MC_CMD_POLL_BIST_PASSED)
7515e111ed8SAndrew Rybchenko 		*resultp = EFX_BIST_RESULT_PASSED;
7525e111ed8SAndrew Rybchenko 	else
7535e111ed8SAndrew Rybchenko 		*resultp = EFX_BIST_RESULT_FAILED;
7545e111ed8SAndrew Rybchenko 
7555e111ed8SAndrew Rybchenko 	return (0);
7565e111ed8SAndrew Rybchenko 
7575e111ed8SAndrew Rybchenko fail2:
7585e111ed8SAndrew Rybchenko 	EFSYS_PROBE(fail2);
7595e111ed8SAndrew Rybchenko fail1:
7605e111ed8SAndrew Rybchenko 	EFSYS_PROBE1(fail1, efx_rc_t, rc);
7615e111ed8SAndrew Rybchenko 
7625e111ed8SAndrew Rybchenko 	return (rc);
7635e111ed8SAndrew Rybchenko }
7645e111ed8SAndrew Rybchenko 
7655e111ed8SAndrew Rybchenko 			void
siena_phy_bist_stop(__in efx_nic_t * enp,__in efx_bist_type_t type)7665e111ed8SAndrew Rybchenko siena_phy_bist_stop(
7675e111ed8SAndrew Rybchenko 	__in		efx_nic_t *enp,
7685e111ed8SAndrew Rybchenko 	__in		efx_bist_type_t type)
7695e111ed8SAndrew Rybchenko {
7705e111ed8SAndrew Rybchenko 	/* There is no way to stop BIST on Siena */
7715e111ed8SAndrew Rybchenko 	_NOTE(ARGUNUSED(enp, type))
7725e111ed8SAndrew Rybchenko }
7735e111ed8SAndrew Rybchenko 
7745e111ed8SAndrew Rybchenko #endif	/* EFSYS_OPT_BIST */
7755e111ed8SAndrew Rybchenko 
7765e111ed8SAndrew Rybchenko #endif	/* EFSYS_OPT_SIENA */
777