| /* |
| * Block driver for Parallels disk image format |
| * |
| * Copyright (c) 2007 Alex Beregszaszi |
| * Copyright (c) 2015 Denis V. Lunev <den@openvz.org> |
| * |
| * This code was originally based on comparing different disk images created |
| * by Parallels. Currently it is based on opened OpenVZ sources |
| * available at |
| * http://git.openvz.org/?p=ploop;a=summary |
| * |
| * Permission is hereby granted, free of charge, to any person obtaining a copy |
| * of this software and associated documentation files (the "Software"), to deal |
| * in the Software without restriction, including without limitation the rights |
| * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |
| * copies of the Software, and to permit persons to whom the Software is |
| * furnished to do so, subject to the following conditions: |
| * |
| * The above copyright notice and this permission notice shall be included in |
| * all copies or substantial portions of the Software. |
| * |
| * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
| * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
| * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL |
| * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
| * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
| * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN |
| * THE SOFTWARE. |
| */ |
| |
| #include "qemu/osdep.h" |
| #include "qemu/error-report.h" |
| #include "qapi/error.h" |
| #include "block/block_int.h" |
| #include "block/qdict.h" |
| #include "sysemu/block-backend.h" |
| #include "qemu/module.h" |
| #include "qemu/option.h" |
| #include "qapi/qmp/qdict.h" |
| #include "qapi/qobject-input-visitor.h" |
| #include "qapi/qapi-visit-block-core.h" |
| #include "qemu/bswap.h" |
| #include "qemu/bitmap.h" |
| #include "qemu/memalign.h" |
| #include "migration/blocker.h" |
| #include "parallels.h" |
| |
| /**************************************************************/ |
| |
| #define HEADER_MAGIC "WithoutFreeSpace" |
| #define HEADER_MAGIC2 "WithouFreSpacExt" |
| #define HEADER_VERSION 2 |
| #define HEADER_INUSE_MAGIC (0x746F6E59) |
| #define MAX_PARALLELS_IMAGE_FACTOR (1ull << 32) |
| |
| static QEnumLookup prealloc_mode_lookup = { |
| .array = (const char *const[]) { |
| "falloc", |
| "truncate", |
| }, |
| .size = PRL_PREALLOC_MODE__MAX |
| }; |
| |
| #define PARALLELS_OPT_PREALLOC_MODE "prealloc-mode" |
| #define PARALLELS_OPT_PREALLOC_SIZE "prealloc-size" |
| |
| static QemuOptsList parallels_runtime_opts = { |
| .name = "parallels", |
| .head = QTAILQ_HEAD_INITIALIZER(parallels_runtime_opts.head), |
| .desc = { |
| { |
| .name = PARALLELS_OPT_PREALLOC_SIZE, |
| .type = QEMU_OPT_SIZE, |
| .help = "Preallocation size on image expansion", |
| .def_value_str = "128M", |
| }, |
| { |
| .name = PARALLELS_OPT_PREALLOC_MODE, |
| .type = QEMU_OPT_STRING, |
| .help = "Preallocation mode on image expansion " |
| "(allowed values: falloc, truncate)", |
| .def_value_str = "falloc", |
| }, |
| { /* end of list */ }, |
| }, |
| }; |
| |
| static QemuOptsList parallels_create_opts = { |
| .name = "parallels-create-opts", |
| .head = QTAILQ_HEAD_INITIALIZER(parallels_create_opts.head), |
| .desc = { |
| { |
| .name = BLOCK_OPT_SIZE, |
| .type = QEMU_OPT_SIZE, |
| .help = "Virtual disk size", |
| }, |
| { |
| .name = BLOCK_OPT_CLUSTER_SIZE, |
| .type = QEMU_OPT_SIZE, |
| .help = "Parallels image cluster size", |
| .def_value_str = stringify(DEFAULT_CLUSTER_SIZE), |
| }, |
| { /* end of list */ } |
| } |
| }; |
| |
| |
| static int64_t bat2sect(BDRVParallelsState *s, uint32_t idx) |
| { |
| return (uint64_t)le32_to_cpu(s->bat_bitmap[idx]) * s->off_multiplier; |
| } |
| |
| static uint32_t bat_entry_off(uint32_t idx) |
| { |
| return sizeof(ParallelsHeader) + sizeof(uint32_t) * idx; |
| } |
| |
| static int64_t seek_to_sector(BDRVParallelsState *s, int64_t sector_num) |
| { |
| uint32_t index, offset; |
| |
| index = sector_num / s->tracks; |
| offset = sector_num % s->tracks; |
| |
| /* not allocated */ |
| if ((index >= s->bat_size) || (s->bat_bitmap[index] == 0)) { |
| return -1; |
| } |
| return bat2sect(s, index) + offset; |
| } |
| |
| static int cluster_remainder(BDRVParallelsState *s, int64_t sector_num, |
| int nb_sectors) |
| { |
| int ret = s->tracks - sector_num % s->tracks; |
| return MIN(nb_sectors, ret); |
| } |
| |
| static uint32_t host_cluster_index(BDRVParallelsState *s, int64_t off) |
| { |
| off -= s->data_start << BDRV_SECTOR_BITS; |
| return off / s->cluster_size; |
| } |
| |
| static int64_t block_status(BDRVParallelsState *s, int64_t sector_num, |
| int nb_sectors, int *pnum) |
| { |
| int64_t start_off = -2, prev_end_off = -2; |
| |
| *pnum = 0; |
| while (nb_sectors > 0 || start_off == -2) { |
| int64_t offset = seek_to_sector(s, sector_num); |
| int to_end; |
| |
| if (start_off == -2) { |
| start_off = offset; |
| prev_end_off = offset; |
| } else if (offset != prev_end_off) { |
| break; |
| } |
| |
| to_end = cluster_remainder(s, sector_num, nb_sectors); |
| nb_sectors -= to_end; |
| sector_num += to_end; |
| *pnum += to_end; |
| |
| if (offset > 0) { |
| prev_end_off += to_end; |
| } |
| } |
| return start_off; |
| } |
| |
| static void parallels_set_bat_entry(BDRVParallelsState *s, |
| uint32_t index, uint32_t offset) |
| { |
| s->bat_bitmap[index] = cpu_to_le32(offset); |
| bitmap_set(s->bat_dirty_bmap, bat_entry_off(index) / s->bat_dirty_block, 1); |
| } |
| |
| static int mark_used(BlockDriverState *bs, unsigned long *bitmap, |
| uint32_t bitmap_size, int64_t off, uint32_t count) |
| { |
| BDRVParallelsState *s = bs->opaque; |
| uint32_t cluster_index = host_cluster_index(s, off); |
| unsigned long next_used; |
| if (cluster_index + count > bitmap_size) { |
| return -E2BIG; |
| } |
| next_used = find_next_bit(bitmap, bitmap_size, cluster_index); |
| if (next_used < cluster_index + count) { |
| return -EBUSY; |
| } |
| bitmap_set(bitmap, cluster_index, count); |
| return 0; |
| } |
| |
| /* |
| * Collect used bitmap. The image can contain errors, we should fill the |
| * bitmap anyway, as much as we can. This information will be used for |
| * error resolution. |
| */ |
| static int GRAPH_RDLOCK parallels_fill_used_bitmap(BlockDriverState *bs) |
| { |
| BDRVParallelsState *s = bs->opaque; |
| int64_t payload_bytes; |
| uint32_t i; |
| int err = 0; |
| |
| payload_bytes = bdrv_getlength(bs->file->bs); |
| if (payload_bytes < 0) { |
| return payload_bytes; |
| } |
| payload_bytes -= s->data_start * BDRV_SECTOR_SIZE; |
| if (payload_bytes < 0) { |
| return -EINVAL; |
| } |
| |
| s->used_bmap_size = DIV_ROUND_UP(payload_bytes, s->cluster_size); |
| if (s->used_bmap_size == 0) { |
| return 0; |
| } |
| s->used_bmap = bitmap_try_new(s->used_bmap_size); |
| if (s->used_bmap == NULL) { |
| return -ENOMEM; |
| } |
| |
| for (i = 0; i < s->bat_size; i++) { |
| int err2; |
| int64_t host_off = bat2sect(s, i) << BDRV_SECTOR_BITS; |
| if (host_off == 0) { |
| continue; |
| } |
| |
| err2 = mark_used(bs, s->used_bmap, s->used_bmap_size, host_off, 1); |
| if (err2 < 0 && err == 0) { |
| err = err2; |
| } |
| } |
| return err; |
| } |
| |
| static void parallels_free_used_bitmap(BlockDriverState *bs) |
| { |
| BDRVParallelsState *s = bs->opaque; |
| s->used_bmap_size = 0; |
| g_free(s->used_bmap); |
| } |
| |
| static int64_t coroutine_fn GRAPH_RDLOCK |
| allocate_clusters(BlockDriverState *bs, int64_t sector_num, |
| int nb_sectors, int *pnum) |
| { |
| int ret = 0; |
| BDRVParallelsState *s = bs->opaque; |
| int64_t i, pos, idx, to_allocate, first_free, host_off; |
| |
| pos = block_status(s, sector_num, nb_sectors, pnum); |
| if (pos > 0) { |
| return pos; |
| } |
| |
| idx = sector_num / s->tracks; |
| to_allocate = DIV_ROUND_UP(sector_num + *pnum, s->tracks) - idx; |
| |
| /* |
| * This function is called only by parallels_co_writev(), which will never |
| * pass a sector_num at or beyond the end of the image (because the block |
| * layer never passes such a sector_num to that function). Therefore, idx |
| * is always below s->bat_size. |
| * block_status() will limit *pnum so that sector_num + *pnum will not |
| * exceed the image end. Therefore, idx + to_allocate cannot exceed |
| * s->bat_size. |
| * Note that s->bat_size is an unsigned int, therefore idx + to_allocate |
| * will always fit into a uint32_t. |
| */ |
| assert(idx < s->bat_size && idx + to_allocate <= s->bat_size); |
| |
| first_free = find_first_zero_bit(s->used_bmap, s->used_bmap_size); |
| if (first_free == s->used_bmap_size) { |
| uint32_t new_usedsize; |
| int64_t bytes = to_allocate * s->cluster_size; |
| bytes += s->prealloc_size * BDRV_SECTOR_SIZE; |
| |
| host_off = s->data_end * BDRV_SECTOR_SIZE; |
| |
| /* |
| * We require the expanded size to read back as zero. If the |
| * user permitted truncation, we try that; but if it fails, we |
| * force the safer-but-slower fallocate. |
| */ |
| if (s->prealloc_mode == PRL_PREALLOC_MODE_TRUNCATE) { |
| ret = bdrv_co_truncate(bs->file, host_off + bytes, |
| false, PREALLOC_MODE_OFF, |
| BDRV_REQ_ZERO_WRITE, NULL); |
| if (ret == -ENOTSUP) { |
| s->prealloc_mode = PRL_PREALLOC_MODE_FALLOCATE; |
| } |
| } |
| if (s->prealloc_mode == PRL_PREALLOC_MODE_FALLOCATE) { |
| ret = bdrv_co_pwrite_zeroes(bs->file, host_off, bytes, 0); |
| } |
| if (ret < 0) { |
| return ret; |
| } |
| |
| new_usedsize = s->used_bmap_size + bytes / s->cluster_size; |
| s->used_bmap = bitmap_zero_extend(s->used_bmap, s->used_bmap_size, |
| new_usedsize); |
| s->used_bmap_size = new_usedsize; |
| } else { |
| int64_t next_used; |
| next_used = find_next_bit(s->used_bmap, s->used_bmap_size, first_free); |
| |
| /* Not enough continuous clusters in the middle, adjust the size */ |
| if (next_used - first_free < to_allocate) { |
| to_allocate = next_used - first_free; |
| *pnum = (idx + to_allocate) * s->tracks - sector_num; |
| } |
| |
| host_off = s->data_start * BDRV_SECTOR_SIZE; |
| host_off += first_free * s->cluster_size; |
| |
| /* |
| * No need to preallocate if we are using tail area from the above |
| * branch. In the other case we are likely re-using hole. Preallocate |
| * the space if required by the prealloc_mode. |
| */ |
| if (s->prealloc_mode == PRL_PREALLOC_MODE_FALLOCATE && |
| host_off < s->data_end * BDRV_SECTOR_SIZE) { |
| ret = bdrv_co_pwrite_zeroes(bs->file, host_off, |
| s->cluster_size * to_allocate, 0); |
| if (ret < 0) { |
| return ret; |
| } |
| } |
| } |
| |
| /* |
| * Try to read from backing to fill empty clusters |
| * FIXME: 1. previous write_zeroes may be redundant |
| * 2. most of data we read from backing will be rewritten by |
| * parallels_co_writev. On aligned-to-cluster write we do not need |
| * this read at all. |
| * 3. it would be good to combine write of data from backing and new |
| * data into one write call. |
| */ |
| if (bs->backing) { |
| int64_t nb_cow_sectors = to_allocate * s->tracks; |
| int64_t nb_cow_bytes = nb_cow_sectors << BDRV_SECTOR_BITS; |
| void *buf = qemu_blockalign(bs, nb_cow_bytes); |
| |
| ret = bdrv_co_pread(bs->backing, idx * s->tracks * BDRV_SECTOR_SIZE, |
| nb_cow_bytes, buf, 0); |
| if (ret < 0) { |
| qemu_vfree(buf); |
| return ret; |
| } |
| |
| ret = bdrv_co_pwrite(bs->file, s->data_end * BDRV_SECTOR_SIZE, |
| nb_cow_bytes, buf, 0); |
| qemu_vfree(buf); |
| if (ret < 0) { |
| return ret; |
| } |
| } |
| |
| ret = mark_used(bs, s->used_bmap, s->used_bmap_size, host_off, to_allocate); |
| if (ret < 0) { |
| /* Image consistency is broken. Alarm! */ |
| return ret; |
| } |
| for (i = 0; i < to_allocate; i++) { |
| parallels_set_bat_entry(s, idx + i, |
| host_off / BDRV_SECTOR_SIZE / s->off_multiplier); |
| host_off += s->cluster_size; |
| } |
| if (host_off > s->data_end * BDRV_SECTOR_SIZE) { |
| s->data_end = host_off / BDRV_SECTOR_SIZE; |
| } |
| |
| return bat2sect(s, idx) + sector_num % s->tracks; |
| } |
| |
| |
| static int coroutine_fn GRAPH_RDLOCK |
| parallels_co_flush_to_os(BlockDriverState *bs) |
| { |
| BDRVParallelsState *s = bs->opaque; |
| unsigned long size = DIV_ROUND_UP(s->header_size, s->bat_dirty_block); |
| unsigned long bit; |
| |
| qemu_co_mutex_lock(&s->lock); |
| |
| bit = find_first_bit(s->bat_dirty_bmap, size); |
| while (bit < size) { |
| uint32_t off = bit * s->bat_dirty_block; |
| uint32_t to_write = s->bat_dirty_block; |
| int ret; |
| |
| if (off + to_write > s->header_size) { |
| to_write = s->header_size - off; |
| } |
| ret = bdrv_co_pwrite(bs->file, off, to_write, |
| (uint8_t *)s->header + off, 0); |
| if (ret < 0) { |
| qemu_co_mutex_unlock(&s->lock); |
| return ret; |
| } |
| bit = find_next_bit(s->bat_dirty_bmap, size, bit + 1); |
| } |
| bitmap_zero(s->bat_dirty_bmap, size); |
| |
| qemu_co_mutex_unlock(&s->lock); |
| return 0; |
| } |
| |
| static int coroutine_fn GRAPH_RDLOCK |
| parallels_co_block_status(BlockDriverState *bs, bool want_zero, int64_t offset, |
| int64_t bytes, int64_t *pnum, int64_t *map, |
| BlockDriverState **file) |
| { |
| BDRVParallelsState *s = bs->opaque; |
| int count; |
| |
| assert(QEMU_IS_ALIGNED(offset | bytes, BDRV_SECTOR_SIZE)); |
| qemu_co_mutex_lock(&s->lock); |
| offset = block_status(s, offset >> BDRV_SECTOR_BITS, |
| bytes >> BDRV_SECTOR_BITS, &count); |
| qemu_co_mutex_unlock(&s->lock); |
| |
| *pnum = count * BDRV_SECTOR_SIZE; |
| if (offset < 0) { |
| return 0; |
| } |
| |
| *map = offset * BDRV_SECTOR_SIZE; |
| *file = bs->file->bs; |
| return BDRV_BLOCK_DATA | BDRV_BLOCK_OFFSET_VALID; |
| } |
| |
| static int coroutine_fn GRAPH_RDLOCK |
| parallels_co_writev(BlockDriverState *bs, int64_t sector_num, int nb_sectors, |
| QEMUIOVector *qiov, int flags) |
| { |
| BDRVParallelsState *s = bs->opaque; |
| uint64_t bytes_done = 0; |
| QEMUIOVector hd_qiov; |
| int ret = 0; |
| |
| qemu_iovec_init(&hd_qiov, qiov->niov); |
| |
| while (nb_sectors > 0) { |
| int64_t position; |
| int n, nbytes; |
| |
| qemu_co_mutex_lock(&s->lock); |
| position = allocate_clusters(bs, sector_num, nb_sectors, &n); |
| qemu_co_mutex_unlock(&s->lock); |
| if (position < 0) { |
| ret = (int)position; |
| break; |
| } |
| |
| nbytes = n << BDRV_SECTOR_BITS; |
| |
| qemu_iovec_reset(&hd_qiov); |
| qemu_iovec_concat(&hd_qiov, qiov, bytes_done, nbytes); |
| |
| ret = bdrv_co_pwritev(bs->file, position * BDRV_SECTOR_SIZE, nbytes, |
| &hd_qiov, 0); |
| if (ret < 0) { |
| break; |
| } |
| |
| nb_sectors -= n; |
| sector_num += n; |
| bytes_done += nbytes; |
| } |
| |
| qemu_iovec_destroy(&hd_qiov); |
| return ret; |
| } |
| |
| static int coroutine_fn GRAPH_RDLOCK |
| parallels_co_readv(BlockDriverState *bs, int64_t sector_num, int nb_sectors, |
| QEMUIOVector *qiov) |
| { |
| BDRVParallelsState *s = bs->opaque; |
| uint64_t bytes_done = 0; |
| QEMUIOVector hd_qiov; |
| int ret = 0; |
| |
| qemu_iovec_init(&hd_qiov, qiov->niov); |
| |
| while (nb_sectors > 0) { |
| int64_t position; |
| int n, nbytes; |
| |
| qemu_co_mutex_lock(&s->lock); |
| position = block_status(s, sector_num, nb_sectors, &n); |
| qemu_co_mutex_unlock(&s->lock); |
| |
| nbytes = n << BDRV_SECTOR_BITS; |
| |
| qemu_iovec_reset(&hd_qiov); |
| qemu_iovec_concat(&hd_qiov, qiov, bytes_done, nbytes); |
| |
| if (position < 0) { |
| if (bs->backing) { |
| ret = bdrv_co_preadv(bs->backing, sector_num * BDRV_SECTOR_SIZE, |
| nbytes, &hd_qiov, 0); |
| if (ret < 0) { |
| break; |
| } |
| } else { |
| qemu_iovec_memset(&hd_qiov, 0, 0, nbytes); |
| } |
| } else { |
| ret = bdrv_co_preadv(bs->file, position * BDRV_SECTOR_SIZE, nbytes, |
| &hd_qiov, 0); |
| if (ret < 0) { |
| break; |
| } |
| } |
| |
| nb_sectors -= n; |
| sector_num += n; |
| bytes_done += nbytes; |
| } |
| |
| qemu_iovec_destroy(&hd_qiov); |
| return ret; |
| } |
| |
| |
| static int coroutine_fn GRAPH_RDLOCK |
| parallels_co_pdiscard(BlockDriverState *bs, int64_t offset, int64_t bytes) |
| { |
| int ret = 0; |
| uint32_t cluster, count; |
| BDRVParallelsState *s = bs->opaque; |
| |
| /* |
| * The image does not support ZERO mark inside the BAT, which means that |
| * stale data could be exposed from the backing file. |
| */ |
| if (bs->backing) { |
| return -ENOTSUP; |
| } |
| |
| if (!QEMU_IS_ALIGNED(offset, s->cluster_size)) { |
| return -ENOTSUP; |
| } else if (!QEMU_IS_ALIGNED(bytes, s->cluster_size)) { |
| return -ENOTSUP; |
| } |
| |
| cluster = offset / s->cluster_size; |
| count = bytes / s->cluster_size; |
| |
| qemu_co_mutex_lock(&s->lock); |
| for (; count > 0; cluster++, count--) { |
| int64_t host_off = bat2sect(s, cluster) << BDRV_SECTOR_BITS; |
| if (host_off == 0) { |
| continue; |
| } |
| |
| ret = bdrv_co_pdiscard(bs->file, host_off, s->cluster_size); |
| if (ret < 0) { |
| goto done; |
| } |
| |
| parallels_set_bat_entry(s, cluster, 0); |
| bitmap_clear(s->used_bmap, host_cluster_index(s, host_off), 1); |
| } |
| done: |
| qemu_co_mutex_unlock(&s->lock); |
| return ret; |
| } |
| |
| static int coroutine_fn GRAPH_RDLOCK |
| parallels_co_pwrite_zeroes(BlockDriverState *bs, int64_t offset, int64_t bytes, |
| BdrvRequestFlags flags) |
| { |
| /* |
| * The zero flag is missed in the Parallels format specification. We can |
| * resort to discard if we have no backing file (this condition is checked |
| * inside parallels_co_pdiscard(). |
| */ |
| return parallels_co_pdiscard(bs, offset, bytes); |
| } |
| |
| |
| static void parallels_check_unclean(BlockDriverState *bs, |
| BdrvCheckResult *res, |
| BdrvCheckMode fix) |
| { |
| BDRVParallelsState *s = bs->opaque; |
| |
| if (!s->header_unclean) { |
| return; |
| } |
| |
| fprintf(stderr, "%s image was not closed correctly\n", |
| fix & BDRV_FIX_ERRORS ? "Repairing" : "ERROR"); |
| res->corruptions++; |
| if (fix & BDRV_FIX_ERRORS) { |
| /* parallels_close will do the job right */ |
| res->corruptions_fixed++; |
| s->header_unclean = false; |
| } |
| } |
| |
| /* |
| * Returns true if data_off is correct, otherwise false. In both cases |
| * correct_offset is set to the proper value. |
| */ |
| static bool parallels_test_data_off(BDRVParallelsState *s, |
| int64_t file_nb_sectors, |
| uint32_t *correct_offset) |
| { |
| uint32_t data_off, min_off; |
| bool old_magic; |
| |
| /* |
| * There are two slightly different image formats: with "WithoutFreeSpace" |
| * or "WithouFreSpacExt" magic words. Call the first one as "old magic". |
| * In such images data_off field can be zero. In this case the offset is |
| * calculated as the end of BAT table plus some padding to ensure sector |
| * size alignment. |
| */ |
| old_magic = !memcmp(s->header->magic, HEADER_MAGIC, 16); |
| |
| min_off = DIV_ROUND_UP(bat_entry_off(s->bat_size), BDRV_SECTOR_SIZE); |
| if (!old_magic) { |
| min_off = ROUND_UP(min_off, s->cluster_size / BDRV_SECTOR_SIZE); |
| } |
| |
| if (correct_offset) { |
| *correct_offset = min_off; |
| } |
| |
| data_off = le32_to_cpu(s->header->data_off); |
| if (data_off == 0 && old_magic) { |
| return true; |
| } |
| |
| if (data_off < min_off || data_off > file_nb_sectors) { |
| return false; |
| } |
| |
| if (correct_offset) { |
| *correct_offset = data_off; |
| } |
| |
| return true; |
| } |
| |
| static int coroutine_fn GRAPH_RDLOCK |
| parallels_check_data_off(BlockDriverState *bs, BdrvCheckResult *res, |
| BdrvCheckMode fix) |
| { |
| BDRVParallelsState *s = bs->opaque; |
| int64_t file_size; |
| uint32_t data_off; |
| |
| file_size = bdrv_co_nb_sectors(bs->file->bs); |
| if (file_size < 0) { |
| res->check_errors++; |
| return file_size; |
| } |
| |
| if (parallels_test_data_off(s, file_size, &data_off)) { |
| return 0; |
| } |
| |
| res->corruptions++; |
| if (fix & BDRV_FIX_ERRORS) { |
| int err; |
| s->header->data_off = cpu_to_le32(data_off); |
| s->data_start = data_off; |
| |
| parallels_free_used_bitmap(bs); |
| err = parallels_fill_used_bitmap(bs); |
| if (err == -ENOMEM) { |
| res->check_errors++; |
| return err; |
| } |
| |
| res->corruptions_fixed++; |
| } |
| |
| fprintf(stderr, "%s data_off field has incorrect value\n", |
| fix & BDRV_FIX_ERRORS ? "Repairing" : "ERROR"); |
| |
| return 0; |
| } |
| |
| static int coroutine_fn GRAPH_RDLOCK |
| parallels_check_outside_image(BlockDriverState *bs, BdrvCheckResult *res, |
| BdrvCheckMode fix) |
| { |
| BDRVParallelsState *s = bs->opaque; |
| uint32_t i; |
| int64_t off, high_off, size; |
| |
| size = bdrv_co_getlength(bs->file->bs); |
| if (size < 0) { |
| res->check_errors++; |
| return size; |
| } |
| |
| high_off = 0; |
| for (i = 0; i < s->bat_size; i++) { |
| off = bat2sect(s, i) << BDRV_SECTOR_BITS; |
| if (off + s->cluster_size > size) { |
| fprintf(stderr, "%s cluster %u is outside image\n", |
| fix & BDRV_FIX_ERRORS ? "Repairing" : "ERROR", i); |
| res->corruptions++; |
| if (fix & BDRV_FIX_ERRORS) { |
| parallels_set_bat_entry(s, i, 0); |
| res->corruptions_fixed++; |
| } |
| continue; |
| } |
| if (high_off < off) { |
| high_off = off; |
| } |
| } |
| |
| if (high_off == 0) { |
| res->image_end_offset = s->data_end << BDRV_SECTOR_BITS; |
| } else { |
| res->image_end_offset = high_off + s->cluster_size; |
| s->data_end = res->image_end_offset >> BDRV_SECTOR_BITS; |
| } |
| |
| return 0; |
| } |
| |
| static int coroutine_fn GRAPH_RDLOCK |
| parallels_check_leak(BlockDriverState *bs, BdrvCheckResult *res, |
| BdrvCheckMode fix, bool explicit) |
| { |
| BDRVParallelsState *s = bs->opaque; |
| int64_t size; |
| int ret; |
| |
| size = bdrv_co_getlength(bs->file->bs); |
| if (size < 0) { |
| res->check_errors++; |
| return size; |
| } |
| |
| if (size > res->image_end_offset) { |
| int64_t count; |
| count = DIV_ROUND_UP(size - res->image_end_offset, s->cluster_size); |
| if (explicit) { |
| fprintf(stderr, |
| "%s space leaked at the end of the image %" PRId64 "\n", |
| fix & BDRV_FIX_LEAKS ? "Repairing" : "ERROR", |
| size - res->image_end_offset); |
| res->leaks += count; |
| } |
| if (fix & BDRV_FIX_LEAKS) { |
| Error *local_err = NULL; |
| |
| /* |
| * In order to really repair the image, we must shrink it. |
| * That means we have to pass exact=true. |
| */ |
| ret = bdrv_co_truncate(bs->file, res->image_end_offset, true, |
| PREALLOC_MODE_OFF, 0, &local_err); |
| if (ret < 0) { |
| error_report_err(local_err); |
| res->check_errors++; |
| return ret; |
| } |
| if (explicit) { |
| res->leaks_fixed += count; |
| } |
| } |
| } |
| |
| return 0; |
| } |
| |
| static int coroutine_fn GRAPH_RDLOCK |
| parallels_check_duplicate(BlockDriverState *bs, BdrvCheckResult *res, |
| BdrvCheckMode fix) |
| { |
| BDRVParallelsState *s = bs->opaque; |
| int64_t host_off, host_sector, guest_sector; |
| unsigned long *bitmap; |
| uint32_t i, bitmap_size, bat_entry; |
| int n, ret = 0; |
| uint64_t *buf = NULL; |
| bool fixed = false; |
| |
| /* |
| * Create a bitmap of used clusters. |
| * If a bit is set, there is a BAT entry pointing to this cluster. |
| * Loop through the BAT entries, check bits relevant to an entry offset. |
| * If bit is set, this entry is duplicated. Otherwise set the bit. |
| * |
| * We shouldn't worry about newly allocated clusters outside the image |
| * because they are created higher then any existing cluster pointed by |
| * a BAT entry. |
| */ |
| bitmap_size = host_cluster_index(s, res->image_end_offset); |
| if (bitmap_size == 0) { |
| return 0; |
| } |
| if (res->image_end_offset % s->cluster_size) { |
| /* A not aligned image end leads to a bitmap shorter by 1 */ |
| bitmap_size++; |
| } |
| |
| bitmap = bitmap_new(bitmap_size); |
| |
| buf = qemu_blockalign(bs, s->cluster_size); |
| |
| for (i = 0; i < s->bat_size; i++) { |
| host_off = bat2sect(s, i) << BDRV_SECTOR_BITS; |
| if (host_off == 0) { |
| continue; |
| } |
| |
| ret = mark_used(bs, bitmap, bitmap_size, host_off, 1); |
| assert(ret != -E2BIG); |
| if (ret == 0) { |
| continue; |
| } |
| |
| /* this cluster duplicates another one */ |
| fprintf(stderr, "%s duplicate offset in BAT entry %u\n", |
| fix & BDRV_FIX_ERRORS ? "Repairing" : "ERROR", i); |
| |
| res->corruptions++; |
| |
| if (!(fix & BDRV_FIX_ERRORS)) { |
| continue; |
| } |
| |
| /* |
| * Reset the entry and allocate a new cluster |
| * for the relevant guest offset. In this way we let |
| * the lower layer to place the new cluster properly. |
| * Copy the original cluster to the allocated one. |
| * But before save the old offset value for repairing |
| * if we have an error. |
| */ |
| bat_entry = s->bat_bitmap[i]; |
| parallels_set_bat_entry(s, i, 0); |
| |
| ret = bdrv_co_pread(bs->file, host_off, s->cluster_size, buf, 0); |
| if (ret < 0) { |
| res->check_errors++; |
| goto out_repair_bat; |
| } |
| |
| guest_sector = (i * (int64_t)s->cluster_size) >> BDRV_SECTOR_BITS; |
| host_sector = allocate_clusters(bs, guest_sector, s->tracks, &n); |
| if (host_sector < 0) { |
| res->check_errors++; |
| goto out_repair_bat; |
| } |
| host_off = host_sector << BDRV_SECTOR_BITS; |
| |
| ret = bdrv_co_pwrite(bs->file, host_off, s->cluster_size, buf, 0); |
| if (ret < 0) { |
| res->check_errors++; |
| goto out_repair_bat; |
| } |
| |
| if (host_off + s->cluster_size > res->image_end_offset) { |
| res->image_end_offset = host_off + s->cluster_size; |
| } |
| |
| /* |
| * In the future allocate_cluster() will reuse holed offsets |
| * inside the image. Keep the used clusters bitmap content |
| * consistent for the new allocated clusters too. |
| * |
| * Note, clusters allocated outside the current image are not |
| * considered, and the bitmap size doesn't change. This specifically |
| * means that -E2BIG is OK. |
| */ |
| ret = mark_used(bs, bitmap, bitmap_size, host_off, 1); |
| if (ret == -EBUSY) { |
| res->check_errors++; |
| goto out_repair_bat; |
| } |
| |
| fixed = true; |
| res->corruptions_fixed++; |
| |
| } |
| |
| if (fixed) { |
| /* |
| * When new clusters are allocated, the file size increases by |
| * 128 Mb. We need to truncate the file to the right size. Let |
| * the leak fix code make its job without res changing. |
| */ |
| ret = parallels_check_leak(bs, res, fix, false); |
| } |
| |
| out_free: |
| g_free(buf); |
| g_free(bitmap); |
| return ret; |
| /* |
| * We can get here only from places where index and old_offset have |
| * meaningful values. |
| */ |
| out_repair_bat: |
| s->bat_bitmap[i] = bat_entry; |
| goto out_free; |
| } |
| |
| static void parallels_collect_statistics(BlockDriverState *bs, |
| BdrvCheckResult *res, |
| BdrvCheckMode fix) |
| { |
| BDRVParallelsState *s = bs->opaque; |
| int64_t off, prev_off; |
| uint32_t i; |
| |
| res->bfi.total_clusters = s->bat_size; |
| res->bfi.compressed_clusters = 0; /* compression is not supported */ |
| |
| prev_off = 0; |
| for (i = 0; i < s->bat_size; i++) { |
| off = bat2sect(s, i) << BDRV_SECTOR_BITS; |
| /* |
| * If BDRV_FIX_ERRORS is not set, out-of-image BAT entries were not |
| * fixed. Skip not allocated and out-of-image BAT entries. |
| */ |
| if (off == 0 || off + s->cluster_size > res->image_end_offset) { |
| prev_off = 0; |
| continue; |
| } |
| |
| if (prev_off != 0 && (prev_off + s->cluster_size) != off) { |
| res->bfi.fragmented_clusters++; |
| } |
| prev_off = off; |
| res->bfi.allocated_clusters++; |
| } |
| } |
| |
| static int coroutine_fn GRAPH_RDLOCK |
| parallels_co_check(BlockDriverState *bs, BdrvCheckResult *res, |
| BdrvCheckMode fix) |
| { |
| BDRVParallelsState *s = bs->opaque; |
| int ret; |
| |
| WITH_QEMU_LOCK_GUARD(&s->lock) { |
| parallels_check_unclean(bs, res, fix); |
| |
| ret = parallels_check_data_off(bs, res, fix); |
| if (ret < 0) { |
| return ret; |
| } |
| |
| ret = parallels_check_outside_image(bs, res, fix); |
| if (ret < 0) { |
| return ret; |
| } |
| |
| ret = parallels_check_leak(bs, res, fix, true); |
| if (ret < 0) { |
| return ret; |
| } |
| |
| ret = parallels_check_duplicate(bs, res, fix); |
| if (ret < 0) { |
| return ret; |
| } |
| |
| parallels_collect_statistics(bs, res, fix); |
| } |
| |
| ret = bdrv_co_flush(bs); |
| if (ret < 0) { |
| res->check_errors++; |
| } |
| |
| return ret; |
| } |
| |
| |
| static int coroutine_fn GRAPH_UNLOCKED |
| parallels_co_create(BlockdevCreateOptions* opts, Error **errp) |
| { |
| BlockdevCreateOptionsParallels *parallels_opts; |
| BlockDriverState *bs; |
| BlockBackend *blk; |
| int64_t total_size, cl_size; |
| uint32_t bat_entries, bat_sectors; |
| ParallelsHeader header; |
| uint8_t tmp[BDRV_SECTOR_SIZE]; |
| int ret; |
| |
| assert(opts->driver == BLOCKDEV_DRIVER_PARALLELS); |
| parallels_opts = &opts->u.parallels; |
| |
| /* Sanity checks */ |
| total_size = parallels_opts->size; |
| |
| if (parallels_opts->has_cluster_size) { |
| cl_size = parallels_opts->cluster_size; |
| } else { |
| cl_size = DEFAULT_CLUSTER_SIZE; |
| } |
| |
| /* XXX What is the real limit here? This is an insanely large maximum. */ |
| if (cl_size >= INT64_MAX / MAX_PARALLELS_IMAGE_FACTOR) { |
| error_setg(errp, "Cluster size is too large"); |
| return -EINVAL; |
| } |
| if (total_size >= MAX_PARALLELS_IMAGE_FACTOR * cl_size) { |
| error_setg(errp, "Image size is too large for this cluster size"); |
| return -E2BIG; |
| } |
| |
| if (!QEMU_IS_ALIGNED(total_size, BDRV_SECTOR_SIZE)) { |
| error_setg(errp, "Image size must be a multiple of 512 bytes"); |
| return -EINVAL; |
| } |
| |
| if (!QEMU_IS_ALIGNED(cl_size, BDRV_SECTOR_SIZE)) { |
| error_setg(errp, "Cluster size must be a multiple of 512 bytes"); |
| return -EINVAL; |
| } |
| |
| /* Create BlockBackend to write to the image */ |
| bs = bdrv_co_open_blockdev_ref(parallels_opts->file, errp); |
| if (bs == NULL) { |
| return -EIO; |
| } |
| |
| blk = blk_co_new_with_bs(bs, BLK_PERM_WRITE | BLK_PERM_RESIZE, BLK_PERM_ALL, |
| errp); |
| if (!blk) { |
| ret = -EPERM; |
| goto out; |
| } |
| blk_set_allow_write_beyond_eof(blk, true); |
| |
| /* Create image format */ |
| bat_entries = DIV_ROUND_UP(total_size, cl_size); |
| bat_sectors = DIV_ROUND_UP(bat_entry_off(bat_entries), cl_size); |
| bat_sectors = (bat_sectors * cl_size) >> BDRV_SECTOR_BITS; |
| |
| memset(&header, 0, sizeof(header)); |
| memcpy(header.magic, HEADER_MAGIC2, sizeof(header.magic)); |
| header.version = cpu_to_le32(HEADER_VERSION); |
| /* don't care much about geometry, it is not used on image level */ |
| header.heads = cpu_to_le32(HEADS_NUMBER); |
| header.cylinders = cpu_to_le32(total_size / BDRV_SECTOR_SIZE |
| / HEADS_NUMBER / SEC_IN_CYL); |
| header.tracks = cpu_to_le32(cl_size >> BDRV_SECTOR_BITS); |
| header.bat_entries = cpu_to_le32(bat_entries); |
| header.nb_sectors = cpu_to_le64(DIV_ROUND_UP(total_size, BDRV_SECTOR_SIZE)); |
| header.data_off = cpu_to_le32(bat_sectors); |
| |
| /* write all the data */ |
| memset(tmp, 0, sizeof(tmp)); |
| memcpy(tmp, &header, sizeof(header)); |
| |
| ret = blk_co_pwrite(blk, 0, BDRV_SECTOR_SIZE, tmp, 0); |
| if (ret < 0) { |
| goto exit; |
| } |
| ret = blk_co_pwrite_zeroes(blk, BDRV_SECTOR_SIZE, |
| (bat_sectors - 1) << BDRV_SECTOR_BITS, 0); |
| if (ret < 0) { |
| goto exit; |
| } |
| |
| ret = 0; |
| out: |
| blk_co_unref(blk); |
| bdrv_co_unref(bs); |
| return ret; |
| |
| exit: |
| error_setg_errno(errp, -ret, "Failed to create Parallels image"); |
| goto out; |
| } |
| |
| static int coroutine_fn GRAPH_UNLOCKED |
| parallels_co_create_opts(BlockDriver *drv, const char *filename, |
| QemuOpts *opts, Error **errp) |
| { |
| BlockdevCreateOptions *create_options = NULL; |
| BlockDriverState *bs = NULL; |
| QDict *qdict; |
| Visitor *v; |
| int ret; |
| |
| static const QDictRenames opt_renames[] = { |
| { BLOCK_OPT_CLUSTER_SIZE, "cluster-size" }, |
| { NULL, NULL }, |
| }; |
| |
| /* Parse options and convert legacy syntax */ |
| qdict = qemu_opts_to_qdict_filtered(opts, NULL, ¶llels_create_opts, |
| true); |
| |
| if (!qdict_rename_keys(qdict, opt_renames, errp)) { |
| ret = -EINVAL; |
| goto done; |
| } |
| |
| /* Create and open the file (protocol layer) */ |
| ret = bdrv_co_create_file(filename, opts, errp); |
| if (ret < 0) { |
| goto done; |
| } |
| |
| bs = bdrv_co_open(filename, NULL, NULL, |
| BDRV_O_RDWR | BDRV_O_RESIZE | BDRV_O_PROTOCOL, errp); |
| if (bs == NULL) { |
| ret = -EIO; |
| goto done; |
| } |
| |
| /* Now get the QAPI type BlockdevCreateOptions */ |
| qdict_put_str(qdict, "driver", "parallels"); |
| qdict_put_str(qdict, "file", bs->node_name); |
| |
| v = qobject_input_visitor_new_flat_confused(qdict, errp); |
| if (!v) { |
| ret = -EINVAL; |
| goto done; |
| } |
| |
| visit_type_BlockdevCreateOptions(v, NULL, &create_options, errp); |
| visit_free(v); |
| if (!create_options) { |
| ret = -EINVAL; |
| goto done; |
| } |
| |
| /* Silently round up sizes */ |
| create_options->u.parallels.size = |
| ROUND_UP(create_options->u.parallels.size, BDRV_SECTOR_SIZE); |
| create_options->u.parallels.cluster_size = |
| ROUND_UP(create_options->u.parallels.cluster_size, BDRV_SECTOR_SIZE); |
| |
| /* Create the Parallels image (format layer) */ |
| ret = parallels_co_create(create_options, errp); |
| if (ret < 0) { |
| goto done; |
| } |
| ret = 0; |
| |
| done: |
| qobject_unref(qdict); |
| bdrv_co_unref(bs); |
| qapi_free_BlockdevCreateOptions(create_options); |
| return ret; |
| } |
| |
| |
| static int parallels_probe(const uint8_t *buf, int buf_size, |
| const char *filename) |
| { |
| const ParallelsHeader *ph = (const void *)buf; |
| |
| if (buf_size < sizeof(ParallelsHeader)) { |
| return 0; |
| } |
| |
| if ((!memcmp(ph->magic, HEADER_MAGIC, 16) || |
| !memcmp(ph->magic, HEADER_MAGIC2, 16)) && |
| (le32_to_cpu(ph->version) == HEADER_VERSION)) { |
| return 100; |
| } |
| |
| return 0; |
| } |
| |
| static int GRAPH_RDLOCK parallels_update_header(BlockDriverState *bs) |
| { |
| BDRVParallelsState *s = bs->opaque; |
| unsigned size = MAX(bdrv_opt_mem_align(bs->file->bs), |
| sizeof(ParallelsHeader)); |
| |
| if (size > s->header_size) { |
| size = s->header_size; |
| } |
| return bdrv_pwrite_sync(bs->file, 0, size, s->header, 0); |
| } |
| |
| |
| static int parallels_opts_prealloc(BlockDriverState *bs, QDict *options, |
| Error **errp) |
| { |
| int err; |
| char *buf; |
| int64_t bytes; |
| BDRVParallelsState *s = bs->opaque; |
| Error *local_err = NULL; |
| QemuOpts *opts = qemu_opts_create(¶llels_runtime_opts, NULL, 0, errp); |
| if (!opts) { |
| return -ENOMEM; |
| } |
| |
| err = -EINVAL; |
| if (!qemu_opts_absorb_qdict(opts, options, errp)) { |
| goto done; |
| } |
| |
| bytes = qemu_opt_get_size_del(opts, PARALLELS_OPT_PREALLOC_SIZE, 0); |
| s->prealloc_size = bytes >> BDRV_SECTOR_BITS; |
| buf = qemu_opt_get_del(opts, PARALLELS_OPT_PREALLOC_MODE); |
| /* prealloc_mode can be downgraded later during allocate_clusters */ |
| s->prealloc_mode = qapi_enum_parse(&prealloc_mode_lookup, buf, |
| PRL_PREALLOC_MODE_FALLOCATE, |
| &local_err); |
| g_free(buf); |
| if (local_err != NULL) { |
| error_propagate(errp, local_err); |
| goto done; |
| } |
| err = 0; |
| |
| done: |
| qemu_opts_del(opts); |
| return err; |
| } |
| |
| static int parallels_open(BlockDriverState *bs, QDict *options, int flags, |
| Error **errp) |
| { |
| BDRVParallelsState *s = bs->opaque; |
| ParallelsHeader ph; |
| int ret, size, i; |
| int64_t file_nb_sectors, sector; |
| uint32_t data_start; |
| bool need_check = false; |
| |
| ret = parallels_opts_prealloc(bs, options, errp); |
| if (ret < 0) { |
| return ret; |
| } |
| |
| ret = bdrv_open_file_child(NULL, options, "file", bs, errp); |
| if (ret < 0) { |
| return ret; |
| } |
| |
| GRAPH_RDLOCK_GUARD_MAINLOOP(); |
| |
| file_nb_sectors = bdrv_nb_sectors(bs->file->bs); |
| if (file_nb_sectors < 0) { |
| return -EINVAL; |
| } |
| |
| ret = bdrv_pread(bs->file, 0, sizeof(ph), &ph, 0); |
| if (ret < 0) { |
| return ret; |
| } |
| |
| bs->total_sectors = le64_to_cpu(ph.nb_sectors); |
| |
| if (le32_to_cpu(ph.version) != HEADER_VERSION) { |
| goto fail_format; |
| } |
| if (!memcmp(ph.magic, HEADER_MAGIC, 16)) { |
| s->off_multiplier = 1; |
| bs->total_sectors = 0xffffffff & bs->total_sectors; |
| } else if (!memcmp(ph.magic, HEADER_MAGIC2, 16)) { |
| s->off_multiplier = le32_to_cpu(ph.tracks); |
| } else { |
| goto fail_format; |
| } |
| |
| s->tracks = le32_to_cpu(ph.tracks); |
| if (s->tracks == 0) { |
| error_setg(errp, "Invalid image: Zero sectors per track"); |
| return -EINVAL; |
| } |
| if (s->tracks > INT32_MAX/513) { |
| error_setg(errp, "Invalid image: Too big cluster"); |
| return -EFBIG; |
| } |
| s->prealloc_size = MAX(s->tracks, s->prealloc_size); |
| s->cluster_size = s->tracks << BDRV_SECTOR_BITS; |
| |
| s->bat_size = le32_to_cpu(ph.bat_entries); |
| if (s->bat_size > INT_MAX / sizeof(uint32_t)) { |
| error_setg(errp, "Catalog too large"); |
| return -EFBIG; |
| } |
| |
| size = bat_entry_off(s->bat_size); |
| s->header_size = ROUND_UP(size, bdrv_opt_mem_align(bs->file->bs)); |
| s->header = qemu_try_blockalign(bs->file->bs, s->header_size); |
| if (s->header == NULL) { |
| return -ENOMEM; |
| } |
| |
| ret = bdrv_pread(bs->file, 0, s->header_size, s->header, 0); |
| if (ret < 0) { |
| goto fail; |
| } |
| s->bat_bitmap = (uint32_t *)(s->header + 1); |
| |
| if (le32_to_cpu(ph.inuse) == HEADER_INUSE_MAGIC) { |
| need_check = s->header_unclean = true; |
| } |
| |
| { |
| bool ok = parallels_test_data_off(s, file_nb_sectors, &data_start); |
| need_check = need_check || !ok; |
| } |
| |
| s->data_start = data_start; |
| s->data_end = s->data_start; |
| if (s->data_end < (s->header_size >> BDRV_SECTOR_BITS)) { |
| /* |
| * There is not enough unused space to fit to block align between BAT |
| * and actual data. We can't avoid read-modify-write... |
| */ |
| s->header_size = size; |
| } |
| |
| if (ph.ext_off) { |
| if (flags & BDRV_O_RDWR) { |
| /* |
| * It's unsafe to open image RW if there is an extension (as we |
| * don't support it). But parallels driver in QEMU historically |
| * ignores the extension, so print warning and don't care. |
| */ |
| warn_report("Format Extension ignored in RW mode"); |
| } else { |
| ret = parallels_read_format_extension( |
| bs, le64_to_cpu(ph.ext_off) << BDRV_SECTOR_BITS, errp); |
| if (ret < 0) { |
| goto fail; |
| } |
| } |
| } |
| |
| if ((flags & BDRV_O_RDWR) && !(flags & BDRV_O_INACTIVE)) { |
| s->header->inuse = cpu_to_le32(HEADER_INUSE_MAGIC); |
| ret = parallels_update_header(bs); |
| if (ret < 0) { |
| goto fail; |
| } |
| } |
| |
| s->bat_dirty_block = 4 * qemu_real_host_page_size(); |
| s->bat_dirty_bmap = |
| bitmap_new(DIV_ROUND_UP(s->header_size, s->bat_dirty_block)); |
| |
| /* Disable migration until bdrv_activate method is added */ |
| error_setg(&s->migration_blocker, "The Parallels format used by node '%s' " |
| "does not support live migration", |
| bdrv_get_device_or_node_name(bs)); |
| |
| ret = migrate_add_blocker_normal(&s->migration_blocker, errp); |
| if (ret < 0) { |
| goto fail; |
| } |
| qemu_co_mutex_init(&s->lock); |
| |
| for (i = 0; i < s->bat_size; i++) { |
| sector = bat2sect(s, i); |
| if (sector + s->tracks > s->data_end) { |
| s->data_end = sector + s->tracks; |
| } |
| } |
| need_check = need_check || s->data_end > file_nb_sectors; |
| |
| if (!need_check) { |
| ret = parallels_fill_used_bitmap(bs); |
| if (ret == -ENOMEM) { |
| goto fail; |
| } |
| need_check = need_check || ret < 0; /* These are correctable errors */ |
| } |
| |
| /* |
| * We don't repair the image here if it's opened for checks. Also we don't |
| * want to change inactive images and can't change readonly images. |
| */ |
| if ((flags & (BDRV_O_CHECK | BDRV_O_INACTIVE)) || !(flags & BDRV_O_RDWR)) { |
| return 0; |
| } |
| |
| /* Repair the image if corruption was detected. */ |
| if (need_check) { |
| BdrvCheckResult res; |
| ret = bdrv_check(bs, &res, BDRV_FIX_ERRORS | BDRV_FIX_LEAKS); |
| if (ret < 0) { |
| error_setg_errno(errp, -ret, "Could not repair corrupted image"); |
| migrate_del_blocker(&s->migration_blocker); |
| goto fail; |
| } |
| } |
| return 0; |
| |
| fail_format: |
| error_setg(errp, "Image not in Parallels format"); |
| return -EINVAL; |
| |
| fail: |
| /* |
| * "s" object was allocated by g_malloc0 so we can safely |
| * try to free its fields even they were not allocated. |
| */ |
| parallels_free_used_bitmap(bs); |
| |
| g_free(s->bat_dirty_bmap); |
| qemu_vfree(s->header); |
| return ret; |
| } |
| |
| |
| static void parallels_close(BlockDriverState *bs) |
| { |
| BDRVParallelsState *s = bs->opaque; |
| |
| GRAPH_RDLOCK_GUARD_MAINLOOP(); |
| |
| if ((bs->open_flags & BDRV_O_RDWR) && !(bs->open_flags & BDRV_O_INACTIVE)) { |
| s->header->inuse = 0; |
| parallels_update_header(bs); |
| |
| /* errors are ignored, so we might as well pass exact=true */ |
| bdrv_truncate(bs->file, s->data_end << BDRV_SECTOR_BITS, true, |
| PREALLOC_MODE_OFF, 0, NULL); |
| } |
| |
| parallels_free_used_bitmap(bs); |
| |
| g_free(s->bat_dirty_bmap); |
| qemu_vfree(s->header); |
| |
| migrate_del_blocker(&s->migration_blocker); |
| } |
| |
| static bool parallels_is_support_dirty_bitmaps(BlockDriverState *bs) |
| { |
| return 1; |
| } |
| |
| static BlockDriver bdrv_parallels = { |
| .format_name = "parallels", |
| .instance_size = sizeof(BDRVParallelsState), |
| .create_opts = ¶llels_create_opts, |
| .is_format = true, |
| .supports_backing = true, |
| |
| .bdrv_has_zero_init = bdrv_has_zero_init_1, |
| .bdrv_supports_persistent_dirty_bitmap = parallels_is_support_dirty_bitmaps, |
| |
| .bdrv_probe = parallels_probe, |
| .bdrv_open = parallels_open, |
| .bdrv_close = parallels_close, |
| .bdrv_child_perm = bdrv_default_perms, |
| .bdrv_co_block_status = parallels_co_block_status, |
| .bdrv_co_flush_to_os = parallels_co_flush_to_os, |
| .bdrv_co_readv = parallels_co_readv, |
| .bdrv_co_writev = parallels_co_writev, |
| .bdrv_co_create = parallels_co_create, |
| .bdrv_co_create_opts = parallels_co_create_opts, |
| .bdrv_co_check = parallels_co_check, |
| .bdrv_co_pdiscard = parallels_co_pdiscard, |
| .bdrv_co_pwrite_zeroes = parallels_co_pwrite_zeroes, |
| }; |
| |
| static void bdrv_parallels_init(void) |
| { |
| bdrv_register(&bdrv_parallels); |
| } |
| |
| block_init(bdrv_parallels_init); |