ubi: Fix possible null-ptr-deref in ubi_free_volume()
[ Upstream commit c15859bfd326c10230f09cb48a17f8a35f190342 ]
It willl cause null-ptr-deref in the following case:
uif_init()
ubi_add_volume()
cdev_add() -> if it fails, call kill_volumes()
device_register()
kill_volumes() -> if ubi_add_volume() fails call this function
ubi_free_volume()
cdev_del()
device_unregister() -> trying to delete a not added device,
it causes null-ptr-deref
So in ubi_free_volume(), it delete devices whether they are added
or not, it will causes null-ptr-deref.
Handle the error case whlie calling ubi_add_volume() to fix this
problem. If add volume fails, set the corresponding vol to null,
so it can not be accessed in kill_volumes() and release the
resource in ubi_add_volume() error path.
Fixes: 801c135ce7
("UBI: Unsorted Block Images")
Suggested-by: Zhihao Cheng <chengzhihao1@huawei.com>
Signed-off-by: Yang Yingliang <yangyingliang@huawei.com>
Reviewed-by: Zhihao Cheng <chengzhihao1@huawei.com>
Signed-off-by: Richard Weinberger <richard@nod.at>
Signed-off-by: Sasha Levin <sashal@kernel.org>
This commit is contained in:

committed by
Greg Kroah-Hartman

parent
bf50229494
commit
fcbc795abe
@@ -467,6 +467,7 @@ static int uif_init(struct ubi_device *ubi)
|
|||||||
err = ubi_add_volume(ubi, ubi->volumes[i]);
|
err = ubi_add_volume(ubi, ubi->volumes[i]);
|
||||||
if (err) {
|
if (err) {
|
||||||
ubi_err(ubi, "cannot add volume %d", i);
|
ubi_err(ubi, "cannot add volume %d", i);
|
||||||
|
ubi->volumes[i] = NULL;
|
||||||
goto out_volumes;
|
goto out_volumes;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -582,6 +582,7 @@ int ubi_add_volume(struct ubi_device *ubi, struct ubi_volume *vol)
|
|||||||
if (err) {
|
if (err) {
|
||||||
ubi_err(ubi, "cannot add character device for volume %d, error %d",
|
ubi_err(ubi, "cannot add character device for volume %d, error %d",
|
||||||
vol_id, err);
|
vol_id, err);
|
||||||
|
vol_release(&vol->dev);
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -592,15 +593,14 @@ int ubi_add_volume(struct ubi_device *ubi, struct ubi_volume *vol)
|
|||||||
vol->dev.groups = volume_dev_groups;
|
vol->dev.groups = volume_dev_groups;
|
||||||
dev_set_name(&vol->dev, "%s_%d", ubi->ubi_name, vol->vol_id);
|
dev_set_name(&vol->dev, "%s_%d", ubi->ubi_name, vol->vol_id);
|
||||||
err = device_register(&vol->dev);
|
err = device_register(&vol->dev);
|
||||||
if (err)
|
if (err) {
|
||||||
goto out_cdev;
|
cdev_del(&vol->cdev);
|
||||||
|
put_device(&vol->dev);
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
self_check_volumes(ubi);
|
self_check_volumes(ubi);
|
||||||
return err;
|
return err;
|
||||||
|
|
||||||
out_cdev:
|
|
||||||
cdev_del(&vol->cdev);
|
|
||||||
return err;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
Reference in New Issue
Block a user