xref: /kvmtool/disk/core.c (revision c1ed214e170343d8ca46770a36ea629bfa2d99ba)
19f532d00SPekka Enberg #include "kvm/disk-image.h"
286835cedSPrasad Joshi #include "kvm/qcow.h"
39f532d00SPekka Enberg 
47d22135fSAsias He struct disk_image *disk_image__new(int fd, u64 size, struct disk_image_operations *ops, int use_mmap)
59f532d00SPekka Enberg {
643835ac9SSasha Levin 	struct disk_image *disk;
79f532d00SPekka Enberg 
843835ac9SSasha Levin 	disk		= malloc(sizeof *disk);
943835ac9SSasha Levin 	if (!disk)
10fffb37a9SPekka Enberg 		return NULL;
119f532d00SPekka Enberg 
1243835ac9SSasha Levin 	disk->fd	= fd;
1343835ac9SSasha Levin 	disk->size	= size;
1443835ac9SSasha Levin 	disk->ops	= ops;
15f4ff38dfSPrasad Joshi 
167d22135fSAsias He 	if (use_mmap == DISK_IMAGE_MMAP) {
177d22135fSAsias He 		/*
187d22135fSAsias He 		 * The write to disk image will be discarded
197d22135fSAsias He 		 */
2037c34ca8SSasha Levin 		disk->priv = mmap(NULL, size, PROT_RW, MAP_PRIVATE | MAP_NORESERVE, fd, 0);
2143835ac9SSasha Levin 		if (disk->priv == MAP_FAILED)
229ac38fe1SSasha Levin 			die("mmap() failed");
237d22135fSAsias He 	}
247d22135fSAsias He 
2543835ac9SSasha Levin 	return disk;
269f532d00SPekka Enberg }
279f532d00SPekka Enberg 
289ac38fe1SSasha Levin struct disk_image *disk_image__open(const char *filename, bool readonly)
29499f3bedSPekka Enberg {
3043835ac9SSasha Levin 	struct disk_image *disk;
316ec60cfdSPekka Enberg 	struct stat st;
32499f3bedSPekka Enberg 	int fd;
33499f3bedSPekka Enberg 
346ec60cfdSPekka Enberg 	if (stat(filename, &st) < 0)
356ec60cfdSPekka Enberg 		return NULL;
366ec60cfdSPekka Enberg 
37441767feSAsias He 	/* blk device ?*/
38441767feSAsias He 	disk		= blkdev__probe(filename, &st);
39441767feSAsias He 	if (disk)
40441767feSAsias He 		return disk;
416ec60cfdSPekka Enberg 
429ac38fe1SSasha Levin 	fd		= open(filename, readonly ? O_RDONLY : O_RDWR);
43499f3bedSPekka Enberg 	if (fd < 0)
44499f3bedSPekka Enberg 		return NULL;
45499f3bedSPekka Enberg 
46441767feSAsias He 	/* qcow image ?*/
47f10860caSPekka Enberg 	disk		= qcow_probe(fd, readonly);
4843835ac9SSasha Levin 	if (disk)
4943835ac9SSasha Levin 		return disk;
5086835cedSPrasad Joshi 
51441767feSAsias He 	/* raw image ?*/
5243835ac9SSasha Levin 	disk		= raw_image__probe(fd, &st, readonly);
5343835ac9SSasha Levin 	if (disk)
5443835ac9SSasha Levin 		return disk;
55499f3bedSPekka Enberg 
56d6c58e5bSPrasad Joshi 	if (close(fd) < 0)
574542f276SCyrill Gorcunov 		pr_warning("close() failed");
58499f3bedSPekka Enberg 
59499f3bedSPekka Enberg 	return NULL;
60499f3bedSPekka Enberg }
61499f3bedSPekka Enberg 
62*c1ed214eSPrasad Joshi struct disk_image **disk_image__open_all(const char **filenames, bool *readonly, int count)
63*c1ed214eSPrasad Joshi {
64*c1ed214eSPrasad Joshi 	struct disk_image **disks;
65*c1ed214eSPrasad Joshi 	int i;
66*c1ed214eSPrasad Joshi 
67*c1ed214eSPrasad Joshi 	if (!count || count > MAX_DISK_IMAGES)
68*c1ed214eSPrasad Joshi 		return NULL;
69*c1ed214eSPrasad Joshi 
70*c1ed214eSPrasad Joshi 	disks = calloc(count, sizeof(*disks));
71*c1ed214eSPrasad Joshi 	if (!disks)
72*c1ed214eSPrasad Joshi 		return NULL;
73*c1ed214eSPrasad Joshi 
74*c1ed214eSPrasad Joshi 	for (i = 0; i < count; i++) {
75*c1ed214eSPrasad Joshi 		if (!filenames[i])
76*c1ed214eSPrasad Joshi 			continue;
77*c1ed214eSPrasad Joshi 
78*c1ed214eSPrasad Joshi 		disks[i] = disk_image__open(filenames[i], readonly[i]);
79*c1ed214eSPrasad Joshi 		if (!disks[i]) {
80*c1ed214eSPrasad Joshi 			pr_error("Loading disk image '%s' failed", filenames[i]);
81*c1ed214eSPrasad Joshi 			goto error;
82*c1ed214eSPrasad Joshi 		}
83*c1ed214eSPrasad Joshi 	}
84*c1ed214eSPrasad Joshi 	return disks;
85*c1ed214eSPrasad Joshi error:
86*c1ed214eSPrasad Joshi 	for (i = 0; i < count; i++)
87*c1ed214eSPrasad Joshi 		disk_image__close(disks[i]);
88*c1ed214eSPrasad Joshi 
89*c1ed214eSPrasad Joshi 	free(disks);
90*c1ed214eSPrasad Joshi 	return NULL;
91*c1ed214eSPrasad Joshi }
92*c1ed214eSPrasad Joshi 
93fda63751SAsias He int disk_image__flush(struct disk_image *disk)
94fda63751SAsias He {
95fda63751SAsias He 	if (disk->ops->flush)
96fda63751SAsias He 		return disk->ops->flush(disk);
97fda63751SAsias He 
98fda63751SAsias He 	return fsync(disk->fd);
99fda63751SAsias He }
100fda63751SAsias He 
10172133dd2SAsias He int disk_image__close(struct disk_image *disk)
102499f3bedSPekka Enberg {
10382d94c50SIngo Molnar 	/* If there was no disk image then there's nothing to do: */
10443835ac9SSasha Levin 	if (!disk)
10572133dd2SAsias He 		return 0;
10682d94c50SIngo Molnar 
10743835ac9SSasha Levin 	if (disk->ops->close)
10872133dd2SAsias He 		return disk->ops->close(disk);
109499f3bedSPekka Enberg 
11043835ac9SSasha Levin 	if (close(disk->fd) < 0)
1114542f276SCyrill Gorcunov 		pr_warning("close() failed");
112499f3bedSPekka Enberg 
11343835ac9SSasha Levin 	free(disk);
11472133dd2SAsias He 
11572133dd2SAsias He 	return 0;
116499f3bedSPekka Enberg }
11770b0d7b0SSasha Levin 
11897a90170SAsias He /*
11997a90170SAsias He  * Fill iov with disk data, starting from sector 'sector'.
12097a90170SAsias He  * Return amount of bytes read.
12197a90170SAsias He  */
122b8861977SAsias He ssize_t disk_image__read(struct disk_image *disk, u64 sector, const struct iovec *iov, int iovcount)
12370b0d7b0SSasha Levin {
12497a90170SAsias He 	ssize_t total = 0;
12597a90170SAsias He 	ssize_t nr;
12670b0d7b0SSasha Levin 
12797a90170SAsias He 	if (disk->ops->read_sector_iov) {
12897a90170SAsias He 		/*
12997a90170SAsias He 		 * Try mulitple buffer based operation first
13097a90170SAsias He 		 */
13197a90170SAsias He 		total		= disk->ops->read_sector_iov(disk, sector, iov, iovcount);
13297a90170SAsias He 		if (total < 0) {
1334542f276SCyrill Gorcunov 			pr_info("disk_image__read error: total=%ld\n", (long)total);
13470b0d7b0SSasha Levin 			return -1;
13597a90170SAsias He 		}
13697a90170SAsias He 	} else if (disk->ops->read_sector) {
13797a90170SAsias He 		/*
13897a90170SAsias He 		 * Fallback to single buffer based operation
13997a90170SAsias He 		 */
14097a90170SAsias He 		while (iovcount--) {
14197a90170SAsias He 			nr 	= disk->ops->read_sector(disk, sector, iov->iov_base, iov->iov_len);
14297a90170SAsias He 			if (nr != (ssize_t)iov->iov_len) {
1434542f276SCyrill Gorcunov 				pr_info("disk_image__read error: nr = %ld iov_len=%ld\n", (long)nr, (long)iov->iov_len);
14497a90170SAsias He 				return -1;
14597a90170SAsias He 			}
14670b0d7b0SSasha Levin 			sector	+= iov->iov_len >> SECTOR_SHIFT;
14770b0d7b0SSasha Levin 			iov++;
14897a90170SAsias He 			total 	+= nr;
14997a90170SAsias He 		}
15097a90170SAsias He 	} else
15197a90170SAsias He 		die("No disk image operation for read\n");
15297a90170SAsias He 
15397a90170SAsias He 	return total;
15470b0d7b0SSasha Levin }
15570b0d7b0SSasha Levin 
15697a90170SAsias He /*
15797a90170SAsias He  * Write iov to disk, starting from sector 'sector'.
15897a90170SAsias He  * Return amount of bytes written.
15997a90170SAsias He  */
160b8861977SAsias He ssize_t disk_image__write(struct disk_image *disk, u64 sector, const struct iovec *iov, int iovcount)
16170b0d7b0SSasha Levin {
16297a90170SAsias He 	ssize_t total = 0;
16397a90170SAsias He 	ssize_t nr;
16470b0d7b0SSasha Levin 
16597a90170SAsias He 	if (disk->ops->write_sector_iov) {
16697a90170SAsias He 		/*
16797a90170SAsias He 		 * Try writev based operation first
16897a90170SAsias He 		 */
16997a90170SAsias He 		total = disk->ops->write_sector_iov(disk, sector, iov, iovcount);
17097a90170SAsias He 		if (total < 0) {
1714542f276SCyrill Gorcunov 			pr_info("disk_image__write error: total=%ld\n", (long)total);
17270b0d7b0SSasha Levin 			return -1;
17397a90170SAsias He 		}
17497a90170SAsias He 	} else if (disk->ops->write_sector) {
17597a90170SAsias He 		/*
17697a90170SAsias He 		 * Fallback to single buffer based operation
17797a90170SAsias He 		 */
17897a90170SAsias He 		while (iovcount--) {
17997a90170SAsias He 			nr	 = disk->ops->write_sector(disk, sector, iov->iov_base, iov->iov_len);
18097a90170SAsias He 			if (nr != (ssize_t)iov->iov_len) {
1814542f276SCyrill Gorcunov 				pr_info("disk_image__write error: nr=%ld iov_len=%ld\n", (long)nr, (long)iov->iov_len);
18297a90170SAsias He 				return -1;
18397a90170SAsias He 			}
18470b0d7b0SSasha Levin 
18570b0d7b0SSasha Levin 			sector	+= iov->iov_len >> SECTOR_SHIFT;
18670b0d7b0SSasha Levin 			iov++;
18797a90170SAsias He 			total	+= nr;
18870b0d7b0SSasha Levin 		}
18997a90170SAsias He 	} else
19097a90170SAsias He 		die("No disk image operation for read\n");
19170b0d7b0SSasha Levin 
19297a90170SAsias He 	return total;
19370b0d7b0SSasha Levin }
194