mirror of https://github.com/torvalds/linux.git
210 lines
6.7 KiB
C
210 lines
6.7 KiB
C
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
/*
|
|
* phy-common-props.c -- Common PHY properties
|
|
*
|
|
* Copyright 2025-2026 NXP
|
|
*/
|
|
#include <linux/export.h>
|
|
#include <linux/fwnode.h>
|
|
#include <linux/phy/phy-common-props.h>
|
|
#include <linux/printk.h>
|
|
#include <linux/property.h>
|
|
#include <linux/slab.h>
|
|
|
|
/**
|
|
* fwnode_get_u32_prop_for_name - Find u32 property by name, or default value
|
|
* @fwnode: Pointer to firmware node, or NULL to use @default_val
|
|
* @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 @fwnode is NULL or @props_title is 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 @fwnode is NULL or @props_title 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 fwnode_get_u32_prop_for_name(struct fwnode_handle *fwnode,
|
|
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("Lookup key inside \"%s\" is mandatory\n", names_title);
|
|
return -EINVAL;
|
|
}
|
|
|
|
n_props = fwnode_property_count_u32(fwnode, props_title);
|
|
if (n_props <= 0) {
|
|
/* fwnode is NULL, or is missing requested property */
|
|
*val = default_val;
|
|
return 0;
|
|
}
|
|
|
|
n_names = fwnode_property_string_array_count(fwnode, names_title);
|
|
if (n_names >= 0 && n_props != n_names) {
|
|
pr_err("%pfw mismatch between \"%s\" and \"%s\" property count (%d vs %d)\n",
|
|
fwnode, props_title, names_title, n_props, n_names);
|
|
return -EINVAL;
|
|
}
|
|
|
|
idx = fwnode_property_match_string(fwnode, names_title, name);
|
|
if (idx < 0)
|
|
idx = fwnode_property_match_string(fwnode, names_title, "default");
|
|
/*
|
|
* 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("%pfw \"%s \" property has %d elements, but cannot find \"%s\" in \"%s\" and there is no default value\n",
|
|
fwnode, props_title, n_props, name, names_title);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (n_props == 1) {
|
|
err = fwnode_property_read_u32(fwnode, props_title, val);
|
|
if (err)
|
|
return err;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/* We implicitly know idx >= 0 here */
|
|
props = kcalloc(n_props, sizeof(*props), GFP_KERNEL);
|
|
if (!props)
|
|
return -ENOMEM;
|
|
|
|
err = fwnode_property_read_u32_array(fwnode, props_title, props, n_props);
|
|
if (err >= 0)
|
|
*val = props[idx];
|
|
|
|
kfree(props);
|
|
|
|
return err;
|
|
}
|
|
|
|
static int phy_get_polarity_for_mode(struct fwnode_handle *fwnode,
|
|
const char *mode_name,
|
|
unsigned int supported,
|
|
unsigned int default_val,
|
|
const char *polarity_prop,
|
|
const char *names_prop,
|
|
unsigned int *val)
|
|
{
|
|
int err;
|
|
|
|
err = fwnode_get_u32_prop_for_name(fwnode, mode_name, polarity_prop,
|
|
names_prop, default_val, val);
|
|
if (err)
|
|
return err;
|
|
|
|
if (!(supported & BIT(*val))) {
|
|
pr_err("%d is not a supported value for %pfw '%s' element '%s'\n",
|
|
*val, fwnode, polarity_prop, mode_name);
|
|
err = -EOPNOTSUPP;
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
/**
|
|
* phy_get_rx_polarity - Get RX polarity for PHY differential lane
|
|
* @fwnode: Pointer to the PHY's firmware 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 __must_check phy_get_rx_polarity(struct fwnode_handle *fwnode,
|
|
const char *mode_name,
|
|
unsigned int supported,
|
|
unsigned int default_val,
|
|
unsigned int *val)
|
|
{
|
|
return phy_get_polarity_for_mode(fwnode, mode_name, supported,
|
|
default_val, "rx-polarity",
|
|
"rx-polarity-names", val);
|
|
}
|
|
EXPORT_SYMBOL_GPL(phy_get_rx_polarity);
|
|
|
|
/**
|
|
* phy_get_tx_polarity - Get TX polarity for PHY differential lane
|
|
* @fwnode: Pointer to the PHY's firmware node.
|
|
* @mode_name: The name of the PHY mode to look up.
|
|
* @supported: Bit mask of PHY_POL_NORMAL and PHY_POL_INVERT
|
|
* @default_val: Default polarity value if property is missing
|
|
* @val: Pointer to returned polarity.
|
|
*
|
|
* Return: zero on success, negative error on failure.
|
|
*/
|
|
int __must_check phy_get_tx_polarity(struct fwnode_handle *fwnode,
|
|
const char *mode_name, unsigned int supported,
|
|
unsigned int default_val, unsigned int *val)
|
|
{
|
|
return phy_get_polarity_for_mode(fwnode, mode_name, supported,
|
|
default_val, "tx-polarity",
|
|
"tx-polarity-names", val);
|
|
}
|
|
EXPORT_SYMBOL_GPL(phy_get_tx_polarity);
|
|
|
|
/**
|
|
* phy_get_manual_rx_polarity - Get manual RX polarity for PHY differential lane
|
|
* @fwnode: Pointer to the PHY's firmware 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 __must_check phy_get_manual_rx_polarity(struct fwnode_handle *fwnode,
|
|
const char *mode_name,
|
|
unsigned int *val)
|
|
{
|
|
return phy_get_rx_polarity(fwnode, mode_name,
|
|
BIT(PHY_POL_NORMAL) | BIT(PHY_POL_INVERT),
|
|
PHY_POL_NORMAL, val);
|
|
}
|
|
EXPORT_SYMBOL_GPL(phy_get_manual_rx_polarity);
|
|
|
|
/**
|
|
* phy_get_manual_tx_polarity - Get manual TX polarity for PHY differential lane
|
|
* @fwnode: Pointer to the PHY's firmware 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 __must_check phy_get_manual_tx_polarity(struct fwnode_handle *fwnode,
|
|
const char *mode_name,
|
|
unsigned int *val)
|
|
{
|
|
return phy_get_tx_polarity(fwnode, mode_name,
|
|
BIT(PHY_POL_NORMAL) | BIT(PHY_POL_INVERT),
|
|
PHY_POL_NORMAL, val);
|
|
}
|
|
EXPORT_SYMBOL_GPL(phy_get_manual_tx_polarity);
|