block: Add permissions to blk_new()

We want every user to be specific about the permissions it needs, so
we'll pass the initial permissions as parameters to blk_new(). A user
only needs to call blk_set_perm() if it wants to change the permissions
after the fact.

The permissions are stored in the BlockBackend and applied whenever a
BlockDriverState should be attached in blk_insert_bs().

This does not include actually choosing the right set of permissions
everywhere yet. Instead, the usual FIXME comment is added to each place
and will be addressed in individual patches.

Signed-off-by: Kevin Wolf <kwolf@redhat.com>
Reviewed-by: Max Reitz <mreitz@redhat.com>
Acked-by: Fam Zheng <famz@redhat.com>
This commit is contained in:
Kevin Wolf 2017-01-20 17:07:26 +01:00
parent 981776b348
commit 6d0eb64d5c
18 changed files with 53 additions and 30 deletions

View File

@ -2193,7 +2193,7 @@ static BlockDriverState *bdrv_open_inherit(const char *filename,
goto fail; goto fail;
} }
if (file_bs != NULL) { if (file_bs != NULL) {
file = blk_new(); file = blk_new(BLK_PERM_CONSISTENT_READ, BLK_PERM_ALL);
blk_insert_bs(file, file_bs); blk_insert_bs(file, file_bs);
bdrv_unref(file_bs); bdrv_unref(file_bs);

View File

@ -624,7 +624,8 @@ BlockJob *backup_job_create(const char *job_id, BlockDriverState *bs,
goto error; goto error;
} }
job->target = blk_new(); /* FIXME Use real permissions */
job->target = blk_new(0, BLK_PERM_ALL);
blk_insert_bs(job->target, target); blk_insert_bs(job->target, target);
job->on_source_error = on_source_error; job->on_source_error = on_source_error;

View File

@ -120,17 +120,23 @@ static const BdrvChildRole child_root = {
/* /*
* Create a new BlockBackend with a reference count of one. * Create a new BlockBackend with a reference count of one.
* Store an error through @errp on failure, unless it's null. *
* @perm is a bitmasks of BLK_PERM_* constants which describes the permissions
* to request for a block driver node that is attached to this BlockBackend.
* @shared_perm is a bitmask which describes which permissions may be granted
* to other users of the attached node.
* Both sets of permissions can be changed later using blk_set_perm().
*
* Return the new BlockBackend on success, null on failure. * Return the new BlockBackend on success, null on failure.
*/ */
BlockBackend *blk_new(void) BlockBackend *blk_new(uint64_t perm, uint64_t shared_perm)
{ {
BlockBackend *blk; BlockBackend *blk;
blk = g_new0(BlockBackend, 1); blk = g_new0(BlockBackend, 1);
blk->refcnt = 1; blk->refcnt = 1;
blk->perm = 0; blk->perm = perm;
blk->shared_perm = BLK_PERM_ALL; blk->shared_perm = shared_perm;
blk_set_enable_write_cache(blk, true); blk_set_enable_write_cache(blk, true);
qemu_co_queue_init(&blk->public.throttled_reqs[0]); qemu_co_queue_init(&blk->public.throttled_reqs[0]);
@ -161,7 +167,7 @@ BlockBackend *blk_new_open(const char *filename, const char *reference,
BlockBackend *blk; BlockBackend *blk;
BlockDriverState *bs; BlockDriverState *bs;
blk = blk_new(); blk = blk_new(0, BLK_PERM_ALL);
bs = bdrv_open(filename, reference, options, flags, errp); bs = bdrv_open(filename, reference, options, flags, errp);
if (!bs) { if (!bs) {
blk_unref(blk); blk_unref(blk);
@ -505,9 +511,10 @@ void blk_remove_bs(BlockBackend *blk)
void blk_insert_bs(BlockBackend *blk, BlockDriverState *bs) void blk_insert_bs(BlockBackend *blk, BlockDriverState *bs)
{ {
bdrv_ref(bs); bdrv_ref(bs);
/* FIXME Use real permissions */ /* FIXME Error handling */
blk->root = bdrv_root_attach_child(bs, "root", &child_root, blk->root = bdrv_root_attach_child(bs, "root", &child_root,
0, BLK_PERM_ALL, blk, &error_abort); blk->perm, blk->shared_perm, blk,
&error_abort);
notifier_list_notify(&blk->insert_bs_notifiers, blk); notifier_list_notify(&blk->insert_bs_notifiers, blk);
if (blk->public.throttle_state) { if (blk->public.throttle_state) {

View File

@ -275,10 +275,12 @@ void commit_start(const char *job_id, BlockDriverState *bs,
block_job_add_bdrv(&s->common, overlay_bs); block_job_add_bdrv(&s->common, overlay_bs);
} }
s->base = blk_new(); /* FIXME Use real permissions */
s->base = blk_new(0, BLK_PERM_ALL);
blk_insert_bs(s->base, base); blk_insert_bs(s->base, base);
s->top = blk_new(); /* FIXME Use real permissions */
s->top = blk_new(0, BLK_PERM_ALL);
blk_insert_bs(s->top, top); blk_insert_bs(s->top, top);
s->active = bs; s->active = bs;
@ -328,10 +330,12 @@ int bdrv_commit(BlockDriverState *bs)
} }
} }
src = blk_new(); /* FIXME Use real permissions */
src = blk_new(0, BLK_PERM_ALL);
blk_insert_bs(src, bs); blk_insert_bs(src, bs);
backing = blk_new(); /* FIXME Use real permissions */
backing = blk_new(0, BLK_PERM_ALL);
blk_insert_bs(backing, bs->backing->bs); blk_insert_bs(backing, bs->backing->bs);
length = blk_getlength(src); length = blk_getlength(src);

View File

@ -1017,7 +1017,8 @@ static void mirror_start_job(const char *job_id, BlockDriverState *bs,
return; return;
} }
s->target = blk_new(); /* FIXME Use real permissions */
s->target = blk_new(0, BLK_PERM_ALL);
blk_insert_bs(s->target, target); blk_insert_bs(s->target, target);
s->replaces = g_strdup(replaces); s->replaces = g_strdup(replaces);

View File

@ -3262,7 +3262,7 @@ static int qcow2_amend_options(BlockDriverState *bs, QemuOpts *opts,
} }
if (new_size) { if (new_size) {
BlockBackend *blk = blk_new(); BlockBackend *blk = blk_new(BLK_PERM_RESIZE, BLK_PERM_ALL);
blk_insert_bs(blk, bs); blk_insert_bs(blk, bs);
ret = blk_truncate(blk, new_size); ret = blk_truncate(blk, new_size);
blk_unref(blk); blk_unref(blk);

View File

@ -558,7 +558,7 @@ static BlockBackend *blockdev_init(const char *file, QDict *bs_opts,
if ((!file || !*file) && !qdict_size(bs_opts)) { if ((!file || !*file) && !qdict_size(bs_opts)) {
BlockBackendRootState *blk_rs; BlockBackendRootState *blk_rs;
blk = blk_new(); blk = blk_new(0, BLK_PERM_ALL);
blk_rs = blk_get_root_state(blk); blk_rs = blk_get_root_state(blk);
blk_rs->open_flags = bdrv_flags; blk_rs->open_flags = bdrv_flags;
blk_rs->read_only = read_only; blk_rs->read_only = read_only;
@ -2890,7 +2890,7 @@ void qmp_block_resize(bool has_device, const char *device,
goto out; goto out;
} }
blk = blk_new(); blk = blk_new(BLK_PERM_RESIZE, BLK_PERM_ALL);
blk_insert_bs(blk, bs); blk_insert_bs(blk, bs);
/* complete all in-flight operations before resizing the device */ /* complete all in-flight operations before resizing the device */

View File

@ -159,7 +159,8 @@ void *block_job_create(const char *job_id, const BlockJobDriver *driver,
} }
} }
blk = blk_new(); /* FIXME Use real permissions */
blk = blk_new(0, BLK_PERM_ALL);
blk_insert_bs(blk, bs); blk_insert_bs(blk, bs);
job = g_malloc0(driver->instance_size); job = g_malloc0(driver->instance_size);

3
hmp.c
View File

@ -2050,7 +2050,8 @@ void hmp_qemu_io(Monitor *mon, const QDict *qdict)
if (!blk) { if (!blk) {
BlockDriverState *bs = bdrv_lookup_bs(NULL, device, &err); BlockDriverState *bs = bdrv_lookup_bs(NULL, device, &err);
if (bs) { if (bs) {
blk = local_blk = blk_new(); /* FIXME Use real permissions */
blk = local_blk = blk_new(0, BLK_PERM_ALL);
blk_insert_bs(blk, bs); blk_insert_bs(blk, bs);
} else { } else {
goto fail; goto fail;

View File

@ -533,7 +533,8 @@ static int floppy_drive_init(DeviceState *qdev)
if (!dev->conf.blk) { if (!dev->conf.blk) {
/* Anonymous BlockBackend for an empty drive */ /* Anonymous BlockBackend for an empty drive */
dev->conf.blk = blk_new(); /* FIXME Use real permissions */
dev->conf.blk = blk_new(0, BLK_PERM_ALL);
ret = blk_attach_dev(dev->conf.blk, qdev); ret = blk_attach_dev(dev->conf.blk, qdev);
assert(ret == 0); assert(ret == 0);
} }

View File

@ -78,7 +78,8 @@ static void parse_drive(DeviceState *dev, const char *str, void **ptr,
if (!blk) { if (!blk) {
BlockDriverState *bs = bdrv_lookup_bs(NULL, str, NULL); BlockDriverState *bs = bdrv_lookup_bs(NULL, str, NULL);
if (bs) { if (bs) {
blk = blk_new(); /* FIXME Use real permissions */
blk = blk_new(0, BLK_PERM_ALL);
blk_insert_bs(blk, bs); blk_insert_bs(blk, bs);
blk_created = true; blk_created = true;
} }

View File

@ -170,7 +170,8 @@ static int ide_dev_initfn(IDEDevice *dev, IDEDriveKind kind)
return -1; return -1;
} else { } else {
/* Anonymous BlockBackend for an empty drive */ /* Anonymous BlockBackend for an empty drive */
dev->conf.blk = blk_new(); /* FIXME Use real permissions */
dev->conf.blk = blk_new(0, BLK_PERM_ALL);
} }
} }

View File

@ -2380,7 +2380,8 @@ static void scsi_cd_realize(SCSIDevice *dev, Error **errp)
SCSIDiskState *s = DO_UPCAST(SCSIDiskState, qdev, dev); SCSIDiskState *s = DO_UPCAST(SCSIDiskState, qdev, dev);
if (!dev->conf.blk) { if (!dev->conf.blk) {
dev->conf.blk = blk_new(); /* FIXME Use real permissions */
dev->conf.blk = blk_new(0, BLK_PERM_ALL);
} }
s->qdev.blocksize = 2048; s->qdev.blocksize = 2048;

View File

@ -84,7 +84,7 @@ typedef struct BlockBackendPublic {
QLIST_ENTRY(BlockBackendPublic) round_robin; QLIST_ENTRY(BlockBackendPublic) round_robin;
} BlockBackendPublic; } BlockBackendPublic;
BlockBackend *blk_new(void); BlockBackend *blk_new(uint64_t perm, uint64_t shared_perm);
BlockBackend *blk_new_open(const char *filename, const char *reference, BlockBackend *blk_new_open(const char *filename, const char *reference,
QDict *options, int flags, Error **errp); QDict *options, int flags, Error **errp);
int blk_get_refcnt(BlockBackend *blk); int blk_get_refcnt(BlockBackend *blk);

View File

@ -415,7 +415,8 @@ static void init_blk_migration(QEMUFile *f)
} }
bmds = g_new0(BlkMigDevState, 1); bmds = g_new0(BlkMigDevState, 1);
bmds->blk = blk_new(); /* FIXME Use real permissions */
bmds->blk = blk_new(0, BLK_PERM_ALL);
bmds->blk_name = g_strdup(bdrv_get_device_name(bs)); bmds->blk_name = g_strdup(bdrv_get_device_name(bs));
bmds->bulk_completed = 0; bmds->bulk_completed = 0;
bmds->total_sectors = sectors; bmds->total_sectors = sectors;

View File

@ -892,7 +892,8 @@ NBDExport *nbd_export_new(BlockDriverState *bs, off_t dev_offset, off_t size,
BlockBackend *blk; BlockBackend *blk;
NBDExport *exp = g_malloc0(sizeof(NBDExport)); NBDExport *exp = g_malloc0(sizeof(NBDExport));
blk = blk_new(); /* FIXME Use real permissions */
blk = blk_new(0, BLK_PERM_ALL);
blk_insert_bs(blk, bs); blk_insert_bs(blk, bs);
blk_set_enable_write_cache(blk, !writethrough); blk_set_enable_write_cache(blk, !writethrough);

View File

@ -53,7 +53,8 @@ static BlockJob *do_test_id(BlockBackend *blk, const char *id,
* BlockDriverState inserted. */ * BlockDriverState inserted. */
static BlockBackend *create_blk(const char *name) static BlockBackend *create_blk(const char *name)
{ {
BlockBackend *blk = blk_new(); /* FIXME Use real permissions */
BlockBackend *blk = blk_new(0, BLK_PERM_ALL);
BlockDriverState *bs; BlockDriverState *bs;
bs = bdrv_open("null-co://", NULL, NULL, 0, &error_abort); bs = bdrv_open("null-co://", NULL, NULL, 0, &error_abort);

View File

@ -593,9 +593,10 @@ static void test_groups(void)
BlockBackend *blk1, *blk2, *blk3; BlockBackend *blk1, *blk2, *blk3;
BlockBackendPublic *blkp1, *blkp2, *blkp3; BlockBackendPublic *blkp1, *blkp2, *blkp3;
blk1 = blk_new(); /* FIXME Use real permissions */
blk2 = blk_new(); blk1 = blk_new(0, BLK_PERM_ALL);
blk3 = blk_new(); blk2 = blk_new(0, BLK_PERM_ALL);
blk3 = blk_new(0, BLK_PERM_ALL);
blkp1 = blk_get_public(blk1); blkp1 = blk_get_public(blk1);
blkp2 = blk_get_public(blk2); blkp2 = blk_get_public(blk2);