⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 block-qcow.c

📁 xen 3.2.2 源码
💻 C
📖 第 1 页 / 共 3 页
字号:
/* block-qcow.c * * Asynchronous Qemu copy-on-write disk implementation. * Code based on the Qemu implementation * (see copyright notice below) * * (c) 2006 Andrew Warfield and Julian Chesterfield * *//* * Block driver for the QCOW format *  * Copyright (c) 2004 Fabrice Bellard *  * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files(the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: */#include <errno.h>#include <fcntl.h>#include <stdio.h>#include <stdlib.h>#include <unistd.h>#include <sys/statvfs.h>#include <sys/stat.h>#include <sys/ioctl.h>#include <linux/fs.h>#include <string.h>#include <zlib.h>#include <inttypes.h>#include <libaio.h>#include <openssl/md5.h>#include "bswap.h"#include "aes.h"#include "tapdisk.h"#include "tapaio.h"#if 1#define ASSERT(_p) \    if ( !(_p) ) { DPRINTF("Assertion '%s' failed, line %d, file %s", #_p , \    __LINE__, __FILE__); *(int*)0=0; }#else#define ASSERT(_p) ((void)0)#endif#define ROUNDUP(l, s) \({ \    (uint64_t)( \        (l + (s - 1)) - ((l + (s - 1)) % s)); \})struct pending_aio {        td_callback_t cb;        int id;        void *private;	int nb_sectors;	char *buf;	uint64_t sector;};#define IOCB_IDX(_s, _io) ((_io) - (_s)->iocb_list)#define ZERO_TEST(_b) (_b | 0x00)/**************************************************************//* QEMU COW block driver with compression and encryption support */#define QCOW_MAGIC (('Q' << 24) | ('F' << 16) | ('I' << 8) | 0xfb)#define XEN_MAGIC  (('X' << 24) | ('E' << 16) | ('N' << 8) | 0xfb)#define QCOW_VERSION 1#define QCOW_CRYPT_NONE 0x00#define QCOW_CRYPT_AES  0x01#define QCOW_OFLAG_COMPRESSED (1LL << 63)#define SPARSE_FILE 0x01#ifndef O_BINARY#define O_BINARY 0#endiftypedef struct QCowHeader {	uint32_t magic;	uint32_t version;	uint64_t backing_file_offset;	uint32_t backing_file_size;	uint32_t mtime;	uint64_t size; /* in bytes */	uint8_t cluster_bits;	uint8_t l2_bits;	uint32_t crypt_method;	uint64_t l1_table_offset;} QCowHeader;/*Extended header for Xen enhancements*/typedef struct QCowHeader_ext {        uint32_t xmagic;        uint32_t cksum;        uint32_t min_cluster_alloc;        uint32_t flags;} QCowHeader_ext;#define L2_CACHE_SIZE 16  /*Fixed allocation in Qemu*/struct tdqcow_state {        int fd;                        /*Main Qcow file descriptor */	uint64_t fd_end;               /*Store a local record of file length */	char *name;                    /*Record of the filename*/	uint32_t backing_file_size;	uint64_t backing_file_offset;	int encrypted;                 /*File contents are encrypted or plain*/	int cluster_bits;              /*Determines length of cluster as 					*indicated by file hdr*/	int cluster_size;              /*Length of cluster*/	int cluster_sectors;           /*Number of sectors per cluster*/	int cluster_alloc;             /*Blktap fix for allocating full 					*extents*/	int min_cluster_alloc;         /*Blktap historical extent alloc*/	int sparse;                    /*Indicates whether to preserve sparseness*/	int l2_bits;                   /*Size of L2 table entry*/	int l2_size;                   /*Full table size*/	int l1_size;                   /*L1 table size*/	uint64_t cluster_offset_mask;    	uint64_t l1_table_offset;      /*L1 table offset from beginning of 					*file*/	uint64_t *l1_table;            /*L1 table entries*/	uint64_t *l2_cache;            /*We maintain a cache of size 					*L2_CACHE_SIZE of most read entries*/	uint64_t l2_cache_offsets[L2_CACHE_SIZE];     /*L2 cache entries*/	uint32_t l2_cache_counts[L2_CACHE_SIZE];      /*Cache access record*/	uint8_t *cluster_cache;          	uint8_t *cluster_data;	uint8_t *sector_lock;          /*Locking bitmap for AIO reads/writes*/	uint64_t cluster_cache_offset; /**/	uint32_t crypt_method;         /*current crypt method, 0 if no 					*key yet */	uint32_t crypt_method_header;  /**/	AES_KEY aes_encrypt_key;       /*AES key*/	AES_KEY aes_decrypt_key;       /*AES key*/        /* libaio state */        tap_aio_context_t   aio_ctx;        int                 max_aio_reqs;        struct iocb        *iocb_list;        struct iocb       **iocb_free;        struct pending_aio *pending_aio;        int                 iocb_free_count;        struct iocb       **iocb_queue;        int                 iocb_queued;        struct io_event    *aio_events;};static int decompress_cluster(struct tdqcow_state *s, uint64_t cluster_offset);static void free_aio_state(struct disk_driver *dd){        struct tdqcow_state *s = (struct tdqcow_state *)dd->private;        if (s->sector_lock)                free(s->sector_lock);        if (s->iocb_list)                free(s->iocb_list);        if (s->pending_aio)                free(s->pending_aio);        if (s->aio_events)                free(s->aio_events);        if (s->iocb_free)                free(s->iocb_free);        if (s->iocb_queue)                free(s->iocb_queue);}static int init_aio_state(struct disk_driver *dd){	int i, ret;	struct td_state     *bs = dd->td_state;	struct tdqcow_state  *s = (struct tdqcow_state *)dd->private;        long     ioidx;        s->iocb_list = NULL;        s->pending_aio = NULL;        s->aio_events = NULL;        s->iocb_free = NULL;        s->iocb_queue = NULL;        /*Initialize Locking bitmap*/	s->sector_lock = calloc(1, bs->size);		if (!s->sector_lock) {		DPRINTF("Failed to allocate sector lock\n");		goto fail;	}        /* A segment (i.e. a page) can span multiple clusters */        s->max_aio_reqs = ((getpagesize() / s->cluster_size) + 1) *            MAX_SEGMENTS_PER_REQ * MAX_REQUESTS;        /* Initialize AIO */        s->iocb_free_count = s->max_aio_reqs;        s->iocb_queued     = 0;        if (!(s->iocb_list = malloc(sizeof(struct iocb) * s->max_aio_reqs)) ||            !(s->pending_aio = malloc(sizeof(struct pending_aio) * s->max_aio_reqs)) ||            !(s->aio_events = malloc(sizeof(struct io_event) * s->max_aio_reqs)) ||            !(s->iocb_free = malloc(sizeof(struct iocb *) * s->max_aio_reqs)) ||            !(s->iocb_queue = malloc(sizeof(struct iocb *) * s->max_aio_reqs))) {                DPRINTF("Failed to allocate AIO structs (max_aio_reqs = %d)\n",                        s->max_aio_reqs);                goto fail;        }	ret = tap_aio_setup(&s->aio_ctx, s->aio_events, s->max_aio_reqs);	if (ret < 0) {                if (ret == -EAGAIN) {                        DPRINTF("Couldn't setup AIO context.  If you are "                                "trying to concurrently use a large number "                                "of blktap-based disks, you may need to "                                "increase the system-wide aio request limit. "                                "(e.g. 'echo echo 1048576 > /proc/sys/fs/"                                "aio-max-nr')\n");                } else {                        DPRINTF("Couldn't setup AIO context.\n");                }		goto fail;	}        for (i=0;i<s->max_aio_reqs;i++)                s->iocb_free[i] = &s->iocb_list[i];        DPRINTF("AIO state initialised\n");        return 0; fail:	return -1;}static uint32_t gen_cksum(char *ptr, int len){	unsigned char *md;	uint32_t ret;	md = malloc(MD5_DIGEST_LENGTH);	if(!md) return 0;	if (MD5((unsigned char *)ptr, len, md) != md) {		free(md);		return 0;	}	memcpy(&ret, md, sizeof(uint32_t));	free(md);	return ret;}static int get_filesize(char *filename, uint64_t *size, struct stat *st){	int fd;	QCowHeader header;	/*Set to the backing file size*/	fd = open(filename, O_RDONLY);	if (fd < 0)		return -1;	if (read(fd, &header, sizeof(header)) < sizeof(header)) {		close(fd);		return -1;	}	close(fd);		be32_to_cpus(&header.magic);	be64_to_cpus(&header.size);	if (header.magic == QCOW_MAGIC) {		*size = header.size >> SECTOR_SHIFT;		return 0;	}	if(S_ISBLK(st->st_mode)) {		fd = open(filename, O_RDONLY);		if (fd < 0)			return -1;		if (ioctl(fd,BLKGETSIZE,size)!=0) {			printf("Unable to get Block device size\n");			close(fd);			return -1;		}		close(fd);	} else *size = (st->st_size >> SECTOR_SHIFT);		return 0;}static int qcow_set_key(struct tdqcow_state *s, const char *key){	uint8_t keybuf[16];	int len, i;		memset(keybuf, 0, 16);	len = strlen(key);	if (len > 16)		len = 16;	/* XXX: we could compress the chars to 7 bits to increase	   entropy */	for (i = 0; i < len; i++) {		keybuf[i] = key[i];	}	s->crypt_method = s->crypt_method_header;		if (AES_set_encrypt_key(keybuf, 128, &s->aes_encrypt_key) != 0)		return -1;	if (AES_set_decrypt_key(keybuf, 128, &s->aes_decrypt_key) != 0)		return -1;#if 0	/* test */	{		uint8_t in[16];		uint8_t out[16];		uint8_t tmp[16];		for (i=0; i<16; i++)			in[i] = i;		AES_encrypt(in, tmp, &s->aes_encrypt_key);		AES_decrypt(tmp, out, &s->aes_decrypt_key);		for (i = 0; i < 16; i++)			DPRINTF(" %02x", tmp[i]);		DPRINTF("\n");		for (i = 0; i < 16; i++)			DPRINTF(" %02x", out[i]);		DPRINTF("\n");	}#endif	return 0;}static int async_read(struct tdqcow_state *s, int size, 		      uint64_t offset, char *buf, td_callback_t cb,		      int id, uint64_t sector, void *private){        struct   iocb *io;        struct   pending_aio *pio;	long     ioidx;        io = s->iocb_free[--s->iocb_free_count];        ioidx = IOCB_IDX(s, io);        pio = &s->pending_aio[ioidx];        pio->cb = cb;        pio->id = id;        pio->private = private;	pio->nb_sectors = size/512;	pio->buf = buf;	pio->sector = sector;        io_prep_pread(io, s->fd, buf, size, offset);        io->data = (void *)ioidx;        s->iocb_queue[s->iocb_queued++] = io;        return 1;}static int async_write(struct tdqcow_state *s, int size,		       uint64_t offset, char *buf, td_callback_t cb,		       int id, uint64_t sector, void *private){        struct   iocb *io;        struct   pending_aio *pio;	long     ioidx;        io = s->iocb_free[--s->iocb_free_count];        ioidx = IOCB_IDX(s, io);        pio = &s->pending_aio[ioidx];        pio->cb = cb;        pio->id = id;        pio->private = private;	pio->nb_sectors = size/512;	pio->buf = buf;	pio->sector = sector;        io_prep_pwrite(io, s->fd, buf, size, offset);        io->data = (void *)ioidx;        s->iocb_queue[s->iocb_queued++] = io;        return 1;}/*TODO: Fix sector span!*/static int aio_can_lock(struct tdqcow_state *s, uint64_t sector){	return (s->sector_lock[sector] ? 0 : 1);}static int aio_lock(struct tdqcow_state *s, uint64_t sector){	return ++s->sector_lock[sector];}static void aio_unlock(struct tdqcow_state *s, uint64_t sector){	if (!s->sector_lock[sector]) return;	--s->sector_lock[sector];	return;}/*  * The crypt function is compatible with the linux cryptoloop * algorithm for < 4 GB images. NOTE: out_buf == in_buf is * supported . */static void encrypt_sectors(struct tdqcow_state *s, int64_t sector_num,                            uint8_t *out_buf, const uint8_t *in_buf,                            int nb_sectors, int enc,                            const AES_KEY *key){	union {		uint64_t ll[2];		uint8_t b[16];	} ivec;	int i;		for (i = 0; i < nb_sectors; i++) {		ivec.ll[0] = cpu_to_le64(sector_num);		ivec.ll[1] = 0;		AES_cbc_encrypt(in_buf, out_buf, 512, key, 				ivec.b, enc);		sector_num++;		in_buf += 512;		out_buf += 512;	}}static int qtruncate(int fd, off_t length, int sparse){	int ret, i; 	int current = 0, rem = 0;	uint64_t sectors;	struct stat st;	char *buf;	/* If length is greater than the current file len	 * we synchronously write zeroes to the end of the 	 * file, otherwise we truncate the length down	 */	ret = fstat(fd, &st);	if (ret == -1) 		return -1;	if (S_ISBLK(st.st_mode))		return 0;	sectors = (length + DEFAULT_SECTOR_SIZE - 1)/DEFAULT_SECTOR_SIZE;	current = (st.st_size + DEFAULT_SECTOR_SIZE - 1)/DEFAULT_SECTOR_SIZE;	rem     = st.st_size % DEFAULT_SECTOR_SIZE;	/* If we are extending this file, we write zeros to the end --	 * this tries to ensure that the extents allocated wind up being	 * contiguous on disk.	 */	if(st.st_size < sectors * DEFAULT_SECTOR_SIZE) {		/*We are extending the file*/		if ((ret = posix_memalign((void **)&buf, 					  512, DEFAULT_SECTOR_SIZE))) {			DPRINTF("posix_memalign failed: %d\n", ret);			return -1;		}		memset(buf, 0x00, DEFAULT_SECTOR_SIZE);		if (lseek(fd, 0, SEEK_END)==-1) {			DPRINTF("Lseek EOF failed (%d), internal error\n",				errno);			free(buf);			return -1;		}		if (rem) {			ret = write(fd, buf, rem);			if (ret != rem) {				DPRINTF("write failed: ret = %d, err = %s\n",					ret, strerror(errno));				free(buf);				return -1;			}		}		for (i = current; i < sectors; i++ ) {			ret = write(fd, buf, DEFAULT_SECTOR_SIZE);			if (ret != DEFAULT_SECTOR_SIZE) {				DPRINTF("write failed: ret = %d, err = %s\n",					ret, strerror(errno));				free(buf);				return -1;			}		}		free(buf);	} else if(sparse && (st.st_size > sectors * DEFAULT_SECTOR_SIZE))		if (ftruncate(fd, (off_t)sectors * DEFAULT_SECTOR_SIZE)==-1) {			DPRINTF("Ftruncate failed (%s)\n", strerror(errno));			return -1;		}	return 0;}

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -