@@ -179,29 +179,34 @@ static struct device_type nsim_bus_dev_type = {
};
static struct nsim_bus_dev *
-nsim_bus_dev_new(unsigned int id, unsigned int port_count);
+nsim_bus_dev_new(unsigned int id, unsigned int port_count,
+ unsigned int linecard_count);
static ssize_t
new_device_store(struct bus_type *bus, const char *buf, size_t count)
{
struct nsim_bus_dev *nsim_bus_dev;
+ unsigned int linecard_count;
unsigned int port_count;
unsigned int id;
int err;
- err = sscanf(buf, "%u %u", &id, &port_count);
+ err = sscanf(buf, "%u %u %u", &id, &port_count, &linecard_count);
switch (err) {
case 1:
port_count = 1;
fallthrough;
case 2:
+ linecard_count = 0;
+ fallthrough;
+ case 3:
if (id > INT_MAX) {
pr_err("Value of \"id\" is too big.\n");
return -EINVAL;
}
break;
default:
- pr_err("Format for adding new device is \"id port_count\" (uint uint).\n");
+ pr_err("Format for adding new device is \"id port_count linecard_count\" (uint uint uint).\n");
return -EINVAL;
}
@@ -212,7 +217,7 @@ new_device_store(struct bus_type *bus, const char *buf, size_t count)
goto err;
}
- nsim_bus_dev = nsim_bus_dev_new(id, port_count);
+ nsim_bus_dev = nsim_bus_dev_new(id, port_count, linecard_count);
if (IS_ERR(nsim_bus_dev)) {
err = PTR_ERR(nsim_bus_dev);
goto err;
@@ -312,7 +317,8 @@ static struct bus_type nsim_bus = {
};
static struct nsim_bus_dev *
-nsim_bus_dev_new(unsigned int id, unsigned int port_count)
+nsim_bus_dev_new(unsigned int id, unsigned int port_count,
+ unsigned int linecard_count)
{
struct nsim_bus_dev *nsim_bus_dev;
int err;
@@ -328,6 +334,7 @@ nsim_bus_dev_new(unsigned int id, unsigned int port_count)
nsim_bus_dev->dev.bus = &nsim_bus;
nsim_bus_dev->dev.type = &nsim_bus_dev_type;
nsim_bus_dev->port_count = port_count;
+ nsim_bus_dev->linecard_count = linecard_count;
nsim_bus_dev->initial_net = current->nsproxy->net_ns;
mutex_init(&nsim_bus_dev->nsim_bus_reload_lock);
/* Disallow using nsim_bus_dev */
@@ -203,6 +203,10 @@ static int nsim_dev_debugfs_init(struct nsim_dev *nsim_dev)
nsim_dev->ports_ddir = debugfs_create_dir("ports", nsim_dev->ddir);
if (IS_ERR(nsim_dev->ports_ddir))
return PTR_ERR(nsim_dev->ports_ddir);
+ nsim_dev->linecards_ddir = debugfs_create_dir("linecards",
+ nsim_dev->ddir);
+ if (IS_ERR(nsim_dev->linecards_ddir))
+ return PTR_ERR(nsim_dev->linecards_ddir);
debugfs_create_bool("fw_update_status", 0600, nsim_dev->ddir,
&nsim_dev->fw_update_status);
debugfs_create_u32("fw_update_overwrite_mask", 0600, nsim_dev->ddir,
@@ -237,6 +241,7 @@ static int nsim_dev_debugfs_init(struct nsim_dev *nsim_dev)
static void nsim_dev_debugfs_exit(struct nsim_dev *nsim_dev)
{
+ debugfs_remove_recursive(nsim_dev->linecards_ddir);
debugfs_remove_recursive(nsim_dev->ports_ddir);
debugfs_remove_recursive(nsim_dev->ddir);
}
@@ -265,6 +270,32 @@ static void nsim_dev_port_debugfs_exit(struct nsim_dev_port *nsim_dev_port)
debugfs_remove_recursive(nsim_dev_port->ddir);
}
+static int
+nsim_dev_linecard_debugfs_init(struct nsim_dev *nsim_dev,
+ struct nsim_dev_linecard *nsim_dev_linecard)
+{
+ char linecard_ddir_name[16];
+ char dev_link_name[32];
+
+ sprintf(linecard_ddir_name, "%u", nsim_dev_linecard->linecard_index);
+ nsim_dev_linecard->ddir = debugfs_create_dir(linecard_ddir_name,
+ nsim_dev->linecards_ddir);
+ if (IS_ERR(nsim_dev_linecard->ddir))
+ return PTR_ERR(nsim_dev_linecard->ddir);
+
+ sprintf(dev_link_name, "../../../" DRV_NAME "%u",
+ nsim_dev->nsim_bus_dev->dev.id);
+ debugfs_create_symlink("dev", nsim_dev_linecard->ddir, dev_link_name);
+
+ return 0;
+}
+
+static void
+nsim_dev_linecard_debugfs_exit(struct nsim_dev_linecard *nsim_dev_linecard)
+{
+ debugfs_remove_recursive(nsim_dev_linecard->ddir);
+}
+
static int nsim_dev_resources_register(struct devlink *devlink)
{
struct devlink_resource_size_params params = {
@@ -998,6 +1029,64 @@ static int nsim_dev_port_add_all(struct nsim_dev *nsim_dev,
return err;
}
+static int __nsim_dev_linecard_add(struct nsim_dev *nsim_dev,
+ unsigned int linecard_index)
+{
+ struct nsim_dev_linecard *nsim_dev_linecard;
+ int err;
+
+ nsim_dev_linecard = kzalloc(sizeof(*nsim_dev_linecard), GFP_KERNEL);
+ if (!nsim_dev_linecard)
+ return -ENOMEM;
+ nsim_dev_linecard->nsim_dev = nsim_dev;
+ nsim_dev_linecard->linecard_index = linecard_index;
+
+ err = nsim_dev_linecard_debugfs_init(nsim_dev, nsim_dev_linecard);
+ if (err)
+ goto err_linecard_free;
+
+ list_add(&nsim_dev_linecard->list, &nsim_dev->linecard_list);
+
+ return 0;
+
+err_linecard_free:
+ kfree(nsim_dev_linecard);
+ return err;
+}
+
+static void __nsim_dev_linecard_del(struct nsim_dev_linecard *nsim_dev_linecard)
+{
+ list_del(&nsim_dev_linecard->list);
+ nsim_dev_linecard_debugfs_exit(nsim_dev_linecard);
+ kfree(nsim_dev_linecard);
+}
+
+static void nsim_dev_linecard_del_all(struct nsim_dev *nsim_dev)
+{
+ struct nsim_dev_linecard *nsim_dev_linecard, *tmp;
+
+ list_for_each_entry_safe(nsim_dev_linecard, tmp,
+ &nsim_dev->linecard_list, list)
+ __nsim_dev_linecard_del(nsim_dev_linecard);
+}
+
+static int nsim_dev_linecard_add_all(struct nsim_dev *nsim_dev,
+ unsigned int linecard_count)
+{
+ int i, err;
+
+ for (i = 0; i < linecard_count; i++) {
+ err = __nsim_dev_linecard_add(nsim_dev, i);
+ if (err)
+ goto err_linecard_del_all;
+ }
+ return 0;
+
+err_linecard_del_all:
+ nsim_dev_linecard_del_all(nsim_dev);
+ return err;
+}
+
static int nsim_dev_reload_create(struct nsim_dev *nsim_dev,
struct netlink_ext_ack *extack)
{
@@ -1009,6 +1098,7 @@ static int nsim_dev_reload_create(struct nsim_dev *nsim_dev,
nsim_dev = devlink_priv(devlink);
INIT_LIST_HEAD(&nsim_dev->port_list);
mutex_init(&nsim_dev->port_list_lock);
+ INIT_LIST_HEAD(&nsim_dev->linecard_list);
nsim_dev->fw_update_status = true;
nsim_dev->fw_update_overwrite_mask = 0;
@@ -1030,10 +1120,14 @@ static int nsim_dev_reload_create(struct nsim_dev *nsim_dev,
if (err)
goto err_traps_exit;
- err = nsim_dev_port_add_all(nsim_dev, nsim_bus_dev->port_count);
+ err = nsim_dev_linecard_add_all(nsim_dev, nsim_bus_dev->linecard_count);
if (err)
goto err_health_exit;
+ err = nsim_dev_port_add_all(nsim_dev, nsim_bus_dev->port_count);
+ if (err)
+ goto err_linecard_del_all;
+
nsim_dev->take_snapshot = debugfs_create_file("take_snapshot",
0200,
nsim_dev->ddir,
@@ -1041,6 +1135,8 @@ static int nsim_dev_reload_create(struct nsim_dev *nsim_dev,
&nsim_dev_take_snapshot_fops);
return 0;
+err_linecard_del_all:
+ nsim_dev_linecard_del_all(nsim_dev);
err_health_exit:
nsim_dev_health_exit(nsim_dev);
err_traps_exit:
@@ -1068,6 +1164,7 @@ int nsim_dev_probe(struct nsim_bus_dev *nsim_bus_dev)
get_random_bytes(nsim_dev->switch_id.id, nsim_dev->switch_id.id_len);
INIT_LIST_HEAD(&nsim_dev->port_list);
mutex_init(&nsim_dev->port_list_lock);
+ INIT_LIST_HEAD(&nsim_dev->linecard_list);
nsim_dev->fw_update_status = true;
nsim_dev->fw_update_overwrite_mask = 0;
nsim_dev->max_macs = NSIM_DEV_MAX_MACS_DEFAULT;
@@ -1116,14 +1213,20 @@ int nsim_dev_probe(struct nsim_bus_dev *nsim_bus_dev)
if (err)
goto err_health_exit;
- err = nsim_dev_port_add_all(nsim_dev, nsim_bus_dev->port_count);
+ err = nsim_dev_linecard_add_all(nsim_dev, nsim_bus_dev->linecard_count);
if (err)
goto err_bpf_dev_exit;
+ err = nsim_dev_port_add_all(nsim_dev, nsim_bus_dev->port_count);
+ if (err)
+ goto err_linecard_del_all;
+
devlink_params_publish(devlink);
devlink_reload_enable(devlink);
return 0;
+err_linecard_del_all:
+ nsim_dev_linecard_del_all(nsim_dev);
err_bpf_dev_exit:
nsim_bpf_dev_exit(nsim_dev);
err_health_exit:
@@ -1156,6 +1259,7 @@ static void nsim_dev_reload_destroy(struct nsim_dev *nsim_dev)
return;
debugfs_remove(nsim_dev->take_snapshot);
nsim_dev_port_del_all(nsim_dev);
+ nsim_dev_linecard_del_all(nsim_dev);
nsim_dev_health_exit(nsim_dev);
nsim_dev_traps_exit(devlink);
nsim_dev_dummy_region_exit(nsim_dev);
@@ -180,20 +180,33 @@ struct nsim_dev_health {
int nsim_dev_health_init(struct nsim_dev *nsim_dev, struct devlink *devlink);
void nsim_dev_health_exit(struct nsim_dev *nsim_dev);
+struct nsim_dev_linecard;
+
struct nsim_dev_port {
struct list_head list;
struct devlink_port devlink_port;
+ struct nsim_dev_linecard *linecard;
unsigned int port_index;
struct dentry *ddir;
struct netdevsim *ns;
};
+struct nsim_dev;
+
+struct nsim_dev_linecard {
+ struct list_head list;
+ struct nsim_dev *nsim_dev;
+ unsigned int linecard_index;
+ struct dentry *ddir;
+};
+
struct nsim_dev {
struct nsim_bus_dev *nsim_bus_dev;
struct nsim_fib_data *fib_data;
struct nsim_trap_data *trap_data;
struct dentry *ddir;
struct dentry *ports_ddir;
+ struct dentry *linecards_ddir;
struct dentry *take_snapshot;
struct bpf_offload_dev *bpf_dev;
bool bpf_bind_accept;
@@ -206,6 +219,7 @@ struct nsim_dev {
struct netdev_phys_item_id switch_id;
struct list_head port_list;
struct mutex port_list_lock; /* protects port list */
+ struct list_head linecard_list;
bool fw_update_status;
u32 fw_update_overwrite_mask;
u32 max_macs;
@@ -287,6 +301,7 @@ struct nsim_bus_dev {
struct device dev;
struct list_head list;
unsigned int port_count;
+ unsigned int linecard_count;
struct net *initial_net; /* Purpose of this is to carry net pointer
* during the probe time only.
*/