aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorArkadi Sharshevsky <arkadis@mellanox.com>2018-01-15 02:59:04 -0500
committerDavid S. Miller <davem@davemloft.net>2018-01-16 14:15:34 -0500
commit2d8dc5bbf4e7603747875eb5cadcd67c1fa8b1bb (patch)
treef47537503b83bd112c2530176ca7321ffe0f81d9
parentd9f9b9a4d05fab693fd23a9ecaa330e03ebe2c31 (diff)
devlink: Add support for reload
Add support for performing driver hot reload. Signed-off-by: Arkadi Sharshevsky <arkadis@mellanox.com> Signed-off-by: Jiri Pirko <jiri@mellanox.com> Signed-off-by: David S. Miller <davem@davemloft.net>
-rw-r--r--include/net/devlink.h1
-rw-r--r--include/uapi/linux/devlink.h5
-rw-r--r--net/core/devlink.c47
3 files changed, 53 insertions, 0 deletions
diff --git a/include/net/devlink.h b/include/net/devlink.h
index ceb1895d119b..c698883fb0bb 100644
--- a/include/net/devlink.h
+++ b/include/net/devlink.h
@@ -281,6 +281,7 @@ struct devlink_resource {
281#define DEVLINK_RESOURCE_ID_PARENT_TOP 0 281#define DEVLINK_RESOURCE_ID_PARENT_TOP 0
282 282
283struct devlink_ops { 283struct devlink_ops {
284 int (*reload)(struct devlink *devlink);
284 int (*port_type_set)(struct devlink_port *devlink_port, 285 int (*port_type_set)(struct devlink_port *devlink_port,
285 enum devlink_port_type port_type); 286 enum devlink_port_type port_type);
286 int (*port_split)(struct devlink *devlink, unsigned int port_index, 287 int (*port_split)(struct devlink *devlink, unsigned int port_index,
diff --git a/include/uapi/linux/devlink.h b/include/uapi/linux/devlink.h
index f89950443e17..555ddcaf0be2 100644
--- a/include/uapi/linux/devlink.h
+++ b/include/uapi/linux/devlink.h
@@ -73,6 +73,11 @@ enum devlink_command {
73 DEVLINK_CMD_RESOURCE_SET, 73 DEVLINK_CMD_RESOURCE_SET,
74 DEVLINK_CMD_RESOURCE_DUMP, 74 DEVLINK_CMD_RESOURCE_DUMP,
75 75
76 /* Hot driver reload, makes configuration changes take place. The
77 * devlink instance is not released during the process.
78 */
79 DEVLINK_CMD_RELOAD,
80
76 /* add new commands above here */ 81 /* add new commands above here */
77 __DEVLINK_CMD_MAX, 82 __DEVLINK_CMD_MAX,
78 DEVLINK_CMD_MAX = __DEVLINK_CMD_MAX - 1 83 DEVLINK_CMD_MAX = __DEVLINK_CMD_MAX - 1
diff --git a/net/core/devlink.c b/net/core/devlink.c
index 89b3704fa450..4c3d85560436 100644
--- a/net/core/devlink.c
+++ b/net/core/devlink.c
@@ -2515,6 +2515,45 @@ static int devlink_nl_cmd_resource_dump(struct sk_buff *skb,
2515 return devlink_resource_fill(info, DEVLINK_CMD_RESOURCE_DUMP, 0); 2515 return devlink_resource_fill(info, DEVLINK_CMD_RESOURCE_DUMP, 0);
2516} 2516}
2517 2517
2518static int
2519devlink_resources_validate(struct devlink *devlink,
2520 struct devlink_resource *resource,
2521 struct genl_info *info)
2522{
2523 struct list_head *resource_list;
2524 int err = 0;
2525
2526 if (resource)
2527 resource_list = &resource->resource_list;
2528 else
2529 resource_list = &devlink->resource_list;
2530
2531 list_for_each_entry(resource, resource_list, list) {
2532 if (!resource->size_valid)
2533 return -EINVAL;
2534 err = devlink_resources_validate(devlink, resource, info);
2535 if (err)
2536 return err;
2537 }
2538 return err;
2539}
2540
2541static int devlink_nl_cmd_reload(struct sk_buff *skb, struct genl_info *info)
2542{
2543 struct devlink *devlink = info->user_ptr[0];
2544 int err;
2545
2546 if (!devlink->ops->reload)
2547 return -EOPNOTSUPP;
2548
2549 err = devlink_resources_validate(devlink, NULL, info);
2550 if (err) {
2551 NL_SET_ERR_MSG_MOD(info->extack, "resources size validation failed");
2552 return err;
2553 }
2554 return devlink->ops->reload(devlink);
2555}
2556
2518static const struct nla_policy devlink_nl_policy[DEVLINK_ATTR_MAX + 1] = { 2557static const struct nla_policy devlink_nl_policy[DEVLINK_ATTR_MAX + 1] = {
2519 [DEVLINK_ATTR_BUS_NAME] = { .type = NLA_NUL_STRING }, 2558 [DEVLINK_ATTR_BUS_NAME] = { .type = NLA_NUL_STRING },
2520 [DEVLINK_ATTR_DEV_NAME] = { .type = NLA_NUL_STRING }, 2559 [DEVLINK_ATTR_DEV_NAME] = { .type = NLA_NUL_STRING },
@@ -2709,6 +2748,14 @@ static const struct genl_ops devlink_nl_ops[] = {
2709 .flags = GENL_ADMIN_PERM, 2748 .flags = GENL_ADMIN_PERM,
2710 .internal_flags = DEVLINK_NL_FLAG_NEED_DEVLINK, 2749 .internal_flags = DEVLINK_NL_FLAG_NEED_DEVLINK,
2711 }, 2750 },
2751 {
2752 .cmd = DEVLINK_CMD_RELOAD,
2753 .doit = devlink_nl_cmd_reload,
2754 .policy = devlink_nl_policy,
2755 .flags = GENL_ADMIN_PERM,
2756 .internal_flags = DEVLINK_NL_FLAG_NEED_DEVLINK |
2757 DEVLINK_NL_FLAG_NO_LOCK,
2758 },
2712}; 2759};
2713 2760
2714static struct genl_family devlink_nl_family __ro_after_init = { 2761static struct genl_family devlink_nl_family __ro_after_init = {