Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 1 | /* |
| 2 | * QEMU backup |
| 3 | * |
| 4 | * Copyright (C) 2013 Proxmox Server Solutions |
Vladimir Sementsov-Ogievskiy | 00e30f0 | 2019-10-01 16:14:09 +0300 | [diff] [blame] | 5 | * Copyright (c) 2019 Virtuozzo International GmbH. |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 6 | * |
| 7 | * Authors: |
| 8 | * Dietmar Maurer (dietmar@proxmox.com) |
| 9 | * |
| 10 | * This work is licensed under the terms of the GNU GPL, version 2 or later. |
| 11 | * See the COPYING file in the top-level directory. |
| 12 | * |
| 13 | */ |
| 14 | |
Peter Maydell | 80c71a2 | 2016-01-18 18:01:42 +0000 | [diff] [blame] | 15 | #include "qemu/osdep.h" |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 16 | |
| 17 | #include "trace.h" |
| 18 | #include "block/block.h" |
| 19 | #include "block/block_int.h" |
John Snow | c87621e | 2016-10-27 12:07:00 -0400 | [diff] [blame] | 20 | #include "block/blockjob_int.h" |
Wen Congyang | 49d3e82 | 2016-07-27 15:01:43 +0800 | [diff] [blame] | 21 | #include "block/block_backup.h" |
Vladimir Sementsov-Ogievskiy | beb5f54 | 2019-09-20 17:20:48 +0300 | [diff] [blame] | 22 | #include "block/block-copy.h" |
Markus Armbruster | da34e65 | 2016-03-14 09:01:28 +0100 | [diff] [blame] | 23 | #include "qapi/error.h" |
Markus Armbruster | cc7a8ea | 2015-03-17 17:22:46 +0100 | [diff] [blame] | 24 | #include "qapi/qmp/qerror.h" |
Veronia Bahaa | f348b6d | 2016-03-20 19:16:19 +0200 | [diff] [blame] | 25 | #include "qemu/cutils.h" |
Max Reitz | 373340b | 2015-10-19 17:53:22 +0200 | [diff] [blame] | 26 | #include "sysemu/block-backend.h" |
Fam Zheng | b2f5646 | 2016-03-08 12:44:52 +0800 | [diff] [blame] | 27 | #include "qemu/bitmap.h" |
Vladimir Sementsov-Ogievskiy | a410a7f | 2017-02-28 22:33:40 +0300 | [diff] [blame] | 28 | #include "qemu/error-report.h" |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 29 | |
Vladimir Sementsov-Ogievskiy | 00e30f0 | 2019-10-01 16:14:09 +0300 | [diff] [blame] | 30 | #include "block/backup-top.h" |
| 31 | |
John Snow | 16096a4 | 2016-02-25 15:58:29 -0500 | [diff] [blame] | 32 | #define BACKUP_CLUSTER_SIZE_DEFAULT (1 << 16) |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 33 | |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 34 | typedef struct BackupBlockJob { |
| 35 | BlockJob common; |
Vladimir Sementsov-Ogievskiy | 00e30f0 | 2019-10-01 16:14:09 +0300 | [diff] [blame] | 36 | BlockDriverState *backup_top; |
Vladimir Sementsov-Ogievskiy | 2c8074c | 2019-09-20 17:20:46 +0300 | [diff] [blame] | 37 | BlockDriverState *source_bs; |
Vladimir Sementsov-Ogievskiy | ff789bf | 2021-02-05 19:37:19 +0300 | [diff] [blame] | 38 | BlockDriverState *target_bs; |
John Snow | 62aa1fb | 2019-07-29 16:35:53 -0400 | [diff] [blame] | 39 | |
John Snow | d58d845 | 2015-04-17 19:49:58 -0400 | [diff] [blame] | 40 | BdrvDirtyBitmap *sync_bitmap; |
John Snow | 62aa1fb | 2019-07-29 16:35:53 -0400 | [diff] [blame] | 41 | |
Ian Main | fc5d3f8 | 2013-07-26 11:39:04 -0700 | [diff] [blame] | 42 | MirrorSyncMode sync_mode; |
John Snow | c8b5650 | 2019-07-29 16:35:52 -0400 | [diff] [blame] | 43 | BitmapSyncMode bitmap_mode; |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 44 | BlockdevOnError on_source_error; |
| 45 | BlockdevOnError on_target_error; |
Kevin Wolf | 05df8a6 | 2018-01-18 18:08:22 +0100 | [diff] [blame] | 46 | uint64_t len; |
John Snow | 16096a4 | 2016-02-25 15:58:29 -0500 | [diff] [blame] | 47 | int64_t cluster_size; |
Vladimir Sementsov-Ogievskiy | 86c6a3b | 2021-01-17 00:46:43 +0300 | [diff] [blame] | 48 | BackupPerf perf; |
Vladimir Sementsov-Ogievskiy | a193b0f | 2017-10-12 16:53:10 +0300 | [diff] [blame] | 49 | |
Vladimir Sementsov-Ogievskiy | 2c8074c | 2019-09-20 17:20:46 +0300 | [diff] [blame] | 50 | BlockCopyState *bcs; |
Vladimir Sementsov-Ogievskiy | 71eed4c | 2021-01-17 00:46:59 +0300 | [diff] [blame] | 51 | |
| 52 | bool wait; |
| 53 | BlockCopyCallState *bg_bcs_call; |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 54 | } BackupBlockJob; |
| 55 | |
Kevin Wolf | bd21935 | 2018-01-19 15:54:40 +0100 | [diff] [blame] | 56 | static const BlockJobDriver backup_job_driver; |
| 57 | |
Fam Zheng | b976ea3 | 2015-11-05 18:13:10 -0500 | [diff] [blame] | 58 | static void backup_cleanup_sync_bitmap(BackupBlockJob *job, int ret) |
| 59 | { |
| 60 | BdrvDirtyBitmap *bm; |
John Snow | c23909e | 2019-07-29 16:35:53 -0400 | [diff] [blame] | 61 | bool sync = (((ret == 0) || (job->bitmap_mode == BITMAP_SYNC_MODE_ALWAYS)) \ |
| 62 | && (job->bitmap_mode != BITMAP_SYNC_MODE_NEVER)); |
Fam Zheng | b976ea3 | 2015-11-05 18:13:10 -0500 | [diff] [blame] | 63 | |
John Snow | c23909e | 2019-07-29 16:35:53 -0400 | [diff] [blame] | 64 | if (sync) { |
John Snow | cf0cd29 | 2019-07-29 16:35:53 -0400 | [diff] [blame] | 65 | /* |
John Snow | c23909e | 2019-07-29 16:35:53 -0400 | [diff] [blame] | 66 | * We succeeded, or we always intended to sync the bitmap. |
| 67 | * Delete this bitmap and install the child. |
| 68 | */ |
Vladimir Sementsov-Ogievskiy | 5deb6cb | 2019-09-16 17:19:09 +0300 | [diff] [blame] | 69 | bm = bdrv_dirty_bitmap_abdicate(job->sync_bitmap, NULL); |
John Snow | c23909e | 2019-07-29 16:35:53 -0400 | [diff] [blame] | 70 | } else { |
| 71 | /* |
| 72 | * We failed, or we never intended to sync the bitmap anyway. |
| 73 | * Merge the successor back into the parent, keeping all data. |
John Snow | cf0cd29 | 2019-07-29 16:35:53 -0400 | [diff] [blame] | 74 | */ |
Vladimir Sementsov-Ogievskiy | 5deb6cb | 2019-09-16 17:19:09 +0300 | [diff] [blame] | 75 | bm = bdrv_reclaim_dirty_bitmap(job->sync_bitmap, NULL); |
John Snow | c23909e | 2019-07-29 16:35:53 -0400 | [diff] [blame] | 76 | } |
| 77 | |
| 78 | assert(bm); |
| 79 | |
| 80 | if (ret < 0 && job->bitmap_mode == BITMAP_SYNC_MODE_ALWAYS) { |
| 81 | /* If we failed and synced, merge in the bits we didn't copy: */ |
Vladimir Sementsov-Ogievskiy | 397f4e9 | 2020-03-11 13:30:04 +0300 | [diff] [blame] | 82 | bdrv_dirty_bitmap_merge_internal(bm, block_copy_dirty_bitmap(job->bcs), |
John Snow | c23909e | 2019-07-29 16:35:53 -0400 | [diff] [blame] | 83 | NULL, true); |
Fam Zheng | b976ea3 | 2015-11-05 18:13:10 -0500 | [diff] [blame] | 84 | } |
| 85 | } |
| 86 | |
Kevin Wolf | 4ad3518 | 2018-04-19 17:30:16 +0200 | [diff] [blame] | 87 | static void backup_commit(Job *job) |
John Snow | c347b2c | 2015-11-05 18:13:16 -0500 | [diff] [blame] | 88 | { |
Kevin Wolf | 4ad3518 | 2018-04-19 17:30:16 +0200 | [diff] [blame] | 89 | BackupBlockJob *s = container_of(job, BackupBlockJob, common.job); |
John Snow | c347b2c | 2015-11-05 18:13:16 -0500 | [diff] [blame] | 90 | if (s->sync_bitmap) { |
| 91 | backup_cleanup_sync_bitmap(s, 0); |
| 92 | } |
| 93 | } |
| 94 | |
Kevin Wolf | 4ad3518 | 2018-04-19 17:30:16 +0200 | [diff] [blame] | 95 | static void backup_abort(Job *job) |
John Snow | c347b2c | 2015-11-05 18:13:16 -0500 | [diff] [blame] | 96 | { |
Kevin Wolf | 4ad3518 | 2018-04-19 17:30:16 +0200 | [diff] [blame] | 97 | BackupBlockJob *s = container_of(job, BackupBlockJob, common.job); |
John Snow | c347b2c | 2015-11-05 18:13:16 -0500 | [diff] [blame] | 98 | if (s->sync_bitmap) { |
| 99 | backup_cleanup_sync_bitmap(s, -1); |
| 100 | } |
| 101 | } |
| 102 | |
Kevin Wolf | 4ad3518 | 2018-04-19 17:30:16 +0200 | [diff] [blame] | 103 | static void backup_clean(Job *job) |
John Snow | e8a40bf | 2016-11-08 01:50:35 -0500 | [diff] [blame] | 104 | { |
Kevin Wolf | 4ad3518 | 2018-04-19 17:30:16 +0200 | [diff] [blame] | 105 | BackupBlockJob *s = container_of(job, BackupBlockJob, common.job); |
Max Reitz | bdc4c4c | 2021-02-19 16:33:46 +0100 | [diff] [blame] | 106 | block_job_remove_all_bdrv(&s->common); |
Vladimir Sementsov-Ogievskiy | 00e30f0 | 2019-10-01 16:14:09 +0300 | [diff] [blame] | 107 | bdrv_backup_top_drop(s->backup_top); |
John Snow | e8a40bf | 2016-11-08 01:50:35 -0500 | [diff] [blame] | 108 | } |
| 109 | |
Wen Congyang | 49d3e82 | 2016-07-27 15:01:43 +0800 | [diff] [blame] | 110 | void backup_do_checkpoint(BlockJob *job, Error **errp) |
| 111 | { |
| 112 | BackupBlockJob *backup_job = container_of(job, BackupBlockJob, common); |
Wen Congyang | 49d3e82 | 2016-07-27 15:01:43 +0800 | [diff] [blame] | 113 | |
Kevin Wolf | bd21935 | 2018-01-19 15:54:40 +0100 | [diff] [blame] | 114 | assert(block_job_driver(job) == &backup_job_driver); |
Wen Congyang | 49d3e82 | 2016-07-27 15:01:43 +0800 | [diff] [blame] | 115 | |
| 116 | if (backup_job->sync_mode != MIRROR_SYNC_MODE_NONE) { |
| 117 | error_setg(errp, "The backup job only supports block checkpoint in" |
| 118 | " sync=none mode"); |
| 119 | return; |
| 120 | } |
| 121 | |
Vladimir Sementsov-Ogievskiy | 397f4e9 | 2020-03-11 13:30:04 +0300 | [diff] [blame] | 122 | bdrv_set_dirty_bitmap(block_copy_dirty_bitmap(backup_job->bcs), 0, |
| 123 | backup_job->len); |
Wen Congyang | 49d3e82 | 2016-07-27 15:01:43 +0800 | [diff] [blame] | 124 | } |
| 125 | |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 126 | static BlockErrorAction backup_error_action(BackupBlockJob *job, |
| 127 | bool read, int error) |
| 128 | { |
| 129 | if (read) { |
Kevin Wolf | 81e254d | 2016-04-18 11:36:38 +0200 | [diff] [blame] | 130 | return block_job_error_action(&job->common, job->on_source_error, |
| 131 | true, error); |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 132 | } else { |
Kevin Wolf | 81e254d | 2016-04-18 11:36:38 +0200 | [diff] [blame] | 133 | return block_job_error_action(&job->common, job->on_target_error, |
| 134 | false, error); |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 135 | } |
| 136 | } |
| 137 | |
Vladimir Sementsov-Ogievskiy | 71eed4c | 2021-01-17 00:46:59 +0300 | [diff] [blame] | 138 | static void coroutine_fn backup_block_copy_callback(void *opaque) |
John Snow | d58d845 | 2015-04-17 19:49:58 -0400 | [diff] [blame] | 139 | { |
Vladimir Sementsov-Ogievskiy | 71eed4c | 2021-01-17 00:46:59 +0300 | [diff] [blame] | 140 | BackupBlockJob *s = opaque; |
Kevin Wolf | dee81d5 | 2018-01-18 21:19:38 +0100 | [diff] [blame] | 141 | |
Vladimir Sementsov-Ogievskiy | 71eed4c | 2021-01-17 00:46:59 +0300 | [diff] [blame] | 142 | if (s->wait) { |
| 143 | s->wait = false; |
| 144 | aio_co_wake(s->common.job.co); |
| 145 | } else { |
| 146 | job_enter(&s->common.job); |
John Snow | d58d845 | 2015-04-17 19:49:58 -0400 | [diff] [blame] | 147 | } |
John Snow | d58d845 | 2015-04-17 19:49:58 -0400 | [diff] [blame] | 148 | } |
| 149 | |
Vladimir Sementsov-Ogievskiy | c334e89 | 2019-04-29 12:08:41 +0300 | [diff] [blame] | 150 | static int coroutine_fn backup_loop(BackupBlockJob *job) |
John Snow | d58d845 | 2015-04-17 19:49:58 -0400 | [diff] [blame] | 151 | { |
Vladimir Sementsov-Ogievskiy | 71eed4c | 2021-01-17 00:46:59 +0300 | [diff] [blame] | 152 | BlockCopyCallState *s = NULL; |
John Snow | 62aa1fb | 2019-07-29 16:35:53 -0400 | [diff] [blame] | 153 | int ret = 0; |
Vladimir Sementsov-Ogievskiy | 71eed4c | 2021-01-17 00:46:59 +0300 | [diff] [blame] | 154 | bool error_is_read; |
| 155 | BlockErrorAction act; |
John Snow | d58d845 | 2015-04-17 19:49:58 -0400 | [diff] [blame] | 156 | |
Vladimir Sementsov-Ogievskiy | 71eed4c | 2021-01-17 00:46:59 +0300 | [diff] [blame] | 157 | while (true) { /* retry loop */ |
| 158 | job->bg_bcs_call = s = block_copy_async(job->bcs, 0, |
| 159 | QEMU_ALIGN_UP(job->len, job->cluster_size), |
| 160 | job->perf.max_workers, job->perf.max_chunk, |
| 161 | backup_block_copy_callback, job); |
| 162 | |
| 163 | while (!block_copy_call_finished(s) && |
| 164 | !job_is_cancelled(&job->common.job)) |
| 165 | { |
| 166 | job_yield(&job->common.job); |
| 167 | } |
| 168 | |
| 169 | if (!block_copy_call_finished(s)) { |
| 170 | assert(job_is_cancelled(&job->common.job)); |
| 171 | /* |
| 172 | * Note that we can't use job_yield() here, as it doesn't work for |
| 173 | * cancelled job. |
| 174 | */ |
| 175 | block_copy_call_cancel(s); |
| 176 | job->wait = true; |
| 177 | qemu_coroutine_yield(); |
| 178 | assert(block_copy_call_finished(s)); |
| 179 | ret = 0; |
| 180 | goto out; |
| 181 | } |
| 182 | |
| 183 | if (job_is_cancelled(&job->common.job) || |
| 184 | block_copy_call_succeeded(s)) |
| 185 | { |
| 186 | ret = 0; |
| 187 | goto out; |
| 188 | } |
| 189 | |
| 190 | if (block_copy_call_cancelled(s)) { |
| 191 | /* |
| 192 | * Job is not cancelled but only block-copy call. This is possible |
| 193 | * after job pause. Now the pause is finished, start new block-copy |
| 194 | * iteration. |
| 195 | */ |
| 196 | block_copy_call_free(s); |
| 197 | continue; |
| 198 | } |
| 199 | |
| 200 | /* The only remaining case is failed block-copy call. */ |
| 201 | assert(block_copy_call_failed(s)); |
| 202 | |
| 203 | ret = block_copy_call_status(s, &error_is_read); |
| 204 | act = backup_error_action(job, error_is_read, -ret); |
| 205 | switch (act) { |
| 206 | case BLOCK_ERROR_ACTION_REPORT: |
| 207 | goto out; |
| 208 | case BLOCK_ERROR_ACTION_STOP: |
| 209 | /* |
| 210 | * Go to pause prior to starting new block-copy call on the next |
| 211 | * iteration. |
| 212 | */ |
| 213 | job_pause_point(&job->common.job); |
| 214 | break; |
| 215 | case BLOCK_ERROR_ACTION_IGNORE: |
| 216 | /* Proceed to new block-copy call to retry. */ |
| 217 | break; |
| 218 | default: |
| 219 | abort(); |
| 220 | } |
| 221 | |
| 222 | block_copy_call_free(s); |
John Snow | d58d845 | 2015-04-17 19:49:58 -0400 | [diff] [blame] | 223 | } |
| 224 | |
Vladimir Sementsov-Ogievskiy | 71eed4c | 2021-01-17 00:46:59 +0300 | [diff] [blame] | 225 | out: |
| 226 | block_copy_call_free(s); |
| 227 | job->bg_bcs_call = NULL; |
John Snow | 62aa1fb | 2019-07-29 16:35:53 -0400 | [diff] [blame] | 228 | return ret; |
John Snow | d58d845 | 2015-04-17 19:49:58 -0400 | [diff] [blame] | 229 | } |
| 230 | |
Vladimir Sementsov-Ogievskiy | 397f4e9 | 2020-03-11 13:30:04 +0300 | [diff] [blame] | 231 | static void backup_init_bcs_bitmap(BackupBlockJob *job) |
Vladimir Sementsov-Ogievskiy | 8cc6dc6 | 2017-10-12 16:53:11 +0300 | [diff] [blame] | 232 | { |
John Snow | 141cdcd | 2019-07-29 16:35:55 -0400 | [diff] [blame] | 233 | bool ret; |
| 234 | uint64_t estimate; |
Vladimir Sementsov-Ogievskiy | 397f4e9 | 2020-03-11 13:30:04 +0300 | [diff] [blame] | 235 | BdrvDirtyBitmap *bcs_bitmap = block_copy_dirty_bitmap(job->bcs); |
Vladimir Sementsov-Ogievskiy | 8cc6dc6 | 2017-10-12 16:53:11 +0300 | [diff] [blame] | 236 | |
John Snow | 141cdcd | 2019-07-29 16:35:55 -0400 | [diff] [blame] | 237 | if (job->sync_mode == MIRROR_SYNC_MODE_BITMAP) { |
Vladimir Sementsov-Ogievskiy | 397f4e9 | 2020-03-11 13:30:04 +0300 | [diff] [blame] | 238 | ret = bdrv_dirty_bitmap_merge_internal(bcs_bitmap, job->sync_bitmap, |
John Snow | 141cdcd | 2019-07-29 16:35:55 -0400 | [diff] [blame] | 239 | NULL, true); |
| 240 | assert(ret); |
| 241 | } else { |
John Snow | 7e30dd6 | 2019-07-29 16:35:55 -0400 | [diff] [blame] | 242 | if (job->sync_mode == MIRROR_SYNC_MODE_TOP) { |
| 243 | /* |
| 244 | * We can't hog the coroutine to initialize this thoroughly. |
| 245 | * Set a flag and resume work when we are able to yield safely. |
| 246 | */ |
Vladimir Sementsov-Ogievskiy | 397f4e9 | 2020-03-11 13:30:04 +0300 | [diff] [blame] | 247 | block_copy_set_skip_unallocated(job->bcs, true); |
John Snow | 7e30dd6 | 2019-07-29 16:35:55 -0400 | [diff] [blame] | 248 | } |
Vladimir Sementsov-Ogievskiy | 397f4e9 | 2020-03-11 13:30:04 +0300 | [diff] [blame] | 249 | bdrv_set_dirty_bitmap(bcs_bitmap, 0, job->len); |
John Snow | 141cdcd | 2019-07-29 16:35:55 -0400 | [diff] [blame] | 250 | } |
| 251 | |
Vladimir Sementsov-Ogievskiy | 397f4e9 | 2020-03-11 13:30:04 +0300 | [diff] [blame] | 252 | estimate = bdrv_get_dirty_count(bcs_bitmap); |
John Snow | 141cdcd | 2019-07-29 16:35:55 -0400 | [diff] [blame] | 253 | job_progress_set_remaining(&job->common.job, estimate); |
Vladimir Sementsov-Ogievskiy | 8cc6dc6 | 2017-10-12 16:53:11 +0300 | [diff] [blame] | 254 | } |
| 255 | |
John Snow | 6870277 | 2018-08-29 21:57:32 -0400 | [diff] [blame] | 256 | static int coroutine_fn backup_run(Job *job, Error **errp) |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 257 | { |
John Snow | 6870277 | 2018-08-29 21:57:32 -0400 | [diff] [blame] | 258 | BackupBlockJob *s = container_of(job, BackupBlockJob, common.job); |
Vladimir Sementsov-Ogievskiy | 511e7d3 | 2021-01-17 00:46:58 +0300 | [diff] [blame] | 259 | int ret; |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 260 | |
Vladimir Sementsov-Ogievskiy | 397f4e9 | 2020-03-11 13:30:04 +0300 | [diff] [blame] | 261 | backup_init_bcs_bitmap(s); |
Vladimir Sementsov-Ogievskiy | 8cc6dc6 | 2017-10-12 16:53:11 +0300 | [diff] [blame] | 262 | |
John Snow | 7e30dd6 | 2019-07-29 16:35:55 -0400 | [diff] [blame] | 263 | if (s->sync_mode == MIRROR_SYNC_MODE_TOP) { |
| 264 | int64_t offset = 0; |
| 265 | int64_t count; |
| 266 | |
| 267 | for (offset = 0; offset < s->len; ) { |
Vladimir Sementsov-Ogievskiy | 71eed4c | 2021-01-17 00:46:59 +0300 | [diff] [blame] | 268 | if (job_is_cancelled(job)) { |
| 269 | return -ECANCELED; |
| 270 | } |
| 271 | |
| 272 | job_pause_point(job); |
| 273 | |
| 274 | if (job_is_cancelled(job)) { |
Vladimir Sementsov-Ogievskiy | 511e7d3 | 2021-01-17 00:46:58 +0300 | [diff] [blame] | 275 | return -ECANCELED; |
John Snow | 7e30dd6 | 2019-07-29 16:35:55 -0400 | [diff] [blame] | 276 | } |
| 277 | |
Vladimir Sementsov-Ogievskiy | 2c8074c | 2019-09-20 17:20:46 +0300 | [diff] [blame] | 278 | ret = block_copy_reset_unallocated(s->bcs, offset, &count); |
John Snow | 7e30dd6 | 2019-07-29 16:35:55 -0400 | [diff] [blame] | 279 | if (ret < 0) { |
Vladimir Sementsov-Ogievskiy | 511e7d3 | 2021-01-17 00:46:58 +0300 | [diff] [blame] | 280 | return ret; |
John Snow | 7e30dd6 | 2019-07-29 16:35:55 -0400 | [diff] [blame] | 281 | } |
| 282 | |
| 283 | offset += count; |
| 284 | } |
Vladimir Sementsov-Ogievskiy | 397f4e9 | 2020-03-11 13:30:04 +0300 | [diff] [blame] | 285 | block_copy_set_skip_unallocated(s->bcs, false); |
John Snow | 7e30dd6 | 2019-07-29 16:35:55 -0400 | [diff] [blame] | 286 | } |
| 287 | |
John Snow | 6870277 | 2018-08-29 21:57:32 -0400 | [diff] [blame] | 288 | if (s->sync_mode == MIRROR_SYNC_MODE_NONE) { |
Vladimir Sementsov-Ogievskiy | 0e23e38 | 2019-09-20 17:20:47 +0300 | [diff] [blame] | 289 | /* |
Vladimir Sementsov-Ogievskiy | 397f4e9 | 2020-03-11 13:30:04 +0300 | [diff] [blame] | 290 | * All bits are set in bcs bitmap to allow any cluster to be copied. |
Vladimir Sementsov-Ogievskiy | 0e23e38 | 2019-09-20 17:20:47 +0300 | [diff] [blame] | 291 | * This does not actually require them to be copied. |
| 292 | */ |
John Snow | 6870277 | 2018-08-29 21:57:32 -0400 | [diff] [blame] | 293 | while (!job_is_cancelled(job)) { |
Vladimir Sementsov-Ogievskiy | 0e23e38 | 2019-09-20 17:20:47 +0300 | [diff] [blame] | 294 | /* |
| 295 | * Yield until the job is cancelled. We just let our before_write |
| 296 | * notify callback service CoW requests. |
| 297 | */ |
John Snow | 6870277 | 2018-08-29 21:57:32 -0400 | [diff] [blame] | 298 | job_yield(job); |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 299 | } |
Ian Main | fc5d3f8 | 2013-07-26 11:39:04 -0700 | [diff] [blame] | 300 | } else { |
Vladimir Sementsov-Ogievskiy | 511e7d3 | 2021-01-17 00:46:58 +0300 | [diff] [blame] | 301 | return backup_loop(s); |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 302 | } |
| 303 | |
Vladimir Sementsov-Ogievskiy | 511e7d3 | 2021-01-17 00:46:58 +0300 | [diff] [blame] | 304 | return 0; |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 305 | } |
| 306 | |
Vladimir Sementsov-Ogievskiy | 71eed4c | 2021-01-17 00:46:59 +0300 | [diff] [blame] | 307 | static void coroutine_fn backup_pause(Job *job) |
| 308 | { |
| 309 | BackupBlockJob *s = container_of(job, BackupBlockJob, common.job); |
| 310 | |
| 311 | if (s->bg_bcs_call && !block_copy_call_finished(s->bg_bcs_call)) { |
| 312 | block_copy_call_cancel(s->bg_bcs_call); |
| 313 | s->wait = true; |
| 314 | qemu_coroutine_yield(); |
| 315 | } |
| 316 | } |
| 317 | |
| 318 | static void coroutine_fn backup_set_speed(BlockJob *job, int64_t speed) |
| 319 | { |
| 320 | BackupBlockJob *s = container_of(job, BackupBlockJob, common); |
| 321 | |
| 322 | /* |
| 323 | * block_job_set_speed() is called first from block_job_create(), when we |
| 324 | * don't yet have s->bcs. |
| 325 | */ |
| 326 | if (s->bcs) { |
| 327 | block_copy_set_speed(s->bcs, speed); |
| 328 | if (s->bg_bcs_call) { |
| 329 | block_copy_kick(s->bg_bcs_call); |
| 330 | } |
| 331 | } |
| 332 | } |
| 333 | |
Vladimir Sementsov-Ogievskiy | 9c785cd | 2021-04-21 10:58:58 +0300 | [diff] [blame] | 334 | static void backup_cancel(Job *job, bool force) |
Vladimir Sementsov-Ogievskiy | ff789bf | 2021-02-05 19:37:19 +0300 | [diff] [blame] | 335 | { |
| 336 | BackupBlockJob *s = container_of(job, BackupBlockJob, common.job); |
| 337 | |
| 338 | bdrv_cancel_in_flight(s->target_bs); |
| 339 | } |
| 340 | |
John Snow | a7815a7 | 2016-11-08 01:50:36 -0500 | [diff] [blame] | 341 | static const BlockJobDriver backup_job_driver = { |
Kevin Wolf | 33e9e9b | 2018-04-12 17:29:59 +0200 | [diff] [blame] | 342 | .job_driver = { |
| 343 | .instance_size = sizeof(BackupBlockJob), |
Kevin Wolf | 252291e | 2018-04-12 17:57:08 +0200 | [diff] [blame] | 344 | .job_type = JOB_TYPE_BACKUP, |
Kevin Wolf | 80fa2c7 | 2018-04-13 18:50:05 +0200 | [diff] [blame] | 345 | .free = block_job_free, |
Kevin Wolf | b15de82 | 2018-04-18 17:10:26 +0200 | [diff] [blame] | 346 | .user_resume = block_job_user_resume, |
John Snow | f67432a | 2018-08-29 21:57:26 -0400 | [diff] [blame] | 347 | .run = backup_run, |
Kevin Wolf | 4ad3518 | 2018-04-19 17:30:16 +0200 | [diff] [blame] | 348 | .commit = backup_commit, |
| 349 | .abort = backup_abort, |
| 350 | .clean = backup_clean, |
Vladimir Sementsov-Ogievskiy | 71eed4c | 2021-01-17 00:46:59 +0300 | [diff] [blame] | 351 | .pause = backup_pause, |
Vladimir Sementsov-Ogievskiy | ff789bf | 2021-02-05 19:37:19 +0300 | [diff] [blame] | 352 | .cancel = backup_cancel, |
Vladimir Sementsov-Ogievskiy | 71eed4c | 2021-01-17 00:46:59 +0300 | [diff] [blame] | 353 | }, |
| 354 | .set_speed = backup_set_speed, |
John Snow | a7815a7 | 2016-11-08 01:50:36 -0500 | [diff] [blame] | 355 | }; |
| 356 | |
Vladimir Sementsov-Ogievskiy | ae6b12f | 2019-04-29 12:08:42 +0300 | [diff] [blame] | 357 | static int64_t backup_calculate_cluster_size(BlockDriverState *target, |
| 358 | Error **errp) |
| 359 | { |
| 360 | int ret; |
| 361 | BlockDriverInfo bdi; |
Max Reitz | 2b088c6 | 2019-06-12 17:46:45 +0200 | [diff] [blame] | 362 | bool target_does_cow = bdrv_backing_chain_next(target); |
Vladimir Sementsov-Ogievskiy | ae6b12f | 2019-04-29 12:08:42 +0300 | [diff] [blame] | 363 | |
| 364 | /* |
| 365 | * If there is no backing file on the target, we cannot rely on COW if our |
| 366 | * backup cluster size is smaller than the target cluster size. Even for |
| 367 | * targets with a backing file, try to avoid COW if possible. |
| 368 | */ |
| 369 | ret = bdrv_get_info(target, &bdi); |
Max Reitz | 2b088c6 | 2019-06-12 17:46:45 +0200 | [diff] [blame] | 370 | if (ret == -ENOTSUP && !target_does_cow) { |
Vladimir Sementsov-Ogievskiy | ae6b12f | 2019-04-29 12:08:42 +0300 | [diff] [blame] | 371 | /* Cluster size is not defined */ |
| 372 | warn_report("The target block device doesn't provide " |
| 373 | "information about the block size and it doesn't have a " |
| 374 | "backing file. The default block size of %u bytes is " |
| 375 | "used. If the actual block size of the target exceeds " |
| 376 | "this default, the backup may be unusable", |
| 377 | BACKUP_CLUSTER_SIZE_DEFAULT); |
| 378 | return BACKUP_CLUSTER_SIZE_DEFAULT; |
Max Reitz | 2b088c6 | 2019-06-12 17:46:45 +0200 | [diff] [blame] | 379 | } else if (ret < 0 && !target_does_cow) { |
Vladimir Sementsov-Ogievskiy | ae6b12f | 2019-04-29 12:08:42 +0300 | [diff] [blame] | 380 | error_setg_errno(errp, -ret, |
| 381 | "Couldn't determine the cluster size of the target image, " |
| 382 | "which has no backing file"); |
| 383 | error_append_hint(errp, |
| 384 | "Aborting, since this may create an unusable destination image\n"); |
| 385 | return ret; |
Max Reitz | 2b088c6 | 2019-06-12 17:46:45 +0200 | [diff] [blame] | 386 | } else if (ret < 0 && target_does_cow) { |
Vladimir Sementsov-Ogievskiy | ae6b12f | 2019-04-29 12:08:42 +0300 | [diff] [blame] | 387 | /* Not fatal; just trudge on ahead. */ |
| 388 | return BACKUP_CLUSTER_SIZE_DEFAULT; |
| 389 | } |
| 390 | |
| 391 | return MAX(BACKUP_CLUSTER_SIZE_DEFAULT, bdi.cluster_size); |
| 392 | } |
| 393 | |
John Snow | 111049a | 2016-11-08 01:50:38 -0500 | [diff] [blame] | 394 | BlockJob *backup_job_create(const char *job_id, BlockDriverState *bs, |
Alberto Garcia | 70559d4 | 2016-07-05 17:28:58 +0300 | [diff] [blame] | 395 | BlockDriverState *target, int64_t speed, |
| 396 | MirrorSyncMode sync_mode, BdrvDirtyBitmap *sync_bitmap, |
John Snow | c8b5650 | 2019-07-29 16:35:52 -0400 | [diff] [blame] | 397 | BitmapSyncMode bitmap_mode, |
Pavel Butsykin | 13b9414 | 2016-07-22 11:17:52 +0300 | [diff] [blame] | 398 | bool compress, |
Vladimir Sementsov-Ogievskiy | 00e30f0 | 2019-10-01 16:14:09 +0300 | [diff] [blame] | 399 | const char *filter_node_name, |
Vladimir Sementsov-Ogievskiy | 86c6a3b | 2021-01-17 00:46:43 +0300 | [diff] [blame] | 400 | BackupPerf *perf, |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 401 | BlockdevOnError on_source_error, |
| 402 | BlockdevOnError on_target_error, |
John Snow | 47970df | 2016-10-27 12:06:57 -0400 | [diff] [blame] | 403 | int creation_flags, |
Markus Armbruster | 097310b | 2014-10-07 13:59:15 +0200 | [diff] [blame] | 404 | BlockCompletionFunc *cb, void *opaque, |
Kevin Wolf | 62c9e41 | 2018-04-19 16:09:52 +0200 | [diff] [blame] | 405 | JobTxn *txn, Error **errp) |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 406 | { |
Kevin Wolf | 958a04b | 2020-04-30 16:27:54 +0200 | [diff] [blame] | 407 | int64_t len, target_len; |
Kevin Wolf | 91ab688 | 2016-04-14 12:59:55 +0200 | [diff] [blame] | 408 | BackupBlockJob *job = NULL; |
Vladimir Sementsov-Ogievskiy | ae6b12f | 2019-04-29 12:08:42 +0300 | [diff] [blame] | 409 | int64_t cluster_size; |
Vladimir Sementsov-Ogievskiy | 2c8074c | 2019-09-20 17:20:46 +0300 | [diff] [blame] | 410 | BdrvRequestFlags write_flags; |
Vladimir Sementsov-Ogievskiy | 00e30f0 | 2019-10-01 16:14:09 +0300 | [diff] [blame] | 411 | BlockDriverState *backup_top = NULL; |
| 412 | BlockCopyState *bcs = NULL; |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 413 | |
| 414 | assert(bs); |
| 415 | assert(target); |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 416 | |
John Snow | a6c9365 | 2019-07-29 16:35:55 -0400 | [diff] [blame] | 417 | /* QMP interface protects us from these cases */ |
| 418 | assert(sync_mode != MIRROR_SYNC_MODE_INCREMENTAL); |
| 419 | assert(sync_bitmap || sync_mode != MIRROR_SYNC_MODE_BITMAP); |
| 420 | |
Fam Zheng | c29c1dd | 2014-12-18 18:37:05 +0800 | [diff] [blame] | 421 | if (bs == target) { |
| 422 | error_setg(errp, "Source and target cannot be the same"); |
John Snow | 111049a | 2016-11-08 01:50:38 -0500 | [diff] [blame] | 423 | return NULL; |
Fam Zheng | c29c1dd | 2014-12-18 18:37:05 +0800 | [diff] [blame] | 424 | } |
| 425 | |
Fam Zheng | c29c1dd | 2014-12-18 18:37:05 +0800 | [diff] [blame] | 426 | if (!bdrv_is_inserted(bs)) { |
| 427 | error_setg(errp, "Device is not inserted: %s", |
| 428 | bdrv_get_device_name(bs)); |
John Snow | 111049a | 2016-11-08 01:50:38 -0500 | [diff] [blame] | 429 | return NULL; |
Fam Zheng | c29c1dd | 2014-12-18 18:37:05 +0800 | [diff] [blame] | 430 | } |
| 431 | |
| 432 | if (!bdrv_is_inserted(target)) { |
| 433 | error_setg(errp, "Device is not inserted: %s", |
| 434 | bdrv_get_device_name(target)); |
John Snow | 111049a | 2016-11-08 01:50:38 -0500 | [diff] [blame] | 435 | return NULL; |
Fam Zheng | c29c1dd | 2014-12-18 18:37:05 +0800 | [diff] [blame] | 436 | } |
| 437 | |
Max Reitz | 2b088c6 | 2019-06-12 17:46:45 +0200 | [diff] [blame] | 438 | if (compress && !bdrv_supports_compressed_writes(target)) { |
Pavel Butsykin | 13b9414 | 2016-07-22 11:17:52 +0300 | [diff] [blame] | 439 | error_setg(errp, "Compression is not supported for this drive %s", |
| 440 | bdrv_get_device_name(target)); |
John Snow | 111049a | 2016-11-08 01:50:38 -0500 | [diff] [blame] | 441 | return NULL; |
Pavel Butsykin | 13b9414 | 2016-07-22 11:17:52 +0300 | [diff] [blame] | 442 | } |
| 443 | |
Fam Zheng | c29c1dd | 2014-12-18 18:37:05 +0800 | [diff] [blame] | 444 | if (bdrv_op_is_blocked(bs, BLOCK_OP_TYPE_BACKUP_SOURCE, errp)) { |
John Snow | 111049a | 2016-11-08 01:50:38 -0500 | [diff] [blame] | 445 | return NULL; |
Fam Zheng | c29c1dd | 2014-12-18 18:37:05 +0800 | [diff] [blame] | 446 | } |
| 447 | |
| 448 | if (bdrv_op_is_blocked(target, BLOCK_OP_TYPE_BACKUP_TARGET, errp)) { |
John Snow | 111049a | 2016-11-08 01:50:38 -0500 | [diff] [blame] | 449 | return NULL; |
Fam Zheng | c29c1dd | 2014-12-18 18:37:05 +0800 | [diff] [blame] | 450 | } |
| 451 | |
Vladimir Sementsov-Ogievskiy | 2c59fd8 | 2021-01-17 00:46:52 +0300 | [diff] [blame] | 452 | cluster_size = backup_calculate_cluster_size(target, errp); |
| 453 | if (cluster_size < 0) { |
| 454 | goto error; |
| 455 | } |
| 456 | |
| 457 | if (perf->max_workers < 1) { |
| 458 | error_setg(errp, "max-workers must be greater than zero"); |
| 459 | return NULL; |
| 460 | } |
| 461 | |
| 462 | if (perf->max_chunk < 0) { |
| 463 | error_setg(errp, "max-chunk must be zero (which means no limit) or " |
| 464 | "positive"); |
| 465 | return NULL; |
| 466 | } |
| 467 | |
| 468 | if (perf->max_chunk && perf->max_chunk < cluster_size) { |
| 469 | error_setg(errp, "Required max-chunk (%" PRIi64 ") is less than backup " |
| 470 | "cluster size (%" PRIi64 ")", perf->max_chunk, cluster_size); |
| 471 | return NULL; |
| 472 | } |
| 473 | |
| 474 | |
John Snow | 1a2b8b4 | 2019-07-29 16:35:55 -0400 | [diff] [blame] | 475 | if (sync_bitmap) { |
John Snow | b30ffbe | 2019-07-29 16:35:54 -0400 | [diff] [blame] | 476 | /* If we need to write to this bitmap, check that we can: */ |
| 477 | if (bitmap_mode != BITMAP_SYNC_MODE_NEVER && |
| 478 | bdrv_dirty_bitmap_check(sync_bitmap, BDRV_BITMAP_DEFAULT, errp)) { |
| 479 | return NULL; |
| 480 | } |
| 481 | |
John Snow | d58d845 | 2015-04-17 19:49:58 -0400 | [diff] [blame] | 482 | /* Create a new bitmap, and freeze/disable this one. */ |
Vladimir Sementsov-Ogievskiy | 5deb6cb | 2019-09-16 17:19:09 +0300 | [diff] [blame] | 483 | if (bdrv_dirty_bitmap_create_successor(sync_bitmap, errp) < 0) { |
John Snow | 111049a | 2016-11-08 01:50:38 -0500 | [diff] [blame] | 484 | return NULL; |
John Snow | d58d845 | 2015-04-17 19:49:58 -0400 | [diff] [blame] | 485 | } |
John Snow | d58d845 | 2015-04-17 19:49:58 -0400 | [diff] [blame] | 486 | } |
| 487 | |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 488 | len = bdrv_getlength(bs); |
| 489 | if (len < 0) { |
Kevin Wolf | 5822663 | 2020-04-30 16:27:53 +0200 | [diff] [blame] | 490 | error_setg_errno(errp, -len, "Unable to get length for '%s'", |
| 491 | bdrv_get_device_or_node_name(bs)); |
John Snow | d58d845 | 2015-04-17 19:49:58 -0400 | [diff] [blame] | 492 | goto error; |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 493 | } |
| 494 | |
Kevin Wolf | 958a04b | 2020-04-30 16:27:54 +0200 | [diff] [blame] | 495 | target_len = bdrv_getlength(target); |
| 496 | if (target_len < 0) { |
| 497 | error_setg_errno(errp, -target_len, "Unable to get length for '%s'", |
| 498 | bdrv_get_device_or_node_name(bs)); |
| 499 | goto error; |
| 500 | } |
| 501 | |
| 502 | if (target_len != len) { |
| 503 | error_setg(errp, "Source and target image have different sizes"); |
| 504 | goto error; |
| 505 | } |
| 506 | |
Vladimir Sementsov-Ogievskiy | a1ed82b | 2019-07-30 19:32:51 +0300 | [diff] [blame] | 507 | /* |
Vladimir Sementsov-Ogievskiy | 372c67e | 2019-09-20 17:20:45 +0300 | [diff] [blame] | 508 | * If source is in backing chain of target assume that target is going to be |
| 509 | * used for "image fleecing", i.e. it should represent a kind of snapshot of |
| 510 | * source at backup-start point in time. And target is going to be read by |
| 511 | * somebody (for example, used as NBD export) during backup job. |
| 512 | * |
| 513 | * In this case, we need to add BDRV_REQ_SERIALISING write flag to avoid |
| 514 | * intersection of backup writes and third party reads from target, |
| 515 | * otherwise reading from target we may occasionally read already updated by |
| 516 | * guest data. |
| 517 | * |
| 518 | * For more information see commit f8d59dfb40bb and test |
| 519 | * tests/qemu-iotests/222 |
Vladimir Sementsov-Ogievskiy | a1ed82b | 2019-07-30 19:32:51 +0300 | [diff] [blame] | 520 | */ |
Vladimir Sementsov-Ogievskiy | 2c8074c | 2019-09-20 17:20:46 +0300 | [diff] [blame] | 521 | write_flags = (bdrv_chain_contains(target, bs) ? BDRV_REQ_SERIALISING : 0) | |
| 522 | (compress ? BDRV_REQ_WRITE_COMPRESSED : 0), |
| 523 | |
Vladimir Sementsov-Ogievskiy | 00e30f0 | 2019-10-01 16:14:09 +0300 | [diff] [blame] | 524 | backup_top = bdrv_backup_top_append(bs, target, filter_node_name, |
Vladimir Sementsov-Ogievskiy | 86c6a3b | 2021-01-17 00:46:43 +0300 | [diff] [blame] | 525 | cluster_size, perf, |
| 526 | write_flags, &bcs, errp); |
Vladimir Sementsov-Ogievskiy | 00e30f0 | 2019-10-01 16:14:09 +0300 | [diff] [blame] | 527 | if (!backup_top) { |
| 528 | goto error; |
| 529 | } |
| 530 | |
Vladimir Sementsov-Ogievskiy | 843670f | 2019-10-01 16:14:06 +0300 | [diff] [blame] | 531 | /* job->len is fixed, so we can't allow resize */ |
Vladimir Sementsov-Ogievskiy | 00e30f0 | 2019-10-01 16:14:09 +0300 | [diff] [blame] | 532 | job = block_job_create(job_id, &backup_job_driver, txn, backup_top, |
| 533 | 0, BLK_PERM_ALL, |
Vladimir Sementsov-Ogievskiy | 843670f | 2019-10-01 16:14:06 +0300 | [diff] [blame] | 534 | speed, creation_flags, cb, opaque, errp); |
| 535 | if (!job) { |
| 536 | goto error; |
| 537 | } |
| 538 | |
Vladimir Sementsov-Ogievskiy | 00e30f0 | 2019-10-01 16:14:09 +0300 | [diff] [blame] | 539 | job->backup_top = backup_top; |
Vladimir Sementsov-Ogievskiy | 843670f | 2019-10-01 16:14:06 +0300 | [diff] [blame] | 540 | job->source_bs = bs; |
Vladimir Sementsov-Ogievskiy | ff789bf | 2021-02-05 19:37:19 +0300 | [diff] [blame] | 541 | job->target_bs = target; |
Vladimir Sementsov-Ogievskiy | 843670f | 2019-10-01 16:14:06 +0300 | [diff] [blame] | 542 | job->on_source_error = on_source_error; |
| 543 | job->on_target_error = on_target_error; |
| 544 | job->sync_mode = sync_mode; |
| 545 | job->sync_bitmap = sync_bitmap; |
| 546 | job->bitmap_mode = bitmap_mode; |
Vladimir Sementsov-Ogievskiy | 00e30f0 | 2019-10-01 16:14:09 +0300 | [diff] [blame] | 547 | job->bcs = bcs; |
Vladimir Sementsov-Ogievskiy | ae6b12f | 2019-04-29 12:08:42 +0300 | [diff] [blame] | 548 | job->cluster_size = cluster_size; |
Vladimir Sementsov-Ogievskiy | 843670f | 2019-10-01 16:14:06 +0300 | [diff] [blame] | 549 | job->len = len; |
Vladimir Sementsov-Ogievskiy | 86c6a3b | 2021-01-17 00:46:43 +0300 | [diff] [blame] | 550 | job->perf = *perf; |
John Snow | 4c9bca7 | 2016-02-25 15:58:30 -0500 | [diff] [blame] | 551 | |
Vladimir Sementsov-Ogievskiy | d0ebeca | 2020-03-11 13:29:57 +0300 | [diff] [blame] | 552 | block_copy_set_progress_meter(bcs, &job->common.job.progress); |
Vladimir Sementsov-Ogievskiy | 71eed4c | 2021-01-17 00:46:59 +0300 | [diff] [blame] | 553 | block_copy_set_speed(bcs, speed); |
Vladimir Sementsov-Ogievskiy | 0f4b02b | 2019-10-01 16:14:07 +0300 | [diff] [blame] | 554 | |
Vladimir Sementsov-Ogievskiy | 00e30f0 | 2019-10-01 16:14:09 +0300 | [diff] [blame] | 555 | /* Required permissions are already taken by backup-top target */ |
Kevin Wolf | 76d554e | 2017-01-17 11:56:42 +0100 | [diff] [blame] | 556 | block_job_add_bdrv(&job->common, "target", target, 0, BLK_PERM_ALL, |
| 557 | &error_abort); |
John Snow | 111049a | 2016-11-08 01:50:38 -0500 | [diff] [blame] | 558 | |
| 559 | return &job->common; |
John Snow | d58d845 | 2015-04-17 19:49:58 -0400 | [diff] [blame] | 560 | |
| 561 | error: |
| 562 | if (sync_bitmap) { |
Vladimir Sementsov-Ogievskiy | 5deb6cb | 2019-09-16 17:19:09 +0300 | [diff] [blame] | 563 | bdrv_reclaim_dirty_bitmap(sync_bitmap, NULL); |
John Snow | d58d845 | 2015-04-17 19:49:58 -0400 | [diff] [blame] | 564 | } |
Vladimir Sementsov-Ogievskiy | 8ccf458 | 2019-10-17 17:21:22 +0300 | [diff] [blame] | 565 | if (backup_top) { |
Vladimir Sementsov-Ogievskiy | 00e30f0 | 2019-10-01 16:14:09 +0300 | [diff] [blame] | 566 | bdrv_backup_top_drop(backup_top); |
Kevin Wolf | 91ab688 | 2016-04-14 12:59:55 +0200 | [diff] [blame] | 567 | } |
John Snow | 111049a | 2016-11-08 01:50:38 -0500 | [diff] [blame] | 568 | |
| 569 | return NULL; |
Dietmar Maurer | 98d2c6f | 2013-06-24 17:13:11 +0200 | [diff] [blame] | 570 | } |