2009-09-25 13:20:13 +00:00
|
|
|
/*
|
2012-12-13 15:25:48 +00:00
|
|
|
* virstoragefile.c: file utility functions for FS storage backend
|
2009-09-25 13:20:13 +00:00
|
|
|
*
|
conf: split network host structs to util/
Continuing the refactoring of host-side storage descriptions out
of conf/domain_conf and into util/virstoragefile, this patch
focuses on details about a host name/port/transport as used by
a network storage volume.
* src/conf/domain_conf.h (virDomainDiskProtocolTransport)
(virDomainDiskHostDef, virDomainDiskHostDefClear)
(virDomainDiskHostDefFree, virDomainDiskHostDefCopy): Move...
* src/util/virstoragefile.h (virStorageNetHostTransport)
(virStorageNetHostDef, virStorageNetHostDefClear)
(virStorageNetHostDefFree, virStorageNetHostDefCopy): ...here,
with better names.
* src/util/virstoragefile.c (virStorageNetHostDefClear)
(virStorageNetHostDefFree, virStorageNetHostDefCopy): Moved from...
* src/conf/domain_conf.c (virDomainDiskHostDefClear)
(virDomainDiskHostDefFree, virDomainDiskHostDefCopy): ...here.
(virDomainDiskSourceDefClear, virDomainDiskSourceDefParse)
(virDomainDiskSourceDefFormatInternal): Adjust callers.
* src/conf/snapshot_conf.h (_virDomainSnapshotDiskDef): Likewise.
* src/conf/snapshot_conf.c (virDomainSnapshotDiskDefClear):
Likewise.
* src/qemu/qemu_command.c (qemuAddRBDHost)
(qemuParseDriveURIString, qemuParseNBDString)
(qemuBuildNetworkDriveURI, qemuParseCommandLineDisk)
(qemuParseCommandLine, qemuGetDriveSourceString): Likewise.
* src/qemu/qemu_command.h: Likewise.
* src/qemu/qemu_conf.c (qemuAddISCSIPoolSourceHost)
(qemuTranslateDiskSourcePool): Likewise.
* src/qemu/qemu_driver.c
(qemuDomainSnapshotCreateSingleDiskActive)
(qemuDomainSnapshotUndoSingleDiskActive): Likewise.
* src/storage/storage_backend_gluster.c
(virStorageFileBackendGlusterInit): Likewise.
* src/storage/storage_driver.c (virStorageFileFree)
(virStorageFileInitInternal): Likewise.
* src/storage/storage_driver.h (_virStorageFile): Likewise.
* src/libvirt_private.syms (domain_conf.h): Move symbols...
(virstoragefile.h): ...as appropriate.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-03-26 22:33:08 +00:00
|
|
|
* Copyright (C) 2007-2014 Red Hat, Inc.
|
2009-09-25 13:20:13 +00:00
|
|
|
* Copyright (C) 2007-2008 Daniel P. Berrange
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
2012-09-20 22:30:55 +00:00
|
|
|
* License along with this library. If not, see
|
2012-07-21 10:06:23 +00:00
|
|
|
* <http://www.gnu.org/licenses/>.
|
2009-09-25 13:20:13 +00:00
|
|
|
*
|
|
|
|
* Author: Daniel P. Berrange <berrange@redhat.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
2012-12-13 15:25:48 +00:00
|
|
|
#include "virstoragefile.h"
|
2009-09-25 13:20:13 +00:00
|
|
|
|
2011-05-26 18:05:32 +00:00
|
|
|
#include <sys/stat.h>
|
2009-09-29 08:34:48 +00:00
|
|
|
#include <unistd.h>
|
2009-09-29 08:41:23 +00:00
|
|
|
#include <fcntl.h>
|
2012-10-09 23:47:42 +00:00
|
|
|
#include <stdlib.h>
|
2012-12-12 18:06:53 +00:00
|
|
|
#include "viralloc.h"
|
2014-06-23 14:40:49 +00:00
|
|
|
#include "virxml.h"
|
|
|
|
#include "viruuid.h"
|
2012-12-13 18:21:53 +00:00
|
|
|
#include "virerror.h"
|
2012-12-12 17:59:27 +00:00
|
|
|
#include "virlog.h"
|
2011-07-19 18:32:58 +00:00
|
|
|
#include "virfile.h"
|
2011-07-20 09:40:53 +00:00
|
|
|
#include "c-ctype.h"
|
2012-12-12 16:27:01 +00:00
|
|
|
#include "vircommand.h"
|
2012-10-13 16:47:15 +00:00
|
|
|
#include "virhash.h"
|
2013-02-07 01:57:13 +00:00
|
|
|
#include "virendian.h"
|
2013-04-03 10:36:23 +00:00
|
|
|
#include "virstring.h"
|
|
|
|
#include "virutil.h"
|
2014-04-25 11:23:50 +00:00
|
|
|
#include "viruri.h"
|
|
|
|
#include "dirname.h"
|
2014-05-02 17:22:17 +00:00
|
|
|
#include "virbuffer.h"
|
2013-05-31 05:16:14 +00:00
|
|
|
#if HAVE_SYS_SYSCALL_H
|
|
|
|
# include <sys/syscall.h>
|
|
|
|
#endif
|
2009-09-29 08:34:48 +00:00
|
|
|
|
|
|
|
#define VIR_FROM_THIS VIR_FROM_STORAGE
|
|
|
|
|
2014-02-28 12:16:17 +00:00
|
|
|
VIR_LOG_INIT("util.storagefile");
|
|
|
|
|
conf: move host disk type to util/
A continuation of the migration of disk details to virstoragefile.
This patch moves a single enum, but converting the name has quite
a bit of fallout.
* src/conf/domain_conf.h (virDomainDiskType): Move...
* src/util/virstoragefile.h (virStorageType): ...and rename.
* src/bhyve/bhyve_command.c (bhyveBuildDiskArgStr)
(virBhyveProcessBuildLoadCmd): Update clients.
* src/conf/domain_conf.c (virDomainDiskSourceDefParse)
(virDomainDiskDefParseXML, virDomainDiskSourceDefFormatInternal)
(virDomainDiskDefFormat, virDomainDiskGetActualType)
(virDomainDiskDefForeachPath, virDomainDiskSourceIsBlockType):
Likewise.
* src/conf/snapshot_conf.h (_virDomainSnapshotDiskDef): Likewise.
* src/conf/snapshot_conf.c (virDomainSnapshotDiskDefParseXML)
(virDomainSnapshotAlignDisks, virDomainSnapshotDiskDefFormat):
Likewise.
* src/esx/esx_driver.c (esxAutodetectSCSIControllerModel)
(esxDomainDefineXML): Likewise.
* src/locking/domain_lock.c (virDomainLockManagerAddDisk):
Likewise.
* src/lxc/lxc_controller.c
(virLXCControllerSetupLoopDeviceDisk)
(virLXCControllerSetupNBDDeviceDisk)
(virLXCControllerSetupLoopDevices, virLXCControllerSetupDisk):
Likewise.
* src/parallels/parallels_driver.c (parallelsGetHddInfo):
Likewise.
* src/phyp/phyp_driver.c (phypDiskType): Likewise.
* src/qemu/qemu_command.c (qemuGetDriveSourceString)
(qemuDomainDiskGetSourceString, qemuBuildDriveStr)
(qemuBuildCommandLine, qemuParseCommandLineDisk)
(qemuParseCommandLine): Likewise.
* src/qemu/qemu_conf.c (qemuCheckSharedDevice)
(qemuTranslateDiskSourcePool)
(qemuTranslateSnapshotDiskSourcePool): Likewise.
* src/qemu/qemu_domain.c (qemuDomainDeviceDefPostParse)
(qemuDomainDetermineDiskChain): Likewise.
* src/qemu/qemu_driver.c (qemuDomainGetBlockInfo)
(qemuDomainSnapshotPrepareDiskExternalBackingInactive)
(qemuDomainSnapshotPrepareDiskExternalBackingActive)
(qemuDomainSnapshotPrepareDiskExternalOverlayActive)
(qemuDomainSnapshotPrepareDiskExternalOverlayInactive)
(qemuDomainSnapshotPrepareDiskInternal)
(qemuDomainSnapshotPrepare)
(qemuDomainSnapshotCreateSingleDiskActive): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainChangeEjectableMedia):
Likewise.
* src/qemu/qemu_migration.c (qemuMigrationIsSafe): Likewise.
* src/security/security_apparmor.c
(AppArmorRestoreSecurityImageLabel)
(AppArmorSetSecurityImageLabel): Likewise.
* src/security/security_dac.c (virSecurityDACSetSecurityImageLabel)
(virSecurityDACRestoreSecurityImageLabelInt)
(virSecurityDACSetSecurityAllLabel): Likewise.
* src/security/security_selinux.c
(virSecuritySELinuxRestoreSecurityImageLabelInt)
(virSecuritySELinuxSetSecurityImageLabel)
(virSecuritySELinuxSetSecurityAllLabel): Likewise.
* src/storage/storage_backend.c (virStorageFileBackendForType):
Likewise.
* src/storage/storage_backend_fs.c (virStorageFileBackendFile)
(virStorageFileBackendBlock): Likewise.
* src/storage/storage_backend_gluster.c
(virStorageFileBackendGluster): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainGetXMLDesc, vboxAttachDrives)
(vboxDomainAttachDeviceImpl, vboxDomainDetachDevice): Likewise.
* src/vmware/vmware_conf.c (vmwareVmxPath): Likewise.
* src/vmx/vmx.c (virVMXParseDisk, virVMXFormatDisk)
(virVMXFormatFloppy): Likewise.
* src/xenxs/xen_sxpr.c (xenParseSxprDisks, xenParseSxpr)
(xenFormatSxprDisk): Likewise.
* src/xenxs/xen_xm.c (xenParseXM, xenFormatXMDisk): Likewise.
* tests/securityselinuxlabeltest.c (testSELinuxLoadDef):
Likewise.
* src/libvirt_private.syms (domain_conf.h): Move symbols...
(virstoragefile.h): ...as appropriate.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-03-27 21:57:49 +00:00
|
|
|
VIR_ENUM_IMPL(virStorage, VIR_STORAGE_TYPE_LAST,
|
2014-04-02 19:01:46 +00:00
|
|
|
"none",
|
conf: move host disk type to util/
A continuation of the migration of disk details to virstoragefile.
This patch moves a single enum, but converting the name has quite
a bit of fallout.
* src/conf/domain_conf.h (virDomainDiskType): Move...
* src/util/virstoragefile.h (virStorageType): ...and rename.
* src/bhyve/bhyve_command.c (bhyveBuildDiskArgStr)
(virBhyveProcessBuildLoadCmd): Update clients.
* src/conf/domain_conf.c (virDomainDiskSourceDefParse)
(virDomainDiskDefParseXML, virDomainDiskSourceDefFormatInternal)
(virDomainDiskDefFormat, virDomainDiskGetActualType)
(virDomainDiskDefForeachPath, virDomainDiskSourceIsBlockType):
Likewise.
* src/conf/snapshot_conf.h (_virDomainSnapshotDiskDef): Likewise.
* src/conf/snapshot_conf.c (virDomainSnapshotDiskDefParseXML)
(virDomainSnapshotAlignDisks, virDomainSnapshotDiskDefFormat):
Likewise.
* src/esx/esx_driver.c (esxAutodetectSCSIControllerModel)
(esxDomainDefineXML): Likewise.
* src/locking/domain_lock.c (virDomainLockManagerAddDisk):
Likewise.
* src/lxc/lxc_controller.c
(virLXCControllerSetupLoopDeviceDisk)
(virLXCControllerSetupNBDDeviceDisk)
(virLXCControllerSetupLoopDevices, virLXCControllerSetupDisk):
Likewise.
* src/parallels/parallels_driver.c (parallelsGetHddInfo):
Likewise.
* src/phyp/phyp_driver.c (phypDiskType): Likewise.
* src/qemu/qemu_command.c (qemuGetDriveSourceString)
(qemuDomainDiskGetSourceString, qemuBuildDriveStr)
(qemuBuildCommandLine, qemuParseCommandLineDisk)
(qemuParseCommandLine): Likewise.
* src/qemu/qemu_conf.c (qemuCheckSharedDevice)
(qemuTranslateDiskSourcePool)
(qemuTranslateSnapshotDiskSourcePool): Likewise.
* src/qemu/qemu_domain.c (qemuDomainDeviceDefPostParse)
(qemuDomainDetermineDiskChain): Likewise.
* src/qemu/qemu_driver.c (qemuDomainGetBlockInfo)
(qemuDomainSnapshotPrepareDiskExternalBackingInactive)
(qemuDomainSnapshotPrepareDiskExternalBackingActive)
(qemuDomainSnapshotPrepareDiskExternalOverlayActive)
(qemuDomainSnapshotPrepareDiskExternalOverlayInactive)
(qemuDomainSnapshotPrepareDiskInternal)
(qemuDomainSnapshotPrepare)
(qemuDomainSnapshotCreateSingleDiskActive): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainChangeEjectableMedia):
Likewise.
* src/qemu/qemu_migration.c (qemuMigrationIsSafe): Likewise.
* src/security/security_apparmor.c
(AppArmorRestoreSecurityImageLabel)
(AppArmorSetSecurityImageLabel): Likewise.
* src/security/security_dac.c (virSecurityDACSetSecurityImageLabel)
(virSecurityDACRestoreSecurityImageLabelInt)
(virSecurityDACSetSecurityAllLabel): Likewise.
* src/security/security_selinux.c
(virSecuritySELinuxRestoreSecurityImageLabelInt)
(virSecuritySELinuxSetSecurityImageLabel)
(virSecuritySELinuxSetSecurityAllLabel): Likewise.
* src/storage/storage_backend.c (virStorageFileBackendForType):
Likewise.
* src/storage/storage_backend_fs.c (virStorageFileBackendFile)
(virStorageFileBackendBlock): Likewise.
* src/storage/storage_backend_gluster.c
(virStorageFileBackendGluster): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainGetXMLDesc, vboxAttachDrives)
(vboxDomainAttachDeviceImpl, vboxDomainDetachDevice): Likewise.
* src/vmware/vmware_conf.c (vmwareVmxPath): Likewise.
* src/vmx/vmx.c (virVMXParseDisk, virVMXFormatDisk)
(virVMXFormatFloppy): Likewise.
* src/xenxs/xen_sxpr.c (xenParseSxprDisks, xenParseSxpr)
(xenFormatSxprDisk): Likewise.
* src/xenxs/xen_xm.c (xenParseXM, xenFormatXMDisk): Likewise.
* tests/securityselinuxlabeltest.c (testSELinuxLoadDef):
Likewise.
* src/libvirt_private.syms (domain_conf.h): Move symbols...
(virstoragefile.h): ...as appropriate.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-03-27 21:57:49 +00:00
|
|
|
"file",
|
2014-04-02 19:01:46 +00:00
|
|
|
"block",
|
conf: move host disk type to util/
A continuation of the migration of disk details to virstoragefile.
This patch moves a single enum, but converting the name has quite
a bit of fallout.
* src/conf/domain_conf.h (virDomainDiskType): Move...
* src/util/virstoragefile.h (virStorageType): ...and rename.
* src/bhyve/bhyve_command.c (bhyveBuildDiskArgStr)
(virBhyveProcessBuildLoadCmd): Update clients.
* src/conf/domain_conf.c (virDomainDiskSourceDefParse)
(virDomainDiskDefParseXML, virDomainDiskSourceDefFormatInternal)
(virDomainDiskDefFormat, virDomainDiskGetActualType)
(virDomainDiskDefForeachPath, virDomainDiskSourceIsBlockType):
Likewise.
* src/conf/snapshot_conf.h (_virDomainSnapshotDiskDef): Likewise.
* src/conf/snapshot_conf.c (virDomainSnapshotDiskDefParseXML)
(virDomainSnapshotAlignDisks, virDomainSnapshotDiskDefFormat):
Likewise.
* src/esx/esx_driver.c (esxAutodetectSCSIControllerModel)
(esxDomainDefineXML): Likewise.
* src/locking/domain_lock.c (virDomainLockManagerAddDisk):
Likewise.
* src/lxc/lxc_controller.c
(virLXCControllerSetupLoopDeviceDisk)
(virLXCControllerSetupNBDDeviceDisk)
(virLXCControllerSetupLoopDevices, virLXCControllerSetupDisk):
Likewise.
* src/parallels/parallels_driver.c (parallelsGetHddInfo):
Likewise.
* src/phyp/phyp_driver.c (phypDiskType): Likewise.
* src/qemu/qemu_command.c (qemuGetDriveSourceString)
(qemuDomainDiskGetSourceString, qemuBuildDriveStr)
(qemuBuildCommandLine, qemuParseCommandLineDisk)
(qemuParseCommandLine): Likewise.
* src/qemu/qemu_conf.c (qemuCheckSharedDevice)
(qemuTranslateDiskSourcePool)
(qemuTranslateSnapshotDiskSourcePool): Likewise.
* src/qemu/qemu_domain.c (qemuDomainDeviceDefPostParse)
(qemuDomainDetermineDiskChain): Likewise.
* src/qemu/qemu_driver.c (qemuDomainGetBlockInfo)
(qemuDomainSnapshotPrepareDiskExternalBackingInactive)
(qemuDomainSnapshotPrepareDiskExternalBackingActive)
(qemuDomainSnapshotPrepareDiskExternalOverlayActive)
(qemuDomainSnapshotPrepareDiskExternalOverlayInactive)
(qemuDomainSnapshotPrepareDiskInternal)
(qemuDomainSnapshotPrepare)
(qemuDomainSnapshotCreateSingleDiskActive): Likewise.
* src/qemu/qemu_hotplug.c (qemuDomainChangeEjectableMedia):
Likewise.
* src/qemu/qemu_migration.c (qemuMigrationIsSafe): Likewise.
* src/security/security_apparmor.c
(AppArmorRestoreSecurityImageLabel)
(AppArmorSetSecurityImageLabel): Likewise.
* src/security/security_dac.c (virSecurityDACSetSecurityImageLabel)
(virSecurityDACRestoreSecurityImageLabelInt)
(virSecurityDACSetSecurityAllLabel): Likewise.
* src/security/security_selinux.c
(virSecuritySELinuxRestoreSecurityImageLabelInt)
(virSecuritySELinuxSetSecurityImageLabel)
(virSecuritySELinuxSetSecurityAllLabel): Likewise.
* src/storage/storage_backend.c (virStorageFileBackendForType):
Likewise.
* src/storage/storage_backend_fs.c (virStorageFileBackendFile)
(virStorageFileBackendBlock): Likewise.
* src/storage/storage_backend_gluster.c
(virStorageFileBackendGluster): Likewise.
* src/vbox/vbox_tmpl.c (vboxDomainGetXMLDesc, vboxAttachDrives)
(vboxDomainAttachDeviceImpl, vboxDomainDetachDevice): Likewise.
* src/vmware/vmware_conf.c (vmwareVmxPath): Likewise.
* src/vmx/vmx.c (virVMXParseDisk, virVMXFormatDisk)
(virVMXFormatFloppy): Likewise.
* src/xenxs/xen_sxpr.c (xenParseSxprDisks, xenParseSxpr)
(xenFormatSxprDisk): Likewise.
* src/xenxs/xen_xm.c (xenParseXM, xenFormatXMDisk): Likewise.
* tests/securityselinuxlabeltest.c (testSELinuxLoadDef):
Likewise.
* src/libvirt_private.syms (domain_conf.h): Move symbols...
(virstoragefile.h): ...as appropriate.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-03-27 21:57:49 +00:00
|
|
|
"dir",
|
|
|
|
"network",
|
|
|
|
"volume")
|
|
|
|
|
2009-09-25 13:20:13 +00:00
|
|
|
VIR_ENUM_IMPL(virStorageFileFormat,
|
|
|
|
VIR_STORAGE_FILE_LAST,
|
storage: list more file types
When an image has no backing file, using VIR_STORAGE_FILE_AUTO
for its type is a bit confusing. Additionally, a future patch
would like to reserve a default value for the case of no file
type specified in the XML, but different from the current use
of -1 to imply probing, since probing is not always safe.
Also, a couple of file types were missing compared to supported
code: libxl supports 'vhd', and qemu supports 'fat' for directories
passed through as a file system.
* src/util/storage_file.h (virStorageFileFormat): Add
VIR_STORAGE_FILE_NONE, VIR_STORAGE_FILE_FAT, VIR_STORAGE_FILE_VHD.
* src/util/storage_file.c (virStorageFileMatchesVersion): Match
documentation when version probing not supported.
(cowGetBackingStore, qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStoreFormat, qedGetBackingStore)
(virStorageFileGetMetadataFromBuf)
(virStorageFileGetMetadataFromFD): Take NONE into account.
* src/conf/domain_conf.c (virDomainDiskDefForeachPath): Likewise.
* src/qemu/qemu_driver.c (qemuDomainGetBlockInfo): Likewise.
* src/conf/storage_conf.c (virStorageVolumeFormatFromString): New
function.
(poolTypeInfo): Use it.
2012-09-28 17:11:07 +00:00
|
|
|
"none",
|
2010-11-22 20:08:17 +00:00
|
|
|
"raw", "dir", "bochs",
|
2014-04-14 22:54:14 +00:00
|
|
|
"cloop", "dmg", "iso",
|
|
|
|
"vpc", "vdi",
|
|
|
|
/* Not direct file formats, but used for various drivers */
|
2014-05-07 18:04:06 +00:00
|
|
|
"fat", "vhd", "ploop",
|
2014-04-14 22:54:14 +00:00
|
|
|
/* Formats with backing file below here */
|
|
|
|
"cow", "qcow", "qcow2", "qed", "vmdk")
|
2009-09-29 08:34:48 +00:00
|
|
|
|
2013-05-07 15:27:43 +00:00
|
|
|
VIR_ENUM_IMPL(virStorageFileFeature,
|
|
|
|
VIR_STORAGE_FILE_FEATURE_LAST,
|
|
|
|
"lazy_refcounts",
|
|
|
|
)
|
|
|
|
|
2014-03-27 22:47:39 +00:00
|
|
|
VIR_ENUM_IMPL(virStorageNetProtocol, VIR_STORAGE_NET_PROTOCOL_LAST,
|
2014-04-24 14:11:44 +00:00
|
|
|
"none",
|
2014-03-27 22:47:39 +00:00
|
|
|
"nbd",
|
|
|
|
"rbd",
|
|
|
|
"sheepdog",
|
|
|
|
"gluster",
|
|
|
|
"iscsi",
|
|
|
|
"http",
|
|
|
|
"https",
|
|
|
|
"ftp",
|
|
|
|
"ftps",
|
|
|
|
"tftp")
|
conf: split network host structs to util/
Continuing the refactoring of host-side storage descriptions out
of conf/domain_conf and into util/virstoragefile, this patch
focuses on details about a host name/port/transport as used by
a network storage volume.
* src/conf/domain_conf.h (virDomainDiskProtocolTransport)
(virDomainDiskHostDef, virDomainDiskHostDefClear)
(virDomainDiskHostDefFree, virDomainDiskHostDefCopy): Move...
* src/util/virstoragefile.h (virStorageNetHostTransport)
(virStorageNetHostDef, virStorageNetHostDefClear)
(virStorageNetHostDefFree, virStorageNetHostDefCopy): ...here,
with better names.
* src/util/virstoragefile.c (virStorageNetHostDefClear)
(virStorageNetHostDefFree, virStorageNetHostDefCopy): Moved from...
* src/conf/domain_conf.c (virDomainDiskHostDefClear)
(virDomainDiskHostDefFree, virDomainDiskHostDefCopy): ...here.
(virDomainDiskSourceDefClear, virDomainDiskSourceDefParse)
(virDomainDiskSourceDefFormatInternal): Adjust callers.
* src/conf/snapshot_conf.h (_virDomainSnapshotDiskDef): Likewise.
* src/conf/snapshot_conf.c (virDomainSnapshotDiskDefClear):
Likewise.
* src/qemu/qemu_command.c (qemuAddRBDHost)
(qemuParseDriveURIString, qemuParseNBDString)
(qemuBuildNetworkDriveURI, qemuParseCommandLineDisk)
(qemuParseCommandLine, qemuGetDriveSourceString): Likewise.
* src/qemu/qemu_command.h: Likewise.
* src/qemu/qemu_conf.c (qemuAddISCSIPoolSourceHost)
(qemuTranslateDiskSourcePool): Likewise.
* src/qemu/qemu_driver.c
(qemuDomainSnapshotCreateSingleDiskActive)
(qemuDomainSnapshotUndoSingleDiskActive): Likewise.
* src/storage/storage_backend_gluster.c
(virStorageFileBackendGlusterInit): Likewise.
* src/storage/storage_driver.c (virStorageFileFree)
(virStorageFileInitInternal): Likewise.
* src/storage/storage_driver.h (_virStorageFile): Likewise.
* src/libvirt_private.syms (domain_conf.h): Move symbols...
(virstoragefile.h): ...as appropriate.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-03-26 22:33:08 +00:00
|
|
|
|
|
|
|
VIR_ENUM_IMPL(virStorageNetHostTransport, VIR_STORAGE_NET_HOST_TRANS_LAST,
|
|
|
|
"tcp",
|
|
|
|
"unix",
|
|
|
|
"rdma")
|
|
|
|
|
2014-03-28 04:33:45 +00:00
|
|
|
VIR_ENUM_IMPL(virStorageSourcePoolMode,
|
|
|
|
VIR_STORAGE_SOURCE_POOL_MODE_LAST,
|
|
|
|
"default",
|
|
|
|
"host",
|
|
|
|
"direct")
|
conf: split network host structs to util/
Continuing the refactoring of host-side storage descriptions out
of conf/domain_conf and into util/virstoragefile, this patch
focuses on details about a host name/port/transport as used by
a network storage volume.
* src/conf/domain_conf.h (virDomainDiskProtocolTransport)
(virDomainDiskHostDef, virDomainDiskHostDefClear)
(virDomainDiskHostDefFree, virDomainDiskHostDefCopy): Move...
* src/util/virstoragefile.h (virStorageNetHostTransport)
(virStorageNetHostDef, virStorageNetHostDefClear)
(virStorageNetHostDefFree, virStorageNetHostDefCopy): ...here,
with better names.
* src/util/virstoragefile.c (virStorageNetHostDefClear)
(virStorageNetHostDefFree, virStorageNetHostDefCopy): Moved from...
* src/conf/domain_conf.c (virDomainDiskHostDefClear)
(virDomainDiskHostDefFree, virDomainDiskHostDefCopy): ...here.
(virDomainDiskSourceDefClear, virDomainDiskSourceDefParse)
(virDomainDiskSourceDefFormatInternal): Adjust callers.
* src/conf/snapshot_conf.h (_virDomainSnapshotDiskDef): Likewise.
* src/conf/snapshot_conf.c (virDomainSnapshotDiskDefClear):
Likewise.
* src/qemu/qemu_command.c (qemuAddRBDHost)
(qemuParseDriveURIString, qemuParseNBDString)
(qemuBuildNetworkDriveURI, qemuParseCommandLineDisk)
(qemuParseCommandLine, qemuGetDriveSourceString): Likewise.
* src/qemu/qemu_command.h: Likewise.
* src/qemu/qemu_conf.c (qemuAddISCSIPoolSourceHost)
(qemuTranslateDiskSourcePool): Likewise.
* src/qemu/qemu_driver.c
(qemuDomainSnapshotCreateSingleDiskActive)
(qemuDomainSnapshotUndoSingleDiskActive): Likewise.
* src/storage/storage_backend_gluster.c
(virStorageFileBackendGlusterInit): Likewise.
* src/storage/storage_driver.c (virStorageFileFree)
(virStorageFileInitInternal): Likewise.
* src/storage/storage_driver.h (_virStorageFile): Likewise.
* src/libvirt_private.syms (domain_conf.h): Move symbols...
(virstoragefile.h): ...as appropriate.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-03-26 22:33:08 +00:00
|
|
|
|
2014-06-23 14:40:49 +00:00
|
|
|
VIR_ENUM_IMPL(virStorageAuth,
|
|
|
|
VIR_STORAGE_AUTH_TYPE_LAST,
|
|
|
|
"none", "chap", "ceph")
|
|
|
|
|
2009-09-29 08:34:48 +00:00
|
|
|
enum lv_endian {
|
|
|
|
LV_LITTLE_ENDIAN = 1, /* 1234 */
|
|
|
|
LV_BIG_ENDIAN /* 4321 */
|
|
|
|
};
|
|
|
|
|
|
|
|
enum {
|
|
|
|
BACKING_STORE_OK,
|
|
|
|
BACKING_STORE_INVALID,
|
|
|
|
BACKING_STORE_ERROR,
|
|
|
|
};
|
|
|
|
|
2013-05-07 15:27:43 +00:00
|
|
|
#define FILE_TYPE_VERSIONS_LAST 2
|
|
|
|
|
2009-09-29 08:34:48 +00:00
|
|
|
/* Either 'magic' or 'extension' *must* be provided */
|
|
|
|
struct FileTypeInfo {
|
2013-02-04 19:16:22 +00:00
|
|
|
int magicOffset; /* Byte offset of the magic */
|
2009-09-29 08:34:48 +00:00
|
|
|
const char *magic; /* Optional string of file magic
|
|
|
|
* to check at head of file */
|
|
|
|
const char *extension; /* Optional file extension to check */
|
|
|
|
enum lv_endian endian; /* Endianness of file format */
|
|
|
|
int versionOffset; /* Byte offset from start of file
|
|
|
|
* where we find version number,
|
2013-06-10 09:44:31 +00:00
|
|
|
* -1 to always fail the version test,
|
|
|
|
* -2 to always pass the version test */
|
2013-05-07 15:27:43 +00:00
|
|
|
int versionNumbers[FILE_TYPE_VERSIONS_LAST];
|
|
|
|
/* Version numbers to validate. Zeroes are ignored. */
|
2009-09-29 08:34:48 +00:00
|
|
|
int sizeOffset; /* Byte offset from start of file
|
|
|
|
* where we find capacity info,
|
|
|
|
* -1 to use st_size as capacity */
|
|
|
|
int sizeBytes; /* Number of bytes for size field */
|
|
|
|
int sizeMultiplier; /* A scaling factor if size is not in bytes */
|
|
|
|
/* Store a COW base image path (possibly relative),
|
|
|
|
* or NULL if there is no COW base image, to RES;
|
|
|
|
* return BACKING_STORE_* */
|
|
|
|
int qcowCryptOffset; /* Byte offset from start of file
|
|
|
|
* where to find encryption mode,
|
|
|
|
* -1 if encryption is not used */
|
2010-06-14 14:53:59 +00:00
|
|
|
int (*getBackingStore)(char **res, int *format,
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
const char *buf, size_t buf_size);
|
2013-05-07 15:27:43 +00:00
|
|
|
int (*getFeatures)(virBitmapPtr *features, int format,
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
char *buf, ssize_t len);
|
2009-09-29 08:34:48 +00:00
|
|
|
};
|
|
|
|
|
2010-06-14 14:53:59 +00:00
|
|
|
static int cowGetBackingStore(char **, int *,
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
const char *, size_t);
|
2010-06-14 14:53:59 +00:00
|
|
|
static int qcow1GetBackingStore(char **, int *,
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
const char *, size_t);
|
2010-06-14 14:53:59 +00:00
|
|
|
static int qcow2GetBackingStore(char **, int *,
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
const char *, size_t);
|
2013-05-07 15:27:43 +00:00
|
|
|
static int qcow2GetFeatures(virBitmapPtr *features, int format,
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
char *buf, ssize_t len);
|
2010-06-14 14:53:59 +00:00
|
|
|
static int vmdk4GetBackingStore(char **, int *,
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
const char *, size_t);
|
2010-11-19 23:19:24 +00:00
|
|
|
static int
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
qedGetBackingStore(char **, int *, const char *, size_t);
|
2010-06-14 14:53:59 +00:00
|
|
|
|
|
|
|
#define QCOWX_HDR_VERSION (4)
|
|
|
|
#define QCOWX_HDR_BACKING_FILE_OFFSET (QCOWX_HDR_VERSION+4)
|
|
|
|
#define QCOWX_HDR_BACKING_FILE_SIZE (QCOWX_HDR_BACKING_FILE_OFFSET+8)
|
|
|
|
#define QCOWX_HDR_IMAGE_SIZE (QCOWX_HDR_BACKING_FILE_SIZE+4+4)
|
|
|
|
|
|
|
|
#define QCOW1_HDR_CRYPT (QCOWX_HDR_IMAGE_SIZE+8+1+1)
|
|
|
|
#define QCOW2_HDR_CRYPT (QCOWX_HDR_IMAGE_SIZE+8)
|
|
|
|
|
|
|
|
#define QCOW1_HDR_TOTAL_SIZE (QCOW1_HDR_CRYPT+4+8)
|
|
|
|
#define QCOW2_HDR_TOTAL_SIZE (QCOW2_HDR_CRYPT+4+4+8+8+4+4+8)
|
|
|
|
|
|
|
|
#define QCOW2_HDR_EXTENSION_END 0
|
|
|
|
#define QCOW2_HDR_EXTENSION_BACKING_FORMAT 0xE2792ACA
|
|
|
|
|
2013-05-07 15:27:43 +00:00
|
|
|
#define QCOW2v3_HDR_FEATURES_INCOMPATIBLE (QCOW2_HDR_TOTAL_SIZE)
|
|
|
|
#define QCOW2v3_HDR_FEATURES_COMPATIBLE (QCOW2v3_HDR_FEATURES_INCOMPATIBLE+8)
|
|
|
|
#define QCOW2v3_HDR_FEATURES_AUTOCLEAR (QCOW2v3_HDR_FEATURES_COMPATIBLE+8)
|
|
|
|
|
|
|
|
/* The location of the header size [4 bytes] */
|
|
|
|
#define QCOW2v3_HDR_SIZE (QCOW2_HDR_TOTAL_SIZE+8+8+8+4)
|
|
|
|
|
2010-11-19 23:19:24 +00:00
|
|
|
#define QED_HDR_FEATURES_OFFSET (4+4+4+4)
|
2010-11-22 20:08:17 +00:00
|
|
|
#define QED_HDR_IMAGE_SIZE (QED_HDR_FEATURES_OFFSET+8+8+8+8)
|
|
|
|
#define QED_HDR_BACKING_FILE_OFFSET (QED_HDR_IMAGE_SIZE+8)
|
2010-11-19 23:19:24 +00:00
|
|
|
#define QED_HDR_BACKING_FILE_SIZE (QED_HDR_BACKING_FILE_OFFSET+4)
|
|
|
|
#define QED_F_BACKING_FILE 0x01
|
|
|
|
#define QED_F_BACKING_FORMAT_NO_PROBE 0x04
|
2010-11-19 16:18:16 +00:00
|
|
|
|
2009-09-29 08:34:48 +00:00
|
|
|
|
|
|
|
static struct FileTypeInfo const fileTypeInfo[] = {
|
2013-02-04 19:16:22 +00:00
|
|
|
[VIR_STORAGE_FILE_NONE] = { 0, NULL, NULL, LV_LITTLE_ENDIAN,
|
2013-05-07 15:27:43 +00:00
|
|
|
-1, {0}, 0, 0, 0, 0, NULL, NULL },
|
2013-02-04 19:16:22 +00:00
|
|
|
[VIR_STORAGE_FILE_RAW] = { 0, NULL, NULL, LV_LITTLE_ENDIAN,
|
2013-05-07 15:27:43 +00:00
|
|
|
-1, {0}, 0, 0, 0, 0, NULL, NULL },
|
2013-02-04 19:16:22 +00:00
|
|
|
[VIR_STORAGE_FILE_DIR] = { 0, NULL, NULL, LV_LITTLE_ENDIAN,
|
2013-05-07 15:27:43 +00:00
|
|
|
-1, {0}, 0, 0, 0, 0, NULL, NULL },
|
2010-06-14 15:39:32 +00:00
|
|
|
[VIR_STORAGE_FILE_BOCHS] = {
|
2013-02-04 19:16:22 +00:00
|
|
|
/*"Bochs Virtual HD Image", */ /* Untested */
|
|
|
|
0, NULL, NULL,
|
2013-05-07 15:27:43 +00:00
|
|
|
LV_LITTLE_ENDIAN, 64, {0x20000},
|
|
|
|
32+16+16+4+4+4+4+4, 8, 1, -1, NULL, NULL
|
2010-06-14 15:39:32 +00:00
|
|
|
},
|
|
|
|
[VIR_STORAGE_FILE_CLOOP] = {
|
2013-02-04 19:16:22 +00:00
|
|
|
/* #!/bin/sh
|
|
|
|
#V2.0 Format
|
|
|
|
modprobe cloop file=$0 && mount -r -t iso9660 /dev/cloop $1
|
|
|
|
*/ /* Untested */
|
|
|
|
0, NULL, NULL,
|
2013-05-07 15:27:43 +00:00
|
|
|
LV_LITTLE_ENDIAN, -1, {0},
|
|
|
|
-1, 0, 0, -1, NULL, NULL
|
2010-06-14 15:39:32 +00:00
|
|
|
},
|
|
|
|
[VIR_STORAGE_FILE_DMG] = {
|
2013-02-04 19:16:22 +00:00
|
|
|
/* XXX QEMU says there's no magic for dmg,
|
|
|
|
* /usr/share/misc/magic lists double magic (both offsets
|
|
|
|
* would have to match) but then disables that check. */
|
|
|
|
0, NULL, ".dmg",
|
2013-05-07 15:27:43 +00:00
|
|
|
0, -1, {0},
|
|
|
|
-1, 0, 0, -1, NULL, NULL
|
2010-06-14 15:39:32 +00:00
|
|
|
},
|
|
|
|
[VIR_STORAGE_FILE_ISO] = {
|
2013-02-04 19:16:22 +00:00
|
|
|
32769, "CD001", ".iso",
|
2013-05-07 15:27:43 +00:00
|
|
|
LV_LITTLE_ENDIAN, -2, {0},
|
|
|
|
-1, 0, 0, -1, NULL, NULL
|
2010-06-14 15:39:32 +00:00
|
|
|
},
|
2014-04-14 22:54:14 +00:00
|
|
|
[VIR_STORAGE_FILE_VPC] = {
|
|
|
|
0, "conectix", NULL,
|
|
|
|
LV_BIG_ENDIAN, 12, {0x10000},
|
|
|
|
8 + 4 + 4 + 8 + 4 + 4 + 2 + 2 + 4, 8, 1, -1, NULL, NULL
|
|
|
|
},
|
|
|
|
/* TODO: add getBackingStore function */
|
|
|
|
[VIR_STORAGE_FILE_VDI] = {
|
|
|
|
64, "\x7f\x10\xda\xbe", ".vdi",
|
|
|
|
LV_LITTLE_ENDIAN, 68, {0x00010001},
|
|
|
|
64 + 5 * 4 + 256 + 7 * 4, 8, 1, -1, NULL, NULL},
|
|
|
|
|
|
|
|
/* Not direct file formats, but used for various drivers */
|
|
|
|
[VIR_STORAGE_FILE_FAT] = { 0, NULL, NULL, LV_LITTLE_ENDIAN,
|
|
|
|
-1, {0}, 0, 0, 0, 0, NULL, NULL },
|
|
|
|
[VIR_STORAGE_FILE_VHD] = { 0, NULL, NULL, LV_LITTLE_ENDIAN,
|
|
|
|
-1, {0}, 0, 0, 0, 0, NULL, NULL },
|
2014-05-07 18:04:06 +00:00
|
|
|
[VIR_STORAGE_FILE_PLOOP] = { 0, NULL, NULL, LV_LITTLE_ENDIAN,
|
|
|
|
-1, {0}, 0, 0, 0, 0, NULL, NULL },
|
2014-04-14 22:54:14 +00:00
|
|
|
|
|
|
|
/* All formats with a backing store probe below here */
|
|
|
|
[VIR_STORAGE_FILE_COW] = {
|
|
|
|
0, "OOOM", NULL,
|
|
|
|
LV_BIG_ENDIAN, 4, {2},
|
|
|
|
4+4+1024+4, 8, 1, -1, cowGetBackingStore, NULL
|
|
|
|
},
|
2010-06-14 15:39:32 +00:00
|
|
|
[VIR_STORAGE_FILE_QCOW] = {
|
2013-02-04 19:16:22 +00:00
|
|
|
0, "QFI", NULL,
|
2013-05-07 15:27:43 +00:00
|
|
|
LV_BIG_ENDIAN, 4, {1},
|
|
|
|
QCOWX_HDR_IMAGE_SIZE, 8, 1, QCOW1_HDR_CRYPT, qcow1GetBackingStore, NULL
|
2010-06-14 15:39:32 +00:00
|
|
|
},
|
|
|
|
[VIR_STORAGE_FILE_QCOW2] = {
|
2013-02-04 19:16:22 +00:00
|
|
|
0, "QFI", NULL,
|
2013-05-07 15:27:43 +00:00
|
|
|
LV_BIG_ENDIAN, 4, {2, 3},
|
2010-06-14 15:39:32 +00:00
|
|
|
QCOWX_HDR_IMAGE_SIZE, 8, 1, QCOW2_HDR_CRYPT, qcow2GetBackingStore,
|
2013-05-07 15:27:43 +00:00
|
|
|
qcow2GetFeatures
|
2010-06-14 15:39:32 +00:00
|
|
|
},
|
2010-11-19 16:18:16 +00:00
|
|
|
[VIR_STORAGE_FILE_QED] = {
|
|
|
|
/* http://wiki.qemu.org/Features/QED */
|
2013-02-04 19:16:22 +00:00
|
|
|
0, "QED", NULL,
|
2013-05-07 15:27:43 +00:00
|
|
|
LV_LITTLE_ENDIAN, -2, {0},
|
|
|
|
QED_HDR_IMAGE_SIZE, 8, 1, -1, qedGetBackingStore, NULL
|
2010-11-19 16:18:16 +00:00
|
|
|
},
|
2010-06-14 15:39:32 +00:00
|
|
|
[VIR_STORAGE_FILE_VMDK] = {
|
2013-02-04 19:16:22 +00:00
|
|
|
0, "KDMV", NULL,
|
2013-07-09 13:16:34 +00:00
|
|
|
LV_LITTLE_ENDIAN, 4, {1, 2},
|
2013-05-07 15:27:43 +00:00
|
|
|
4+4+4, 8, 512, -1, vmdk4GetBackingStore, NULL
|
2010-06-14 15:39:32 +00:00
|
|
|
},
|
2009-09-29 08:34:48 +00:00
|
|
|
};
|
2010-06-14 15:39:32 +00:00
|
|
|
verify(ARRAY_CARDINALITY(fileTypeInfo) == VIR_STORAGE_FILE_LAST);
|
2009-09-29 08:34:48 +00:00
|
|
|
|
2013-05-07 15:27:43 +00:00
|
|
|
/* qcow2 compatible features in the order they appear on-disk */
|
|
|
|
enum qcow2CompatibleFeature {
|
|
|
|
QCOW2_COMPATIBLE_FEATURE_LAZY_REFCOUNTS = 0,
|
|
|
|
|
|
|
|
QCOW2_COMPATIBLE_FEATURE_LAST
|
|
|
|
};
|
|
|
|
|
|
|
|
/* conversion to virStorageFileFeature */
|
|
|
|
static const int qcow2CompatibleFeatureArray[] = {
|
|
|
|
VIR_STORAGE_FILE_FEATURE_LAZY_REFCOUNTS,
|
|
|
|
};
|
|
|
|
verify(ARRAY_CARDINALITY(qcow2CompatibleFeatureArray) ==
|
|
|
|
QCOW2_COMPATIBLE_FEATURE_LAST);
|
|
|
|
|
2009-09-29 08:34:48 +00:00
|
|
|
static int
|
2010-02-04 22:46:55 +00:00
|
|
|
cowGetBackingStore(char **res,
|
2010-06-14 14:53:59 +00:00
|
|
|
int *format,
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
const char *buf,
|
2009-09-29 08:34:48 +00:00
|
|
|
size_t buf_size)
|
|
|
|
{
|
|
|
|
#define COW_FILENAME_MAXLEN 1024
|
|
|
|
*res = NULL;
|
2010-06-14 14:53:59 +00:00
|
|
|
*format = VIR_STORAGE_FILE_AUTO;
|
|
|
|
|
2009-09-29 08:34:48 +00:00
|
|
|
if (buf_size < 4+4+ COW_FILENAME_MAXLEN)
|
|
|
|
return BACKING_STORE_INVALID;
|
storage: list more file types
When an image has no backing file, using VIR_STORAGE_FILE_AUTO
for its type is a bit confusing. Additionally, a future patch
would like to reserve a default value for the case of no file
type specified in the XML, but different from the current use
of -1 to imply probing, since probing is not always safe.
Also, a couple of file types were missing compared to supported
code: libxl supports 'vhd', and qemu supports 'fat' for directories
passed through as a file system.
* src/util/storage_file.h (virStorageFileFormat): Add
VIR_STORAGE_FILE_NONE, VIR_STORAGE_FILE_FAT, VIR_STORAGE_FILE_VHD.
* src/util/storage_file.c (virStorageFileMatchesVersion): Match
documentation when version probing not supported.
(cowGetBackingStore, qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStoreFormat, qedGetBackingStore)
(virStorageFileGetMetadataFromBuf)
(virStorageFileGetMetadataFromFD): Take NONE into account.
* src/conf/domain_conf.c (virDomainDiskDefForeachPath): Likewise.
* src/qemu/qemu_driver.c (qemuDomainGetBlockInfo): Likewise.
* src/conf/storage_conf.c (virStorageVolumeFormatFromString): New
function.
(poolTypeInfo): Use it.
2012-09-28 17:11:07 +00:00
|
|
|
if (buf[4+4] == '\0') { /* cow_header_v2.backing_file[0] */
|
|
|
|
*format = VIR_STORAGE_FILE_NONE;
|
2009-09-29 08:34:48 +00:00
|
|
|
return BACKING_STORE_OK;
|
storage: list more file types
When an image has no backing file, using VIR_STORAGE_FILE_AUTO
for its type is a bit confusing. Additionally, a future patch
would like to reserve a default value for the case of no file
type specified in the XML, but different from the current use
of -1 to imply probing, since probing is not always safe.
Also, a couple of file types were missing compared to supported
code: libxl supports 'vhd', and qemu supports 'fat' for directories
passed through as a file system.
* src/util/storage_file.h (virStorageFileFormat): Add
VIR_STORAGE_FILE_NONE, VIR_STORAGE_FILE_FAT, VIR_STORAGE_FILE_VHD.
* src/util/storage_file.c (virStorageFileMatchesVersion): Match
documentation when version probing not supported.
(cowGetBackingStore, qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStoreFormat, qedGetBackingStore)
(virStorageFileGetMetadataFromBuf)
(virStorageFileGetMetadataFromFD): Take NONE into account.
* src/conf/domain_conf.c (virDomainDiskDefForeachPath): Likewise.
* src/qemu/qemu_driver.c (qemuDomainGetBlockInfo): Likewise.
* src/conf/storage_conf.c (virStorageVolumeFormatFromString): New
function.
(poolTypeInfo): Use it.
2012-09-28 17:11:07 +00:00
|
|
|
}
|
2009-09-29 08:34:48 +00:00
|
|
|
|
2013-05-24 07:19:51 +00:00
|
|
|
if (VIR_STRNDUP(*res, (const char*)buf + 4 + 4, COW_FILENAME_MAXLEN) < 0)
|
2009-09-29 08:34:48 +00:00
|
|
|
return BACKING_STORE_ERROR;
|
|
|
|
return BACKING_STORE_OK;
|
|
|
|
}
|
|
|
|
|
2010-06-14 14:53:59 +00:00
|
|
|
|
|
|
|
static int
|
|
|
|
qcow2GetBackingStoreFormat(int *format,
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
const char *buf,
|
2010-06-14 14:53:59 +00:00
|
|
|
size_t buf_size,
|
|
|
|
size_t extension_start,
|
|
|
|
size_t extension_end)
|
|
|
|
{
|
|
|
|
size_t offset = extension_start;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* The extensions take format of
|
|
|
|
*
|
|
|
|
* int32: magic
|
|
|
|
* int32: length
|
|
|
|
* byte[length]: payload
|
|
|
|
*
|
|
|
|
* Unknown extensions can be ignored by skipping
|
|
|
|
* over "length" bytes in the data stream.
|
|
|
|
*/
|
|
|
|
while (offset < (buf_size-8) &&
|
|
|
|
offset < (extension_end-8)) {
|
2013-02-07 01:57:13 +00:00
|
|
|
unsigned int magic = virReadBufInt32BE(buf + offset);
|
|
|
|
unsigned int len = virReadBufInt32BE(buf + offset + 4);
|
2010-06-14 14:53:59 +00:00
|
|
|
|
|
|
|
offset += 8;
|
|
|
|
|
|
|
|
if ((offset + len) < offset)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if ((offset + len) > buf_size)
|
|
|
|
break;
|
|
|
|
|
|
|
|
switch (magic) {
|
|
|
|
case QCOW2_HDR_EXTENSION_END:
|
|
|
|
goto done;
|
|
|
|
|
|
|
|
case QCOW2_HDR_EXTENSION_BACKING_FORMAT:
|
|
|
|
if (buf[offset+len] != '\0')
|
|
|
|
break;
|
|
|
|
*format = virStorageFileFormatTypeFromString(
|
|
|
|
((const char *)buf)+offset);
|
storage: list more file types
When an image has no backing file, using VIR_STORAGE_FILE_AUTO
for its type is a bit confusing. Additionally, a future patch
would like to reserve a default value for the case of no file
type specified in the XML, but different from the current use
of -1 to imply probing, since probing is not always safe.
Also, a couple of file types were missing compared to supported
code: libxl supports 'vhd', and qemu supports 'fat' for directories
passed through as a file system.
* src/util/storage_file.h (virStorageFileFormat): Add
VIR_STORAGE_FILE_NONE, VIR_STORAGE_FILE_FAT, VIR_STORAGE_FILE_VHD.
* src/util/storage_file.c (virStorageFileMatchesVersion): Match
documentation when version probing not supported.
(cowGetBackingStore, qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStoreFormat, qedGetBackingStore)
(virStorageFileGetMetadataFromBuf)
(virStorageFileGetMetadataFromFD): Take NONE into account.
* src/conf/domain_conf.c (virDomainDiskDefForeachPath): Likewise.
* src/qemu/qemu_driver.c (qemuDomainGetBlockInfo): Likewise.
* src/conf/storage_conf.c (virStorageVolumeFormatFromString): New
function.
(poolTypeInfo): Use it.
2012-09-28 17:11:07 +00:00
|
|
|
if (*format <= VIR_STORAGE_FILE_NONE)
|
|
|
|
return -1;
|
2010-06-14 14:53:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
offset += len;
|
|
|
|
}
|
|
|
|
|
2014-03-25 06:53:22 +00:00
|
|
|
done:
|
2010-06-14 14:53:59 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-09-29 08:34:48 +00:00
|
|
|
static int
|
2010-02-04 22:46:55 +00:00
|
|
|
qcowXGetBackingStore(char **res,
|
2010-06-14 14:53:59 +00:00
|
|
|
int *format,
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
const char *buf,
|
2010-06-14 14:53:59 +00:00
|
|
|
size_t buf_size,
|
|
|
|
bool isQCow2)
|
2009-09-29 08:34:48 +00:00
|
|
|
{
|
|
|
|
unsigned long long offset;
|
2011-06-02 23:52:16 +00:00
|
|
|
unsigned int size;
|
2013-05-07 15:27:43 +00:00
|
|
|
unsigned long long start;
|
|
|
|
int version;
|
2009-09-29 08:34:48 +00:00
|
|
|
|
|
|
|
*res = NULL;
|
2010-06-14 14:53:59 +00:00
|
|
|
if (format)
|
|
|
|
*format = VIR_STORAGE_FILE_AUTO;
|
|
|
|
|
|
|
|
if (buf_size < QCOWX_HDR_BACKING_FILE_OFFSET+8+4)
|
2009-09-29 08:34:48 +00:00
|
|
|
return BACKING_STORE_INVALID;
|
2013-02-07 01:57:13 +00:00
|
|
|
offset = virReadBufInt64BE(buf + QCOWX_HDR_BACKING_FILE_OFFSET);
|
2009-09-29 08:34:48 +00:00
|
|
|
if (offset > buf_size)
|
|
|
|
return BACKING_STORE_INVALID;
|
2014-09-15 14:16:25 +00:00
|
|
|
|
|
|
|
if (offset == 0) {
|
|
|
|
if (format)
|
|
|
|
*format = VIR_STORAGE_FILE_NONE;
|
|
|
|
return BACKING_STORE_OK;
|
|
|
|
}
|
|
|
|
|
2013-02-07 01:57:13 +00:00
|
|
|
size = virReadBufInt32BE(buf + QCOWX_HDR_BACKING_FILE_SIZE);
|
storage: list more file types
When an image has no backing file, using VIR_STORAGE_FILE_AUTO
for its type is a bit confusing. Additionally, a future patch
would like to reserve a default value for the case of no file
type specified in the XML, but different from the current use
of -1 to imply probing, since probing is not always safe.
Also, a couple of file types were missing compared to supported
code: libxl supports 'vhd', and qemu supports 'fat' for directories
passed through as a file system.
* src/util/storage_file.h (virStorageFileFormat): Add
VIR_STORAGE_FILE_NONE, VIR_STORAGE_FILE_FAT, VIR_STORAGE_FILE_VHD.
* src/util/storage_file.c (virStorageFileMatchesVersion): Match
documentation when version probing not supported.
(cowGetBackingStore, qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStoreFormat, qedGetBackingStore)
(virStorageFileGetMetadataFromBuf)
(virStorageFileGetMetadataFromFD): Take NONE into account.
* src/conf/domain_conf.c (virDomainDiskDefForeachPath): Likewise.
* src/qemu/qemu_driver.c (qemuDomainGetBlockInfo): Likewise.
* src/conf/storage_conf.c (virStorageVolumeFormatFromString): New
function.
(poolTypeInfo): Use it.
2012-09-28 17:11:07 +00:00
|
|
|
if (size == 0) {
|
|
|
|
if (format)
|
|
|
|
*format = VIR_STORAGE_FILE_NONE;
|
2009-09-29 08:34:48 +00:00
|
|
|
return BACKING_STORE_OK;
|
storage: list more file types
When an image has no backing file, using VIR_STORAGE_FILE_AUTO
for its type is a bit confusing. Additionally, a future patch
would like to reserve a default value for the case of no file
type specified in the XML, but different from the current use
of -1 to imply probing, since probing is not always safe.
Also, a couple of file types were missing compared to supported
code: libxl supports 'vhd', and qemu supports 'fat' for directories
passed through as a file system.
* src/util/storage_file.h (virStorageFileFormat): Add
VIR_STORAGE_FILE_NONE, VIR_STORAGE_FILE_FAT, VIR_STORAGE_FILE_VHD.
* src/util/storage_file.c (virStorageFileMatchesVersion): Match
documentation when version probing not supported.
(cowGetBackingStore, qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStoreFormat, qedGetBackingStore)
(virStorageFileGetMetadataFromBuf)
(virStorageFileGetMetadataFromFD): Take NONE into account.
* src/conf/domain_conf.c (virDomainDiskDefForeachPath): Likewise.
* src/qemu/qemu_driver.c (qemuDomainGetBlockInfo): Likewise.
* src/conf/storage_conf.c (virStorageVolumeFormatFromString): New
function.
(poolTypeInfo): Use it.
2012-09-28 17:11:07 +00:00
|
|
|
}
|
2014-09-15 14:16:25 +00:00
|
|
|
if (size > 1023)
|
2009-09-29 08:34:48 +00:00
|
|
|
return BACKING_STORE_INVALID;
|
2014-09-15 14:16:25 +00:00
|
|
|
if (offset + size > buf_size || offset + size < offset)
|
2009-09-29 08:34:48 +00:00
|
|
|
return BACKING_STORE_INVALID;
|
2013-07-04 10:17:18 +00:00
|
|
|
if (VIR_ALLOC_N(*res, size + 1) < 0)
|
2009-09-29 08:34:48 +00:00
|
|
|
return BACKING_STORE_ERROR;
|
|
|
|
memcpy(*res, buf + offset, size);
|
|
|
|
(*res)[size] = '\0';
|
2010-06-14 14:53:59 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Traditionally QCow2 files had a layout of
|
|
|
|
*
|
|
|
|
* [header]
|
|
|
|
* [backingStoreName]
|
|
|
|
*
|
|
|
|
* Although the backingStoreName typically followed
|
|
|
|
* the header immediately, this was not required by
|
|
|
|
* the format. By specifying a higher byte offset for
|
|
|
|
* the backing file offset in the header, it was
|
|
|
|
* possible to leave space between the header and
|
|
|
|
* start of backingStore.
|
|
|
|
*
|
|
|
|
* This hack is now used to store extensions to the
|
|
|
|
* qcow2 format:
|
|
|
|
*
|
|
|
|
* [header]
|
|
|
|
* [extensions]
|
|
|
|
* [backingStoreName]
|
|
|
|
*
|
|
|
|
* Thus the file region to search for extensions is
|
|
|
|
* between the end of the header (QCOW2_HDR_TOTAL_SIZE)
|
|
|
|
* and the start of the backingStoreName (offset)
|
2013-05-07 15:27:43 +00:00
|
|
|
*
|
|
|
|
* for qcow2 v3 images, the length of the header
|
|
|
|
* is stored at QCOW2v3_HDR_SIZE
|
2010-06-14 14:53:59 +00:00
|
|
|
*/
|
2013-05-07 15:27:43 +00:00
|
|
|
if (isQCow2 && format) {
|
|
|
|
version = virReadBufInt32BE(buf + QCOWX_HDR_VERSION);
|
|
|
|
if (version == 2)
|
|
|
|
start = QCOW2_HDR_TOTAL_SIZE;
|
|
|
|
else
|
|
|
|
start = virReadBufInt32BE(buf + QCOW2v3_HDR_SIZE);
|
|
|
|
if (qcow2GetBackingStoreFormat(format, buf, buf_size,
|
|
|
|
start, offset) < 0)
|
|
|
|
return BACKING_STORE_INVALID;
|
|
|
|
}
|
2010-06-14 14:53:59 +00:00
|
|
|
|
2009-09-29 08:34:48 +00:00
|
|
|
return BACKING_STORE_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-06-14 14:53:59 +00:00
|
|
|
static int
|
|
|
|
qcow1GetBackingStore(char **res,
|
|
|
|
int *format,
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
const char *buf,
|
2010-06-14 14:53:59 +00:00
|
|
|
size_t buf_size)
|
|
|
|
{
|
storage: list more file types
When an image has no backing file, using VIR_STORAGE_FILE_AUTO
for its type is a bit confusing. Additionally, a future patch
would like to reserve a default value for the case of no file
type specified in the XML, but different from the current use
of -1 to imply probing, since probing is not always safe.
Also, a couple of file types were missing compared to supported
code: libxl supports 'vhd', and qemu supports 'fat' for directories
passed through as a file system.
* src/util/storage_file.h (virStorageFileFormat): Add
VIR_STORAGE_FILE_NONE, VIR_STORAGE_FILE_FAT, VIR_STORAGE_FILE_VHD.
* src/util/storage_file.c (virStorageFileMatchesVersion): Match
documentation when version probing not supported.
(cowGetBackingStore, qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStoreFormat, qedGetBackingStore)
(virStorageFileGetMetadataFromBuf)
(virStorageFileGetMetadataFromFD): Take NONE into account.
* src/conf/domain_conf.c (virDomainDiskDefForeachPath): Likewise.
* src/qemu/qemu_driver.c (qemuDomainGetBlockInfo): Likewise.
* src/conf/storage_conf.c (virStorageVolumeFormatFromString): New
function.
(poolTypeInfo): Use it.
2012-09-28 17:11:07 +00:00
|
|
|
int ret;
|
|
|
|
|
2010-06-14 14:53:59 +00:00
|
|
|
/* QCow1 doesn't have the extensions capability
|
|
|
|
* used to store backing format */
|
|
|
|
*format = VIR_STORAGE_FILE_AUTO;
|
storage: list more file types
When an image has no backing file, using VIR_STORAGE_FILE_AUTO
for its type is a bit confusing. Additionally, a future patch
would like to reserve a default value for the case of no file
type specified in the XML, but different from the current use
of -1 to imply probing, since probing is not always safe.
Also, a couple of file types were missing compared to supported
code: libxl supports 'vhd', and qemu supports 'fat' for directories
passed through as a file system.
* src/util/storage_file.h (virStorageFileFormat): Add
VIR_STORAGE_FILE_NONE, VIR_STORAGE_FILE_FAT, VIR_STORAGE_FILE_VHD.
* src/util/storage_file.c (virStorageFileMatchesVersion): Match
documentation when version probing not supported.
(cowGetBackingStore, qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStoreFormat, qedGetBackingStore)
(virStorageFileGetMetadataFromBuf)
(virStorageFileGetMetadataFromFD): Take NONE into account.
* src/conf/domain_conf.c (virDomainDiskDefForeachPath): Likewise.
* src/qemu/qemu_driver.c (qemuDomainGetBlockInfo): Likewise.
* src/conf/storage_conf.c (virStorageVolumeFormatFromString): New
function.
(poolTypeInfo): Use it.
2012-09-28 17:11:07 +00:00
|
|
|
ret = qcowXGetBackingStore(res, NULL, buf, buf_size, false);
|
|
|
|
if (ret == 0 && *buf == '\0')
|
|
|
|
*format = VIR_STORAGE_FILE_NONE;
|
|
|
|
return ret;
|
2010-06-14 14:53:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
qcow2GetBackingStore(char **res,
|
|
|
|
int *format,
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
const char *buf,
|
2010-06-14 14:53:59 +00:00
|
|
|
size_t buf_size)
|
|
|
|
{
|
|
|
|
return qcowXGetBackingStore(res, format, buf, buf_size, true);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-09-29 08:34:48 +00:00
|
|
|
static int
|
2010-02-04 22:46:55 +00:00
|
|
|
vmdk4GetBackingStore(char **res,
|
2010-06-14 14:53:59 +00:00
|
|
|
int *format,
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
const char *buf,
|
2009-09-29 08:34:48 +00:00
|
|
|
size_t buf_size)
|
|
|
|
{
|
|
|
|
static const char prefix[] = "parentFileNameHint=\"";
|
2010-06-15 13:58:10 +00:00
|
|
|
char *desc, *start, *end;
|
2009-09-29 08:34:48 +00:00
|
|
|
size_t len;
|
2010-06-15 13:58:10 +00:00
|
|
|
int ret = BACKING_STORE_ERROR;
|
|
|
|
|
2013-11-05 21:12:02 +00:00
|
|
|
if (VIR_ALLOC_N(desc, VIR_STORAGE_MAX_HEADER) < 0)
|
2010-06-15 13:58:10 +00:00
|
|
|
goto cleanup;
|
2009-09-29 08:34:48 +00:00
|
|
|
|
|
|
|
*res = NULL;
|
2010-06-14 14:53:59 +00:00
|
|
|
/*
|
|
|
|
* Technically this should have been VMDK, since
|
|
|
|
* VMDK spec / VMWare impl only support VMDK backed
|
|
|
|
* by VMDK. QEMU isn't following this though and
|
|
|
|
* does probing on VMDK backing files, hence we set
|
|
|
|
* AUTO
|
|
|
|
*/
|
|
|
|
*format = VIR_STORAGE_FILE_AUTO;
|
2009-09-29 08:34:48 +00:00
|
|
|
|
2010-06-15 13:58:10 +00:00
|
|
|
if (buf_size <= 0x200) {
|
|
|
|
ret = BACKING_STORE_INVALID;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-09-29 08:34:48 +00:00
|
|
|
len = buf_size - 0x200;
|
2013-11-05 21:12:02 +00:00
|
|
|
if (len > VIR_STORAGE_MAX_HEADER)
|
|
|
|
len = VIR_STORAGE_MAX_HEADER;
|
2009-09-29 08:34:48 +00:00
|
|
|
memcpy(desc, buf + 0x200, len);
|
|
|
|
desc[len] = '\0';
|
|
|
|
start = strstr(desc, prefix);
|
2010-06-15 13:58:10 +00:00
|
|
|
if (start == NULL) {
|
storage: list more file types
When an image has no backing file, using VIR_STORAGE_FILE_AUTO
for its type is a bit confusing. Additionally, a future patch
would like to reserve a default value for the case of no file
type specified in the XML, but different from the current use
of -1 to imply probing, since probing is not always safe.
Also, a couple of file types were missing compared to supported
code: libxl supports 'vhd', and qemu supports 'fat' for directories
passed through as a file system.
* src/util/storage_file.h (virStorageFileFormat): Add
VIR_STORAGE_FILE_NONE, VIR_STORAGE_FILE_FAT, VIR_STORAGE_FILE_VHD.
* src/util/storage_file.c (virStorageFileMatchesVersion): Match
documentation when version probing not supported.
(cowGetBackingStore, qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStoreFormat, qedGetBackingStore)
(virStorageFileGetMetadataFromBuf)
(virStorageFileGetMetadataFromFD): Take NONE into account.
* src/conf/domain_conf.c (virDomainDiskDefForeachPath): Likewise.
* src/qemu/qemu_driver.c (qemuDomainGetBlockInfo): Likewise.
* src/conf/storage_conf.c (virStorageVolumeFormatFromString): New
function.
(poolTypeInfo): Use it.
2012-09-28 17:11:07 +00:00
|
|
|
*format = VIR_STORAGE_FILE_NONE;
|
2010-06-15 13:58:10 +00:00
|
|
|
ret = BACKING_STORE_OK;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-09-29 08:34:48 +00:00
|
|
|
start += strlen(prefix);
|
|
|
|
end = strchr(start, '"');
|
2010-06-15 13:58:10 +00:00
|
|
|
if (end == NULL) {
|
|
|
|
ret = BACKING_STORE_INVALID;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (end == start) {
|
storage: list more file types
When an image has no backing file, using VIR_STORAGE_FILE_AUTO
for its type is a bit confusing. Additionally, a future patch
would like to reserve a default value for the case of no file
type specified in the XML, but different from the current use
of -1 to imply probing, since probing is not always safe.
Also, a couple of file types were missing compared to supported
code: libxl supports 'vhd', and qemu supports 'fat' for directories
passed through as a file system.
* src/util/storage_file.h (virStorageFileFormat): Add
VIR_STORAGE_FILE_NONE, VIR_STORAGE_FILE_FAT, VIR_STORAGE_FILE_VHD.
* src/util/storage_file.c (virStorageFileMatchesVersion): Match
documentation when version probing not supported.
(cowGetBackingStore, qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStoreFormat, qedGetBackingStore)
(virStorageFileGetMetadataFromBuf)
(virStorageFileGetMetadataFromFD): Take NONE into account.
* src/conf/domain_conf.c (virDomainDiskDefForeachPath): Likewise.
* src/qemu/qemu_driver.c (qemuDomainGetBlockInfo): Likewise.
* src/conf/storage_conf.c (virStorageVolumeFormatFromString): New
function.
(poolTypeInfo): Use it.
2012-09-28 17:11:07 +00:00
|
|
|
*format = VIR_STORAGE_FILE_NONE;
|
2010-06-15 13:58:10 +00:00
|
|
|
ret = BACKING_STORE_OK;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2009-09-29 08:34:48 +00:00
|
|
|
*end = '\0';
|
2013-05-24 07:19:51 +00:00
|
|
|
if (VIR_STRDUP(*res, start) < 0)
|
2010-06-15 13:58:10 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = BACKING_STORE_OK;
|
|
|
|
|
2014-03-25 06:53:22 +00:00
|
|
|
cleanup:
|
2010-06-15 13:58:10 +00:00
|
|
|
VIR_FREE(desc);
|
|
|
|
return ret;
|
2009-09-29 08:34:48 +00:00
|
|
|
}
|
|
|
|
|
2010-11-19 23:19:24 +00:00
|
|
|
static int
|
|
|
|
qedGetBackingStore(char **res,
|
|
|
|
int *format,
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
const char *buf,
|
2010-11-19 23:19:24 +00:00
|
|
|
size_t buf_size)
|
|
|
|
{
|
|
|
|
unsigned long long flags;
|
|
|
|
unsigned long offset, size;
|
|
|
|
|
|
|
|
*res = NULL;
|
|
|
|
/* Check if this image has a backing file */
|
|
|
|
if (buf_size < QED_HDR_FEATURES_OFFSET+8)
|
|
|
|
return BACKING_STORE_INVALID;
|
2013-02-07 01:57:13 +00:00
|
|
|
flags = virReadBufInt64LE(buf + QED_HDR_FEATURES_OFFSET);
|
storage: list more file types
When an image has no backing file, using VIR_STORAGE_FILE_AUTO
for its type is a bit confusing. Additionally, a future patch
would like to reserve a default value for the case of no file
type specified in the XML, but different from the current use
of -1 to imply probing, since probing is not always safe.
Also, a couple of file types were missing compared to supported
code: libxl supports 'vhd', and qemu supports 'fat' for directories
passed through as a file system.
* src/util/storage_file.h (virStorageFileFormat): Add
VIR_STORAGE_FILE_NONE, VIR_STORAGE_FILE_FAT, VIR_STORAGE_FILE_VHD.
* src/util/storage_file.c (virStorageFileMatchesVersion): Match
documentation when version probing not supported.
(cowGetBackingStore, qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStoreFormat, qedGetBackingStore)
(virStorageFileGetMetadataFromBuf)
(virStorageFileGetMetadataFromFD): Take NONE into account.
* src/conf/domain_conf.c (virDomainDiskDefForeachPath): Likewise.
* src/qemu/qemu_driver.c (qemuDomainGetBlockInfo): Likewise.
* src/conf/storage_conf.c (virStorageVolumeFormatFromString): New
function.
(poolTypeInfo): Use it.
2012-09-28 17:11:07 +00:00
|
|
|
if (!(flags & QED_F_BACKING_FILE)) {
|
|
|
|
*format = VIR_STORAGE_FILE_NONE;
|
2010-11-19 23:19:24 +00:00
|
|
|
return BACKING_STORE_OK;
|
storage: list more file types
When an image has no backing file, using VIR_STORAGE_FILE_AUTO
for its type is a bit confusing. Additionally, a future patch
would like to reserve a default value for the case of no file
type specified in the XML, but different from the current use
of -1 to imply probing, since probing is not always safe.
Also, a couple of file types were missing compared to supported
code: libxl supports 'vhd', and qemu supports 'fat' for directories
passed through as a file system.
* src/util/storage_file.h (virStorageFileFormat): Add
VIR_STORAGE_FILE_NONE, VIR_STORAGE_FILE_FAT, VIR_STORAGE_FILE_VHD.
* src/util/storage_file.c (virStorageFileMatchesVersion): Match
documentation when version probing not supported.
(cowGetBackingStore, qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStoreFormat, qedGetBackingStore)
(virStorageFileGetMetadataFromBuf)
(virStorageFileGetMetadataFromFD): Take NONE into account.
* src/conf/domain_conf.c (virDomainDiskDefForeachPath): Likewise.
* src/qemu/qemu_driver.c (qemuDomainGetBlockInfo): Likewise.
* src/conf/storage_conf.c (virStorageVolumeFormatFromString): New
function.
(poolTypeInfo): Use it.
2012-09-28 17:11:07 +00:00
|
|
|
}
|
2010-11-19 23:19:24 +00:00
|
|
|
|
|
|
|
/* Parse the backing file */
|
|
|
|
if (buf_size < QED_HDR_BACKING_FILE_OFFSET+8)
|
|
|
|
return BACKING_STORE_INVALID;
|
2013-02-07 01:57:13 +00:00
|
|
|
offset = virReadBufInt32LE(buf + QED_HDR_BACKING_FILE_OFFSET);
|
2010-11-19 23:19:24 +00:00
|
|
|
if (offset > buf_size)
|
|
|
|
return BACKING_STORE_INVALID;
|
2013-02-07 01:57:13 +00:00
|
|
|
size = virReadBufInt32LE(buf + QED_HDR_BACKING_FILE_SIZE);
|
2010-11-19 23:19:24 +00:00
|
|
|
if (size == 0)
|
|
|
|
return BACKING_STORE_OK;
|
|
|
|
if (offset + size > buf_size || offset + size < offset)
|
|
|
|
return BACKING_STORE_INVALID;
|
2013-07-04 10:17:18 +00:00
|
|
|
if (VIR_ALLOC_N(*res, size + 1) < 0)
|
2010-11-19 23:19:24 +00:00
|
|
|
return BACKING_STORE_ERROR;
|
|
|
|
memcpy(*res, buf + offset, size);
|
|
|
|
(*res)[size] = '\0';
|
|
|
|
|
storage: list more file types
When an image has no backing file, using VIR_STORAGE_FILE_AUTO
for its type is a bit confusing. Additionally, a future patch
would like to reserve a default value for the case of no file
type specified in the XML, but different from the current use
of -1 to imply probing, since probing is not always safe.
Also, a couple of file types were missing compared to supported
code: libxl supports 'vhd', and qemu supports 'fat' for directories
passed through as a file system.
* src/util/storage_file.h (virStorageFileFormat): Add
VIR_STORAGE_FILE_NONE, VIR_STORAGE_FILE_FAT, VIR_STORAGE_FILE_VHD.
* src/util/storage_file.c (virStorageFileMatchesVersion): Match
documentation when version probing not supported.
(cowGetBackingStore, qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStoreFormat, qedGetBackingStore)
(virStorageFileGetMetadataFromBuf)
(virStorageFileGetMetadataFromFD): Take NONE into account.
* src/conf/domain_conf.c (virDomainDiskDefForeachPath): Likewise.
* src/qemu/qemu_driver.c (qemuDomainGetBlockInfo): Likewise.
* src/conf/storage_conf.c (virStorageVolumeFormatFromString): New
function.
(poolTypeInfo): Use it.
2012-09-28 17:11:07 +00:00
|
|
|
if (flags & QED_F_BACKING_FORMAT_NO_PROBE)
|
|
|
|
*format = VIR_STORAGE_FILE_RAW;
|
|
|
|
else
|
|
|
|
*format = VIR_STORAGE_FILE_AUTO_SAFE;
|
2010-11-19 23:19:24 +00:00
|
|
|
|
|
|
|
return BACKING_STORE_OK;
|
|
|
|
}
|
|
|
|
|
2010-06-15 13:58:10 +00:00
|
|
|
|
|
|
|
static bool
|
|
|
|
virStorageFileMatchesMagic(int format,
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
char *buf,
|
2010-06-15 13:58:10 +00:00
|
|
|
size_t buflen)
|
2009-09-29 08:34:48 +00:00
|
|
|
{
|
2010-06-15 13:58:10 +00:00
|
|
|
int mlen;
|
2013-02-04 19:16:22 +00:00
|
|
|
int magicOffset = fileTypeInfo[format].magicOffset;
|
|
|
|
const char *magic = fileTypeInfo[format].magic;
|
2009-09-29 08:34:48 +00:00
|
|
|
|
2013-02-04 19:16:22 +00:00
|
|
|
if (magic == NULL)
|
2010-06-15 13:58:10 +00:00
|
|
|
return false;
|
2010-05-18 05:53:31 +00:00
|
|
|
|
2010-06-15 13:58:10 +00:00
|
|
|
/* Validate magic data */
|
2013-02-04 19:16:22 +00:00
|
|
|
mlen = strlen(magic);
|
|
|
|
if (magicOffset + mlen > buflen)
|
2010-06-15 13:58:10 +00:00
|
|
|
return false;
|
2009-09-29 08:34:48 +00:00
|
|
|
|
2013-02-04 19:16:22 +00:00
|
|
|
if (memcmp(buf + magicOffset, magic, mlen) != 0)
|
2010-06-15 13:58:10 +00:00
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static bool
|
|
|
|
virStorageFileMatchesExtension(int format,
|
|
|
|
const char *path)
|
|
|
|
{
|
|
|
|
if (fileTypeInfo[format].extension == NULL)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (virFileHasSuffix(path, fileTypeInfo[format].extension))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static bool
|
|
|
|
virStorageFileMatchesVersion(int format,
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
char *buf,
|
2010-06-15 13:58:10 +00:00
|
|
|
size_t buflen)
|
|
|
|
{
|
Convert 'int i' to 'size_t i' in src/util/ files
Convert the type of loop iterators named 'i', 'j', k',
'ii', 'jj', 'kk', to be 'size_t' instead of 'int' or
'unsigned int', also santizing 'ii', 'jj', 'kk' to use
the normal 'i', 'j', 'k' naming
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2013-07-08 14:09:33 +00:00
|
|
|
int version;
|
|
|
|
size_t i;
|
2010-06-15 13:58:10 +00:00
|
|
|
|
|
|
|
/* Validate version number info */
|
|
|
|
if (fileTypeInfo[format].versionOffset == -1)
|
storage: list more file types
When an image has no backing file, using VIR_STORAGE_FILE_AUTO
for its type is a bit confusing. Additionally, a future patch
would like to reserve a default value for the case of no file
type specified in the XML, but different from the current use
of -1 to imply probing, since probing is not always safe.
Also, a couple of file types were missing compared to supported
code: libxl supports 'vhd', and qemu supports 'fat' for directories
passed through as a file system.
* src/util/storage_file.h (virStorageFileFormat): Add
VIR_STORAGE_FILE_NONE, VIR_STORAGE_FILE_FAT, VIR_STORAGE_FILE_VHD.
* src/util/storage_file.c (virStorageFileMatchesVersion): Match
documentation when version probing not supported.
(cowGetBackingStore, qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStoreFormat, qedGetBackingStore)
(virStorageFileGetMetadataFromBuf)
(virStorageFileGetMetadataFromFD): Take NONE into account.
* src/conf/domain_conf.c (virDomainDiskDefForeachPath): Likewise.
* src/qemu/qemu_driver.c (qemuDomainGetBlockInfo): Likewise.
* src/conf/storage_conf.c (virStorageVolumeFormatFromString): New
function.
(poolTypeInfo): Use it.
2012-09-28 17:11:07 +00:00
|
|
|
return false;
|
2010-06-15 13:58:10 +00:00
|
|
|
|
2012-12-13 14:25:10 +00:00
|
|
|
/* -2 == non-versioned file format, so trivially match */
|
|
|
|
if (fileTypeInfo[format].versionOffset == -2)
|
|
|
|
return true;
|
|
|
|
|
2010-06-15 13:58:10 +00:00
|
|
|
if ((fileTypeInfo[format].versionOffset + 4) > buflen)
|
|
|
|
return false;
|
|
|
|
|
2013-02-07 01:57:13 +00:00
|
|
|
if (fileTypeInfo[format].endian == LV_LITTLE_ENDIAN)
|
|
|
|
version = virReadBufInt32LE(buf + fileTypeInfo[format].versionOffset);
|
|
|
|
else
|
|
|
|
version = virReadBufInt32BE(buf + fileTypeInfo[format].versionOffset);
|
2012-12-13 14:23:50 +00:00
|
|
|
|
2013-05-07 15:27:43 +00:00
|
|
|
for (i = 0;
|
|
|
|
i < FILE_TYPE_VERSIONS_LAST && fileTypeInfo[format].versionNumbers[i];
|
|
|
|
i++) {
|
|
|
|
VIR_DEBUG("Compare detected version %d vs one of the expected versions %d",
|
|
|
|
version, fileTypeInfo[format].versionNumbers[i]);
|
|
|
|
if (version == fileTypeInfo[format].versionNumbers[i])
|
|
|
|
return true;
|
|
|
|
}
|
2009-09-29 08:34:48 +00:00
|
|
|
|
2013-05-07 15:27:43 +00:00
|
|
|
return false;
|
2010-06-15 13:58:10 +00:00
|
|
|
}
|
2009-09-29 08:34:48 +00:00
|
|
|
|
2014-04-09 22:08:42 +00:00
|
|
|
bool
|
|
|
|
virStorageIsFile(const char *backing)
|
2010-11-03 15:50:11 +00:00
|
|
|
{
|
2014-04-09 22:08:42 +00:00
|
|
|
char *colon;
|
|
|
|
char *slash;
|
|
|
|
|
|
|
|
if (!backing)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
colon = strchr(backing, ':');
|
|
|
|
slash = strchr(backing, '/');
|
2013-11-04 21:27:42 +00:00
|
|
|
|
|
|
|
/* Reject anything that looks like a protocol (such as nbd: or
|
|
|
|
* rbd:); if someone really does want a relative file name that
|
|
|
|
* includes ':', they can always prefix './'. */
|
|
|
|
if (colon && (!slash || colon < slash))
|
2010-11-03 15:50:11 +00:00
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
2009-09-29 08:34:48 +00:00
|
|
|
|
2014-04-29 03:40:22 +00:00
|
|
|
|
2014-04-25 11:23:50 +00:00
|
|
|
static bool
|
|
|
|
virStorageIsRelative(const char *backing)
|
|
|
|
{
|
|
|
|
if (backing[0] == '/')
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (!virStorageIsFile(backing))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-07-07 10:56:23 +00:00
|
|
|
int
|
2013-02-06 23:54:08 +00:00
|
|
|
virStorageFileProbeFormatFromBuf(const char *path,
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
char *buf,
|
2013-02-06 23:54:08 +00:00
|
|
|
size_t buflen)
|
|
|
|
{
|
|
|
|
int format = VIR_STORAGE_FILE_RAW;
|
Convert 'int i' to 'size_t i' in src/util/ files
Convert the type of loop iterators named 'i', 'j', k',
'ii', 'jj', 'kk', to be 'size_t' instead of 'int' or
'unsigned int', also santizing 'ii', 'jj', 'kk' to use
the normal 'i', 'j', 'k' naming
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2013-07-08 14:09:33 +00:00
|
|
|
size_t i;
|
2013-02-06 23:54:08 +00:00
|
|
|
int possibleFormat = VIR_STORAGE_FILE_RAW;
|
2013-11-05 21:12:02 +00:00
|
|
|
VIR_DEBUG("path=%s, buf=%p, buflen=%zu", path, buf, buflen);
|
2013-02-06 23:54:08 +00:00
|
|
|
|
|
|
|
/* First check file magic */
|
2013-05-21 07:58:16 +00:00
|
|
|
for (i = 0; i < VIR_STORAGE_FILE_LAST; i++) {
|
2013-02-06 23:54:08 +00:00
|
|
|
if (virStorageFileMatchesMagic(i, buf, buflen)) {
|
|
|
|
if (!virStorageFileMatchesVersion(i, buf, buflen)) {
|
|
|
|
possibleFormat = i;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
format = i;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (possibleFormat != VIR_STORAGE_FILE_RAW)
|
|
|
|
VIR_WARN("File %s matches %s magic, but version is wrong. "
|
|
|
|
"Please report new version to libvir-list@redhat.com",
|
|
|
|
path, virStorageFileFormatTypeToString(possibleFormat));
|
|
|
|
|
|
|
|
/* No magic, so check file extension */
|
2013-05-21 07:58:16 +00:00
|
|
|
for (i = 0; i < VIR_STORAGE_FILE_LAST; i++) {
|
2013-02-06 23:54:08 +00:00
|
|
|
if (virStorageFileMatchesExtension(i, path)) {
|
|
|
|
format = i;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-25 06:53:22 +00:00
|
|
|
cleanup:
|
2013-02-06 23:54:08 +00:00
|
|
|
VIR_DEBUG("format=%d", format);
|
|
|
|
return format;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-05-07 15:27:43 +00:00
|
|
|
static int
|
|
|
|
qcow2GetFeatures(virBitmapPtr *features,
|
|
|
|
int format,
|
storage: use simpler 'char *'
'unsigned char *' makes sense if you are doing math on bytes and
don't want to worry about wraparound from a signed 'char'; but
since all we are doing is memcmp() or virReadBufInt*[LB]E(), which
are both safe on either type of char, and since read() prefers to
operate on 'char *', it's simpler to avoid casts by just typing
things as 'char *' from the get-go. [Technically, read can
operate on an 'unsigned char *' thanks to the C rule that any
pointer can be implicitly converted to 'char *' for legacy K&R
compatibility; but where this patch saves us is if we try to use
virfile.h functions that take 'char **' in order to allocate the
buffer, where the compiler would barf on type mismatch.]
* src/util/virstoragefile.c (FileTypeInfo): Avoid unsigned char.
(cowGetBackingStore, qcow2GetBackingStoreFormat)
(qcowXGetBackingStore, qcow1GetBackingStore)
(qcow2GetBackingStore, vmdk4GetBackingStore, qedGetBackingStore)
(virStorageFileMatchesMagic, virStorageFileMatchesVersion)
(virStorageFileProbeFormatFromBuf, qcow2GetFeatures)
(virStorageFileGetMetadataInternal)
(virStorageFileProbeFormatFromFD): Simplify clients.
Signed-off-by: Eric Blake <eblake@redhat.com>
2013-11-05 17:44:14 +00:00
|
|
|
char *buf,
|
2013-05-07 15:27:43 +00:00
|
|
|
ssize_t len)
|
|
|
|
{
|
|
|
|
int version = -1;
|
|
|
|
virBitmapPtr feat = NULL;
|
|
|
|
uint64_t bits;
|
Convert 'int i' to 'size_t i' in src/util/ files
Convert the type of loop iterators named 'i', 'j', k',
'ii', 'jj', 'kk', to be 'size_t' instead of 'int' or
'unsigned int', also santizing 'ii', 'jj', 'kk' to use
the normal 'i', 'j', 'k' naming
Signed-off-by: Daniel P. Berrange <berrange@redhat.com>
2013-07-08 14:09:33 +00:00
|
|
|
size_t i;
|
2013-05-07 15:27:43 +00:00
|
|
|
|
|
|
|
version = virReadBufInt32BE(buf + fileTypeInfo[format].versionOffset);
|
|
|
|
|
|
|
|
if (version == 2)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (len < QCOW2v3_HDR_SIZE)
|
|
|
|
return -1;
|
|
|
|
|
2013-07-04 10:17:18 +00:00
|
|
|
if (!(feat = virBitmapNew(VIR_STORAGE_FILE_FEATURE_LAST)))
|
2013-05-07 15:27:43 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
/* todo: check for incompatible or autoclear features? */
|
|
|
|
bits = virReadBufInt64BE(buf + QCOW2v3_HDR_FEATURES_COMPATIBLE);
|
|
|
|
for (i = 0; i < QCOW2_COMPATIBLE_FEATURE_LAST; i++) {
|
|
|
|
if (bits & ((uint64_t) 1 << i))
|
|
|
|
ignore_value(virBitmapSetBit(feat, qcow2CompatibleFeatureArray[i]));
|
|
|
|
}
|
|
|
|
|
|
|
|
*features = feat;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-15 12:25:10 +00:00
|
|
|
/* Given a header in BUF with length LEN, as parsed from the storage file
|
|
|
|
* assuming it has the given FORMAT, populate information into META
|
|
|
|
* with information about the file and its backing store. Return format
|
|
|
|
* of the backing store as BACKING_FORMAT. PATH and FORMAT have to be
|
|
|
|
* pre-populated in META */
|
2014-04-25 19:38:40 +00:00
|
|
|
int
|
2014-04-17 14:05:16 +00:00
|
|
|
virStorageFileGetMetadataInternal(virStorageSourcePtr meta,
|
2013-11-05 20:50:29 +00:00
|
|
|
char *buf,
|
|
|
|
size_t len,
|
2014-04-16 13:44:06 +00:00
|
|
|
int *backingFormat)
|
2010-06-15 13:58:10 +00:00
|
|
|
{
|
2014-04-08 21:20:36 +00:00
|
|
|
int ret = -1;
|
2013-02-06 23:33:45 +00:00
|
|
|
|
2014-05-27 08:05:57 +00:00
|
|
|
VIR_DEBUG("path=%s, buf=%p, len=%zu, meta->format=%d",
|
|
|
|
meta->path, buf, len, meta->format);
|
2013-02-09 13:41:01 +00:00
|
|
|
|
2014-04-15 12:25:10 +00:00
|
|
|
if (meta->format == VIR_STORAGE_FILE_AUTO)
|
2014-05-27 08:05:57 +00:00
|
|
|
meta->format = virStorageFileProbeFormatFromBuf(meta->path, buf, len);
|
conf: track more fields in backing chain metadata
The current use of virStorageFileMetadata is awkward; to learn
some of the information about a child node, you have to read
fields in the parent node. This does not lend itself well to
modifying backing chains (whether inserting a new node in the
chain, or consolidating existing nodes); better would be to
learn about a child node directly in that node. This patch
sets up some new fields which contain redundant information,
although not necessarily in the final desired state for the
new fields (see the next patch for actual tests of what is there
now). Then later patches will do any refactoring necessary to
get the fields to their desired states, and update clients to
get the information from the new fields, so we can finally
delete the fields that are tracking information about the wrong
node.
More concretely, compare these three example backing chains:
good <- one
missing <- two
gluster://server/vol/img <- three
Pre-patch, querying the chains gives:
{ .backingStore = "/path/to/good",
.backingStoreRaw = "good",
.backingStoreIsFile = true,
.backingStoreFormat = VIR_STORAGE_FILE_RAW,
.backingMeta = {
.backingStore = NULL,
.backingStoreRaw = NULL,
.backingStoreIsFile = false,
.backingMeta = NULL,
}
}
{ .backingStore = NULL,
.backingStoreRaw = "missing",
.backingStoreIsFile = false,
.backingStoreFormat = VIR_STORAGE_FILE_NONE,
.backingMeta = NULL,
}
{ .backingStore = "gluster://server/vol/img",
.backingStoreRaw = NULL,
.backingStoreIsFile = false,
.backingStoreFormat = VIR_STORAGE_FILE_RAW,
.backingMeta = NULL,
}
Deciding whether to ignore a missing backing file (as in virsh
vol-dumpxml) or report an error (as in security manager sVirt
labeling) requires reading multiple fields. Plus, the format
is hard-coded to treat all network protocols as end-of-the-chain,
as if they were raw. By the end of this patch series, the goal
is to instead represent these three situations as:
{ .path = "one",
.canonPath = "/path/to/one",
.type = VIR_STORAGE_TYPE_FILE,
.format = VIR_STORAGE_FILE_QCOW2,
.backingStoreRaw = "good",
.backingMeta = {
.path = "good",
.canonPath = "/path/to/good",
.type = VIR_STORAGE_TYPE_FILE,
.format = VIR_STORAGE_FILE_RAW,
.backingStoreRaw = NULL,
.backingMeta = NULL,
}
}
{ .path = "two",
.canonPath = "/path/to/two",
.type = VIR_STORAGE_TYPE_FILE,
.format = VIR_STORAGE_FILE_QCOW2,
.backingStoreRaw = "missing",
.backingMeta = NULL,
}
{ .path = "three",
.canonPath = "/path/to/three",
.type = VIR_STORAGE_TYPE_FILE,
.format = VIR_STORAGE_FILE_QCOW2,
.backingStoreRaw = "gluster://server/vol/img",
.backingMeta = {
.path = "gluster://server/vol/img",
.canonPath = "gluster://server/vol/img",
.type = VIR_STORAGE_TYPE_NETWORK,
.format = VIR_STORAGE_FILE_RAW,
.backingStoreRaw = NULL,
.backingMeta = NULL,
}
}
or, for the second file, maybe also allowing:
{ .path = "two",
.canonPath = "/path/to/two",
.type = VIR_STORAGE_TYPE_FILE,
.format = VIR_STORAGE_FILE_QCOW2,
.backingStoreRaw = "missing",
.backingMeta = {
.path = "missing",
.canonPath = NULL,
.type = VIR_STORAGE_TYPE_NONE,
.format = VIR_STORAGE_FILE_NONE,
.backingStoreRaw = NULL,
.backingMeta = NULL,
}
}
* src/util/virstoragefile.h (_virStorageFileMetadata): Add
path, canonPath, relDir, type, and format fields. Reorder
existing fields, and add lots of comments.
* src/util/virstoragefile.c (virStorageFileFreeMetadata): Clean
new fields.
(virStorageFileGetMetadataInternal)
(virStorageFileGetMetadataFromFDInternal): Start populating new
fields.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-04-08 22:09:05 +00:00
|
|
|
|
2014-04-15 12:25:10 +00:00
|
|
|
if (meta->format <= VIR_STORAGE_FILE_NONE ||
|
|
|
|
meta->format >= VIR_STORAGE_FILE_LAST) {
|
|
|
|
virReportSystemError(EINVAL, _("unknown storage file meta->format %d"),
|
|
|
|
meta->format);
|
2013-02-06 23:33:45 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2012-12-13 14:23:50 +00:00
|
|
|
|
2010-06-15 13:58:10 +00:00
|
|
|
/* XXX we should consider moving virStorageBackendUpdateVolInfo
|
|
|
|
* code into this method, for non-magic files
|
|
|
|
*/
|
2014-04-15 12:25:10 +00:00
|
|
|
if (!fileTypeInfo[meta->format].magic)
|
2013-02-09 13:41:01 +00:00
|
|
|
goto done;
|
2009-09-29 08:34:48 +00:00
|
|
|
|
2010-06-15 13:58:10 +00:00
|
|
|
/* Optionally extract capacity from file */
|
2014-04-15 12:25:10 +00:00
|
|
|
if (fileTypeInfo[meta->format].sizeOffset != -1) {
|
|
|
|
if ((fileTypeInfo[meta->format].sizeOffset + 8) > len)
|
2013-02-09 13:41:01 +00:00
|
|
|
goto done;
|
2010-06-15 13:58:10 +00:00
|
|
|
|
2014-04-15 12:25:10 +00:00
|
|
|
if (fileTypeInfo[meta->format].endian == LV_LITTLE_ENDIAN)
|
2013-02-07 01:57:13 +00:00
|
|
|
meta->capacity = virReadBufInt64LE(buf +
|
2014-04-15 12:25:10 +00:00
|
|
|
fileTypeInfo[meta->format].sizeOffset);
|
2013-02-07 01:57:13 +00:00
|
|
|
else
|
|
|
|
meta->capacity = virReadBufInt64BE(buf +
|
2014-04-15 12:25:10 +00:00
|
|
|
fileTypeInfo[meta->format].sizeOffset);
|
2010-06-15 13:58:10 +00:00
|
|
|
/* Avoid unlikely, but theoretically possible overflow */
|
2013-02-07 01:57:13 +00:00
|
|
|
if (meta->capacity > (ULLONG_MAX /
|
2014-04-15 12:25:10 +00:00
|
|
|
fileTypeInfo[meta->format].sizeMultiplier))
|
2013-02-09 13:41:01 +00:00
|
|
|
goto done;
|
2014-04-15 12:25:10 +00:00
|
|
|
meta->capacity *= fileTypeInfo[meta->format].sizeMultiplier;
|
2010-06-15 13:58:10 +00:00
|
|
|
}
|
2009-09-29 08:34:48 +00:00
|
|
|
|
2014-04-15 12:25:10 +00:00
|
|
|
if (fileTypeInfo[meta->format].qcowCryptOffset != -1) {
|
2010-06-15 13:58:10 +00:00
|
|
|
int crypt_format;
|
2009-09-29 08:34:48 +00:00
|
|
|
|
2013-02-07 01:57:13 +00:00
|
|
|
crypt_format = virReadBufInt32BE(buf +
|
2014-04-15 12:25:10 +00:00
|
|
|
fileTypeInfo[meta->format].qcowCryptOffset);
|
storage: fix memory leak with encrypted images
Jim Fehlig reported a regression found by libvirt-TCK tests:
> ~ # perl /usr/share/libvirt-tck/tests/qemu/100-disk-encryption.t
...
> ok 4 - defined persistent domain config
> # Starting inactive domain config
> libvirt error code: 1, message: internal error: unable to execute QEMU command
> 'cont': 'drive-ide0-0-1'
> (/var/cache/libvirt-tck/300-disk-encryption/demo.qcow2) is encrypted
Commit 2279d560 converted a boolean into a pointer with the intent of
transferring that pointer out of a temporary object into the caller's
data structure. The temporary structure meant that meta->encryption
was always NULL on entry, so we could get away with blindly allocating
the pointer when the header said so. But later, commit 8823272d
tweaked things to do backing chain detection in-place, rather than via
a temporary object; this has the net result that meta->encryption can
be non-NULL on entry. Not only did this turn the latent behavior into
a memory leak, it is also a behavior regression: blindly allocating a
new pointer wipes out what secrets we already knew about the chain,
making it impossible to restart the domain.
Of course, no one in their right mind should be relying on qcow2
encryption - it is fundamentally flawed. And sadly, the TCK tests
don't get run often enough, and this shows that our virstoragetest
does not exercise encrypted images at all. Otherwise, we could
have avoided a release containing this regression.
* src/util/virstoragefile.c (virStorageFileGetMetadataInternal):
Don't nuke an already-existing encryption.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-06-09 21:36:58 +00:00
|
|
|
if (crypt_format && !meta->encryption &&
|
|
|
|
VIR_ALLOC(meta->encryption) < 0)
|
2014-04-02 01:13:51 +00:00
|
|
|
goto cleanup;
|
2010-06-15 13:58:10 +00:00
|
|
|
}
|
2009-09-29 08:34:48 +00:00
|
|
|
|
2014-08-06 20:48:59 +00:00
|
|
|
VIR_FREE(meta->backingStoreRaw);
|
2014-04-15 12:25:10 +00:00
|
|
|
if (fileTypeInfo[meta->format].getBackingStore != NULL) {
|
|
|
|
int store = fileTypeInfo[meta->format].getBackingStore(&meta->backingStoreRaw,
|
2014-04-10 01:49:07 +00:00
|
|
|
backingFormat,
|
2013-02-09 13:41:01 +00:00
|
|
|
buf, len);
|
|
|
|
if (store == BACKING_STORE_INVALID)
|
|
|
|
goto done;
|
2010-06-15 13:58:10 +00:00
|
|
|
|
2013-02-09 13:41:01 +00:00
|
|
|
if (store == BACKING_STORE_ERROR)
|
|
|
|
goto cleanup;
|
2010-06-15 13:58:10 +00:00
|
|
|
}
|
|
|
|
|
2014-04-15 12:25:10 +00:00
|
|
|
if (fileTypeInfo[meta->format].getFeatures != NULL &&
|
|
|
|
fileTypeInfo[meta->format].getFeatures(&meta->features, meta->format, buf, len) < 0)
|
2013-05-07 15:27:43 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2014-04-15 12:25:10 +00:00
|
|
|
if (meta->format == VIR_STORAGE_FILE_QCOW2 && meta->features &&
|
2013-05-07 15:27:43 +00:00
|
|
|
VIR_STRDUP(meta->compat, "1.1") < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2014-03-25 06:53:22 +00:00
|
|
|
done:
|
2014-04-08 21:20:36 +00:00
|
|
|
ret = 0;
|
2013-02-06 23:33:45 +00:00
|
|
|
|
2014-03-25 06:53:22 +00:00
|
|
|
cleanup:
|
2013-02-09 13:41:01 +00:00
|
|
|
return ret;
|
2010-06-15 13:58:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2013-11-05 21:12:02 +00:00
|
|
|
* virStorageFileProbeFormat:
|
2010-06-15 13:58:10 +00:00
|
|
|
*
|
2013-11-05 21:12:02 +00:00
|
|
|
* Probe for the format of 'path', returning the detected
|
|
|
|
* disk format.
|
2010-06-15 13:58:10 +00:00
|
|
|
*
|
|
|
|
* Callers are advised never to trust the returned 'format'
|
|
|
|
* unless it is listed as VIR_STORAGE_FILE_RAW, since a
|
2013-11-05 21:12:02 +00:00
|
|
|
* malicious guest can turn a raw file into any other non-raw
|
2010-06-15 13:58:10 +00:00
|
|
|
* format at will.
|
|
|
|
*
|
|
|
|
* Best option: Don't use this function
|
|
|
|
*/
|
|
|
|
int
|
2013-11-05 21:12:02 +00:00
|
|
|
virStorageFileProbeFormat(const char *path, uid_t uid, gid_t gid)
|
2010-06-15 13:58:10 +00:00
|
|
|
{
|
2013-11-05 21:12:02 +00:00
|
|
|
int fd;
|
2010-06-15 13:58:10 +00:00
|
|
|
int ret = -1;
|
2011-05-26 18:05:32 +00:00
|
|
|
struct stat sb;
|
2013-11-05 21:12:02 +00:00
|
|
|
ssize_t len = VIR_STORAGE_MAX_HEADER;
|
|
|
|
char *header = NULL;
|
2011-05-26 18:05:32 +00:00
|
|
|
|
2013-11-05 21:12:02 +00:00
|
|
|
if ((fd = virFileOpenAs(path, O_RDONLY, 0, uid, gid, 0)) < 0) {
|
|
|
|
virReportSystemError(-fd, _("Failed to open file '%s'"), path);
|
2011-05-26 18:05:32 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-11-05 21:12:02 +00:00
|
|
|
if (fstat(fd, &sb) < 0) {
|
|
|
|
virReportSystemError(errno, _("cannot stat file '%s'"), path);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2011-05-26 18:05:32 +00:00
|
|
|
/* No header to probe for directories */
|
|
|
|
if (S_ISDIR(sb.st_mode)) {
|
2013-11-05 21:12:02 +00:00
|
|
|
ret = VIR_STORAGE_FILE_DIR;
|
|
|
|
goto cleanup;
|
2011-05-26 18:05:32 +00:00
|
|
|
}
|
2010-06-15 13:58:10 +00:00
|
|
|
|
|
|
|
if (lseek(fd, 0, SEEK_SET) == (off_t)-1) {
|
|
|
|
virReportSystemError(errno, _("cannot set to start of '%s'"), path);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2013-11-05 21:12:02 +00:00
|
|
|
if ((len = virFileReadHeaderFD(fd, len, &header)) < 0) {
|
2010-06-15 13:58:10 +00:00
|
|
|
virReportSystemError(errno, _("cannot read header '%s'"), path);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2013-11-05 21:12:02 +00:00
|
|
|
ret = virStorageFileProbeFormatFromBuf(path, header, len);
|
2010-06-15 13:58:10 +00:00
|
|
|
|
2014-03-25 06:53:22 +00:00
|
|
|
cleanup:
|
2013-11-05 21:12:02 +00:00
|
|
|
VIR_FREE(header);
|
2010-11-09 20:48:48 +00:00
|
|
|
VIR_FORCE_CLOSE(fd);
|
2010-06-15 13:58:10 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2013-11-05 20:50:29 +00:00
|
|
|
|
2014-04-17 14:05:16 +00:00
|
|
|
static virStorageSourcePtr
|
2014-04-15 12:25:10 +00:00
|
|
|
virStorageFileMetadataNew(const char *path,
|
|
|
|
int format)
|
|
|
|
{
|
2014-04-17 14:05:16 +00:00
|
|
|
virStorageSourcePtr ret = NULL;
|
2014-04-15 12:25:10 +00:00
|
|
|
|
|
|
|
if (VIR_ALLOC(ret) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
ret->format = format;
|
|
|
|
ret->type = VIR_STORAGE_TYPE_FILE;
|
|
|
|
|
2014-05-28 13:17:11 +00:00
|
|
|
if (VIR_STRDUP(ret->path, path) < 0)
|
|
|
|
goto error;
|
2014-04-15 12:25:10 +00:00
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
error:
|
2014-04-17 14:05:16 +00:00
|
|
|
virStorageSourceFree(ret);
|
2014-04-15 12:25:10 +00:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-11-05 20:50:29 +00:00
|
|
|
/**
|
|
|
|
* virStorageFileGetMetadataFromBuf:
|
|
|
|
* @path: name of file, for error messages
|
|
|
|
* @buf: header bytes from @path
|
|
|
|
* @len: length of @buf
|
2014-07-07 09:38:28 +00:00
|
|
|
* @format: format of the storage file
|
2014-04-10 01:49:07 +00:00
|
|
|
* @backingFormat: format of @backing
|
2013-11-05 20:50:29 +00:00
|
|
|
*
|
2014-07-07 09:38:28 +00:00
|
|
|
* Extract metadata about the storage volume with the specified image format.
|
|
|
|
* If image format is VIR_STORAGE_FILE_AUTO, it will probe to automatically
|
|
|
|
* identify the format. Does not recurse.
|
2013-11-05 20:50:29 +00:00
|
|
|
*
|
2014-07-07 09:38:28 +00:00
|
|
|
* Callers are advised never to use VIR_STORAGE_FILE_AUTO as a format on a file
|
|
|
|
* that might be raw if that file will then be passed to a guest, since a
|
|
|
|
* malicious guest can turn a raw file into any other non-raw format at will.
|
|
|
|
*
|
|
|
|
* If the returned @backingFormat is VIR_STORAGE_FILE_AUTO it indicates the
|
|
|
|
* image didn't specify an explicit format for its backing store. Callers are
|
|
|
|
* advised against probing for the backing store format in this case.
|
2013-11-05 20:50:29 +00:00
|
|
|
*
|
2014-04-17 14:05:16 +00:00
|
|
|
* Caller MUST free the result after use via virStorageSourceFree.
|
2013-11-05 20:50:29 +00:00
|
|
|
*/
|
2014-04-17 14:05:16 +00:00
|
|
|
virStorageSourcePtr
|
2013-11-05 20:50:29 +00:00
|
|
|
virStorageFileGetMetadataFromBuf(const char *path,
|
|
|
|
char *buf,
|
|
|
|
size_t len,
|
2014-07-07 09:38:28 +00:00
|
|
|
int format,
|
2014-04-10 01:49:07 +00:00
|
|
|
int *backingFormat)
|
2013-11-05 20:50:29 +00:00
|
|
|
{
|
2014-04-17 14:05:16 +00:00
|
|
|
virStorageSourcePtr ret = NULL;
|
2014-07-09 08:00:58 +00:00
|
|
|
int dummy;
|
|
|
|
|
|
|
|
if (!backingFormat)
|
|
|
|
backingFormat = &dummy;
|
2014-04-08 20:26:02 +00:00
|
|
|
|
2014-07-07 09:38:28 +00:00
|
|
|
if (!(ret = virStorageFileMetadataNew(path, format)))
|
2014-04-08 20:26:02 +00:00
|
|
|
return NULL;
|
2014-04-08 21:20:36 +00:00
|
|
|
|
2014-07-07 09:38:28 +00:00
|
|
|
if (virStorageFileGetMetadataInternal(ret, buf, len,
|
|
|
|
backingFormat) < 0) {
|
|
|
|
virStorageSourceFree(ret);
|
|
|
|
return NULL;
|
|
|
|
}
|
2014-04-15 12:28:10 +00:00
|
|
|
|
2014-04-08 20:26:02 +00:00
|
|
|
return ret;
|
2013-11-05 20:50:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-07-07 09:24:14 +00:00
|
|
|
/**
|
|
|
|
* virStorageFileGetMetadataFromFD:
|
|
|
|
*
|
|
|
|
* Extract metadata about the storage volume with the specified
|
|
|
|
* image format. If image format is VIR_STORAGE_FILE_AUTO, it
|
|
|
|
* will probe to automatically identify the format. Does not recurse.
|
|
|
|
*
|
|
|
|
* Callers are advised never to use VIR_STORAGE_FILE_AUTO as a
|
|
|
|
* format, since a malicious guest can turn a raw file into any
|
|
|
|
* other non-raw format at will.
|
|
|
|
*
|
|
|
|
* Caller MUST free the result after use via virStorageSourceFree.
|
|
|
|
*/
|
|
|
|
virStorageSourcePtr
|
|
|
|
virStorageFileGetMetadataFromFD(const char *path,
|
|
|
|
int fd,
|
|
|
|
int format,
|
|
|
|
int *backingFormat)
|
|
|
|
|
2013-11-05 20:50:29 +00:00
|
|
|
{
|
2014-07-07 09:24:14 +00:00
|
|
|
virStorageSourcePtr ret = NULL;
|
|
|
|
virStorageSourcePtr meta = NULL;
|
2013-11-05 20:50:29 +00:00
|
|
|
char *buf = NULL;
|
2013-11-05 21:12:02 +00:00
|
|
|
ssize_t len = VIR_STORAGE_MAX_HEADER;
|
2013-11-05 20:50:29 +00:00
|
|
|
struct stat sb;
|
2014-04-10 02:37:16 +00:00
|
|
|
int dummy;
|
2013-11-05 20:50:29 +00:00
|
|
|
|
2014-04-10 02:37:16 +00:00
|
|
|
if (!backingFormat)
|
|
|
|
backingFormat = &dummy;
|
2014-04-15 12:25:10 +00:00
|
|
|
|
2014-04-10 02:37:16 +00:00
|
|
|
*backingFormat = VIR_STORAGE_FILE_NONE;
|
2014-04-15 12:25:10 +00:00
|
|
|
|
2013-11-05 20:50:29 +00:00
|
|
|
if (fstat(fd, &sb) < 0) {
|
|
|
|
virReportSystemError(errno,
|
2014-07-07 09:24:14 +00:00
|
|
|
_("cannot stat file '%s'"), path);
|
|
|
|
return NULL;
|
2013-11-05 20:50:29 +00:00
|
|
|
}
|
|
|
|
|
2014-07-07 09:24:14 +00:00
|
|
|
if (!(meta = virStorageFileMetadataNew(path, format)))
|
|
|
|
return NULL;
|
|
|
|
|
2013-11-05 20:50:29 +00:00
|
|
|
if (S_ISDIR(sb.st_mode)) {
|
2014-04-18 08:07:41 +00:00
|
|
|
/* No header to probe for directories, but also no backing file. Just
|
|
|
|
* update the metadata.*/
|
conf: track more fields in backing chain metadata
The current use of virStorageFileMetadata is awkward; to learn
some of the information about a child node, you have to read
fields in the parent node. This does not lend itself well to
modifying backing chains (whether inserting a new node in the
chain, or consolidating existing nodes); better would be to
learn about a child node directly in that node. This patch
sets up some new fields which contain redundant information,
although not necessarily in the final desired state for the
new fields (see the next patch for actual tests of what is there
now). Then later patches will do any refactoring necessary to
get the fields to their desired states, and update clients to
get the information from the new fields, so we can finally
delete the fields that are tracking information about the wrong
node.
More concretely, compare these three example backing chains:
good <- one
missing <- two
gluster://server/vol/img <- three
Pre-patch, querying the chains gives:
{ .backingStore = "/path/to/good",
.backingStoreRaw = "good",
.backingStoreIsFile = true,
.backingStoreFormat = VIR_STORAGE_FILE_RAW,
.backingMeta = {
.backingStore = NULL,
.backingStoreRaw = NULL,
.backingStoreIsFile = false,
.backingMeta = NULL,
}
}
{ .backingStore = NULL,
.backingStoreRaw = "missing",
.backingStoreIsFile = false,
.backingStoreFormat = VIR_STORAGE_FILE_NONE,
.backingMeta = NULL,
}
{ .backingStore = "gluster://server/vol/img",
.backingStoreRaw = NULL,
.backingStoreIsFile = false,
.backingStoreFormat = VIR_STORAGE_FILE_RAW,
.backingMeta = NULL,
}
Deciding whether to ignore a missing backing file (as in virsh
vol-dumpxml) or report an error (as in security manager sVirt
labeling) requires reading multiple fields. Plus, the format
is hard-coded to treat all network protocols as end-of-the-chain,
as if they were raw. By the end of this patch series, the goal
is to instead represent these three situations as:
{ .path = "one",
.canonPath = "/path/to/one",
.type = VIR_STORAGE_TYPE_FILE,
.format = VIR_STORAGE_FILE_QCOW2,
.backingStoreRaw = "good",
.backingMeta = {
.path = "good",
.canonPath = "/path/to/good",
.type = VIR_STORAGE_TYPE_FILE,
.format = VIR_STORAGE_FILE_RAW,
.backingStoreRaw = NULL,
.backingMeta = NULL,
}
}
{ .path = "two",
.canonPath = "/path/to/two",
.type = VIR_STORAGE_TYPE_FILE,
.format = VIR_STORAGE_FILE_QCOW2,
.backingStoreRaw = "missing",
.backingMeta = NULL,
}
{ .path = "three",
.canonPath = "/path/to/three",
.type = VIR_STORAGE_TYPE_FILE,
.format = VIR_STORAGE_FILE_QCOW2,
.backingStoreRaw = "gluster://server/vol/img",
.backingMeta = {
.path = "gluster://server/vol/img",
.canonPath = "gluster://server/vol/img",
.type = VIR_STORAGE_TYPE_NETWORK,
.format = VIR_STORAGE_FILE_RAW,
.backingStoreRaw = NULL,
.backingMeta = NULL,
}
}
or, for the second file, maybe also allowing:
{ .path = "two",
.canonPath = "/path/to/two",
.type = VIR_STORAGE_TYPE_FILE,
.format = VIR_STORAGE_FILE_QCOW2,
.backingStoreRaw = "missing",
.backingMeta = {
.path = "missing",
.canonPath = NULL,
.type = VIR_STORAGE_TYPE_NONE,
.format = VIR_STORAGE_FILE_NONE,
.backingStoreRaw = NULL,
.backingMeta = NULL,
}
}
* src/util/virstoragefile.h (_virStorageFileMetadata): Add
path, canonPath, relDir, type, and format fields. Reorder
existing fields, and add lots of comments.
* src/util/virstoragefile.c (virStorageFileFreeMetadata): Clean
new fields.
(virStorageFileGetMetadataInternal)
(virStorageFileGetMetadataFromFDInternal): Start populating new
fields.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-04-08 22:09:05 +00:00
|
|
|
meta->type = VIR_STORAGE_TYPE_DIR;
|
|
|
|
meta->format = VIR_STORAGE_FILE_DIR;
|
2014-07-07 09:24:14 +00:00
|
|
|
ret = meta;
|
|
|
|
meta = NULL;
|
2013-11-05 20:50:29 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lseek(fd, 0, SEEK_SET) == (off_t)-1) {
|
2014-04-17 11:36:59 +00:00
|
|
|
virReportSystemError(errno, _("cannot seek to start of '%s'"), meta->relPath);
|
2013-11-05 20:50:29 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((len = virFileReadHeaderFD(fd, len, &buf)) < 0) {
|
2014-04-17 11:36:59 +00:00
|
|
|
virReportSystemError(errno, _("cannot read header '%s'"), meta->relPath);
|
2013-11-05 20:50:29 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2014-07-07 09:24:14 +00:00
|
|
|
if (virStorageFileGetMetadataInternal(meta, buf, len, backingFormat) < 0)
|
|
|
|
goto cleanup;
|
2014-05-28 13:17:11 +00:00
|
|
|
|
2014-07-07 09:24:14 +00:00
|
|
|
if (S_ISREG(sb.st_mode))
|
|
|
|
meta->type = VIR_STORAGE_TYPE_FILE;
|
|
|
|
else if (S_ISBLK(sb.st_mode))
|
|
|
|
meta->type = VIR_STORAGE_TYPE_BLOCK;
|
2014-04-15 12:25:10 +00:00
|
|
|
|
2014-07-07 09:24:14 +00:00
|
|
|
ret = meta;
|
|
|
|
meta = NULL;
|
2014-04-08 21:20:36 +00:00
|
|
|
|
2014-07-07 09:24:14 +00:00
|
|
|
cleanup:
|
|
|
|
virStorageSourceFree(meta);
|
|
|
|
VIR_FREE(buf);
|
2014-04-08 20:26:02 +00:00
|
|
|
return ret;
|
2010-06-15 13:58:10 +00:00
|
|
|
}
|
|
|
|
|
2013-11-05 20:50:29 +00:00
|
|
|
|
2013-07-29 12:51:15 +00:00
|
|
|
/**
|
|
|
|
* virStorageFileChainCheckBroken
|
|
|
|
*
|
|
|
|
* If CHAIN is broken, set *brokenFile to the broken file name,
|
|
|
|
* otherwise set it to NULL. Caller MUST free *brokenFile after use.
|
2014-04-09 21:36:30 +00:00
|
|
|
* Return 0 on success (including when brokenFile is set), negative on
|
|
|
|
* error (allocation failure).
|
2013-07-29 12:51:15 +00:00
|
|
|
*/
|
|
|
|
int
|
2014-04-17 14:05:16 +00:00
|
|
|
virStorageFileChainGetBroken(virStorageSourcePtr chain,
|
2013-07-29 12:51:15 +00:00
|
|
|
char **brokenFile)
|
|
|
|
{
|
2014-04-17 14:05:16 +00:00
|
|
|
virStorageSourcePtr tmp;
|
2013-07-29 12:51:15 +00:00
|
|
|
|
2014-04-09 21:36:30 +00:00
|
|
|
*brokenFile = NULL;
|
|
|
|
|
2013-07-29 12:51:15 +00:00
|
|
|
if (!chain)
|
|
|
|
return 0;
|
|
|
|
|
2014-04-17 20:46:18 +00:00
|
|
|
for (tmp = chain; tmp; tmp = tmp->backingStore) {
|
2014-04-09 21:36:30 +00:00
|
|
|
/* Break when we hit end of chain; report error if we detected
|
|
|
|
* a missing backing file, infinite loop, or other error */
|
2014-04-17 20:46:18 +00:00
|
|
|
if (!tmp->backingStore && tmp->backingStoreRaw) {
|
2014-04-14 13:49:28 +00:00
|
|
|
if (VIR_STRDUP(*brokenFile, tmp->backingStoreRaw) < 0)
|
|
|
|
return -1;
|
2013-07-29 12:51:15 +00:00
|
|
|
|
2014-04-14 13:49:28 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
2013-07-29 12:51:15 +00:00
|
|
|
|
2014-04-14 13:49:28 +00:00
|
|
|
return 0;
|
2013-07-29 12:51:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-01-30 07:40:00 +00:00
|
|
|
/**
|
|
|
|
* virStorageFileResize:
|
|
|
|
*
|
|
|
|
* Change the capacity of the raw storage file at 'path'.
|
|
|
|
*/
|
|
|
|
int
|
2013-05-31 05:16:14 +00:00
|
|
|
virStorageFileResize(const char *path,
|
|
|
|
unsigned long long capacity,
|
|
|
|
unsigned long long orig_capacity,
|
|
|
|
bool pre_allocate)
|
2012-01-30 07:40:00 +00:00
|
|
|
{
|
2012-02-08 14:03:29 +00:00
|
|
|
int fd = -1;
|
|
|
|
int ret = -1;
|
2013-06-06 02:33:15 +00:00
|
|
|
int rc ATTRIBUTE_UNUSED;
|
|
|
|
off_t offset ATTRIBUTE_UNUSED;
|
|
|
|
off_t len ATTRIBUTE_UNUSED;
|
|
|
|
|
|
|
|
offset = orig_capacity;
|
|
|
|
len = capacity - orig_capacity;
|
2012-02-08 14:03:29 +00:00
|
|
|
|
|
|
|
if ((fd = open(path, O_RDWR)) < 0) {
|
|
|
|
virReportSystemError(errno, _("Unable to open '%s'"), path);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2013-05-31 05:16:14 +00:00
|
|
|
if (pre_allocate) {
|
|
|
|
#if HAVE_POSIX_FALLOCATE
|
|
|
|
if ((rc = posix_fallocate(fd, offset, len)) != 0) {
|
|
|
|
virReportSystemError(rc,
|
|
|
|
_("Failed to pre-allocate space for "
|
|
|
|
"file '%s'"), path);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
#elif HAVE_SYS_SYSCALL_H && defined(SYS_fallocate)
|
|
|
|
if (syscall(SYS_fallocate, fd, 0, offset, len) != 0) {
|
|
|
|
virReportSystemError(errno,
|
2013-06-06 02:33:15 +00:00
|
|
|
_("Failed to pre-allocate space for "
|
2013-05-31 05:16:14 +00:00
|
|
|
"file '%s'"), path);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
virReportError(VIR_ERR_OPERATION_UNSUPPORTED, "%s",
|
2013-06-06 02:33:15 +00:00
|
|
|
_("preallocate is not supported on this platform"));
|
2012-02-08 14:03:29 +00:00
|
|
|
goto cleanup;
|
2013-05-31 05:16:14 +00:00
|
|
|
#endif
|
|
|
|
} else {
|
|
|
|
if (ftruncate(fd, capacity) < 0) {
|
|
|
|
virReportSystemError(errno,
|
|
|
|
_("Failed to truncate file '%s'"), path);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2012-01-30 07:40:00 +00:00
|
|
|
}
|
|
|
|
|
2012-02-08 14:03:29 +00:00
|
|
|
if (VIR_CLOSE(fd) < 0) {
|
|
|
|
virReportSystemError(errno, _("Unable to save '%s'"), path);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
2014-03-25 06:53:22 +00:00
|
|
|
cleanup:
|
2012-02-08 14:03:29 +00:00
|
|
|
VIR_FORCE_CLOSE(fd);
|
|
|
|
return ret;
|
2012-01-30 07:40:00 +00:00
|
|
|
}
|
|
|
|
|
2012-02-21 21:58:50 +00:00
|
|
|
|
|
|
|
int virStorageFileIsClusterFS(const char *path)
|
|
|
|
{
|
|
|
|
/* These are coherent cluster filesystems known to be safe for
|
|
|
|
* migration with cache != none
|
|
|
|
*/
|
2014-03-29 20:15:33 +00:00
|
|
|
return virFileIsSharedFSType(path,
|
|
|
|
VIR_FILE_SHFS_GFS2 |
|
|
|
|
VIR_FILE_SHFS_OCFS);
|
2012-02-21 21:58:50 +00:00
|
|
|
}
|
2011-07-20 09:40:53 +00:00
|
|
|
|
|
|
|
#ifdef LVS
|
2012-12-11 19:10:51 +00:00
|
|
|
int virStorageFileGetLVMKey(const char *path,
|
|
|
|
char **key)
|
2011-07-20 09:40:53 +00:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* # lvs --noheadings --unbuffered --nosuffix --options "uuid" LVNAME
|
|
|
|
* 06UgP5-2rhb-w3Bo-3mdR-WeoL-pytO-SAa2ky
|
|
|
|
*/
|
2012-12-11 19:10:51 +00:00
|
|
|
int status;
|
2011-07-20 09:40:53 +00:00
|
|
|
virCommandPtr cmd = virCommandNewArgList(
|
|
|
|
LVS,
|
|
|
|
"--noheadings", "--unbuffered", "--nosuffix",
|
|
|
|
"--options", "uuid", path,
|
|
|
|
NULL
|
|
|
|
);
|
2012-12-11 19:10:51 +00:00
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
*key = NULL;
|
2011-07-20 09:40:53 +00:00
|
|
|
|
|
|
|
/* Run the program and capture its output */
|
2012-12-11 19:10:51 +00:00
|
|
|
virCommandSetOutputBuffer(cmd, key);
|
|
|
|
if (virCommandRun(cmd, &status) < 0)
|
2011-07-20 09:40:53 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2012-12-11 19:10:51 +00:00
|
|
|
/* Explicitly check status == 0, rather than passing NULL
|
|
|
|
* to virCommandRun because we don't want to raise an actual
|
|
|
|
* error in this scenario, just return a NULL key.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (status == 0 && *key) {
|
2011-07-20 09:40:53 +00:00
|
|
|
char *nl;
|
2012-12-11 19:10:51 +00:00
|
|
|
char *tmp = *key;
|
2011-07-20 09:40:53 +00:00
|
|
|
|
|
|
|
/* Find first non-space character */
|
2014-11-13 14:28:18 +00:00
|
|
|
while (*tmp && c_isspace(*tmp))
|
2011-07-20 09:40:53 +00:00
|
|
|
tmp++;
|
|
|
|
/* Kill leading spaces */
|
2012-12-11 19:10:51 +00:00
|
|
|
if (tmp != *key)
|
|
|
|
memmove(*key, tmp, strlen(tmp)+1);
|
2011-07-20 09:40:53 +00:00
|
|
|
|
|
|
|
/* Kill trailing newline */
|
2012-12-11 19:10:51 +00:00
|
|
|
if ((nl = strchr(*key, '\n')))
|
2011-07-20 09:40:53 +00:00
|
|
|
*nl = '\0';
|
|
|
|
}
|
|
|
|
|
2012-12-11 19:10:51 +00:00
|
|
|
ret = 0;
|
2011-07-20 09:40:53 +00:00
|
|
|
|
2014-03-25 06:53:22 +00:00
|
|
|
cleanup:
|
2012-12-11 19:10:51 +00:00
|
|
|
if (*key && STREQ(*key, ""))
|
|
|
|
VIR_FREE(*key);
|
|
|
|
|
2011-07-20 09:40:53 +00:00
|
|
|
virCommandFree(cmd);
|
|
|
|
|
2012-12-11 19:10:51 +00:00
|
|
|
return ret;
|
2011-07-20 09:40:53 +00:00
|
|
|
}
|
|
|
|
#else
|
2012-12-11 19:10:51 +00:00
|
|
|
int virStorageFileGetLVMKey(const char *path,
|
|
|
|
char **key ATTRIBUTE_UNUSED)
|
2011-07-20 09:40:53 +00:00
|
|
|
{
|
|
|
|
virReportSystemError(ENOSYS, _("Unable to get LVM key for %s"), path);
|
2012-12-11 19:10:51 +00:00
|
|
|
return -1;
|
2011-07-20 09:40:53 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2012-09-20 14:24:47 +00:00
|
|
|
#ifdef WITH_UDEV
|
2012-12-11 19:10:51 +00:00
|
|
|
int virStorageFileGetSCSIKey(const char *path,
|
|
|
|
char **key)
|
2011-07-20 09:40:53 +00:00
|
|
|
{
|
2012-12-11 19:10:51 +00:00
|
|
|
int status;
|
2011-07-20 09:40:53 +00:00
|
|
|
virCommandPtr cmd = virCommandNewArgList(
|
|
|
|
"/lib/udev/scsi_id",
|
|
|
|
"--replace-whitespace",
|
|
|
|
"--whitelisted",
|
|
|
|
"--device", path,
|
|
|
|
NULL
|
|
|
|
);
|
2012-12-11 19:10:51 +00:00
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
*key = NULL;
|
2011-07-20 09:40:53 +00:00
|
|
|
|
|
|
|
/* Run the program and capture its output */
|
2012-12-11 19:10:51 +00:00
|
|
|
virCommandSetOutputBuffer(cmd, key);
|
|
|
|
if (virCommandRun(cmd, &status) < 0)
|
2011-07-20 09:40:53 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2012-12-11 19:10:51 +00:00
|
|
|
/* Explicitly check status == 0, rather than passing NULL
|
|
|
|
* to virCommandRun because we don't want to raise an actual
|
|
|
|
* error in this scenario, just return a NULL key.
|
|
|
|
*/
|
|
|
|
if (status == 0 && *key) {
|
|
|
|
char *nl = strchr(*key, '\n');
|
2011-07-20 09:40:53 +00:00
|
|
|
if (nl)
|
|
|
|
*nl = '\0';
|
|
|
|
}
|
|
|
|
|
2012-12-11 19:10:51 +00:00
|
|
|
ret = 0;
|
|
|
|
|
2014-03-25 06:53:22 +00:00
|
|
|
cleanup:
|
2012-12-11 19:10:51 +00:00
|
|
|
if (*key && STREQ(*key, ""))
|
|
|
|
VIR_FREE(*key);
|
|
|
|
|
2011-07-20 09:40:53 +00:00
|
|
|
virCommandFree(cmd);
|
|
|
|
|
2012-12-11 19:10:51 +00:00
|
|
|
return ret;
|
2011-07-20 09:40:53 +00:00
|
|
|
}
|
|
|
|
#else
|
2012-12-11 19:10:51 +00:00
|
|
|
int virStorageFileGetSCSIKey(const char *path,
|
|
|
|
char **key ATTRIBUTE_UNUSED)
|
2011-07-20 09:40:53 +00:00
|
|
|
{
|
|
|
|
virReportSystemError(ENOSYS, _("Unable to get SCSI key for %s"), path);
|
2012-12-11 19:10:51 +00:00
|
|
|
return -1;
|
2011-07-20 09:40:53 +00:00
|
|
|
}
|
|
|
|
#endif
|
2012-10-12 22:29:14 +00:00
|
|
|
|
2014-04-18 12:35:33 +00:00
|
|
|
int
|
|
|
|
virStorageFileParseChainIndex(const char *diskTarget,
|
|
|
|
const char *name,
|
|
|
|
unsigned int *chainIndex)
|
|
|
|
{
|
|
|
|
char **strings = NULL;
|
|
|
|
unsigned int idx = 0;
|
|
|
|
char *suffix;
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
*chainIndex = 0;
|
|
|
|
|
|
|
|
if (name && diskTarget)
|
|
|
|
strings = virStringSplit(name, "[", 2);
|
|
|
|
|
|
|
|
if (virStringListLength(strings) != 2)
|
|
|
|
goto cleanup;
|
|
|
|
|
2014-05-01 02:17:42 +00:00
|
|
|
if (virStrToLong_uip(strings[1], &suffix, 10, &idx) < 0 ||
|
2014-04-18 12:35:33 +00:00
|
|
|
STRNEQ(suffix, "]"))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (STRNEQ(diskTarget, strings[0])) {
|
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("requested target '%s' does not match target '%s'"),
|
|
|
|
strings[0], diskTarget);
|
|
|
|
ret = -1;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
*chainIndex = idx;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virStringFreeList(strings);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
blockcommit: require base below top
The block commit code looks for an explicit base file relative
to the discovered top file; so for a chain of:
base <- snap1 <- snap2 <- snap3
and a command of:
virsh blockcommit $dom vda --base snap2 --top snap1
we got a sane message (here from libvirt 1.0.5):
error: invalid argument: could not find base 'snap2' below 'snap1' in chain for 'vda'
Meanwhile, recent refactoring has slightly reduced the quality of the
libvirt error messages, by losing the phrase 'below xyz':
error: invalid argument: could not find image 'snap2' in chain for 'snap3'
But we had a one-off, where we were not excluding the top file
itself in searching for the base; thankfully qemu still reports
the error, but the quality is worse:
virsh blockcommit $dom vda --base snap2 --top snap2
error: internal error unable to execute QEMU command 'block-commit': Base '/snap2' not found
Fix the one-off in blockcommit by changing the semantics of name
lookup - if a starting point is specified, then the result must
be below that point, rather than including that point. The only
other call to chain lookup was blockpull code, which was already
forcing the lookup to omit the active layer and only needs a
tweak to use the new semantics.
This also fixes the bug exposed in the testsuite, where when doing
a lookup pinned to an intermediate point in the chain, we were
unable to return the name of the parent also in the chain.
* src/util/virstoragefile.c (virStorageFileChainLookup): Change
semantics for non-NULL startFrom.
* src/qemu/qemu_driver.c (qemuDomainBlockJobImpl): Adjust caller,
to keep existing semantics.
* tests/virstoragetest.c (mymain): Adjust to expose new semantics.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-06-11 22:22:57 +00:00
|
|
|
/* Given a @chain, look for the backing store @name that is a backing file
|
|
|
|
* of @startFrom (or any member of @chain if @startFrom is NULL) and return
|
|
|
|
* that location within the chain. @chain must always point to the top of
|
|
|
|
* the chain. Pass NULL for @name and 0 for @idx to find the base of the
|
|
|
|
* chain. Pass nonzero @idx to find the backing source according to its
|
|
|
|
* position in the backing chain. If @parent is not NULL, set *@parent to
|
|
|
|
* the preferred name of the parent (or to NULL if @name matches the start
|
|
|
|
* of the chain). Since the results point within @chain, they must not be
|
|
|
|
* independently freed. Reports an error and returns NULL if @name is not
|
|
|
|
* found.
|
2014-04-18 12:35:33 +00:00
|
|
|
*/
|
2014-04-18 13:25:19 +00:00
|
|
|
virStorageSourcePtr
|
2014-04-17 14:05:16 +00:00
|
|
|
virStorageFileChainLookup(virStorageSourcePtr chain,
|
2014-04-18 12:35:33 +00:00
|
|
|
virStorageSourcePtr startFrom,
|
2014-04-18 13:25:19 +00:00
|
|
|
const char *name,
|
2014-04-18 12:35:33 +00:00
|
|
|
unsigned int idx,
|
2014-06-25 13:43:04 +00:00
|
|
|
virStorageSourcePtr *parent)
|
2012-10-12 22:29:14 +00:00
|
|
|
{
|
2014-06-25 13:43:04 +00:00
|
|
|
virStorageSourcePtr prev;
|
2014-04-17 11:47:41 +00:00
|
|
|
const char *start = chain->path;
|
2014-05-27 13:32:21 +00:00
|
|
|
char *parentDir = NULL;
|
2014-04-11 03:44:45 +00:00
|
|
|
bool nameIsFile = virStorageIsFile(name);
|
2014-06-25 11:45:54 +00:00
|
|
|
size_t i = 0;
|
2012-10-12 22:29:14 +00:00
|
|
|
|
|
|
|
if (!parent)
|
2014-06-25 13:43:04 +00:00
|
|
|
parent = &prev;
|
2012-10-12 22:29:14 +00:00
|
|
|
*parent = NULL;
|
2014-04-18 12:35:33 +00:00
|
|
|
|
|
|
|
if (startFrom) {
|
blockcommit: require base below top
The block commit code looks for an explicit base file relative
to the discovered top file; so for a chain of:
base <- snap1 <- snap2 <- snap3
and a command of:
virsh blockcommit $dom vda --base snap2 --top snap1
we got a sane message (here from libvirt 1.0.5):
error: invalid argument: could not find base 'snap2' below 'snap1' in chain for 'vda'
Meanwhile, recent refactoring has slightly reduced the quality of the
libvirt error messages, by losing the phrase 'below xyz':
error: invalid argument: could not find image 'snap2' in chain for 'snap3'
But we had a one-off, where we were not excluding the top file
itself in searching for the base; thankfully qemu still reports
the error, but the quality is worse:
virsh blockcommit $dom vda --base snap2 --top snap2
error: internal error unable to execute QEMU command 'block-commit': Base '/snap2' not found
Fix the one-off in blockcommit by changing the semantics of name
lookup - if a starting point is specified, then the result must
be below that point, rather than including that point. The only
other call to chain lookup was blockpull code, which was already
forcing the lookup to omit the active layer and only needs a
tweak to use the new semantics.
This also fixes the bug exposed in the testsuite, where when doing
a lookup pinned to an intermediate point in the chain, we were
unable to return the name of the parent also in the chain.
* src/util/virstoragefile.c (virStorageFileChainLookup): Change
semantics for non-NULL startFrom.
* src/qemu/qemu_driver.c (qemuDomainBlockJobImpl): Adjust caller,
to keep existing semantics.
* tests/virstoragetest.c (mymain): Adjust to expose new semantics.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-06-11 22:22:57 +00:00
|
|
|
while (chain && chain != startFrom->backingStore) {
|
2014-04-18 12:35:33 +00:00
|
|
|
chain = chain->backingStore;
|
|
|
|
i++;
|
|
|
|
}
|
2014-06-25 13:43:04 +00:00
|
|
|
*parent = startFrom;
|
2014-04-18 12:35:33 +00:00
|
|
|
}
|
|
|
|
|
2014-04-11 03:44:45 +00:00
|
|
|
while (chain) {
|
2014-04-18 12:35:33 +00:00
|
|
|
if (!name && !idx) {
|
2014-04-17 20:46:18 +00:00
|
|
|
if (!chain->backingStore)
|
2012-10-12 22:29:14 +00:00
|
|
|
break;
|
2014-04-18 12:35:33 +00:00
|
|
|
} else if (idx) {
|
|
|
|
VIR_DEBUG("%zu: %s", i, chain->path);
|
|
|
|
if (idx == i)
|
|
|
|
break;
|
2014-04-11 03:44:45 +00:00
|
|
|
} else {
|
2014-05-27 08:05:57 +00:00
|
|
|
if (STREQ_NULLABLE(name, chain->relPath) ||
|
|
|
|
STREQ(name, chain->path))
|
2012-10-12 22:29:14 +00:00
|
|
|
break;
|
2014-06-25 11:45:54 +00:00
|
|
|
|
|
|
|
if (nameIsFile && virStorageSourceIsLocalStorage(chain)) {
|
2014-06-25 13:43:04 +00:00
|
|
|
if (*parent && virStorageSourceIsLocalStorage(*parent))
|
|
|
|
parentDir = mdir_name((*parent)->path);
|
2014-06-25 11:45:54 +00:00
|
|
|
else
|
|
|
|
ignore_value(VIR_STRDUP_QUIET(parentDir, "."));
|
|
|
|
|
|
|
|
if (!parentDir) {
|
|
|
|
virReportOOMError();
|
|
|
|
goto error;
|
2014-05-27 13:32:21 +00:00
|
|
|
}
|
|
|
|
|
2014-04-11 03:44:45 +00:00
|
|
|
int result = virFileRelLinkPointsTo(parentDir, name,
|
2014-04-17 11:47:41 +00:00
|
|
|
chain->path);
|
2014-05-27 13:32:21 +00:00
|
|
|
|
|
|
|
VIR_FREE(parentDir);
|
2014-06-25 11:45:54 +00:00
|
|
|
|
2014-04-11 03:44:45 +00:00
|
|
|
if (result < 0)
|
|
|
|
goto error;
|
2014-06-25 11:45:54 +00:00
|
|
|
|
2014-04-11 03:44:45 +00:00
|
|
|
if (result > 0)
|
|
|
|
break;
|
|
|
|
}
|
2012-10-12 22:29:14 +00:00
|
|
|
}
|
2014-06-25 13:43:04 +00:00
|
|
|
*parent = chain;
|
2014-04-17 20:46:18 +00:00
|
|
|
chain = chain->backingStore;
|
2014-04-18 12:35:33 +00:00
|
|
|
i++;
|
2012-10-12 22:29:14 +00:00
|
|
|
}
|
2014-04-18 13:25:19 +00:00
|
|
|
|
2014-04-11 03:44:45 +00:00
|
|
|
if (!chain)
|
2012-10-12 22:29:14 +00:00
|
|
|
goto error;
|
2014-06-25 11:45:54 +00:00
|
|
|
|
2014-04-18 13:25:19 +00:00
|
|
|
return chain;
|
2012-10-12 22:29:14 +00:00
|
|
|
|
2014-03-25 06:53:22 +00:00
|
|
|
error:
|
2014-04-18 12:35:33 +00:00
|
|
|
if (idx) {
|
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("could not find backing store %u in chain for '%s'"),
|
|
|
|
idx, start);
|
|
|
|
} else if (name) {
|
blockcommit: require base below top
The block commit code looks for an explicit base file relative
to the discovered top file; so for a chain of:
base <- snap1 <- snap2 <- snap3
and a command of:
virsh blockcommit $dom vda --base snap2 --top snap1
we got a sane message (here from libvirt 1.0.5):
error: invalid argument: could not find base 'snap2' below 'snap1' in chain for 'vda'
Meanwhile, recent refactoring has slightly reduced the quality of the
libvirt error messages, by losing the phrase 'below xyz':
error: invalid argument: could not find image 'snap2' in chain for 'snap3'
But we had a one-off, where we were not excluding the top file
itself in searching for the base; thankfully qemu still reports
the error, but the quality is worse:
virsh blockcommit $dom vda --base snap2 --top snap2
error: internal error unable to execute QEMU command 'block-commit': Base '/snap2' not found
Fix the one-off in blockcommit by changing the semantics of name
lookup - if a starting point is specified, then the result must
be below that point, rather than including that point. The only
other call to chain lookup was blockpull code, which was already
forcing the lookup to omit the active layer and only needs a
tweak to use the new semantics.
This also fixes the bug exposed in the testsuite, where when doing
a lookup pinned to an intermediate point in the chain, we were
unable to return the name of the parent also in the chain.
* src/util/virstoragefile.c (virStorageFileChainLookup): Change
semantics for non-NULL startFrom.
* src/qemu/qemu_driver.c (qemuDomainBlockJobImpl): Adjust caller,
to keep existing semantics.
* tests/virstoragetest.c (mymain): Adjust to expose new semantics.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-06-11 22:22:57 +00:00
|
|
|
if (startFrom)
|
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("could not find image '%s' beneath '%s' in "
|
|
|
|
"chain for '%s'"), name, startFrom->path, start);
|
|
|
|
else
|
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("could not find image '%s' in chain for '%s'"),
|
|
|
|
name, start);
|
2014-04-18 13:25:19 +00:00
|
|
|
} else {
|
2014-04-11 01:03:01 +00:00
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("could not find base image in chain for '%s'"),
|
|
|
|
start);
|
2014-04-18 13:25:19 +00:00
|
|
|
}
|
2012-10-12 22:29:14 +00:00
|
|
|
*parent = NULL;
|
|
|
|
return NULL;
|
|
|
|
}
|
conf: split network host structs to util/
Continuing the refactoring of host-side storage descriptions out
of conf/domain_conf and into util/virstoragefile, this patch
focuses on details about a host name/port/transport as used by
a network storage volume.
* src/conf/domain_conf.h (virDomainDiskProtocolTransport)
(virDomainDiskHostDef, virDomainDiskHostDefClear)
(virDomainDiskHostDefFree, virDomainDiskHostDefCopy): Move...
* src/util/virstoragefile.h (virStorageNetHostTransport)
(virStorageNetHostDef, virStorageNetHostDefClear)
(virStorageNetHostDefFree, virStorageNetHostDefCopy): ...here,
with better names.
* src/util/virstoragefile.c (virStorageNetHostDefClear)
(virStorageNetHostDefFree, virStorageNetHostDefCopy): Moved from...
* src/conf/domain_conf.c (virDomainDiskHostDefClear)
(virDomainDiskHostDefFree, virDomainDiskHostDefCopy): ...here.
(virDomainDiskSourceDefClear, virDomainDiskSourceDefParse)
(virDomainDiskSourceDefFormatInternal): Adjust callers.
* src/conf/snapshot_conf.h (_virDomainSnapshotDiskDef): Likewise.
* src/conf/snapshot_conf.c (virDomainSnapshotDiskDefClear):
Likewise.
* src/qemu/qemu_command.c (qemuAddRBDHost)
(qemuParseDriveURIString, qemuParseNBDString)
(qemuBuildNetworkDriveURI, qemuParseCommandLineDisk)
(qemuParseCommandLine, qemuGetDriveSourceString): Likewise.
* src/qemu/qemu_command.h: Likewise.
* src/qemu/qemu_conf.c (qemuAddISCSIPoolSourceHost)
(qemuTranslateDiskSourcePool): Likewise.
* src/qemu/qemu_driver.c
(qemuDomainSnapshotCreateSingleDiskActive)
(qemuDomainSnapshotUndoSingleDiskActive): Likewise.
* src/storage/storage_backend_gluster.c
(virStorageFileBackendGlusterInit): Likewise.
* src/storage/storage_driver.c (virStorageFileFree)
(virStorageFileInitInternal): Likewise.
* src/storage/storage_driver.h (_virStorageFile): Likewise.
* src/libvirt_private.syms (domain_conf.h): Move symbols...
(virstoragefile.h): ...as appropriate.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-03-26 22:33:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
virStorageNetHostDefClear(virStorageNetHostDefPtr def)
|
|
|
|
{
|
|
|
|
if (!def)
|
|
|
|
return;
|
|
|
|
|
|
|
|
VIR_FREE(def->name);
|
|
|
|
VIR_FREE(def->port);
|
|
|
|
VIR_FREE(def->socket);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
virStorageNetHostDefFree(size_t nhosts,
|
|
|
|
virStorageNetHostDefPtr hosts)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (!hosts)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (i = 0; i < nhosts; i++)
|
|
|
|
virStorageNetHostDefClear(&hosts[i]);
|
|
|
|
|
|
|
|
VIR_FREE(hosts);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-06-12 14:11:43 +00:00
|
|
|
static void
|
|
|
|
virStoragePermsFree(virStoragePermsPtr def)
|
|
|
|
{
|
|
|
|
if (!def)
|
|
|
|
return;
|
|
|
|
|
|
|
|
VIR_FREE(def->label);
|
|
|
|
VIR_FREE(def);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
conf: split network host structs to util/
Continuing the refactoring of host-side storage descriptions out
of conf/domain_conf and into util/virstoragefile, this patch
focuses on details about a host name/port/transport as used by
a network storage volume.
* src/conf/domain_conf.h (virDomainDiskProtocolTransport)
(virDomainDiskHostDef, virDomainDiskHostDefClear)
(virDomainDiskHostDefFree, virDomainDiskHostDefCopy): Move...
* src/util/virstoragefile.h (virStorageNetHostTransport)
(virStorageNetHostDef, virStorageNetHostDefClear)
(virStorageNetHostDefFree, virStorageNetHostDefCopy): ...here,
with better names.
* src/util/virstoragefile.c (virStorageNetHostDefClear)
(virStorageNetHostDefFree, virStorageNetHostDefCopy): Moved from...
* src/conf/domain_conf.c (virDomainDiskHostDefClear)
(virDomainDiskHostDefFree, virDomainDiskHostDefCopy): ...here.
(virDomainDiskSourceDefClear, virDomainDiskSourceDefParse)
(virDomainDiskSourceDefFormatInternal): Adjust callers.
* src/conf/snapshot_conf.h (_virDomainSnapshotDiskDef): Likewise.
* src/conf/snapshot_conf.c (virDomainSnapshotDiskDefClear):
Likewise.
* src/qemu/qemu_command.c (qemuAddRBDHost)
(qemuParseDriveURIString, qemuParseNBDString)
(qemuBuildNetworkDriveURI, qemuParseCommandLineDisk)
(qemuParseCommandLine, qemuGetDriveSourceString): Likewise.
* src/qemu/qemu_command.h: Likewise.
* src/qemu/qemu_conf.c (qemuAddISCSIPoolSourceHost)
(qemuTranslateDiskSourcePool): Likewise.
* src/qemu/qemu_driver.c
(qemuDomainSnapshotCreateSingleDiskActive)
(qemuDomainSnapshotUndoSingleDiskActive): Likewise.
* src/storage/storage_backend_gluster.c
(virStorageFileBackendGlusterInit): Likewise.
* src/storage/storage_driver.c (virStorageFileFree)
(virStorageFileInitInternal): Likewise.
* src/storage/storage_driver.h (_virStorageFile): Likewise.
* src/libvirt_private.syms (domain_conf.h): Move symbols...
(virstoragefile.h): ...as appropriate.
Signed-off-by: Eric Blake <eblake@redhat.com>
2014-03-26 22:33:08 +00:00
|
|
|
virStorageNetHostDefPtr
|
|
|
|
virStorageNetHostDefCopy(size_t nhosts,
|
|
|
|
virStorageNetHostDefPtr hosts)
|
|
|
|
{
|
|
|
|
virStorageNetHostDefPtr ret = NULL;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (VIR_ALLOC_N(ret, nhosts) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
for (i = 0; i < nhosts; i++) {
|
|
|
|
virStorageNetHostDefPtr src = &hosts[i];
|
|
|
|
virStorageNetHostDefPtr dst = &ret[i];
|
|
|
|
|
|
|
|
dst->transport = src->transport;
|
|
|
|
|
|
|
|
if (VIR_STRDUP(dst->name, src->name) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (VIR_STRDUP(dst->port, src->port) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (VIR_STRDUP(dst->socket, src->socket) < 0)
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
error:
|
|
|
|
virStorageNetHostDefFree(nhosts, ret);
|
|
|
|
return NULL;
|
|
|
|
}
|
2014-03-29 18:00:38 +00:00
|
|
|
|
|
|
|
|
2014-06-23 14:40:49 +00:00
|
|
|
void
|
|
|
|
virStorageAuthDefFree(virStorageAuthDefPtr authdef)
|
|
|
|
{
|
|
|
|
if (!authdef)
|
|
|
|
return;
|
|
|
|
|
|
|
|
VIR_FREE(authdef->username);
|
|
|
|
VIR_FREE(authdef->secrettype);
|
|
|
|
if (authdef->secretType == VIR_STORAGE_SECRET_TYPE_USAGE)
|
|
|
|
VIR_FREE(authdef->secret.usage);
|
|
|
|
VIR_FREE(authdef);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
virStorageAuthDefPtr
|
|
|
|
virStorageAuthDefCopy(const virStorageAuthDef *src)
|
|
|
|
{
|
|
|
|
virStorageAuthDefPtr ret;
|
|
|
|
|
|
|
|
if (VIR_ALLOC(ret) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (VIR_STRDUP(ret->username, src->username) < 0)
|
|
|
|
goto error;
|
|
|
|
/* Not present for storage pool, but used for disk source */
|
|
|
|
if (VIR_STRDUP(ret->secrettype, src->secrettype) < 0)
|
|
|
|
goto error;
|
|
|
|
ret->authType = src->authType;
|
|
|
|
ret->secretType = src->secretType;
|
|
|
|
if (ret->secretType == VIR_STORAGE_SECRET_TYPE_UUID) {
|
|
|
|
memcpy(ret->secret.uuid, src->secret.uuid, sizeof(ret->secret.uuid));
|
|
|
|
} else if (ret->secretType == VIR_STORAGE_SECRET_TYPE_USAGE) {
|
|
|
|
if (VIR_STRDUP(ret->secret.usage, src->secret.usage) < 0)
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
error:
|
|
|
|
virStorageAuthDefFree(ret);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
virStorageAuthDefParseSecret(xmlXPathContextPtr ctxt,
|
|
|
|
virStorageAuthDefPtr authdef)
|
|
|
|
{
|
|
|
|
char *uuid;
|
|
|
|
char *usage;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
/* Used by the domain disk xml parsing in order to ensure the
|
|
|
|
* <secret type='%s' value matches the expected secret type for
|
|
|
|
* the style of disk (iscsi is chap, nbd is ceph). For some reason
|
|
|
|
* the virSecretUsageType{From|To}String() cannot be linked here
|
|
|
|
* and because only the domain parsing code cares - just keep
|
|
|
|
* it as a string.
|
|
|
|
*/
|
|
|
|
authdef->secrettype = virXPathString("string(./secret/@type)", ctxt);
|
|
|
|
|
|
|
|
uuid = virXPathString("string(./secret/@uuid)", ctxt);
|
|
|
|
usage = virXPathString("string(./secret/@usage)", ctxt);
|
|
|
|
if (uuid == NULL && usage == NULL) {
|
|
|
|
virReportError(VIR_ERR_XML_ERROR, "%s",
|
|
|
|
_("missing auth secret uuid or usage attribute"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (uuid && usage) {
|
|
|
|
virReportError(VIR_ERR_XML_ERROR, "%s",
|
|
|
|
_("either auth secret uuid or usage expected"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (uuid) {
|
|
|
|
if (virUUIDParse(uuid, authdef->secret.uuid) < 0) {
|
|
|
|
virReportError(VIR_ERR_XML_ERROR, "%s",
|
|
|
|
_("invalid auth secret uuid"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
authdef->secretType = VIR_STORAGE_SECRET_TYPE_UUID;
|
|
|
|
} else {
|
|
|
|
authdef->secret.usage = usage;
|
|
|
|
usage = NULL;
|
|
|
|
authdef->secretType = VIR_STORAGE_SECRET_TYPE_USAGE;
|
|
|
|
}
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(uuid);
|
|
|
|
VIR_FREE(usage);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static virStorageAuthDefPtr
|
|
|
|
virStorageAuthDefParseXML(xmlXPathContextPtr ctxt)
|
|
|
|
{
|
|
|
|
virStorageAuthDefPtr authdef = NULL;
|
|
|
|
char *username = NULL;
|
|
|
|
char *authtype = NULL;
|
|
|
|
|
|
|
|
if (VIR_ALLOC(authdef) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (!(username = virXPathString("string(./@username)", ctxt))) {
|
|
|
|
virReportError(VIR_ERR_XML_ERROR, "%s",
|
|
|
|
_("missing username for auth"));
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
authdef->username = username;
|
|
|
|
username = NULL;
|
|
|
|
|
|
|
|
authdef->authType = VIR_STORAGE_AUTH_TYPE_NONE;
|
|
|
|
authtype = virXPathString("string(./@type)", ctxt);
|
|
|
|
if (authtype) {
|
|
|
|
/* Used by the storage pool instead of the secret type field
|
|
|
|
* to define whether chap or ceph being used
|
|
|
|
*/
|
|
|
|
if ((authdef->authType = virStorageAuthTypeFromString(authtype)) < 0) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
_("unknown auth type '%s'"), authtype);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
VIR_FREE(authtype);
|
|
|
|
}
|
|
|
|
|
|
|
|
authdef->secretType = VIR_STORAGE_SECRET_TYPE_NONE;
|
|
|
|
if (virStorageAuthDefParseSecret(ctxt, authdef) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
return authdef;
|
|
|
|
|
|
|
|
error:
|
|
|
|
VIR_FREE(authtype);
|
|
|
|
VIR_FREE(username);
|
|
|
|
virStorageAuthDefFree(authdef);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
virStorageAuthDefPtr
|
|
|
|
virStorageAuthDefParse(xmlDocPtr xml, xmlNodePtr root)
|
|
|
|
{
|
|
|
|
xmlXPathContextPtr ctxt = NULL;
|
|
|
|
virStorageAuthDefPtr authdef = NULL;
|
|
|
|
|
|
|
|
ctxt = xmlXPathNewContext(xml);
|
|
|
|
if (ctxt == NULL) {
|
|
|
|
virReportOOMError();
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ctxt->node = root;
|
|
|
|
authdef = virStorageAuthDefParseXML(ctxt);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
xmlXPathFreeContext(ctxt);
|
|
|
|
return authdef;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
virStorageAuthDefFormat(virBufferPtr buf,
|
|
|
|
virStorageAuthDefPtr authdef)
|
|
|
|
{
|
|
|
|
char uuidstr[VIR_UUID_STRING_BUFLEN];
|
|
|
|
|
|
|
|
if (authdef->authType == VIR_STORAGE_AUTH_TYPE_NONE) {
|
|
|
|
virBufferEscapeString(buf, "<auth username='%s'>\n", authdef->username);
|
|
|
|
} else {
|
|
|
|
virBufferAsprintf(buf, "<auth type='%s' ",
|
|
|
|
virStorageAuthTypeToString(authdef->authType));
|
|
|
|
virBufferEscapeString(buf, "username='%s'>\n", authdef->username);
|
|
|
|
}
|
|
|
|
|
|
|
|
virBufferAdjustIndent(buf, 2);
|
|
|
|
if (authdef->secrettype)
|
|
|
|
virBufferAsprintf(buf, "<secret type='%s'", authdef->secrettype);
|
|
|
|
else
|
|
|
|
virBufferAddLit(buf, "<secret");
|
|
|
|
|
|
|
|
if (authdef->secretType == VIR_STORAGE_SECRET_TYPE_UUID) {
|
|
|
|
virUUIDFormat(authdef->secret.uuid, uuidstr);
|
|
|
|
virBufferAsprintf(buf, " uuid='%s'/>\n", uuidstr);
|
|
|
|
} else if (authdef->secretType == VIR_STORAGE_SECRET_TYPE_USAGE) {
|
|
|
|
virBufferEscapeString(buf, " usage='%s'/>\n",
|
|
|
|
authdef->secret.usage);
|
|
|
|
} else {
|
|
|
|
virBufferAddLit(buf, "/>\n");
|
|
|
|
}
|
|
|
|
virBufferAdjustIndent(buf, -2);
|
|
|
|
virBufferAddLit(buf, "</auth>\n");
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-06-18 16:39:47 +00:00
|
|
|
virSecurityDeviceLabelDefPtr
|
|
|
|
virStorageSourceGetSecurityLabelDef(virStorageSourcePtr src,
|
|
|
|
const char *model)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < src->nseclabels; i++) {
|
|
|
|
if (STREQ_NULLABLE(src->seclabels[i]->model, model))
|
|
|
|
return src->seclabels[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-06-16 11:10:42 +00:00
|
|
|
static void
|
|
|
|
virStorageSourceSeclabelsClear(virStorageSourcePtr def)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (def->seclabels) {
|
|
|
|
for (i = 0; i < def->nseclabels; i++)
|
|
|
|
virSecurityDeviceLabelDefFree(def->seclabels[i]);
|
|
|
|
VIR_FREE(def->seclabels);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
virStorageSourceSeclabelsCopy(virStorageSourcePtr to,
|
|
|
|
const virStorageSource *from)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
if (from->nseclabels == 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (VIR_ALLOC_N(to->seclabels, from->nseclabels) < 0)
|
|
|
|
return -1;
|
|
|
|
to->nseclabels = from->nseclabels;
|
|
|
|
|
|
|
|
for (i = 0; i < to->nseclabels; i++) {
|
|
|
|
if (!(to->seclabels[i] = virSecurityDeviceLabelDefCopy(from->seclabels[i])))
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error:
|
|
|
|
virStorageSourceSeclabelsClear(to);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static virStorageTimestampsPtr
|
|
|
|
virStorageTimestampsCopy(const virStorageTimestamps *src)
|
|
|
|
{
|
|
|
|
virStorageTimestampsPtr ret;
|
|
|
|
|
|
|
|
if (VIR_ALLOC(ret) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
memcpy(ret, src, sizeof(*src));
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static virStoragePermsPtr
|
|
|
|
virStoragePermsCopy(const virStoragePerms *src)
|
|
|
|
{
|
|
|
|
virStoragePermsPtr ret;
|
|
|
|
|
|
|
|
if (VIR_ALLOC(ret) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
ret->mode = src->mode;
|
|
|
|
ret->uid = src->uid;
|
|
|
|
ret->gid = src->gid;
|
|
|
|
|
|
|
|
if (VIR_STRDUP(ret->label, src->label))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
error:
|
|
|
|
virStoragePermsFree(ret);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static virStorageSourcePoolDefPtr
|
|
|
|
virStorageSourcePoolDefCopy(const virStorageSourcePoolDef *src)
|
|
|
|
{
|
|
|
|
virStorageSourcePoolDefPtr ret;
|
|
|
|
|
|
|
|
if (VIR_ALLOC(ret) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
ret->voltype = src->voltype;
|
|
|
|
ret->pooltype = src->pooltype;
|
|
|
|
ret->actualtype = src->actualtype;
|
|
|
|
ret->mode = src->mode;
|
|
|
|
|
|
|
|
if (VIR_STRDUP(ret->pool, src->pool) < 0 ||
|
|
|
|
VIR_STRDUP(ret->volume, src->volume) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
error:
|
|
|
|
virStorageSourcePoolDefFree(ret);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virStorageSourcePtr:
|
|
|
|
*
|
|
|
|
* Deep-copies a virStorageSource structure. If @backing chain is true
|
|
|
|
* then also copies the backing chain recursively, otherwise just
|
|
|
|
* the top element is copied. This function doesn't copy the
|
|
|
|
* storage driver access structure and thus the struct needs to be initialized
|
|
|
|
* separately.
|
|
|
|
*/
|
|
|
|
virStorageSourcePtr
|
|
|
|
virStorageSourceCopy(const virStorageSource *src,
|
|
|
|
bool backingChain)
|
|
|
|
{
|
|
|
|
virStorageSourcePtr ret = NULL;
|
|
|
|
|
|
|
|
if (VIR_ALLOC(ret) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
ret->type = src->type;
|
|
|
|
ret->protocol = src->protocol;
|
|
|
|
ret->format = src->format;
|
|
|
|
ret->allocation = src->allocation;
|
|
|
|
ret->capacity = src->capacity;
|
|
|
|
ret->readonly = src->readonly;
|
|
|
|
ret->shared = src->shared;
|
|
|
|
|
|
|
|
/* storage driver metadata are not copied */
|
|
|
|
ret->drv = NULL;
|
|
|
|
|
|
|
|
if (VIR_STRDUP(ret->path, src->path) < 0 ||
|
|
|
|
VIR_STRDUP(ret->volume, src->volume) < 0 ||
|
|
|
|
VIR_STRDUP(ret->driverName, src->driverName) < 0 ||
|
|
|
|
VIR_STRDUP(ret->relPath, src->relPath) < 0 ||
|
|
|
|
VIR_STRDUP(ret->backingStoreRaw, src->backingStoreRaw) < 0 ||
|
2014-11-11 10:35:25 +00:00
|
|
|
VIR_STRDUP(ret->snapshot, src->snapshot) < 0 ||
|
2014-11-11 16:31:24 +00:00
|
|
|
VIR_STRDUP(ret->configFile, src->configFile) < 0 ||
|
2014-06-16 11:10:42 +00:00
|
|
|
VIR_STRDUP(ret->compat, src->compat) < 0)
|
|
|
|
goto error;
|
|
|
|
|
2014-10-30 10:52:17 +00:00
|
|
|
if (src->nhosts) {
|
|
|
|
if (!(ret->hosts = virStorageNetHostDefCopy(src->nhosts, src->hosts)))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
ret->nhosts = src->nhosts;
|
|
|
|
}
|
2014-06-16 11:10:42 +00:00
|
|
|
|
|
|
|
if (src->srcpool &&
|
|
|
|
!(ret->srcpool = virStorageSourcePoolDefCopy(src->srcpool)))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (src->features &&
|
|
|
|
!(ret->features = virBitmapNewCopy(src->features)))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (src->encryption &&
|
|
|
|
!(ret->encryption = virStorageEncryptionCopy(src->encryption)))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (src->perms &&
|
|
|
|
!(ret->perms = virStoragePermsCopy(src->perms)))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (src->timestamps &&
|
|
|
|
!(ret->timestamps = virStorageTimestampsCopy(src->timestamps)))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (virStorageSourceSeclabelsCopy(ret, src) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (src->auth &&
|
|
|
|
!(ret->auth = virStorageAuthDefCopy(src->auth)))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (backingChain && src->backingStore) {
|
|
|
|
if (!(ret->backingStore = virStorageSourceCopy(src->backingStore,
|
|
|
|
true)))
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
error:
|
|
|
|
virStorageSourceFree(ret);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-06-25 16:48:27 +00:00
|
|
|
/**
|
|
|
|
* virStorageSourceInitChainElement:
|
|
|
|
* @newelem: New backing chain element disk source
|
|
|
|
* @old: Existing top level disk source
|
2014-11-19 17:54:43 +00:00
|
|
|
* @transferLabels: Transfer security lables.
|
2014-06-25 16:48:27 +00:00
|
|
|
*
|
|
|
|
* Transfers relevant information from the existing disk source to the new
|
|
|
|
* backing chain element if they weren't supplied so that labelling info
|
|
|
|
* and possibly other stuff is correct.
|
|
|
|
*
|
2014-11-19 17:54:43 +00:00
|
|
|
* If @transferLabels is true, security labels from the existing disk are copied
|
|
|
|
* to the new disk. Otherwise the default domain imagelabel label will be used.
|
2014-06-25 16:48:27 +00:00
|
|
|
*
|
|
|
|
* Returns 0 on success, -1 on error.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
virStorageSourceInitChainElement(virStorageSourcePtr newelem,
|
|
|
|
virStorageSourcePtr old,
|
2014-11-19 17:54:43 +00:00
|
|
|
bool transferLabels)
|
2014-06-25 16:48:27 +00:00
|
|
|
{
|
|
|
|
int ret = -1;
|
|
|
|
|
2014-11-19 17:54:43 +00:00
|
|
|
if (transferLabels &&
|
|
|
|
!newelem->seclabels &&
|
2014-06-25 16:48:27 +00:00
|
|
|
virStorageSourceSeclabelsCopy(newelem, old) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2014-09-16 10:55:32 +00:00
|
|
|
if (!newelem->driverName &&
|
|
|
|
VIR_STRDUP(newelem->driverName, old->driverName) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2014-06-25 16:48:27 +00:00
|
|
|
newelem->shared = old->shared;
|
|
|
|
newelem->readonly = old->readonly;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-29 18:00:38 +00:00
|
|
|
void
|
|
|
|
virStorageSourcePoolDefFree(virStorageSourcePoolDefPtr def)
|
|
|
|
{
|
|
|
|
if (!def)
|
|
|
|
return;
|
|
|
|
|
|
|
|
VIR_FREE(def->pool);
|
|
|
|
VIR_FREE(def->volume);
|
|
|
|
|
|
|
|
VIR_FREE(def);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-08 06:51:06 +00:00
|
|
|
int
|
|
|
|
virStorageSourceGetActualType(virStorageSourcePtr def)
|
|
|
|
{
|
|
|
|
if (def->type == VIR_STORAGE_TYPE_VOLUME && def->srcpool)
|
|
|
|
return def->srcpool->actualtype;
|
|
|
|
|
|
|
|
return def->type;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-06-25 11:47:32 +00:00
|
|
|
bool
|
|
|
|
virStorageSourceIsLocalStorage(virStorageSourcePtr src)
|
|
|
|
{
|
2014-09-03 16:54:56 +00:00
|
|
|
virStorageType type = virStorageSourceGetActualType(src);
|
|
|
|
|
|
|
|
switch (type) {
|
|
|
|
case VIR_STORAGE_TYPE_FILE:
|
|
|
|
case VIR_STORAGE_TYPE_BLOCK:
|
|
|
|
case VIR_STORAGE_TYPE_DIR:
|
|
|
|
return true;
|
|
|
|
|
|
|
|
case VIR_STORAGE_TYPE_NETWORK:
|
|
|
|
case VIR_STORAGE_TYPE_VOLUME:
|
|
|
|
case VIR_STORAGE_TYPE_LAST:
|
|
|
|
case VIR_STORAGE_TYPE_NONE:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2014-06-25 11:47:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-09-11 17:43:53 +00:00
|
|
|
/**
|
|
|
|
* virStorageSourceIsEmpty:
|
|
|
|
*
|
|
|
|
* @src: disk source to check
|
|
|
|
*
|
|
|
|
* Returns true if the guest disk has no associated host storage source
|
|
|
|
* (such as an empty cdrom drive).
|
|
|
|
*/
|
|
|
|
bool
|
|
|
|
virStorageSourceIsEmpty(virStorageSourcePtr src)
|
|
|
|
{
|
|
|
|
if (virStorageSourceIsLocalStorage(src) && !src->path)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (src->type == VIR_STORAGE_TYPE_NONE)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-26 06:27:58 +00:00
|
|
|
/**
|
2014-06-20 08:40:45 +00:00
|
|
|
* virStorageSourceBackingStoreClear:
|
2014-04-26 06:27:58 +00:00
|
|
|
*
|
|
|
|
* @src: disk source to clear
|
|
|
|
*
|
|
|
|
* Clears information about backing store of the current storage file.
|
|
|
|
*/
|
|
|
|
void
|
2014-06-20 08:40:45 +00:00
|
|
|
virStorageSourceBackingStoreClear(virStorageSourcePtr def)
|
2014-04-26 06:27:58 +00:00
|
|
|
{
|
|
|
|
if (!def)
|
|
|
|
return;
|
|
|
|
|
|
|
|
VIR_FREE(def->relPath);
|
|
|
|
VIR_FREE(def->backingStoreRaw);
|
|
|
|
|
|
|
|
/* recursively free backing chain */
|
|
|
|
virStorageSourceFree(def->backingStore);
|
|
|
|
def->backingStore = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-29 18:00:38 +00:00
|
|
|
void
|
|
|
|
virStorageSourceClear(virStorageSourcePtr def)
|
|
|
|
{
|
|
|
|
if (!def)
|
|
|
|
return;
|
|
|
|
|
|
|
|
VIR_FREE(def->path);
|
2014-05-02 14:17:42 +00:00
|
|
|
VIR_FREE(def->volume);
|
2014-03-29 18:00:38 +00:00
|
|
|
virStorageSourcePoolDefFree(def->srcpool);
|
|
|
|
VIR_FREE(def->driverName);
|
2014-03-30 03:21:06 +00:00
|
|
|
virBitmapFree(def->features);
|
|
|
|
VIR_FREE(def->compat);
|
2014-03-29 18:00:38 +00:00
|
|
|
virStorageEncryptionFree(def->encryption);
|
2014-06-16 11:10:42 +00:00
|
|
|
virStorageSourceSeclabelsClear(def);
|
2014-06-12 14:11:43 +00:00
|
|
|
virStoragePermsFree(def->perms);
|
2014-03-30 03:21:06 +00:00
|
|
|
VIR_FREE(def->timestamps);
|
2014-03-29 18:00:38 +00:00
|
|
|
|
|
|
|
virStorageNetHostDefFree(def->nhosts, def->hosts);
|
2014-06-24 13:46:23 +00:00
|
|
|
virStorageAuthDefFree(def->auth);
|
2014-04-17 14:04:33 +00:00
|
|
|
|
2014-06-20 08:40:45 +00:00
|
|
|
virStorageSourceBackingStoreClear(def);
|
2014-03-29 18:00:38 +00:00
|
|
|
}
|
2014-04-07 15:58:08 +00:00
|
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
virStorageSourceFree(virStorageSourcePtr def)
|
|
|
|
{
|
|
|
|
if (!def)
|
|
|
|
return;
|
|
|
|
|
|
|
|
virStorageSourceClear(def);
|
|
|
|
VIR_FREE(def);
|
|
|
|
}
|
2014-04-25 11:23:50 +00:00
|
|
|
|
|
|
|
|
|
|
|
static virStorageSourcePtr
|
|
|
|
virStorageSourceNewFromBackingRelative(virStorageSourcePtr parent,
|
|
|
|
const char *rel)
|
|
|
|
{
|
|
|
|
char *dirname = NULL;
|
|
|
|
virStorageSourcePtr ret;
|
|
|
|
|
|
|
|
if (VIR_ALLOC(ret) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
2014-05-27 08:05:57 +00:00
|
|
|
/* store relative name */
|
|
|
|
if (VIR_STRDUP(ret->relPath, parent->backingStoreRaw) < 0)
|
|
|
|
goto error;
|
|
|
|
|
2014-05-27 13:32:21 +00:00
|
|
|
if (!(dirname = mdir_name(parent->path))) {
|
|
|
|
virReportOOMError();
|
2014-04-25 11:23:50 +00:00
|
|
|
goto error;
|
2014-05-27 13:32:21 +00:00
|
|
|
}
|
2014-04-25 11:23:50 +00:00
|
|
|
|
2014-05-27 13:32:21 +00:00
|
|
|
if (STRNEQ(dirname, "/")) {
|
|
|
|
if (virAsprintf(&ret->path, "%s/%s", dirname, rel) < 0)
|
2014-04-25 11:23:50 +00:00
|
|
|
goto error;
|
|
|
|
} else {
|
2014-05-27 13:32:21 +00:00
|
|
|
if (virAsprintf(&ret->path, "/%s", rel) < 0)
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virStorageSourceGetActualType(parent) == VIR_STORAGE_TYPE_NETWORK) {
|
2014-04-25 11:23:50 +00:00
|
|
|
ret->type = VIR_STORAGE_TYPE_NETWORK;
|
|
|
|
|
|
|
|
/* copy the host network part */
|
|
|
|
ret->protocol = parent->protocol;
|
2014-10-30 10:52:17 +00:00
|
|
|
if (parent->nhosts) {
|
|
|
|
if (!(ret->hosts = virStorageNetHostDefCopy(parent->nhosts,
|
|
|
|
parent->hosts)))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
ret->nhosts = parent->nhosts;
|
|
|
|
}
|
2014-04-25 11:23:50 +00:00
|
|
|
|
|
|
|
if (VIR_STRDUP(ret->volume, parent->volume) < 0)
|
|
|
|
goto error;
|
2014-05-27 13:32:21 +00:00
|
|
|
} else {
|
|
|
|
/* set the type to _FILE, the caller shall update it to the actual type */
|
|
|
|
ret->type = VIR_STORAGE_TYPE_FILE;
|
2014-04-25 11:23:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(dirname);
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
error:
|
|
|
|
virStorageSourceFree(ret);
|
|
|
|
ret = NULL;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
virStorageSourceParseBackingURI(virStorageSourcePtr src,
|
|
|
|
const char *path)
|
|
|
|
{
|
|
|
|
virURIPtr uri = NULL;
|
|
|
|
char **scheme = NULL;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (!(uri = virURIParse(path))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("failed to parse backing file location '%s'"),
|
|
|
|
path);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2014-10-29 09:55:23 +00:00
|
|
|
if (VIR_ALLOC(src->hosts) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
src->nhosts = 1;
|
|
|
|
|
2014-04-25 11:23:50 +00:00
|
|
|
if (!(scheme = virStringSplit(uri->scheme, "+", 2)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (!scheme[0] ||
|
|
|
|
(src->protocol = virStorageNetProtocolTypeFromString(scheme[0])) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("invalid backing protocol '%s'"),
|
|
|
|
NULLSTR(scheme[0]));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (scheme[1] &&
|
|
|
|
(src->hosts->transport = virStorageNetHostTransportTypeFromString(scheme[1])) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("invalid protocol transport type '%s'"),
|
|
|
|
scheme[1]);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* handle socket stored as a query */
|
|
|
|
if (uri->query) {
|
|
|
|
if (VIR_STRDUP(src->hosts->socket, STRSKIP(uri->query, "socket=")) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* XXX We currently don't support auth, so don't bother parsing it */
|
|
|
|
|
|
|
|
/* possibly skip the leading slash */
|
|
|
|
if (VIR_STRDUP(src->path,
|
|
|
|
*uri->path == '/' ? uri->path + 1 : uri->path) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (src->protocol == VIR_STORAGE_NET_PROTOCOL_GLUSTER) {
|
|
|
|
char *tmp;
|
|
|
|
if (!(tmp = strchr(src->path, '/')) ||
|
|
|
|
tmp == src->path) {
|
|
|
|
virReportError(VIR_ERR_XML_ERROR,
|
|
|
|
_("missing volume name or file name in "
|
|
|
|
"gluster source path '%s'"), src->path);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
src->volume = src->path;
|
|
|
|
|
|
|
|
if (VIR_STRDUP(src->path, tmp) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
tmp[0] = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (uri->port > 0) {
|
|
|
|
if (virAsprintf(&src->hosts->port, "%d", uri->port) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (VIR_STRDUP(src->hosts->name, uri->server) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virURIFree(uri);
|
|
|
|
virStringFreeList(scheme);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-10-31 16:49:56 +00:00
|
|
|
static int
|
|
|
|
virStorageSourceRBDAddHost(virStorageSourcePtr src,
|
|
|
|
char *hostport)
|
|
|
|
{
|
|
|
|
char *port;
|
|
|
|
size_t skip;
|
|
|
|
char **parts;
|
|
|
|
|
|
|
|
if (VIR_EXPAND_N(src->hosts, src->nhosts, 1) < 0)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if ((port = strchr(hostport, ']'))) {
|
|
|
|
/* ipv6, strip brackets */
|
|
|
|
hostport += 1;
|
|
|
|
skip = 3;
|
|
|
|
} else {
|
|
|
|
port = strstr(hostport, "\\:");
|
|
|
|
skip = 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (port) {
|
|
|
|
*port = '\0';
|
|
|
|
port += skip;
|
|
|
|
if (VIR_STRDUP(src->hosts[src->nhosts - 1].port, port) < 0)
|
|
|
|
goto error;
|
|
|
|
} else {
|
|
|
|
if (VIR_STRDUP(src->hosts[src->nhosts - 1].port, "6789") < 0)
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
parts = virStringSplit(hostport, "\\:", 0);
|
|
|
|
if (!parts)
|
|
|
|
goto error;
|
|
|
|
src->hosts[src->nhosts-1].name = virStringJoin((const char **)parts, ":");
|
|
|
|
virStringFreeList(parts);
|
|
|
|
if (!src->hosts[src->nhosts-1].name)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
src->hosts[src->nhosts-1].transport = VIR_STORAGE_NET_HOST_TRANS_TCP;
|
|
|
|
src->hosts[src->nhosts-1].socket = NULL;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error:
|
|
|
|
VIR_FREE(src->hosts[src->nhosts-1].port);
|
|
|
|
VIR_FREE(src->hosts[src->nhosts-1].name);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
virStorageSourceParseRBDColonString(const char *rbdstr,
|
|
|
|
virStorageSourcePtr src)
|
|
|
|
{
|
|
|
|
char *options = NULL;
|
|
|
|
char *p, *e, *next;
|
|
|
|
virStorageAuthDefPtr authdef = NULL;
|
|
|
|
|
|
|
|
/* optionally skip the "rbd:" prefix if provided */
|
|
|
|
if (STRPREFIX(rbdstr, "rbd:"))
|
|
|
|
rbdstr += strlen("rbd:");
|
|
|
|
|
|
|
|
if (VIR_STRDUP(src->path, rbdstr) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
p = strchr(src->path, ':');
|
|
|
|
if (p) {
|
|
|
|
if (VIR_STRDUP(options, p + 1) < 0)
|
|
|
|
goto error;
|
|
|
|
*p = '\0';
|
|
|
|
}
|
|
|
|
|
2014-11-11 10:35:25 +00:00
|
|
|
/* snapshot name */
|
|
|
|
if ((p = strchr(src->path, '@'))) {
|
|
|
|
if (VIR_STRDUP(src->snapshot, p + 1) < 0)
|
|
|
|
goto error;
|
|
|
|
*p = '\0';
|
|
|
|
}
|
|
|
|
|
2014-10-31 16:49:56 +00:00
|
|
|
/* options */
|
|
|
|
if (!options)
|
|
|
|
return 0; /* all done */
|
|
|
|
|
|
|
|
p = options;
|
|
|
|
while (*p) {
|
|
|
|
/* find : delimiter or end of string */
|
|
|
|
for (e = p; *e && *e != ':'; ++e) {
|
|
|
|
if (*e == '\\') {
|
|
|
|
e++;
|
|
|
|
if (*e == '\0')
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (*e == '\0') {
|
|
|
|
next = e; /* last kv pair */
|
|
|
|
} else {
|
|
|
|
next = e + 1;
|
|
|
|
*e = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (STRPREFIX(p, "id=")) {
|
|
|
|
/* formulate authdef for src->auth */
|
|
|
|
if (VIR_ALLOC(authdef) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (VIR_STRDUP(authdef->username, p + strlen("id=")) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (VIR_STRDUP(authdef->secrettype,
|
|
|
|
virStorageAuthTypeToString(VIR_STORAGE_AUTH_TYPE_CEPHX)) < 0)
|
|
|
|
goto error;
|
|
|
|
src->auth = authdef;
|
|
|
|
authdef = NULL;
|
|
|
|
|
|
|
|
/* Cannot formulate a secretType (eg, usage or uuid) given
|
|
|
|
* what is provided.
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
if (STRPREFIX(p, "mon_host=")) {
|
|
|
|
char *h, *sep;
|
|
|
|
|
|
|
|
h = p + strlen("mon_host=");
|
|
|
|
while (h < e) {
|
|
|
|
for (sep = h; sep < e; ++sep) {
|
|
|
|
if (*sep == '\\' && (sep[1] == ',' ||
|
|
|
|
sep[1] == ';' ||
|
|
|
|
sep[1] == ' ')) {
|
|
|
|
*sep = '\0';
|
|
|
|
sep += 2;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virStorageSourceRBDAddHost(src, h) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
h = sep;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-11-11 16:31:24 +00:00
|
|
|
if (STRPREFIX(p, "conf=") &&
|
|
|
|
VIR_STRDUP(src->configFile, p + strlen("conf=")) < 0)
|
|
|
|
goto error;
|
|
|
|
|
2014-10-31 16:49:56 +00:00
|
|
|
p = next;
|
|
|
|
}
|
|
|
|
VIR_FREE(options);
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
error:
|
|
|
|
VIR_FREE(options);
|
|
|
|
virStorageAuthDefFree(authdef);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-04-25 11:23:50 +00:00
|
|
|
static int
|
2014-11-04 13:07:53 +00:00
|
|
|
virStorageSourceParseNBDColonString(const char *nbdstr,
|
|
|
|
virStorageSourcePtr src)
|
2014-04-25 11:23:50 +00:00
|
|
|
{
|
|
|
|
char **backing = NULL;
|
|
|
|
int ret = -1;
|
|
|
|
|
2014-11-04 13:07:53 +00:00
|
|
|
if (!(backing = virStringSplit(nbdstr, ":", 0)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
/* we know that backing[0] now equals to "nbd" */
|
|
|
|
|
|
|
|
if (VIR_ALLOC_N(src->hosts, 1) < 0)
|
2014-04-25 11:23:50 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2014-11-04 13:07:53 +00:00
|
|
|
src->nhosts = 1;
|
|
|
|
src->hosts->transport = VIR_STORAGE_NET_HOST_TRANS_TCP;
|
|
|
|
|
|
|
|
/* format: [] denotes optional sections, uppercase are variable strings
|
|
|
|
* nbd:unix:/PATH/TO/SOCKET[:exportname=EXPORTNAME]
|
|
|
|
* nbd:HOSTNAME:PORT[:exportname=EXPORTNAME]
|
|
|
|
*/
|
|
|
|
if (!backing[1]) {
|
2014-04-25 11:23:50 +00:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
2014-11-04 13:07:53 +00:00
|
|
|
_("missing remote information in '%s' for protocol nbd"),
|
|
|
|
nbdstr);
|
2014-04-25 11:23:50 +00:00
|
|
|
goto cleanup;
|
2014-11-04 13:07:53 +00:00
|
|
|
} else if (STREQ(backing[1], "unix")) {
|
|
|
|
if (!backing[2]) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("missing unix socket path in nbd backing string %s"),
|
|
|
|
nbdstr);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2014-04-25 11:23:50 +00:00
|
|
|
|
2014-11-04 13:07:53 +00:00
|
|
|
if (VIR_STRDUP(src->hosts->socket, backing[2]) < 0)
|
2014-04-25 11:23:50 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2014-11-04 13:07:53 +00:00
|
|
|
} else {
|
2014-04-25 11:23:50 +00:00
|
|
|
if (!backing[1]) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
2014-11-04 13:07:53 +00:00
|
|
|
_("missing host name in nbd string '%s'"),
|
|
|
|
nbdstr);
|
2014-04-25 11:23:50 +00:00
|
|
|
goto cleanup;
|
2014-11-04 13:07:53 +00:00
|
|
|
}
|
2014-04-25 11:23:50 +00:00
|
|
|
|
2014-11-04 13:07:53 +00:00
|
|
|
if (VIR_STRDUP(src->hosts->name, backing[1]) < 0)
|
|
|
|
goto cleanup;
|
2014-04-25 11:23:50 +00:00
|
|
|
|
2014-11-04 13:07:53 +00:00
|
|
|
if (!backing[2]) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("missing port in nbd string '%s'"),
|
|
|
|
nbdstr);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2014-04-25 11:23:50 +00:00
|
|
|
|
2014-11-04 13:07:53 +00:00
|
|
|
if (VIR_STRDUP(src->hosts->port, backing[2]) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2014-04-25 11:23:50 +00:00
|
|
|
|
2014-11-04 13:07:53 +00:00
|
|
|
if (backing[3] && STRPREFIX(backing[3], "exportname=")) {
|
|
|
|
if (VIR_STRDUP(src->path, backing[3] + strlen("exportname=")) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
}
|
2014-04-25 11:23:50 +00:00
|
|
|
|
2014-11-04 13:07:53 +00:00
|
|
|
ret = 0;
|
2014-04-25 11:23:50 +00:00
|
|
|
|
2014-11-04 13:07:53 +00:00
|
|
|
cleanup:
|
|
|
|
virStringFreeList(backing);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
virStorageSourceParseBackingColon(virStorageSourcePtr src,
|
|
|
|
const char *path)
|
|
|
|
{
|
|
|
|
char *protocol = NULL;
|
|
|
|
const char *p;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (!(p = strchr(path, ':'))) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("invalid backing protocol string '%s'"),
|
|
|
|
path);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (VIR_STRNDUP(protocol, path, p - path) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if ((src->protocol = virStorageNetProtocolTypeFromString(protocol)) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("invalid backing protocol '%s'"),
|
|
|
|
protocol);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
switch ((virStorageNetProtocol) src->protocol) {
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_NBD:
|
|
|
|
if (virStorageSourceParseNBDColonString(path, src) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
break;
|
2014-04-25 11:23:50 +00:00
|
|
|
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_RBD:
|
2014-11-04 16:35:16 +00:00
|
|
|
if (virStorageSourceParseRBDColonString(path, src) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_SHEEPDOG:
|
2014-04-25 11:23:50 +00:00
|
|
|
case VIR_STORAGE_NET_PROTOCOL_LAST:
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_NONE:
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("backing store parser is not implemented for protocol %s"),
|
2014-11-04 13:07:53 +00:00
|
|
|
protocol);
|
2014-04-25 11:23:50 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_HTTP:
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_HTTPS:
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_FTP:
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_FTPS:
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_TFTP:
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_ISCSI:
|
|
|
|
case VIR_STORAGE_NET_PROTOCOL_GLUSTER:
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("malformed backing store path for protocol %s"),
|
2014-11-04 13:07:53 +00:00
|
|
|
protocol);
|
2014-04-25 11:23:50 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
2014-11-04 13:07:53 +00:00
|
|
|
VIR_FREE(protocol);
|
2014-04-25 11:23:50 +00:00
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static virStorageSourcePtr
|
|
|
|
virStorageSourceNewFromBackingAbsolute(const char *path)
|
|
|
|
{
|
|
|
|
virStorageSourcePtr ret;
|
|
|
|
|
|
|
|
if (VIR_ALLOC(ret) < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (virStorageIsFile(path)) {
|
|
|
|
ret->type = VIR_STORAGE_TYPE_FILE;
|
|
|
|
|
2014-05-27 12:49:41 +00:00
|
|
|
if (VIR_STRDUP(ret->path, path) < 0)
|
|
|
|
goto error;
|
2014-04-25 11:23:50 +00:00
|
|
|
} else {
|
|
|
|
ret->type = VIR_STORAGE_TYPE_NETWORK;
|
|
|
|
|
|
|
|
/* handle URI formatted backing stores */
|
|
|
|
if (strstr(path, "://")) {
|
|
|
|
if (virStorageSourceParseBackingURI(ret, path) < 0)
|
|
|
|
goto error;
|
|
|
|
} else {
|
|
|
|
if (virStorageSourceParseBackingColon(ret, path) < 0)
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
error:
|
|
|
|
virStorageSourceFree(ret);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
virStorageSourcePtr
|
|
|
|
virStorageSourceNewFromBacking(virStorageSourcePtr parent)
|
|
|
|
{
|
|
|
|
struct stat st;
|
|
|
|
virStorageSourcePtr ret;
|
|
|
|
|
|
|
|
if (virStorageIsRelative(parent->backingStoreRaw))
|
|
|
|
ret = virStorageSourceNewFromBackingRelative(parent,
|
|
|
|
parent->backingStoreRaw);
|
|
|
|
else
|
|
|
|
ret = virStorageSourceNewFromBackingAbsolute(parent->backingStoreRaw);
|
|
|
|
|
|
|
|
if (ret) {
|
|
|
|
/* possibly update local type */
|
|
|
|
if (ret->type == VIR_STORAGE_TYPE_FILE) {
|
|
|
|
if (stat(ret->path, &st) == 0) {
|
|
|
|
if (S_ISDIR(st.st_mode)) {
|
|
|
|
ret->type = VIR_STORAGE_TYPE_DIR;
|
|
|
|
ret->format = VIR_STORAGE_FILE_DIR;
|
|
|
|
} else if (S_ISBLK(st.st_mode)) {
|
|
|
|
ret->type = VIR_STORAGE_TYPE_BLOCK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-06-27 12:33:01 +00:00
|
|
|
|
|
|
|
/* copy parent's labelling and other top level stuff */
|
2014-11-19 17:54:43 +00:00
|
|
|
if (virStorageSourceInitChainElement(ret, parent, true) < 0)
|
2014-06-27 12:33:01 +00:00
|
|
|
goto error;
|
2014-04-25 11:23:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
2014-06-27 12:33:01 +00:00
|
|
|
|
|
|
|
error:
|
|
|
|
virStorageSourceFree(ret);
|
|
|
|
return NULL;
|
2014-04-25 11:23:50 +00:00
|
|
|
}
|
2014-05-02 17:22:17 +00:00
|
|
|
|
|
|
|
|
|
|
|
static char *
|
|
|
|
virStorageFileCanonicalizeFormatPath(char **components,
|
|
|
|
size_t ncomponents,
|
|
|
|
bool beginSlash,
|
|
|
|
bool beginDoubleSlash)
|
|
|
|
{
|
|
|
|
virBuffer buf = VIR_BUFFER_INITIALIZER;
|
|
|
|
size_t i;
|
|
|
|
char *ret = NULL;
|
|
|
|
|
|
|
|
if (beginSlash)
|
|
|
|
virBufferAddLit(&buf, "/");
|
|
|
|
|
|
|
|
if (beginDoubleSlash)
|
|
|
|
virBufferAddLit(&buf, "/");
|
|
|
|
|
|
|
|
for (i = 0; i < ncomponents; i++) {
|
|
|
|
if (i != 0)
|
|
|
|
virBufferAddLit(&buf, "/");
|
|
|
|
|
|
|
|
virBufferAdd(&buf, components[i], -1);
|
|
|
|
}
|
|
|
|
|
2014-06-27 08:40:15 +00:00
|
|
|
if (virBufferCheckError(&buf) < 0)
|
2014-05-02 17:22:17 +00:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
/* if the output string is empty just return an empty string */
|
|
|
|
if (!(ret = virBufferContentAndReset(&buf)))
|
|
|
|
ignore_value(VIR_STRDUP(ret, ""));
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
virStorageFileCanonicalizeInjectSymlink(const char *path,
|
|
|
|
size_t at,
|
|
|
|
char ***components,
|
|
|
|
size_t *ncomponents)
|
|
|
|
{
|
|
|
|
char **tmp = NULL;
|
|
|
|
char **next;
|
|
|
|
size_t ntmp = 0;
|
|
|
|
int ret = -1;
|
|
|
|
|
|
|
|
if (!(tmp = virStringSplitCount(path, "/", 0, &ntmp)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
/* prepend */
|
|
|
|
for (next = tmp; *next; next++) {
|
|
|
|
if (VIR_INSERT_ELEMENT(*components, at, *ncomponents, *next) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
at++;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virStringFreeListCount(tmp, ntmp);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
char *
|
|
|
|
virStorageFileCanonicalizePath(const char *path,
|
|
|
|
virStorageFileSimplifyPathReadlinkCallback cb,
|
|
|
|
void *cbdata)
|
|
|
|
{
|
|
|
|
virHashTablePtr cycle = NULL;
|
|
|
|
bool beginSlash = false;
|
|
|
|
bool beginDoubleSlash = false;
|
|
|
|
char **components = NULL;
|
|
|
|
size_t ncomponents = 0;
|
|
|
|
char *linkpath = NULL;
|
|
|
|
char *currentpath = NULL;
|
|
|
|
size_t i = 0;
|
|
|
|
size_t j = 0;
|
|
|
|
int rc;
|
|
|
|
char *ret = NULL;
|
|
|
|
|
|
|
|
if (path[0] == '/') {
|
|
|
|
beginSlash = true;
|
|
|
|
|
|
|
|
if (path[1] == '/' && path[2] != '/')
|
|
|
|
beginDoubleSlash = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(cycle = virHashCreate(10, NULL)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (!(components = virStringSplitCount(path, "/", 0, &ncomponents)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
j = 0;
|
|
|
|
while (j < ncomponents) {
|
|
|
|
/* skip slashes */
|
|
|
|
if (STREQ(components[j], "")) {
|
|
|
|
VIR_FREE(components[j]);
|
|
|
|
VIR_DELETE_ELEMENT(components, j, ncomponents);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
j++;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (i < ncomponents) {
|
|
|
|
/* skip '.'s unless it's the last one remaining */
|
|
|
|
if (STREQ(components[i], ".") &&
|
|
|
|
(beginSlash || ncomponents > 1)) {
|
|
|
|
VIR_FREE(components[i]);
|
|
|
|
VIR_DELETE_ELEMENT(components, i, ncomponents);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* resolve changes to parent directory */
|
|
|
|
if (STREQ(components[i], "..")) {
|
|
|
|
if (!beginSlash &&
|
|
|
|
(i == 0 || STREQ(components[i - 1], ".."))) {
|
|
|
|
i++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
VIR_FREE(components[i]);
|
|
|
|
VIR_DELETE_ELEMENT(components, i, ncomponents);
|
|
|
|
|
|
|
|
if (i != 0) {
|
|
|
|
VIR_FREE(components[i - 1]);
|
|
|
|
VIR_DELETE_ELEMENT(components, i - 1, ncomponents);
|
|
|
|
i--;
|
|
|
|
}
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* check if the actual path isn't resulting into a symlink */
|
|
|
|
if (!(currentpath = virStorageFileCanonicalizeFormatPath(components,
|
|
|
|
i + 1,
|
|
|
|
beginSlash,
|
|
|
|
beginDoubleSlash)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if ((rc = cb(currentpath, &linkpath, cbdata)) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (rc == 0) {
|
|
|
|
if (virHashLookup(cycle, currentpath)) {
|
|
|
|
virReportSystemError(ELOOP,
|
|
|
|
_("Failed to canonicalize path '%s'"), path);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virHashAddEntry(cycle, currentpath, (void *) 1) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
if (linkpath[0] == '/') {
|
|
|
|
/* kill everything from the beginning including the actual component */
|
|
|
|
i++;
|
|
|
|
while (i--) {
|
|
|
|
VIR_FREE(components[0]);
|
|
|
|
VIR_DELETE_ELEMENT(components, 0, ncomponents);
|
|
|
|
}
|
|
|
|
beginSlash = true;
|
|
|
|
|
|
|
|
if (linkpath[1] == '/' && linkpath[2] != '/')
|
|
|
|
beginDoubleSlash = true;
|
|
|
|
else
|
|
|
|
beginDoubleSlash = false;
|
|
|
|
|
|
|
|
i = 0;
|
|
|
|
} else {
|
|
|
|
VIR_FREE(components[i]);
|
|
|
|
VIR_DELETE_ELEMENT(components, i, ncomponents);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virStorageFileCanonicalizeInjectSymlink(linkpath,
|
|
|
|
i,
|
|
|
|
&components,
|
|
|
|
&ncomponents) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
j = 0;
|
|
|
|
while (j < ncomponents) {
|
|
|
|
/* skip slashes */
|
|
|
|
if (STREQ(components[j], "")) {
|
|
|
|
VIR_FREE(components[j]);
|
|
|
|
VIR_DELETE_ELEMENT(components, j, ncomponents);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
j++;
|
|
|
|
}
|
|
|
|
|
|
|
|
VIR_FREE(linkpath);
|
|
|
|
VIR_FREE(currentpath);
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
VIR_FREE(currentpath);
|
|
|
|
|
|
|
|
i++;
|
|
|
|
}
|
|
|
|
|
|
|
|
ret = virStorageFileCanonicalizeFormatPath(components, ncomponents,
|
|
|
|
beginSlash, beginDoubleSlash);
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
virHashFree(cycle);
|
|
|
|
virStringFreeListCount(components, ncomponents);
|
|
|
|
VIR_FREE(linkpath);
|
|
|
|
VIR_FREE(currentpath);
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2014-05-13 08:10:56 +00:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* virStorageFileRemoveLastPathComponent:
|
|
|
|
*
|
|
|
|
* @path: Path string to remove the last component from
|
|
|
|
*
|
|
|
|
* Removes the last path component of a path. This function is designed to be
|
|
|
|
* called on file paths only (no trailing slashes in @path). Caller is
|
|
|
|
* responsible to free the returned string.
|
|
|
|
*/
|
|
|
|
static char *
|
|
|
|
virStorageFileRemoveLastPathComponent(const char *path)
|
|
|
|
{
|
|
|
|
char *tmp;
|
|
|
|
char *ret;
|
|
|
|
|
|
|
|
if (VIR_STRDUP(ret, path ? path : "") < 0)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if ((tmp = strrchr(ret, '/')))
|
|
|
|
tmp[1] = '\0';
|
|
|
|
else
|
|
|
|
ret[0] = '\0';
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* virStorageFileGetRelativeBackingPath:
|
|
|
|
*
|
|
|
|
* Resolve relative path to be written to the overlay of @top image when
|
|
|
|
* collapsing the backing chain between @top and @base.
|
|
|
|
*
|
|
|
|
* Returns 0 on success; 1 if backing chain isn't relative and -1 on error.
|
|
|
|
*/
|
|
|
|
int
|
|
|
|
virStorageFileGetRelativeBackingPath(virStorageSourcePtr top,
|
|
|
|
virStorageSourcePtr base,
|
|
|
|
char **relpath)
|
|
|
|
{
|
|
|
|
virStorageSourcePtr next;
|
|
|
|
char *tmp = NULL;
|
|
|
|
char *path = NULL;
|
|
|
|
char ret = -1;
|
|
|
|
|
|
|
|
*relpath = NULL;
|
|
|
|
|
|
|
|
for (next = top; next; next = next->backingStore) {
|
2014-05-27 08:40:59 +00:00
|
|
|
if (!next->relPath) {
|
2014-05-13 08:10:56 +00:00
|
|
|
ret = 1;
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(tmp = virStorageFileRemoveLastPathComponent(path)))
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
VIR_FREE(path);
|
|
|
|
|
|
|
|
if (virAsprintf(&path, "%s%s", tmp, next->relPath) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
|
|
|
VIR_FREE(tmp);
|
|
|
|
|
|
|
|
if (next == base)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (next != base) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("failed to resolve relative backing name: "
|
|
|
|
"base image is not in backing chain"));
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
*relpath = path;
|
|
|
|
path = NULL;
|
|
|
|
|
|
|
|
ret = 0;
|
|
|
|
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(path);
|
|
|
|
VIR_FREE(tmp);
|
|
|
|
return ret;
|
|
|
|
}
|