mirror of
https://source.denx.de/u-boot/u-boot.git
synced 2026-06-02 09:46:37 +03:00
Add a new PHY_COMMON_PROPS library that provides helper functions for PHY drivers to read standardized polarity properties from the device tree node: - phy_get_rx_polarity() / phy_get_tx_polarity() - phy_get_manual_rx_polarity() / phy_get_manual_tx_polarity() The dt-bindings/phy/phy.h header with PHY_POL_NORMAL, PHY_POL_INVERT, and PHY_POL_AUTO constants is provided via dts/upstream/include, which is already in the build include path. Ported from Merge tag 'phy-for-7.0': git://git.kernel.org/pub/scm/linux/kernel/git/phy/linux-phy Link: https://git.kernel.org/linus/e7556b59ba65179612bce3fa56bb53d1b4fb20db Signed-off-by: Lucien.Jheng <lucienzx159@gmail.com>
287 lines
9.1 KiB
C
287 lines
9.1 KiB
C
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
/*
|
|
* phy-common-props.c -- Common PHY properties
|
|
*
|
|
* Copyright 2025-2026 NXP
|
|
*/
|
|
#include <dm/ofnode.h>
|
|
#include <log.h>
|
|
#include <malloc.h>
|
|
#include <linux/phy/phy-common-props.h>
|
|
#include <linux/bitops.h>
|
|
#include <linux/err.h>
|
|
#include <linux/errno.h>
|
|
|
|
/**
|
|
* ofnode_count_u32_prop - Count number of u32 elements in a property
|
|
* @node: Device tree node
|
|
* @propname: Property name
|
|
*
|
|
* Return: number of u32 elements, or negative error code
|
|
*/
|
|
static int ofnode_count_u32_prop(ofnode node, const char *propname)
|
|
{
|
|
int size;
|
|
|
|
size = ofnode_read_size(node, propname);
|
|
if (size < 0) {
|
|
pr_debug("%s: property '%s' not found (err=%d)\n",
|
|
__func__, propname, size);
|
|
return size;
|
|
}
|
|
|
|
pr_debug("%s: property '%s' has %zu bytes (%zu elements)\n",
|
|
__func__, propname, (size_t)size, (size_t)(size / sizeof(u32)));
|
|
|
|
return size / sizeof(u32);
|
|
}
|
|
|
|
/**
|
|
* ofnode_get_u32_prop_for_name - Find u32 property by name, or default value
|
|
* @node: Device tree node; if invalid or @props_title is absent, @default_val is used
|
|
* @name: Property name used as lookup key in @names_title (must not be NULL)
|
|
* @props_title: Name of u32 array property holding values
|
|
* @names_title: Name of string array property holding lookup keys
|
|
* @default_val: Default value if @node is invalid, @props_title is absent, or empty
|
|
* @val: Pointer to store the returned value
|
|
*
|
|
* This function retrieves a u32 value from @props_title based on a name lookup
|
|
* in @names_title. The value stored in @val is determined as follows:
|
|
*
|
|
* - If @node is invalid or @props_title is absent: @default_val is used
|
|
* - If @props_title exists but is empty: @default_val is used
|
|
* - If @props_title has exactly one element and @names_title is empty:
|
|
* that element is used
|
|
* - Otherwise: @val is set to the element at the same index where @name is
|
|
* found in @names_title.
|
|
* - If @name is not found, the function looks for a "default" entry in
|
|
* @names_title and uses the corresponding value from @props_title
|
|
*
|
|
* When both @props_title and @names_title are present, they must have the
|
|
* same number of elements (except when @props_title has exactly one element).
|
|
*
|
|
* Return: zero on success, negative error on failure.
|
|
*/
|
|
static int ofnode_get_u32_prop_for_name(ofnode node, const char *name,
|
|
const char *props_title,
|
|
const char *names_title,
|
|
unsigned int default_val,
|
|
unsigned int *val)
|
|
{
|
|
int err, n_props, n_names, idx;
|
|
u32 *props;
|
|
|
|
if (!name) {
|
|
pr_err("Error: Lookup key inside \"%s\" is mandatory\n",
|
|
names_title);
|
|
return -EINVAL;
|
|
}
|
|
|
|
pr_debug("%s: looking up '%s' in props='%s' names='%s' default=%u\n",
|
|
__func__, name, props_title, names_title, default_val);
|
|
|
|
n_props = ofnode_count_u32_prop(node, props_title);
|
|
if (n_props < 0) {
|
|
/* property is absent */
|
|
pr_debug("%s: '%s' is absent, using default value %u\n",
|
|
__func__, props_title, default_val);
|
|
*val = default_val;
|
|
return 0;
|
|
}
|
|
if (n_props == 0) {
|
|
/* property exists but is empty, use default */
|
|
pr_debug("%s: '%s' is empty, using default value %u\n",
|
|
__func__, props_title, default_val);
|
|
*val = default_val;
|
|
return 0;
|
|
}
|
|
|
|
n_names = ofnode_read_string_count(node, names_title);
|
|
|
|
pr_debug("%s: '%s' has %d elements, '%s' has %d entries\n",
|
|
__func__, props_title, n_props, names_title, n_names);
|
|
if (n_names >= 0 && n_props != n_names) {
|
|
pr_err("Error: mismatch between \"%s\" and \"%s\" property count (%d vs %d)\n",
|
|
props_title, names_title, n_props, n_names);
|
|
return -EINVAL;
|
|
}
|
|
|
|
idx = ofnode_stringlist_search(node, names_title, name);
|
|
if (idx >= 0) {
|
|
pr_debug("%s: found '%s' at index %d in '%s'\n",
|
|
__func__, name, idx, names_title);
|
|
} else {
|
|
pr_debug("%s: '%s' not found in '%s', trying 'default'\n",
|
|
__func__, name, names_title);
|
|
idx = ofnode_stringlist_search(node, names_title, "default");
|
|
if (idx >= 0)
|
|
pr_debug("%s: 'default' entry found at index %d\n",
|
|
__func__, idx);
|
|
else
|
|
pr_debug("%s: 'default' entry not found in '%s'\n",
|
|
__func__, names_title);
|
|
}
|
|
/*
|
|
* If the mode name is missing, it can only mean the specified property
|
|
* is the default one for all modes, so reject any other property count
|
|
* than 1.
|
|
*/
|
|
if (idx < 0 && n_props != 1) {
|
|
pr_err("Error: \"%s\" property has %d elements, but cannot find \"%s\" in \"%s\" and there is no default value\n",
|
|
props_title, n_props, name, names_title);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (n_props == 1) {
|
|
pr_debug("%s: single-element '%s', reading directly\n",
|
|
__func__, props_title);
|
|
err = ofnode_read_u32(node, props_title, val);
|
|
if (err) {
|
|
pr_debug("%s: failed to read '%s' (err=%d)\n",
|
|
__func__, props_title, err);
|
|
return err;
|
|
}
|
|
pr_debug("%s: resolved value %u for name '%s' from '%s'\n",
|
|
__func__, *val, name, props_title);
|
|
return 0;
|
|
}
|
|
|
|
/* We implicitly know idx >= 0 here */
|
|
props = calloc(n_props, sizeof(*props));
|
|
if (!props)
|
|
return -ENOMEM;
|
|
|
|
err = ofnode_read_u32_array(node, props_title, props, n_props);
|
|
if (err >= 0) {
|
|
*val = props[idx];
|
|
pr_debug("%s: resolved value %u at index %d for name '%s' from '%s'\n",
|
|
__func__, *val, idx, name, props_title);
|
|
} else {
|
|
pr_debug("%s: failed to read u32 array '%s' (err=%d)\n",
|
|
__func__, props_title, err);
|
|
}
|
|
|
|
free(props);
|
|
|
|
return err;
|
|
}
|
|
|
|
/**
|
|
* phy_get_polarity_for_mode - Get polarity for a specific PHY mode
|
|
* @node: Device tree node
|
|
* @mode_name: The name of the PHY mode to look up
|
|
* @supported: Bit mask of supported polarity values
|
|
* @default_val: Default polarity value if property is missing
|
|
* @polarity_prop: Name of the polarity property
|
|
* @names_prop: Name of the names property
|
|
* @val: Pointer to returned polarity
|
|
*
|
|
* Return: zero on success, negative error on failure.
|
|
*/
|
|
static int phy_get_polarity_for_mode(ofnode node, const char *mode_name,
|
|
unsigned int supported,
|
|
unsigned int default_val,
|
|
const char *polarity_prop,
|
|
const char *names_prop,
|
|
unsigned int *val)
|
|
{
|
|
int err;
|
|
|
|
pr_debug("%s: querying '%s' for mode '%s' (supported=0x%x, default=%u)\n",
|
|
__func__, polarity_prop, mode_name, supported, default_val);
|
|
|
|
err = ofnode_get_u32_prop_for_name(node, mode_name, polarity_prop,
|
|
names_prop, default_val, val);
|
|
if (err) {
|
|
pr_debug("%s: '%s' lookup failed for mode '%s' (err=%d)\n",
|
|
__func__, polarity_prop, mode_name, err);
|
|
return err;
|
|
}
|
|
|
|
pr_debug("%s: '%s' for mode '%s' = %u\n",
|
|
__func__, polarity_prop, mode_name, *val);
|
|
|
|
if (!(supported & BIT(*val))) {
|
|
pr_err("Error: %d is not a supported value for '%s' element '%s'\n",
|
|
*val, polarity_prop, mode_name);
|
|
err = -EOPNOTSUPP;
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
/**
|
|
* phy_get_rx_polarity - Get RX polarity for PHY differential lane
|
|
* @node: Pointer to the PHY's device tree node.
|
|
* @mode_name: The name of the PHY mode to look up.
|
|
* @supported: Bit mask of PHY_POL_NORMAL, PHY_POL_INVERT and PHY_POL_AUTO
|
|
* @default_val: Default polarity value if property is missing
|
|
* @val: Pointer to returned polarity.
|
|
*
|
|
* Return: zero on success, negative error on failure.
|
|
*/
|
|
int phy_get_rx_polarity(ofnode node, const char *mode_name,
|
|
unsigned int supported, unsigned int default_val,
|
|
unsigned int *val)
|
|
{
|
|
return phy_get_polarity_for_mode(node, mode_name, supported,
|
|
default_val, "rx-polarity",
|
|
"rx-polarity-names", val);
|
|
}
|
|
|
|
/**
|
|
* phy_get_tx_polarity - Get TX polarity for PHY differential lane
|
|
* @node: Pointer to the PHY's device tree node.
|
|
* @mode_name: The name of the PHY mode to look up.
|
|
* @supported: Bit mask of PHY_POL_NORMAL, PHY_POL_INVERT and PHY_POL_AUTO
|
|
* @default_val: Default polarity value if property is missing
|
|
* @val: Pointer to returned polarity.
|
|
*
|
|
* Return: zero on success, negative error on failure.
|
|
*/
|
|
int phy_get_tx_polarity(ofnode node, const char *mode_name,
|
|
unsigned int supported, unsigned int default_val,
|
|
unsigned int *val)
|
|
{
|
|
return phy_get_polarity_for_mode(node, mode_name, supported,
|
|
default_val, "tx-polarity",
|
|
"tx-polarity-names", val);
|
|
}
|
|
|
|
/**
|
|
* phy_get_manual_rx_polarity - Get manual RX polarity for PHY differential lane
|
|
* @node: Pointer to the PHY's device tree node.
|
|
* @mode_name: The name of the PHY mode to look up.
|
|
* @val: Pointer to returned polarity.
|
|
*
|
|
* Helper for PHYs which do not support protocols with automatic RX polarity
|
|
* detection and correction.
|
|
*
|
|
* Return: zero on success, negative error on failure.
|
|
*/
|
|
int phy_get_manual_rx_polarity(ofnode node, const char *mode_name,
|
|
unsigned int *val)
|
|
{
|
|
return phy_get_rx_polarity(node, mode_name,
|
|
BIT(PHY_POL_NORMAL) | BIT(PHY_POL_INVERT),
|
|
PHY_POL_NORMAL, val);
|
|
}
|
|
|
|
/**
|
|
* phy_get_manual_tx_polarity - Get manual TX polarity for PHY differential lane
|
|
* @node: Pointer to the PHY's device tree node.
|
|
* @mode_name: The name of the PHY mode to look up.
|
|
* @val: Pointer to returned polarity.
|
|
*
|
|
* Helper for PHYs without any custom default value for the TX polarity.
|
|
*
|
|
* Return: zero on success, negative error on failure.
|
|
*/
|
|
int phy_get_manual_tx_polarity(ofnode node, const char *mode_name,
|
|
unsigned int *val)
|
|
{
|
|
return phy_get_tx_polarity(node, mode_name,
|
|
BIT(PHY_POL_NORMAL) | BIT(PHY_POL_INVERT),
|
|
PHY_POL_NORMAL, val);
|
|
}
|