linux-stable/drivers/net/phy/open_alliance_helpers.c
Oleksij Rempel 9e7c1a9b90 phy: Add Open Alliance helpers for the PHY framework
Introduce helper functions specific to Open Alliance diagnostics,
integrating them into the PHY framework. Currently, these helpers
are limited to 1000BaseT1 specific TDR functionality.

Signed-off-by: Oleksij Rempel <o.rempel@pengutronix.de>
Link: https://patch.msgid.link/20240812073046.1728288-2-o.rempel@pengutronix.de
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
2024-08-16 10:16:26 -07:00

78 lines
2.9 KiB
C

// SPDX-License-Identifier: GPL-2.0+
/*
* open_alliance_helpers.c - OPEN Alliance specific PHY diagnostic helpers
*
* This file contains helper functions for implementing advanced diagnostic
* features as specified by the OPEN Alliance for automotive Ethernet PHYs.
* These helpers include functionality for Time Delay Reflection (TDR), dynamic
* channel quality assessment, and other PHY diagnostics.
*
* For more information on the specifications, refer to the OPEN Alliance
* documentation: https://opensig.org/automotive-ethernet-specifications/
* Currently following specifications are partially or fully implemented:
* - Advanced diagnostic features for 1000BASE-T1 automotive Ethernet PHYs.
* TC12 - advanced PHY features.
* https://opensig.org/wp-content/uploads/2024/03/Advanced_PHY_features_for_automotive_Ethernet_v2.0_fin.pdf
*/
#include <linux/bitfield.h>
#include <linux/ethtool_netlink.h>
#include "open_alliance_helpers.h"
/**
* oa_1000bt1_get_ethtool_cable_result_code - Convert TDR status to ethtool
* result code
* @reg_value: Value read from the TDR register
*
* This function takes a register value from the HDD.TDR register and converts
* the TDR status to the corresponding ethtool cable test result code.
*
* Return: The appropriate ethtool result code based on the TDR status
*/
int oa_1000bt1_get_ethtool_cable_result_code(u16 reg_value)
{
u8 tdr_status = FIELD_GET(OA_1000BT1_HDD_TDR_STATUS_MASK, reg_value);
u8 dist_val = FIELD_GET(OA_1000BT1_HDD_TDR_DISTANCE_MASK, reg_value);
switch (tdr_status) {
case OA_1000BT1_HDD_TDR_STATUS_CABLE_OK:
return ETHTOOL_A_CABLE_RESULT_CODE_OK;
case OA_1000BT1_HDD_TDR_STATUS_OPEN:
return ETHTOOL_A_CABLE_RESULT_CODE_OPEN;
case OA_1000BT1_HDD_TDR_STATUS_SHORT:
return ETHTOOL_A_CABLE_RESULT_CODE_SAME_SHORT;
case OA_1000BT1_HDD_TDR_STATUS_NOISE:
return ETHTOOL_A_CABLE_RESULT_CODE_NOISE;
default:
if (dist_val == OA_1000BT1_HDD_TDR_DISTANCE_RESOLUTION_NOT_POSSIBLE)
return ETHTOOL_A_CABLE_RESULT_CODE_RESOLUTION_NOT_POSSIBLE;
return ETHTOOL_A_CABLE_RESULT_CODE_UNSPEC;
}
}
EXPORT_SYMBOL_GPL(oa_1000bt1_get_ethtool_cable_result_code);
/**
* oa_1000bt1_get_tdr_distance - Get distance to the main fault from TDR
* register value
* @reg_value: Value read from the TDR register
*
* This function takes a register value from the HDD.TDR register and extracts
* the distance to the main fault detected by the TDR feature. The distance is
* measured in centimeters and ranges from 0 to 3100 centimeters. If the
* distance is not available (0x3f), the function returns -ERANGE.
*
* Return: The distance to the main fault in centimeters, or -ERANGE if the
* resolution is not possible.
*/
int oa_1000bt1_get_tdr_distance(u16 reg_value)
{
u8 dist_val = FIELD_GET(OA_1000BT1_HDD_TDR_DISTANCE_MASK, reg_value);
if (dist_val == OA_1000BT1_HDD_TDR_DISTANCE_RESOLUTION_NOT_POSSIBLE)
return -ERANGE;
return dist_val * 100;
}
EXPORT_SYMBOL_GPL(oa_1000bt1_get_tdr_distance);