Commit b9fa33a6e4cb65c55798d8cef604f3b57383280e
1 parent
f55761a0
block-vpc: Split up struct vpc_subheader (Kevin Wolf)
struct vpc_subheader currently is a union of two completely different data structures (the Hard Disk Footer and the Dynamic Disk Header). That doesn't make too much sense, so split them up. Signed-off-by: Kevin Wolf <kwolf@suse.de> Signed-off-by: Anthony Liguori <aliguori@us.ibm.com> git-svn-id: svn://svn.savannah.nongnu.org/qemu/trunk@6454 c046a42c-6fe2-441c-8c8c-71466251a162
Showing
1 changed file
with
43 additions
and
42 deletions
block-vpc.c
| ... | ... | @@ -31,36 +31,31 @@ |
| 31 | 31 | //#define CACHE |
| 32 | 32 | |
| 33 | 33 | // always big-endian |
| 34 | -struct vpc_subheader { | |
| 35 | - char magic[8]; // "conectix" / "cxsparse" | |
| 36 | - union { | |
| 37 | - struct { | |
| 38 | - uint32_t unk1[2]; | |
| 39 | - uint32_t unk2; // always zero? | |
| 40 | - uint32_t subheader_offset; | |
| 41 | - uint32_t unk3; // some size? | |
| 42 | - char creator[4]; // "vpc " | |
| 43 | - uint16_t major; | |
| 44 | - uint16_t minor; | |
| 45 | - char guest[4]; // "Wi2k" | |
| 46 | - uint32_t unk4[7]; | |
| 47 | - uint8_t vnet_id[16]; // virtual network id, purpose unknown | |
| 48 | - // next 16 longs are used, but dunno the purpose | |
| 49 | - // next 6 longs unknown, following 7 long maybe a serial | |
| 50 | - char padding[HEADER_SIZE - 84]; | |
| 51 | - } main; | |
| 52 | - struct { | |
| 53 | - uint32_t unk1[2]; // all bits set | |
| 54 | - uint32_t unk2; // always zero? | |
| 55 | - uint32_t pagetable_offset; | |
| 56 | - uint32_t unk3; | |
| 57 | - uint32_t pagetable_entries; // 32bit/entry | |
| 58 | - uint32_t pageentry_size; // 512*8*512 | |
| 59 | - uint32_t nb_sectors; | |
| 60 | - char padding[HEADER_SIZE - 40]; | |
| 61 | - } sparse; | |
| 62 | - char padding[HEADER_SIZE - 8]; | |
| 63 | - } type; | |
| 34 | +struct vhd_footer { | |
| 35 | + char creator[8]; // "conectix | |
| 36 | + uint32_t unk1[2]; | |
| 37 | + uint32_t unk2; // always zero? | |
| 38 | + uint32_t subheader_offset; | |
| 39 | + uint32_t unk3; // some size? | |
| 40 | + char creator_app[4]; // "vpc " | |
| 41 | + uint16_t major; | |
| 42 | + uint16_t minor; | |
| 43 | + char guest[4]; // "Wi2k" | |
| 44 | + uint32_t unk4[7]; | |
| 45 | + uint8_t vnet_id[16]; // virtual network id, purpose unknown | |
| 46 | + // next 16 longs are used, but dunno the purpose | |
| 47 | + // next 6 longs unknown, following 7 long maybe a serial | |
| 48 | +}; | |
| 49 | + | |
| 50 | +struct vhd_dyndisk_header { | |
| 51 | + char magic[8]; // "cxsparse" | |
| 52 | + uint32_t unk1[2]; // all bits set | |
| 53 | + uint32_t unk2; // always zero? | |
| 54 | + uint32_t pagetable_offset; | |
| 55 | + uint32_t unk3; | |
| 56 | + uint32_t pagetable_entries; // 32bit/entry | |
| 57 | + uint32_t pageentry_size; // 512*8*512 | |
| 58 | + uint32_t nb_sectors; | |
| 64 | 59 | }; |
| 65 | 60 | |
| 66 | 61 | typedef struct BDRVVPCState { |
| ... | ... | @@ -90,7 +85,9 @@ static int vpc_open(BlockDriverState *bs, const char *filename, int flags) |
| 90 | 85 | { |
| 91 | 86 | BDRVVPCState *s = bs->opaque; |
| 92 | 87 | int fd, i; |
| 93 | - struct vpc_subheader header; | |
| 88 | + struct vhd_footer* footer; | |
| 89 | + struct vhd_dyndisk_header* dyndisk_header; | |
| 90 | + uint8_t buf[HEADER_SIZE]; | |
| 94 | 91 | |
| 95 | 92 | fd = open(filename, O_RDONLY | O_BINARY); |
| 96 | 93 | if (fd < 0) |
| ... | ... | @@ -100,25 +97,29 @@ static int vpc_open(BlockDriverState *bs, const char *filename, int flags) |
| 100 | 97 | |
| 101 | 98 | s->fd = fd; |
| 102 | 99 | |
| 103 | - if (read(fd, &header, HEADER_SIZE) != HEADER_SIZE) | |
| 100 | + if (read(fd, buf, HEADER_SIZE) != HEADER_SIZE) | |
| 104 | 101 | goto fail; |
| 105 | 102 | |
| 106 | - if (strncmp(header.magic, "conectix", 8)) | |
| 103 | + footer = (struct vhd_footer*) buf; | |
| 104 | + if (strncmp(footer->creator, "conectix", 8)) | |
| 107 | 105 | goto fail; |
| 108 | - lseek(s->fd, be32_to_cpu(header.type.main.subheader_offset), SEEK_SET); | |
| 109 | 106 | |
| 110 | - if (read(fd, &header, HEADER_SIZE) != HEADER_SIZE) | |
| 107 | + lseek(s->fd, be32_to_cpu(footer->subheader_offset), SEEK_SET); | |
| 108 | + if (read(fd, buf, HEADER_SIZE) != HEADER_SIZE) | |
| 111 | 109 | goto fail; |
| 112 | 110 | |
| 113 | - if (strncmp(header.magic, "cxsparse", 8)) | |
| 114 | - goto fail; | |
| 111 | + footer = NULL; | |
| 112 | + dyndisk_header = (struct vhd_dyndisk_header*) buf; | |
| 113 | + | |
| 114 | + if (strncmp(dyndisk_header->magic, "cxsparse", 8)) | |
| 115 | + goto fail; | |
| 115 | 116 | |
| 116 | - bs->total_sectors = ((uint64_t)be32_to_cpu(header.type.sparse.pagetable_entries) * | |
| 117 | - be32_to_cpu(header.type.sparse.pageentry_size)) / 512; | |
| 117 | + bs->total_sectors = ((uint64_t)be32_to_cpu(dyndisk_header->pagetable_entries) * | |
| 118 | + be32_to_cpu(dyndisk_header->pageentry_size)) / 512; | |
| 118 | 119 | |
| 119 | - lseek(s->fd, be32_to_cpu(header.type.sparse.pagetable_offset), SEEK_SET); | |
| 120 | + lseek(s->fd, be32_to_cpu(dyndisk_header->pagetable_offset), SEEK_SET); | |
| 120 | 121 | |
| 121 | - s->pagetable_entries = be32_to_cpu(header.type.sparse.pagetable_entries); | |
| 122 | + s->pagetable_entries = be32_to_cpu(dyndisk_header->pagetable_entries); | |
| 122 | 123 | s->pagetable = qemu_malloc(s->pagetable_entries * 4); |
| 123 | 124 | if (!s->pagetable) |
| 124 | 125 | goto fail; |
| ... | ... | @@ -128,7 +129,7 @@ static int vpc_open(BlockDriverState *bs, const char *filename, int flags) |
| 128 | 129 | for (i = 0; i < s->pagetable_entries; i++) |
| 129 | 130 | be32_to_cpus(&s->pagetable[i]); |
| 130 | 131 | |
| 131 | - s->pageentry_size = be32_to_cpu(header.type.sparse.pageentry_size); | |
| 132 | + s->pageentry_size = be32_to_cpu(dyndisk_header->pageentry_size); | |
| 132 | 133 | #ifdef CACHE |
| 133 | 134 | s->pageentry_u8 = qemu_malloc(512); |
| 134 | 135 | if (!s->pageentry_u8) | ... | ... |