Commit 52cc5f3a authored by Jacob Keller's avatar Jacob Keller Committed by Jakub Kicinski

devlink: move flash end and begin to core devlink

When performing a flash update via devlink, device drivers may inform
user space of status updates via
devlink_flash_update_(begin|end|timeout|status)_notify functions.

It is expected that drivers do not send any status notifications unless
they send a begin and end message. If a driver sends a status
notification without sending the appropriate end notification upon
finishing (regardless of success or failure), the current implementation
of the devlink userspace program can get stuck endlessly waiting for the
end notification that will never come.

The current ice driver implementation may send such a status message
without the appropriate end notification in rare cases.

Fixing the ice driver is relatively simple: we just need to send the
begin_notify at the start of the function and always send an end_notify
no matter how the function exits.

Rather than assuming driver authors will always get this right in the
future, lets just fix the API so that it is not possible to get wrong.
Make devlink_flash_update_begin_notify and
devlink_flash_update_end_notify static, and call them in devlink.c core
code. Always send the begin_notify just before calling the driver's
flash_update routine. Always send the end_notify just after the routine
returns regardless of success or failure.

Doing this makes the status notification easier to use from the driver,
as it no longer needs to worry about catching failures and cleaning up
by calling devlink_flash_update_end_notify. It is now no longer possible
to do the wrong thing in this regard. We also save a couple of lines of
code in each driver.
Signed-off-by: default avatarJacob Keller <jacob.e.keller@intel.com>
Acked-by: default avatarVasundhara Volam <vasundhara-v.volam@broadcom.com>
Reviewed-by: default avatarJiri Pirko <jiri@nvidia.com>
Signed-off-by: default avatarJakub Kicinski <kuba@kernel.org>
parent b44cfd4f
...@@ -30,14 +30,12 @@ bnxt_dl_flash_update(struct devlink *dl, ...@@ -30,14 +30,12 @@ bnxt_dl_flash_update(struct devlink *dl,
return -EPERM; return -EPERM;
} }
devlink_flash_update_begin_notify(dl);
devlink_flash_update_status_notify(dl, "Preparing to flash", NULL, 0, 0); devlink_flash_update_status_notify(dl, "Preparing to flash", NULL, 0, 0);
rc = bnxt_flash_package_from_fw_obj(bp->dev, params->fw, 0); rc = bnxt_flash_package_from_fw_obj(bp->dev, params->fw, 0);
if (!rc) if (!rc)
devlink_flash_update_status_notify(dl, "Flashing done", NULL, 0, 0); devlink_flash_update_status_notify(dl, "Flashing done", NULL, 0, 0);
else else
devlink_flash_update_status_notify(dl, "Flashing failed", NULL, 0, 0); devlink_flash_update_status_notify(dl, "Flashing failed", NULL, 0, 0);
devlink_flash_update_end_notify(dl);
return rc; return rc;
} }
......
...@@ -275,12 +275,9 @@ ice_devlink_flash_update(struct devlink *devlink, ...@@ -275,12 +275,9 @@ ice_devlink_flash_update(struct devlink *devlink,
if (err) if (err)
return err; return err;
devlink_flash_update_begin_notify(devlink);
devlink_flash_update_status_notify(devlink, "Preparing to flash", NULL, 0, 0); devlink_flash_update_status_notify(devlink, "Preparing to flash", NULL, 0, 0);
err = ice_flash_pldm_image(pf, params->fw, preservation, extack);
devlink_flash_update_end_notify(devlink);
return err; return ice_flash_pldm_image(pf, params->fw, preservation, extack);
} }
static const struct devlink_ops ice_devlink_ops = { static const struct devlink_ops ice_devlink_ops = {
......
...@@ -368,7 +368,6 @@ int mlxfw_firmware_flash(struct mlxfw_dev *mlxfw_dev, ...@@ -368,7 +368,6 @@ int mlxfw_firmware_flash(struct mlxfw_dev *mlxfw_dev,
} }
mlxfw_info(mlxfw_dev, "Initialize firmware flash process\n"); mlxfw_info(mlxfw_dev, "Initialize firmware flash process\n");
devlink_flash_update_begin_notify(mlxfw_dev->devlink);
mlxfw_status_notify(mlxfw_dev, "Initializing firmware flash process", mlxfw_status_notify(mlxfw_dev, "Initializing firmware flash process",
NULL, 0, 0); NULL, 0, 0);
err = mlxfw_dev->ops->fsm_lock(mlxfw_dev, &fwhandle); err = mlxfw_dev->ops->fsm_lock(mlxfw_dev, &fwhandle);
...@@ -417,7 +416,6 @@ int mlxfw_firmware_flash(struct mlxfw_dev *mlxfw_dev, ...@@ -417,7 +416,6 @@ int mlxfw_firmware_flash(struct mlxfw_dev *mlxfw_dev,
mlxfw_info(mlxfw_dev, "Firmware flash done\n"); mlxfw_info(mlxfw_dev, "Firmware flash done\n");
mlxfw_status_notify(mlxfw_dev, "Firmware flash done", NULL, 0, 0); mlxfw_status_notify(mlxfw_dev, "Firmware flash done", NULL, 0, 0);
mlxfw_mfa2_file_fini(mfa2_file); mlxfw_mfa2_file_fini(mfa2_file);
devlink_flash_update_end_notify(mlxfw_dev->devlink);
return 0; return 0;
err_state_wait_activate_to_locked: err_state_wait_activate_to_locked:
...@@ -429,7 +427,6 @@ int mlxfw_firmware_flash(struct mlxfw_dev *mlxfw_dev, ...@@ -429,7 +427,6 @@ int mlxfw_firmware_flash(struct mlxfw_dev *mlxfw_dev,
mlxfw_dev->ops->fsm_release(mlxfw_dev, fwhandle); mlxfw_dev->ops->fsm_release(mlxfw_dev, fwhandle);
err_fsm_lock: err_fsm_lock:
mlxfw_mfa2_file_fini(mfa2_file); mlxfw_mfa2_file_fini(mfa2_file);
devlink_flash_update_end_notify(mlxfw_dev->devlink);
return err; return err;
} }
EXPORT_SYMBOL(mlxfw_firmware_flash); EXPORT_SYMBOL(mlxfw_firmware_flash);
......
...@@ -107,7 +107,6 @@ int ionic_firmware_update(struct ionic_lif *lif, const struct firmware *fw, ...@@ -107,7 +107,6 @@ int ionic_firmware_update(struct ionic_lif *lif, const struct firmware *fw,
netdev_info(netdev, "Installing firmware\n"); netdev_info(netdev, "Installing firmware\n");
dl = priv_to_devlink(ionic); dl = priv_to_devlink(ionic);
devlink_flash_update_begin_notify(dl);
devlink_flash_update_status_notify(dl, "Preparing to flash", NULL, 0, 0); devlink_flash_update_status_notify(dl, "Preparing to flash", NULL, 0, 0);
buf_sz = sizeof(idev->dev_cmd_regs->data); buf_sz = sizeof(idev->dev_cmd_regs->data);
...@@ -193,6 +192,5 @@ int ionic_firmware_update(struct ionic_lif *lif, const struct firmware *fw, ...@@ -193,6 +192,5 @@ int ionic_firmware_update(struct ionic_lif *lif, const struct firmware *fw,
devlink_flash_update_status_notify(dl, "Flash failed", NULL, 0, 0); devlink_flash_update_status_notify(dl, "Flash failed", NULL, 0, 0);
else else
devlink_flash_update_status_notify(dl, "Flash done", NULL, 0, 0); devlink_flash_update_status_notify(dl, "Flash done", NULL, 0, 0);
devlink_flash_update_end_notify(dl);
return err; return err;
} }
...@@ -766,7 +766,6 @@ static int nsim_dev_flash_update(struct devlink *devlink, ...@@ -766,7 +766,6 @@ static int nsim_dev_flash_update(struct devlink *devlink,
return -EOPNOTSUPP; return -EOPNOTSUPP;
if (nsim_dev->fw_update_status) { if (nsim_dev->fw_update_status) {
devlink_flash_update_begin_notify(devlink);
devlink_flash_update_status_notify(devlink, devlink_flash_update_status_notify(devlink,
"Preparing to flash", "Preparing to flash",
params->component, 0, 0); params->component, 0, 0);
...@@ -790,7 +789,6 @@ static int nsim_dev_flash_update(struct devlink *devlink, ...@@ -790,7 +789,6 @@ static int nsim_dev_flash_update(struct devlink *devlink,
params->component, 81); params->component, 81);
devlink_flash_update_status_notify(devlink, "Flashing done", devlink_flash_update_status_notify(devlink, "Flashing done",
params->component, 0, 0); params->component, 0, 0);
devlink_flash_update_end_notify(devlink);
} }
return 0; return 0;
......
...@@ -1577,8 +1577,6 @@ void devlink_remote_reload_actions_performed(struct devlink *devlink, ...@@ -1577,8 +1577,6 @@ void devlink_remote_reload_actions_performed(struct devlink *devlink,
enum devlink_reload_limit limit, enum devlink_reload_limit limit,
u32 actions_performed); u32 actions_performed);
void devlink_flash_update_begin_notify(struct devlink *devlink);
void devlink_flash_update_end_notify(struct devlink *devlink);
void devlink_flash_update_status_notify(struct devlink *devlink, void devlink_flash_update_status_notify(struct devlink *devlink,
const char *status_msg, const char *status_msg,
const char *component, const char *component,
......
...@@ -3372,7 +3372,7 @@ static void __devlink_flash_update_notify(struct devlink *devlink, ...@@ -3372,7 +3372,7 @@ static void __devlink_flash_update_notify(struct devlink *devlink,
nlmsg_free(msg); nlmsg_free(msg);
} }
void devlink_flash_update_begin_notify(struct devlink *devlink) static void devlink_flash_update_begin_notify(struct devlink *devlink)
{ {
struct devlink_flash_notify params = { 0 }; struct devlink_flash_notify params = { 0 };
...@@ -3380,9 +3380,8 @@ void devlink_flash_update_begin_notify(struct devlink *devlink) ...@@ -3380,9 +3380,8 @@ void devlink_flash_update_begin_notify(struct devlink *devlink)
DEVLINK_CMD_FLASH_UPDATE, DEVLINK_CMD_FLASH_UPDATE,
&params); &params);
} }
EXPORT_SYMBOL_GPL(devlink_flash_update_begin_notify);
void devlink_flash_update_end_notify(struct devlink *devlink) static void devlink_flash_update_end_notify(struct devlink *devlink)
{ {
struct devlink_flash_notify params = { 0 }; struct devlink_flash_notify params = { 0 };
...@@ -3390,7 +3389,6 @@ void devlink_flash_update_end_notify(struct devlink *devlink) ...@@ -3390,7 +3389,6 @@ void devlink_flash_update_end_notify(struct devlink *devlink)
DEVLINK_CMD_FLASH_UPDATE_END, DEVLINK_CMD_FLASH_UPDATE_END,
&params); &params);
} }
EXPORT_SYMBOL_GPL(devlink_flash_update_end_notify);
void devlink_flash_update_status_notify(struct devlink *devlink, void devlink_flash_update_status_notify(struct devlink *devlink,
const char *status_msg, const char *status_msg,
...@@ -3477,7 +3475,9 @@ static int devlink_nl_cmd_flash_update(struct sk_buff *skb, ...@@ -3477,7 +3475,9 @@ static int devlink_nl_cmd_flash_update(struct sk_buff *skb,
return ret; return ret;
} }
devlink_flash_update_begin_notify(devlink);
ret = devlink->ops->flash_update(devlink, &params, info->extack); ret = devlink->ops->flash_update(devlink, &params, info->extack);
devlink_flash_update_end_notify(devlink);
release_firmware(params.fw); release_firmware(params.fw);
...@@ -10244,7 +10244,9 @@ int devlink_compat_flash_update(struct net_device *dev, const char *file_name) ...@@ -10244,7 +10244,9 @@ int devlink_compat_flash_update(struct net_device *dev, const char *file_name)
goto out; goto out;
mutex_lock(&devlink->lock); mutex_lock(&devlink->lock);
devlink_flash_update_begin_notify(devlink);
ret = devlink->ops->flash_update(devlink, &params, NULL); ret = devlink->ops->flash_update(devlink, &params, NULL);
devlink_flash_update_end_notify(devlink);
mutex_unlock(&devlink->lock); mutex_unlock(&devlink->lock);
release_firmware(params.fw); release_firmware(params.fw);
......
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