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: Tom Herbert <therbert@google.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
committed by
David S. Miller
parent
ed9af2e839
commit
fe8222406c
@@ -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)
|
|||||||
}
|
}
|
||||||
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)
|
|||||||
|
|
||||||
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,
|
|||||||
#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,
|
|||||||
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)
|
|||||||
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[] = {
|
|||||||
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)
|
|||||||
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 = {
|
|||||||
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)
|
|||||||
}
|
}
|
||||||
|
|
||||||
kobject_uevent(kobj, KOBJ_ADD);
|
kobject_uevent(kobj, KOBJ_ADD);
|
||||||
atomic_inc(&first->count);
|
dev_hold(queue->dev);
|
||||||
|
|
||||||
return error;
|
return error;
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user