2008-02-20 15:52:17 +00:00
|
|
|
/*
|
|
|
|
* storage_backend_disk.c: storage backend for disk handling
|
|
|
|
*
|
storage: Add new flag for libvirt_parthelper
https://bugzilla.redhat.com/show_bug.cgi?id=1265694
In order to be able to process disk storage pool's using a multipath
device to handle the partitions, libvirt_parthelper will need a way to
not automatically add a partition separator "p" to the generated device
name for each partition found. This is designed to mimic the multipath
features known as 'user_friendly_names' and custom 'alias' name.
If the part_separator attribute is set to "no", then generation of the
multipath partition name will not include the "p" partition separator
unless the source device path name ends with a number. The generated
partition names that get passed back to libvirt are processed in order
to find the device mapper multipath (dm-#) path device.
For example, device path "/dev/mapper/mpatha" would create partitions
"/dev/mapper/mpatha1", "/dev/mapper/mpatha2", etc. instead of
"/dev/mapper/mpathap1", "/dev/mapper/mpathap2", etc. If the device
path ends with a number "/dev/mapper/mpatha1", then the algorithm
to generate names "/dev/mapper/mpatha1p1", "/dev/mapper/mpatha1p2", etc.
would be utilized.
Signed-off-by: John Ferlan <jferlan@redhat.com>
2016-01-07 12:34:51 +00:00
|
|
|
* Copyright (C) 2007-2016 Red Hat, Inc.
|
2008-02-20 15:52:17 +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/>.
|
2008-02-20 15:52:17 +00:00
|
|
|
*
|
|
|
|
* Author: Daniel P. Berrange <berrange@redhat.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
2008-09-16 16:46:08 +00:00
|
|
|
#include <string.h>
|
2008-09-17 14:29:47 +00:00
|
|
|
#include <unistd.h>
|
2008-11-17 11:19:33 +00:00
|
|
|
#include <stdio.h>
|
2008-02-20 15:52:17 +00:00
|
|
|
|
2013-04-25 20:24:42 +00:00
|
|
|
#include "dirname.h"
|
2012-12-13 18:21:53 +00:00
|
|
|
#include "virerror.h"
|
2012-12-12 17:59:27 +00:00
|
|
|
#include "virlog.h"
|
2008-02-20 15:52:17 +00:00
|
|
|
#include "storage_backend_disk.h"
|
2017-01-11 17:04:15 +00:00
|
|
|
#include "storage_util.h"
|
2012-12-12 18:06:53 +00:00
|
|
|
#include "viralloc.h"
|
2012-12-12 16:27:01 +00:00
|
|
|
#include "vircommand.h"
|
2013-05-09 18:59:04 +00:00
|
|
|
#include "virfile.h"
|
2010-11-16 14:54:17 +00:00
|
|
|
#include "configmake.h"
|
2013-04-03 10:36:23 +00:00
|
|
|
#include "virstring.h"
|
2008-02-20 15:52:17 +00:00
|
|
|
|
2009-01-20 17:13:33 +00:00
|
|
|
#define VIR_FROM_THIS VIR_FROM_STORAGE
|
|
|
|
|
2014-02-28 12:16:17 +00:00
|
|
|
VIR_LOG_INIT("storage.storage_backend_disk");
|
|
|
|
|
2009-06-26 16:18:59 +00:00
|
|
|
#define SECTOR_SIZE 512
|
|
|
|
|
2017-05-09 12:18:33 +00:00
|
|
|
static bool
|
|
|
|
virStorageVolPartFindExtended(virStorageVolDefPtr def,
|
|
|
|
const void *opaque ATTRIBUTE_UNUSED)
|
|
|
|
{
|
|
|
|
if (def->source.partType == VIR_STORAGE_VOL_DISK_TYPE_EXTENDED)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-02-20 15:52:17 +00:00
|
|
|
static int
|
2010-02-10 11:42:56 +00:00
|
|
|
virStorageBackendDiskMakeDataVol(virStoragePoolObjPtr pool,
|
2008-02-20 15:52:17 +00:00
|
|
|
char **const groups,
|
|
|
|
virStorageVolDefPtr vol)
|
|
|
|
{
|
2015-01-16 00:12:42 +00:00
|
|
|
char *tmp, *devpath, *partname;
|
|
|
|
|
|
|
|
/* Prepended path will be same for all partitions, so we can
|
|
|
|
* strip the path to form a reasonable pool-unique name
|
|
|
|
*/
|
|
|
|
if ((tmp = strrchr(groups[0], '/')))
|
|
|
|
partname = tmp + 1;
|
|
|
|
else
|
|
|
|
partname = groups[0];
|
2008-02-20 15:52:17 +00:00
|
|
|
|
|
|
|
if (vol == NULL) {
|
2015-01-16 00:12:42 +00:00
|
|
|
/* This is typically a reload/restart/refresh path where
|
|
|
|
* we're discovering the existing partitions for the pool
|
|
|
|
*/
|
2013-07-04 10:16:29 +00:00
|
|
|
if (VIR_ALLOC(vol) < 0)
|
2008-02-20 15:52:17 +00:00
|
|
|
return -1;
|
2015-01-16 00:12:42 +00:00
|
|
|
if (VIR_STRDUP(vol->name, partname) < 0 ||
|
2017-05-09 12:05:16 +00:00
|
|
|
virStoragePoolObjAddVol(pool, vol) < 0) {
|
2014-03-07 08:33:31 +00:00
|
|
|
virStorageVolDefFree(vol);
|
2008-02-20 15:52:17 +00:00
|
|
|
return -1;
|
2014-03-07 08:33:31 +00:00
|
|
|
}
|
2008-02-20 15:52:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (vol->target.path == NULL) {
|
2013-05-03 12:49:08 +00:00
|
|
|
if (VIR_STRDUP(devpath, groups[0]) < 0)
|
2008-02-20 15:52:17 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
/* Now figure out the stable path
|
|
|
|
*
|
|
|
|
* XXX this method is O(N) because it scans the pool target
|
|
|
|
* dir every time its run. Should figure out a more efficient
|
|
|
|
* way of doing this...
|
|
|
|
*/
|
2012-10-21 16:53:20 +00:00
|
|
|
vol->target.path = virStorageBackendStablePath(pool, devpath, true);
|
2008-11-03 11:37:11 +00:00
|
|
|
VIR_FREE(devpath);
|
2009-11-10 11:56:11 +00:00
|
|
|
if (vol->target.path == NULL)
|
|
|
|
return -1;
|
2008-02-20 15:52:17 +00:00
|
|
|
}
|
|
|
|
|
2015-01-16 00:12:42 +00:00
|
|
|
/* Enforce provided vol->name is the same as what parted created.
|
|
|
|
* We do this after filling target.path so that we have a chance at
|
|
|
|
* deleting the partition with this failure from CreateVol path
|
|
|
|
*/
|
|
|
|
if (STRNEQ(vol->name, partname)) {
|
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("invalid partition name '%s', expected '%s'"),
|
|
|
|
vol->name, partname);
|
2016-11-15 19:37:01 +00:00
|
|
|
|
|
|
|
/* Let's see if by chance parthelper created a name that won't be
|
|
|
|
* found later when we try to delete. We tell parthelper to add a 'p'
|
|
|
|
* to the output via the part_separator flag, but if devmapper has
|
|
|
|
* user_friendly_names set, the creation won't happen that way, thus
|
|
|
|
* our deletion will fail because the name we generated is wrong.
|
|
|
|
* Check for our conditions and see if the generated name is the
|
|
|
|
* same as StablePath returns and has the 'p' in it */
|
|
|
|
if (pool->def->source.devices[0].part_separator ==
|
|
|
|
VIR_TRISTATE_BOOL_YES &&
|
|
|
|
!virIsDevMapperDevice(vol->target.path) &&
|
|
|
|
STREQ(groups[0], vol->target.path) &&
|
|
|
|
(tmp = strrchr(groups[0], 'p'))) {
|
|
|
|
|
|
|
|
/* If we remove the 'p' from groups[0] and the resulting
|
|
|
|
* device is a devmapper device, then we know parthelper
|
|
|
|
* was told to create the wrong name based on the results.
|
|
|
|
* So just remove the 'p' from the vol->target.path too. */
|
|
|
|
memmove(tmp, tmp + 1, strlen(tmp));
|
|
|
|
if (virIsDevMapperDevice(groups[0]) &&
|
|
|
|
(tmp = strrchr(vol->target.path, 'p')))
|
|
|
|
memmove(tmp, tmp + 1, strlen(tmp));
|
|
|
|
}
|
2015-01-16 00:12:42 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2008-02-20 15:52:17 +00:00
|
|
|
if (vol->key == NULL) {
|
|
|
|
/* XXX base off a unique key of the underlying disk */
|
2013-05-03 12:49:08 +00:00
|
|
|
if (VIR_STRDUP(vol->key, vol->target.path) < 0)
|
2008-02-20 15:52:17 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vol->source.extents == NULL) {
|
2013-07-04 10:16:29 +00:00
|
|
|
if (VIR_ALLOC(vol->source.extents) < 0)
|
2008-02-20 15:52:17 +00:00
|
|
|
return -1;
|
|
|
|
vol->source.nextent = 1;
|
|
|
|
|
|
|
|
if (virStrToLong_ull(groups[3], NULL, 10,
|
|
|
|
&vol->source.extents[0].start) < 0) {
|
2012-07-18 11:38:29 +00:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("cannot parse device start location"));
|
2008-02-20 15:52:17 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (virStrToLong_ull(groups[4], NULL, 10,
|
|
|
|
&vol->source.extents[0].end) < 0) {
|
2012-07-18 11:38:29 +00:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("cannot parse device end location"));
|
2008-02-20 15:52:17 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-05-03 12:49:08 +00:00
|
|
|
if (VIR_STRDUP(vol->source.extents[0].path,
|
|
|
|
pool->def->source.devices[0].path) < 0)
|
2008-02-20 15:52:17 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2009-06-26 16:18:59 +00:00
|
|
|
/* set partition type */
|
2010-08-18 20:54:48 +00:00
|
|
|
if (STREQ(groups[1], "normal"))
|
2014-03-30 02:27:44 +00:00
|
|
|
vol->source.partType = VIR_STORAGE_VOL_DISK_TYPE_PRIMARY;
|
2010-08-18 20:54:48 +00:00
|
|
|
else if (STREQ(groups[1], "logical"))
|
2014-03-30 02:27:44 +00:00
|
|
|
vol->source.partType = VIR_STORAGE_VOL_DISK_TYPE_LOGICAL;
|
2010-08-18 20:54:48 +00:00
|
|
|
else if (STREQ(groups[1], "extended"))
|
2014-03-30 02:27:44 +00:00
|
|
|
vol->source.partType = VIR_STORAGE_VOL_DISK_TYPE_EXTENDED;
|
2009-06-26 16:18:59 +00:00
|
|
|
else
|
2014-03-30 02:27:44 +00:00
|
|
|
vol->source.partType = VIR_STORAGE_VOL_DISK_TYPE_NONE;
|
2009-06-26 16:18:59 +00:00
|
|
|
|
2008-11-17 11:19:33 +00:00
|
|
|
vol->type = VIR_STORAGE_VOL_BLOCK;
|
|
|
|
|
2015-01-22 14:12:04 +00:00
|
|
|
/* Refresh allocation/capacity/perms
|
|
|
|
*
|
|
|
|
* For an extended partition, virStorageBackendUpdateVolInfo will
|
|
|
|
* return incorrect values for allocation and capacity, so use the
|
|
|
|
* extent information captured above instead.
|
|
|
|
*
|
|
|
|
* Also once a logical partition exists or another primary partition
|
|
|
|
* after an extended partition is created an open on the extended
|
|
|
|
* partition will fail, so pass the NOERROR flag and only error if a
|
|
|
|
* -1 was returned indicating some other error than an open error.
|
|
|
|
*/
|
|
|
|
if (vol->source.partType == VIR_STORAGE_VOL_DISK_TYPE_EXTENDED) {
|
2015-02-19 12:43:03 +00:00
|
|
|
if (virStorageBackendUpdateVolInfo(vol, false,
|
2015-01-22 14:12:04 +00:00
|
|
|
VIR_STORAGE_VOL_OPEN_DEFAULT |
|
2015-11-24 15:08:29 +00:00
|
|
|
VIR_STORAGE_VOL_OPEN_NOERROR,
|
|
|
|
0) == -1)
|
2015-01-22 14:12:04 +00:00
|
|
|
return -1;
|
2015-10-01 15:37:17 +00:00
|
|
|
vol->target.allocation = 0;
|
|
|
|
vol->target.capacity =
|
2015-01-22 14:12:04 +00:00
|
|
|
(vol->source.extents[0].end - vol->source.extents[0].start);
|
|
|
|
} else {
|
2015-02-19 12:43:03 +00:00
|
|
|
if (virStorageBackendUpdateVolInfo(vol, false,
|
2015-11-24 15:08:29 +00:00
|
|
|
VIR_STORAGE_VOL_OPEN_DEFAULT, 0) < 0)
|
2015-01-22 14:12:04 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2008-02-20 15:52:17 +00:00
|
|
|
|
2015-10-01 15:37:17 +00:00
|
|
|
/* Find the extended partition and increase the allocation value */
|
|
|
|
if (vol->source.partType == VIR_STORAGE_VOL_DISK_TYPE_LOGICAL) {
|
2017-05-09 12:18:33 +00:00
|
|
|
virStorageVolDefPtr voldef;
|
2015-10-01 15:37:17 +00:00
|
|
|
|
2017-05-09 12:18:33 +00:00
|
|
|
voldef = virStoragePoolObjSearchVolume(pool,
|
|
|
|
virStorageVolPartFindExtended,
|
|
|
|
NULL);
|
|
|
|
if (voldef)
|
|
|
|
voldef->target.allocation += vol->target.allocation;
|
2015-10-01 15:37:17 +00:00
|
|
|
}
|
|
|
|
|
2008-02-20 15:52:17 +00:00
|
|
|
if (STRNEQ(groups[2], "metadata"))
|
conf: track sizes directly in source struct
One of the features of qcow2 is that a wrapper file can have
more capacity than its backing file from the guest's perspective;
what's more, sparse files make tracking allocation of both
the active and backing file worthwhile. As such, it makes
more sense to show allocation numbers for each file in a chain,
and not just the top-level file. This sets up the fields for
the tracking, although it does not modify XML to display any
new information.
* src/util/virstoragefile.h (_virStorageSource): Add fields.
* src/conf/storage_conf.h (_virStorageVolDef): Drop redundant
fields.
* src/storage/storage_backend.c (virStorageBackendCreateBlockFrom)
(createRawFile, virStorageBackendCreateQemuImgCmd)
(virStorageBackendCreateQcowCreate): Update clients.
* src/storage/storage_driver.c (storageVolDelete)
(storageVolCreateXML, storageVolCreateXMLFrom, storageVolResize)
(storageVolWipeInternal, storageVolGetInfo): Likewise.
* src/storage/storage_backend_fs.c (virStorageBackendProbeTarget)
(virStorageBackendFileSystemRefresh)
(virStorageBackendFileSystemVolResize)
(virStorageBackendFileSystemVolRefresh): Likewise.
* src/storage/storage_backend_logical.c
(virStorageBackendLogicalMakeVol)
(virStorageBackendLogicalCreateVol): Likewise.
* src/storage/storage_backend_scsi.c
(virStorageBackendSCSINewLun): Likewise.
* src/storage/storage_backend_mpath.c
(virStorageBackendMpathNewVol): Likewise.
* src/storage/storage_backend_rbd.c
(volStorageBackendRBDRefreshVolInfo)
(virStorageBackendRBDCreateImage): Likewise.
* src/storage/storage_backend_disk.c
(virStorageBackendDiskMakeDataVol)
(virStorageBackendDiskCreateVol): Likewise.
* src/storage/storage_backend_sheepdog.c
(virStorageBackendSheepdogBuildVol)
(virStorageBackendSheepdogParseVdiList): Likewise.
* src/storage/storage_backend_gluster.c
(virStorageBackendGlusterRefreshVol): Likewise.
* src/conf/storage_conf.c (virStorageVolDefFormat)
(virStorageVolDefParseXML): Likewise.
* src/test/test_driver.c (testOpenVolumesForPool)
(testStorageVolCreateXML, testStorageVolCreateXMLFrom)
(testStorageVolDelete, testStorageVolGetInfo): Likewise.
* src/esx/esx_storage_backend_iscsi.c (esxStorageVolGetXMLDesc):
Likewise.
* src/esx/esx_storage_backend_vmfs.c (esxStorageVolGetXMLDesc)
(esxStorageVolCreateXML): Likewise.
* src/parallels/parallels_driver.c (parallelsAddHddByVolume):
Likewise.
* src/parallels/parallels_storage.c (parallelsDiskDescParseNode)
(parallelsStorageVolDefineXML, parallelsStorageVolCreateXMLFrom)
(parallelsStorageVolDefRemove, parallelsStorageVolGetInfo):
Likewise.
* src/vbox/vbox_tmpl.c (vboxStorageVolCreateXML)
(vboxStorageVolGetXMLDesc): Likewise.
* tests/storagebackendsheepdogtest.c (test_vdi_list_parser):
Likewise.
* src/phyp/phyp_driver.c (phypStorageVolCreateXML): Likewise.
2014-04-01 23:43:36 +00:00
|
|
|
pool->def->allocation += vol->target.allocation;
|
2008-02-20 15:52:17 +00:00
|
|
|
if (vol->source.extents[0].end > pool->def->capacity)
|
|
|
|
pool->def->capacity = vol->source.extents[0].end;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2010-02-10 11:42:56 +00:00
|
|
|
virStorageBackendDiskMakeFreeExtent(virStoragePoolObjPtr pool,
|
2008-02-20 15:52:17 +00:00
|
|
|
char **const groups)
|
|
|
|
{
|
|
|
|
virStoragePoolSourceDevicePtr dev = &pool->def->source.devices[0];
|
|
|
|
|
2008-06-06 11:09:57 +00:00
|
|
|
if (VIR_REALLOC_N(dev->freeExtents,
|
|
|
|
dev->nfreeExtent + 1) < 0)
|
2008-02-20 15:52:17 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
memset(dev->freeExtents +
|
2008-06-06 11:09:57 +00:00
|
|
|
dev->nfreeExtent, 0,
|
|
|
|
sizeof(dev->freeExtents[0]));
|
2008-02-20 15:52:17 +00:00
|
|
|
|
2009-06-26 16:18:59 +00:00
|
|
|
/* set type of free area */
|
2010-08-18 20:54:48 +00:00
|
|
|
if (STREQ(groups[1], "logical")) {
|
2009-06-26 16:18:59 +00:00
|
|
|
dev->freeExtents[dev->nfreeExtent].type = VIR_STORAGE_FREE_LOGICAL;
|
|
|
|
} else {
|
|
|
|
dev->freeExtents[dev->nfreeExtent].type = VIR_STORAGE_FREE_NORMAL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-02-20 15:52:17 +00:00
|
|
|
if (virStrToLong_ull(groups[3], NULL, 10,
|
|
|
|
&dev->freeExtents[dev->nfreeExtent].start) < 0)
|
|
|
|
return -1; /* Don't bother to re-alloc freeExtents - it'll be free'd shortly */
|
|
|
|
|
|
|
|
if (virStrToLong_ull(groups[4], NULL, 10,
|
|
|
|
&dev->freeExtents[dev->nfreeExtent].end) < 0)
|
|
|
|
return -1; /* Don't bother to re-alloc freeExtents - it'll be free'd shortly */
|
|
|
|
|
2009-06-26 16:18:59 +00:00
|
|
|
/* first block reported as free, even if it is not */
|
2014-11-13 14:25:27 +00:00
|
|
|
if (dev->freeExtents[dev->nfreeExtent].start == 0)
|
2009-06-26 16:18:59 +00:00
|
|
|
dev->freeExtents[dev->nfreeExtent].start = SECTOR_SIZE;
|
|
|
|
|
2008-02-20 15:52:17 +00:00
|
|
|
pool->def->available +=
|
|
|
|
(dev->freeExtents[dev->nfreeExtent].end -
|
|
|
|
dev->freeExtents[dev->nfreeExtent].start);
|
|
|
|
if (dev->freeExtents[dev->nfreeExtent].end > pool->def->capacity)
|
|
|
|
pool->def->capacity = dev->freeExtents[dev->nfreeExtent].end;
|
|
|
|
|
|
|
|
dev->nfreeExtent++;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-03-18 14:35:01 +00:00
|
|
|
struct virStorageBackendDiskPoolVolData {
|
|
|
|
virStoragePoolObjPtr pool;
|
|
|
|
virStorageVolDefPtr vol;
|
|
|
|
};
|
|
|
|
|
2008-02-20 15:52:17 +00:00
|
|
|
static int
|
2014-03-18 14:35:01 +00:00
|
|
|
virStorageBackendDiskMakeVol(size_t ntok ATTRIBUTE_UNUSED,
|
2008-02-20 15:52:17 +00:00
|
|
|
char **const groups,
|
2014-03-18 14:35:01 +00:00
|
|
|
void *opaque)
|
2008-02-20 15:52:17 +00:00
|
|
|
{
|
2014-03-18 14:35:01 +00:00
|
|
|
struct virStorageBackendDiskPoolVolData *data = opaque;
|
|
|
|
virStoragePoolObjPtr pool = data->pool;
|
2008-02-20 15:52:17 +00:00
|
|
|
/*
|
|
|
|
* Ignore normal+metadata, and logical+metadata partitions
|
|
|
|
* since they're basically internal book-keeping regions
|
|
|
|
* we have no control over. Do keep extended+metadata though
|
|
|
|
* because that's the MS-DOS extended partition region we
|
|
|
|
* need to be able to view/create/delete
|
|
|
|
*/
|
|
|
|
if ((STREQ(groups[1], "normal") ||
|
|
|
|
STREQ(groups[1], "logical")) &&
|
|
|
|
STREQ(groups[2], "metadata"))
|
|
|
|
return 0;
|
|
|
|
|
2008-02-27 10:37:19 +00:00
|
|
|
/* Remaining data / metadata parts get turn into volumes... */
|
2008-02-20 15:52:17 +00:00
|
|
|
if (STREQ(groups[2], "metadata") ||
|
|
|
|
STREQ(groups[2], "data")) {
|
2014-03-18 14:35:01 +00:00
|
|
|
virStorageVolDefPtr vol = data->vol;
|
2009-07-10 01:51:36 +00:00
|
|
|
|
|
|
|
if (vol) {
|
|
|
|
/* We're searching for a specific vol only */
|
|
|
|
if (vol->key) {
|
|
|
|
if (STRNEQ(vol->key, groups[0]))
|
|
|
|
return 0;
|
|
|
|
} else if (virStorageVolDefFindByKey(pool, groups[0]) != NULL) {
|
|
|
|
/* If no key, the volume must be newly created. If groups[0]
|
|
|
|
* isn't already a volume, assume it's the path we want */
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
2008-02-20 15:52:17 +00:00
|
|
|
|
2010-02-10 11:42:56 +00:00
|
|
|
return virStorageBackendDiskMakeDataVol(pool, groups, vol);
|
2008-02-20 15:52:17 +00:00
|
|
|
} else if (STREQ(groups[2], "free")) {
|
|
|
|
/* ....or free space extents */
|
2010-02-10 11:42:56 +00:00
|
|
|
return virStorageBackendDiskMakeFreeExtent(pool, groups);
|
2008-02-20 15:52:17 +00:00
|
|
|
} else {
|
2008-02-27 10:37:19 +00:00
|
|
|
/* This code path should never happen unless someone changed
|
2008-02-20 15:52:17 +00:00
|
|
|
* libvirt_parthelper forgot to change this code */
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* To get a list of partitions we run an external helper
|
|
|
|
* tool which then uses parted APIs. This is because
|
|
|
|
* parted's API is not compatible with libvirt's license
|
|
|
|
* but we really really want to use parted because the
|
|
|
|
* other options all suck :-)
|
|
|
|
*
|
|
|
|
* All the other storage backends run an external tool for
|
|
|
|
* listing volumes so this really isn't too much of a pain,
|
|
|
|
* and we can even ensure the output is friendly.
|
|
|
|
*/
|
|
|
|
static int
|
2010-02-10 11:42:56 +00:00
|
|
|
virStorageBackendDiskReadPartitions(virStoragePoolObjPtr pool,
|
2008-02-20 15:52:17 +00:00
|
|
|
virStorageVolDefPtr vol)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* # libvirt_parthelper DEVICE
|
|
|
|
* /dev/sda1 normal data 32256 106928128 106896384
|
|
|
|
* /dev/sda2 normal data 106928640 100027629568 99920701440
|
|
|
|
* - normal metadata 100027630080 100030242304 2612736
|
|
|
|
*
|
|
|
|
*/
|
2014-03-25 08:23:14 +00:00
|
|
|
|
|
|
|
char *parthelper_path;
|
|
|
|
virCommandPtr cmd;
|
2014-03-18 14:35:01 +00:00
|
|
|
struct virStorageBackendDiskPoolVolData cbdata = {
|
|
|
|
.pool = pool,
|
|
|
|
.vol = vol,
|
|
|
|
};
|
2012-07-11 14:56:55 +00:00
|
|
|
int ret;
|
2008-02-20 15:52:17 +00:00
|
|
|
|
2014-03-25 08:23:14 +00:00
|
|
|
if (!(parthelper_path = virFileFindResource("libvirt_parthelper",
|
2015-02-13 13:25:27 +00:00
|
|
|
abs_topbuilddir "/src",
|
2014-03-25 08:23:14 +00:00
|
|
|
LIBEXECDIR)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
cmd = virCommandNewArgList(parthelper_path,
|
|
|
|
pool->def->source.devices[0].path,
|
|
|
|
NULL);
|
|
|
|
|
2016-05-09 18:57:17 +00:00
|
|
|
/* Check for the presence of the part_separator='yes'. Pass this
|
storage: Add new flag for libvirt_parthelper
https://bugzilla.redhat.com/show_bug.cgi?id=1265694
In order to be able to process disk storage pool's using a multipath
device to handle the partitions, libvirt_parthelper will need a way to
not automatically add a partition separator "p" to the generated device
name for each partition found. This is designed to mimic the multipath
features known as 'user_friendly_names' and custom 'alias' name.
If the part_separator attribute is set to "no", then generation of the
multipath partition name will not include the "p" partition separator
unless the source device path name ends with a number. The generated
partition names that get passed back to libvirt are processed in order
to find the device mapper multipath (dm-#) path device.
For example, device path "/dev/mapper/mpatha" would create partitions
"/dev/mapper/mpatha1", "/dev/mapper/mpatha2", etc. instead of
"/dev/mapper/mpathap1", "/dev/mapper/mpathap2", etc. If the device
path ends with a number "/dev/mapper/mpatha1", then the algorithm
to generate names "/dev/mapper/mpatha1p1", "/dev/mapper/mpatha1p2", etc.
would be utilized.
Signed-off-by: John Ferlan <jferlan@redhat.com>
2016-01-07 12:34:51 +00:00
|
|
|
* along to the libvirt_parthelper as option '-p'. This will cause
|
2016-05-09 18:57:17 +00:00
|
|
|
* libvirt_parthelper to append the "p" partition separator to
|
|
|
|
* the generated device name for a source device which ends with
|
|
|
|
* a non-numeric value (e.g. mpatha would generate mpathap#).
|
storage: Add new flag for libvirt_parthelper
https://bugzilla.redhat.com/show_bug.cgi?id=1265694
In order to be able to process disk storage pool's using a multipath
device to handle the partitions, libvirt_parthelper will need a way to
not automatically add a partition separator "p" to the generated device
name for each partition found. This is designed to mimic the multipath
features known as 'user_friendly_names' and custom 'alias' name.
If the part_separator attribute is set to "no", then generation of the
multipath partition name will not include the "p" partition separator
unless the source device path name ends with a number. The generated
partition names that get passed back to libvirt are processed in order
to find the device mapper multipath (dm-#) path device.
For example, device path "/dev/mapper/mpatha" would create partitions
"/dev/mapper/mpatha1", "/dev/mapper/mpatha2", etc. instead of
"/dev/mapper/mpathap1", "/dev/mapper/mpathap2", etc. If the device
path ends with a number "/dev/mapper/mpatha1", then the algorithm
to generate names "/dev/mapper/mpatha1p1", "/dev/mapper/mpatha1p2", etc.
would be utilized.
Signed-off-by: John Ferlan <jferlan@redhat.com>
2016-01-07 12:34:51 +00:00
|
|
|
*/
|
|
|
|
if (pool->def->source.devices[0].part_separator ==
|
2016-05-09 18:57:17 +00:00
|
|
|
VIR_TRISTATE_BOOL_YES)
|
storage: Add new flag for libvirt_parthelper
https://bugzilla.redhat.com/show_bug.cgi?id=1265694
In order to be able to process disk storage pool's using a multipath
device to handle the partitions, libvirt_parthelper will need a way to
not automatically add a partition separator "p" to the generated device
name for each partition found. This is designed to mimic the multipath
features known as 'user_friendly_names' and custom 'alias' name.
If the part_separator attribute is set to "no", then generation of the
multipath partition name will not include the "p" partition separator
unless the source device path name ends with a number. The generated
partition names that get passed back to libvirt are processed in order
to find the device mapper multipath (dm-#) path device.
For example, device path "/dev/mapper/mpatha" would create partitions
"/dev/mapper/mpatha1", "/dev/mapper/mpatha2", etc. instead of
"/dev/mapper/mpathap1", "/dev/mapper/mpathap2", etc. If the device
path ends with a number "/dev/mapper/mpatha1", then the algorithm
to generate names "/dev/mapper/mpatha1p1", "/dev/mapper/mpatha1p2", etc.
would be utilized.
Signed-off-by: John Ferlan <jferlan@redhat.com>
2016-01-07 12:34:51 +00:00
|
|
|
virCommandAddArg(cmd, "-p");
|
|
|
|
|
2015-05-22 01:10:56 +00:00
|
|
|
/* If a volume is passed, virStorageBackendDiskMakeVol only updates the
|
|
|
|
* pool allocation for that single volume.
|
|
|
|
*/
|
|
|
|
if (!vol)
|
|
|
|
pool->def->allocation = 0;
|
|
|
|
pool->def->capacity = pool->def->available = 0;
|
2008-02-20 15:52:17 +00:00
|
|
|
|
2014-03-18 14:35:01 +00:00
|
|
|
ret = virCommandRunNul(cmd,
|
|
|
|
6,
|
|
|
|
virStorageBackendDiskMakeVol,
|
|
|
|
&cbdata);
|
2012-07-11 14:56:55 +00:00
|
|
|
virCommandFree(cmd);
|
2014-03-25 08:23:14 +00:00
|
|
|
VIR_FREE(parthelper_path);
|
2012-07-11 14:56:55 +00:00
|
|
|
return ret;
|
2008-02-20 15:52:17 +00:00
|
|
|
}
|
|
|
|
|
2009-06-26 16:18:59 +00:00
|
|
|
static int
|
2014-03-18 14:35:01 +00:00
|
|
|
virStorageBackendDiskMakePoolGeometry(size_t ntok ATTRIBUTE_UNUSED,
|
2010-02-10 11:42:56 +00:00
|
|
|
char **const groups,
|
2014-03-18 14:35:01 +00:00
|
|
|
void *data)
|
2009-06-26 16:18:59 +00:00
|
|
|
{
|
2014-03-18 14:35:01 +00:00
|
|
|
virStoragePoolObjPtr pool = data;
|
2013-11-14 16:14:26 +00:00
|
|
|
virStoragePoolSourceDevicePtr device = &(pool->def->source.devices[0]);
|
|
|
|
if (virStrToLong_i(groups[0], NULL, 0, &device->geometry.cylinders) < 0 ||
|
|
|
|
virStrToLong_i(groups[1], NULL, 0, &device->geometry.heads) < 0 ||
|
|
|
|
virStrToLong_i(groups[2], NULL, 0, &device->geometry.sectors) < 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("Failed to create disk pool geometry"));
|
|
|
|
return -1;
|
|
|
|
}
|
2009-06-26 16:18:59 +00:00
|
|
|
|
2013-11-14 16:14:26 +00:00
|
|
|
return 0;
|
2009-06-26 16:18:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2010-02-10 11:42:56 +00:00
|
|
|
virStorageBackendDiskReadGeometry(virStoragePoolObjPtr pool)
|
2009-06-26 16:18:59 +00:00
|
|
|
{
|
2014-03-25 08:23:14 +00:00
|
|
|
char *parthelper_path;
|
|
|
|
virCommandPtr cmd;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
if (!(parthelper_path = virFileFindResource("libvirt_parthelper",
|
2015-02-13 13:25:27 +00:00
|
|
|
abs_topbuilddir "/src",
|
2014-03-25 08:23:14 +00:00
|
|
|
LIBEXECDIR)))
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
cmd = virCommandNewArgList(parthelper_path,
|
2012-07-11 14:56:55 +00:00
|
|
|
pool->def->source.devices[0].path,
|
|
|
|
"-g",
|
|
|
|
NULL);
|
|
|
|
|
2014-03-18 14:35:01 +00:00
|
|
|
ret = virCommandRunNul(cmd,
|
|
|
|
3,
|
|
|
|
virStorageBackendDiskMakePoolGeometry,
|
|
|
|
pool);
|
2012-07-11 14:56:55 +00:00
|
|
|
virCommandFree(cmd);
|
2014-03-25 08:23:14 +00:00
|
|
|
VIR_FREE(parthelper_path);
|
2012-07-11 14:56:55 +00:00
|
|
|
return ret;
|
2009-06-26 16:18:59 +00:00
|
|
|
}
|
2008-02-20 15:52:17 +00:00
|
|
|
|
|
|
|
static int
|
2010-02-10 11:42:56 +00:00
|
|
|
virStorageBackendDiskRefreshPool(virConnectPtr conn ATTRIBUTE_UNUSED,
|
2008-02-20 15:52:17 +00:00
|
|
|
virStoragePoolObjPtr pool)
|
|
|
|
{
|
2008-06-06 11:09:57 +00:00
|
|
|
VIR_FREE(pool->def->source.devices[0].freeExtents);
|
2008-02-20 15:52:17 +00:00
|
|
|
pool->def->source.devices[0].nfreeExtent = 0;
|
|
|
|
|
2017-02-20 12:00:51 +00:00
|
|
|
virWaitForDevices();
|
2008-11-28 07:50:20 +00:00
|
|
|
|
2011-09-19 07:35:15 +00:00
|
|
|
if (!virFileExists(pool->def->source.devices[0].path)) {
|
2012-07-18 11:38:29 +00:00
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("device path '%s' doesn't exist"),
|
|
|
|
pool->def->source.devices[0].path);
|
2011-09-19 07:35:15 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2014-11-13 14:25:27 +00:00
|
|
|
if (virStorageBackendDiskReadGeometry(pool) != 0)
|
2009-06-26 16:18:59 +00:00
|
|
|
return -1;
|
|
|
|
|
2010-02-10 11:42:56 +00:00
|
|
|
return virStorageBackendDiskReadPartitions(pool, NULL);
|
2008-02-20 15:52:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-10-01 12:38:58 +00:00
|
|
|
static int
|
|
|
|
virStorageBackendDiskStartPool(virConnectPtr conn ATTRIBUTE_UNUSED,
|
|
|
|
virStoragePoolObjPtr pool)
|
|
|
|
{
|
2017-09-11 21:11:25 +00:00
|
|
|
const char *format;
|
2016-12-14 23:34:24 +00:00
|
|
|
const char *path = pool->def->source.devices[0].path;
|
|
|
|
|
2017-02-20 12:00:51 +00:00
|
|
|
virWaitForDevices();
|
2015-10-01 12:38:58 +00:00
|
|
|
|
2016-12-14 23:34:24 +00:00
|
|
|
if (!virFileExists(path)) {
|
2015-10-01 12:38:58 +00:00
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
2016-12-14 23:34:24 +00:00
|
|
|
_("device path '%s' doesn't exist"), path);
|
2015-10-01 12:38:58 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2017-09-11 21:11:25 +00:00
|
|
|
if (pool->def->source.format == VIR_STORAGE_POOL_DISK_UNKNOWN)
|
|
|
|
pool->def->source.format = VIR_STORAGE_POOL_DISK_DOS;
|
|
|
|
format = virStoragePoolFormatDiskTypeToString(pool->def->source.format);
|
2016-12-14 23:34:24 +00:00
|
|
|
if (!virStorageBackendDeviceIsEmpty(path, format, false))
|
2015-10-01 12:38:58 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-02-20 15:52:17 +00:00
|
|
|
/**
|
|
|
|
* Write a new partition table header
|
|
|
|
*/
|
|
|
|
static int
|
2010-02-04 22:41:52 +00:00
|
|
|
virStorageBackendDiskBuildPool(virConnectPtr conn ATTRIBUTE_UNUSED,
|
2008-02-20 15:52:17 +00:00
|
|
|
virStoragePoolObjPtr pool,
|
2011-07-06 22:51:23 +00:00
|
|
|
unsigned int flags)
|
2008-02-20 15:52:17 +00:00
|
|
|
{
|
2016-12-14 23:34:24 +00:00
|
|
|
int format = pool->def->source.format;
|
|
|
|
const char *fmt;
|
2011-11-12 12:31:52 +00:00
|
|
|
bool ok_to_mklabel = false;
|
|
|
|
int ret = -1;
|
2013-01-30 12:02:15 +00:00
|
|
|
virCommandPtr cmd = NULL;
|
2008-02-20 15:52:17 +00:00
|
|
|
|
2011-11-12 12:31:52 +00:00
|
|
|
virCheckFlags(VIR_STORAGE_POOL_BUILD_OVERWRITE |
|
|
|
|
VIR_STORAGE_POOL_BUILD_NO_OVERWRITE, ret);
|
2011-07-06 22:51:23 +00:00
|
|
|
|
2015-04-28 17:14:07 +00:00
|
|
|
VIR_EXCLUSIVE_FLAGS_GOTO(VIR_STORAGE_POOL_BUILD_OVERWRITE,
|
|
|
|
VIR_STORAGE_POOL_BUILD_NO_OVERWRITE,
|
|
|
|
error);
|
2008-02-20 15:52:17 +00:00
|
|
|
|
2016-12-14 23:34:24 +00:00
|
|
|
fmt = virStoragePoolFormatDiskTypeToString(format);
|
|
|
|
if (flags & VIR_STORAGE_POOL_BUILD_OVERWRITE) {
|
2011-11-12 12:31:52 +00:00
|
|
|
ok_to_mklabel = true;
|
2016-12-14 23:34:24 +00:00
|
|
|
} else {
|
|
|
|
if (virStorageBackendDeviceIsEmpty(pool->def->source.devices[0].path,
|
2017-04-07 13:36:05 +00:00
|
|
|
fmt, true))
|
2016-12-14 23:34:24 +00:00
|
|
|
ok_to_mklabel = true;
|
|
|
|
}
|
2011-11-12 12:31:52 +00:00
|
|
|
|
2013-01-30 12:02:15 +00:00
|
|
|
if (ok_to_mklabel) {
|
2017-04-07 13:36:05 +00:00
|
|
|
if (virStorageBackendZeroPartitionTable(pool->def->source.devices[0].path,
|
|
|
|
1024 * 1024) < 0)
|
|
|
|
goto error;
|
|
|
|
|
2015-02-27 00:39:36 +00:00
|
|
|
/* eg parted /dev/sda mklabel --script msdos */
|
2015-06-08 12:16:58 +00:00
|
|
|
if (format == VIR_STORAGE_POOL_DISK_UNKNOWN)
|
|
|
|
format = pool->def->source.format = VIR_STORAGE_POOL_DISK_DOS;
|
|
|
|
if (format == VIR_STORAGE_POOL_DISK_DOS)
|
2015-02-27 00:39:36 +00:00
|
|
|
fmt = "msdos";
|
|
|
|
else
|
|
|
|
fmt = virStoragePoolFormatDiskTypeToString(format);
|
|
|
|
|
2013-01-30 12:02:15 +00:00
|
|
|
cmd = virCommandNewArgList(PARTED,
|
|
|
|
pool->def->source.devices[0].path,
|
|
|
|
"mklabel",
|
|
|
|
"--script",
|
2015-02-27 00:39:36 +00:00
|
|
|
fmt,
|
2013-01-30 12:02:15 +00:00
|
|
|
NULL);
|
2012-07-11 14:56:55 +00:00
|
|
|
ret = virCommandRun(cmd, NULL);
|
2013-01-30 12:02:15 +00:00
|
|
|
}
|
2011-11-12 12:31:52 +00:00
|
|
|
|
2014-03-25 06:52:40 +00:00
|
|
|
error:
|
2012-07-11 14:56:55 +00:00
|
|
|
virCommandFree(cmd);
|
2011-11-12 12:31:52 +00:00
|
|
|
return ret;
|
2008-02-20 15:52:17 +00:00
|
|
|
}
|
|
|
|
|
2017-05-09 12:18:33 +00:00
|
|
|
|
|
|
|
struct virStorageVolNumData {
|
|
|
|
int count;
|
|
|
|
};
|
|
|
|
|
|
|
|
static int
|
|
|
|
virStorageVolNumOfPartTypes(virStorageVolDefPtr def,
|
|
|
|
const void *opaque)
|
|
|
|
{
|
|
|
|
struct virStorageVolNumData *data = (struct virStorageVolNumData *)opaque;
|
|
|
|
|
|
|
|
if (def->source.partType == VIR_STORAGE_VOL_DISK_TYPE_PRIMARY ||
|
|
|
|
def->source.partType == VIR_STORAGE_VOL_DISK_TYPE_EXTENDED)
|
|
|
|
data->count++;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-06-26 16:18:59 +00:00
|
|
|
/**
|
|
|
|
* Decides what kind of partition type that should be created.
|
|
|
|
* Important when the partition table is of msdos type
|
|
|
|
*/
|
2008-02-20 15:52:17 +00:00
|
|
|
static int
|
2009-06-26 16:18:59 +00:00
|
|
|
virStorageBackendDiskPartTypeToCreate(virStoragePoolObjPtr pool)
|
|
|
|
{
|
2017-05-09 12:18:33 +00:00
|
|
|
struct virStorageVolNumData data = { .count = 0 };
|
|
|
|
|
2009-06-26 16:18:59 +00:00
|
|
|
if (pool->def->source.format == VIR_STORAGE_POOL_DISK_DOS) {
|
2014-04-01 22:26:59 +00:00
|
|
|
/* count primary and extended partitions,
|
2009-06-26 16:18:59 +00:00
|
|
|
can't be more than 3 to create a new primary partition */
|
2017-05-09 12:18:33 +00:00
|
|
|
if (virStoragePoolObjForEachVolume(pool, virStorageVolNumOfPartTypes,
|
|
|
|
&data) == 0) {
|
|
|
|
if (data.count >= 4)
|
|
|
|
return VIR_STORAGE_VOL_DISK_TYPE_LOGICAL;
|
2009-06-26 16:18:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* for all other cases, all partitions are primary */
|
|
|
|
return VIR_STORAGE_VOL_DISK_TYPE_PRIMARY;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2010-02-10 11:42:56 +00:00
|
|
|
virStorageBackendDiskPartFormat(virStoragePoolObjPtr pool,
|
2009-06-26 16:18:59 +00:00
|
|
|
virStorageVolDefPtr vol,
|
2010-08-18 21:54:11 +00:00
|
|
|
char** partFormat)
|
2009-06-26 16:18:59 +00:00
|
|
|
{
|
|
|
|
if (pool->def->source.format == VIR_STORAGE_POOL_DISK_DOS) {
|
2010-08-18 20:54:48 +00:00
|
|
|
const char *partedFormat;
|
2014-05-14 19:48:15 +00:00
|
|
|
partedFormat = virStoragePartedFsTypeToString(vol->target.format);
|
2010-08-18 20:54:48 +00:00
|
|
|
if (partedFormat == NULL) {
|
2012-07-18 11:38:29 +00:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("Invalid partition type"));
|
2010-08-18 20:54:48 +00:00
|
|
|
return -1;
|
2009-06-26 16:18:59 +00:00
|
|
|
}
|
|
|
|
if (vol->target.format == VIR_STORAGE_VOL_DISK_EXTENDED) {
|
2016-12-01 20:50:08 +00:00
|
|
|
/* make sure we don't have an extended partition already */
|
2017-05-09 12:18:33 +00:00
|
|
|
if (virStoragePoolObjSearchVolume(pool,
|
|
|
|
virStorageVolPartFindExtended,
|
|
|
|
NULL)) {
|
2012-07-18 11:38:29 +00:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("extended partition already exists"));
|
2010-08-18 20:54:48 +00:00
|
|
|
return -1;
|
2009-06-26 16:18:59 +00:00
|
|
|
}
|
2013-05-03 12:49:08 +00:00
|
|
|
if (VIR_STRDUP(*partFormat, partedFormat) < 0)
|
2010-08-18 21:54:11 +00:00
|
|
|
return -1;
|
2009-06-26 16:18:59 +00:00
|
|
|
} else {
|
|
|
|
/* create primary partition as long as it is possible
|
|
|
|
and after that check if an extended partition exists
|
|
|
|
to create logical partitions. */
|
|
|
|
/* XXX Only support one extended partition */
|
|
|
|
switch (virStorageBackendDiskPartTypeToCreate(pool)) {
|
2010-08-18 20:54:48 +00:00
|
|
|
case VIR_STORAGE_VOL_DISK_TYPE_PRIMARY:
|
2013-07-04 10:16:29 +00:00
|
|
|
if (virAsprintf(partFormat, "primary %s", partedFormat) < 0)
|
2010-08-18 21:54:11 +00:00
|
|
|
return -1;
|
2010-08-18 20:54:48 +00:00
|
|
|
break;
|
|
|
|
case VIR_STORAGE_VOL_DISK_TYPE_LOGICAL:
|
2016-12-01 20:50:08 +00:00
|
|
|
/* make sure we have an extended partition */
|
2017-05-09 12:18:33 +00:00
|
|
|
if (virStoragePoolObjSearchVolume(pool,
|
|
|
|
virStorageVolPartFindExtended,
|
|
|
|
NULL)) {
|
|
|
|
if (virAsprintf(partFormat, "logical %s",
|
|
|
|
partedFormat) < 0)
|
|
|
|
return -1;
|
|
|
|
} else {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
|
|
|
_("no extended partition found and no "
|
|
|
|
"primary partition available"));
|
2010-08-18 20:54:48 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
default:
|
2012-07-18 11:38:29 +00:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("unknown partition type"));
|
2010-08-18 21:54:11 +00:00
|
|
|
return -1;
|
2009-06-26 16:18:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2013-05-03 12:49:08 +00:00
|
|
|
if (VIR_STRDUP(*partFormat, "primary") < 0)
|
2010-08-18 21:54:11 +00:00
|
|
|
return -1;
|
2009-06-26 16:18:59 +00:00
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2012-10-11 16:31:20 +00:00
|
|
|
* Aligns a new partition to nearest cylinder boundary
|
2010-08-18 20:54:48 +00:00
|
|
|
* when having a msdos partition table type
|
2012-10-11 16:31:20 +00:00
|
|
|
* to avoid any problem with already existing
|
2009-06-26 16:18:59 +00:00
|
|
|
* partitions
|
|
|
|
*/
|
|
|
|
static int
|
2014-04-01 22:26:59 +00:00
|
|
|
virStorageBackendDiskPartBoundaries(virStoragePoolObjPtr pool,
|
|
|
|
unsigned long long *start,
|
|
|
|
unsigned long long *end,
|
|
|
|
unsigned long long allocation)
|
2008-02-20 15:52:17 +00:00
|
|
|
{
|
Convert 'int i' to 'size_t i' in src/storage/ 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;
|
2008-02-20 15:52:17 +00:00
|
|
|
int smallestExtent = -1;
|
2009-06-26 16:18:59 +00:00
|
|
|
unsigned long long smallestSize = 0;
|
|
|
|
unsigned long long extraBytes = 0;
|
|
|
|
unsigned long long alignedAllocation = allocation;
|
2008-02-20 15:52:17 +00:00
|
|
|
virStoragePoolSourceDevicePtr dev = &pool->def->source.devices[0];
|
2014-09-04 14:34:43 +00:00
|
|
|
unsigned long long cylinderSize = (unsigned long long)dev->geometry.heads *
|
2009-06-26 16:18:59 +00:00
|
|
|
dev->geometry.sectors * SECTOR_SIZE;
|
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("find free area: allocation %llu, cyl size %llu", allocation,
|
2010-08-18 20:54:48 +00:00
|
|
|
cylinderSize);
|
2009-06-26 16:18:59 +00:00
|
|
|
int partType = virStorageBackendDiskPartTypeToCreate(pool);
|
|
|
|
|
|
|
|
/* how many extra bytes we have since we allocate
|
2012-10-11 16:31:20 +00:00
|
|
|
aligned to the cylinder boundary */
|
2009-06-26 16:18:59 +00:00
|
|
|
extraBytes = cylinderSize - (allocation % cylinderSize);
|
|
|
|
|
2013-05-21 07:21:21 +00:00
|
|
|
for (i = 0; i < dev->nfreeExtent; i++) {
|
2009-06-26 16:18:59 +00:00
|
|
|
unsigned long long size =
|
|
|
|
dev->freeExtents[i].end -
|
|
|
|
dev->freeExtents[i].start;
|
|
|
|
unsigned long long neededSize = allocation;
|
|
|
|
|
|
|
|
if (pool->def->source.format == VIR_STORAGE_POOL_DISK_DOS) {
|
2012-10-11 16:31:20 +00:00
|
|
|
/* align to cylinder boundary */
|
2009-06-26 16:18:59 +00:00
|
|
|
neededSize += extraBytes;
|
|
|
|
if ((*start % cylinderSize) > extraBytes) {
|
|
|
|
/* add an extra cylinder if the offset can't fit within
|
|
|
|
the extra bytes we have */
|
|
|
|
neededSize += cylinderSize;
|
|
|
|
}
|
2014-04-01 22:26:59 +00:00
|
|
|
/* if we are creating a logical partition, we need one extra
|
2009-06-26 16:18:59 +00:00
|
|
|
block between partitions (or actually move start one block) */
|
2014-11-13 14:25:27 +00:00
|
|
|
if (partType == VIR_STORAGE_VOL_DISK_TYPE_LOGICAL)
|
2009-06-26 16:18:59 +00:00
|
|
|
size -= SECTOR_SIZE;
|
|
|
|
}
|
|
|
|
if (size > neededSize &&
|
|
|
|
(smallestSize == 0 ||
|
|
|
|
size < smallestSize)) {
|
|
|
|
/* for logical partition, the free extent
|
|
|
|
must be within a logical free area */
|
|
|
|
if (partType == VIR_STORAGE_VOL_DISK_TYPE_LOGICAL &&
|
|
|
|
dev->freeExtents[i].type != VIR_STORAGE_FREE_LOGICAL) {
|
|
|
|
continue;
|
|
|
|
/* for primary partition, the free extent
|
|
|
|
must not be within a logical free area */
|
2010-08-18 20:54:48 +00:00
|
|
|
} else if (partType == VIR_STORAGE_VOL_DISK_TYPE_PRIMARY &&
|
|
|
|
dev->freeExtents[i].type != VIR_STORAGE_FREE_NORMAL) {
|
|
|
|
continue;
|
2009-06-26 16:18:59 +00:00
|
|
|
}
|
|
|
|
smallestSize = size;
|
|
|
|
smallestExtent = i;
|
|
|
|
alignedAllocation = neededSize;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (smallestExtent == -1) {
|
2012-07-18 11:38:29 +00:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
"%s", _("no large enough free extent"));
|
2009-06-26 16:18:59 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("aligned alloc %llu", alignedAllocation);
|
2009-06-26 16:18:59 +00:00
|
|
|
*start = dev->freeExtents[smallestExtent].start;
|
|
|
|
|
|
|
|
if (partType == VIR_STORAGE_VOL_DISK_TYPE_LOGICAL) {
|
|
|
|
/* for logical partition, skip one block */
|
|
|
|
*start += SECTOR_SIZE;
|
|
|
|
}
|
|
|
|
|
|
|
|
*end = *start + alignedAllocation;
|
|
|
|
if (pool->def->source.format == VIR_STORAGE_POOL_DISK_DOS) {
|
2012-10-11 16:31:20 +00:00
|
|
|
/* adjust our allocation if start is not at a cylinder boundary */
|
2009-06-26 16:18:59 +00:00
|
|
|
*end -= (*start % cylinderSize);
|
|
|
|
}
|
|
|
|
|
2014-03-30 02:27:44 +00:00
|
|
|
/* counting in bytes, we want the last byte of the current sector */
|
2009-06-26 16:18:59 +00:00
|
|
|
*end -= 1;
|
2011-02-16 23:37:57 +00:00
|
|
|
VIR_DEBUG("final aligned start %llu, end %llu", *start, *end);
|
2009-06-26 16:18:59 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-04-26 12:53:57 +00:00
|
|
|
/* virStorageBackendDiskDeleteVol
|
|
|
|
* @conn: Pointer to a libvirt connection
|
|
|
|
* @pool: Pointer to the storage pool
|
|
|
|
* @vol: Pointer to the volume definition
|
|
|
|
* @flags: flags (unused for now)
|
|
|
|
*
|
|
|
|
* This API will remove the disk volume partition either from direct
|
|
|
|
* API call or as an error path during creation when the partition
|
|
|
|
* name provided during create doesn't match the name read from
|
|
|
|
* virStorageBackendDiskReadPartitions.
|
|
|
|
*
|
|
|
|
* For a device mapper device, device respresentation is dependant upon
|
|
|
|
* device mapper configuration, but the general rule of thumb is that at
|
|
|
|
* creation if a device name ends with a number, then a partition separator
|
|
|
|
* "p" is added to the created name; otherwise, if the device name doesn't
|
|
|
|
* end with a number, then there is no partition separator. This name is
|
|
|
|
* what ends up in the vol->target.path. This ends up being a link to a
|
|
|
|
* /dev/mapper/dm-# device which cannot be used in the algorithm to determine
|
|
|
|
* which partition to remove, but a properly handled target.path can be.
|
|
|
|
*
|
|
|
|
* For non device mapper devices, just need to resolve the link of the
|
|
|
|
* vol->target.path in order to get the path.
|
|
|
|
*
|
|
|
|
* Returns 0 on success, -1 on failure with error message set.
|
|
|
|
*/
|
2015-01-21 19:41:02 +00:00
|
|
|
static int
|
2015-01-22 16:23:10 +00:00
|
|
|
virStorageBackendDiskDeleteVol(virConnectPtr conn,
|
2015-01-21 19:41:02 +00:00
|
|
|
virStoragePoolObjPtr pool,
|
|
|
|
virStorageVolDefPtr vol,
|
|
|
|
unsigned int flags)
|
|
|
|
{
|
|
|
|
char *part_num = NULL;
|
|
|
|
char *devpath = NULL;
|
2016-04-26 12:53:57 +00:00
|
|
|
char *dev_name;
|
|
|
|
char *src_path = pool->def->source.devices[0].path;
|
|
|
|
char *srcname = last_component(src_path);
|
2015-01-21 19:41:02 +00:00
|
|
|
virCommandPtr cmd = NULL;
|
|
|
|
bool isDevMapperDevice;
|
|
|
|
int rc = -1;
|
|
|
|
|
|
|
|
virCheckFlags(0, -1);
|
|
|
|
|
2015-01-21 20:38:05 +00:00
|
|
|
if (!vol->target.path) {
|
|
|
|
virReportError(VIR_ERR_INVALID_ARG,
|
|
|
|
_("volume target path empty for source path '%s'"),
|
2016-04-26 12:53:57 +00:00
|
|
|
src_path);
|
2015-01-21 20:38:05 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2016-04-26 12:53:57 +00:00
|
|
|
/* NB: This is the corollary to the algorithm in libvirt_parthelper
|
|
|
|
* (parthelper.c) that is used to generate the target.path name
|
|
|
|
* for use by libvirt. Changes to either, need to be reflected
|
|
|
|
* in both places */
|
|
|
|
isDevMapperDevice = virIsDevMapperDevice(vol->target.path);
|
|
|
|
if (isDevMapperDevice) {
|
|
|
|
dev_name = last_component(vol->target.path);
|
|
|
|
} else {
|
|
|
|
if (virFileResolveLink(vol->target.path, &devpath) < 0) {
|
|
|
|
virReportSystemError(errno,
|
|
|
|
_("Couldn't read volume target path '%s'"),
|
|
|
|
vol->target.path);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
dev_name = last_component(devpath);
|
2015-01-21 19:41:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
VIR_DEBUG("dev_name=%s, srcname=%s", dev_name, srcname);
|
|
|
|
|
2016-04-26 12:53:57 +00:00
|
|
|
if (!STRPREFIX(dev_name, srcname)) {
|
2015-01-21 19:41:02 +00:00
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("Volume path '%s' did not start with parent "
|
|
|
|
"pool source device name."), dev_name);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2016-04-26 12:53:57 +00:00
|
|
|
part_num = dev_name + strlen(srcname);
|
2015-01-21 19:41:02 +00:00
|
|
|
|
2016-04-26 12:53:57 +00:00
|
|
|
/* For device mapper and we have a partition character 'p' as the
|
|
|
|
* current character, let's move beyond that before checking part_num */
|
|
|
|
if (isDevMapperDevice && *part_num == 'p')
|
|
|
|
part_num++;
|
2015-01-21 19:41:02 +00:00
|
|
|
|
2016-04-26 12:53:57 +00:00
|
|
|
if (*part_num == 0) {
|
|
|
|
virReportError(VIR_ERR_INTERNAL_ERROR,
|
|
|
|
_("cannot parse partition number from target "
|
|
|
|
"'%s'"), dev_name);
|
|
|
|
goto cleanup;
|
2015-01-21 19:41:02 +00:00
|
|
|
}
|
|
|
|
|
2016-04-26 12:53:57 +00:00
|
|
|
/* eg parted /dev/sda rm 2 or /dev/mapper/mpathc rm 2 */
|
|
|
|
cmd = virCommandNewArgList(PARTED,
|
|
|
|
src_path,
|
|
|
|
"rm",
|
|
|
|
"--script",
|
|
|
|
part_num,
|
|
|
|
NULL);
|
|
|
|
if (virCommandRun(cmd, NULL) < 0)
|
|
|
|
goto cleanup;
|
|
|
|
|
2015-10-01 15:37:17 +00:00
|
|
|
/* Refreshing the pool is the easiest option as LOGICAL and EXTENDED
|
|
|
|
* partition allocation/capacity management is handled within
|
|
|
|
* virStorageBackendDiskMakeDataVol and trying to redo that logic
|
|
|
|
* here is pointless
|
2015-01-22 16:23:10 +00:00
|
|
|
*/
|
2015-10-01 15:37:17 +00:00
|
|
|
virStoragePoolObjClearVols(pool);
|
|
|
|
if (virStorageBackendDiskRefreshPool(conn, pool) < 0)
|
|
|
|
goto cleanup;
|
2015-01-22 16:23:10 +00:00
|
|
|
|
2015-01-21 19:41:02 +00:00
|
|
|
rc = 0;
|
|
|
|
cleanup:
|
|
|
|
VIR_FREE(devpath);
|
|
|
|
virCommandFree(cmd);
|
|
|
|
return rc;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-06-26 16:18:59 +00:00
|
|
|
static int
|
2015-01-21 20:38:05 +00:00
|
|
|
virStorageBackendDiskCreateVol(virConnectPtr conn,
|
2014-02-12 13:54:05 +00:00
|
|
|
virStoragePoolObjPtr pool,
|
2009-06-26 16:18:59 +00:00
|
|
|
virStorageVolDefPtr vol)
|
|
|
|
{
|
2010-08-18 21:54:11 +00:00
|
|
|
int res = -1;
|
2013-01-30 12:02:15 +00:00
|
|
|
char *partFormat = NULL;
|
2009-06-26 16:18:59 +00:00
|
|
|
unsigned long long startOffset = 0, endOffset = 0;
|
2014-02-12 13:54:05 +00:00
|
|
|
virCommandPtr cmd = virCommandNewArgList(PARTED,
|
|
|
|
pool->def->source.devices[0].path,
|
|
|
|
"mkpart",
|
|
|
|
"--script",
|
|
|
|
NULL);
|
2009-07-21 02:40:50 +00:00
|
|
|
|
2014-02-12 13:54:05 +00:00
|
|
|
if (vol->target.encryption != NULL) {
|
|
|
|
virReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
|
|
|
"%s", _("storage pool does not support encrypted "
|
|
|
|
"volumes"));
|
2013-01-30 12:02:15 +00:00
|
|
|
goto cleanup;
|
2014-02-12 13:54:05 +00:00
|
|
|
}
|
2013-12-11 16:04:24 +00:00
|
|
|
|
2014-11-13 14:25:27 +00:00
|
|
|
if (virStorageBackendDiskPartFormat(pool, vol, &partFormat) != 0)
|
2014-02-12 13:54:05 +00:00
|
|
|
goto cleanup;
|
2012-07-11 14:56:55 +00:00
|
|
|
virCommandAddArg(cmd, partFormat);
|
2009-06-26 16:18:59 +00:00
|
|
|
|
2014-04-01 22:26:59 +00:00
|
|
|
if (virStorageBackendDiskPartBoundaries(pool, &startOffset,
|
|
|
|
&endOffset,
|
conf: track sizes directly in source struct
One of the features of qcow2 is that a wrapper file can have
more capacity than its backing file from the guest's perspective;
what's more, sparse files make tracking allocation of both
the active and backing file worthwhile. As such, it makes
more sense to show allocation numbers for each file in a chain,
and not just the top-level file. This sets up the fields for
the tracking, although it does not modify XML to display any
new information.
* src/util/virstoragefile.h (_virStorageSource): Add fields.
* src/conf/storage_conf.h (_virStorageVolDef): Drop redundant
fields.
* src/storage/storage_backend.c (virStorageBackendCreateBlockFrom)
(createRawFile, virStorageBackendCreateQemuImgCmd)
(virStorageBackendCreateQcowCreate): Update clients.
* src/storage/storage_driver.c (storageVolDelete)
(storageVolCreateXML, storageVolCreateXMLFrom, storageVolResize)
(storageVolWipeInternal, storageVolGetInfo): Likewise.
* src/storage/storage_backend_fs.c (virStorageBackendProbeTarget)
(virStorageBackendFileSystemRefresh)
(virStorageBackendFileSystemVolResize)
(virStorageBackendFileSystemVolRefresh): Likewise.
* src/storage/storage_backend_logical.c
(virStorageBackendLogicalMakeVol)
(virStorageBackendLogicalCreateVol): Likewise.
* src/storage/storage_backend_scsi.c
(virStorageBackendSCSINewLun): Likewise.
* src/storage/storage_backend_mpath.c
(virStorageBackendMpathNewVol): Likewise.
* src/storage/storage_backend_rbd.c
(volStorageBackendRBDRefreshVolInfo)
(virStorageBackendRBDCreateImage): Likewise.
* src/storage/storage_backend_disk.c
(virStorageBackendDiskMakeDataVol)
(virStorageBackendDiskCreateVol): Likewise.
* src/storage/storage_backend_sheepdog.c
(virStorageBackendSheepdogBuildVol)
(virStorageBackendSheepdogParseVdiList): Likewise.
* src/storage/storage_backend_gluster.c
(virStorageBackendGlusterRefreshVol): Likewise.
* src/conf/storage_conf.c (virStorageVolDefFormat)
(virStorageVolDefParseXML): Likewise.
* src/test/test_driver.c (testOpenVolumesForPool)
(testStorageVolCreateXML, testStorageVolCreateXMLFrom)
(testStorageVolDelete, testStorageVolGetInfo): Likewise.
* src/esx/esx_storage_backend_iscsi.c (esxStorageVolGetXMLDesc):
Likewise.
* src/esx/esx_storage_backend_vmfs.c (esxStorageVolGetXMLDesc)
(esxStorageVolCreateXML): Likewise.
* src/parallels/parallels_driver.c (parallelsAddHddByVolume):
Likewise.
* src/parallels/parallels_storage.c (parallelsDiskDescParseNode)
(parallelsStorageVolDefineXML, parallelsStorageVolCreateXMLFrom)
(parallelsStorageVolDefRemove, parallelsStorageVolGetInfo):
Likewise.
* src/vbox/vbox_tmpl.c (vboxStorageVolCreateXML)
(vboxStorageVolGetXMLDesc): Likewise.
* tests/storagebackendsheepdogtest.c (test_vdi_list_parser):
Likewise.
* src/phyp/phyp_driver.c (phypStorageVolCreateXML): Likewise.
2014-04-01 23:43:36 +00:00
|
|
|
vol->target.capacity) != 0) {
|
2010-08-18 21:54:11 +00:00
|
|
|
goto cleanup;
|
2008-02-20 15:52:17 +00:00
|
|
|
}
|
|
|
|
|
2012-07-11 14:56:55 +00:00
|
|
|
virCommandAddArgFormat(cmd, "%lluB", startOffset);
|
|
|
|
virCommandAddArgFormat(cmd, "%lluB", endOffset);
|
2008-02-20 15:52:17 +00:00
|
|
|
|
2012-07-11 14:56:55 +00:00
|
|
|
if (virCommandRun(cmd, NULL) < 0)
|
2010-08-18 21:54:11 +00:00
|
|
|
goto cleanup;
|
2008-02-20 15:52:17 +00:00
|
|
|
|
2009-06-26 16:18:59 +00:00
|
|
|
/* wait for device node to show up */
|
2017-02-20 12:00:51 +00:00
|
|
|
virWaitForDevices();
|
2009-06-26 16:18:59 +00:00
|
|
|
|
2008-02-20 15:52:17 +00:00
|
|
|
/* Blow away free extent info, as we're about to re-populate it */
|
2008-06-06 11:09:57 +00:00
|
|
|
VIR_FREE(pool->def->source.devices[0].freeExtents);
|
2008-02-20 15:52:17 +00:00
|
|
|
pool->def->source.devices[0].nfreeExtent = 0;
|
|
|
|
|
2009-07-10 01:51:36 +00:00
|
|
|
/* Specifying a target path is meaningless */
|
|
|
|
VIR_FREE(vol->target.path);
|
|
|
|
|
|
|
|
/* Fetch actual extent info, generate key */
|
2015-01-21 20:38:05 +00:00
|
|
|
if (virStorageBackendDiskReadPartitions(pool, vol) < 0) {
|
|
|
|
/* Best effort to remove the partition. Ignore any errors
|
|
|
|
* since we could be calling this with vol->target.path == NULL
|
|
|
|
*/
|
|
|
|
virErrorPtr save_err = virSaveLastError();
|
|
|
|
ignore_value(virStorageBackendDiskDeleteVol(conn, pool, vol, 0));
|
|
|
|
virSetError(save_err);
|
|
|
|
virFreeError(save_err);
|
2010-08-18 21:54:11 +00:00
|
|
|
goto cleanup;
|
2015-01-21 20:38:05 +00:00
|
|
|
}
|
2008-02-20 15:52:17 +00:00
|
|
|
|
2010-08-18 21:54:11 +00:00
|
|
|
res = 0;
|
|
|
|
|
2014-03-25 06:52:40 +00:00
|
|
|
cleanup:
|
2010-08-18 21:54:11 +00:00
|
|
|
VIR_FREE(partFormat);
|
2012-07-11 14:56:55 +00:00
|
|
|
virCommandFree(cmd);
|
2010-08-18 21:54:11 +00:00
|
|
|
return res;
|
2008-02-20 15:52:17 +00:00
|
|
|
}
|
|
|
|
|
2009-07-09 23:17:31 +00:00
|
|
|
static int
|
|
|
|
virStorageBackendDiskBuildVolFrom(virConnectPtr conn,
|
2010-01-20 23:41:52 +00:00
|
|
|
virStoragePoolObjPtr pool,
|
2009-07-09 23:17:31 +00:00
|
|
|
virStorageVolDefPtr vol,
|
|
|
|
virStorageVolDefPtr inputvol,
|
|
|
|
unsigned int flags)
|
|
|
|
{
|
|
|
|
virStorageBackendBuildVolFrom build_func;
|
|
|
|
|
2010-02-10 11:42:56 +00:00
|
|
|
build_func = virStorageBackendGetBuildVolFromFunction(vol, inputvol);
|
2009-07-09 23:17:31 +00:00
|
|
|
if (!build_func)
|
|
|
|
return -1;
|
|
|
|
|
2010-01-20 23:41:52 +00:00
|
|
|
return build_func(conn, pool, vol, inputvol, flags);
|
2009-07-09 23:17:31 +00:00
|
|
|
}
|
2008-02-20 15:52:17 +00:00
|
|
|
|
|
|
|
|
2015-06-09 22:15:39 +00:00
|
|
|
static int
|
|
|
|
virStorageBackendDiskVolWipe(virConnectPtr conn,
|
|
|
|
virStoragePoolObjPtr pool,
|
|
|
|
virStorageVolDefPtr vol,
|
|
|
|
unsigned int algorithm,
|
|
|
|
unsigned int flags)
|
|
|
|
{
|
|
|
|
if (vol->source.partType != VIR_STORAGE_VOL_DISK_TYPE_EXTENDED)
|
|
|
|
return virStorageBackendVolWipeLocal(conn, pool, vol, algorithm, flags);
|
|
|
|
|
|
|
|
/* Wiping an extended partition is not support */
|
|
|
|
virReportError(VIR_ERR_NO_SUPPORT,
|
|
|
|
_("cannot wipe extended partition '%s'"),
|
|
|
|
vol->target.path);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-02-20 15:52:17 +00:00
|
|
|
virStorageBackend virStorageBackendDisk = {
|
|
|
|
.type = VIR_STORAGE_POOL_DISK,
|
|
|
|
|
2015-10-01 12:38:58 +00:00
|
|
|
.startPool = virStorageBackendDiskStartPool,
|
2008-02-20 15:52:17 +00:00
|
|
|
.buildPool = virStorageBackendDiskBuildPool,
|
|
|
|
.refreshPool = virStorageBackendDiskRefreshPool,
|
|
|
|
|
|
|
|
.createVol = virStorageBackendDiskCreateVol,
|
|
|
|
.deleteVol = virStorageBackendDiskDeleteVol,
|
2009-07-09 23:17:31 +00:00
|
|
|
.buildVolFrom = virStorageBackendDiskBuildVolFrom,
|
2014-07-07 14:50:11 +00:00
|
|
|
.uploadVol = virStorageBackendVolUploadLocal,
|
|
|
|
.downloadVol = virStorageBackendVolDownloadLocal,
|
2015-06-09 22:15:39 +00:00
|
|
|
.wipeVol = virStorageBackendDiskVolWipe,
|
2008-02-20 15:52:17 +00:00
|
|
|
};
|
2017-01-13 15:50:11 +00:00
|
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
virStorageBackendDiskRegister(void)
|
|
|
|
{
|
|
|
|
return virStorageBackendRegister(&virStorageBackendDisk);
|
|
|
|
}
|