Commit d8e18ee0 authored by stephen hemminger's avatar stephen hemminger Committed by David S. Miller

netvsc: enhance transmit select_queue

The netvsc select queue function was missing many of the flow caching
features that exist in default tx queue selection. Add the same
logic to remember queue based on socket and implement two level
mapping (like RSS).
Signed-off-by: default avatarStephen Hemminger <sthemmin@microsoft.com>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent ff4a4419
...@@ -191,22 +191,41 @@ static void *init_ppi_data(struct rndis_message *msg, u32 ppi_size, ...@@ -191,22 +191,41 @@ static void *init_ppi_data(struct rndis_message *msg, u32 ppi_size,
return ppi; return ppi;
} }
/*
* Select queue for transmit.
*
* If a valid queue has already been assigned, then use that.
* Otherwise compute tx queue based on hash and the send table.
*
* This is basically similar to default (__netdev_pick_tx) with the added step
* of using the host send_table when no other queue has been assigned.
*
* TODO support XPS - but get_xps_queue not exported
*/
static u16 netvsc_select_queue(struct net_device *ndev, struct sk_buff *skb, static u16 netvsc_select_queue(struct net_device *ndev, struct sk_buff *skb,
void *accel_priv, select_queue_fallback_t fallback) void *accel_priv, select_queue_fallback_t fallback)
{ {
struct net_device_context *net_device_ctx = netdev_priv(ndev); struct net_device_context *net_device_ctx = netdev_priv(ndev);
struct netvsc_device *nvsc_dev = net_device_ctx->nvdev; struct netvsc_device *nvsc_dev = net_device_ctx->nvdev;
u32 hash; struct sock *sk = skb->sk;
u16 q_idx = 0; int q_idx = sk_tx_queue_get(sk);
if (nvsc_dev == NULL || ndev->real_num_tx_queues <= 1) if (q_idx < 0 || skb->ooo_okay ||
return 0; q_idx >= ndev->real_num_tx_queues) {
u16 hash = __skb_tx_hash(ndev, skb, VRSS_SEND_TAB_SIZE);
int new_idx;
new_idx = nvsc_dev->send_table[hash]
% nvsc_dev->num_chn;
hash = skb_get_hash(skb); if (q_idx != new_idx && sk &&
q_idx = nvsc_dev->send_table[hash % VRSS_SEND_TAB_SIZE] % sk_fullsock(sk) && rcu_access_pointer(sk->sk_dst_cache))
ndev->real_num_tx_queues; sk_tx_queue_set(sk, new_idx);
q_idx = new_idx;
}
if (!nvsc_dev->chn_table[q_idx]) if (unlikely(!nvsc_dev->chn_table[q_idx]))
q_idx = 0; q_idx = 0;
return q_idx; return q_idx;
......
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