|  | /* | 
|  | * Block layer qmp and info dump related functions | 
|  | * | 
|  | * Copyright (c) 2003-2008 Fabrice Bellard | 
|  | * | 
|  | * 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 "block/qapi.h" | 
|  | #include "block/block_int.h" | 
|  | #include "qmp-commands.h" | 
|  | #include "qapi-visit.h" | 
|  | #include "qapi/qmp-output-visitor.h" | 
|  | #include "qapi/qmp/types.h" | 
|  |  | 
|  | BlockDeviceInfo *bdrv_block_device_info(BlockDriverState *bs) | 
|  | { | 
|  | BlockDeviceInfo *info = g_malloc0(sizeof(*info)); | 
|  |  | 
|  | info->file                   = g_strdup(bs->filename); | 
|  | info->ro                     = bs->read_only; | 
|  | info->drv                    = g_strdup(bs->drv->format_name); | 
|  | info->encrypted              = bs->encrypted; | 
|  | info->encryption_key_missing = bdrv_key_required(bs); | 
|  |  | 
|  | if (bs->node_name[0]) { | 
|  | info->has_node_name = true; | 
|  | info->node_name = g_strdup(bs->node_name); | 
|  | } | 
|  |  | 
|  | if (bs->backing_file[0]) { | 
|  | info->has_backing_file = true; | 
|  | info->backing_file = g_strdup(bs->backing_file); | 
|  | } | 
|  |  | 
|  | info->backing_file_depth = bdrv_get_backing_file_depth(bs); | 
|  |  | 
|  | if (bs->io_limits_enabled) { | 
|  | ThrottleConfig cfg; | 
|  | throttle_get_config(&bs->throttle_state, &cfg); | 
|  | info->bps     = cfg.buckets[THROTTLE_BPS_TOTAL].avg; | 
|  | info->bps_rd  = cfg.buckets[THROTTLE_BPS_READ].avg; | 
|  | info->bps_wr  = cfg.buckets[THROTTLE_BPS_WRITE].avg; | 
|  |  | 
|  | info->iops    = cfg.buckets[THROTTLE_OPS_TOTAL].avg; | 
|  | info->iops_rd = cfg.buckets[THROTTLE_OPS_READ].avg; | 
|  | info->iops_wr = cfg.buckets[THROTTLE_OPS_WRITE].avg; | 
|  |  | 
|  | info->has_bps_max     = cfg.buckets[THROTTLE_BPS_TOTAL].max; | 
|  | info->bps_max         = cfg.buckets[THROTTLE_BPS_TOTAL].max; | 
|  | info->has_bps_rd_max  = cfg.buckets[THROTTLE_BPS_READ].max; | 
|  | info->bps_rd_max      = cfg.buckets[THROTTLE_BPS_READ].max; | 
|  | info->has_bps_wr_max  = cfg.buckets[THROTTLE_BPS_WRITE].max; | 
|  | info->bps_wr_max      = cfg.buckets[THROTTLE_BPS_WRITE].max; | 
|  |  | 
|  | info->has_iops_max    = cfg.buckets[THROTTLE_OPS_TOTAL].max; | 
|  | info->iops_max        = cfg.buckets[THROTTLE_OPS_TOTAL].max; | 
|  | info->has_iops_rd_max = cfg.buckets[THROTTLE_OPS_READ].max; | 
|  | info->iops_rd_max     = cfg.buckets[THROTTLE_OPS_READ].max; | 
|  | info->has_iops_wr_max = cfg.buckets[THROTTLE_OPS_WRITE].max; | 
|  | info->iops_wr_max     = cfg.buckets[THROTTLE_OPS_WRITE].max; | 
|  |  | 
|  | info->has_iops_size = cfg.op_size; | 
|  | info->iops_size = cfg.op_size; | 
|  | } | 
|  |  | 
|  | return info; | 
|  | } | 
|  |  | 
|  | /* | 
|  | * Returns 0 on success, with *p_list either set to describe snapshot | 
|  | * information, or NULL because there are no snapshots.  Returns -errno on | 
|  | * error, with *p_list untouched. | 
|  | */ | 
|  | int bdrv_query_snapshot_info_list(BlockDriverState *bs, | 
|  | SnapshotInfoList **p_list, | 
|  | Error **errp) | 
|  | { | 
|  | int i, sn_count; | 
|  | QEMUSnapshotInfo *sn_tab = NULL; | 
|  | SnapshotInfoList *info_list, *cur_item = NULL, *head = NULL; | 
|  | SnapshotInfo *info; | 
|  |  | 
|  | sn_count = bdrv_snapshot_list(bs, &sn_tab); | 
|  | if (sn_count < 0) { | 
|  | const char *dev = bdrv_get_device_name(bs); | 
|  | switch (sn_count) { | 
|  | case -ENOMEDIUM: | 
|  | error_setg(errp, "Device '%s' is not inserted", dev); | 
|  | break; | 
|  | case -ENOTSUP: | 
|  | error_setg(errp, | 
|  | "Device '%s' does not support internal snapshots", | 
|  | dev); | 
|  | break; | 
|  | default: | 
|  | error_setg_errno(errp, -sn_count, | 
|  | "Can't list snapshots of device '%s'", dev); | 
|  | break; | 
|  | } | 
|  | return sn_count; | 
|  | } | 
|  |  | 
|  | for (i = 0; i < sn_count; i++) { | 
|  | info = g_new0(SnapshotInfo, 1); | 
|  | info->id            = g_strdup(sn_tab[i].id_str); | 
|  | info->name          = g_strdup(sn_tab[i].name); | 
|  | info->vm_state_size = sn_tab[i].vm_state_size; | 
|  | info->date_sec      = sn_tab[i].date_sec; | 
|  | info->date_nsec     = sn_tab[i].date_nsec; | 
|  | info->vm_clock_sec  = sn_tab[i].vm_clock_nsec / 1000000000; | 
|  | info->vm_clock_nsec = sn_tab[i].vm_clock_nsec % 1000000000; | 
|  |  | 
|  | info_list = g_new0(SnapshotInfoList, 1); | 
|  | info_list->value = info; | 
|  |  | 
|  | /* XXX: waiting for the qapi to support qemu-queue.h types */ | 
|  | if (!cur_item) { | 
|  | head = cur_item = info_list; | 
|  | } else { | 
|  | cur_item->next = info_list; | 
|  | cur_item = info_list; | 
|  | } | 
|  |  | 
|  | } | 
|  |  | 
|  | g_free(sn_tab); | 
|  | *p_list = head; | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * bdrv_query_image_info: | 
|  | * @bs: block device to examine | 
|  | * @p_info: location to store image information | 
|  | * @errp: location to store error information | 
|  | * | 
|  | * Store "flat" image information in @p_info. | 
|  | * | 
|  | * "Flat" means it does *not* query backing image information, | 
|  | * i.e. (*pinfo)->has_backing_image will be set to false and | 
|  | * (*pinfo)->backing_image to NULL even when the image does in fact have | 
|  | * a backing image. | 
|  | * | 
|  | * @p_info will be set only on success. On error, store error in @errp. | 
|  | */ | 
|  | void bdrv_query_image_info(BlockDriverState *bs, | 
|  | ImageInfo **p_info, | 
|  | Error **errp) | 
|  | { | 
|  | uint64_t total_sectors; | 
|  | const char *backing_filename; | 
|  | char backing_filename2[1024]; | 
|  | BlockDriverInfo bdi; | 
|  | int ret; | 
|  | Error *err = NULL; | 
|  | ImageInfo *info = g_new0(ImageInfo, 1); | 
|  |  | 
|  | bdrv_get_geometry(bs, &total_sectors); | 
|  |  | 
|  | info->filename        = g_strdup(bs->filename); | 
|  | info->format          = g_strdup(bdrv_get_format_name(bs)); | 
|  | info->virtual_size    = total_sectors * 512; | 
|  | info->actual_size     = bdrv_get_allocated_file_size(bs); | 
|  | info->has_actual_size = info->actual_size >= 0; | 
|  | if (bdrv_is_encrypted(bs)) { | 
|  | info->encrypted = true; | 
|  | info->has_encrypted = true; | 
|  | } | 
|  | if (bdrv_get_info(bs, &bdi) >= 0) { | 
|  | if (bdi.cluster_size != 0) { | 
|  | info->cluster_size = bdi.cluster_size; | 
|  | info->has_cluster_size = true; | 
|  | } | 
|  | info->dirty_flag = bdi.is_dirty; | 
|  | info->has_dirty_flag = true; | 
|  | } | 
|  | info->format_specific     = bdrv_get_specific_info(bs); | 
|  | info->has_format_specific = info->format_specific != NULL; | 
|  |  | 
|  | backing_filename = bs->backing_file; | 
|  | if (backing_filename[0] != '\0') { | 
|  | info->backing_filename = g_strdup(backing_filename); | 
|  | info->has_backing_filename = true; | 
|  | bdrv_get_full_backing_filename(bs, backing_filename2, | 
|  | sizeof(backing_filename2)); | 
|  |  | 
|  | if (strcmp(backing_filename, backing_filename2) != 0) { | 
|  | info->full_backing_filename = | 
|  | g_strdup(backing_filename2); | 
|  | info->has_full_backing_filename = true; | 
|  | } | 
|  |  | 
|  | if (bs->backing_format[0]) { | 
|  | info->backing_filename_format = g_strdup(bs->backing_format); | 
|  | info->has_backing_filename_format = true; | 
|  | } | 
|  | } | 
|  |  | 
|  | ret = bdrv_query_snapshot_info_list(bs, &info->snapshots, &err); | 
|  | switch (ret) { | 
|  | case 0: | 
|  | if (info->snapshots) { | 
|  | info->has_snapshots = true; | 
|  | } | 
|  | break; | 
|  | /* recoverable error */ | 
|  | case -ENOMEDIUM: | 
|  | case -ENOTSUP: | 
|  | error_free(err); | 
|  | break; | 
|  | default: | 
|  | error_propagate(errp, err); | 
|  | qapi_free_ImageInfo(info); | 
|  | return; | 
|  | } | 
|  |  | 
|  | *p_info = info; | 
|  | } | 
|  |  | 
|  | /* @p_info will be set only on success. */ | 
|  | void bdrv_query_info(BlockDriverState *bs, | 
|  | BlockInfo **p_info, | 
|  | Error **errp) | 
|  | { | 
|  | BlockInfo *info = g_malloc0(sizeof(*info)); | 
|  | BlockDriverState *bs0; | 
|  | ImageInfo **p_image_info; | 
|  | Error *local_err = NULL; | 
|  | info->device = g_strdup(bs->device_name); | 
|  | info->type = g_strdup("unknown"); | 
|  | info->locked = bdrv_dev_is_medium_locked(bs); | 
|  | info->removable = bdrv_dev_has_removable_media(bs); | 
|  |  | 
|  | if (bdrv_dev_has_removable_media(bs)) { | 
|  | info->has_tray_open = true; | 
|  | info->tray_open = bdrv_dev_is_tray_open(bs); | 
|  | } | 
|  |  | 
|  | if (bdrv_iostatus_is_enabled(bs)) { | 
|  | info->has_io_status = true; | 
|  | info->io_status = bs->iostatus; | 
|  | } | 
|  |  | 
|  | if (!QLIST_EMPTY(&bs->dirty_bitmaps)) { | 
|  | info->has_dirty_bitmaps = true; | 
|  | info->dirty_bitmaps = bdrv_query_dirty_bitmaps(bs); | 
|  | } | 
|  |  | 
|  | if (bs->drv) { | 
|  | info->has_inserted = true; | 
|  | info->inserted = bdrv_block_device_info(bs); | 
|  |  | 
|  | bs0 = bs; | 
|  | p_image_info = &info->inserted->image; | 
|  | while (1) { | 
|  | bdrv_query_image_info(bs0, p_image_info, &local_err); | 
|  | if (local_err) { | 
|  | error_propagate(errp, local_err); | 
|  | goto err; | 
|  | } | 
|  | if (bs0->drv && bs0->backing_hd) { | 
|  | bs0 = bs0->backing_hd; | 
|  | (*p_image_info)->has_backing_image = true; | 
|  | p_image_info = &((*p_image_info)->backing_image); | 
|  | } else { | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | *p_info = info; | 
|  | return; | 
|  |  | 
|  | err: | 
|  | qapi_free_BlockInfo(info); | 
|  | } | 
|  |  | 
|  | BlockStats *bdrv_query_stats(const BlockDriverState *bs) | 
|  | { | 
|  | BlockStats *s; | 
|  |  | 
|  | s = g_malloc0(sizeof(*s)); | 
|  |  | 
|  | if (bs->device_name[0]) { | 
|  | s->has_device = true; | 
|  | s->device = g_strdup(bs->device_name); | 
|  | } | 
|  |  | 
|  | s->stats = g_malloc0(sizeof(*s->stats)); | 
|  | s->stats->rd_bytes = bs->nr_bytes[BDRV_ACCT_READ]; | 
|  | s->stats->wr_bytes = bs->nr_bytes[BDRV_ACCT_WRITE]; | 
|  | s->stats->rd_operations = bs->nr_ops[BDRV_ACCT_READ]; | 
|  | s->stats->wr_operations = bs->nr_ops[BDRV_ACCT_WRITE]; | 
|  | s->stats->wr_highest_offset = bs->wr_highest_sector * BDRV_SECTOR_SIZE; | 
|  | s->stats->flush_operations = bs->nr_ops[BDRV_ACCT_FLUSH]; | 
|  | s->stats->wr_total_time_ns = bs->total_time_ns[BDRV_ACCT_WRITE]; | 
|  | s->stats->rd_total_time_ns = bs->total_time_ns[BDRV_ACCT_READ]; | 
|  | s->stats->flush_total_time_ns = bs->total_time_ns[BDRV_ACCT_FLUSH]; | 
|  |  | 
|  | if (bs->file) { | 
|  | s->has_parent = true; | 
|  | s->parent = bdrv_query_stats(bs->file); | 
|  | } | 
|  |  | 
|  | if (bs->backing_hd) { | 
|  | s->has_backing = true; | 
|  | s->backing = bdrv_query_stats(bs->backing_hd); | 
|  | } | 
|  |  | 
|  | return s; | 
|  | } | 
|  |  | 
|  | BlockInfoList *qmp_query_block(Error **errp) | 
|  | { | 
|  | BlockInfoList *head = NULL, **p_next = &head; | 
|  | BlockDriverState *bs = NULL; | 
|  | Error *local_err = NULL; | 
|  |  | 
|  | while ((bs = bdrv_next(bs))) { | 
|  | BlockInfoList *info = g_malloc0(sizeof(*info)); | 
|  | bdrv_query_info(bs, &info->value, &local_err); | 
|  | if (local_err) { | 
|  | error_propagate(errp, local_err); | 
|  | goto err; | 
|  | } | 
|  |  | 
|  | *p_next = info; | 
|  | p_next = &info->next; | 
|  | } | 
|  |  | 
|  | return head; | 
|  |  | 
|  | err: | 
|  | qapi_free_BlockInfoList(head); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | BlockStatsList *qmp_query_blockstats(Error **errp) | 
|  | { | 
|  | BlockStatsList *head = NULL, **p_next = &head; | 
|  | BlockDriverState *bs = NULL; | 
|  |  | 
|  | while ((bs = bdrv_next(bs))) { | 
|  | BlockStatsList *info = g_malloc0(sizeof(*info)); | 
|  | info->value = bdrv_query_stats(bs); | 
|  |  | 
|  | *p_next = info; | 
|  | p_next = &info->next; | 
|  | } | 
|  |  | 
|  | return head; | 
|  | } | 
|  |  | 
|  | #define NB_SUFFIXES 4 | 
|  |  | 
|  | static char *get_human_readable_size(char *buf, int buf_size, int64_t size) | 
|  | { | 
|  | static const char suffixes[NB_SUFFIXES] = "KMGT"; | 
|  | int64_t base; | 
|  | int i; | 
|  |  | 
|  | if (size <= 999) { | 
|  | snprintf(buf, buf_size, "%" PRId64, size); | 
|  | } else { | 
|  | base = 1024; | 
|  | for (i = 0; i < NB_SUFFIXES; i++) { | 
|  | if (size < (10 * base)) { | 
|  | snprintf(buf, buf_size, "%0.1f%c", | 
|  | (double)size / base, | 
|  | suffixes[i]); | 
|  | break; | 
|  | } else if (size < (1000 * base) || i == (NB_SUFFIXES - 1)) { | 
|  | snprintf(buf, buf_size, "%" PRId64 "%c", | 
|  | ((size + (base >> 1)) / base), | 
|  | suffixes[i]); | 
|  | break; | 
|  | } | 
|  | base = base * 1024; | 
|  | } | 
|  | } | 
|  | return buf; | 
|  | } | 
|  |  | 
|  | void bdrv_snapshot_dump(fprintf_function func_fprintf, void *f, | 
|  | QEMUSnapshotInfo *sn) | 
|  | { | 
|  | char buf1[128], date_buf[128], clock_buf[128]; | 
|  | struct tm tm; | 
|  | time_t ti; | 
|  | int64_t secs; | 
|  |  | 
|  | if (!sn) { | 
|  | func_fprintf(f, | 
|  | "%-10s%-20s%7s%20s%15s", | 
|  | "ID", "TAG", "VM SIZE", "DATE", "VM CLOCK"); | 
|  | } else { | 
|  | ti = sn->date_sec; | 
|  | localtime_r(&ti, &tm); | 
|  | strftime(date_buf, sizeof(date_buf), | 
|  | "%Y-%m-%d %H:%M:%S", &tm); | 
|  | secs = sn->vm_clock_nsec / 1000000000; | 
|  | snprintf(clock_buf, sizeof(clock_buf), | 
|  | "%02d:%02d:%02d.%03d", | 
|  | (int)(secs / 3600), | 
|  | (int)((secs / 60) % 60), | 
|  | (int)(secs % 60), | 
|  | (int)((sn->vm_clock_nsec / 1000000) % 1000)); | 
|  | func_fprintf(f, | 
|  | "%-10s%-20s%7s%20s%15s", | 
|  | sn->id_str, sn->name, | 
|  | get_human_readable_size(buf1, sizeof(buf1), | 
|  | sn->vm_state_size), | 
|  | date_buf, | 
|  | clock_buf); | 
|  | } | 
|  | } | 
|  |  | 
|  | static void dump_qdict(fprintf_function func_fprintf, void *f, int indentation, | 
|  | QDict *dict); | 
|  | static void dump_qlist(fprintf_function func_fprintf, void *f, int indentation, | 
|  | QList *list); | 
|  |  | 
|  | static void dump_qobject(fprintf_function func_fprintf, void *f, | 
|  | int comp_indent, QObject *obj) | 
|  | { | 
|  | switch (qobject_type(obj)) { | 
|  | case QTYPE_QINT: { | 
|  | QInt *value = qobject_to_qint(obj); | 
|  | func_fprintf(f, "%" PRId64, qint_get_int(value)); | 
|  | break; | 
|  | } | 
|  | case QTYPE_QSTRING: { | 
|  | QString *value = qobject_to_qstring(obj); | 
|  | func_fprintf(f, "%s", qstring_get_str(value)); | 
|  | break; | 
|  | } | 
|  | case QTYPE_QDICT: { | 
|  | QDict *value = qobject_to_qdict(obj); | 
|  | dump_qdict(func_fprintf, f, comp_indent, value); | 
|  | break; | 
|  | } | 
|  | case QTYPE_QLIST: { | 
|  | QList *value = qobject_to_qlist(obj); | 
|  | dump_qlist(func_fprintf, f, comp_indent, value); | 
|  | break; | 
|  | } | 
|  | case QTYPE_QFLOAT: { | 
|  | QFloat *value = qobject_to_qfloat(obj); | 
|  | func_fprintf(f, "%g", qfloat_get_double(value)); | 
|  | break; | 
|  | } | 
|  | case QTYPE_QBOOL: { | 
|  | QBool *value = qobject_to_qbool(obj); | 
|  | func_fprintf(f, "%s", qbool_get_int(value) ? "true" : "false"); | 
|  | break; | 
|  | } | 
|  | case QTYPE_QERROR: { | 
|  | QString *value = qerror_human((QError *)obj); | 
|  | func_fprintf(f, "%s", qstring_get_str(value)); | 
|  | break; | 
|  | } | 
|  | case QTYPE_NONE: | 
|  | break; | 
|  | case QTYPE_MAX: | 
|  | default: | 
|  | abort(); | 
|  | } | 
|  | } | 
|  |  | 
|  | static void dump_qlist(fprintf_function func_fprintf, void *f, int indentation, | 
|  | QList *list) | 
|  | { | 
|  | const QListEntry *entry; | 
|  | int i = 0; | 
|  |  | 
|  | for (entry = qlist_first(list); entry; entry = qlist_next(entry), i++) { | 
|  | qtype_code type = qobject_type(entry->value); | 
|  | bool composite = (type == QTYPE_QDICT || type == QTYPE_QLIST); | 
|  | const char *format = composite ? "%*s[%i]:\n" : "%*s[%i]: "; | 
|  |  | 
|  | func_fprintf(f, format, indentation * 4, "", i); | 
|  | dump_qobject(func_fprintf, f, indentation + 1, entry->value); | 
|  | if (!composite) { | 
|  | func_fprintf(f, "\n"); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | static void dump_qdict(fprintf_function func_fprintf, void *f, int indentation, | 
|  | QDict *dict) | 
|  | { | 
|  | const QDictEntry *entry; | 
|  |  | 
|  | for (entry = qdict_first(dict); entry; entry = qdict_next(dict, entry)) { | 
|  | qtype_code type = qobject_type(entry->value); | 
|  | bool composite = (type == QTYPE_QDICT || type == QTYPE_QLIST); | 
|  | const char *format = composite ? "%*s%s:\n" : "%*s%s: "; | 
|  | char key[strlen(entry->key) + 1]; | 
|  | int i; | 
|  |  | 
|  | /* replace dashes with spaces in key (variable) names */ | 
|  | for (i = 0; entry->key[i]; i++) { | 
|  | key[i] = entry->key[i] == '-' ? ' ' : entry->key[i]; | 
|  | } | 
|  | key[i] = 0; | 
|  |  | 
|  | func_fprintf(f, format, indentation * 4, "", key); | 
|  | dump_qobject(func_fprintf, f, indentation + 1, entry->value); | 
|  | if (!composite) { | 
|  | func_fprintf(f, "\n"); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void bdrv_image_info_specific_dump(fprintf_function func_fprintf, void *f, | 
|  | ImageInfoSpecific *info_spec) | 
|  | { | 
|  | Error *local_err = NULL; | 
|  | QmpOutputVisitor *ov = qmp_output_visitor_new(); | 
|  | QObject *obj, *data; | 
|  |  | 
|  | visit_type_ImageInfoSpecific(qmp_output_get_visitor(ov), &info_spec, NULL, | 
|  | &local_err); | 
|  | obj = qmp_output_get_qobject(ov); | 
|  | assert(qobject_type(obj) == QTYPE_QDICT); | 
|  | data = qdict_get(qobject_to_qdict(obj), "data"); | 
|  | dump_qobject(func_fprintf, f, 1, data); | 
|  | qmp_output_visitor_cleanup(ov); | 
|  | } | 
|  |  | 
|  | void bdrv_image_info_dump(fprintf_function func_fprintf, void *f, | 
|  | ImageInfo *info) | 
|  | { | 
|  | char size_buf[128], dsize_buf[128]; | 
|  | if (!info->has_actual_size) { | 
|  | snprintf(dsize_buf, sizeof(dsize_buf), "unavailable"); | 
|  | } else { | 
|  | get_human_readable_size(dsize_buf, sizeof(dsize_buf), | 
|  | info->actual_size); | 
|  | } | 
|  | get_human_readable_size(size_buf, sizeof(size_buf), info->virtual_size); | 
|  | func_fprintf(f, | 
|  | "image: %s\n" | 
|  | "file format: %s\n" | 
|  | "virtual size: %s (%" PRId64 " bytes)\n" | 
|  | "disk size: %s\n", | 
|  | info->filename, info->format, size_buf, | 
|  | info->virtual_size, | 
|  | dsize_buf); | 
|  |  | 
|  | if (info->has_encrypted && info->encrypted) { | 
|  | func_fprintf(f, "encrypted: yes\n"); | 
|  | } | 
|  |  | 
|  | if (info->has_cluster_size) { | 
|  | func_fprintf(f, "cluster_size: %" PRId64 "\n", | 
|  | info->cluster_size); | 
|  | } | 
|  |  | 
|  | if (info->has_dirty_flag && info->dirty_flag) { | 
|  | func_fprintf(f, "cleanly shut down: no\n"); | 
|  | } | 
|  |  | 
|  | if (info->has_backing_filename) { | 
|  | func_fprintf(f, "backing file: %s", info->backing_filename); | 
|  | if (info->has_full_backing_filename) { | 
|  | func_fprintf(f, " (actual path: %s)", info->full_backing_filename); | 
|  | } | 
|  | func_fprintf(f, "\n"); | 
|  | if (info->has_backing_filename_format) { | 
|  | func_fprintf(f, "backing file format: %s\n", | 
|  | info->backing_filename_format); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (info->has_snapshots) { | 
|  | SnapshotInfoList *elem; | 
|  |  | 
|  | func_fprintf(f, "Snapshot list:\n"); | 
|  | bdrv_snapshot_dump(func_fprintf, f, NULL); | 
|  | func_fprintf(f, "\n"); | 
|  |  | 
|  | /* Ideally bdrv_snapshot_dump() would operate on SnapshotInfoList but | 
|  | * we convert to the block layer's native QEMUSnapshotInfo for now. | 
|  | */ | 
|  | for (elem = info->snapshots; elem; elem = elem->next) { | 
|  | QEMUSnapshotInfo sn = { | 
|  | .vm_state_size = elem->value->vm_state_size, | 
|  | .date_sec = elem->value->date_sec, | 
|  | .date_nsec = elem->value->date_nsec, | 
|  | .vm_clock_nsec = elem->value->vm_clock_sec * 1000000000ULL + | 
|  | elem->value->vm_clock_nsec, | 
|  | }; | 
|  |  | 
|  | pstrcpy(sn.id_str, sizeof(sn.id_str), elem->value->id); | 
|  | pstrcpy(sn.name, sizeof(sn.name), elem->value->name); | 
|  | bdrv_snapshot_dump(func_fprintf, f, &sn); | 
|  | func_fprintf(f, "\n"); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (info->has_format_specific) { | 
|  | func_fprintf(f, "Format specific information:\n"); | 
|  | bdrv_image_info_specific_dump(func_fprintf, f, info->format_specific); | 
|  | } | 
|  | } |