mirror of
https://source.denx.de/u-boot/u-boot.git
synced 2026-06-02 09:46:37 +03:00
phy: add common PHY properties support
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>
This commit is contained in:
committed by
Jerome Forissier
parent
5bb7ae8921
commit
9eca7fd0d3
@@ -350,4 +350,12 @@ source "drivers/phy/qcom/Kconfig"
|
||||
source "drivers/phy/renesas/Kconfig"
|
||||
source "drivers/phy/starfive/Kconfig"
|
||||
|
||||
config PHY_COMMON_PROPS
|
||||
bool "Common PHY properties support"
|
||||
help
|
||||
Enable support for common PHY properties defined in the device tree,
|
||||
such as rx-polarity and tx-polarity. This provides helpers for PHY
|
||||
drivers to read polarity and other standard PHY properties from the
|
||||
device tree node.
|
||||
|
||||
endmenu
|
||||
|
||||
@@ -43,6 +43,7 @@ obj-$(CONFIG_PHY_NPCM_USB) += phy-npcm-usb.o
|
||||
obj-$(CONFIG_$(PHASE_)PHY_IMX8MQ_USB) += phy-imx8mq-usb.o
|
||||
obj-$(CONFIG_PHY_IMX8M_PCIE) += phy-imx8m-pcie.o
|
||||
obj-$(CONFIG_PHY_XILINX_ZYNQMP) += phy-zynqmp.o
|
||||
obj-$(CONFIG_PHY_COMMON_PROPS) += phy-common-props.o
|
||||
obj-y += cadence/
|
||||
obj-y += ti/
|
||||
obj-y += qcom/
|
||||
|
||||
286
drivers/phy/phy-common-props.c
Normal file
286
drivers/phy/phy-common-props.c
Normal file
@@ -0,0 +1,286 @@
|
||||
// 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);
|
||||
}
|
||||
Reference in New Issue
Block a user