Commit fe36cbe0 authored by Michael S. Tsirkin's avatar Michael S. Tsirkin

virtio_net: clear MTU when out of range

virtio attempts to clear the MTU feature bit if the value is out of the
supported range, but this has no real effect since FEATURES_OK has
already been set.

Fix this up by checking the MTU in the new validate callback.

Fixes: 14de9d11 ("virtio-net: Add initial MTU advice feature")
Signed-off-by: default avatarMichael S. Tsirkin <mst@redhat.com>
parent 404123c2
...@@ -2230,14 +2230,8 @@ static bool virtnet_validate_features(struct virtio_device *vdev) ...@@ -2230,14 +2230,8 @@ static bool virtnet_validate_features(struct virtio_device *vdev)
#define MIN_MTU ETH_MIN_MTU #define MIN_MTU ETH_MIN_MTU
#define MAX_MTU ETH_MAX_MTU #define MAX_MTU ETH_MAX_MTU
static int virtnet_probe(struct virtio_device *vdev) static int virtnet_validate(struct virtio_device *vdev)
{ {
int i, err;
struct net_device *dev;
struct virtnet_info *vi;
u16 max_queue_pairs;
int mtu;
if (!vdev->config->get) { if (!vdev->config->get) {
dev_err(&vdev->dev, "%s failure: config access disabled\n", dev_err(&vdev->dev, "%s failure: config access disabled\n",
__func__); __func__);
...@@ -2247,6 +2241,25 @@ static int virtnet_probe(struct virtio_device *vdev) ...@@ -2247,6 +2241,25 @@ static int virtnet_probe(struct virtio_device *vdev)
if (!virtnet_validate_features(vdev)) if (!virtnet_validate_features(vdev))
return -EINVAL; return -EINVAL;
if (virtio_has_feature(vdev, VIRTIO_NET_F_MTU)) {
int mtu = virtio_cread16(vdev,
offsetof(struct virtio_net_config,
mtu));
if (mtu < MIN_MTU)
__virtio_clear_bit(vdev, VIRTIO_NET_F_MTU);
}
return 0;
}
static int virtnet_probe(struct virtio_device *vdev)
{
int i, err;
struct net_device *dev;
struct virtnet_info *vi;
u16 max_queue_pairs;
int mtu;
/* Find if host supports multiqueue virtio_net device */ /* Find if host supports multiqueue virtio_net device */
err = virtio_cread_feature(vdev, VIRTIO_NET_F_MQ, err = virtio_cread_feature(vdev, VIRTIO_NET_F_MQ,
struct virtio_net_config, struct virtio_net_config,
...@@ -2362,11 +2375,16 @@ static int virtnet_probe(struct virtio_device *vdev) ...@@ -2362,11 +2375,16 @@ static int virtnet_probe(struct virtio_device *vdev)
offsetof(struct virtio_net_config, offsetof(struct virtio_net_config,
mtu)); mtu));
if (mtu < dev->min_mtu) { if (mtu < dev->min_mtu) {
__virtio_clear_bit(vdev, VIRTIO_NET_F_MTU); /* Should never trigger: MTU was previously validated
} else { * in virtnet_validate.
*/
dev_err(&vdev->dev, "device MTU appears to have changed "
"it is now %d < %d", mtu, dev->min_mtu);
goto free_stats;
}
dev->mtu = mtu; dev->mtu = mtu;
dev->max_mtu = mtu; dev->max_mtu = mtu;
}
/* TODO: size buffers correctly in this case. */ /* TODO: size buffers correctly in this case. */
if (dev->mtu > ETH_DATA_LEN) if (dev->mtu > ETH_DATA_LEN)
...@@ -2548,6 +2566,7 @@ static struct virtio_driver virtio_net_driver = { ...@@ -2548,6 +2566,7 @@ static struct virtio_driver virtio_net_driver = {
.driver.name = KBUILD_MODNAME, .driver.name = KBUILD_MODNAME,
.driver.owner = THIS_MODULE, .driver.owner = THIS_MODULE,
.id_table = id_table, .id_table = id_table,
.validate = virtnet_validate,
.probe = virtnet_probe, .probe = virtnet_probe,
.remove = virtnet_remove, .remove = virtnet_remove,
.config_changed = virtnet_config_changed, .config_changed = virtnet_config_changed,
......
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