2008-02-20 15:45:33 +00:00
|
|
|
/*
|
2011-04-17 08:34:10 +02:00
|
|
|
* storage_backend_logical.c: storage backend for logical volume handling
|
2008-02-20 15:45:33 +00:00
|
|
|
*
|
2011-07-06 16:51:23 -06:00
|
|
|
* Copyright (C) 2007-2009, 2011 Red Hat, Inc.
|
2008-02-20 15:45:33 +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
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
*
|
|
|
|
* Author: Daniel P. Berrange <berrange@redhat.com>
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
#include <sys/wait.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <regex.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
2008-11-04 22:30:33 +00:00
|
|
|
#include "virterror_internal.h"
|
2008-02-20 15:45:33 +00:00
|
|
|
#include "storage_backend_logical.h"
|
|
|
|
#include "storage_conf.h"
|
2011-05-10 14:42:59 -04:00
|
|
|
#include "command.h"
|
2008-06-06 11:09:57 +00:00
|
|
|
#include "memory.h"
|
2009-10-16 11:09:13 +01:00
|
|
|
#include "logging.h"
|
2011-07-19 12:32:58 -06:00
|
|
|
#include "virfile.h"
|
2008-02-20 15:45:33 +00:00
|
|
|
|
2009-01-20 17:13:33 +00:00
|
|
|
#define VIR_FROM_THIS VIR_FROM_STORAGE
|
|
|
|
|
2008-02-20 15:45:33 +00:00
|
|
|
#define PV_BLANK_SECTOR_SIZE 512
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
2010-02-04 23:41:52 +01:00
|
|
|
virStorageBackendLogicalSetActive(virStoragePoolObjPtr pool,
|
2008-02-20 15:45:33 +00:00
|
|
|
int on)
|
|
|
|
{
|
|
|
|
const char *cmdargv[4];
|
|
|
|
|
|
|
|
cmdargv[0] = VGCHANGE;
|
|
|
|
cmdargv[1] = on ? "-ay" : "-an";
|
2008-09-02 14:15:42 +00:00
|
|
|
cmdargv[2] = pool->def->source.name;
|
2008-02-20 15:45:33 +00:00
|
|
|
cmdargv[3] = NULL;
|
|
|
|
|
2010-02-04 23:41:52 +01:00
|
|
|
if (virRun(cmdargv, NULL) < 0)
|
2008-02-20 15:45:33 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
2010-02-10 11:42:56 +00:00
|
|
|
virStorageBackendLogicalMakeVol(virStoragePoolObjPtr pool,
|
2008-02-20 15:45:33 +00:00
|
|
|
char **const groups,
|
|
|
|
void *data)
|
|
|
|
{
|
|
|
|
virStorageVolDefPtr vol = NULL;
|
|
|
|
unsigned long long offset, size, length;
|
|
|
|
|
|
|
|
/* See if we're only looking for a specific volume */
|
|
|
|
if (data != NULL) {
|
|
|
|
vol = data;
|
|
|
|
if (STRNEQ(vol->name, groups[0]))
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Or filling in more data on an existing volume */
|
|
|
|
if (vol == NULL)
|
|
|
|
vol = virStorageVolDefFindByName(pool, groups[0]);
|
|
|
|
|
|
|
|
/* Or a completely new volume */
|
|
|
|
if (vol == NULL) {
|
2008-06-06 11:09:57 +00:00
|
|
|
if (VIR_ALLOC(vol) < 0) {
|
2010-02-04 19:19:08 +01:00
|
|
|
virReportOOMError();
|
2008-02-20 15:45:33 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2008-11-17 11:19:33 +00:00
|
|
|
vol->type = VIR_STORAGE_VOL_BLOCK;
|
|
|
|
|
2008-02-20 15:45:33 +00:00
|
|
|
if ((vol->name = strdup(groups[0])) == NULL) {
|
2010-02-04 19:19:08 +01:00
|
|
|
virReportOOMError();
|
2008-10-10 15:13:28 +00:00
|
|
|
virStorageVolDefFree(vol);
|
2008-02-20 15:45:33 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2008-10-10 15:13:28 +00:00
|
|
|
if (VIR_REALLOC_N(pool->volumes.objs,
|
|
|
|
pool->volumes.count + 1)) {
|
2010-02-04 19:19:08 +01:00
|
|
|
virReportOOMError();
|
2008-10-10 15:13:28 +00:00
|
|
|
virStorageVolDefFree(vol);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
pool->volumes.objs[pool->volumes.count++] = vol;
|
2008-02-20 15:45:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (vol->target.path == NULL) {
|
2009-01-27 18:30:03 +00:00
|
|
|
if (virAsprintf(&vol->target.path, "%s/%s",
|
|
|
|
pool->def->target.path, vol->name) < 0) {
|
2010-02-04 19:19:08 +01:00
|
|
|
virReportOOMError();
|
2009-01-27 18:30:03 +00:00
|
|
|
virStorageVolDefFree(vol);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (groups[1] && !STREQ(groups[1], "")) {
|
|
|
|
if (virAsprintf(&vol->backingStore.path, "%s/%s",
|
|
|
|
pool->def->target.path, groups[1]) < 0) {
|
2010-02-04 19:19:08 +01:00
|
|
|
virReportOOMError();
|
2009-01-27 18:30:03 +00:00
|
|
|
virStorageVolDefFree(vol);
|
2008-02-20 15:45:33 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2009-01-27 18:30:03 +00:00
|
|
|
|
|
|
|
vol->backingStore.format = VIR_STORAGE_POOL_LOGICAL_LVM2;
|
2008-02-20 15:45:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (vol->key == NULL &&
|
2009-01-27 18:30:03 +00:00
|
|
|
(vol->key = strdup(groups[2])) == NULL) {
|
2010-02-04 19:19:08 +01:00
|
|
|
virReportOOMError();
|
2008-02-20 15:45:33 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-02-04 21:02:58 +01:00
|
|
|
if (virStorageBackendUpdateVolInfo(vol, 1) < 0)
|
2008-02-20 15:45:33 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
|
|
/* Finally fill in extents information */
|
2008-06-06 11:09:57 +00:00
|
|
|
if (VIR_REALLOC_N(vol->source.extents,
|
|
|
|
vol->source.nextent + 1) < 0) {
|
2010-02-04 19:19:08 +01:00
|
|
|
virReportOOMError();
|
2008-02-20 15:45:33 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((vol->source.extents[vol->source.nextent].path =
|
2009-01-27 18:30:03 +00:00
|
|
|
strdup(groups[3])) == NULL) {
|
2010-02-04 19:19:08 +01:00
|
|
|
virReportOOMError();
|
2008-02-20 15:45:33 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2009-01-27 18:30:03 +00:00
|
|
|
if (virStrToLong_ull(groups[4], NULL, 10, &offset) < 0) {
|
2010-02-10 11:42:56 +00:00
|
|
|
virStorageReportError(VIR_ERR_INTERNAL_ERROR,
|
2008-02-22 16:26:13 +00:00
|
|
|
"%s", _("malformed volume extent offset value"));
|
2008-02-20 15:45:33 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2009-01-27 18:30:03 +00:00
|
|
|
if (virStrToLong_ull(groups[5], NULL, 10, &length) < 0) {
|
2010-02-10 11:42:56 +00:00
|
|
|
virStorageReportError(VIR_ERR_INTERNAL_ERROR,
|
2008-02-22 16:26:13 +00:00
|
|
|
"%s", _("malformed volume extent length value"));
|
2008-02-20 15:45:33 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2009-01-27 18:30:03 +00:00
|
|
|
if (virStrToLong_ull(groups[6], NULL, 10, &size) < 0) {
|
2010-02-10 11:42:56 +00:00
|
|
|
virStorageReportError(VIR_ERR_INTERNAL_ERROR,
|
2008-02-22 16:26:13 +00:00
|
|
|
"%s", _("malformed volume extent size value"));
|
2008-02-20 15:45:33 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
vol->source.extents[vol->source.nextent].start = offset * size;
|
|
|
|
vol->source.extents[vol->source.nextent].end = (offset * size) + length;
|
|
|
|
vol->source.nextent++;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2010-02-10 11:42:56 +00:00
|
|
|
virStorageBackendLogicalFindLVs(virStoragePoolObjPtr pool,
|
2008-02-20 15:45:33 +00:00
|
|
|
virStorageVolDefPtr vol)
|
|
|
|
{
|
|
|
|
/*
|
2009-01-27 18:30:03 +00:00
|
|
|
* # lvs --separator , --noheadings --units b --unbuffered --nosuffix --options "lv_name,origin,uuid,devices,seg_size,vg_extent_size" VGNAME
|
|
|
|
* RootLV,,06UgP5-2rhb-w3Bo-3mdR-WeoL-pytO-SAa2ky,/dev/hda2(0),5234491392,33554432
|
|
|
|
* SwapLV,,oHviCK-8Ik0-paqS-V20c-nkhY-Bm1e-zgzU0M,/dev/hda2(156),1040187392,33554432
|
|
|
|
* Test2,,3pg3he-mQsA-5Sui-h0i6-HNmc-Cz7W-QSndcR,/dev/hda2(219),1073741824,33554432
|
|
|
|
* Test3,,UB5hFw-kmlm-LSoX-EI1t-ioVd-h7GL-M0W8Ht,/dev/hda2(251),2181038080,33554432
|
|
|
|
* Test3,Test2,UB5hFw-kmlm-LSoX-EI1t-ioVd-h7GL-M0W8Ht,/dev/hda2(187),1040187392,33554432
|
2008-02-20 15:45:33 +00:00
|
|
|
*
|
2009-01-27 18:30:03 +00:00
|
|
|
* Pull out name, origin, & uuid, device, device extent start #, segment size, extent size.
|
2008-02-20 15:45:33 +00:00
|
|
|
*
|
|
|
|
* NB can be multiple rows per volume if they have many extents
|
2008-07-30 08:52:44 +00:00
|
|
|
*
|
2008-11-14 16:10:47 +00:00
|
|
|
* NB lvs from some distros (e.g. SLES10 SP2) outputs trailing "," on each line
|
|
|
|
*
|
|
|
|
* NB Encrypted logical volumes can print ':' in their name, so it is
|
|
|
|
* not a suitable separator (rhbz 470693).
|
2008-02-20 15:45:33 +00:00
|
|
|
*/
|
|
|
|
const char *regexes[] = {
|
2009-01-27 18:30:03 +00:00
|
|
|
"^\\s*(\\S+),(\\S*),(\\S+),(\\S+)\\((\\S+)\\),(\\S+),([0-9]+),?\\s*$"
|
2008-02-20 15:45:33 +00:00
|
|
|
};
|
|
|
|
int vars[] = {
|
2009-01-27 18:30:03 +00:00
|
|
|
7
|
2008-02-20 15:45:33 +00:00
|
|
|
};
|
|
|
|
const char *prog[] = {
|
2008-11-14 16:10:47 +00:00
|
|
|
LVS, "--separator", ",", "--noheadings", "--units", "b",
|
2008-02-20 15:45:33 +00:00
|
|
|
"--unbuffered", "--nosuffix", "--options",
|
2009-01-27 18:30:03 +00:00
|
|
|
"lv_name,origin,uuid,devices,seg_size,vg_extent_size",
|
2008-09-02 14:15:42 +00:00
|
|
|
pool->def->source.name, NULL
|
2008-02-20 15:45:33 +00:00
|
|
|
};
|
|
|
|
|
2010-02-10 11:42:56 +00:00
|
|
|
if (virStorageBackendRunProgRegex(pool,
|
2008-06-17 12:45:24 +00:00
|
|
|
prog,
|
|
|
|
1,
|
|
|
|
regexes,
|
|
|
|
vars,
|
|
|
|
virStorageBackendLogicalMakeVol,
|
2011-05-10 15:36:50 -04:00
|
|
|
vol) < 0) {
|
2008-06-17 12:45:24 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
2008-02-20 15:45:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2010-02-10 11:42:56 +00:00
|
|
|
virStorageBackendLogicalRefreshPoolFunc(virStoragePoolObjPtr pool ATTRIBUTE_UNUSED,
|
2008-02-20 15:45:33 +00:00
|
|
|
char **const groups,
|
|
|
|
void *data ATTRIBUTE_UNUSED)
|
|
|
|
{
|
|
|
|
if (virStrToLong_ull(groups[0], NULL, 10, &pool->def->capacity) < 0)
|
|
|
|
return -1;
|
|
|
|
if (virStrToLong_ull(groups[1], NULL, 10, &pool->def->available) < 0)
|
|
|
|
return -1;
|
|
|
|
pool->def->allocation = pool->def->capacity - pool->def->available;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-08-27 20:05:58 +00:00
|
|
|
static int
|
2010-02-10 11:42:56 +00:00
|
|
|
virStorageBackendLogicalFindPoolSourcesFunc(virStoragePoolObjPtr pool ATTRIBUTE_UNUSED,
|
2008-08-27 20:05:58 +00:00
|
|
|
char **const groups,
|
|
|
|
void *data)
|
|
|
|
{
|
2008-10-23 11:39:53 +00:00
|
|
|
virStoragePoolSourceListPtr sourceList = data;
|
|
|
|
char *pvname = NULL;
|
|
|
|
char *vgname = NULL;
|
|
|
|
int i;
|
|
|
|
virStoragePoolSourceDevicePtr dev;
|
|
|
|
virStoragePoolSource *thisSource;
|
|
|
|
|
|
|
|
pvname = strdup(groups[0]);
|
|
|
|
vgname = strdup(groups[1]);
|
|
|
|
|
|
|
|
if (pvname == NULL || vgname == NULL) {
|
2010-02-04 19:19:08 +01:00
|
|
|
virReportOOMError();
|
2008-10-23 11:39:53 +00:00
|
|
|
goto err_no_memory;
|
|
|
|
}
|
|
|
|
|
|
|
|
thisSource = NULL;
|
|
|
|
for (i = 0 ; i < sourceList->nsources; i++) {
|
|
|
|
if (STREQ(sourceList->sources[i].name, vgname)) {
|
|
|
|
thisSource = &sourceList->sources[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2008-08-27 20:05:58 +00:00
|
|
|
|
2008-10-23 11:39:53 +00:00
|
|
|
if (thisSource == NULL) {
|
2010-02-10 11:42:56 +00:00
|
|
|
if (!(thisSource = virStoragePoolSourceListNewSource(sourceList)))
|
2008-10-23 11:39:53 +00:00
|
|
|
goto err_no_memory;
|
|
|
|
|
|
|
|
thisSource->name = vgname;
|
2008-08-27 20:05:58 +00:00
|
|
|
}
|
2008-10-23 11:39:53 +00:00
|
|
|
else
|
|
|
|
VIR_FREE(vgname);
|
2008-08-27 20:05:58 +00:00
|
|
|
|
2008-10-23 11:39:53 +00:00
|
|
|
if (VIR_REALLOC_N(thisSource->devices, thisSource->ndevice + 1) != 0) {
|
2010-02-04 19:19:08 +01:00
|
|
|
virReportOOMError();
|
2008-10-23 11:39:53 +00:00
|
|
|
goto err_no_memory;
|
2008-08-27 20:05:58 +00:00
|
|
|
}
|
|
|
|
|
2008-10-23 11:39:53 +00:00
|
|
|
dev = &thisSource->devices[thisSource->ndevice];
|
|
|
|
thisSource->ndevice++;
|
2008-11-04 21:54:21 +00:00
|
|
|
thisSource->format = VIR_STORAGE_POOL_LOGICAL_LVM2;
|
2008-10-23 11:39:53 +00:00
|
|
|
|
|
|
|
memset(dev, 0, sizeof(*dev));
|
|
|
|
dev->path = pvname;
|
2008-08-27 20:05:58 +00:00
|
|
|
|
|
|
|
return 0;
|
2008-10-23 11:39:53 +00:00
|
|
|
|
|
|
|
err_no_memory:
|
|
|
|
VIR_FREE(pvname);
|
|
|
|
VIR_FREE(vgname);
|
|
|
|
|
|
|
|
return -1;
|
2008-08-27 20:05:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static char *
|
2010-02-10 11:42:56 +00:00
|
|
|
virStorageBackendLogicalFindPoolSources(virConnectPtr conn ATTRIBUTE_UNUSED,
|
2008-08-27 20:05:58 +00:00
|
|
|
const char *srcSpec ATTRIBUTE_UNUSED,
|
2011-07-06 16:51:23 -06:00
|
|
|
unsigned int flags)
|
2008-08-27 20:05:58 +00:00
|
|
|
{
|
|
|
|
/*
|
2008-10-23 11:39:53 +00:00
|
|
|
* # pvs --noheadings -o pv_name,vg_name
|
|
|
|
* /dev/sdb
|
|
|
|
* /dev/sdc VolGroup00
|
2008-08-27 20:05:58 +00:00
|
|
|
*/
|
|
|
|
const char *regexes[] = {
|
2008-10-23 11:39:53 +00:00
|
|
|
"^\\s*(\\S+)\\s+(\\S+)\\s*$"
|
2008-08-27 20:05:58 +00:00
|
|
|
};
|
|
|
|
int vars[] = {
|
2008-10-23 11:39:53 +00:00
|
|
|
2
|
2008-08-27 20:05:58 +00:00
|
|
|
};
|
2008-10-23 11:39:53 +00:00
|
|
|
const char *const prog[] = { PVS, "--noheadings", "-o", "pv_name,vg_name", NULL };
|
2008-11-05 11:41:43 +00:00
|
|
|
const char *const scanprog[] = { VGSCAN, NULL };
|
2008-08-27 20:05:58 +00:00
|
|
|
char *retval = NULL;
|
2008-10-23 11:39:53 +00:00
|
|
|
virStoragePoolSourceList sourceList;
|
|
|
|
int i;
|
2008-08-27 20:05:58 +00:00
|
|
|
|
2011-07-06 16:51:23 -06:00
|
|
|
virCheckFlags(0, NULL);
|
|
|
|
|
2008-11-05 11:41:43 +00:00
|
|
|
/*
|
|
|
|
* NOTE: ignoring errors here; this is just to "touch" any logical volumes
|
|
|
|
* that might be hanging around, so if this fails for some reason, the
|
|
|
|
* worst that happens is that scanning doesn't pick everything up
|
|
|
|
*/
|
2011-05-10 15:36:50 -04:00
|
|
|
if (virRun(scanprog, NULL) < 0) {
|
2011-05-09 17:24:09 +08:00
|
|
|
VIR_WARN("Failure when running vgscan to refresh physical volumes");
|
2009-10-16 11:09:13 +01:00
|
|
|
}
|
2008-11-05 11:41:43 +00:00
|
|
|
|
2008-10-23 11:39:53 +00:00
|
|
|
memset(&sourceList, 0, sizeof(sourceList));
|
2008-11-04 21:54:21 +00:00
|
|
|
sourceList.type = VIR_STORAGE_POOL_LOGICAL;
|
|
|
|
|
2010-02-10 11:42:56 +00:00
|
|
|
if (virStorageBackendRunProgRegex(NULL, prog, 1, regexes, vars,
|
2011-05-10 15:36:50 -04:00
|
|
|
virStorageBackendLogicalFindPoolSourcesFunc,
|
|
|
|
&sourceList) < 0)
|
2008-08-27 20:05:58 +00:00
|
|
|
return NULL;
|
|
|
|
|
2010-02-10 11:42:56 +00:00
|
|
|
retval = virStoragePoolSourceListFormat(&sourceList);
|
2008-08-27 20:05:58 +00:00
|
|
|
if (retval == NULL) {
|
2010-02-10 11:42:56 +00:00
|
|
|
virStorageReportError(VIR_ERR_INTERNAL_ERROR, "%s",
|
2008-10-23 11:39:53 +00:00
|
|
|
_("failed to get source from sourceList"));
|
2008-08-27 20:05:58 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
cleanup:
|
2008-10-23 11:39:53 +00:00
|
|
|
for (i = 0; i < sourceList.nsources; i++)
|
|
|
|
virStoragePoolSourceFree(&sourceList.sources[i]);
|
|
|
|
|
|
|
|
VIR_FREE(sourceList.sources);
|
2008-08-27 20:05:58 +00:00
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-11-11 20:09:20 +00:00
|
|
|
static int
|
|
|
|
virStorageBackendLogicalCheckPool(virConnectPtr conn ATTRIBUTE_UNUSED,
|
|
|
|
virStoragePoolObjPtr pool,
|
|
|
|
bool *isActive)
|
|
|
|
{
|
|
|
|
char *path;
|
|
|
|
|
|
|
|
*isActive = false;
|
|
|
|
if (virAsprintf(&path, "/dev/%s", pool->def->source.name) < 0) {
|
|
|
|
virReportOOMError();
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (access(path, F_OK) == 0)
|
|
|
|
*isActive = true;
|
|
|
|
|
|
|
|
VIR_FREE(path);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-02-20 15:45:33 +00:00
|
|
|
static int
|
2010-02-04 23:41:52 +01:00
|
|
|
virStorageBackendLogicalStartPool(virConnectPtr conn ATTRIBUTE_UNUSED,
|
2008-02-20 15:45:33 +00:00
|
|
|
virStoragePoolObjPtr pool)
|
|
|
|
{
|
2010-02-04 23:41:52 +01:00
|
|
|
if (virStorageBackendLogicalSetActive(pool, 1) < 0)
|
2008-02-20 15:45:33 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
2010-02-04 23:41:52 +01:00
|
|
|
virStorageBackendLogicalBuildPool(virConnectPtr conn ATTRIBUTE_UNUSED,
|
2008-02-20 15:45:33 +00:00
|
|
|
virStoragePoolObjPtr pool,
|
2011-07-06 16:51:23 -06:00
|
|
|
unsigned int flags)
|
2008-02-20 15:45:33 +00:00
|
|
|
{
|
|
|
|
const char **vgargv;
|
|
|
|
const char *pvargv[3];
|
|
|
|
int n = 0, i, fd;
|
|
|
|
char zeros[PV_BLANK_SECTOR_SIZE];
|
|
|
|
|
2011-07-06 16:51:23 -06:00
|
|
|
virCheckFlags(0, -1);
|
|
|
|
|
2008-02-20 15:45:33 +00:00
|
|
|
memset(zeros, 0, sizeof(zeros));
|
|
|
|
|
2008-09-04 13:13:32 +00:00
|
|
|
if (VIR_ALLOC_N(vgargv, 3 + pool->def->source.ndevice) < 0) {
|
2010-02-04 19:19:08 +01:00
|
|
|
virReportOOMError();
|
2008-02-20 15:45:33 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
vgargv[n++] = VGCREATE;
|
2008-09-02 14:15:42 +00:00
|
|
|
vgargv[n++] = pool->def->source.name;
|
2008-02-20 15:45:33 +00:00
|
|
|
|
|
|
|
pvargv[0] = PVCREATE;
|
|
|
|
pvargv[2] = NULL;
|
|
|
|
for (i = 0 ; i < pool->def->source.ndevice ; i++) {
|
|
|
|
/*
|
|
|
|
* LVM requires that the first sector is blanked if using
|
|
|
|
* a whole disk as a PV. So we just blank them out regardless
|
|
|
|
* rather than trying to figure out if we're a disk or partition
|
|
|
|
*/
|
|
|
|
if ((fd = open(pool->def->source.devices[i].path, O_WRONLY)) < 0) {
|
2010-02-04 21:02:58 +01:00
|
|
|
virReportSystemError(errno,
|
2009-01-20 17:13:33 +00:00
|
|
|
_("cannot open device '%s'"),
|
|
|
|
pool->def->source.devices[i].path);
|
2008-02-20 15:45:33 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2008-02-25 10:00:16 +00:00
|
|
|
if (safewrite(fd, zeros, sizeof(zeros)) < 0) {
|
2010-02-04 21:02:58 +01:00
|
|
|
virReportSystemError(errno,
|
2009-01-20 17:13:33 +00:00
|
|
|
_("cannot clear device header of '%s'"),
|
|
|
|
pool->def->source.devices[i].path);
|
2011-08-18 14:40:03 +02:00
|
|
|
VIR_FORCE_CLOSE(fd);
|
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
if (fsync(fd) < 0) {
|
|
|
|
virReportSystemError(errno,
|
|
|
|
_("cannot flush header of device'%s'"),
|
|
|
|
pool->def->source.devices[i].path);
|
2010-10-28 11:43:16 -04:00
|
|
|
VIR_FORCE_CLOSE(fd);
|
2008-02-20 15:45:33 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
2010-10-28 11:43:16 -04:00
|
|
|
if (VIR_CLOSE(fd) < 0) {
|
2010-02-04 21:02:58 +01:00
|
|
|
virReportSystemError(errno,
|
2009-01-20 17:13:33 +00:00
|
|
|
_("cannot close device '%s'"),
|
|
|
|
pool->def->source.devices[i].path);
|
2008-02-20 15:45:33 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialize the physical volume because vgcreate is not
|
|
|
|
* clever enough todo this for us :-(
|
|
|
|
*/
|
|
|
|
vgargv[n++] = pool->def->source.devices[i].path;
|
|
|
|
pvargv[1] = pool->def->source.devices[i].path;
|
2010-02-04 23:41:52 +01:00
|
|
|
if (virRun(pvargv, NULL) < 0)
|
2008-02-20 15:45:33 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2009-09-03 18:14:48 +02:00
|
|
|
vgargv[n] = NULL;
|
2008-02-20 15:45:33 +00:00
|
|
|
|
|
|
|
/* Now create the volume group itself */
|
2010-02-04 23:41:52 +01:00
|
|
|
if (virRun(vgargv, NULL) < 0)
|
2008-02-20 15:45:33 +00:00
|
|
|
goto cleanup;
|
|
|
|
|
2008-06-06 11:09:57 +00:00
|
|
|
VIR_FREE(vgargv);
|
2008-02-20 15:45:33 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
cleanup:
|
2008-06-06 11:09:57 +00:00
|
|
|
VIR_FREE(vgargv);
|
2008-02-20 15:45:33 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
2010-02-10 11:42:56 +00:00
|
|
|
virStorageBackendLogicalRefreshPool(virConnectPtr conn ATTRIBUTE_UNUSED,
|
2008-02-20 15:45:33 +00:00
|
|
|
virStoragePoolObjPtr pool)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* # vgs --separator : --noheadings --units b --unbuffered --nosuffix --options "vg_size,vg_free" VGNAME
|
|
|
|
* 10603200512:4328521728
|
|
|
|
*
|
|
|
|
* Pull out size & free
|
2008-07-30 08:52:44 +00:00
|
|
|
*
|
|
|
|
* NB vgs from some distros (e.g. SLES10 SP2) outputs trailing ":" on each line
|
2008-02-20 15:45:33 +00:00
|
|
|
*/
|
|
|
|
const char *regexes[] = {
|
2008-07-30 08:52:44 +00:00
|
|
|
"^\\s*(\\S+):([0-9]+):?\\s*$"
|
2008-02-20 15:45:33 +00:00
|
|
|
};
|
|
|
|
int vars[] = {
|
|
|
|
2
|
|
|
|
};
|
|
|
|
const char *prog[] = {
|
|
|
|
VGS, "--separator", ":", "--noheadings", "--units", "b", "--unbuffered",
|
|
|
|
"--nosuffix", "--options", "vg_size,vg_free",
|
2008-09-02 14:15:42 +00:00
|
|
|
pool->def->source.name, NULL
|
2008-02-20 15:45:33 +00:00
|
|
|
};
|
|
|
|
|
2010-02-04 23:41:52 +01:00
|
|
|
virFileWaitForDevices();
|
2008-11-28 07:50:20 +00:00
|
|
|
|
2008-02-20 15:45:33 +00:00
|
|
|
/* Get list of all logical volumes */
|
2010-02-10 11:42:56 +00:00
|
|
|
if (virStorageBackendLogicalFindLVs(pool, NULL) < 0) {
|
2008-02-20 15:45:33 +00:00
|
|
|
virStoragePoolObjClearVols(pool);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Now get basic volgrp metadata */
|
2010-02-10 11:42:56 +00:00
|
|
|
if (virStorageBackendRunProgRegex(pool,
|
2008-02-20 15:45:33 +00:00
|
|
|
prog,
|
|
|
|
1,
|
|
|
|
regexes,
|
|
|
|
vars,
|
|
|
|
virStorageBackendLogicalRefreshPoolFunc,
|
2011-05-10 15:36:50 -04:00
|
|
|
NULL) < 0) {
|
2008-02-20 15:45:33 +00:00
|
|
|
virStoragePoolObjClearVols(pool);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2008-09-22 20:22:35 +00:00
|
|
|
/*
|
|
|
|
* This is actually relatively safe; if you happen to try to "stop" the
|
|
|
|
* pool that your / is on, for instance, you will get failure like:
|
|
|
|
* "Can't deactivate volume group "VolGroup00" with 3 open logical volume(s)"
|
2008-02-20 15:45:33 +00:00
|
|
|
*/
|
|
|
|
static int
|
2010-02-04 23:41:52 +01:00
|
|
|
virStorageBackendLogicalStopPool(virConnectPtr conn ATTRIBUTE_UNUSED,
|
2008-02-20 15:45:33 +00:00
|
|
|
virStoragePoolObjPtr pool)
|
|
|
|
{
|
2010-02-04 23:41:52 +01:00
|
|
|
if (virStorageBackendLogicalSetActive(pool, 0) < 0)
|
2008-02-20 15:45:33 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
2010-02-04 23:41:52 +01:00
|
|
|
virStorageBackendLogicalDeletePool(virConnectPtr conn ATTRIBUTE_UNUSED,
|
2008-02-20 15:45:33 +00:00
|
|
|
virStoragePoolObjPtr pool,
|
2011-07-06 16:51:23 -06:00
|
|
|
unsigned int flags)
|
2008-02-20 15:45:33 +00:00
|
|
|
{
|
|
|
|
const char *cmdargv[] = {
|
2008-09-23 20:48:49 +00:00
|
|
|
VGREMOVE, "-f", pool->def->source.name, NULL
|
2008-02-20 15:45:33 +00:00
|
|
|
};
|
2008-09-22 20:22:35 +00:00
|
|
|
const char *pvargv[3];
|
|
|
|
int i, error;
|
2008-02-20 15:45:33 +00:00
|
|
|
|
2011-07-06 16:51:23 -06:00
|
|
|
virCheckFlags(0, -1);
|
|
|
|
|
2008-09-22 20:22:35 +00:00
|
|
|
/* first remove the volume group */
|
2010-02-04 23:41:52 +01:00
|
|
|
if (virRun(cmdargv, NULL) < 0)
|
2008-02-20 15:45:33 +00:00
|
|
|
return -1;
|
|
|
|
|
2008-09-22 20:22:35 +00:00
|
|
|
/* now remove the pv devices and clear them out */
|
|
|
|
error = 0;
|
|
|
|
pvargv[0] = PVREMOVE;
|
|
|
|
pvargv[2] = NULL;
|
|
|
|
for (i = 0 ; i < pool->def->source.ndevice ; i++) {
|
|
|
|
pvargv[1] = pool->def->source.devices[i].path;
|
2010-02-04 23:41:52 +01:00
|
|
|
if (virRun(pvargv, NULL) < 0) {
|
2008-09-22 20:22:35 +00:00
|
|
|
error = -1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2008-02-20 15:45:33 +00:00
|
|
|
|
2008-09-22 20:22:35 +00:00
|
|
|
return error;
|
2008-02-20 15:45:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
virStorageBackendLogicalDeleteVol(virConnectPtr conn,
|
|
|
|
virStoragePoolObjPtr pool,
|
|
|
|
virStorageVolDefPtr vol,
|
|
|
|
unsigned int flags);
|
|
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
virStorageBackendLogicalCreateVol(virConnectPtr conn,
|
|
|
|
virStoragePoolObjPtr pool,
|
|
|
|
virStorageVolDefPtr vol)
|
|
|
|
{
|
storage: Check for invalid storage mode before opening
If a directory pool contains pipes or sockets, a pool start can fail or hang:
https://bugzilla.redhat.com/show_bug.cgi?id=589577
We already try to avoid these special files, but only attempt after
opening the path, which is where the problems lie. Unify volume opening
into helper functions, which use the proper open() flags to avoid error,
followed by fstat to validate storage mode.
Previously, virStorageBackendUpdateVolTargetInfoFD attempted to enforce the
storage mode check, but allowed callers to detect this case and silently
continue. In practice, only the FS backend was using this feature, the rest
were treating unknown mode as an error condition. Unfortunately the InfoFD
function wasn't raising an error message here, so error reporting was
busted.
This patch adds 2 functions: virStorageBackendVolOpen, and
virStorageBackendVolOpenModeSkip. The latter retains the original opt out
semantics, the former now throws an explicit error.
This patch maintains the previous volume mode checks: allowing specific
modes for specific pool types requires a bit of surgery, since VolOpen
is called through several different helper functions.
v2: Use ATTRIBUTE_NONNULL. Drop stat check, just open with
O_NONBLOCK|O_NOCTTY.
v3: Move mode check logic back to VolOpen. Use 2 VolOpen functions with
different error semantics.
v4: Make second VolOpen function more extensible. Didn't opt to change
FS backend defaults, this can just be to fix the original bug.
v5: Prefix default flags with VIR_, use ATTRIBUTE_RETURN_CHECK
2010-05-20 14:25:01 -04:00
|
|
|
int fdret, fd = -1;
|
2008-02-20 15:45:33 +00:00
|
|
|
char size[100];
|
2009-01-27 18:30:03 +00:00
|
|
|
const char *cmdargvnew[] = {
|
2008-02-20 15:45:33 +00:00
|
|
|
LVCREATE, "--name", vol->name, "-L", size,
|
|
|
|
pool->def->target.path, NULL
|
|
|
|
};
|
2009-01-27 18:30:03 +00:00
|
|
|
const char *cmdargvsnap[] = {
|
|
|
|
LVCREATE, "--name", vol->name, "-L", size,
|
|
|
|
"-s", vol->backingStore.path, NULL
|
|
|
|
};
|
|
|
|
const char **cmdargv = cmdargvnew;
|
|
|
|
|
2009-07-21 04:40:50 +02:00
|
|
|
if (vol->target.encryption != NULL) {
|
2011-09-01 15:24:59 +08:00
|
|
|
virStorageReportError(VIR_ERR_CONFIG_UNSUPPORTED,
|
2009-07-21 04:40:50 +02:00
|
|
|
"%s", _("storage pool does not support encrypted "
|
|
|
|
"volumes"));
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2009-01-27 18:30:03 +00:00
|
|
|
if (vol->backingStore.path) {
|
|
|
|
cmdargv = cmdargvsnap;
|
|
|
|
}
|
2008-02-20 15:45:33 +00:00
|
|
|
|
2011-01-28 22:03:24 +01:00
|
|
|
snprintf(size, sizeof(size)-1, "%lluK", VIR_DIV_UP(vol->capacity, 1024));
|
2008-02-20 15:45:33 +00:00
|
|
|
size[sizeof(size)-1] = '\0';
|
|
|
|
|
2008-11-17 11:19:33 +00:00
|
|
|
vol->type = VIR_STORAGE_VOL_BLOCK;
|
|
|
|
|
2008-08-20 13:33:01 +00:00
|
|
|
if (vol->target.path != NULL) {
|
|
|
|
/* A target path passed to CreateVol has no meaning */
|
|
|
|
VIR_FREE(vol->target.path);
|
|
|
|
}
|
2009-05-19 13:15:50 +00:00
|
|
|
|
|
|
|
if (virAsprintf(&vol->target.path, "%s/%s",
|
|
|
|
pool->def->target.path,
|
|
|
|
vol->name) == -1) {
|
2010-02-04 19:19:08 +01:00
|
|
|
virReportOOMError();
|
2008-08-20 13:33:01 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2010-02-04 23:41:52 +01:00
|
|
|
if (virRun(cmdargv, NULL) < 0)
|
2008-02-20 15:45:33 +00:00
|
|
|
return -1;
|
|
|
|
|
storage: Check for invalid storage mode before opening
If a directory pool contains pipes or sockets, a pool start can fail or hang:
https://bugzilla.redhat.com/show_bug.cgi?id=589577
We already try to avoid these special files, but only attempt after
opening the path, which is where the problems lie. Unify volume opening
into helper functions, which use the proper open() flags to avoid error,
followed by fstat to validate storage mode.
Previously, virStorageBackendUpdateVolTargetInfoFD attempted to enforce the
storage mode check, but allowed callers to detect this case and silently
continue. In practice, only the FS backend was using this feature, the rest
were treating unknown mode as an error condition. Unfortunately the InfoFD
function wasn't raising an error message here, so error reporting was
busted.
This patch adds 2 functions: virStorageBackendVolOpen, and
virStorageBackendVolOpenModeSkip. The latter retains the original opt out
semantics, the former now throws an explicit error.
This patch maintains the previous volume mode checks: allowing specific
modes for specific pool types requires a bit of surgery, since VolOpen
is called through several different helper functions.
v2: Use ATTRIBUTE_NONNULL. Drop stat check, just open with
O_NONBLOCK|O_NOCTTY.
v3: Move mode check logic back to VolOpen. Use 2 VolOpen functions with
different error semantics.
v4: Make second VolOpen function more extensible. Didn't opt to change
FS backend defaults, this can just be to fix the original bug.
v5: Prefix default flags with VIR_, use ATTRIBUTE_RETURN_CHECK
2010-05-20 14:25:01 -04:00
|
|
|
if ((fdret = virStorageBackendVolOpen(vol->target.path)) < 0)
|
2008-02-20 15:45:33 +00:00
|
|
|
goto cleanup;
|
storage: Check for invalid storage mode before opening
If a directory pool contains pipes or sockets, a pool start can fail or hang:
https://bugzilla.redhat.com/show_bug.cgi?id=589577
We already try to avoid these special files, but only attempt after
opening the path, which is where the problems lie. Unify volume opening
into helper functions, which use the proper open() flags to avoid error,
followed by fstat to validate storage mode.
Previously, virStorageBackendUpdateVolTargetInfoFD attempted to enforce the
storage mode check, but allowed callers to detect this case and silently
continue. In practice, only the FS backend was using this feature, the rest
were treating unknown mode as an error condition. Unfortunately the InfoFD
function wasn't raising an error message here, so error reporting was
busted.
This patch adds 2 functions: virStorageBackendVolOpen, and
virStorageBackendVolOpenModeSkip. The latter retains the original opt out
semantics, the former now throws an explicit error.
This patch maintains the previous volume mode checks: allowing specific
modes for specific pool types requires a bit of surgery, since VolOpen
is called through several different helper functions.
v2: Use ATTRIBUTE_NONNULL. Drop stat check, just open with
O_NONBLOCK|O_NOCTTY.
v3: Move mode check logic back to VolOpen. Use 2 VolOpen functions with
different error semantics.
v4: Make second VolOpen function more extensible. Didn't opt to change
FS backend defaults, this can just be to fix the original bug.
v5: Prefix default flags with VIR_, use ATTRIBUTE_RETURN_CHECK
2010-05-20 14:25:01 -04:00
|
|
|
fd = fdret;
|
2008-02-20 15:45:33 +00:00
|
|
|
|
|
|
|
/* We can only chown/grp if root */
|
|
|
|
if (getuid() == 0) {
|
|
|
|
if (fchown(fd, vol->target.perms.uid, vol->target.perms.gid) < 0) {
|
2010-02-04 21:02:58 +01:00
|
|
|
virReportSystemError(errno,
|
2009-01-20 17:13:33 +00:00
|
|
|
_("cannot set file owner '%s'"),
|
|
|
|
vol->target.path);
|
2008-02-20 15:45:33 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (fchmod(fd, vol->target.perms.mode) < 0) {
|
2010-02-04 21:02:58 +01:00
|
|
|
virReportSystemError(errno,
|
2009-01-20 17:13:33 +00:00
|
|
|
_("cannot set file mode '%s'"),
|
|
|
|
vol->target.path);
|
2008-02-20 15:45:33 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
2010-10-28 11:43:16 -04:00
|
|
|
if (VIR_CLOSE(fd) < 0) {
|
2010-02-04 21:02:58 +01:00
|
|
|
virReportSystemError(errno,
|
2009-01-20 17:13:33 +00:00
|
|
|
_("cannot close file '%s'"),
|
|
|
|
vol->target.path);
|
2008-02-20 15:45:33 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
fd = -1;
|
|
|
|
|
|
|
|
/* Fill in data about this new vol */
|
2010-02-10 11:42:56 +00:00
|
|
|
if (virStorageBackendLogicalFindLVs(pool, vol) < 0) {
|
2010-02-04 21:02:58 +01:00
|
|
|
virReportSystemError(errno,
|
2009-01-20 17:13:33 +00:00
|
|
|
_("cannot find newly created volume '%s'"),
|
|
|
|
vol->target.path);
|
2008-02-20 15:45:33 +00:00
|
|
|
goto cleanup;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
cleanup:
|
2010-10-28 11:43:16 -04:00
|
|
|
VIR_FORCE_CLOSE(fd);
|
2008-02-20 15:45:33 +00:00
|
|
|
virStorageBackendLogicalDeleteVol(conn, pool, vol, 0);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2009-07-09 19:17:31 -04:00
|
|
|
static int
|
|
|
|
virStorageBackendLogicalBuildVolFrom(virConnectPtr conn,
|
2010-01-21 00:41:52 +01:00
|
|
|
virStoragePoolObjPtr pool,
|
2009-07-09 19:17:31 -04: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 19:17:31 -04:00
|
|
|
if (!build_func)
|
|
|
|
return -1;
|
|
|
|
|
2010-01-21 00:41:52 +01:00
|
|
|
return build_func(conn, pool, vol, inputvol, flags);
|
2009-07-09 19:17:31 -04:00
|
|
|
}
|
|
|
|
|
2008-02-20 15:45:33 +00:00
|
|
|
static int
|
2010-02-04 23:41:52 +01:00
|
|
|
virStorageBackendLogicalDeleteVol(virConnectPtr conn ATTRIBUTE_UNUSED,
|
2008-02-20 15:45:33 +00:00
|
|
|
virStoragePoolObjPtr pool ATTRIBUTE_UNUSED,
|
|
|
|
virStorageVolDefPtr vol,
|
2011-07-06 16:51:23 -06:00
|
|
|
unsigned int flags)
|
2008-02-20 15:45:33 +00:00
|
|
|
{
|
|
|
|
const char *cmdargv[] = {
|
|
|
|
LVREMOVE, "-f", vol->target.path, NULL
|
|
|
|
};
|
|
|
|
|
2011-07-06 16:51:23 -06:00
|
|
|
virCheckFlags(0, -1);
|
|
|
|
|
2010-02-04 23:41:52 +01:00
|
|
|
if (virRun(cmdargv, NULL) < 0)
|
2008-02-20 15:45:33 +00:00
|
|
|
return -1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
virStorageBackend virStorageBackendLogical = {
|
|
|
|
.type = VIR_STORAGE_POOL_LOGICAL,
|
|
|
|
|
2008-08-27 20:05:58 +00:00
|
|
|
.findPoolSources = virStorageBackendLogicalFindPoolSources,
|
2010-11-11 20:09:20 +00:00
|
|
|
.checkPool = virStorageBackendLogicalCheckPool,
|
2008-02-20 15:45:33 +00:00
|
|
|
.startPool = virStorageBackendLogicalStartPool,
|
|
|
|
.buildPool = virStorageBackendLogicalBuildPool,
|
|
|
|
.refreshPool = virStorageBackendLogicalRefreshPool,
|
|
|
|
.stopPool = virStorageBackendLogicalStopPool,
|
|
|
|
.deletePool = virStorageBackendLogicalDeletePool,
|
2009-07-09 19:17:31 -04:00
|
|
|
.buildVol = NULL,
|
|
|
|
.buildVolFrom = virStorageBackendLogicalBuildVolFrom,
|
2008-02-20 15:45:33 +00:00
|
|
|
.createVol = virStorageBackendLogicalCreateVol,
|
|
|
|
.deleteVol = virStorageBackendLogicalDeleteVol,
|
|
|
|
};
|