OSDN Git Service

devlink: protect health reporter operation with instance lock
authorJiri Pirko <jiri@nvidia.com>
Wed, 18 Jan 2023 15:21:08 +0000 (16:21 +0100)
committerJakub Kicinski <kuba@kernel.org>
Fri, 20 Jan 2023 03:08:37 +0000 (19:08 -0800)
Similar to other devlink objects, protect the reporters list
by devlink instance lock. Alongside add unlocked versions
of health reporter create/destroy functions and use them in drivers
on call paths where the instance lock is held.

Signed-off-by: Jiri Pirko <jiri@nvidia.com>
Reviewed-by: Jacob Keller <jacob.e.keller@intel.com>
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
drivers/net/ethernet/mellanox/mlxsw/core.c
drivers/net/netdevsim/health.c
include/net/devlink.h
net/devlink/leftover.c

index a0a06e2..33ef726 100644 (file)
@@ -2051,8 +2051,8 @@ static int mlxsw_core_health_init(struct mlxsw_core *mlxsw_core)
        if (!(mlxsw_core->bus->features & MLXSW_BUS_F_TXRX))
                return 0;
 
-       fw_fatal = devlink_health_reporter_create(devlink, &mlxsw_core_health_fw_fatal_ops,
-                                                 0, mlxsw_core);
+       fw_fatal = devl_health_reporter_create(devlink, &mlxsw_core_health_fw_fatal_ops,
+                                              0, mlxsw_core);
        if (IS_ERR(fw_fatal)) {
                dev_err(mlxsw_core->bus_info->dev, "Failed to create fw fatal reporter");
                return PTR_ERR(fw_fatal);
@@ -2072,7 +2072,7 @@ static int mlxsw_core_health_init(struct mlxsw_core *mlxsw_core)
 err_fw_fatal_config:
        mlxsw_core_trap_unregister(mlxsw_core, &mlxsw_core_health_listener, mlxsw_core);
 err_trap_register:
-       devlink_health_reporter_destroy(mlxsw_core->health.fw_fatal);
+       devl_health_reporter_destroy(mlxsw_core->health.fw_fatal);
        return err;
 }
 
@@ -2085,7 +2085,7 @@ static void mlxsw_core_health_fini(struct mlxsw_core *mlxsw_core)
        mlxsw_core_trap_unregister(mlxsw_core, &mlxsw_core_health_listener, mlxsw_core);
        /* Make sure there is no more event work scheduled */
        mlxsw_core_flush_owq();
-       devlink_health_reporter_destroy(mlxsw_core->health.fw_fatal);
+       devl_health_reporter_destroy(mlxsw_core->health.fw_fatal);
 }
 
 static void mlxsw_core_irq_event_handler_init(struct mlxsw_core *mlxsw_core)
index aa77af4..eb04ed7 100644 (file)
@@ -233,16 +233,16 @@ int nsim_dev_health_init(struct nsim_dev *nsim_dev, struct devlink *devlink)
        int err;
 
        health->empty_reporter =
-               devlink_health_reporter_create(devlink,
-                                              &nsim_dev_empty_reporter_ops,
-                                              0, health);
+               devl_health_reporter_create(devlink,
+                                           &nsim_dev_empty_reporter_ops,
+                                           0, health);
        if (IS_ERR(health->empty_reporter))
                return PTR_ERR(health->empty_reporter);
 
        health->dummy_reporter =
-               devlink_health_reporter_create(devlink,
-                                              &nsim_dev_dummy_reporter_ops,
-                                              0, health);
+               devl_health_reporter_create(devlink,
+                                           &nsim_dev_dummy_reporter_ops,
+                                           0, health);
        if (IS_ERR(health->dummy_reporter)) {
                err = PTR_ERR(health->dummy_reporter);
                goto err_empty_reporter_destroy;
@@ -266,9 +266,9 @@ int nsim_dev_health_init(struct nsim_dev *nsim_dev, struct devlink *devlink)
        return 0;
 
 err_dummy_reporter_destroy:
-       devlink_health_reporter_destroy(health->dummy_reporter);
+       devl_health_reporter_destroy(health->dummy_reporter);
 err_empty_reporter_destroy:
-       devlink_health_reporter_destroy(health->empty_reporter);
+       devl_health_reporter_destroy(health->empty_reporter);
        return err;
 }
 
@@ -278,6 +278,6 @@ void nsim_dev_health_exit(struct nsim_dev *nsim_dev)
 
        debugfs_remove_recursive(health->ddir);
        kfree(health->recovered_break_msg);
-       devlink_health_reporter_destroy(health->dummy_reporter);
-       devlink_health_reporter_destroy(health->empty_reporter);
+       devl_health_reporter_destroy(health->dummy_reporter);
+       devl_health_reporter_destroy(health->empty_reporter);
 }
index d7c9572..0d64fea 100644 (file)
@@ -1865,19 +1865,35 @@ int devlink_fmsg_binary_pair_put(struct devlink_fmsg *fmsg, const char *name,
                                 const void *value, u32 value_len);
 
 struct devlink_health_reporter *
-devlink_health_reporter_create(struct devlink *devlink,
-                              const struct devlink_health_reporter_ops *ops,
-                              u64 graceful_period, void *priv);
+devl_port_health_reporter_create(struct devlink_port *port,
+                                const struct devlink_health_reporter_ops *ops,
+                                u64 graceful_period, void *priv);
 
 struct devlink_health_reporter *
 devlink_port_health_reporter_create(struct devlink_port *port,
                                    const struct devlink_health_reporter_ops *ops,
                                    u64 graceful_period, void *priv);
 
+struct devlink_health_reporter *
+devl_health_reporter_create(struct devlink *devlink,
+                           const struct devlink_health_reporter_ops *ops,
+                           u64 graceful_period, void *priv);
+
+struct devlink_health_reporter *
+devlink_health_reporter_create(struct devlink *devlink,
+                              const struct devlink_health_reporter_ops *ops,
+                              u64 graceful_period, void *priv);
+
+void
+devl_health_reporter_destroy(struct devlink_health_reporter *reporter);
+
 void
 devlink_health_reporter_destroy(struct devlink_health_reporter *reporter);
 
 void
+devl_port_health_reporter_destroy(struct devlink_health_reporter *reporter);
+
+void
 devlink_port_health_reporter_destroy(struct devlink_health_reporter *reporter);
 
 void *
index f029e1b..0263dc1 100644 (file)
@@ -7337,8 +7337,8 @@ __devlink_health_reporter_create(struct devlink *devlink,
 }
 
 /**
- *     devlink_port_health_reporter_create - create devlink health reporter for
- *                                           specified port instance
+ *     devl_port_health_reporter_create - create devlink health reporter for
+ *                                        specified port instance
  *
  *     @port: devlink_port which should contain the new reporter
  *     @ops: ops
@@ -7346,12 +7346,13 @@ __devlink_health_reporter_create(struct devlink *devlink,
  *     @priv: priv
  */
 struct devlink_health_reporter *
-devlink_port_health_reporter_create(struct devlink_port *port,
-                                   const struct devlink_health_reporter_ops *ops,
-                                   u64 graceful_period, void *priv)
+devl_port_health_reporter_create(struct devlink_port *port,
+                                const struct devlink_health_reporter_ops *ops,
+                                u64 graceful_period, void *priv)
 {
        struct devlink_health_reporter *reporter;
 
+       devl_assert_locked(port->devlink);
        mutex_lock(&port->reporters_lock);
        if (__devlink_health_reporter_find_by_name(&port->reporter_list,
                                                   &port->reporters_lock, ops->name)) {
@@ -7370,10 +7371,26 @@ unlock:
        mutex_unlock(&port->reporters_lock);
        return reporter;
 }
+EXPORT_SYMBOL_GPL(devl_port_health_reporter_create);
+
+struct devlink_health_reporter *
+devlink_port_health_reporter_create(struct devlink_port *port,
+                                   const struct devlink_health_reporter_ops *ops,
+                                   u64 graceful_period, void *priv)
+{
+       struct devlink_health_reporter *reporter;
+       struct devlink *devlink = port->devlink;
+
+       devl_lock(devlink);
+       reporter = devl_port_health_reporter_create(port, ops,
+                                                   graceful_period, priv);
+       devl_unlock(devlink);
+       return reporter;
+}
 EXPORT_SYMBOL_GPL(devlink_port_health_reporter_create);
 
 /**
- *     devlink_health_reporter_create - create devlink health reporter
+ *     devl_health_reporter_create - create devlink health reporter
  *
  *     @devlink: devlink
  *     @ops: ops
@@ -7381,12 +7398,13 @@ EXPORT_SYMBOL_GPL(devlink_port_health_reporter_create);
  *     @priv: priv
  */
 struct devlink_health_reporter *
-devlink_health_reporter_create(struct devlink *devlink,
-                              const struct devlink_health_reporter_ops *ops,
-                              u64 graceful_period, void *priv)
+devl_health_reporter_create(struct devlink *devlink,
+                           const struct devlink_health_reporter_ops *ops,
+                           u64 graceful_period, void *priv)
 {
        struct devlink_health_reporter *reporter;
 
+       devl_assert_locked(devlink);
        mutex_lock(&devlink->reporters_lock);
        if (devlink_health_reporter_find_by_name(devlink, ops->name)) {
                reporter = ERR_PTR(-EEXIST);
@@ -7403,6 +7421,21 @@ unlock:
        mutex_unlock(&devlink->reporters_lock);
        return reporter;
 }
+EXPORT_SYMBOL_GPL(devl_health_reporter_create);
+
+struct devlink_health_reporter *
+devlink_health_reporter_create(struct devlink *devlink,
+                              const struct devlink_health_reporter_ops *ops,
+                              u64 graceful_period, void *priv)
+{
+       struct devlink_health_reporter *reporter;
+
+       devl_lock(devlink);
+       reporter = devl_health_reporter_create(devlink, ops,
+                                              graceful_period, priv);
+       devl_unlock(devlink);
+       return reporter;
+}
 EXPORT_SYMBOL_GPL(devlink_health_reporter_create);
 
 static void
@@ -7429,35 +7462,61 @@ __devlink_health_reporter_destroy(struct devlink_health_reporter *reporter)
 }
 
 /**
- *     devlink_health_reporter_destroy - destroy devlink health reporter
+ *     devl_health_reporter_destroy - destroy devlink health reporter
  *
  *     @reporter: devlink health reporter to destroy
  */
 void
-devlink_health_reporter_destroy(struct devlink_health_reporter *reporter)
+devl_health_reporter_destroy(struct devlink_health_reporter *reporter)
 {
        struct mutex *lock = &reporter->devlink->reporters_lock;
 
+       devl_assert_locked(reporter->devlink);
+
        mutex_lock(lock);
        __devlink_health_reporter_destroy(reporter);
        mutex_unlock(lock);
 }
+EXPORT_SYMBOL_GPL(devl_health_reporter_destroy);
+
+void
+devlink_health_reporter_destroy(struct devlink_health_reporter *reporter)
+{
+       struct devlink *devlink = reporter->devlink;
+
+       devl_lock(devlink);
+       devl_health_reporter_destroy(reporter);
+       devl_unlock(devlink);
+}
 EXPORT_SYMBOL_GPL(devlink_health_reporter_destroy);
 
 /**
- *     devlink_port_health_reporter_destroy - destroy devlink port health reporter
+ *     devl_port_health_reporter_destroy - destroy devlink port health reporter
  *
  *     @reporter: devlink health reporter to destroy
  */
 void
-devlink_port_health_reporter_destroy(struct devlink_health_reporter *reporter)
+devl_port_health_reporter_destroy(struct devlink_health_reporter *reporter)
 {
        struct mutex *lock = &reporter->devlink_port->reporters_lock;
 
+       devl_assert_locked(reporter->devlink);
+
        mutex_lock(lock);
        __devlink_health_reporter_destroy(reporter);
        mutex_unlock(lock);
 }
+EXPORT_SYMBOL_GPL(devl_port_health_reporter_destroy);
+
+void
+devlink_port_health_reporter_destroy(struct devlink_health_reporter *reporter)
+{
+       struct devlink *devlink = reporter->devlink;
+
+       devl_lock(devlink);
+       devl_port_health_reporter_destroy(reporter);
+       devl_unlock(devlink);
+}
 EXPORT_SYMBOL_GPL(devlink_port_health_reporter_destroy);
 
 static int
@@ -7805,12 +7864,11 @@ devlink_nl_cmd_health_reporter_get_dumpit(struct sk_buff *msg,
                unsigned long port_index;
                int idx = 0;
 
+               devl_lock(devlink);
+               if (!devl_is_registered(devlink))
+                       goto next_devlink;
+
                mutex_lock(&devlink->reporters_lock);
-               if (!devl_is_registered(devlink)) {
-                       mutex_unlock(&devlink->reporters_lock);
-                       devlink_put(devlink);
-                       continue;
-               }
 
                list_for_each_entry(reporter, &devlink->reporter_list,
                                    list) {
@@ -7824,6 +7882,7 @@ devlink_nl_cmd_health_reporter_get_dumpit(struct sk_buff *msg,
                                NLM_F_MULTI);
                        if (err) {
                                mutex_unlock(&devlink->reporters_lock);
+                               devl_unlock(devlink);
                                devlink_put(devlink);
                                state->idx = idx;
                                goto out;
@@ -7832,10 +7891,6 @@ devlink_nl_cmd_health_reporter_get_dumpit(struct sk_buff *msg,
                }
                mutex_unlock(&devlink->reporters_lock);
 
-               devl_lock(devlink);
-               if (!devl_is_registered(devlink))
-                       goto next_devlink;
-
                xa_for_each(&devlink->ports, port_index, port) {
                        mutex_lock(&port->reporters_lock);
                        list_for_each_entry(reporter, &port->reporter_list, list) {