Commit c624896e authored by Rusty Russell's avatar Rusty Russell

virtio: Rename set_features to finalize_features

Rather than explicitly handing the features to the lower-level, we just
hand the virtio_device and have it set the features.  This make it clear
that it has the chance to manipulate the features of the device at this
point (and that all feature negotiation is already done).
Signed-off-by: default avatarRusty Russell <rusty@rustcorp.com.au>
parent dd7c7bc4
...@@ -98,16 +98,17 @@ static u32 lg_get_features(struct virtio_device *vdev) ...@@ -98,16 +98,17 @@ static u32 lg_get_features(struct virtio_device *vdev)
return features; return features;
} }
static void lg_set_features(struct virtio_device *vdev, u32 features) static void lg_finalize_features(struct virtio_device *vdev)
{ {
unsigned int i; unsigned int i, bits;
struct lguest_device_desc *desc = to_lgdev(vdev)->desc; struct lguest_device_desc *desc = to_lgdev(vdev)->desc;
/* Second half of bitmap is features we accept. */ /* Second half of bitmap is features we accept. */
u8 *out_features = lg_features(desc) + desc->feature_len; u8 *out_features = lg_features(desc) + desc->feature_len;
memset(out_features, 0, desc->feature_len); memset(out_features, 0, desc->feature_len);
for (i = 0; i < min(desc->feature_len * 8, 32); i++) { bits = min_t(unsigned, desc->feature_len, sizeof(vdev->features)) * 8;
if (features & (1 << i)) for (i = 0; i < bits; i++) {
if (test_bit(i, vdev->features))
out_features[i / 8] |= (1 << (i % 8)); out_features[i / 8] |= (1 << (i % 8));
} }
} }
...@@ -297,7 +298,7 @@ static void lg_del_vq(struct virtqueue *vq) ...@@ -297,7 +298,7 @@ static void lg_del_vq(struct virtqueue *vq)
/* The ops structure which hooks everything together. */ /* The ops structure which hooks everything together. */
static struct virtio_config_ops lguest_config_ops = { static struct virtio_config_ops lguest_config_ops = {
.get_features = lg_get_features, .get_features = lg_get_features,
.set_features = lg_set_features, .finalize_features = lg_finalize_features,
.get = lg_get, .get = lg_get,
.set = lg_set, .set = lg_set,
.get_status = lg_get_status, .get_status = lg_get_status,
......
...@@ -88,16 +88,17 @@ static u32 kvm_get_features(struct virtio_device *vdev) ...@@ -88,16 +88,17 @@ static u32 kvm_get_features(struct virtio_device *vdev)
return features; return features;
} }
static void kvm_set_features(struct virtio_device *vdev, u32 features) static void kvm_finalize_features(struct virtio_device *vdev)
{ {
unsigned int i; unsigned int i, bits;
struct kvm_device_desc *desc = to_kvmdev(vdev)->desc; struct kvm_device_desc *desc = to_kvmdev(vdev)->desc;
/* Second half of bitmap is features we accept. */ /* Second half of bitmap is features we accept. */
u8 *out_features = kvm_vq_features(desc) + desc->feature_len; u8 *out_features = kvm_vq_features(desc) + desc->feature_len;
memset(out_features, 0, desc->feature_len); memset(out_features, 0, desc->feature_len);
for (i = 0; i < min(desc->feature_len * 8, 32); i++) { bits = min_t(unsigned, desc->feature_len, sizeof(vdev->features)) * 8;
if (features & (1 << i)) for (i = 0; i < bits; i++) {
if (test_bit(i, vdev->features))
out_features[i / 8] |= (1 << (i % 8)); out_features[i / 8] |= (1 << (i % 8));
} }
} }
...@@ -223,7 +224,7 @@ static void kvm_del_vq(struct virtqueue *vq) ...@@ -223,7 +224,7 @@ static void kvm_del_vq(struct virtqueue *vq)
*/ */
static struct virtio_config_ops kvm_vq_configspace_ops = { static struct virtio_config_ops kvm_vq_configspace_ops = {
.get_features = kvm_get_features, .get_features = kvm_get_features,
.set_features = kvm_set_features, .finalize_features = kvm_finalize_features,
.get = kvm_get, .get = kvm_get,
.set = kvm_set, .set = kvm_set,
.get_status = kvm_get_status, .get_status = kvm_get_status,
......
...@@ -113,7 +113,7 @@ static int virtio_dev_probe(struct device *_d) ...@@ -113,7 +113,7 @@ static int virtio_dev_probe(struct device *_d)
set_bit(f, dev->features); set_bit(f, dev->features);
} }
/* Transport features are always preserved to pass to set_features. */ /* Transport features always preserved to pass to finalize_features. */
for (i = VIRTIO_TRANSPORT_F_START; i < VIRTIO_TRANSPORT_F_END; i++) for (i = VIRTIO_TRANSPORT_F_START; i < VIRTIO_TRANSPORT_F_END; i++)
if (device_features & (1 << i)) if (device_features & (1 << i))
set_bit(i, dev->features); set_bit(i, dev->features);
...@@ -122,8 +122,7 @@ static int virtio_dev_probe(struct device *_d) ...@@ -122,8 +122,7 @@ static int virtio_dev_probe(struct device *_d)
if (err) if (err)
add_status(dev, VIRTIO_CONFIG_S_FAILED); add_status(dev, VIRTIO_CONFIG_S_FAILED);
else { else {
/* They should never have set feature bits beyond 32 */ dev->config->finalize_features(dev);
dev->config->set_features(dev, dev->features[0]);
add_status(dev, VIRTIO_CONFIG_S_DRIVER_OK); add_status(dev, VIRTIO_CONFIG_S_DRIVER_OK);
} }
return err; return err;
......
...@@ -94,12 +94,14 @@ static u32 vp_get_features(struct virtio_device *vdev) ...@@ -94,12 +94,14 @@ static u32 vp_get_features(struct virtio_device *vdev)
return ioread32(vp_dev->ioaddr + VIRTIO_PCI_HOST_FEATURES); return ioread32(vp_dev->ioaddr + VIRTIO_PCI_HOST_FEATURES);
} }
/* virtio config->set_features() implementation */ /* virtio config->finalize_features() implementation */
static void vp_set_features(struct virtio_device *vdev, u32 features) static void vp_finalize_features(struct virtio_device *vdev)
{ {
struct virtio_pci_device *vp_dev = to_vp_device(vdev); struct virtio_pci_device *vp_dev = to_vp_device(vdev);
iowrite32(features, vp_dev->ioaddr + VIRTIO_PCI_GUEST_FEATURES); /* We only support 32 feature bits. */
BUILD_BUG_ON(ARRAY_SIZE(vdev->features) != 1);
iowrite32(vdev->features[0], vp_dev->ioaddr+VIRTIO_PCI_GUEST_FEATURES);
} }
/* virtio config->get() implementation */ /* virtio config->get() implementation */
...@@ -297,7 +299,7 @@ static struct virtio_config_ops virtio_pci_config_ops = { ...@@ -297,7 +299,7 @@ static struct virtio_config_ops virtio_pci_config_ops = {
.find_vq = vp_find_vq, .find_vq = vp_find_vq,
.del_vq = vp_del_vq, .del_vq = vp_del_vq,
.get_features = vp_get_features, .get_features = vp_get_features,
.set_features = vp_set_features, .finalize_features = vp_finalize_features,
}; };
/* the PCI probing function */ /* the PCI probing function */
......
...@@ -61,9 +61,10 @@ ...@@ -61,9 +61,10 @@
* @get_features: get the array of feature bits for this device. * @get_features: get the array of feature bits for this device.
* vdev: the virtio_device * vdev: the virtio_device
* Returns the first 32 feature bits (all we currently need). * Returns the first 32 feature bits (all we currently need).
* @set_features: confirm what device features we'll be using. * @finalize_features: confirm what device features we'll be using.
* vdev: the virtio_device * vdev: the virtio_device
* feature: the first 32 feature bits * This gives the final feature bits for the device: it can change
* the dev->feature bits if it wants.
*/ */
struct virtio_config_ops struct virtio_config_ops
{ {
...@@ -79,7 +80,7 @@ struct virtio_config_ops ...@@ -79,7 +80,7 @@ struct virtio_config_ops
void (*callback)(struct virtqueue *)); void (*callback)(struct virtqueue *));
void (*del_vq)(struct virtqueue *vq); void (*del_vq)(struct virtqueue *vq);
u32 (*get_features)(struct virtio_device *vdev); u32 (*get_features)(struct virtio_device *vdev);
void (*set_features)(struct virtio_device *vdev, u32 features); void (*finalize_features)(struct virtio_device *vdev);
}; };
/* If driver didn't advertise the feature, it will never appear. */ /* If driver didn't advertise the feature, it will never appear. */
......
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