mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/stable/linux.git
synced 2025-01-04 12:16:41 +00:00
9e7c1a9b90
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>
78 lines
2.9 KiB
C
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);
|