Commit afd888c3 authored by Jiri Pirko's avatar Jiri Pirko Committed by David S. Miller

devlink: make sure driver does not read updated driverinit param before reload

The driverinit param purpose is to serve the driver during init/reload
time to provide a value, either default or set by user.

Make sure that driver does not read value updated by user before the
reload is performed. Hold the new value in a separate struct and switch
it during reload.

Note that this is required to be eventually possible to call
devl_param_driverinit_value_get() without holding instance lock.
Signed-off-by: default avatarJiri Pirko <jiri@nvidia.com>
Reviewed-by: default avatarSimon Horman <simon.horman@corigine.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent fa2f921f
...@@ -489,6 +489,10 @@ struct devlink_param_item { ...@@ -489,6 +489,10 @@ struct devlink_param_item {
const struct devlink_param *param; const struct devlink_param *param;
union devlink_param_value driverinit_value; union devlink_param_value driverinit_value;
bool driverinit_value_valid; bool driverinit_value_valid;
union devlink_param_value driverinit_value_new; /* Not reachable
* until reload.
*/
bool driverinit_value_new_valid;
}; };
enum devlink_param_generic_id { enum devlink_param_generic_id {
......
...@@ -369,6 +369,9 @@ int devlink_reload(struct devlink *devlink, struct net *dest_net, ...@@ -369,6 +369,9 @@ int devlink_reload(struct devlink *devlink, struct net *dest_net,
if (dest_net && !net_eq(dest_net, curr_net)) if (dest_net && !net_eq(dest_net, curr_net))
devlink_reload_netns_change(devlink, curr_net, dest_net); devlink_reload_netns_change(devlink, curr_net, dest_net);
if (action == DEVLINK_RELOAD_ACTION_DRIVER_REINIT)
devlink_params_driverinit_load_new(devlink);
err = devlink->ops->reload_up(devlink, action, limit, actions_performed, extack); err = devlink->ops->reload_up(devlink, action, limit, actions_performed, extack);
devlink_reload_failed_set(devlink, !!err); devlink_reload_failed_set(devlink, !!err);
if (err) if (err)
......
...@@ -189,6 +189,9 @@ static inline bool devlink_reload_supported(const struct devlink_ops *ops) ...@@ -189,6 +189,9 @@ static inline bool devlink_reload_supported(const struct devlink_ops *ops)
return ops->reload_down && ops->reload_up; return ops->reload_down && ops->reload_up;
} }
/* Params */
void devlink_params_driverinit_load_new(struct devlink *devlink);
/* Resources */ /* Resources */
struct devlink_resource; struct devlink_resource;
int devlink_resources_validate(struct devlink *devlink, int devlink_resources_validate(struct devlink *devlink,
......
...@@ -4097,9 +4097,12 @@ static int devlink_nl_param_fill(struct sk_buff *msg, struct devlink *devlink, ...@@ -4097,9 +4097,12 @@ static int devlink_nl_param_fill(struct sk_buff *msg, struct devlink *devlink,
if (!devlink_param_cmode_is_supported(param, i)) if (!devlink_param_cmode_is_supported(param, i))
continue; continue;
if (i == DEVLINK_PARAM_CMODE_DRIVERINIT) { if (i == DEVLINK_PARAM_CMODE_DRIVERINIT) {
if (!param_item->driverinit_value_valid) if (param_item->driverinit_value_new_valid)
return -EOPNOTSUPP; param_value[i] = param_item->driverinit_value_new;
else if (param_item->driverinit_value_valid)
param_value[i] = param_item->driverinit_value; param_value[i] = param_item->driverinit_value;
else
return -EOPNOTSUPP;
} else { } else {
ctx.cmode = i; ctx.cmode = i;
err = devlink_param_get(devlink, param, &ctx); err = devlink_param_get(devlink, param, &ctx);
...@@ -4387,8 +4390,8 @@ static int __devlink_nl_cmd_param_set_doit(struct devlink *devlink, ...@@ -4387,8 +4390,8 @@ static int __devlink_nl_cmd_param_set_doit(struct devlink *devlink,
return -EOPNOTSUPP; return -EOPNOTSUPP;
if (cmode == DEVLINK_PARAM_CMODE_DRIVERINIT) { if (cmode == DEVLINK_PARAM_CMODE_DRIVERINIT) {
param_item->driverinit_value = value; param_item->driverinit_value_new = value;
param_item->driverinit_value_valid = true; param_item->driverinit_value_new_valid = true;
} else { } else {
if (!param->set) if (!param->set)
return -EOPNOTSUPP; return -EOPNOTSUPP;
...@@ -9690,6 +9693,21 @@ void devl_param_driverinit_value_set(struct devlink *devlink, u32 param_id, ...@@ -9690,6 +9693,21 @@ void devl_param_driverinit_value_set(struct devlink *devlink, u32 param_id,
} }
EXPORT_SYMBOL_GPL(devl_param_driverinit_value_set); EXPORT_SYMBOL_GPL(devl_param_driverinit_value_set);
void devlink_params_driverinit_load_new(struct devlink *devlink)
{
struct devlink_param_item *param_item;
list_for_each_entry(param_item, &devlink->param_list, list) {
if (!devlink_param_cmode_is_supported(param_item->param,
DEVLINK_PARAM_CMODE_DRIVERINIT) ||
!param_item->driverinit_value_new_valid)
continue;
param_item->driverinit_value = param_item->driverinit_value_new;
param_item->driverinit_value_valid = true;
param_item->driverinit_value_new_valid = false;
}
}
/** /**
* devl_param_value_changed - notify devlink on a parameter's value * devl_param_value_changed - notify devlink on a parameter's value
* change. Should be called by the driver * change. Should be called by the driver
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment