Commit fe822240 authored by Tom Herbert's avatar Tom Herbert Committed by David S. Miller

net: Simplify RX queue allocation

This patch move RX queue allocation to alloc_netdev_mq and freeing of
the queues to free_netdev (symmetric to TX queue allocation).  Each
kobject RX queue takes a reference to the queue's device so that the
device can't be freed before all the kobjects have been released-- this
obviates the need for reference counts specific to RX queues.
Signed-off-by: default avatarTom Herbert <therbert@google.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent ed9af2e8
...@@ -592,8 +592,7 @@ struct netdev_rx_queue { ...@@ -592,8 +592,7 @@ struct netdev_rx_queue {
struct rps_map __rcu *rps_map; struct rps_map __rcu *rps_map;
struct rps_dev_flow_table __rcu *rps_flow_table; struct rps_dev_flow_table __rcu *rps_flow_table;
struct kobject kobj; struct kobject kobj;
struct netdev_rx_queue *first; struct net_device *dev;
atomic_t count;
} ____cacheline_aligned_in_smp; } ____cacheline_aligned_in_smp;
#endif /* CONFIG_RPS */ #endif /* CONFIG_RPS */
......
...@@ -5051,12 +5051,8 @@ static int netif_alloc_rx_queues(struct net_device *dev) ...@@ -5051,12 +5051,8 @@ static int netif_alloc_rx_queues(struct net_device *dev)
} }
dev->_rx = rx; dev->_rx = rx;
/*
* Set a pointer to first element in the array which holds the
* reference count.
*/
for (i = 0; i < count; i++) for (i = 0; i < count; i++)
rx[i].first = rx; rx[i].dev = dev;
#endif #endif
return 0; return 0;
} }
...@@ -5132,10 +5128,6 @@ int register_netdevice(struct net_device *dev) ...@@ -5132,10 +5128,6 @@ int register_netdevice(struct net_device *dev)
dev->iflink = -1; dev->iflink = -1;
ret = netif_alloc_rx_queues(dev);
if (ret)
goto out;
netdev_init_queues(dev); netdev_init_queues(dev);
/* Init, if this function is available */ /* Init, if this function is available */
...@@ -5601,6 +5593,8 @@ struct net_device *alloc_netdev_mq(int sizeof_priv, const char *name, ...@@ -5601,6 +5593,8 @@ struct net_device *alloc_netdev_mq(int sizeof_priv, const char *name,
#ifdef CONFIG_RPS #ifdef CONFIG_RPS
dev->num_rx_queues = queue_count; dev->num_rx_queues = queue_count;
dev->real_num_rx_queues = queue_count; dev->real_num_rx_queues = queue_count;
if (netif_alloc_rx_queues(dev))
goto free_pcpu;
#endif #endif
dev->gso_max_size = GSO_MAX_SIZE; dev->gso_max_size = GSO_MAX_SIZE;
...@@ -5618,6 +5612,10 @@ struct net_device *alloc_netdev_mq(int sizeof_priv, const char *name, ...@@ -5618,6 +5612,10 @@ struct net_device *alloc_netdev_mq(int sizeof_priv, const char *name,
free_pcpu: free_pcpu:
free_percpu(dev->pcpu_refcnt); free_percpu(dev->pcpu_refcnt);
kfree(dev->_tx); kfree(dev->_tx);
#ifdef CONFIG_RPS
kfree(dev->_rx);
#endif
free_p: free_p:
kfree(p); kfree(p);
return NULL; return NULL;
...@@ -5639,6 +5637,9 @@ void free_netdev(struct net_device *dev) ...@@ -5639,6 +5637,9 @@ void free_netdev(struct net_device *dev)
release_net(dev_net(dev)); release_net(dev_net(dev));
kfree(dev->_tx); kfree(dev->_tx);
#ifdef CONFIG_RPS
kfree(dev->_rx);
#endif
kfree(rcu_dereference_raw(dev->ingress_queue)); kfree(rcu_dereference_raw(dev->ingress_queue));
......
...@@ -706,7 +706,6 @@ static struct attribute *rx_queue_default_attrs[] = { ...@@ -706,7 +706,6 @@ static struct attribute *rx_queue_default_attrs[] = {
static void rx_queue_release(struct kobject *kobj) static void rx_queue_release(struct kobject *kobj)
{ {
struct netdev_rx_queue *queue = to_rx_queue(kobj); struct netdev_rx_queue *queue = to_rx_queue(kobj);
struct netdev_rx_queue *first = queue->first;
struct rps_map *map; struct rps_map *map;
struct rps_dev_flow_table *flow_table; struct rps_dev_flow_table *flow_table;
...@@ -719,8 +718,7 @@ static void rx_queue_release(struct kobject *kobj) ...@@ -719,8 +718,7 @@ static void rx_queue_release(struct kobject *kobj)
if (flow_table) if (flow_table)
call_rcu(&flow_table->rcu, rps_dev_flow_table_release); call_rcu(&flow_table->rcu, rps_dev_flow_table_release);
if (atomic_dec_and_test(&first->count)) dev_put(queue->dev);
kfree(first);
} }
static struct kobj_type rx_queue_ktype = { static struct kobj_type rx_queue_ktype = {
...@@ -732,7 +730,6 @@ static struct kobj_type rx_queue_ktype = { ...@@ -732,7 +730,6 @@ static struct kobj_type rx_queue_ktype = {
static int rx_queue_add_kobject(struct net_device *net, int index) static int rx_queue_add_kobject(struct net_device *net, int index)
{ {
struct netdev_rx_queue *queue = net->_rx + index; struct netdev_rx_queue *queue = net->_rx + index;
struct netdev_rx_queue *first = queue->first;
struct kobject *kobj = &queue->kobj; struct kobject *kobj = &queue->kobj;
int error = 0; int error = 0;
...@@ -745,7 +742,7 @@ static int rx_queue_add_kobject(struct net_device *net, int index) ...@@ -745,7 +742,7 @@ static int rx_queue_add_kobject(struct net_device *net, int index)
} }
kobject_uevent(kobj, KOBJ_ADD); kobject_uevent(kobj, KOBJ_ADD);
atomic_inc(&first->count); dev_hold(queue->dev);
return error; return error;
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment