You need to sign in or sign up before continuing.
提交 ce94d6a1 编写于 作者: X Xie Yongji 提交者: Yang Yingliang

virtio-blk: Add validation for block size in config space

mainline inclusion
from mainline-v5.14-rc7
commit 82e89ea0
category: bufgix
bugzilla: NA
CVE: NA

--------------------------------

An untrusted device might presents an invalid block size
in configuration space. This tries to add validation for it
in the validate callback and clear the VIRTIO_BLK_F_BLK_SIZE
feature bit if the value is out of the supported range.

And we also double check the value in virtblk_probe() in
case that it's changed after the validation.
Signed-off-by: NXie Yongji <xieyongji@bytedance.com>
Link: https://lore.kernel.org/r/20210809101609.148-1-xieyongji@bytedance.comSigned-off-by: NMichael S. Tsirkin <mst@redhat.com>
Acked-by: NJason Wang <jasowang@redhat.com>

Conflict: drivers/block/virtio_blk.c
a. commit d1e9aa9c ("virtio-blk: support per-device queue depth")
is not backported, 'unsigned int queue_depth' do not exist.
b. commit e982c4d0 ("virtio-blk: modernize sysfs attribute creation")
is not backported, 'virtblk_attr_groups' do not exist.
c. commit f525464a ("block: add blk_alloc_disk and blk_cleanup_disk APIs")
and commit 89a5f065 ("virtio-blk: use blk_mq_alloc_disk") is not
backported, 'blk_cleanup_disk' is needed to repleace with 'blk_cleanup_queue',
'out_free_vq' is needed to replace with 'out_put_disk'.
Signed-off-by: NLi Jinlin <lijinlin3@huawei.com>
Reviewed-by: Nlouhongxiang <louhongxiang@huawei.com>
Signed-off-by: NYang Yingliang <yangyingliang@huawei.com>
上级 ed9f7876
...@@ -718,6 +718,28 @@ static const struct blk_mq_ops virtio_mq_ops = { ...@@ -718,6 +718,28 @@ static const struct blk_mq_ops virtio_mq_ops = {
static unsigned int virtblk_queue_depth; static unsigned int virtblk_queue_depth;
module_param_named(queue_depth, virtblk_queue_depth, uint, 0444); module_param_named(queue_depth, virtblk_queue_depth, uint, 0444);
static int virtblk_validate(struct virtio_device *vdev)
{
u32 blk_size;
if (!vdev->config->get) {
dev_err(&vdev->dev, "%s failure: config access disabled\n",
__func__);
return -EINVAL;
}
if (!virtio_has_feature(vdev, VIRTIO_BLK_F_BLK_SIZE))
return 0;
blk_size = virtio_cread32(vdev,
offsetof(struct virtio_blk_config, blk_size));
if (blk_size < SECTOR_SIZE || blk_size > PAGE_SIZE)
__virtio_clear_bit(vdev, VIRTIO_BLK_F_BLK_SIZE);
return 0;
}
static int virtblk_probe(struct virtio_device *vdev) static int virtblk_probe(struct virtio_device *vdev)
{ {
struct virtio_blk *vblk; struct virtio_blk *vblk;
...@@ -728,12 +750,6 @@ static int virtblk_probe(struct virtio_device *vdev) ...@@ -728,12 +750,6 @@ static int virtblk_probe(struct virtio_device *vdev)
u16 min_io_size; u16 min_io_size;
u8 physical_block_exp, alignment_offset; u8 physical_block_exp, alignment_offset;
if (!vdev->config->get) {
dev_err(&vdev->dev, "%s failure: config access disabled\n",
__func__);
return -EINVAL;
}
err = ida_simple_get(&vd_index_ida, 0, minor_to_index(1 << MINORBITS), err = ida_simple_get(&vd_index_ida, 0, minor_to_index(1 << MINORBITS),
GFP_KERNEL); GFP_KERNEL);
if (err < 0) if (err < 0)
...@@ -849,6 +865,14 @@ static int virtblk_probe(struct virtio_device *vdev) ...@@ -849,6 +865,14 @@ static int virtblk_probe(struct virtio_device *vdev)
else else
blk_size = queue_logical_block_size(q); blk_size = queue_logical_block_size(q);
if (unlikely(blk_size < SECTOR_SIZE || blk_size > PAGE_SIZE)) {
dev_err(&vdev->dev,
"block size is changed unexpectedly, now is %u\n",
blk_size);
err = -EINVAL;
goto err_cleanup_disk;
}
/* Use topology information if available */ /* Use topology information if available */
err = virtio_cread_feature(vdev, VIRTIO_BLK_F_TOPOLOGY, err = virtio_cread_feature(vdev, VIRTIO_BLK_F_TOPOLOGY,
struct virtio_blk_config, physical_block_exp, struct virtio_blk_config, physical_block_exp,
...@@ -882,6 +906,8 @@ static int virtblk_probe(struct virtio_device *vdev) ...@@ -882,6 +906,8 @@ static int virtblk_probe(struct virtio_device *vdev)
device_add_disk(&vdev->dev, vblk->disk); device_add_disk(&vdev->dev, vblk->disk);
return 0; return 0;
err_cleanup_disk:
blk_cleanup_queue(vblk->disk->queue);
out_free_tags: out_free_tags:
blk_mq_free_tag_set(&vblk->tag_set); blk_mq_free_tag_set(&vblk->tag_set);
out_put_disk: out_put_disk:
...@@ -991,6 +1017,7 @@ static struct virtio_driver virtio_blk = { ...@@ -991,6 +1017,7 @@ static struct virtio_driver virtio_blk = {
.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 = virtblk_validate,
.probe = virtblk_probe, .probe = virtblk_probe,
.remove = virtblk_remove, .remove = virtblk_remove,
.config_changed = virtblk_config_changed, .config_changed = virtblk_config_changed,
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册