aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMichael S. Tsirkin <mst@redhat.com>2014-10-23 11:07:47 -0400
committerMichael S. Tsirkin <mst@redhat.com>2014-12-09 05:05:26 -0500
commitb3bb62d11950eb6ac87403cacd667f84fa9495bc (patch)
treef3aee72f5e622722d19c484c7f5925990e97cffb
parentc102659d690d382171bd2e40f35c5c811f0cdcac (diff)
virtio: add legacy feature table support
virtio-blk has some legacy feature bits that modern drivers must not negotiate, but are needed for old legacy hosts (that e.g. don't support virtio-scsi). Allow a separate legacy feature table for such cases. Signed-off-by: Michael S. Tsirkin <mst@redhat.com> Reviewed-by: Cornelia Huck <cornelia.huck@de.ibm.com>
-rw-r--r--drivers/virtio/virtio.c25
-rw-r--r--include/linux/virtio.h4
2 files changed, 28 insertions, 1 deletions
diff --git a/drivers/virtio/virtio.c b/drivers/virtio/virtio.c
index 3e78f4bcfa8e..f9ad99c8b352 100644
--- a/drivers/virtio/virtio.c
+++ b/drivers/virtio/virtio.c
@@ -113,6 +113,13 @@ void virtio_check_driver_offered_feature(const struct virtio_device *vdev,
113 for (i = 0; i < drv->feature_table_size; i++) 113 for (i = 0; i < drv->feature_table_size; i++)
114 if (drv->feature_table[i] == fbit) 114 if (drv->feature_table[i] == fbit)
115 return; 115 return;
116
117 if (drv->feature_table_legacy) {
118 for (i = 0; i < drv->feature_table_size_legacy; i++)
119 if (drv->feature_table_legacy[i] == fbit)
120 return;
121 }
122
116 BUG(); 123 BUG();
117} 124}
118EXPORT_SYMBOL_GPL(virtio_check_driver_offered_feature); 125EXPORT_SYMBOL_GPL(virtio_check_driver_offered_feature);
@@ -161,6 +168,7 @@ static int virtio_dev_probe(struct device *_d)
161 struct virtio_driver *drv = drv_to_virtio(dev->dev.driver); 168 struct virtio_driver *drv = drv_to_virtio(dev->dev.driver);
162 u64 device_features; 169 u64 device_features;
163 u64 driver_features; 170 u64 driver_features;
171 u64 driver_features_legacy;
164 unsigned status; 172 unsigned status;
165 173
166 /* We have a driver! */ 174 /* We have a driver! */
@@ -177,7 +185,22 @@ static int virtio_dev_probe(struct device *_d)
177 driver_features |= (1ULL << f); 185 driver_features |= (1ULL << f);
178 } 186 }
179 187
180 dev->features = driver_features & device_features; 188 /* Some drivers have a separate feature table for virtio v1.0 */
189 if (drv->feature_table_legacy) {
190 driver_features_legacy = 0;
191 for (i = 0; i < drv->feature_table_size_legacy; i++) {
192 unsigned int f = drv->feature_table_legacy[i];
193 BUG_ON(f >= 64);
194 driver_features_legacy |= (1ULL << f);
195 }
196 } else {
197 driver_features_legacy = driver_features;
198 }
199
200 if (driver_features & device_features & (1ULL << VIRTIO_F_VERSION_1))
201 dev->features = driver_features & device_features;
202 else
203 dev->features = driver_features_legacy & device_features;
181 204
182 /* Transport features always preserved to pass to finalize_features. */ 205 /* Transport features always preserved to pass to finalize_features. */
183 for (i = VIRTIO_TRANSPORT_F_START; i < VIRTIO_TRANSPORT_F_END; i++) 206 for (i = VIRTIO_TRANSPORT_F_START; i < VIRTIO_TRANSPORT_F_END; i++)
diff --git a/include/linux/virtio.h b/include/linux/virtio.h
index d6359a5d5d4e..f70411eb9d27 100644
--- a/include/linux/virtio.h
+++ b/include/linux/virtio.h
@@ -130,6 +130,8 @@ int virtio_device_restore(struct virtio_device *dev);
130 * @id_table: the ids serviced by this driver. 130 * @id_table: the ids serviced by this driver.
131 * @feature_table: an array of feature numbers supported by this driver. 131 * @feature_table: an array of feature numbers supported by this driver.
132 * @feature_table_size: number of entries in the feature table array. 132 * @feature_table_size: number of entries in the feature table array.
133 * @feature_table_legacy: same as feature_table but when working in legacy mode.
134 * @feature_table_size_legacy: number of entries in feature table legacy array.
133 * @probe: the function to call when a device is found. Returns 0 or -errno. 135 * @probe: the function to call when a device is found. Returns 0 or -errno.
134 * @remove: the function to call when a device is removed. 136 * @remove: the function to call when a device is removed.
135 * @config_changed: optional function to call when the device configuration 137 * @config_changed: optional function to call when the device configuration
@@ -140,6 +142,8 @@ struct virtio_driver {
140 const struct virtio_device_id *id_table; 142 const struct virtio_device_id *id_table;
141 const unsigned int *feature_table; 143 const unsigned int *feature_table;
142 unsigned int feature_table_size; 144 unsigned int feature_table_size;
145 const unsigned int *feature_table_legacy;
146 unsigned int feature_table_size_legacy;
143 int (*probe)(struct virtio_device *dev); 147 int (*probe)(struct virtio_device *dev);
144 void (*scan)(struct virtio_device *dev); 148 void (*scan)(struct virtio_device *dev);
145 void (*remove)(struct virtio_device *dev); 149 void (*remove)(struct virtio_device *dev);