xref: /kvmtool/disk/core.c (revision 30a9aa69300118907de7735f5695f8b142d74455)
19f532d00SPekka Enberg #include "kvm/disk-image.h"
286835cedSPrasad Joshi #include "kvm/qcow.h"
3f41a132bSSasha Levin #include "kvm/virtio-blk.h"
43b55dcdeSSasha Levin #include "kvm/kvm.h"
5f41a132bSSasha Levin 
69f9207c5SSasha Levin #include <linux/err.h>
752c22e6eSAndre Przywara #include <poll.h>
8f41a132bSSasha Levin 
9aa400b00SPrasad Joshi int debug_iodelay;
10aa400b00SPrasad Joshi 
113b55dcdeSSasha Levin static int disk_image__close(struct disk_image *disk);
123b55dcdeSSasha Levin 
133b55dcdeSSasha Levin int disk_img_name_parser(const struct option *opt, const char *arg, int unset)
143b55dcdeSSasha Levin {
153b55dcdeSSasha Levin 	const char *cur;
163b55dcdeSSasha Levin 	char *sep;
173b55dcdeSSasha Levin 	struct kvm *kvm = opt->ptr;
183b55dcdeSSasha Levin 
193b55dcdeSSasha Levin 	if (kvm->cfg.image_count >= MAX_DISK_IMAGES)
203b55dcdeSSasha Levin 		die("Currently only 4 images are supported");
213b55dcdeSSasha Levin 
223b55dcdeSSasha Levin 	kvm->cfg.disk_image[kvm->cfg.image_count].filename = arg;
233b55dcdeSSasha Levin 	cur = arg;
243b55dcdeSSasha Levin 
253b55dcdeSSasha Levin 	if (strncmp(arg, "scsi:", 5) == 0) {
263b55dcdeSSasha Levin 		sep = strstr(arg, ":");
273b55dcdeSSasha Levin 		if (sep)
283b55dcdeSSasha Levin 			kvm->cfg.disk_image[kvm->cfg.image_count].wwpn = sep + 1;
293b55dcdeSSasha Levin 		sep = strstr(sep + 1, ":");
303b55dcdeSSasha Levin 		if (sep) {
313b55dcdeSSasha Levin 			*sep = 0;
323b55dcdeSSasha Levin 			kvm->cfg.disk_image[kvm->cfg.image_count].tpgt = sep + 1;
333b55dcdeSSasha Levin 		}
343b55dcdeSSasha Levin 		cur = sep + 1;
353b55dcdeSSasha Levin 	}
363b55dcdeSSasha Levin 
373b55dcdeSSasha Levin 	do {
383b55dcdeSSasha Levin 		sep = strstr(cur, ",");
393b55dcdeSSasha Levin 		if (sep) {
403b55dcdeSSasha Levin 			if (strncmp(sep + 1, "ro", 2) == 0)
413b55dcdeSSasha Levin 				kvm->cfg.disk_image[kvm->cfg.image_count].readonly = true;
423b55dcdeSSasha Levin 			else if (strncmp(sep + 1, "direct", 6) == 0)
433b55dcdeSSasha Levin 				kvm->cfg.disk_image[kvm->cfg.image_count].direct = true;
443b55dcdeSSasha Levin 			*sep = 0;
453b55dcdeSSasha Levin 			cur = sep + 1;
463b55dcdeSSasha Levin 		}
473b55dcdeSSasha Levin 	} while (sep);
483b55dcdeSSasha Levin 
493b55dcdeSSasha Levin 	kvm->cfg.image_count++;
503b55dcdeSSasha Levin 
513b55dcdeSSasha Levin 	return 0;
523b55dcdeSSasha Levin }
533b55dcdeSSasha Levin 
5454c84566SAsias He struct disk_image *disk_image__new(int fd, u64 size,
5554c84566SAsias He 				   struct disk_image_operations *ops,
5654c84566SAsias He 				   int use_mmap)
579f532d00SPekka Enberg {
5843835ac9SSasha Levin 	struct disk_image *disk;
599f9207c5SSasha Levin 	int r;
609f532d00SPekka Enberg 
6143835ac9SSasha Levin 	disk = malloc(sizeof *disk);
6243835ac9SSasha Levin 	if (!disk)
639f9207c5SSasha Levin 		return ERR_PTR(-ENOMEM);
649f532d00SPekka Enberg 
659f9207c5SSasha Levin 	*disk = (struct disk_image) {
669f9207c5SSasha Levin 		.fd	= fd,
679f9207c5SSasha Levin 		.size	= size,
689f9207c5SSasha Levin 		.ops	= ops,
699f9207c5SSasha Levin 	};
70f4ff38dfSPrasad Joshi 
717d22135fSAsias He 	if (use_mmap == DISK_IMAGE_MMAP) {
727d22135fSAsias He 		/*
737d22135fSAsias He 		 * The write to disk image will be discarded
747d22135fSAsias He 		 */
7537c34ca8SSasha Levin 		disk->priv = mmap(NULL, size, PROT_RW, MAP_PRIVATE | MAP_NORESERVE, fd, 0);
769d15c39aSSasha Levin 		if (disk->priv == MAP_FAILED) {
779f9207c5SSasha Levin 			r = -errno;
78*30a9aa69SJean-Philippe Brucker 			goto err_free_disk;
799d15c39aSSasha Levin 		}
807d22135fSAsias He 	}
817d22135fSAsias He 
82*30a9aa69SJean-Philippe Brucker 	r = disk_aio_setup(disk);
83*30a9aa69SJean-Philippe Brucker 	if (r)
84*30a9aa69SJean-Philippe Brucker 		goto err_unmap_disk;
85f41a132bSSasha Levin 
8643835ac9SSasha Levin 	return disk;
87*30a9aa69SJean-Philippe Brucker 
88*30a9aa69SJean-Philippe Brucker err_unmap_disk:
89*30a9aa69SJean-Philippe Brucker 	if (disk->priv)
90*30a9aa69SJean-Philippe Brucker 		munmap(disk->priv, size);
91*30a9aa69SJean-Philippe Brucker err_free_disk:
92*30a9aa69SJean-Philippe Brucker 	free(disk);
93*30a9aa69SJean-Philippe Brucker 	return ERR_PTR(r);
949f532d00SPekka Enberg }
959f532d00SPekka Enberg 
963b55dcdeSSasha Levin static struct disk_image *disk_image__open(const char *filename, bool readonly, bool direct)
97499f3bedSPekka Enberg {
9843835ac9SSasha Levin 	struct disk_image *disk;
996ec60cfdSPekka Enberg 	struct stat st;
1005236b505SAsias He 	int fd, flags;
1015236b505SAsias He 
1025236b505SAsias He 	if (readonly)
1035236b505SAsias He 		flags = O_RDONLY;
1045236b505SAsias He 	else
1055236b505SAsias He 		flags = O_RDWR;
1065236b505SAsias He 	if (direct)
1075236b505SAsias He 		flags |= O_DIRECT;
108499f3bedSPekka Enberg 
1096ec60cfdSPekka Enberg 	if (stat(filename, &st) < 0)
1109f9207c5SSasha Levin 		return ERR_PTR(-errno);
1116ec60cfdSPekka Enberg 
112441767feSAsias He 	/* blk device ?*/
1135236b505SAsias He 	disk = blkdev__probe(filename, flags, &st);
1145c5cae75SJean-Philippe Brucker 	if (!IS_ERR_OR_NULL(disk)) {
1155c5cae75SJean-Philippe Brucker 		disk->readonly = readonly;
116441767feSAsias He 		return disk;
1175c5cae75SJean-Philippe Brucker 	}
1186ec60cfdSPekka Enberg 
1195236b505SAsias He 	fd = open(filename, flags);
120499f3bedSPekka Enberg 	if (fd < 0)
1219f9207c5SSasha Levin 		return ERR_PTR(fd);
122499f3bedSPekka Enberg 
123441767feSAsias He 	/* qcow image ?*/
12417f68274SPekka Enberg 	disk = qcow_probe(fd, true);
1259f9207c5SSasha Levin 	if (!IS_ERR_OR_NULL(disk)) {
12617f68274SPekka Enberg 		pr_warning("Forcing read-only support for QCOW");
1275c5cae75SJean-Philippe Brucker 		disk->readonly = true;
12843835ac9SSasha Levin 		return disk;
12917f68274SPekka Enberg 	}
13086835cedSPrasad Joshi 
131441767feSAsias He 	/* raw image ?*/
13243835ac9SSasha Levin 	disk = raw_image__probe(fd, &st, readonly);
1335c5cae75SJean-Philippe Brucker 	if (!IS_ERR_OR_NULL(disk)) {
1345c5cae75SJean-Philippe Brucker 		disk->readonly = readonly;
13543835ac9SSasha Levin 		return disk;
1365c5cae75SJean-Philippe Brucker 	}
137499f3bedSPekka Enberg 
138d6c58e5bSPrasad Joshi 	if (close(fd) < 0)
1394542f276SCyrill Gorcunov 		pr_warning("close() failed");
140499f3bedSPekka Enberg 
1419f9207c5SSasha Levin 	return ERR_PTR(-ENOSYS);
142499f3bedSPekka Enberg }
143499f3bedSPekka Enberg 
1443b55dcdeSSasha Levin static struct disk_image **disk_image__open_all(struct kvm *kvm)
145c1ed214eSPrasad Joshi {
146c1ed214eSPrasad Joshi 	struct disk_image **disks;
14797f16d66SAsias He 	const char *filename;
148a67da3beSAsias He 	const char *wwpn;
149a67da3beSAsias He 	const char *tpgt;
15097f16d66SAsias He 	bool readonly;
1515236b505SAsias He 	bool direct;
1529f9207c5SSasha Levin 	void *err;
15397f16d66SAsias He 	int i;
1543b55dcdeSSasha Levin 	struct disk_image_params *params = (struct disk_image_params *)&kvm->cfg.disk_image;
1553b55dcdeSSasha Levin 	int count = kvm->cfg.image_count;
156c1ed214eSPrasad Joshi 
1579f9207c5SSasha Levin 	if (!count)
1589f9207c5SSasha Levin 		return ERR_PTR(-EINVAL);
1599f9207c5SSasha Levin 	if (count > MAX_DISK_IMAGES)
1609f9207c5SSasha Levin 		return ERR_PTR(-ENOSPC);
161c1ed214eSPrasad Joshi 
162c1ed214eSPrasad Joshi 	disks = calloc(count, sizeof(*disks));
163c1ed214eSPrasad Joshi 	if (!disks)
1649f9207c5SSasha Levin 		return ERR_PTR(-ENOMEM);
165c1ed214eSPrasad Joshi 
166c1ed214eSPrasad Joshi 	for (i = 0; i < count; i++) {
16797f16d66SAsias He 		filename = params[i].filename;
16897f16d66SAsias He 		readonly = params[i].readonly;
1695236b505SAsias He 		direct = params[i].direct;
170a67da3beSAsias He 		wwpn = params[i].wwpn;
171a67da3beSAsias He 		tpgt = params[i].tpgt;
172a67da3beSAsias He 
173a67da3beSAsias He 		if (wwpn) {
174a67da3beSAsias He 			disks[i] = malloc(sizeof(struct disk_image));
175a67da3beSAsias He 			if (!disks[i])
176a67da3beSAsias He 				return ERR_PTR(-ENOMEM);
177a67da3beSAsias He 			disks[i]->wwpn = wwpn;
178a67da3beSAsias He 			disks[i]->tpgt = tpgt;
179a67da3beSAsias He 			continue;
180a67da3beSAsias He 		}
181a67da3beSAsias He 
18297f16d66SAsias He 		if (!filename)
183c1ed214eSPrasad Joshi 			continue;
184c1ed214eSPrasad Joshi 
1855236b505SAsias He 		disks[i] = disk_image__open(filename, readonly, direct);
1869f9207c5SSasha Levin 		if (IS_ERR_OR_NULL(disks[i])) {
18797f16d66SAsias He 			pr_err("Loading disk image '%s' failed", filename);
1889f9207c5SSasha Levin 			err = disks[i];
189c1ed214eSPrasad Joshi 			goto error;
190c1ed214eSPrasad Joshi 		}
1913b55dcdeSSasha Levin 		disks[i]->debug_iodelay = kvm->cfg.debug_iodelay;
192c1ed214eSPrasad Joshi 	}
193f41a132bSSasha Levin 
194c1ed214eSPrasad Joshi 	return disks;
195c1ed214eSPrasad Joshi error:
196c1ed214eSPrasad Joshi 	for (i = 0; i < count; i++)
1979f9207c5SSasha Levin 		if (!IS_ERR_OR_NULL(disks[i]))
198c1ed214eSPrasad Joshi 			disk_image__close(disks[i]);
199c1ed214eSPrasad Joshi 
200c1ed214eSPrasad Joshi 	free(disks);
2019f9207c5SSasha Levin 	return err;
202c1ed214eSPrasad Joshi }
203c1ed214eSPrasad Joshi 
204fda63751SAsias He int disk_image__flush(struct disk_image *disk)
205fda63751SAsias He {
206fda63751SAsias He 	if (disk->ops->flush)
207fda63751SAsias He 		return disk->ops->flush(disk);
208fda63751SAsias He 
209fda63751SAsias He 	return fsync(disk->fd);
210fda63751SAsias He }
211fda63751SAsias He 
2123b55dcdeSSasha Levin static int disk_image__close(struct disk_image *disk)
213499f3bedSPekka Enberg {
21482d94c50SIngo Molnar 	/* If there was no disk image then there's nothing to do: */
21543835ac9SSasha Levin 	if (!disk)
21672133dd2SAsias He 		return 0;
21782d94c50SIngo Molnar 
218*30a9aa69SJean-Philippe Brucker 	disk_aio_destroy(disk);
219*30a9aa69SJean-Philippe Brucker 
22043835ac9SSasha Levin 	if (disk->ops->close)
22172133dd2SAsias He 		return disk->ops->close(disk);
222499f3bedSPekka Enberg 
22343835ac9SSasha Levin 	if (close(disk->fd) < 0)
2244542f276SCyrill Gorcunov 		pr_warning("close() failed");
225499f3bedSPekka Enberg 
22643835ac9SSasha Levin 	free(disk);
22772133dd2SAsias He 
22872133dd2SAsias He 	return 0;
229499f3bedSPekka Enberg }
23070b0d7b0SSasha Levin 
2313b55dcdeSSasha Levin static int disk_image__close_all(struct disk_image **disks, int count)
2329df47d00SPrasad Joshi {
2339df47d00SPrasad Joshi 	while (count)
2349df47d00SPrasad Joshi 		disk_image__close(disks[--count]);
2359df47d00SPrasad Joshi 
2369df47d00SPrasad Joshi 	free(disks);
2379f9207c5SSasha Levin 
2389f9207c5SSasha Levin 	return 0;
2399df47d00SPrasad Joshi }
2409df47d00SPrasad Joshi 
24197a90170SAsias He /*
24297a90170SAsias He  * Fill iov with disk data, starting from sector 'sector'.
24397a90170SAsias He  * Return amount of bytes read.
24497a90170SAsias He  */
24554c84566SAsias He ssize_t disk_image__read(struct disk_image *disk, u64 sector,
24654c84566SAsias He 			 const struct iovec *iov, int iovcount, void *param)
24770b0d7b0SSasha Levin {
24897a90170SAsias He 	ssize_t total = 0;
24970b0d7b0SSasha Levin 
250aa400b00SPrasad Joshi 	if (debug_iodelay)
251aa400b00SPrasad Joshi 		msleep(debug_iodelay);
252aa400b00SPrasad Joshi 
253dcd3cd8eSAsias He 	if (disk->ops->read) {
254dcd3cd8eSAsias He 		total = disk->ops->read(disk, sector, iov, iovcount, param);
25597a90170SAsias He 		if (total < 0) {
2564542f276SCyrill Gorcunov 			pr_info("disk_image__read error: total=%ld\n", (long)total);
2579f9207c5SSasha Levin 			return total;
25897a90170SAsias He 		}
2592534c9b6SSasha Levin 	}
26097a90170SAsias He 
261f41a132bSSasha Levin 	if (!disk->async && disk->disk_req_cb)
2628b52f877SSasha Levin 		disk->disk_req_cb(param, total);
2635af21162SSasha Levin 
26497a90170SAsias He 	return total;
26570b0d7b0SSasha Levin }
26670b0d7b0SSasha Levin 
26797a90170SAsias He /*
26897a90170SAsias He  * Write iov to disk, starting from sector 'sector'.
26997a90170SAsias He  * Return amount of bytes written.
27097a90170SAsias He  */
27154c84566SAsias He ssize_t disk_image__write(struct disk_image *disk, u64 sector,
27254c84566SAsias He 			  const struct iovec *iov, int iovcount, void *param)
27370b0d7b0SSasha Levin {
27497a90170SAsias He 	ssize_t total = 0;
27570b0d7b0SSasha Levin 
276aa400b00SPrasad Joshi 	if (debug_iodelay)
277aa400b00SPrasad Joshi 		msleep(debug_iodelay);
278aa400b00SPrasad Joshi 
279dcd3cd8eSAsias He 	if (disk->ops->write) {
28097a90170SAsias He 		/*
28197a90170SAsias He 		 * Try writev based operation first
28297a90170SAsias He 		 */
2835af21162SSasha Levin 
284dcd3cd8eSAsias He 		total = disk->ops->write(disk, sector, iov, iovcount, param);
28597a90170SAsias He 		if (total < 0) {
2864542f276SCyrill Gorcunov 			pr_info("disk_image__write error: total=%ld\n", (long)total);
2879f9207c5SSasha Levin 			return total;
28897a90170SAsias He 		}
2892534c9b6SSasha Levin 	} else {
290b41ca15aSSasha Levin 		/* Do nothing */
2912534c9b6SSasha Levin 	}
29270b0d7b0SSasha Levin 
293f41a132bSSasha Levin 	if (!disk->async && disk->disk_req_cb)
2948b52f877SSasha Levin 		disk->disk_req_cb(param, total);
2955af21162SSasha Levin 
29697a90170SAsias He 	return total;
29770b0d7b0SSasha Levin }
298ff6462e8SSasha Levin 
299ff6462e8SSasha Levin ssize_t disk_image__get_serial(struct disk_image *disk, void *buffer, ssize_t *len)
300ff6462e8SSasha Levin {
301ff6462e8SSasha Levin 	struct stat st;
3029f9207c5SSasha Levin 	int r;
303ff6462e8SSasha Levin 
3049f9207c5SSasha Levin 	r = fstat(disk->fd, &st);
3059f9207c5SSasha Levin 	if (r)
3069f9207c5SSasha Levin 		return r;
307ff6462e8SSasha Levin 
30854c84566SAsias He 	*len = snprintf(buffer, *len, "%llu%llu%llu",
30969f50425SAndreas Herrmann 			(unsigned long long)st.st_dev,
31069f50425SAndreas Herrmann 			(unsigned long long)st.st_rdev,
31169f50425SAndreas Herrmann 			(unsigned long long)st.st_ino);
312ff6462e8SSasha Levin 	return *len;
313ff6462e8SSasha Levin }
3145af21162SSasha Levin 
31554c84566SAsias He void disk_image__set_callback(struct disk_image *disk,
31654c84566SAsias He 			      void (*disk_req_cb)(void *param, long len))
3175af21162SSasha Levin {
3185af21162SSasha Levin 	disk->disk_req_cb = disk_req_cb;
3195af21162SSasha Levin }
3203b55dcdeSSasha Levin 
3213b55dcdeSSasha Levin int disk_image__init(struct kvm *kvm)
3223b55dcdeSSasha Levin {
3233b55dcdeSSasha Levin 	if (kvm->cfg.image_count) {
3243b55dcdeSSasha Levin 		kvm->disks = disk_image__open_all(kvm);
3253b55dcdeSSasha Levin 		if (IS_ERR(kvm->disks))
3263b55dcdeSSasha Levin 			return PTR_ERR(kvm->disks);
3273b55dcdeSSasha Levin 	}
3283b55dcdeSSasha Levin 
3293b55dcdeSSasha Levin 	return 0;
3303b55dcdeSSasha Levin }
33149a8afd1SSasha Levin dev_base_init(disk_image__init);
3323b55dcdeSSasha Levin 
3333b55dcdeSSasha Levin int disk_image__exit(struct kvm *kvm)
3343b55dcdeSSasha Levin {
3353b55dcdeSSasha Levin 	return disk_image__close_all(kvm->disks, kvm->nr_disks);
3363b55dcdeSSasha Levin }
33749a8afd1SSasha Levin dev_base_exit(disk_image__exit);
338