diff options
Diffstat (limited to 'usr.sbin/vmctl/vmctl.c')
-rw-r--r-- | usr.sbin/vmctl/vmctl.c | 187 |
1 files changed, 5 insertions, 182 deletions
diff --git a/usr.sbin/vmctl/vmctl.c b/usr.sbin/vmctl/vmctl.c index 1feca4754d4..66200a6ab28 100644 --- a/usr.sbin/vmctl/vmctl.c +++ b/usr.sbin/vmctl/vmctl.c @@ -1,4 +1,4 @@ -/* $OpenBSD: vmctl.c,v 1.62 2018/10/19 10:12:39 reyk Exp $ */ +/* $OpenBSD: vmctl.c,v 1.63 2018/11/26 10:39:30 reyk Exp $ */ /* * Copyright (c) 2014 Mike Larkin <mlarkin@openbsd.org> @@ -851,10 +851,10 @@ open_imagefile(int type, const char *imgfile_path, int flags, return (-1); } } - ret = virtio_init_qcow2(file, sz, basefd, nfd); + ret = virtio_qcow2_init(file, sz, basefd, nfd); break; default: - ret = virtio_init_raw(file, sz, &fd, 1); + ret = virtio_raw_init(file, sz, &fd, 1); break; } @@ -893,191 +893,14 @@ create_imagefile(int type, const char *imgfile_path, const char *base_path, switch (type) { case VMDF_QCOW2: *format = "qcow2"; - ret = create_qc2_imagefile(imgfile_path, base_path, imgsize); + ret = virtio_qcow2_create(imgfile_path, base_path, imgsize); break; default: *format = "raw"; - ret = create_raw_imagefile(imgfile_path, imgsize); + ret = virtio_raw_create(imgfile_path, imgsize); break; } return (ret); } -/* - * create_raw_imagefile - * - * Create an empty imagefile with the specified path and size. - * - * Parameters: - * imgfile_path: path to the image file to create - * imgsize : size of the image file to create (in MB) - * - * Return: - * EEXIST: The requested image file already exists - * 0 : Image file successfully created - * Exxxx : Various other Exxxx errno codes due to other I/O errors - */ -int -create_raw_imagefile(const char *imgfile_path, long imgsize) -{ - int fd, ret; - - /* Refuse to overwrite an existing image */ - fd = open(imgfile_path, O_RDWR | O_CREAT | O_TRUNC | O_EXCL, - S_IRUSR | S_IWUSR); - if (fd == -1) - return (errno); - - /* Extend to desired size */ - if (ftruncate(fd, (off_t)imgsize * 1024 * 1024) == -1) { - ret = errno; - close(fd); - unlink(imgfile_path); - return (ret); - } - - ret = close(fd); - return (ret); -} - -/* - * create_qc2_imagefile - * - * Create an empty qcow2 imagefile with the specified path and size. - * - * Parameters: - * imgfile_path: path to the image file to create - * imgsize : size of the image file to create (in MB) - * - * Return: - * EEXIST: The requested image file already exists - * 0 : Image file successfully created - * Exxxx : Various other Exxxx errno codes due to other I/O errors - */ -int -create_qc2_imagefile(const char *imgfile_path, - const char *base_path, long imgsize) -{ - struct qcheader { - char magic[4]; - uint32_t version; - uint64_t backingoff; - uint32_t backingsz; - uint32_t clustershift; - uint64_t disksz; - uint32_t cryptmethod; - uint32_t l1sz; - uint64_t l1off; - uint64_t refoff; - uint32_t refsz; - uint32_t snapcount; - uint64_t snapsz; - /* v3 additions */ - uint64_t incompatfeatures; - uint64_t compatfeatures; - uint64_t autoclearfeatures; - uint32_t reforder; - uint32_t headersz; - } __packed hdr, basehdr; - int fd, ret; - ssize_t base_len; - uint64_t l1sz, refsz, disksz, initsz, clustersz; - uint64_t l1off, refoff, v, i, l1entrysz, refentrysz; - uint16_t refs; - - disksz = 1024 * 1024 * imgsize; - - if (base_path) { - fd = open(base_path, O_RDONLY); - if (read(fd, &basehdr, sizeof(basehdr)) != sizeof(basehdr)) - err(1, "failure to read base image header"); - close(fd); - if (strncmp(basehdr.magic, - VM_MAGIC_QCOW, strlen(VM_MAGIC_QCOW)) != 0) - errx(1, "base image is not a qcow2 file"); - if (!disksz) - disksz = betoh64(basehdr.disksz); - else if (disksz != betoh64(basehdr.disksz)) - errx(1, "base size does not match requested size"); - } - if (!base_path && !disksz) - errx(1, "missing disk size"); - - clustersz = (1<<16); - l1off = ALIGN(sizeof(hdr), clustersz); - - l1entrysz = clustersz * clustersz / 8; - l1sz = (disksz + l1entrysz - 1) / l1entrysz; - - refoff = ALIGN(l1off + 8*l1sz, clustersz); - refentrysz = clustersz * clustersz * clustersz / 2; - refsz = (disksz + refentrysz - 1) / refentrysz; - - initsz = ALIGN(refoff + refsz*clustersz, clustersz); - base_len = base_path ? strlen(base_path) : 0; - - memcpy(hdr.magic, VM_MAGIC_QCOW, strlen(VM_MAGIC_QCOW)); - hdr.version = htobe32(3); - hdr.backingoff = htobe64(base_path ? sizeof(hdr) : 0); - hdr.backingsz = htobe32(base_len); - hdr.clustershift = htobe32(16); - hdr.disksz = htobe64(disksz); - hdr.cryptmethod = htobe32(0); - hdr.l1sz = htobe32(l1sz); - hdr.l1off = htobe64(l1off); - hdr.refoff = htobe64(refoff); - hdr.refsz = htobe32(refsz); - hdr.snapcount = htobe32(0); - hdr.snapsz = htobe64(0); - hdr.incompatfeatures = htobe64(0); - hdr.compatfeatures = htobe64(0); - hdr.autoclearfeatures = htobe64(0); - hdr.reforder = htobe32(4); - hdr.headersz = htobe32(sizeof(hdr)); - - /* Refuse to overwrite an existing image */ - fd = open(imgfile_path, O_RDWR | O_CREAT | O_TRUNC | O_EXCL, - S_IRUSR | S_IWUSR); - if (fd == -1) - return (errno); - - /* Write out the header */ - if (write(fd, &hdr, sizeof(hdr)) != sizeof(hdr)) - goto error; - - /* Add the base image */ - if (base_path && write(fd, base_path, base_len) != base_len) - goto error; - - /* Extend to desired size, and add one refcount cluster */ - if (ftruncate(fd, (off_t)initsz + clustersz) == -1) - goto error; - - /* - * Paranoia: if our disk image takes more than one cluster - * to refcount the initial image, fail. - */ - if (initsz/clustersz > clustersz/2) { - errno = ERANGE; - goto error; - } - - /* Add a refcount block, and refcount ourselves. */ - v = htobe64(initsz); - if (pwrite(fd, &v, 8, refoff) != 8) - goto error; - for (i = 0; i < initsz/clustersz + 1; i++) { - refs = htobe16(1); - if (pwrite(fd, &refs, 2, initsz + 2*i) != 2) - goto error; - } - - ret = close(fd); - return (ret); -error: - ret = errno; - close(fd); - unlink(imgfile_path); - return (errno); -} |