Skip to content

Commit d14ed18

Browse files
LaneWolfkevmw
authored andcommittedFeb 22, 2013
qemu-img: Add compare subcommand
This patch adds new qemu-img subcommand that compares content of two disk images. Signed-off-by: Miroslav Rezanina <[email protected]> Reviewed-by: Kevin Wolf <[email protected]> Signed-off-by: Stefan Hajnoczi <[email protected]>
1 parent f382d43 commit d14ed18

File tree

3 files changed

+348
-1
lines changed

3 files changed

+348
-1
lines changed
 

‎qemu-img-cmds.hx

+6
Original file line numberDiff line numberDiff line change
@@ -27,6 +27,12 @@ STEXI
2727
@item commit [-q] [-f @var{fmt}] [-t @var{cache}] @var{filename}
2828
ETEXI
2929

30+
DEF("compare", img_compare,
31+
"compare [-f fmt] [-F fmt] [-p] [-q] [-s] filename1 filename2")
32+
STEXI
33+
@item compare [-f @var{fmt}] [-F @var{fmt}] [-p] [-q] [-s] @var{filename1} @var{filename2}
34+
ETEXI
35+
3036
DEF("convert", img_convert,
3137
"convert [-c] [-p] [-q] [-f fmt] [-t cache] [-O output_fmt] [-o options] [-s snapshot_name] [-S sparse_size] filename [filename2 [...]] output_filename")
3238
STEXI

‎qemu-img.c

+289-1
Original file line numberDiff line numberDiff line change
@@ -113,7 +113,12 @@ static void help(void)
113113
" '-a' applies a snapshot (revert disk to saved state)\n"
114114
" '-c' creates a snapshot\n"
115115
" '-d' deletes a snapshot\n"
116-
" '-l' lists all snapshots in the given image\n";
116+
" '-l' lists all snapshots in the given image\n"
117+
"\n"
118+
"Parameters to compare subcommand:\n"
119+
" '-f' first image format\n"
120+
" '-F' second image format\n"
121+
" '-s' run in Strict mode - fail on different image size or sector allocation\n";
117122

118123
printf("%s\nSupported formats:", help_msg);
119124
bdrv_iterate_format(format_print, NULL);
@@ -820,6 +825,289 @@ static int compare_sectors(const uint8_t *buf1, const uint8_t *buf2, int n,
820825

821826
#define IO_BUF_SIZE (2 * 1024 * 1024)
822827

828+
static int64_t sectors_to_bytes(int64_t sectors)
829+
{
830+
return sectors << BDRV_SECTOR_BITS;
831+
}
832+
833+
static int64_t sectors_to_process(int64_t total, int64_t from)
834+
{
835+
return MIN(total - from, IO_BUF_SIZE >> BDRV_SECTOR_BITS);
836+
}
837+
838+
/*
839+
* Check if passed sectors are empty (not allocated or contain only 0 bytes)
840+
*
841+
* Returns 0 in case sectors are filled with 0, 1 if sectors contain non-zero
842+
* data and negative value on error.
843+
*
844+
* @param bs: Driver used for accessing file
845+
* @param sect_num: Number of first sector to check
846+
* @param sect_count: Number of sectors to check
847+
* @param filename: Name of disk file we are checking (logging purpose)
848+
* @param buffer: Allocated buffer for storing read data
849+
* @param quiet: Flag for quiet mode
850+
*/
851+
static int check_empty_sectors(BlockDriverState *bs, int64_t sect_num,
852+
int sect_count, const char *filename,
853+
uint8_t *buffer, bool quiet)
854+
{
855+
int pnum, ret = 0;
856+
ret = bdrv_read(bs, sect_num, buffer, sect_count);
857+
if (ret < 0) {
858+
error_report("Error while reading offset %" PRId64 " of %s: %s",
859+
sectors_to_bytes(sect_num), filename, strerror(-ret));
860+
return ret;
861+
}
862+
ret = is_allocated_sectors(buffer, sect_count, &pnum);
863+
if (ret || pnum != sect_count) {
864+
qprintf(quiet, "Content mismatch at offset %" PRId64 "!\n",
865+
sectors_to_bytes(ret ? sect_num : sect_num + pnum));
866+
return 1;
867+
}
868+
869+
return 0;
870+
}
871+
872+
/*
873+
* Compares two images. Exit codes:
874+
*
875+
* 0 - Images are identical
876+
* 1 - Images differ
877+
* >1 - Error occurred
878+
*/
879+
static int img_compare(int argc, char **argv)
880+
{
881+
const char *fmt1 = NULL, *fmt2 = NULL, *filename1, *filename2;
882+
BlockDriverState *bs1, *bs2;
883+
int64_t total_sectors1, total_sectors2;
884+
uint8_t *buf1 = NULL, *buf2 = NULL;
885+
int pnum1, pnum2;
886+
int allocated1, allocated2;
887+
int ret = 0; /* return value - 0 Ident, 1 Different, >1 Error */
888+
bool progress = false, quiet = false, strict = false;
889+
int64_t total_sectors;
890+
int64_t sector_num = 0;
891+
int64_t nb_sectors;
892+
int c, pnum;
893+
uint64_t bs_sectors;
894+
uint64_t progress_base;
895+
896+
for (;;) {
897+
c = getopt(argc, argv, "hpf:F:sq");
898+
if (c == -1) {
899+
break;
900+
}
901+
switch (c) {
902+
case '?':
903+
case 'h':
904+
help();
905+
break;
906+
case 'f':
907+
fmt1 = optarg;
908+
break;
909+
case 'F':
910+
fmt2 = optarg;
911+
break;
912+
case 'p':
913+
progress = true;
914+
break;
915+
case 'q':
916+
quiet = true;
917+
break;
918+
case 's':
919+
strict = true;
920+
break;
921+
}
922+
}
923+
924+
/* Progress is not shown in Quiet mode */
925+
if (quiet) {
926+
progress = false;
927+
}
928+
929+
930+
if (optind > argc - 2) {
931+
help();
932+
}
933+
filename1 = argv[optind++];
934+
filename2 = argv[optind++];
935+
936+
/* Initialize before goto out */
937+
qemu_progress_init(progress, 2.0);
938+
939+
bs1 = bdrv_new_open(filename1, fmt1, BDRV_O_FLAGS, true, quiet);
940+
if (!bs1) {
941+
error_report("Can't open file %s", filename1);
942+
ret = 2;
943+
goto out3;
944+
}
945+
946+
bs2 = bdrv_new_open(filename2, fmt2, BDRV_O_FLAGS, true, quiet);
947+
if (!bs2) {
948+
error_report("Can't open file %s", filename2);
949+
ret = 2;
950+
goto out2;
951+
}
952+
953+
buf1 = qemu_blockalign(bs1, IO_BUF_SIZE);
954+
buf2 = qemu_blockalign(bs2, IO_BUF_SIZE);
955+
bdrv_get_geometry(bs1, &bs_sectors);
956+
total_sectors1 = bs_sectors;
957+
bdrv_get_geometry(bs2, &bs_sectors);
958+
total_sectors2 = bs_sectors;
959+
total_sectors = MIN(total_sectors1, total_sectors2);
960+
progress_base = MAX(total_sectors1, total_sectors2);
961+
962+
qemu_progress_print(0, 100);
963+
964+
if (strict && total_sectors1 != total_sectors2) {
965+
ret = 1;
966+
qprintf(quiet, "Strict mode: Image size mismatch!\n");
967+
goto out;
968+
}
969+
970+
for (;;) {
971+
nb_sectors = sectors_to_process(total_sectors, sector_num);
972+
if (nb_sectors <= 0) {
973+
break;
974+
}
975+
allocated1 = bdrv_is_allocated_above(bs1, NULL, sector_num, nb_sectors,
976+
&pnum1);
977+
if (allocated1 < 0) {
978+
ret = 3;
979+
error_report("Sector allocation test failed for %s", filename1);
980+
goto out;
981+
}
982+
983+
allocated2 = bdrv_is_allocated_above(bs2, NULL, sector_num, nb_sectors,
984+
&pnum2);
985+
if (allocated2 < 0) {
986+
ret = 3;
987+
error_report("Sector allocation test failed for %s", filename2);
988+
goto out;
989+
}
990+
nb_sectors = MIN(pnum1, pnum2);
991+
992+
if (allocated1 == allocated2) {
993+
if (allocated1) {
994+
ret = bdrv_read(bs1, sector_num, buf1, nb_sectors);
995+
if (ret < 0) {
996+
error_report("Error while reading offset %" PRId64 " of %s:"
997+
" %s", sectors_to_bytes(sector_num), filename1,
998+
strerror(-ret));
999+
ret = 4;
1000+
goto out;
1001+
}
1002+
ret = bdrv_read(bs2, sector_num, buf2, nb_sectors);
1003+
if (ret < 0) {
1004+
error_report("Error while reading offset %" PRId64
1005+
" of %s: %s", sectors_to_bytes(sector_num),
1006+
filename2, strerror(-ret));
1007+
ret = 4;
1008+
goto out;
1009+
}
1010+
ret = compare_sectors(buf1, buf2, nb_sectors, &pnum);
1011+
if (ret || pnum != nb_sectors) {
1012+
ret = 1;
1013+
qprintf(quiet, "Content mismatch at offset %" PRId64 "!\n",
1014+
sectors_to_bytes(
1015+
ret ? sector_num : sector_num + pnum));
1016+
goto out;
1017+
}
1018+
}
1019+
} else {
1020+
if (strict) {
1021+
ret = 1;
1022+
qprintf(quiet, "Strict mode: Offset %" PRId64
1023+
" allocation mismatch!\n",
1024+
sectors_to_bytes(sector_num));
1025+
goto out;
1026+
}
1027+
1028+
if (allocated1) {
1029+
ret = check_empty_sectors(bs1, sector_num, nb_sectors,
1030+
filename1, buf1, quiet);
1031+
} else {
1032+
ret = check_empty_sectors(bs2, sector_num, nb_sectors,
1033+
filename2, buf1, quiet);
1034+
}
1035+
if (ret) {
1036+
if (ret < 0) {
1037+
ret = 4;
1038+
error_report("Error while reading offset %" PRId64 ": %s",
1039+
sectors_to_bytes(sector_num), strerror(-ret));
1040+
}
1041+
goto out;
1042+
}
1043+
}
1044+
sector_num += nb_sectors;
1045+
qemu_progress_print(((float) nb_sectors / progress_base)*100, 100);
1046+
}
1047+
1048+
if (total_sectors1 != total_sectors2) {
1049+
BlockDriverState *bs_over;
1050+
int64_t total_sectors_over;
1051+
const char *filename_over;
1052+
1053+
qprintf(quiet, "Warning: Image size mismatch!\n");
1054+
if (total_sectors1 > total_sectors2) {
1055+
total_sectors_over = total_sectors1;
1056+
bs_over = bs1;
1057+
filename_over = filename1;
1058+
} else {
1059+
total_sectors_over = total_sectors2;
1060+
bs_over = bs2;
1061+
filename_over = filename2;
1062+
}
1063+
1064+
for (;;) {
1065+
nb_sectors = sectors_to_process(total_sectors_over, sector_num);
1066+
if (nb_sectors <= 0) {
1067+
break;
1068+
}
1069+
ret = bdrv_is_allocated_above(bs_over, NULL, sector_num,
1070+
nb_sectors, &pnum);
1071+
if (ret < 0) {
1072+
ret = 3;
1073+
error_report("Sector allocation test failed for %s",
1074+
filename_over);
1075+
goto out;
1076+
1077+
}
1078+
nb_sectors = pnum;
1079+
if (ret) {
1080+
ret = check_empty_sectors(bs_over, sector_num, nb_sectors,
1081+
filename_over, buf1, quiet);
1082+
if (ret) {
1083+
if (ret < 0) {
1084+
ret = 4;
1085+
error_report("Error while reading offset %" PRId64
1086+
" of %s: %s", sectors_to_bytes(sector_num),
1087+
filename_over, strerror(-ret));
1088+
}
1089+
goto out;
1090+
}
1091+
}
1092+
sector_num += nb_sectors;
1093+
qemu_progress_print(((float) nb_sectors / progress_base)*100, 100);
1094+
}
1095+
}
1096+
1097+
qprintf(quiet, "Images are identical.\n");
1098+
ret = 0;
1099+
1100+
out:
1101+
bdrv_delete(bs2);
1102+
qemu_vfree(buf1);
1103+
qemu_vfree(buf2);
1104+
out2:
1105+
bdrv_delete(bs1);
1106+
out3:
1107+
qemu_progress_end();
1108+
return ret;
1109+
}
1110+
8231111
static int img_convert(int argc, char **argv)
8241112
{
8251113
int c, ret = 0, n, n1, bs_n, bs_i, compress, cluster_size, cluster_sectors;

‎qemu-img.texi

+53
Original file line numberDiff line numberDiff line change
@@ -84,6 +84,18 @@ deletes a snapshot
8484
lists all snapshots in the given image
8585
@end table
8686

87+
Parameters to compare subcommand:
88+
89+
@table @option
90+
91+
@item -f
92+
First image format
93+
@item -F
94+
Second image format
95+
@item -s
96+
Strict mode - fail on on different image size or sector allocation
97+
@end table
98+
8799
Command description:
88100

89101
@table @option
@@ -118,6 +130,47 @@ it doesn't need to be specified separately in this case.
118130

119131
Commit the changes recorded in @var{filename} in its base image.
120132

133+
@item compare [-f @var{fmt}] [-F @var{fmt}] [-p] [-s] [-q] @var{filename1} @var{filename2}
134+
135+
Check if two images have the same content. You can compare images with
136+
different format or settings.
137+
138+
The format is probed unless you specify it by @var{-f} (used for
139+
@var{filename1}) and/or @var{-F} (used for @var{filename2}) option.
140+
141+
By default, images with different size are considered identical if the larger
142+
image contains only unallocated and/or zeroed sectors in the area after the end
143+
of the other image. In addition, if any sector is not allocated in one image
144+
and contains only zero bytes in the second one, it is evaluated as equal. You
145+
can use Strict mode by specifying the @var{-s} option. When compare runs in
146+
Strict mode, it fails in case image size differs or a sector is allocated in
147+
one image and is not allocated in the second one.
148+
149+
By default, compare prints out a result message. This message displays
150+
information that both images are same or the position of the first different
151+
byte. In addition, result message can report different image size in case
152+
Strict mode is used.
153+
154+
Compare exits with @code{0} in case the images are equal and with @code{1}
155+
in case the images differ. Other exit codes mean an error occurred during
156+
execution and standard error output should contain an error message.
157+
The following table sumarizes all exit codes of the compare subcommand:
158+
159+
@table @option
160+
161+
@item 0
162+
Images are identical
163+
@item 1
164+
Images differ
165+
@item 2
166+
Error on opening an image
167+
@item 3
168+
Error on checking a sector allocation
169+
@item 4
170+
Error on reading data
171+
172+
@end table
173+
121174
@item convert [-c] [-p] [-f @var{fmt}] [-t @var{cache}] [-O @var{output_fmt}] [-o @var{options}] [-s @var{snapshot_name}] [-S @var{sparse_size}] @var{filename} [@var{filename2} [...]] @var{output_filename}
122175

123176
Convert the disk image @var{filename} or a snapshot @var{snapshot_name} to disk image @var{output_filename}

0 commit comments

Comments
 (0)
Please sign in to comment.