242 lines
5.5 KiB
C
242 lines
5.5 KiB
C
// SPDX-License-Identifier: GPL-2.0
|
|
/* Marvell RVU PF/VF Netdev Devlink
|
|
*
|
|
* Copyright (C) 2021 Marvell.
|
|
*/
|
|
|
|
#include "otx2_common.h"
|
|
|
|
/* Devlink Params APIs */
|
|
static int otx2_dl_mcam_count_validate(struct devlink *devlink, u32 id,
|
|
union devlink_param_value val,
|
|
struct netlink_ext_ack *extack)
|
|
{
|
|
struct otx2_devlink *otx2_dl = devlink_priv(devlink);
|
|
struct otx2_nic *pfvf = otx2_dl->pfvf;
|
|
struct otx2_flow_config *flow_cfg;
|
|
|
|
if (!pfvf->flow_cfg) {
|
|
NL_SET_ERR_MSG_MOD(extack,
|
|
"pfvf->flow_cfg not initialized");
|
|
return -EINVAL;
|
|
}
|
|
|
|
flow_cfg = pfvf->flow_cfg;
|
|
if (flow_cfg && flow_cfg->nr_flows) {
|
|
NL_SET_ERR_MSG_MOD(extack,
|
|
"Cannot modify count when there are active rules");
|
|
return -EINVAL;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int otx2_dl_mcam_count_set(struct devlink *devlink, u32 id,
|
|
struct devlink_param_gset_ctx *ctx,
|
|
struct netlink_ext_ack *extack)
|
|
{
|
|
struct otx2_devlink *otx2_dl = devlink_priv(devlink);
|
|
struct otx2_nic *pfvf = otx2_dl->pfvf;
|
|
|
|
if (!pfvf->flow_cfg)
|
|
return 0;
|
|
|
|
otx2_alloc_mcam_entries(pfvf, ctx->val.vu16);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int otx2_dl_mcam_count_get(struct devlink *devlink, u32 id,
|
|
struct devlink_param_gset_ctx *ctx)
|
|
{
|
|
struct otx2_devlink *otx2_dl = devlink_priv(devlink);
|
|
struct otx2_nic *pfvf = otx2_dl->pfvf;
|
|
struct otx2_flow_config *flow_cfg;
|
|
|
|
if (!pfvf->flow_cfg) {
|
|
ctx->val.vu16 = 0;
|
|
return 0;
|
|
}
|
|
|
|
flow_cfg = pfvf->flow_cfg;
|
|
ctx->val.vu16 = flow_cfg->max_flows;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int otx2_dl_ucast_flt_cnt_set(struct devlink *devlink, u32 id,
|
|
struct devlink_param_gset_ctx *ctx,
|
|
struct netlink_ext_ack *extack)
|
|
{
|
|
struct otx2_devlink *otx2_dl = devlink_priv(devlink);
|
|
struct otx2_nic *pfvf = otx2_dl->pfvf;
|
|
int err;
|
|
|
|
pfvf->flow_cfg->ucast_flt_cnt = ctx->val.vu8;
|
|
|
|
otx2_mcam_flow_del(pfvf);
|
|
err = otx2_mcam_entry_init(pfvf);
|
|
if (err)
|
|
return err;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int otx2_dl_ucast_flt_cnt_get(struct devlink *devlink, u32 id,
|
|
struct devlink_param_gset_ctx *ctx)
|
|
{
|
|
struct otx2_devlink *otx2_dl = devlink_priv(devlink);
|
|
struct otx2_nic *pfvf = otx2_dl->pfvf;
|
|
|
|
ctx->val.vu8 = pfvf->flow_cfg ? pfvf->flow_cfg->ucast_flt_cnt : 0;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int otx2_dl_ucast_flt_cnt_validate(struct devlink *devlink, u32 id,
|
|
union devlink_param_value val,
|
|
struct netlink_ext_ack *extack)
|
|
{
|
|
struct otx2_devlink *otx2_dl = devlink_priv(devlink);
|
|
struct otx2_nic *pfvf = otx2_dl->pfvf;
|
|
|
|
/* Check for UNICAST filter support*/
|
|
if (!(pfvf->flags & OTX2_FLAG_UCAST_FLTR_SUPPORT)) {
|
|
NL_SET_ERR_MSG_MOD(extack,
|
|
"Unicast filter not enabled");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (!pfvf->flow_cfg) {
|
|
NL_SET_ERR_MSG_MOD(extack,
|
|
"pfvf->flow_cfg not initialized");
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (pfvf->flow_cfg->nr_flows) {
|
|
NL_SET_ERR_MSG_MOD(extack,
|
|
"Cannot modify count when there are active rules");
|
|
return -EINVAL;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
enum otx2_dl_param_id {
|
|
OTX2_DEVLINK_PARAM_ID_BASE = DEVLINK_PARAM_GENERIC_ID_MAX,
|
|
OTX2_DEVLINK_PARAM_ID_MCAM_COUNT,
|
|
OTX2_DEVLINK_PARAM_ID_UCAST_FLT_CNT,
|
|
};
|
|
|
|
static const struct devlink_param otx2_dl_params[] = {
|
|
DEVLINK_PARAM_DRIVER(OTX2_DEVLINK_PARAM_ID_MCAM_COUNT,
|
|
"mcam_count", DEVLINK_PARAM_TYPE_U16,
|
|
BIT(DEVLINK_PARAM_CMODE_RUNTIME),
|
|
otx2_dl_mcam_count_get, otx2_dl_mcam_count_set,
|
|
otx2_dl_mcam_count_validate),
|
|
DEVLINK_PARAM_DRIVER(OTX2_DEVLINK_PARAM_ID_UCAST_FLT_CNT,
|
|
"unicast_filter_count", DEVLINK_PARAM_TYPE_U8,
|
|
BIT(DEVLINK_PARAM_CMODE_RUNTIME),
|
|
otx2_dl_ucast_flt_cnt_get, otx2_dl_ucast_flt_cnt_set,
|
|
otx2_dl_ucast_flt_cnt_validate),
|
|
};
|
|
|
|
#ifdef CONFIG_RVU_ESWITCH
|
|
static int otx2_devlink_eswitch_mode_get(struct devlink *devlink, u16 *mode)
|
|
{
|
|
struct otx2_devlink *otx2_dl = devlink_priv(devlink);
|
|
struct otx2_nic *pfvf = otx2_dl->pfvf;
|
|
|
|
if (!otx2_rep_dev(pfvf->pdev))
|
|
return -EOPNOTSUPP;
|
|
|
|
*mode = pfvf->esw_mode;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int otx2_devlink_eswitch_mode_set(struct devlink *devlink, u16 mode,
|
|
struct netlink_ext_ack *extack)
|
|
{
|
|
struct otx2_devlink *otx2_dl = devlink_priv(devlink);
|
|
struct otx2_nic *pfvf = otx2_dl->pfvf;
|
|
int ret = 0;
|
|
|
|
if (!otx2_rep_dev(pfvf->pdev))
|
|
return -EOPNOTSUPP;
|
|
|
|
if (pfvf->esw_mode == mode)
|
|
return 0;
|
|
|
|
switch (mode) {
|
|
case DEVLINK_ESWITCH_MODE_LEGACY:
|
|
rvu_rep_destroy(pfvf);
|
|
break;
|
|
case DEVLINK_ESWITCH_MODE_SWITCHDEV:
|
|
ret = rvu_rep_create(pfvf, extack);
|
|
break;
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (!ret)
|
|
pfvf->esw_mode = mode;
|
|
|
|
return ret;
|
|
}
|
|
#endif
|
|
|
|
static const struct devlink_ops otx2_devlink_ops = {
|
|
#ifdef CONFIG_RVU_ESWITCH
|
|
.eswitch_mode_get = otx2_devlink_eswitch_mode_get,
|
|
.eswitch_mode_set = otx2_devlink_eswitch_mode_set,
|
|
#endif
|
|
};
|
|
|
|
int otx2_register_dl(struct otx2_nic *pfvf)
|
|
{
|
|
struct otx2_devlink *otx2_dl;
|
|
struct devlink *dl;
|
|
int err;
|
|
|
|
dl = devlink_alloc(&otx2_devlink_ops,
|
|
sizeof(struct otx2_devlink), pfvf->dev);
|
|
if (!dl) {
|
|
dev_warn(pfvf->dev, "devlink_alloc failed\n");
|
|
return -ENOMEM;
|
|
}
|
|
|
|
otx2_dl = devlink_priv(dl);
|
|
otx2_dl->dl = dl;
|
|
otx2_dl->pfvf = pfvf;
|
|
pfvf->dl = otx2_dl;
|
|
|
|
err = devlink_params_register(dl, otx2_dl_params,
|
|
ARRAY_SIZE(otx2_dl_params));
|
|
if (err) {
|
|
dev_err(pfvf->dev,
|
|
"devlink params register failed with error %d", err);
|
|
goto err_dl;
|
|
}
|
|
|
|
devlink_register(dl);
|
|
return 0;
|
|
|
|
err_dl:
|
|
devlink_free(dl);
|
|
return err;
|
|
}
|
|
EXPORT_SYMBOL(otx2_register_dl);
|
|
|
|
void otx2_unregister_dl(struct otx2_nic *pfvf)
|
|
{
|
|
struct otx2_devlink *otx2_dl = pfvf->dl;
|
|
struct devlink *dl = otx2_dl->dl;
|
|
|
|
devlink_unregister(dl);
|
|
devlink_params_unregister(dl, otx2_dl_params,
|
|
ARRAY_SIZE(otx2_dl_params));
|
|
devlink_free(dl);
|
|
}
|
|
EXPORT_SYMBOL(otx2_unregister_dl);
|