8957261cd8
The ETHTOOL_A_PLCA_ENABLED data type is u8. But while parsing the value from the attribute, nla_get_u32() is used in the plca_update_sint() function instead of nla_get_u8(). So plca_cfg.enabled variable is updated with some garbage value instead of 0 or 1 and always enables plca even though plca is disabled through ethtool application. This bug has been fixed by parsing the values based on the attributes type in the policy. Fixes: 8580e16c28f3 ("net/ethtool: add netlink interface for the PLCA RS") Signed-off-by: Parthiban Veerasooran <Parthiban.Veerasooran@microchip.com> Reviewed-by: Andrew Lunn <andrew@lunn.ch> Link: https://lore.kernel.org/r/20230908044548.5878-1-Parthiban.Veerasooran@microchip.com Signed-off-by: Jakub Kicinski <kuba@kernel.org>
262 lines
7.4 KiB
C
262 lines
7.4 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
|
|
#include <linux/phy.h>
|
|
#include <linux/ethtool_netlink.h>
|
|
|
|
#include "netlink.h"
|
|
#include "common.h"
|
|
|
|
struct plca_req_info {
|
|
struct ethnl_req_info base;
|
|
};
|
|
|
|
struct plca_reply_data {
|
|
struct ethnl_reply_data base;
|
|
struct phy_plca_cfg plca_cfg;
|
|
struct phy_plca_status plca_st;
|
|
};
|
|
|
|
// Helpers ------------------------------------------------------------------ //
|
|
|
|
#define PLCA_REPDATA(__reply_base) \
|
|
container_of(__reply_base, struct plca_reply_data, base)
|
|
|
|
// PLCA get configuration message ------------------------------------------- //
|
|
|
|
const struct nla_policy ethnl_plca_get_cfg_policy[] = {
|
|
[ETHTOOL_A_PLCA_HEADER] =
|
|
NLA_POLICY_NESTED(ethnl_header_policy),
|
|
};
|
|
|
|
static void plca_update_sint(int *dst, struct nlattr **tb, u32 attrid,
|
|
bool *mod)
|
|
{
|
|
const struct nlattr *attr = tb[attrid];
|
|
|
|
if (!attr ||
|
|
WARN_ON_ONCE(attrid >= ARRAY_SIZE(ethnl_plca_set_cfg_policy)))
|
|
return;
|
|
|
|
switch (ethnl_plca_set_cfg_policy[attrid].type) {
|
|
case NLA_U8:
|
|
*dst = nla_get_u8(attr);
|
|
break;
|
|
case NLA_U32:
|
|
*dst = nla_get_u32(attr);
|
|
break;
|
|
default:
|
|
WARN_ON_ONCE(1);
|
|
}
|
|
|
|
*mod = true;
|
|
}
|
|
|
|
static int plca_get_cfg_prepare_data(const struct ethnl_req_info *req_base,
|
|
struct ethnl_reply_data *reply_base,
|
|
const struct genl_info *info)
|
|
{
|
|
struct plca_reply_data *data = PLCA_REPDATA(reply_base);
|
|
struct net_device *dev = reply_base->dev;
|
|
const struct ethtool_phy_ops *ops;
|
|
int ret;
|
|
|
|
// check that the PHY device is available and connected
|
|
if (!dev->phydev) {
|
|
ret = -EOPNOTSUPP;
|
|
goto out;
|
|
}
|
|
|
|
// note: rtnl_lock is held already by ethnl_default_doit
|
|
ops = ethtool_phy_ops;
|
|
if (!ops || !ops->get_plca_cfg) {
|
|
ret = -EOPNOTSUPP;
|
|
goto out;
|
|
}
|
|
|
|
ret = ethnl_ops_begin(dev);
|
|
if (ret < 0)
|
|
goto out;
|
|
|
|
memset(&data->plca_cfg, 0xff,
|
|
sizeof_field(struct plca_reply_data, plca_cfg));
|
|
|
|
ret = ops->get_plca_cfg(dev->phydev, &data->plca_cfg);
|
|
ethnl_ops_complete(dev);
|
|
|
|
out:
|
|
return ret;
|
|
}
|
|
|
|
static int plca_get_cfg_reply_size(const struct ethnl_req_info *req_base,
|
|
const struct ethnl_reply_data *reply_base)
|
|
{
|
|
return nla_total_size(sizeof(u16)) + /* _VERSION */
|
|
nla_total_size(sizeof(u8)) + /* _ENABLED */
|
|
nla_total_size(sizeof(u32)) + /* _NODE_CNT */
|
|
nla_total_size(sizeof(u32)) + /* _NODE_ID */
|
|
nla_total_size(sizeof(u32)) + /* _TO_TIMER */
|
|
nla_total_size(sizeof(u32)) + /* _BURST_COUNT */
|
|
nla_total_size(sizeof(u32)); /* _BURST_TIMER */
|
|
}
|
|
|
|
static int plca_get_cfg_fill_reply(struct sk_buff *skb,
|
|
const struct ethnl_req_info *req_base,
|
|
const struct ethnl_reply_data *reply_base)
|
|
{
|
|
const struct plca_reply_data *data = PLCA_REPDATA(reply_base);
|
|
const struct phy_plca_cfg *plca = &data->plca_cfg;
|
|
|
|
if ((plca->version >= 0 &&
|
|
nla_put_u16(skb, ETHTOOL_A_PLCA_VERSION, plca->version)) ||
|
|
(plca->enabled >= 0 &&
|
|
nla_put_u8(skb, ETHTOOL_A_PLCA_ENABLED, !!plca->enabled)) ||
|
|
(plca->node_id >= 0 &&
|
|
nla_put_u32(skb, ETHTOOL_A_PLCA_NODE_ID, plca->node_id)) ||
|
|
(plca->node_cnt >= 0 &&
|
|
nla_put_u32(skb, ETHTOOL_A_PLCA_NODE_CNT, plca->node_cnt)) ||
|
|
(plca->to_tmr >= 0 &&
|
|
nla_put_u32(skb, ETHTOOL_A_PLCA_TO_TMR, plca->to_tmr)) ||
|
|
(plca->burst_cnt >= 0 &&
|
|
nla_put_u32(skb, ETHTOOL_A_PLCA_BURST_CNT, plca->burst_cnt)) ||
|
|
(plca->burst_tmr >= 0 &&
|
|
nla_put_u32(skb, ETHTOOL_A_PLCA_BURST_TMR, plca->burst_tmr)))
|
|
return -EMSGSIZE;
|
|
|
|
return 0;
|
|
};
|
|
|
|
// PLCA set configuration message ------------------------------------------- //
|
|
|
|
const struct nla_policy ethnl_plca_set_cfg_policy[] = {
|
|
[ETHTOOL_A_PLCA_HEADER] =
|
|
NLA_POLICY_NESTED(ethnl_header_policy),
|
|
[ETHTOOL_A_PLCA_ENABLED] = NLA_POLICY_MAX(NLA_U8, 1),
|
|
[ETHTOOL_A_PLCA_NODE_ID] = NLA_POLICY_MAX(NLA_U32, 255),
|
|
[ETHTOOL_A_PLCA_NODE_CNT] = NLA_POLICY_RANGE(NLA_U32, 1, 255),
|
|
[ETHTOOL_A_PLCA_TO_TMR] = NLA_POLICY_MAX(NLA_U32, 255),
|
|
[ETHTOOL_A_PLCA_BURST_CNT] = NLA_POLICY_MAX(NLA_U32, 255),
|
|
[ETHTOOL_A_PLCA_BURST_TMR] = NLA_POLICY_MAX(NLA_U32, 255),
|
|
};
|
|
|
|
static int
|
|
ethnl_set_plca(struct ethnl_req_info *req_info, struct genl_info *info)
|
|
{
|
|
struct net_device *dev = req_info->dev;
|
|
const struct ethtool_phy_ops *ops;
|
|
struct nlattr **tb = info->attrs;
|
|
struct phy_plca_cfg plca_cfg;
|
|
bool mod = false;
|
|
int ret;
|
|
|
|
// check that the PHY device is available and connected
|
|
if (!dev->phydev)
|
|
return -EOPNOTSUPP;
|
|
|
|
ops = ethtool_phy_ops;
|
|
if (!ops || !ops->set_plca_cfg)
|
|
return -EOPNOTSUPP;
|
|
|
|
memset(&plca_cfg, 0xff, sizeof(plca_cfg));
|
|
plca_update_sint(&plca_cfg.enabled, tb, ETHTOOL_A_PLCA_ENABLED, &mod);
|
|
plca_update_sint(&plca_cfg.node_id, tb, ETHTOOL_A_PLCA_NODE_ID, &mod);
|
|
plca_update_sint(&plca_cfg.node_cnt, tb, ETHTOOL_A_PLCA_NODE_CNT, &mod);
|
|
plca_update_sint(&plca_cfg.to_tmr, tb, ETHTOOL_A_PLCA_TO_TMR, &mod);
|
|
plca_update_sint(&plca_cfg.burst_cnt, tb, ETHTOOL_A_PLCA_BURST_CNT,
|
|
&mod);
|
|
plca_update_sint(&plca_cfg.burst_tmr, tb, ETHTOOL_A_PLCA_BURST_TMR,
|
|
&mod);
|
|
if (!mod)
|
|
return 0;
|
|
|
|
ret = ops->set_plca_cfg(dev->phydev, &plca_cfg, info->extack);
|
|
return ret < 0 ? ret : 1;
|
|
}
|
|
|
|
const struct ethnl_request_ops ethnl_plca_cfg_request_ops = {
|
|
.request_cmd = ETHTOOL_MSG_PLCA_GET_CFG,
|
|
.reply_cmd = ETHTOOL_MSG_PLCA_GET_CFG_REPLY,
|
|
.hdr_attr = ETHTOOL_A_PLCA_HEADER,
|
|
.req_info_size = sizeof(struct plca_req_info),
|
|
.reply_data_size = sizeof(struct plca_reply_data),
|
|
|
|
.prepare_data = plca_get_cfg_prepare_data,
|
|
.reply_size = plca_get_cfg_reply_size,
|
|
.fill_reply = plca_get_cfg_fill_reply,
|
|
|
|
.set = ethnl_set_plca,
|
|
.set_ntf_cmd = ETHTOOL_MSG_PLCA_NTF,
|
|
};
|
|
|
|
// PLCA get status message -------------------------------------------------- //
|
|
|
|
const struct nla_policy ethnl_plca_get_status_policy[] = {
|
|
[ETHTOOL_A_PLCA_HEADER] =
|
|
NLA_POLICY_NESTED(ethnl_header_policy),
|
|
};
|
|
|
|
static int plca_get_status_prepare_data(const struct ethnl_req_info *req_base,
|
|
struct ethnl_reply_data *reply_base,
|
|
const struct genl_info *info)
|
|
{
|
|
struct plca_reply_data *data = PLCA_REPDATA(reply_base);
|
|
struct net_device *dev = reply_base->dev;
|
|
const struct ethtool_phy_ops *ops;
|
|
int ret;
|
|
|
|
// check that the PHY device is available and connected
|
|
if (!dev->phydev) {
|
|
ret = -EOPNOTSUPP;
|
|
goto out;
|
|
}
|
|
|
|
// note: rtnl_lock is held already by ethnl_default_doit
|
|
ops = ethtool_phy_ops;
|
|
if (!ops || !ops->get_plca_status) {
|
|
ret = -EOPNOTSUPP;
|
|
goto out;
|
|
}
|
|
|
|
ret = ethnl_ops_begin(dev);
|
|
if (ret < 0)
|
|
goto out;
|
|
|
|
memset(&data->plca_st, 0xff,
|
|
sizeof_field(struct plca_reply_data, plca_st));
|
|
|
|
ret = ops->get_plca_status(dev->phydev, &data->plca_st);
|
|
ethnl_ops_complete(dev);
|
|
out:
|
|
return ret;
|
|
}
|
|
|
|
static int plca_get_status_reply_size(const struct ethnl_req_info *req_base,
|
|
const struct ethnl_reply_data *reply_base)
|
|
{
|
|
return nla_total_size(sizeof(u8)); /* _STATUS */
|
|
}
|
|
|
|
static int plca_get_status_fill_reply(struct sk_buff *skb,
|
|
const struct ethnl_req_info *req_base,
|
|
const struct ethnl_reply_data *reply_base)
|
|
{
|
|
const struct plca_reply_data *data = PLCA_REPDATA(reply_base);
|
|
const u8 status = data->plca_st.pst;
|
|
|
|
if (nla_put_u8(skb, ETHTOOL_A_PLCA_STATUS, !!status))
|
|
return -EMSGSIZE;
|
|
|
|
return 0;
|
|
};
|
|
|
|
const struct ethnl_request_ops ethnl_plca_status_request_ops = {
|
|
.request_cmd = ETHTOOL_MSG_PLCA_GET_STATUS,
|
|
.reply_cmd = ETHTOOL_MSG_PLCA_GET_STATUS_REPLY,
|
|
.hdr_attr = ETHTOOL_A_PLCA_HEADER,
|
|
.req_info_size = sizeof(struct plca_req_info),
|
|
.reply_data_size = sizeof(struct plca_reply_data),
|
|
|
|
.prepare_data = plca_get_status_prepare_data,
|
|
.reply_size = plca_get_status_reply_size,
|
|
.fill_reply = plca_get_status_fill_reply,
|
|
};
|