blob: 1a704b3de9d6cdea9d46ef5a274cf56594da8372 [file] [log] [blame]
#include "kvm/disk-image.h"
#include "kvm/read-write.h"
#include "kvm/qcow.h"
#include "kvm/util.h"
#include <linux/fs.h> /* for BLKGETSIZE64 */
#include <sys/ioctl.h>
#include <sys/types.h>
#include <linux/types.h>
#include <sys/mman.h>
#include <sys/stat.h>
#include <stdbool.h>
#include <stddef.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
struct disk_image *disk_image__new(int fd, u64 size, struct disk_image_operations *ops)
{
struct disk_image *disk;
disk = malloc(sizeof *disk);
if (!disk)
return NULL;
disk->fd = fd;
disk->size = size;
disk->ops = ops;
return disk;
}
struct disk_image *disk_image__new_readonly(int fd, u64 size, struct disk_image_operations *ops)
{
struct disk_image *disk;
disk = disk_image__new(fd, size, ops);
if (!disk)
return NULL;
disk->priv = mmap(NULL, size, PROT_RW, MAP_PRIVATE | MAP_NORESERVE, fd, 0);
if (disk->priv == MAP_FAILED)
die("mmap() failed");
return disk;
}
static ssize_t raw_image__read_sector_iov(struct disk_image *disk, u64 sector, const struct iovec *iov, int iovcount)
{
u64 offset = sector << SECTOR_SHIFT;
return preadv_in_full(disk->fd, iov, iovcount, offset);
}
static ssize_t raw_image__write_sector_iov(struct disk_image *disk, u64 sector, const struct iovec *iov, int iovcount)
{
u64 offset = sector << SECTOR_SHIFT;
return pwritev_in_full(disk->fd, iov, iovcount, offset);
}
static int raw_image__read_sector_ro_mmap(struct disk_image *disk, u64 sector, void *dst, u32 dst_len)
{
u64 offset = sector << SECTOR_SHIFT;
if (offset + dst_len > disk->size)
return -1;
memcpy(dst, disk->priv + offset, dst_len);
return 0;
}
static int raw_image__write_sector_ro_mmap(struct disk_image *disk, u64 sector, void *src, u32 src_len)
{
u64 offset = sector << SECTOR_SHIFT;
if (offset + src_len > disk->size)
return -1;
memcpy(disk->priv + offset, src, src_len);
return 0;
}
static void raw_image__close_ro_mmap(struct disk_image *disk)
{
if (disk->priv != MAP_FAILED)
munmap(disk->priv, disk->size);
}
static struct disk_image_operations raw_image_ops = {
.read_sector_iov = raw_image__read_sector_iov,
.write_sector_iov = raw_image__write_sector_iov
};
static struct disk_image_operations raw_image_ro_mmap_ops = {
.read_sector = raw_image__read_sector_ro_mmap,
.write_sector = raw_image__write_sector_ro_mmap,
.close = raw_image__close_ro_mmap,
};
static struct disk_image *raw_image__probe(int fd, struct stat *st, bool readonly)
{
if (readonly)
return disk_image__new_readonly(fd, st->st_size, &raw_image_ro_mmap_ops);
else
return disk_image__new(fd, st->st_size, &raw_image_ops);
}
static struct disk_image *blkdev__probe(const char *filename, struct stat *st)
{
u64 size;
int fd;
if (!S_ISBLK(st->st_mode))
return NULL;
fd = open(filename, O_RDONLY);
if (fd < 0)
return NULL;
if (ioctl(fd, BLKGETSIZE64, &size) < 0) {
close(fd);
return NULL;
}
return disk_image__new_readonly(fd, size, &raw_image_ro_mmap_ops);
}
struct disk_image *disk_image__open(const char *filename, bool readonly)
{
struct disk_image *disk;
struct stat st;
int fd;
if (stat(filename, &st) < 0)
return NULL;
if (S_ISBLK(st.st_mode))
return blkdev__probe(filename, &st);
fd = open(filename, readonly ? O_RDONLY : O_RDWR);
if (fd < 0)
return NULL;
disk = qcow_probe(fd, readonly);
if (disk)
return disk;
disk = raw_image__probe(fd, &st, readonly);
if (disk)
return disk;
if (close(fd) < 0)
warning("close() failed");
return NULL;
}
void disk_image__close(struct disk_image *disk)
{
/* If there was no disk image then there's nothing to do: */
if (!disk)
return;
if (disk->ops->close)
disk->ops->close(disk);
if (close(disk->fd) < 0)
warning("close() failed");
free(disk);
}