blockResize: add flag for bytes

Qemu supports sizing by bytes; we shouldn't force the user to
round up if they really wanted an unaligned total size.

* include/libvirt/libvirt.h.in (VIR_DOMAIN_BLOCK_RESIZE_BYTES):
New flag.
* src/libvirt.c (virDomainBlockResize): Document it.
* src/qemu/qemu_monitor_json.c (qemuMonitorJSONBlockResize): Take
size in bytes.
* src/qemu/qemu_monitor_text.c (qemuMonitorTextBlockResize):
Likewise.  Pass bytes, not megabytes, to monitor.
* src/qemu/qemu_driver.c (qemuDomainBlockResize): Implement new
flag.
This commit is contained in:
Eric Blake 2012-03-03 07:43:22 -07:00
parent 07dd6fb610
commit 877fd769b9
5 changed files with 41 additions and 21 deletions

View File

@ -1636,6 +1636,16 @@ int virDomainBlockPeek (virDomainPtr dom,
size_t size, size_t size,
void *buffer, void *buffer,
unsigned int flags); unsigned int flags);
/**
* virDomainBlockResizeFlags:
*
* Flags available for virDomainBlockResize().
*/
typedef enum {
VIR_DOMAIN_BLOCK_RESIZE_BYTES = 1 << 0, /* size in bytes instead of KiB */
} virDomainBlockResizeFlags;
int virDomainBlockResize (virDomainPtr dom, int virDomainBlockResize (virDomainPtr dom,
const char *disk, const char *disk,
unsigned long long size, unsigned long long size,

View File

@ -7601,12 +7601,15 @@ error:
* virDomainBlockResize: * virDomainBlockResize:
* @dom: pointer to the domain object * @dom: pointer to the domain object
* @disk: path to the block image, or shorthand * @disk: path to the block image, or shorthand
* @size: new size of the block image in kilobytes * @size: new size of the block image, see below for unit
* @flags: extra flags; not used yet, so callers should always pass 0 * @flags: bitwise-OR of virDomainBlockResizeFlags
* *
* Note that this call may fail if the underlying virtualization hypervisor * Resize a block device of domain while the domain is running. If
* does not support it. And this call requires privileged access to the * @flags is 0, then @size is in kibibytes (blocks of 1024); since
* hypervisor. * 0.9.11, if @flags includes VIR_DOMAIN_BLOCK_RESIZE_BYTES, @size is
* in bytes instead. @size is taken directly as the new size.
* Depending on the file format, the hypervisor may round up to the
* next alignment boundary.
* *
* The @disk parameter is either an unambiguous source name of the * The @disk parameter is either an unambiguous source name of the
* block device (the <source file='...'/> sub-element, such as * block device (the <source file='...'/> sub-element, such as
@ -7615,7 +7618,9 @@ error:
* can be found by calling virDomainGetXMLDesc() and inspecting * can be found by calling virDomainGetXMLDesc() and inspecting
* elements within //domain/devices/disk. * elements within //domain/devices/disk.
* *
* Resize a block device of domain while the domain is running. * Note that this call may fail if the underlying virtualization hypervisor
* does not support it; this call requires privileged access to the
* hypervisor.
* *
* Returns: 0 in case of success or -1 in case of failure. * Returns: 0 in case of success or -1 in case of failure.
*/ */

View File

@ -7348,10 +7348,10 @@ qemuGetSchedulerParameters(virDomainPtr dom,
* like LVM volumes. * like LVM volumes.
*/ */
static int static int
qemuDomainBlockResize (virDomainPtr dom, qemuDomainBlockResize(virDomainPtr dom,
const char *path, const char *path,
unsigned long long size, unsigned long long size,
unsigned int flags) unsigned int flags)
{ {
struct qemud_driver *driver = dom->conn->privateData; struct qemud_driver *driver = dom->conn->privateData;
virDomainObjPtr vm; virDomainObjPtr vm;
@ -7360,7 +7360,7 @@ qemuDomainBlockResize (virDomainPtr dom,
char *device = NULL; char *device = NULL;
virDomainDiskDefPtr disk = NULL; virDomainDiskDefPtr disk = NULL;
virCheckFlags(0, -1); virCheckFlags(VIR_DOMAIN_BLOCK_RESIZE_BYTES, -1);
if (path[0] == '\0') { if (path[0] == '\0') {
qemuReportError(VIR_ERR_INVALID_ARG, qemuReportError(VIR_ERR_INVALID_ARG,
@ -7368,11 +7368,15 @@ qemuDomainBlockResize (virDomainPtr dom,
return -1; return -1;
} }
if (size > ULLONG_MAX / 1024) { /* We prefer operating on bytes. */
qemuReportError(VIR_ERR_INVALID_ARG, if ((flags & VIR_DOMAIN_BLOCK_RESIZE_BYTES) == 0) {
_("size must be less than %llu"), if (size > ULLONG_MAX / 1024) {
ULLONG_MAX / 1024); qemuReportError(VIR_ERR_INVALID_ARG,
return -1; _("size must be less than %llu"),
ULLONG_MAX / 1024);
return -1;
}
size *= 1024;
} }
qemuDriverLock(driver); qemuDriverLock(driver);

View File

@ -1783,7 +1783,8 @@ cleanup:
return ret; return ret;
} }
/* Return 0 on success, -1 on failure, or -2 if not supported. */ /* Return 0 on success, -1 on failure, or -2 if not supported. Size
* is in bytes. */
int qemuMonitorJSONBlockResize(qemuMonitorPtr mon, int qemuMonitorJSONBlockResize(qemuMonitorPtr mon,
const char *device, const char *device,
unsigned long long size) unsigned long long size)
@ -1794,7 +1795,7 @@ int qemuMonitorJSONBlockResize(qemuMonitorPtr mon,
cmd = qemuMonitorJSONMakeCommand("block_resize", cmd = qemuMonitorJSONMakeCommand("block_resize",
"s:device", device, "s:device", device,
"U:size", size * 1024, "U:size", size,
NULL); NULL);
if (!cmd) if (!cmd)
return -1; return -1;

View File

@ -1097,7 +1097,8 @@ int qemuMonitorTextGetBlockExtent(qemuMonitorPtr mon ATTRIBUTE_UNUSED,
return -1; return -1;
} }
/* Return 0 on success, -1 on failure, or -2 if not supported. */ /* Return 0 on success, -1 on failure, or -2 if not supported. Size
* is in bytes. */
int qemuMonitorTextBlockResize(qemuMonitorPtr mon, int qemuMonitorTextBlockResize(qemuMonitorPtr mon,
const char *device, const char *device,
unsigned long long size) unsigned long long size)
@ -1106,8 +1107,7 @@ int qemuMonitorTextBlockResize(qemuMonitorPtr mon,
char *reply = NULL; char *reply = NULL;
int ret = -1; int ret = -1;
if (virAsprintf(&cmd, "block_resize %s %llu", if (virAsprintf(&cmd, "block_resize %s %lluB", device, size) < 0) {
device, VIR_DIV_UP(size, 1024)) < 0) {
virReportOOMError(); virReportOOMError();
goto cleanup; goto cleanup;
} }