|  | /* | 
|  | * Copyright (c) 2018 Cumulus Networks. All rights reserved. | 
|  | * Copyright (c) 2018 David Ahern <dsa@cumulusnetworks.com> | 
|  | * | 
|  | * This software is licensed under the GNU General License Version 2, | 
|  | * June 1991 as shown in the file COPYING in the top-level directory of this | 
|  | * source tree. | 
|  | * | 
|  | * THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES PROVIDE THE PROGRAM "AS IS" | 
|  | * WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED OR IMPLIED, INCLUDING, | 
|  | * BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS | 
|  | * FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS TO THE QUALITY AND PERFORMANCE | 
|  | * OF THE PROGRAM IS WITH YOU. SHOULD THE PROGRAM PROVE DEFECTIVE, YOU ASSUME | 
|  | * THE COST OF ALL NECESSARY SERVICING, REPAIR OR CORRECTION. | 
|  | */ | 
|  |  | 
|  | #include <linux/device.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; | 
|  | } | 
|  |  | 
|  | /* IPv4 | 
|  | */ | 
|  | static u64 nsim_ipv4_fib_resource_occ_get(void *priv) | 
|  | { | 
|  | struct net *net = priv; | 
|  |  | 
|  | return nsim_fib_get_val(net, NSIM_RESOURCE_IPV4_FIB, false); | 
|  | } | 
|  |  | 
|  | static u64 nsim_ipv4_fib_rules_res_occ_get(void *priv) | 
|  | { | 
|  | struct net *net = priv; | 
|  |  | 
|  | return nsim_fib_get_val(net, NSIM_RESOURCE_IPV4_FIB_RULES, false); | 
|  | } | 
|  |  | 
|  | /* IPv6 | 
|  | */ | 
|  | static u64 nsim_ipv6_fib_resource_occ_get(void *priv) | 
|  | { | 
|  | struct net *net = priv; | 
|  |  | 
|  | return nsim_fib_get_val(net, NSIM_RESOURCE_IPV6_FIB, false); | 
|  | } | 
|  |  | 
|  | static u64 nsim_ipv6_fib_rules_res_occ_get(void *priv) | 
|  | { | 
|  | struct net *net = priv; | 
|  |  | 
|  | return nsim_fib_get_val(net, NSIM_RESOURCE_IPV6_FIB_RULES, false); | 
|  | } | 
|  |  | 
|  | static int devlink_resources_register(struct devlink *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; | 
|  |  | 
|  | /* Resources for IPv4 */ | 
|  | err = devlink_resource_register(devlink, "IPv4", (u64)-1, | 
|  | NSIM_RESOURCE_IPV4, | 
|  | DEVLINK_RESOURCE_ID_PARENT_TOP, | 
|  | ¶ms); | 
|  | if (err) { | 
|  | pr_err("Failed to register IPv4 top resource\n"); | 
|  | goto out; | 
|  | } | 
|  |  | 
|  | n = nsim_fib_get_val(net, NSIM_RESOURCE_IPV4_FIB, true); | 
|  | err = devlink_resource_register(devlink, "fib", n, | 
|  | NSIM_RESOURCE_IPV4_FIB, | 
|  | NSIM_RESOURCE_IPV4, ¶ms); | 
|  | if (err) { | 
|  | pr_err("Failed to register IPv4 FIB resource\n"); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | n = nsim_fib_get_val(net, NSIM_RESOURCE_IPV4_FIB_RULES, true); | 
|  | err = devlink_resource_register(devlink, "fib-rules", n, | 
|  | NSIM_RESOURCE_IPV4_FIB_RULES, | 
|  | NSIM_RESOURCE_IPV4, ¶ms); | 
|  | if (err) { | 
|  | pr_err("Failed to register IPv4 FIB rules resource\n"); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | /* Resources for IPv6 */ | 
|  | err = devlink_resource_register(devlink, "IPv6", (u64)-1, | 
|  | NSIM_RESOURCE_IPV6, | 
|  | DEVLINK_RESOURCE_ID_PARENT_TOP, | 
|  | ¶ms); | 
|  | if (err) { | 
|  | pr_err("Failed to register IPv6 top resource\n"); | 
|  | goto out; | 
|  | } | 
|  |  | 
|  | n = nsim_fib_get_val(net, NSIM_RESOURCE_IPV6_FIB, true); | 
|  | err = devlink_resource_register(devlink, "fib", n, | 
|  | NSIM_RESOURCE_IPV6_FIB, | 
|  | NSIM_RESOURCE_IPV6, ¶ms); | 
|  | if (err) { | 
|  | pr_err("Failed to register IPv6 FIB resource\n"); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | n = nsim_fib_get_val(net, NSIM_RESOURCE_IPV6_FIB_RULES, true); | 
|  | err = devlink_resource_register(devlink, "fib-rules", n, | 
|  | NSIM_RESOURCE_IPV6_FIB_RULES, | 
|  | NSIM_RESOURCE_IPV6, ¶ms); | 
|  | if (err) { | 
|  | pr_err("Failed to register IPv6 FIB rules resource\n"); | 
|  | return err; | 
|  | } | 
|  |  | 
|  | devlink_resource_occ_get_register(devlink, | 
|  | NSIM_RESOURCE_IPV4_FIB, | 
|  | nsim_ipv4_fib_resource_occ_get, | 
|  | net); | 
|  | devlink_resource_occ_get_register(devlink, | 
|  | NSIM_RESOURCE_IPV4_FIB_RULES, | 
|  | nsim_ipv4_fib_rules_res_occ_get, | 
|  | net); | 
|  | devlink_resource_occ_get_register(devlink, | 
|  | NSIM_RESOURCE_IPV6_FIB, | 
|  | nsim_ipv6_fib_resource_occ_get, | 
|  | net); | 
|  | devlink_resource_occ_get_register(devlink, | 
|  | NSIM_RESOURCE_IPV6_FIB_RULES, | 
|  | nsim_ipv6_fib_rules_res_occ_get, | 
|  | net); | 
|  | out: | 
|  | return err; | 
|  | } | 
|  |  | 
|  | static int nsim_devlink_reload(struct devlink *devlink, | 
|  | struct netlink_ext_ack *extack) | 
|  | { | 
|  | 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) { | 
|  | int err; | 
|  | u64 val; | 
|  |  | 
|  | err = devlink_resource_size_get(devlink, res_ids[i], &val); | 
|  | if (!err) { | 
|  | err = nsim_fib_set_max(net, res_ids[i], val, extack); | 
|  | if (err) | 
|  | return err; | 
|  | } | 
|  | } | 
|  |  | 
|  | 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) | 
|  | { | 
|  | 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 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); | 
|  | if (!devlink) | 
|  | return -ENOMEM; | 
|  |  | 
|  | err = devlink_register(devlink, &ns->dev); | 
|  | if (err) | 
|  | goto err_devlink_free; | 
|  |  | 
|  | err = devlink_resources_register(devlink); | 
|  | if (err) | 
|  | goto err_dl_unregister; | 
|  |  | 
|  | ns->devlink = devlink; | 
|  |  | 
|  | *reg_devlink = false; | 
|  |  | 
|  | return 0; | 
|  |  | 
|  | err_dl_unregister: | 
|  | devlink_unregister(devlink); | 
|  | err_devlink_free: | 
|  | devlink_free(devlink); | 
|  |  | 
|  | return err; | 
|  | } | 
|  |  | 
|  | /* Initialize per network namespace state */ | 
|  | static int __net_init nsim_devlink_netns_init(struct net *net) | 
|  | { | 
|  | 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), | 
|  | }; | 
|  |  | 
|  | void nsim_devlink_exit(void) | 
|  | { | 
|  | unregister_pernet_subsys(&nsim_devlink_net_ops); | 
|  | nsim_fib_exit(); | 
|  | } | 
|  |  | 
|  | int nsim_devlink_init(void) | 
|  | { | 
|  | int err; | 
|  |  | 
|  | 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; | 
|  | } |