xref: /qemu/block/snapshot-access.c (revision a4b740db)
11c14eaabSVladimir Sementsov-Ogievskiy /*
21c14eaabSVladimir Sementsov-Ogievskiy  * snapshot_access block driver
31c14eaabSVladimir Sementsov-Ogievskiy  *
41c14eaabSVladimir Sementsov-Ogievskiy  * Copyright (c) 2022 Virtuozzo International GmbH.
51c14eaabSVladimir Sementsov-Ogievskiy  *
61c14eaabSVladimir Sementsov-Ogievskiy  * Author:
71c14eaabSVladimir Sementsov-Ogievskiy  *  Sementsov-Ogievskiy Vladimir <vsementsov@virtuozzo.com>
81c14eaabSVladimir Sementsov-Ogievskiy  *
91c14eaabSVladimir Sementsov-Ogievskiy  * This program is free software; you can redistribute it and/or modify
101c14eaabSVladimir Sementsov-Ogievskiy  * it under the terms of the GNU General Public License as published by
111c14eaabSVladimir Sementsov-Ogievskiy  * the Free Software Foundation; either version 2 of the License, or
121c14eaabSVladimir Sementsov-Ogievskiy  * (at your option) any later version.
131c14eaabSVladimir Sementsov-Ogievskiy  *
141c14eaabSVladimir Sementsov-Ogievskiy  * This program is distributed in the hope that it will be useful,
151c14eaabSVladimir Sementsov-Ogievskiy  * but WITHOUT ANY WARRANTY; without even the implied warranty of
161c14eaabSVladimir Sementsov-Ogievskiy  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
171c14eaabSVladimir Sementsov-Ogievskiy  * GNU General Public License for more details.
181c14eaabSVladimir Sementsov-Ogievskiy  *
191c14eaabSVladimir Sementsov-Ogievskiy  * You should have received a copy of the GNU General Public License
201c14eaabSVladimir Sementsov-Ogievskiy  * along with this program. If not, see <http://www.gnu.org/licenses/>.
211c14eaabSVladimir Sementsov-Ogievskiy  */
221c14eaabSVladimir Sementsov-Ogievskiy 
231c14eaabSVladimir Sementsov-Ogievskiy #include "qemu/osdep.h"
241c14eaabSVladimir Sementsov-Ogievskiy 
251c14eaabSVladimir Sementsov-Ogievskiy #include "sysemu/block-backend.h"
261c14eaabSVladimir Sementsov-Ogievskiy #include "qemu/cutils.h"
271c14eaabSVladimir Sementsov-Ogievskiy #include "block/block_int.h"
281c14eaabSVladimir Sementsov-Ogievskiy 
297b9e8b22SKevin Wolf static int coroutine_fn GRAPH_RDLOCK
snapshot_access_co_preadv_part(BlockDriverState * bs,int64_t offset,int64_t bytes,QEMUIOVector * qiov,size_t qiov_offset,BdrvRequestFlags flags)301c14eaabSVladimir Sementsov-Ogievskiy snapshot_access_co_preadv_part(BlockDriverState *bs,
311c14eaabSVladimir Sementsov-Ogievskiy                                int64_t offset, int64_t bytes,
321c14eaabSVladimir Sementsov-Ogievskiy                                QEMUIOVector *qiov, size_t qiov_offset,
331c14eaabSVladimir Sementsov-Ogievskiy                                BdrvRequestFlags flags)
341c14eaabSVladimir Sementsov-Ogievskiy {
351c14eaabSVladimir Sementsov-Ogievskiy     if (flags) {
361c14eaabSVladimir Sementsov-Ogievskiy         return -ENOTSUP;
371c14eaabSVladimir Sementsov-Ogievskiy     }
381c14eaabSVladimir Sementsov-Ogievskiy 
391c14eaabSVladimir Sementsov-Ogievskiy     return bdrv_co_preadv_snapshot(bs->file, offset, bytes, qiov, qiov_offset);
401c14eaabSVladimir Sementsov-Ogievskiy }
411c14eaabSVladimir Sementsov-Ogievskiy 
427b9e8b22SKevin Wolf static int coroutine_fn GRAPH_RDLOCK
snapshot_access_co_block_status(BlockDriverState * bs,bool want_zero,int64_t offset,int64_t bytes,int64_t * pnum,int64_t * map,BlockDriverState ** file)431c14eaabSVladimir Sementsov-Ogievskiy snapshot_access_co_block_status(BlockDriverState *bs,
441c14eaabSVladimir Sementsov-Ogievskiy                                 bool want_zero, int64_t offset,
451c14eaabSVladimir Sementsov-Ogievskiy                                 int64_t bytes, int64_t *pnum,
461c14eaabSVladimir Sementsov-Ogievskiy                                 int64_t *map, BlockDriverState **file)
471c14eaabSVladimir Sementsov-Ogievskiy {
481c14eaabSVladimir Sementsov-Ogievskiy     return bdrv_co_snapshot_block_status(bs->file->bs, want_zero, offset,
491c14eaabSVladimir Sementsov-Ogievskiy                                          bytes, pnum, map, file);
501c14eaabSVladimir Sementsov-Ogievskiy }
511c14eaabSVladimir Sementsov-Ogievskiy 
529a5a1c62SEmanuele Giuseppe Esposito static int coroutine_fn GRAPH_RDLOCK
snapshot_access_co_pdiscard(BlockDriverState * bs,int64_t offset,int64_t bytes)539a5a1c62SEmanuele Giuseppe Esposito snapshot_access_co_pdiscard(BlockDriverState *bs, int64_t offset, int64_t bytes)
541c14eaabSVladimir Sementsov-Ogievskiy {
551c14eaabSVladimir Sementsov-Ogievskiy     return bdrv_co_pdiscard_snapshot(bs->file->bs, offset, bytes);
561c14eaabSVladimir Sementsov-Ogievskiy }
571c14eaabSVladimir Sementsov-Ogievskiy 
581c14eaabSVladimir Sementsov-Ogievskiy static int coroutine_fn
snapshot_access_co_pwrite_zeroes(BlockDriverState * bs,int64_t offset,int64_t bytes,BdrvRequestFlags flags)591c14eaabSVladimir Sementsov-Ogievskiy snapshot_access_co_pwrite_zeroes(BlockDriverState *bs,
601c14eaabSVladimir Sementsov-Ogievskiy                                  int64_t offset, int64_t bytes,
611c14eaabSVladimir Sementsov-Ogievskiy                                  BdrvRequestFlags flags)
621c14eaabSVladimir Sementsov-Ogievskiy {
631c14eaabSVladimir Sementsov-Ogievskiy     return -ENOTSUP;
641c14eaabSVladimir Sementsov-Ogievskiy }
651c14eaabSVladimir Sementsov-Ogievskiy 
661c14eaabSVladimir Sementsov-Ogievskiy static coroutine_fn int
snapshot_access_co_pwritev_part(BlockDriverState * bs,int64_t offset,int64_t bytes,QEMUIOVector * qiov,size_t qiov_offset,BdrvRequestFlags flags)671c14eaabSVladimir Sementsov-Ogievskiy snapshot_access_co_pwritev_part(BlockDriverState *bs,
681c14eaabSVladimir Sementsov-Ogievskiy                                 int64_t offset, int64_t bytes,
691c14eaabSVladimir Sementsov-Ogievskiy                                 QEMUIOVector *qiov, size_t qiov_offset,
701c14eaabSVladimir Sementsov-Ogievskiy                                 BdrvRequestFlags flags)
711c14eaabSVladimir Sementsov-Ogievskiy {
721c14eaabSVladimir Sementsov-Ogievskiy     return -ENOTSUP;
731c14eaabSVladimir Sementsov-Ogievskiy }
741c14eaabSVladimir Sementsov-Ogievskiy 
751c14eaabSVladimir Sementsov-Ogievskiy 
snapshot_access_refresh_filename(BlockDriverState * bs)7679a55866SKevin Wolf static void GRAPH_RDLOCK snapshot_access_refresh_filename(BlockDriverState *bs)
771c14eaabSVladimir Sementsov-Ogievskiy {
781c14eaabSVladimir Sementsov-Ogievskiy     pstrcpy(bs->exact_filename, sizeof(bs->exact_filename),
791c14eaabSVladimir Sementsov-Ogievskiy             bs->file->bs->filename);
801c14eaabSVladimir Sementsov-Ogievskiy }
811c14eaabSVladimir Sementsov-Ogievskiy 
snapshot_access_open(BlockDriverState * bs,QDict * options,int flags,Error ** errp)821c14eaabSVladimir Sementsov-Ogievskiy static int snapshot_access_open(BlockDriverState *bs, QDict *options, int flags,
831c14eaabSVladimir Sementsov-Ogievskiy                                 Error **errp)
841c14eaabSVladimir Sementsov-Ogievskiy {
855bb04747SVladimir Sementsov-Ogievskiy     bdrv_open_child(NULL, options, "file", bs, &child_of_bds,
861c14eaabSVladimir Sementsov-Ogievskiy                     BDRV_CHILD_DATA | BDRV_CHILD_PRIMARY,
871c14eaabSVladimir Sementsov-Ogievskiy                     false, errp);
88a4b740dbSKevin Wolf 
89a4b740dbSKevin Wolf     GRAPH_RDLOCK_GUARD_MAINLOOP();
90a4b740dbSKevin Wolf 
911c14eaabSVladimir Sementsov-Ogievskiy     if (!bs->file) {
921c14eaabSVladimir Sementsov-Ogievskiy         return -EINVAL;
931c14eaabSVladimir Sementsov-Ogievskiy     }
941c14eaabSVladimir Sementsov-Ogievskiy 
951c14eaabSVladimir Sementsov-Ogievskiy     bs->total_sectors = bs->file->bs->total_sectors;
961c14eaabSVladimir Sementsov-Ogievskiy 
971c14eaabSVladimir Sementsov-Ogievskiy     return 0;
981c14eaabSVladimir Sementsov-Ogievskiy }
991c14eaabSVladimir Sementsov-Ogievskiy 
snapshot_access_child_perm(BlockDriverState * bs,BdrvChild * c,BdrvChildRole role,BlockReopenQueue * reopen_queue,uint64_t perm,uint64_t shared,uint64_t * nperm,uint64_t * nshared)1001c14eaabSVladimir Sementsov-Ogievskiy static void snapshot_access_child_perm(BlockDriverState *bs, BdrvChild *c,
1011c14eaabSVladimir Sementsov-Ogievskiy                                 BdrvChildRole role,
1021c14eaabSVladimir Sementsov-Ogievskiy                                 BlockReopenQueue *reopen_queue,
1031c14eaabSVladimir Sementsov-Ogievskiy                                 uint64_t perm, uint64_t shared,
1041c14eaabSVladimir Sementsov-Ogievskiy                                 uint64_t *nperm, uint64_t *nshared)
1051c14eaabSVladimir Sementsov-Ogievskiy {
1061c14eaabSVladimir Sementsov-Ogievskiy     /*
1071c14eaabSVladimir Sementsov-Ogievskiy      * Currently, we don't need any permissions. If bs->file provides
1081c14eaabSVladimir Sementsov-Ogievskiy      * snapshot-access API, we can use it.
1091c14eaabSVladimir Sementsov-Ogievskiy      */
1101c14eaabSVladimir Sementsov-Ogievskiy     *nperm = 0;
1111c14eaabSVladimir Sementsov-Ogievskiy     *nshared = BLK_PERM_ALL;
1121c14eaabSVladimir Sementsov-Ogievskiy }
1131c14eaabSVladimir Sementsov-Ogievskiy 
1149ea473fbSKevin Wolf static BlockDriver bdrv_snapshot_access_drv = {
1151c14eaabSVladimir Sementsov-Ogievskiy     .format_name = "snapshot-access",
1161c14eaabSVladimir Sementsov-Ogievskiy 
1171c14eaabSVladimir Sementsov-Ogievskiy     .bdrv_open                  = snapshot_access_open,
1181c14eaabSVladimir Sementsov-Ogievskiy 
1191c14eaabSVladimir Sementsov-Ogievskiy     .bdrv_co_preadv_part        = snapshot_access_co_preadv_part,
1201c14eaabSVladimir Sementsov-Ogievskiy     .bdrv_co_pwritev_part       = snapshot_access_co_pwritev_part,
1211c14eaabSVladimir Sementsov-Ogievskiy     .bdrv_co_pwrite_zeroes      = snapshot_access_co_pwrite_zeroes,
1221c14eaabSVladimir Sementsov-Ogievskiy     .bdrv_co_pdiscard           = snapshot_access_co_pdiscard,
1231c14eaabSVladimir Sementsov-Ogievskiy     .bdrv_co_block_status       = snapshot_access_co_block_status,
1241c14eaabSVladimir Sementsov-Ogievskiy 
1251c14eaabSVladimir Sementsov-Ogievskiy     .bdrv_refresh_filename      = snapshot_access_refresh_filename,
1261c14eaabSVladimir Sementsov-Ogievskiy 
1271c14eaabSVladimir Sementsov-Ogievskiy     .bdrv_child_perm            = snapshot_access_child_perm,
1281c14eaabSVladimir Sementsov-Ogievskiy };
1291c14eaabSVladimir Sementsov-Ogievskiy 
snapshot_access_init(void)1301c14eaabSVladimir Sementsov-Ogievskiy static void snapshot_access_init(void)
1311c14eaabSVladimir Sementsov-Ogievskiy {
1321c14eaabSVladimir Sementsov-Ogievskiy     bdrv_register(&bdrv_snapshot_access_drv);
1331c14eaabSVladimir Sementsov-Ogievskiy }
1341c14eaabSVladimir Sementsov-Ogievskiy 
1351c14eaabSVladimir Sementsov-Ogievskiy block_init(snapshot_access_init);
136