📄 block-qcow.c
字号:
uint64_t cluster_offset, sector, nb_sectors;
struct qcow_prv* prv;
td_request_t clone = treq;
char* buf = treq.buf;
sector = treq.sec;
nb_sectors = treq.secs;
/*We store a local record of the request*/
while (nb_sectors > 0) {
cluster_offset =
get_cluster_offset(s, sector << 9, 0, 0, 0, 0);
index_in_cluster = sector & (s->cluster_sectors - 1);
n = s->cluster_sectors - index_in_cluster;
if (n > nb_sectors)
n = nb_sectors;
if (s->aio_free_count == 0) {
td_complete_request(treq, -EBUSY);
return;
}
if(!cluster_offset) {
treq.buf = buf;
treq.sec = sector;
treq.secs = n;
td_forward_request(treq);
} else if (cluster_offset & QCOW_OFLAG_COMPRESSED) {
if (decompress_cluster(s, cluster_offset) < 0) {
td_complete_request(treq, -EIO);
goto done;
}
memcpy(buf, s->cluster_cache + index_in_cluster * 512,
512 * n);
treq.buf = buf;
treq.sec = sector;
treq.secs = n;
td_complete_request(treq, 0);
} else {
clone.buf = buf;
clone.sec = (cluster_offset>>9)+index_in_cluster;
clone.secs = n;
async_read(driver, clone);
}
nb_sectors -= n;
sector += n;
buf += n * 512;
}
done:
return;
}
void tdqcow_queue_write(td_driver_t *driver, td_request_t treq)
{
struct tdqcow_state *s = (struct tdqcow_state *)driver->data;
int ret = 0, index_in_cluster, n, i;
uint64_t cluster_offset, sector, nb_sectors;
td_callback_t cb;
struct qcow_prv* prv;
char* buf = treq.buf;
td_request_t clone=treq;
sector = treq.sec;
nb_sectors = treq.secs;
/*We store a local record of the request*/
while (nb_sectors > 0) {
index_in_cluster = sector & (s->cluster_sectors - 1);
n = s->cluster_sectors - index_in_cluster;
if (n > nb_sectors)
n = nb_sectors;
if (s->aio_free_count == 0) {
td_complete_request(treq, -EBUSY);
return;
}
cluster_offset = get_cluster_offset(s, sector << 9, 1, 0,
index_in_cluster,
index_in_cluster+n);
if (!cluster_offset) {
DPRINTF("Ooops, no write cluster offset!\n");
td_complete_request(treq, -EIO);
return;
}
if (s->crypt_method) {
encrypt_sectors(s, sector, s->cluster_data,
(unsigned char *)buf, n, 1,
&s->aes_encrypt_key);
clone.buf = buf;
clone.sec = (cluster_offset>>9) + index_in_cluster;
clone.secs = n;
async_write(driver, clone);
} else {
clone.buf = buf;
clone.sec = (cluster_offset>>9) + index_in_cluster;
clone.secs = n;
async_write(driver, clone);
}
nb_sectors -= n;
sector += n;
buf += n * 512;
}
s->cluster_cache_offset = -1; /* disable compressed cache */
return;
}
static int
tdqcow_update_checksum(struct tdqcow_state *s)
{
int i, fd, err;
uint32_t offset, cksum, out;
if (!s->extended)
return 0;
fd = open(s->name, O_WRONLY | O_LARGEFILE); /* open without O_DIRECT */
if (fd == -1) {
err = errno;
goto out;
}
offset = sizeof(QCowHeader) + offsetof(QCowHeader_ext, cksum);
if (lseek(fd, offset, SEEK_SET) == (off_t)-1) {
err = errno;
goto out;
}
/* convert to big endian for checksum */
for (i = 0; i < s->l1_size; i++)
cpu_to_be64s(&s->l1_table[i]);
cksum = gen_cksum((char *)s->l1_table, s->l1_size * sizeof(uint64_t));
/* and back again... */
for (i = 0; i < s->l1_size; i++)
be64_to_cpus(&s->l1_table[i]);
DPRINTF("Writing cksum: %d", cksum);
out = cpu_to_be32(cksum);
if (write(fd, &out, sizeof(out)) != sizeof(out)) {
err = errno;
goto out;
}
err = 0;
out:
if (err)
DPRINTF("failed to update checksum: %d\n", err);
if (fd != -1)
close(fd);
return err;
}
int tdqcow_close(td_driver_t *driver)
{
struct tdqcow_state *s = (struct tdqcow_state *)driver->data;
/*Update the hdr cksum*/
tdqcow_update_checksum(s);
free_aio_state(s);
free(s->name);
free(s->l1_table);
free(s->l2_cache);
free(s->cluster_cache);
free(s->cluster_data);
close(s->fd);
return 0;
}
int qcow_create(const char *filename, uint64_t total_size,
const char *backing_file, int sparse)
{
int fd, header_size, backing_filename_len, l1_size, i;
int shift, length, adjust, flags = 0, ret = 0;
QCowHeader header;
QCowHeader_ext exthdr;
char backing_filename[PATH_MAX], *ptr;
uint64_t tmp, size, total_length;
struct stat st;
DPRINTF("Qcow_create: size %"PRIu64"\n",(long long unsigned)total_size);
fd = open(filename,
O_WRONLY | O_CREAT | O_TRUNC | O_BINARY | O_LARGEFILE,
0644);
if (fd < 0)
return -1;
memset(&header, 0, sizeof(header));
header.magic = cpu_to_be32(QCOW_MAGIC);
header.version = cpu_to_be32(QCOW_VERSION);
/*Create extended header fields*/
exthdr.xmagic = cpu_to_be32(XEN_MAGIC);
header_size = sizeof(header) + sizeof(QCowHeader_ext);
backing_filename_len = 0;
size = (total_size >> SECTOR_SHIFT);
if (backing_file) {
if (strcmp(backing_file, "fat:")) {
const char *p;
/* XXX: this is a hack: we do not attempt to
*check for URL like syntax */
p = strchr(backing_file, ':');
if (p && (p - backing_file) >= 2) {
/* URL like but exclude "c:" like filenames */
strncpy(backing_filename, backing_file,
sizeof(backing_filename));
} else {
if (realpath(backing_file, backing_filename) == NULL ||
stat(backing_filename, &st) != 0) {
return -1;
}
}
header.backing_file_offset = cpu_to_be64(header_size);
backing_filename_len = strlen(backing_filename);
header.backing_file_size = cpu_to_be32(
backing_filename_len);
header_size += backing_filename_len;
/*Set to the backing file size*/
if(get_filesize(backing_filename, &size, &st)) {
return -1;
}
DPRINTF("Backing file size detected: %"PRId64" sectors"
"(total %"PRId64" [%"PRId64" MB])\n",
(long long)size,
(long long)(size << SECTOR_SHIFT),
(long long)(size >> 11));
} else {
backing_file = NULL;
DPRINTF("Setting file size: %"PRId64" (total %"PRId64")\n",
(long long) total_size,
(long long) (total_size << SECTOR_SHIFT));
}
header.mtime = cpu_to_be32(st.st_mtime);
header.cluster_bits = 9; /* 512 byte cluster to avoid copying
unmodifyed sectors */
header.l2_bits = 12; /* 32 KB L2 tables */
exthdr.min_cluster_alloc = cpu_to_be32(1);
} else {
DPRINTF("Setting file size: %"PRId64" sectors"
"(total %"PRId64" [%"PRId64" MB])\n",
(long long) size,
(long long) (size << SECTOR_SHIFT),
(long long) (size >> 11));
header.cluster_bits = 12; /* 4 KB clusters */
header.l2_bits = 9; /* 4 KB L2 tables */
exthdr.min_cluster_alloc = cpu_to_be32(1 << 9);
}
/*Set the header size value*/
header.size = cpu_to_be64(size * 512);
header_size = (header_size + 7) & ~7;
if (header_size % 4096 > 0) {
header_size = ((header_size >> 12) + 1) << 12;
}
shift = header.cluster_bits + header.l2_bits;
l1_size = ((size * 512) + (1LL << shift) - 1) >> shift;
header.l1_table_offset = cpu_to_be64(header_size);
DPRINTF("L1 Table offset: %d, size %d\n",
header_size,
(int)(l1_size * sizeof(uint64_t)));
header.crypt_method = cpu_to_be32(QCOW_CRYPT_NONE);
ptr = calloc(1, l1_size * sizeof(uint64_t));
exthdr.cksum = cpu_to_be32(gen_cksum(ptr, l1_size * sizeof(uint64_t)));
printf("Created cksum: %d\n",exthdr.cksum);
free(ptr);
/*adjust file length to system page size boundary*/
length = ROUNDUP(header_size + (l1_size * sizeof(uint64_t)),
getpagesize());
if (qtruncate(fd, length, 0)!=0) {
DPRINTF("ERROR truncating file\n");
return -1;
}
if (sparse == 0) {
/*Filesize is length+l1_size*(1 << s->l2_bits)+(size*512)*/
total_length = length + (l1_size * (1 << 9)) + (size * 512);
if (qtruncate(fd, total_length, 0)!=0) {
DPRINTF("ERROR truncating file\n");
return -1;
}
printf("File truncated to length %"PRIu64"\n",total_length);
} else
flags = SPARSE_FILE;
flags |= EXTHDR_L1_BIG_ENDIAN;
exthdr.flags = cpu_to_be32(flags);
/* write all the data */
lseek(fd, 0, SEEK_SET);
ret += write(fd, &header, sizeof(header));
ret += write(fd, &exthdr, sizeof(exthdr));
if (backing_file)
ret += write(fd, backing_filename, backing_filename_len);
lseek(fd, header_size, SEEK_SET);
tmp = 0;
for (i = 0;i < l1_size; i++) {
ret += write(fd, &tmp, sizeof(tmp));
}
close(fd);
return 0;
}
static int qcow_make_empty(struct tdqcow_state *s)
{
uint32_t l1_length = s->l1_size * sizeof(uint64_t);
memset(s->l1_table, 0, l1_length);
lseek(s->fd, s->l1_table_offset, SEEK_SET);
if (write(s->fd, s->l1_table, l1_length) < 0)
return -1;
if (qtruncate(s->fd, s->l1_table_offset + l1_length, s->sparse)!=0) {
DPRINTF("ERROR truncating file\n");
return -1;
}
memset(s->l2_cache, 0, s->l2_size * L2_CACHE_SIZE * sizeof(uint64_t));
memset(s->l2_cache_offsets, 0, L2_CACHE_SIZE * sizeof(uint64_t));
memset(s->l2_cache_counts, 0, L2_CACHE_SIZE * sizeof(uint32_t));
return 0;
}
static int qcow_get_cluster_size(struct tdqcow_state *s)
{
return s->cluster_size;
}
/* XXX: put compressed sectors first, then all the cluster aligned
tables to avoid losing bytes in alignment */
static int qcow_compress_cluster(struct tdqcow_state *s, int64_t sector_num,
const uint8_t *buf)
{
z_stream strm;
int ret, out_len;
uint8_t *out_buf;
uint64_t cluster_offset;
out_buf = malloc(s->cluster_size + (s->cluster_size / 1000) + 128);
if (!out_buf)
return -1;
/* best compression, small window, no zlib header */
memset(&strm, 0, sizeof(strm));
ret = deflateInit2(&strm, Z_DEFAULT_COMPRESSION,
Z_DEFLATED, -12,
9, Z_DEFAULT_STRATEGY);
if (ret != 0) {
free(out_buf);
return -1;
}
strm.avail_in = s->cluster_size;
strm.next_in = (uint8_t *)buf;
strm.avail_out = s->cluster_size;
strm.next_out = out_buf;
ret = deflate(&strm, Z_FINISH);
if (ret != Z_STREAM_END && ret != Z_OK) {
free(out_buf);
deflateEnd(&strm);
return -1;
}
out_len = strm.next_out - out_buf;
deflateEnd(&strm);
if (ret != Z_STREAM_END || out_len >= s->cluster_size) {
/* could not compress: write normal cluster */
//tdqcow_queue_write(bs, sector_num, buf, s->cluster_sectors);
} else {
cluster_offset = get_cluster_offset(s, sector_num << 9, 2,
out_len, 0, 0);
cluster_offset &= s->cluster_offset_mask;
lseek(s->fd, cluster_offset, SEEK_SET);
if (write(s->fd, out_buf, out_len) != out_len) {
free(out_buf);
return -1;
}
}
free(out_buf);
return 0;
}
static int
tdqcow_get_image_type(const char *file, int *type)
{
int fd;
size_t size;
QCowHeader header;
fd = open(file, O_RDONLY);
if (fd == -1)
return -errno;
size = read(fd, &header, sizeof(header));
close(fd);
if (size != sizeof(header))
return (errno ? -errno : -EIO);
be32_to_cpus(&header.magic);
if (header.magic == QCOW_MAGIC)
*type = DISK_TYPE_QCOW;
else
*type = DISK_TYPE_AIO;
return 0;
}
int tdqcow_get_parent_id(td_driver_t *driver, td_disk_id_t *id)
{
off_t off;
char *buf, *filename;
int len, secs, type, err = -EINVAL;
struct tdqcow_state *child = (struct tdqcow_state *)driver->data;
if (!child->backing_file_offset)
return TD_NO_PARENT;
/* read the backing file name */
len = child->backing_file_size;
off = child->backing_file_offset - (child->backing_file_offset % 512);
secs = (len + (child->backing_file_offset - off) + 511) >> 9;
if (posix_memalign((void **)&buf, 512, secs << 9))
return -1;
if (lseek(child->fd, off, SEEK_SET) == (off_t)-1)
goto out;
if (read(child->fd, buf, secs << 9) != secs << 9)
goto out;
filename = buf + (child->backing_file_offset - off);
filename[len] = '\0';
if (tdqcow_get_image_type(filename, &type))
goto out;
id->name = strdup(filename);
id->drivertype = type;
err = 0;
out:
free(buf);
return err;
}
int tdqcow_validate_parent(td_driver_t *driver,
td_driver_t *pdriver, td_flag_t flags)
{
struct stat stats;
uint64_t psize, csize;
struct tdqcow_state *c = (struct tdqcow_state *)driver->data;
struct tdqcow_state *p = (struct tdqcow_state *)pdriver->data;
if (stat(p->name, &stats))
return -EINVAL;
if (get_filesize(p->name, &psize, &stats))
return -EINVAL;
if (stat(c->name, &stats))
return -EINVAL;
if (get_filesize(c->name, &csize, &stats))
return -EINVAL;
if (csize != psize)
return -EINVAL;
return 0;
}
struct tap_disk tapdisk_qcow = {
.disk_type = "tapdisk_qcow",
.flags = 0,
.private_data_size = sizeof(struct tdqcow_state),
.td_open = tdqcow_open,
.td_close = tdqcow_close,
.td_queue_read = tdqcow_queue_read,
.td_queue_write = tdqcow_queue_write,
.td_get_parent_id = tdqcow_get_parent_id,
.td_validate_parent = tdqcow_validate_parent,
.td_debug = NULL,
};
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -