Commit 184c449f authored by Alexander Duyck's avatar Alexander Duyck Committed by David S. Miller

net: Add support for XPS with QoS via traffic classes

This patch adds support for setting and using XPS when QoS via traffic
classes is enabled.  With this change we will factor in the priority and
traffic class mapping of the packet and use that information to correctly
select the queue.

This allows us to define a set of queues for a given traffic class via
mqprio and then configure the XPS mapping for those queues so that the
traffic flows can avoid head-of-line blocking between the individual CPUs
if so desired.
Signed-off-by: default avatarAlexander Duyck <alexander.h.duyck@intel.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent 6234f874
...@@ -732,8 +732,8 @@ struct xps_dev_maps { ...@@ -732,8 +732,8 @@ struct xps_dev_maps {
struct rcu_head rcu; struct rcu_head rcu;
struct xps_map __rcu *cpu_map[0]; struct xps_map __rcu *cpu_map[0];
}; };
#define XPS_DEV_MAPS_SIZE (sizeof(struct xps_dev_maps) + \ #define XPS_DEV_MAPS_SIZE(_tcs) (sizeof(struct xps_dev_maps) + \
(nr_cpu_ids * sizeof(struct xps_map *))) (nr_cpu_ids * (_tcs) * sizeof(struct xps_map *)))
#endif /* CONFIG_XPS */ #endif /* CONFIG_XPS */
#define TC_MAX_QUEUE 16 #define TC_MAX_QUEUE 16
......
...@@ -2002,6 +2002,11 @@ static bool remove_xps_queue_cpu(struct net_device *dev, ...@@ -2002,6 +2002,11 @@ static bool remove_xps_queue_cpu(struct net_device *dev,
struct xps_dev_maps *dev_maps, struct xps_dev_maps *dev_maps,
int cpu, u16 offset, u16 count) int cpu, u16 offset, u16 count)
{ {
int num_tc = dev->num_tc ? : 1;
bool active = false;
int tci;
for (tci = cpu * num_tc; num_tc--; tci++) {
int i, j; int i, j;
for (i = count, j = offset; i--; j++) { for (i = count, j = offset; i--; j++) {
...@@ -2009,7 +2014,10 @@ static bool remove_xps_queue_cpu(struct net_device *dev, ...@@ -2009,7 +2014,10 @@ static bool remove_xps_queue_cpu(struct net_device *dev,
break; break;
} }
return i < 0; active |= i < 0;
}
return active;
} }
static void netif_reset_xps_queues(struct net_device *dev, u16 offset, static void netif_reset_xps_queues(struct net_device *dev, u16 offset,
...@@ -2086,20 +2094,28 @@ int netif_set_xps_queue(struct net_device *dev, const struct cpumask *mask, ...@@ -2086,20 +2094,28 @@ int netif_set_xps_queue(struct net_device *dev, const struct cpumask *mask,
u16 index) u16 index)
{ {
struct xps_dev_maps *dev_maps, *new_dev_maps = NULL; struct xps_dev_maps *dev_maps, *new_dev_maps = NULL;
int i, cpu, tci, numa_node_id = -2;
int maps_sz, num_tc = 1, tc = 0;
struct xps_map *map, *new_map; struct xps_map *map, *new_map;
int maps_sz = max_t(unsigned int, XPS_DEV_MAPS_SIZE, L1_CACHE_BYTES);
int cpu, numa_node_id = -2;
bool active = false; bool active = false;
if (dev->num_tc) {
num_tc = dev->num_tc;
tc = netdev_txq_to_tc(dev, index);
if (tc < 0)
return -EINVAL;
}
maps_sz = XPS_DEV_MAPS_SIZE(num_tc);
if (maps_sz < L1_CACHE_BYTES)
maps_sz = L1_CACHE_BYTES;
mutex_lock(&xps_map_mutex); mutex_lock(&xps_map_mutex);
dev_maps = xmap_dereference(dev->xps_maps); dev_maps = xmap_dereference(dev->xps_maps);
/* allocate memory for queue storage */ /* allocate memory for queue storage */
for_each_online_cpu(cpu) { for_each_cpu_and(cpu, cpu_online_mask, mask) {
if (!cpumask_test_cpu(cpu, mask))
continue;
if (!new_dev_maps) if (!new_dev_maps)
new_dev_maps = kzalloc(maps_sz, GFP_KERNEL); new_dev_maps = kzalloc(maps_sz, GFP_KERNEL);
if (!new_dev_maps) { if (!new_dev_maps) {
...@@ -2107,25 +2123,38 @@ int netif_set_xps_queue(struct net_device *dev, const struct cpumask *mask, ...@@ -2107,25 +2123,38 @@ int netif_set_xps_queue(struct net_device *dev, const struct cpumask *mask,
return -ENOMEM; return -ENOMEM;
} }
map = dev_maps ? xmap_dereference(dev_maps->cpu_map[cpu]) : tci = cpu * num_tc + tc;
map = dev_maps ? xmap_dereference(dev_maps->cpu_map[tci]) :
NULL; NULL;
map = expand_xps_map(map, cpu, index); map = expand_xps_map(map, cpu, index);
if (!map) if (!map)
goto error; goto error;
RCU_INIT_POINTER(new_dev_maps->cpu_map[cpu], map); RCU_INIT_POINTER(new_dev_maps->cpu_map[tci], map);
} }
if (!new_dev_maps) if (!new_dev_maps)
goto out_no_new_maps; goto out_no_new_maps;
for_each_possible_cpu(cpu) { for_each_possible_cpu(cpu) {
/* copy maps belonging to foreign traffic classes */
for (i = tc, tci = cpu * num_tc; dev_maps && i--; tci++) {
/* fill in the new device map from the old device map */
map = xmap_dereference(dev_maps->cpu_map[tci]);
RCU_INIT_POINTER(new_dev_maps->cpu_map[tci], map);
}
/* We need to explicitly update tci as prevous loop
* could break out early if dev_maps is NULL.
*/
tci = cpu * num_tc + tc;
if (cpumask_test_cpu(cpu, mask) && cpu_online(cpu)) { if (cpumask_test_cpu(cpu, mask) && cpu_online(cpu)) {
/* add queue to CPU maps */ /* add queue to CPU maps */
int pos = 0; int pos = 0;
map = xmap_dereference(new_dev_maps->cpu_map[cpu]); map = xmap_dereference(new_dev_maps->cpu_map[tci]);
while ((pos < map->len) && (map->queues[pos] != index)) while ((pos < map->len) && (map->queues[pos] != index))
pos++; pos++;
...@@ -2139,26 +2168,36 @@ int netif_set_xps_queue(struct net_device *dev, const struct cpumask *mask, ...@@ -2139,26 +2168,36 @@ int netif_set_xps_queue(struct net_device *dev, const struct cpumask *mask,
#endif #endif
} else if (dev_maps) { } else if (dev_maps) {
/* fill in the new device map from the old device map */ /* fill in the new device map from the old device map */
map = xmap_dereference(dev_maps->cpu_map[cpu]); map = xmap_dereference(dev_maps->cpu_map[tci]);
RCU_INIT_POINTER(new_dev_maps->cpu_map[cpu], map); RCU_INIT_POINTER(new_dev_maps->cpu_map[tci], map);
} }
/* copy maps belonging to foreign traffic classes */
for (i = num_tc - tc, tci++; dev_maps && --i; tci++) {
/* fill in the new device map from the old device map */
map = xmap_dereference(dev_maps->cpu_map[tci]);
RCU_INIT_POINTER(new_dev_maps->cpu_map[tci], map);
}
} }
rcu_assign_pointer(dev->xps_maps, new_dev_maps); rcu_assign_pointer(dev->xps_maps, new_dev_maps);
/* Cleanup old maps */ /* Cleanup old maps */
if (dev_maps) { if (!dev_maps)
goto out_no_old_maps;
for_each_possible_cpu(cpu) { for_each_possible_cpu(cpu) {
new_map = xmap_dereference(new_dev_maps->cpu_map[cpu]); for (i = num_tc, tci = cpu * num_tc; i--; tci++) {
map = xmap_dereference(dev_maps->cpu_map[cpu]); new_map = xmap_dereference(new_dev_maps->cpu_map[tci]);
map = xmap_dereference(dev_maps->cpu_map[tci]);
if (map && map != new_map) if (map && map != new_map)
kfree_rcu(map, rcu); kfree_rcu(map, rcu);
} }
}
kfree_rcu(dev_maps, rcu); kfree_rcu(dev_maps, rcu);
}
out_no_old_maps:
dev_maps = new_dev_maps; dev_maps = new_dev_maps;
active = true; active = true;
...@@ -2173,11 +2212,12 @@ int netif_set_xps_queue(struct net_device *dev, const struct cpumask *mask, ...@@ -2173,11 +2212,12 @@ int netif_set_xps_queue(struct net_device *dev, const struct cpumask *mask,
/* removes queue from unused CPUs */ /* removes queue from unused CPUs */
for_each_possible_cpu(cpu) { for_each_possible_cpu(cpu) {
if (cpumask_test_cpu(cpu, mask) && cpu_online(cpu)) for (i = tc, tci = cpu * num_tc; i--; tci++)
continue; active |= remove_xps_queue(dev_maps, tci, index);
if (!cpumask_test_cpu(cpu, mask) || !cpu_online(cpu))
if (remove_xps_queue(dev_maps, cpu, index)) active |= remove_xps_queue(dev_maps, tci, index);
active = true; for (i = num_tc - tc, tci++; --i; tci++)
active |= remove_xps_queue(dev_maps, tci, index);
} }
/* free map if not active */ /* free map if not active */
...@@ -2193,12 +2233,15 @@ int netif_set_xps_queue(struct net_device *dev, const struct cpumask *mask, ...@@ -2193,12 +2233,15 @@ int netif_set_xps_queue(struct net_device *dev, const struct cpumask *mask,
error: error:
/* remove any maps that we added */ /* remove any maps that we added */
for_each_possible_cpu(cpu) { for_each_possible_cpu(cpu) {
new_map = xmap_dereference(new_dev_maps->cpu_map[cpu]); for (i = num_tc, tci = cpu * num_tc; i--; tci++) {
map = dev_maps ? xmap_dereference(dev_maps->cpu_map[cpu]) : new_map = xmap_dereference(new_dev_maps->cpu_map[tci]);
map = dev_maps ?
xmap_dereference(dev_maps->cpu_map[tci]) :
NULL; NULL;
if (new_map && new_map != map) if (new_map && new_map != map)
kfree(new_map); kfree(new_map);
} }
}
mutex_unlock(&xps_map_mutex); mutex_unlock(&xps_map_mutex);
...@@ -3158,8 +3201,14 @@ static inline int get_xps_queue(struct net_device *dev, struct sk_buff *skb) ...@@ -3158,8 +3201,14 @@ static inline int get_xps_queue(struct net_device *dev, struct sk_buff *skb)
rcu_read_lock(); rcu_read_lock();
dev_maps = rcu_dereference(dev->xps_maps); dev_maps = rcu_dereference(dev->xps_maps);
if (dev_maps) { if (dev_maps) {
map = rcu_dereference( unsigned int tci = skb->sender_cpu - 1;
dev_maps->cpu_map[skb->sender_cpu - 1]);
if (dev->num_tc) {
tci *= dev->num_tc;
tci += netdev_get_prio_tc_map(dev, skb->priority);
}
map = rcu_dereference(dev_maps->cpu_map[tci]);
if (map) { if (map) {
if (map->len == 1) if (map->len == 1)
queue_index = map->queues[0]; queue_index = map->queues[0];
......
...@@ -1210,33 +1210,42 @@ static ssize_t show_xps_map(struct netdev_queue *queue, ...@@ -1210,33 +1210,42 @@ static ssize_t show_xps_map(struct netdev_queue *queue,
struct netdev_queue_attribute *attribute, char *buf) struct netdev_queue_attribute *attribute, char *buf)
{ {
struct net_device *dev = queue->dev; struct net_device *dev = queue->dev;
int cpu, len, num_tc = 1, tc = 0;
struct xps_dev_maps *dev_maps; struct xps_dev_maps *dev_maps;
cpumask_var_t mask; cpumask_var_t mask;
unsigned long index; unsigned long index;
int i, len;
if (!zalloc_cpumask_var(&mask, GFP_KERNEL)) if (!zalloc_cpumask_var(&mask, GFP_KERNEL))
return -ENOMEM; return -ENOMEM;
index = get_netdev_queue_index(queue); index = get_netdev_queue_index(queue);
if (dev->num_tc) {
num_tc = dev->num_tc;
tc = netdev_txq_to_tc(dev, index);
if (tc < 0)
return -EINVAL;
}
rcu_read_lock(); rcu_read_lock();
dev_maps = rcu_dereference(dev->xps_maps); dev_maps = rcu_dereference(dev->xps_maps);
if (dev_maps) { if (dev_maps) {
for_each_possible_cpu(i) { for_each_possible_cpu(cpu) {
struct xps_map *map = int i, tci = cpu * num_tc + tc;
rcu_dereference(dev_maps->cpu_map[i]); struct xps_map *map;
if (map) {
int j; map = rcu_dereference(dev_maps->cpu_map[tci]);
for (j = 0; j < map->len; j++) { if (!map)
if (map->queues[j] == index) { continue;
cpumask_set_cpu(i, mask);
for (i = map->len; i--;) {
if (map->queues[i] == index) {
cpumask_set_cpu(cpu, mask);
break; break;
} }
} }
} }
} }
}
rcu_read_unlock(); rcu_read_unlock();
len = snprintf(buf, PAGE_SIZE, "%*pb\n", cpumask_pr_args(mask)); len = snprintf(buf, PAGE_SIZE, "%*pb\n", cpumask_pr_args(mask));
......
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