diff options
author | Wanlong Gao <gaowanlong@cn.fujitsu.com> | 2013-01-24 18:51:31 -0500 |
---|---|---|
committer | David S. Miller <davem@davemloft.net> | 2013-01-27 01:22:51 -0500 |
commit | 8de4b2f3ae90c8fc0f17eeaab87d5a951b66ee17 (patch) | |
tree | f46a5ec6dd1e2654d61ee72606bdd5b672c739c3 /drivers/net/virtio_net.c | |
parent | 8898c21cf37d04041863e7ecf53707dff504bda0 (diff) |
virtio-net: reset virtqueue affinity when doing cpu hotplug
Add a cpu notifier to virtio-net, so that we can reset the
virtqueue affinity if the cpu hotplug happens. It improve
the performance through enabling or disabling the virtqueue
affinity after doing cpu hotplug.
Cc: Rusty Russell <rusty@rustcorp.com.au>
Cc: "Michael S. Tsirkin" <mst@redhat.com>
Cc: Jason Wang <jasowang@redhat.com>
Cc: Eric Dumazet <erdnetdev@gmail.com>
Cc: "David S. Miller" <davem@davemloft.net>
Cc: virtualization@lists.linux-foundation.org
Cc: netdev@vger.kernel.org
Signed-off-by: Wanlong Gao <gaowanlong@cn.fujitsu.com>
Acked-by: Michael S. Tsirkin <mst@redhat.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'drivers/net/virtio_net.c')
-rw-r--r-- | drivers/net/virtio_net.c | 33 |
1 files changed, 33 insertions, 0 deletions
diff --git a/drivers/net/virtio_net.c b/drivers/net/virtio_net.c index 2f6fe9b6b17b..35c00c5ea02a 100644 --- a/drivers/net/virtio_net.c +++ b/drivers/net/virtio_net.c | |||
@@ -26,6 +26,7 @@ | |||
26 | #include <linux/scatterlist.h> | 26 | #include <linux/scatterlist.h> |
27 | #include <linux/if_vlan.h> | 27 | #include <linux/if_vlan.h> |
28 | #include <linux/slab.h> | 28 | #include <linux/slab.h> |
29 | #include <linux/cpu.h> | ||
29 | 30 | ||
30 | static int napi_weight = 128; | 31 | static int napi_weight = 128; |
31 | module_param(napi_weight, int, 0444); | 32 | module_param(napi_weight, int, 0444); |
@@ -126,6 +127,9 @@ struct virtnet_info { | |||
126 | 127 | ||
127 | /* Per-cpu variable to show the mapping from CPU to virtqueue */ | 128 | /* Per-cpu variable to show the mapping from CPU to virtqueue */ |
128 | int __percpu *vq_index; | 129 | int __percpu *vq_index; |
130 | |||
131 | /* CPU hot plug notifier */ | ||
132 | struct notifier_block nb; | ||
129 | }; | 133 | }; |
130 | 134 | ||
131 | struct skb_vnet_hdr { | 135 | struct skb_vnet_hdr { |
@@ -1067,6 +1071,26 @@ static void virtnet_set_affinity(struct virtnet_info *vi) | |||
1067 | vi->affinity_hint_set = true; | 1071 | vi->affinity_hint_set = true; |
1068 | } | 1072 | } |
1069 | 1073 | ||
1074 | static int virtnet_cpu_callback(struct notifier_block *nfb, | ||
1075 | unsigned long action, void *hcpu) | ||
1076 | { | ||
1077 | struct virtnet_info *vi = container_of(nfb, struct virtnet_info, nb); | ||
1078 | |||
1079 | switch(action & ~CPU_TASKS_FROZEN) { | ||
1080 | case CPU_ONLINE: | ||
1081 | case CPU_DOWN_FAILED: | ||
1082 | case CPU_DEAD: | ||
1083 | virtnet_set_affinity(vi); | ||
1084 | break; | ||
1085 | case CPU_DOWN_PREPARE: | ||
1086 | virtnet_clean_affinity(vi, (long)hcpu); | ||
1087 | break; | ||
1088 | default: | ||
1089 | break; | ||
1090 | } | ||
1091 | return NOTIFY_OK; | ||
1092 | } | ||
1093 | |||
1070 | static void virtnet_get_ringparam(struct net_device *dev, | 1094 | static void virtnet_get_ringparam(struct net_device *dev, |
1071 | struct ethtool_ringparam *ring) | 1095 | struct ethtool_ringparam *ring) |
1072 | { | 1096 | { |
@@ -1541,6 +1565,13 @@ static int virtnet_probe(struct virtio_device *vdev) | |||
1541 | } | 1565 | } |
1542 | } | 1566 | } |
1543 | 1567 | ||
1568 | vi->nb.notifier_call = &virtnet_cpu_callback; | ||
1569 | err = register_hotcpu_notifier(&vi->nb); | ||
1570 | if (err) { | ||
1571 | pr_debug("virtio_net: registering cpu notifier failed\n"); | ||
1572 | goto free_recv_bufs; | ||
1573 | } | ||
1574 | |||
1544 | /* Assume link up if device can't report link status, | 1575 | /* Assume link up if device can't report link status, |
1545 | otherwise get link status from config. */ | 1576 | otherwise get link status from config. */ |
1546 | if (virtio_has_feature(vi->vdev, VIRTIO_NET_F_STATUS)) { | 1577 | if (virtio_has_feature(vi->vdev, VIRTIO_NET_F_STATUS)) { |
@@ -1587,6 +1618,8 @@ static void virtnet_remove(struct virtio_device *vdev) | |||
1587 | { | 1618 | { |
1588 | struct virtnet_info *vi = vdev->priv; | 1619 | struct virtnet_info *vi = vdev->priv; |
1589 | 1620 | ||
1621 | unregister_hotcpu_notifier(&vi->nb); | ||
1622 | |||
1590 | /* Prevent config work handler from accessing the device. */ | 1623 | /* Prevent config work handler from accessing the device. */ |
1591 | mutex_lock(&vi->config_lock); | 1624 | mutex_lock(&vi->config_lock); |
1592 | vi->config_enable = false; | 1625 | vi->config_enable = false; |