@@ -113,7 +113,12 @@ static void help(void)
113
113
" '-a' applies a snapshot (revert disk to saved state)\n"
114
114
" '-c' creates a snapshot\n"
115
115
" '-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" ;
117
122
118
123
printf ("%s\nSupported formats:" , help_msg );
119
124
bdrv_iterate_format (format_print , NULL );
@@ -820,6 +825,289 @@ static int compare_sectors(const uint8_t *buf1, const uint8_t *buf2, int n,
820
825
821
826
#define IO_BUF_SIZE (2 * 1024 * 1024)
822
827
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
+
823
1111
static int img_convert (int argc , char * * argv )
824
1112
{
825
1113
int c , ret = 0 , n , n1 , bs_n , bs_i , compress , cluster_size , cluster_sectors ;
0 commit comments