summaryrefslogtreecommitdiffstats
path: root/drivers/net/netdevsim/devlink.c
diff options
context:
space:
mode:
authorJiri Pirko2019-04-25 15:59:42 +0200
committerDavid S. Miller2019-04-26 07:52:02 +0200
commit5fc494225c1eb81309cc4c91f183cd30e4edb674 (patch)
tree4369d3393e958b7ffb3cf7cd4d5eba271d7f78ad /drivers/net/netdevsim/devlink.c
parentnetdevsim: move device registration on bus to be done earlier in init (diff)
downloadkernel-qcow2-linux-5fc494225c1eb81309cc4c91f183cd30e4edb674.tar.gz
kernel-qcow2-linux-5fc494225c1eb81309cc4c91f183cd30e4edb674.tar.xz
kernel-qcow2-linux-5fc494225c1eb81309cc4c91f183cd30e4edb674.zip
netdevsim: create devlink instance per netdevsim instance
Currently there is one devlink instance created per network namespace. That is quite odd considering the fact that devlink instance should represent an ASIC. The following patches are going to move the devlink instance even more down to a bus device, but until then, have one devlink instance per netdevsim instance. Struct nsim_devlink is introduced to hold fib setting. The changes in the fib code are only related to holding the configuration per devlink instance instead of network namespace. Signed-off-by: Jiri Pirko <jiri@mellanox.com> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'drivers/net/netdevsim/devlink.c')
-rw-r--r--drivers/net/netdevsim/devlink.c179
1 files changed, 64 insertions, 115 deletions
diff --git a/drivers/net/netdevsim/devlink.c b/drivers/net/netdevsim/devlink.c
index 5135fc371f01..f718912fa52d 100644
--- a/drivers/net/netdevsim/devlink.c
+++ b/drivers/net/netdevsim/devlink.c
@@ -15,59 +15,59 @@
*/
#include <linux/device.h>
+#include <linux/rtnetlink.h>
#include <net/devlink.h>
-#include <net/netns/generic.h>
#include "netdevsim.h"
-static unsigned int nsim_devlink_id;
-
-/* place holder until devlink and namespaces is sorted out */
-static struct net *nsim_devlink_net(struct devlink *devlink)
-{
- return &init_net;
-}
+struct nsim_devlink {
+ struct nsim_fib_data *fib_data;
+};
/* IPv4
*/
static u64 nsim_ipv4_fib_resource_occ_get(void *priv)
{
- struct net *net = priv;
+ struct nsim_devlink *nsim_devlink = priv;
- return nsim_fib_get_val(net, NSIM_RESOURCE_IPV4_FIB, false);
+ return nsim_fib_get_val(nsim_devlink->fib_data,
+ NSIM_RESOURCE_IPV4_FIB, false);
}
static u64 nsim_ipv4_fib_rules_res_occ_get(void *priv)
{
- struct net *net = priv;
+ struct nsim_devlink *nsim_devlink = priv;
- return nsim_fib_get_val(net, NSIM_RESOURCE_IPV4_FIB_RULES, false);
+ return nsim_fib_get_val(nsim_devlink->fib_data,
+ NSIM_RESOURCE_IPV4_FIB_RULES, false);
}
/* IPv6
*/
static u64 nsim_ipv6_fib_resource_occ_get(void *priv)
{
- struct net *net = priv;
+ struct nsim_devlink *nsim_devlink = priv;
- return nsim_fib_get_val(net, NSIM_RESOURCE_IPV6_FIB, false);
+ return nsim_fib_get_val(nsim_devlink->fib_data,
+ NSIM_RESOURCE_IPV6_FIB, false);
}
static u64 nsim_ipv6_fib_rules_res_occ_get(void *priv)
{
- struct net *net = priv;
+ struct nsim_devlink *nsim_devlink = priv;
- return nsim_fib_get_val(net, NSIM_RESOURCE_IPV6_FIB_RULES, false);
+ return nsim_fib_get_val(nsim_devlink->fib_data,
+ NSIM_RESOURCE_IPV6_FIB_RULES, false);
}
static int devlink_resources_register(struct devlink *devlink)
{
+ struct nsim_devlink *nsim_devlink = devlink_priv(devlink);
struct devlink_resource_size_params params = {
.size_max = (u64)-1,
.size_granularity = 1,
.unit = DEVLINK_RESOURCE_UNIT_ENTRY
};
- struct net *net = nsim_devlink_net(devlink);
int err;
u64 n;
@@ -81,7 +81,8 @@ static int devlink_resources_register(struct devlink *devlink)
goto out;
}
- n = nsim_fib_get_val(net, NSIM_RESOURCE_IPV4_FIB, true);
+ n = nsim_fib_get_val(nsim_devlink->fib_data,
+ NSIM_RESOURCE_IPV4_FIB, true);
err = devlink_resource_register(devlink, "fib", n,
NSIM_RESOURCE_IPV4_FIB,
NSIM_RESOURCE_IPV4, &params);
@@ -90,7 +91,8 @@ static int devlink_resources_register(struct devlink *devlink)
return err;
}
- n = nsim_fib_get_val(net, NSIM_RESOURCE_IPV4_FIB_RULES, true);
+ n = nsim_fib_get_val(nsim_devlink->fib_data,
+ NSIM_RESOURCE_IPV4_FIB_RULES, true);
err = devlink_resource_register(devlink, "fib-rules", n,
NSIM_RESOURCE_IPV4_FIB_RULES,
NSIM_RESOURCE_IPV4, &params);
@@ -109,7 +111,8 @@ static int devlink_resources_register(struct devlink *devlink)
goto out;
}
- n = nsim_fib_get_val(net, NSIM_RESOURCE_IPV6_FIB, true);
+ n = nsim_fib_get_val(nsim_devlink->fib_data,
+ NSIM_RESOURCE_IPV6_FIB, true);
err = devlink_resource_register(devlink, "fib", n,
NSIM_RESOURCE_IPV6_FIB,
NSIM_RESOURCE_IPV6, &params);
@@ -118,7 +121,8 @@ static int devlink_resources_register(struct devlink *devlink)
return err;
}
- n = nsim_fib_get_val(net, NSIM_RESOURCE_IPV6_FIB_RULES, true);
+ n = nsim_fib_get_val(nsim_devlink->fib_data,
+ NSIM_RESOURCE_IPV6_FIB_RULES, true);
err = devlink_resource_register(devlink, "fib-rules", n,
NSIM_RESOURCE_IPV6_FIB_RULES,
NSIM_RESOURCE_IPV6, &params);
@@ -130,19 +134,19 @@ static int devlink_resources_register(struct devlink *devlink)
devlink_resource_occ_get_register(devlink,
NSIM_RESOURCE_IPV4_FIB,
nsim_ipv4_fib_resource_occ_get,
- net);
+ nsim_devlink);
devlink_resource_occ_get_register(devlink,
NSIM_RESOURCE_IPV4_FIB_RULES,
nsim_ipv4_fib_rules_res_occ_get,
- net);
+ nsim_devlink);
devlink_resource_occ_get_register(devlink,
NSIM_RESOURCE_IPV6_FIB,
nsim_ipv6_fib_resource_occ_get,
- net);
+ nsim_devlink);
devlink_resource_occ_get_register(devlink,
NSIM_RESOURCE_IPV6_FIB_RULES,
nsim_ipv6_fib_rules_res_occ_get,
- net);
+ nsim_devlink);
out:
return err;
}
@@ -150,11 +154,11 @@ out:
static int nsim_devlink_reload(struct devlink *devlink,
struct netlink_ext_ack *extack)
{
+ struct nsim_devlink *nsim_devlink = devlink_priv(devlink);
enum nsim_resource_id res_ids[] = {
NSIM_RESOURCE_IPV4_FIB, NSIM_RESOURCE_IPV4_FIB_RULES,
NSIM_RESOURCE_IPV6_FIB, NSIM_RESOURCE_IPV6_FIB_RULES
};
- struct net *net = nsim_devlink_net(devlink);
int i;
for (i = 0; i < ARRAY_SIZE(res_ids); ++i) {
@@ -163,7 +167,8 @@ static int nsim_devlink_reload(struct devlink *devlink,
err = devlink_resource_size_get(devlink, res_ids[i], &val);
if (!err) {
- err = nsim_fib_set_max(net, res_ids[i], val, extack);
+ err = nsim_fib_set_max(nsim_devlink->fib_data,
+ res_ids[i], val, extack);
if (err)
return err;
}
@@ -172,124 +177,68 @@ static int nsim_devlink_reload(struct devlink *devlink,
return 0;
}
-static void nsim_devlink_net_reset(struct net *net)
-{
- enum nsim_resource_id res_ids[] = {
- NSIM_RESOURCE_IPV4_FIB, NSIM_RESOURCE_IPV4_FIB_RULES,
- NSIM_RESOURCE_IPV6_FIB, NSIM_RESOURCE_IPV6_FIB_RULES
- };
- int i;
-
- for (i = 0; i < ARRAY_SIZE(res_ids); ++i) {
- if (nsim_fib_set_max(net, res_ids[i], (u64)-1, NULL)) {
- pr_err("Failed to reset limit for resource %u\n",
- res_ids[i]);
- }
- }
-}
-
static const struct devlink_ops nsim_devlink_ops = {
.reload = nsim_devlink_reload,
};
-/* once devlink / namespace issues are sorted out
- * this needs to be net in which a devlink instance
- * is to be created. e.g., dev_net(ns->netdev)
- */
-static struct net *nsim_to_net(struct netdevsim *ns)
+static int __nsim_devlink_init(struct netdevsim *ns)
{
- return &init_net;
-}
-
-void nsim_devlink_teardown(struct netdevsim *ns)
-{
- if (ns->devlink) {
- struct net *net = nsim_to_net(ns);
- bool *reg_devlink = net_generic(net, nsim_devlink_id);
-
- devlink_resources_unregister(ns->devlink, NULL);
- devlink_unregister(ns->devlink);
- devlink_free(ns->devlink);
- ns->devlink = NULL;
-
- nsim_devlink_net_reset(net);
- *reg_devlink = true;
- }
-}
-
-int nsim_devlink_setup(struct netdevsim *ns)
-{
- struct net *net = nsim_to_net(ns);
- bool *reg_devlink = net_generic(net, nsim_devlink_id);
+ struct nsim_devlink *nsim_devlink;
struct devlink *devlink;
int err;
- /* only one device per namespace controls devlink */
- if (!*reg_devlink) {
- ns->devlink = NULL;
- return 0;
- }
-
- devlink = devlink_alloc(&nsim_devlink_ops, 0);
+ devlink = devlink_alloc(&nsim_devlink_ops, sizeof(*nsim_devlink));
if (!devlink)
return -ENOMEM;
+ nsim_devlink = devlink_priv(devlink);
- err = devlink_register(devlink, &ns->dev);
- if (err)
+ nsim_devlink->fib_data = nsim_fib_create();
+ if (IS_ERR(nsim_devlink->fib_data)) {
+ err = PTR_ERR(nsim_devlink->fib_data);
goto err_devlink_free;
+ }
err = devlink_resources_register(devlink);
if (err)
- goto err_dl_unregister;
+ goto err_fib_destroy;
- ns->devlink = devlink;
+ err = devlink_register(devlink, &ns->dev);
+ if (err)
+ goto err_resources_unregister;
- *reg_devlink = false;
+ ns->devlink = devlink;
return 0;
-err_dl_unregister:
- devlink_unregister(devlink);
+err_resources_unregister:
+ devlink_resources_unregister(devlink, NULL);
+err_fib_destroy:
+ nsim_fib_destroy(nsim_devlink->fib_data);
err_devlink_free:
devlink_free(devlink);
return err;
}
-/* Initialize per network namespace state */
-static int __net_init nsim_devlink_netns_init(struct net *net)
+int nsim_devlink_init(struct netdevsim *ns)
{
- bool *reg_devlink = net_generic(net, nsim_devlink_id);
-
- *reg_devlink = true;
-
- return 0;
-}
-
-static struct pernet_operations nsim_devlink_net_ops = {
- .init = nsim_devlink_netns_init,
- .id = &nsim_devlink_id,
- .size = sizeof(bool),
-};
+ int err;
-void nsim_devlink_exit(void)
-{
- unregister_pernet_subsys(&nsim_devlink_net_ops);
- nsim_fib_exit();
+ dev_hold(ns->netdev);
+ rtnl_unlock();
+ err = __nsim_devlink_init(ns);
+ rtnl_lock();
+ dev_put(ns->netdev);
+ return err;
}
-int nsim_devlink_init(void)
+void nsim_devlink_exit(struct netdevsim *ns)
{
- int err;
+ struct devlink *devlink = ns->devlink;
+ struct nsim_devlink *nsim_devlink = devlink_priv(devlink);
- err = nsim_fib_init();
- if (err)
- goto err_out;
-
- err = register_pernet_subsys(&nsim_devlink_net_ops);
- if (err)
- nsim_fib_exit();
-
-err_out:
- return err;
+ devlink_unregister(devlink);
+ devlink_resources_unregister(devlink, NULL);
+ nsim_fib_destroy(nsim_devlink->fib_data);
+ devlink_free(devlink);
}