Commit 9b80ddf3460f5513b917720a07a1d32ec4ba1817

Authored by aliguori
1 parent 64bf3f4e

Introducing qcow2 extensions (Uri Lublin)

Qcow2 extensions are build of magic (id) len (in bytes) and data.
They reside right after the qcow2 header.
If a backing filename exists it follows the qcow2 extension (if exist)

Qcow2 extensions are read upon image open.
Qcow2 extensions are identified by their magic.
Unknown qcow2 extensions (unknown magic) are skipped.
A Special magic of 0 means end-of-qcow2-extensions.

In this patchset, to be used to keep backing file format.

Based on a work done by Shahar Frank <sfrank@redhat.com>.

Signed-off-by: Uri Lublin <uril@redhat.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>


git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@6907 c046a42c-6fe2-441c-8c8c-71466251a162
Showing 1 changed file with 72 additions and 0 deletions
block-qcow2.c
@@ -45,6 +45,7 @@ @@ -45,6 +45,7 @@
45 45
46 //#define DEBUG_ALLOC 46 //#define DEBUG_ALLOC
47 //#define DEBUG_ALLOC2 47 //#define DEBUG_ALLOC2
  48 +//#define DEBUG_EXT
48 49
49 #define QCOW_MAGIC (('Q' << 24) | ('F' << 16) | ('I' << 8) | 0xfb) 50 #define QCOW_MAGIC (('Q' << 24) | ('F' << 16) | ('I' << 8) | 0xfb)
50 #define QCOW_VERSION 2 51 #define QCOW_VERSION 2
@@ -77,6 +78,14 @@ typedef struct QCowHeader { @@ -77,6 +78,14 @@ typedef struct QCowHeader {
77 uint64_t snapshots_offset; 78 uint64_t snapshots_offset;
78 } QCowHeader; 79 } QCowHeader;
79 80
  81 +
  82 +typedef struct {
  83 + uint32_t magic;
  84 + uint32_t len;
  85 +} QCowExtension;
  86 +#define QCOW_EXT_MAGIC_END 0
  87 +
  88 +
80 typedef struct __attribute__((packed)) QCowSnapshotHeader { 89 typedef struct __attribute__((packed)) QCowSnapshotHeader {
81 /* header is 8 byte aligned */ 90 /* header is 8 byte aligned */
82 uint64_t l1_table_offset; 91 uint64_t l1_table_offset;
@@ -183,11 +192,66 @@ static int qcow_probe(const uint8_t *buf, int buf_size, const char *filename) @@ -183,11 +192,66 @@ static int qcow_probe(const uint8_t *buf, int buf_size, const char *filename)
183 return 0; 192 return 0;
184 } 193 }
185 194
  195 +
  196 +/*
  197 + * read qcow2 extension and fill bs
  198 + * start reading from start_offset
  199 + * finish reading upon magic of value 0 or when end_offset reached
  200 + * unknown magic is skipped (future extension this version knows nothing about)
  201 + * return 0 upon success, non-0 otherwise
  202 + */
  203 +static int qcow_read_extensions(BlockDriverState *bs, uint64_t start_offset,
  204 + uint64_t end_offset)
  205 +{
  206 + BDRVQcowState *s = bs->opaque;
  207 + QCowExtension ext;
  208 + uint64_t offset;
  209 +
  210 +#ifdef DEBUG_EXT
  211 + printf("qcow_read_extensions: start=%ld end=%ld\n", start_offset, end_offset);
  212 +#endif
  213 + offset = start_offset;
  214 + while (offset < end_offset) {
  215 +
  216 +#ifdef DEBUG_EXT
  217 + /* Sanity check */
  218 + if (offset > s->cluster_size)
  219 + printf("qcow_handle_extension: suspicious offset %lu\n", offset);
  220 +
  221 + printf("attemting to read extended header in offset %lu\n", offset);
  222 +#endif
  223 +
  224 + if (bdrv_pread(s->hd, offset, &ext, sizeof(ext)) != sizeof(ext)) {
  225 + fprintf(stderr, "qcow_handle_extension: ERROR: pread fail from offset %lu\n",
  226 + offset);
  227 + return 1;
  228 + }
  229 + be32_to_cpus(&ext.magic);
  230 + be32_to_cpus(&ext.len);
  231 + offset += sizeof(ext);
  232 +#ifdef DEBUG_EXT
  233 + printf("ext.magic = 0x%x\n", ext.magic);
  234 +#endif
  235 + switch (ext.magic) {
  236 + case QCOW_EXT_MAGIC_END:
  237 + return 0;
  238 + default:
  239 + /* unknown magic -- just skip it */
  240 + offset += ((ext.len + 7) & ~7);
  241 + break;
  242 + }
  243 + }
  244 +
  245 + return 0;
  246 +}
  247 +
  248 +
186 static int qcow_open(BlockDriverState *bs, const char *filename, int flags) 249 static int qcow_open(BlockDriverState *bs, const char *filename, int flags)
187 { 250 {
188 BDRVQcowState *s = bs->opaque; 251 BDRVQcowState *s = bs->opaque;
189 int len, i, shift, ret; 252 int len, i, shift, ret;
190 QCowHeader header; 253 QCowHeader header;
  254 + uint64_t ext_end;
191 255
192 /* Performance is terrible right now with cache=writethrough due mainly 256 /* Performance is terrible right now with cache=writethrough due mainly
193 * to reference count updates. If the user does not explicitly specify 257 * to reference count updates. If the user does not explicitly specify
@@ -270,6 +334,14 @@ static int qcow_open(BlockDriverState *bs, const char *filename, int flags) @@ -270,6 +334,14 @@ static int qcow_open(BlockDriverState *bs, const char *filename, int flags)
270 if (refcount_init(bs) < 0) 334 if (refcount_init(bs) < 0)
271 goto fail; 335 goto fail;
272 336
  337 + /* read qcow2 extensions */
  338 + if (header.backing_file_offset)
  339 + ext_end = header.backing_file_offset;
  340 + else
  341 + ext_end = s->cluster_size;
  342 + if (qcow_read_extensions(bs, sizeof(header), ext_end))
  343 + goto fail;
  344 +
273 /* read the backing file name */ 345 /* read the backing file name */
274 if (header.backing_file_offset != 0) { 346 if (header.backing_file_offset != 0) {
275 len = header.backing_file_size; 347 len = header.backing_file_size;