On Wed, 2 Oct 2019 18:12:27 +0200, Jiri Pirko wrote: > From: Jiri Pirko <j...@mellanox.com> > > During devlink reload, all driver objects should be reinstantiated with > the exception of devlink instance and devlink resources and params. > Move existing devlink_resource_size_get() calls into fib_create() just > before fib notifier is registered. Also, make sure that extack is > propagated down to fib_notifier_register() call. > > Signed-off-by: Jiri Pirko <j...@mellanox.com> > --- > v1->v2: > - don't reinstantiate debugfs during reload. > --- > drivers/net/netdevsim/dev.c | 160 +++++++++++++++++------------- > drivers/net/netdevsim/fib.c | 53 +++++----- > drivers/net/netdevsim/netdevsim.h | 8 +- > 3 files changed, 124 insertions(+), 97 deletions(-) > > diff --git a/drivers/net/netdevsim/dev.c b/drivers/net/netdevsim/dev.c > index 3cc101aee991..c04312359e4f 100644 > --- a/drivers/net/netdevsim/dev.c > +++ b/drivers/net/netdevsim/dev.c > @@ -469,37 +469,28 @@ static void nsim_dev_traps_exit(struct devlink *devlink) > kfree(nsim_dev->trap_data); > } > > +static struct nsim_dev * > +nsim_dev_create(struct nsim_bus_dev *nsim_bus_dev, struct nsim_dev *nsim_dev, > + struct netlink_ext_ack *extack); > +static void nsim_dev_destroy(struct nsim_dev *nsim_dev, bool reload); > + > static int nsim_dev_reload_down(struct devlink *devlink, > struct netlink_ext_ack *extack) > { > + struct nsim_dev *nsim_dev = devlink_priv(devlink); > + > + nsim_dev_destroy(nsim_dev, true); > return 0; > } > > static int nsim_dev_reload_up(struct devlink *devlink, > struct netlink_ext_ack *extack) > +
Stray new line. > { > struct nsim_dev *nsim_dev = 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 > - }; > - 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(nsim_dev->fib_data, > - res_ids[i], val, extack); > - if (err) > - return err; > - } > - } > - nsim_devlink_param_load_driverinit_values(devlink); > - > - return 0; > + nsim_dev = nsim_dev_create(nsim_dev->nsim_bus_dev, nsim_dev, extack); > + return PTR_ERR_OR_ZERO(nsim_dev); > } > > #define NSIM_DEV_FLASH_SIZE 500000 > @@ -688,15 +679,21 @@ static int nsim_dev_port_add_all(struct nsim_dev > *nsim_dev, > } > > static struct nsim_dev * > -nsim_dev_create(struct nsim_bus_dev *nsim_bus_dev) > +nsim_dev_create(struct nsim_bus_dev *nsim_bus_dev, struct nsim_dev *nsim_dev, > + struct netlink_ext_ack *extack) > { > - struct nsim_dev *nsim_dev; > + bool reload = !!nsim_dev; > struct devlink *devlink; > int err; > > - devlink = devlink_alloc(&nsim_dev_devlink_ops, sizeof(*nsim_dev)); > - if (!devlink) > - return ERR_PTR(-ENOMEM); > + if (!reload) { > + devlink = devlink_alloc(&nsim_dev_devlink_ops, > + sizeof(*nsim_dev)); > + if (!devlink) > + return ERR_PTR(-ENOMEM); > + } else { > + devlink = priv_to_devlink(nsim_dev); > + } > nsim_dev = devlink_priv(devlink); > nsim_dev->nsim_bus_dev = nsim_bus_dev; > nsim_dev->switch_id.id_len = sizeof(nsim_dev->switch_id.id); > @@ -704,28 +701,35 @@ nsim_dev_create(struct nsim_bus_dev *nsim_bus_dev) > INIT_LIST_HEAD(&nsim_dev->port_list); > mutex_init(&nsim_dev->port_list_lock); > nsim_dev->fw_update_status = true; > - nsim_dev->max_macs = NSIM_DEV_MAX_MACS_DEFAULT; > - nsim_dev->test1 = NSIM_DEV_TEST1_DEFAULT; > > - err = nsim_dev_resources_register(devlink); > - if (err) > - goto err_devlink_free; > + if (!reload) { > + err = nsim_dev_resources_register(devlink); > + if (err) > + goto err_devlink_free; > + } > > - nsim_dev->fib_data = nsim_fib_create(devlink); > + nsim_dev->fib_data = nsim_fib_create(devlink, extack); > if (IS_ERR(nsim_dev->fib_data)) { > err = PTR_ERR(nsim_dev->fib_data); > goto err_resources_unregister; > } > > - err = devlink_register(devlink, &nsim_bus_dev->dev); > - if (err) > - goto err_fib_destroy; > + if (!reload) { > + nsim_dev->max_macs = NSIM_DEV_MAX_MACS_DEFAULT; > + nsim_dev->test1 = NSIM_DEV_TEST1_DEFAULT; > > - err = devlink_params_register(devlink, nsim_devlink_params, > - ARRAY_SIZE(nsim_devlink_params)); > - if (err) > - goto err_dl_unregister; > - nsim_devlink_set_params_init_values(nsim_dev, devlink); > + err = devlink_register(devlink, &nsim_bus_dev->dev); > + if (err) > + goto err_fib_destroy; > + > + err = devlink_params_register(devlink, nsim_devlink_params, > + ARRAY_SIZE(nsim_devlink_params)); > + if (err) > + goto err_dl_unregister; > + nsim_devlink_set_params_init_values(nsim_dev, devlink); > + } else { > + nsim_devlink_param_load_driverinit_values(devlink); > + } > > err = nsim_dev_dummy_region_init(nsim_dev, devlink); > if (err) > @@ -735,66 +739,86 @@ nsim_dev_create(struct nsim_bus_dev *nsim_bus_dev) > if (err) > goto err_dummy_region_exit; > > - err = nsim_dev_debugfs_init(nsim_dev); > - if (err) > - goto err_traps_exit; > + if (!reload) { > + err = nsim_dev_debugfs_init(nsim_dev); > + if (err) > + goto err_traps_exit; > > - err = nsim_bpf_dev_init(nsim_dev); > - if (err) > - goto err_debugfs_exit; > + err = nsim_bpf_dev_init(nsim_dev); > + if (err) > + goto err_debugfs_exit; > + } > > err = nsim_dev_port_add_all(nsim_dev, nsim_bus_dev->port_count); > if (err) > goto err_bpf_dev_exit; > > - devlink_params_publish(devlink); > + if (reload) > + devlink_params_publish(devlink); > > return nsim_dev; > > err_bpf_dev_exit: > - nsim_bpf_dev_exit(nsim_dev); > + if (!reload) { > + nsim_bpf_dev_exit(nsim_dev); > err_debugfs_exit: > - nsim_dev_debugfs_exit(nsim_dev); > + nsim_dev_debugfs_exit(nsim_dev); > + } > err_traps_exit: > nsim_dev_traps_exit(devlink); > err_dummy_region_exit: > nsim_dev_dummy_region_exit(nsim_dev); > err_params_unregister: > - devlink_params_unregister(devlink, nsim_devlink_params, > - ARRAY_SIZE(nsim_devlink_params)); > + if (!reload) { > + devlink_params_unregister(devlink, nsim_devlink_params, > + ARRAY_SIZE(nsim_devlink_params)); > err_dl_unregister: > - devlink_unregister(devlink); > + devlink_unregister(devlink); > + } > err_fib_destroy: > nsim_fib_destroy(devlink, nsim_dev->fib_data); > err_resources_unregister: > - devlink_resources_unregister(devlink, NULL); > + if (!reload) { > + devlink_resources_unregister(devlink, NULL); > err_devlink_free: > - devlink_free(devlink); > + devlink_free(devlink); > + } > return ERR_PTR(err); All this if (reload) code looks pretty ugly :/ I'd say create function should create, and destroy destroy, rather than do only half of the job based on some extra parameter :( > } > > -static void nsim_dev_destroy(struct nsim_dev *nsim_dev) > +static void nsim_dev_destroy(struct nsim_dev *nsim_dev, bool reload) > { > struct devlink *devlink = priv_to_devlink(nsim_dev); > > - nsim_dev_port_del_all(nsim_dev); > - nsim_bpf_dev_exit(nsim_dev); > - nsim_dev_debugfs_exit(nsim_dev); > - nsim_dev_traps_exit(devlink); > - nsim_dev_dummy_region_exit(nsim_dev); > - devlink_params_unregister(devlink, nsim_devlink_params, > - ARRAY_SIZE(nsim_devlink_params)); > - devlink_unregister(devlink); > - nsim_fib_destroy(devlink, nsim_dev->fib_data); > - devlink_resources_unregister(devlink, NULL); > - mutex_destroy(&nsim_dev->port_list_lock); > - devlink_free(devlink); > + if (!devlink_is_reload_failed(devlink)) > + nsim_dev_port_del_all(nsim_dev); > + if (!reload) { > + nsim_bpf_dev_exit(nsim_dev); > + nsim_dev_debugfs_exit(nsim_dev); > + } > + if (!devlink_is_reload_failed(devlink)) { > + nsim_dev_traps_exit(devlink); > + nsim_dev_dummy_region_exit(nsim_dev); > + mutex_destroy(&nsim_dev->port_list_lock); > + } > + if (!reload) { > + devlink_params_unregister(devlink, nsim_devlink_params, > + ARRAY_SIZE(nsim_devlink_params)); > + devlink_unregister(devlink); > + } > + if (!devlink_is_reload_failed(devlink)) > + nsim_fib_destroy(devlink, nsim_dev->fib_data); > + if (!reload) { > + devlink_resources_unregister(devlink, NULL); > + mutex_destroy(&nsim_dev->port_list_lock); > + devlink_free(devlink); > + } > } > > int nsim_dev_probe(struct nsim_bus_dev *nsim_bus_dev) > { > struct nsim_dev *nsim_dev; > > - nsim_dev = nsim_dev_create(nsim_bus_dev); > + nsim_dev = nsim_dev_create(nsim_bus_dev, NULL, NULL); > if (IS_ERR(nsim_dev)) > return PTR_ERR(nsim_dev); > dev_set_drvdata(&nsim_bus_dev->dev, nsim_dev); > @@ -806,7 +830,7 @@ void nsim_dev_remove(struct nsim_bus_dev *nsim_bus_dev) > { > struct nsim_dev *nsim_dev = dev_get_drvdata(&nsim_bus_dev->dev); > > - nsim_dev_destroy(nsim_dev); > + nsim_dev_destroy(nsim_dev, false); > } > > static struct nsim_dev_port * > diff --git a/drivers/net/netdevsim/fib.c b/drivers/net/netdevsim/fib.c > index d2aeac0f4c2c..fdc682f3a09a 100644 > --- a/drivers/net/netdevsim/fib.c > +++ b/drivers/net/netdevsim/fib.c > @@ -63,12 +63,10 @@ u64 nsim_fib_get_val(struct nsim_fib_data *fib_data, > return max ? entry->max : entry->num; > } > > -int nsim_fib_set_max(struct nsim_fib_data *fib_data, > - enum nsim_resource_id res_id, u64 val, > - struct netlink_ext_ack *extack) > +static void nsim_fib_set_max(struct nsim_fib_data *fib_data, > + enum nsim_resource_id res_id, u64 val) > { > struct nsim_fib_entry *entry; > - int err = 0; > > switch (res_id) { > case NSIM_RESOURCE_IPV4_FIB: > @@ -84,20 +82,10 @@ int nsim_fib_set_max(struct nsim_fib_data *fib_data, > entry = &fib_data->ipv6.rules; > break; > default: > - return 0; > - } > - > - /* not allowing a new max to be less than curren occupancy > - * --> no means of evicting entries > - */ > - if (val < entry->num) { > - NL_SET_ERR_MSG_MOD(extack, "New size is less than current > occupancy"); > - err = -EINVAL; > - } else { > - entry->max = val; > + WARN_ON(1); > + return; > } > - > - return err; > + entry->max = val; > } > > static int nsim_fib_rule_account(struct nsim_fib_entry *entry, bool add, > @@ -239,7 +227,28 @@ static u64 nsim_fib_ipv6_rules_res_occ_get(void *priv) > return nsim_fib_get_val(data, NSIM_RESOURCE_IPV6_FIB_RULES, false); > } > > -struct nsim_fib_data *nsim_fib_create(struct devlink *devlink) > +static void nsim_fib_set_max_all(struct nsim_fib_data *data, > + struct devlink *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 > + }; > + 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) > + val = (u64) -1; > + nsim_fib_set_max(data, res_ids[i], val); > + } > +} > + > +struct nsim_fib_data *nsim_fib_create(struct devlink *devlink, > + struct netlink_ext_ack *extack) > { > struct nsim_fib_data *data; > int err; > @@ -248,15 +257,11 @@ struct nsim_fib_data *nsim_fib_create(struct devlink > *devlink) > if (!data) > return ERR_PTR(-ENOMEM); > > - data->ipv4.fib.max = (u64)-1; > - data->ipv4.rules.max = (u64)-1; > - > - data->ipv6.fib.max = (u64)-1; > - data->ipv6.rules.max = (u64)-1; > + nsim_fib_set_max_all(data, devlink); > > data->fib_nb.notifier_call = nsim_fib_event_nb; > err = register_fib_notifier(&init_net, &data->fib_nb, > - nsim_fib_dump_inconsistent, NULL); > + nsim_fib_dump_inconsistent, extack); > if (err) { > pr_err("Failed to register fib notifier\n"); > goto err_out; > diff --git a/drivers/net/netdevsim/netdevsim.h > b/drivers/net/netdevsim/netdevsim.h > index ac506cf253b6..702d951fe160 100644 > --- a/drivers/net/netdevsim/netdevsim.h > +++ b/drivers/net/netdevsim/netdevsim.h > @@ -173,13 +173,11 @@ int nsim_dev_port_add(struct nsim_bus_dev *nsim_bus_dev, > int nsim_dev_port_del(struct nsim_bus_dev *nsim_bus_dev, > unsigned int port_index); > > -struct nsim_fib_data *nsim_fib_create(struct devlink *devlink); > -void nsim_fib_destroy(struct devlink *devlink, struct nsim_fib_data *data); > +struct nsim_fib_data *nsim_fib_create(struct devlink *devlink, > + struct netlink_ext_ack *extack); > +void nsim_fib_destroy(struct devlink *devlink, struct nsim_fib_data > *fib_data); > u64 nsim_fib_get_val(struct nsim_fib_data *fib_data, > enum nsim_resource_id res_id, bool max); > -int nsim_fib_set_max(struct nsim_fib_data *fib_data, > - enum nsim_resource_id res_id, u64 val, > - struct netlink_ext_ack *extack); > > #if IS_ENABLED(CONFIG_XFRM_OFFLOAD) > void nsim_ipsec_init(struct netdevsim *ns);