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 | |
| 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>
| -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; |
