fbnic_csr_get_regs(fbn->fbd, data, ®s->version);
 }
 
+static struct fbnic_net *fbnic_clone_create(struct fbnic_net *orig)
+{
+       struct fbnic_net *clone;
+
+       clone = kmemdup(orig, sizeof(*orig), GFP_KERNEL);
+       if (!clone)
+               return NULL;
+
+       memset(clone->tx, 0, sizeof(clone->tx));
+       memset(clone->rx, 0, sizeof(clone->rx));
+       memset(clone->napi, 0, sizeof(clone->napi));
+       return clone;
+}
+
+static void fbnic_clone_swap_cfg(struct fbnic_net *orig,
+                                struct fbnic_net *clone)
+{
+       swap(clone->rcq_size, orig->rcq_size);
+       swap(clone->hpq_size, orig->hpq_size);
+       swap(clone->ppq_size, orig->ppq_size);
+       swap(clone->txq_size, orig->txq_size);
+       swap(clone->num_rx_queues, orig->num_rx_queues);
+       swap(clone->num_tx_queues, orig->num_tx_queues);
+       swap(clone->num_napi, orig->num_napi);
+}
+
+static void fbnic_aggregate_vector_counters(struct fbnic_net *fbn,
+                                           struct fbnic_napi_vector *nv)
+{
+       int i, j;
+
+       for (i = 0; i < nv->txt_count; i++) {
+               fbnic_aggregate_ring_tx_counters(fbn, &nv->qt[i].sub0);
+               fbnic_aggregate_ring_tx_counters(fbn, &nv->qt[i].sub1);
+               fbnic_aggregate_ring_tx_counters(fbn, &nv->qt[i].cmpl);
+       }
+
+       for (j = 0; j < nv->rxt_count; j++, i++) {
+               fbnic_aggregate_ring_rx_counters(fbn, &nv->qt[i].sub0);
+               fbnic_aggregate_ring_rx_counters(fbn, &nv->qt[i].sub1);
+               fbnic_aggregate_ring_rx_counters(fbn, &nv->qt[i].cmpl);
+       }
+}
+
+static void fbnic_clone_swap(struct fbnic_net *orig,
+                            struct fbnic_net *clone)
+{
+       struct fbnic_dev *fbd = orig->fbd;
+       unsigned int i;
+
+       for (i = 0; i < max(clone->num_napi, orig->num_napi); i++)
+               fbnic_synchronize_irq(fbd, FBNIC_NON_NAPI_VECTORS + i);
+       for (i = 0; i < orig->num_napi; i++)
+               fbnic_aggregate_vector_counters(orig, orig->napi[i]);
+
+       fbnic_clone_swap_cfg(orig, clone);
+
+       for (i = 0; i < ARRAY_SIZE(orig->napi); i++)
+               swap(clone->napi[i], orig->napi[i]);
+       for (i = 0; i < ARRAY_SIZE(orig->tx); i++)
+               swap(clone->tx[i], orig->tx[i]);
+       for (i = 0; i < ARRAY_SIZE(orig->rx); i++)
+               swap(clone->rx[i], orig->rx[i]);
+}
+
+static void fbnic_clone_free(struct fbnic_net *clone)
+{
+       kfree(clone);
+}
+
 static void fbnic_get_strings(struct net_device *dev, u32 sset, u8 *data)
 {
        int i;
        struct fbnic_net *fbn = netdev_priv(netdev);
        unsigned int max_napis, standalone;
        struct fbnic_dev *fbd = fbn->fbd;
+       struct fbnic_net *clone;
+       int err;
 
        max_napis = fbd->num_irqs - FBNIC_NON_NAPI_VECTORS;
        standalone = ch->rx_count + ch->tx_count;
                return 0;
        }
 
-       return -EBUSY;
+       clone = fbnic_clone_create(fbn);
+       if (!clone)
+               return -ENOMEM;
+
+       fbnic_set_queues(clone, ch, max_napis);
+
+       err = fbnic_alloc_napi_vectors(clone);
+       if (err)
+               goto err_free_clone;
+
+       err = fbnic_alloc_resources(clone);
+       if (err)
+               goto err_free_napis;
+
+       fbnic_down_noidle(fbn);
+       err = fbnic_wait_all_queues_idle(fbn->fbd, true);
+       if (err)
+               goto err_start_stack;
+
+       err = fbnic_set_netif_queues(clone);
+       if (err)
+               goto err_start_stack;
+
+       /* Nothing can fail past this point */
+       fbnic_flush(fbn);
+
+       fbnic_clone_swap(fbn, clone);
+
+       /* Reset RSS indirection table */
+       fbnic_reset_indir_tbl(fbn);
+
+       fbnic_up(fbn);
+
+       fbnic_free_resources(clone);
+       fbnic_free_napi_vectors(clone);
+       fbnic_clone_free(clone);
+
+       return 0;
+
+err_start_stack:
+       fbnic_flush(fbn);
+       fbnic_up(fbn);
+       fbnic_free_resources(clone);
+err_free_napis:
+       fbnic_free_napi_vectors(clone);
+err_free_clone:
+       fbnic_clone_free(clone);
+       return err;
 }
 
 static int