mirror of
https://gitlab.com/libvirt/libvirt.git
synced 2025-01-12 15:52:55 +00:00
storage: Introduce storagePoolUpdateAllState function
The 'checkPool' callback was originally part of the storageDriverAutostart function, but the pools need to be checked earlier during initialization phase, otherwise we can't start a domain which mounts a volume after the libvirtd daemon restarted. This is because qemuProcessReconnect is called earlier than storageDriverAutostart. Therefore the 'checkPool' logic has been moved to storagePoolUpdateAllState which is called inside storageDriverInitialize. We also need a valid 'conn' reference to be able to execute 'refreshPool' during initialization phase. Though it isn't available until storageDriverAutostart all of our storage backends do ignore 'conn' pointer, except for RBD, but RBD doesn't support 'checkPool' callback, so it's safe to pass conn = NULL in this case. Resolves: https://bugzilla.redhat.com/show_bug.cgi?id=1177733
This commit is contained in:
parent
a9700771f5
commit
2a31c5f030
@ -74,6 +74,65 @@ static void storageDriverUnlock(void)
|
|||||||
virMutexUnlock(&driver->lock);
|
virMutexUnlock(&driver->lock);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void
|
||||||
|
storagePoolUpdateAllState(void)
|
||||||
|
{
|
||||||
|
size_t i;
|
||||||
|
bool active;
|
||||||
|
|
||||||
|
for (i = 0; i < driver->pools.count; i++) {
|
||||||
|
virStoragePoolObjPtr pool = driver->pools.objs[i];
|
||||||
|
virStorageBackendPtr backend;
|
||||||
|
|
||||||
|
virStoragePoolObjLock(pool);
|
||||||
|
if (!virStoragePoolObjIsActive(pool)) {
|
||||||
|
virStoragePoolObjUnlock(pool);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ((backend = virStorageBackendForType(pool->def->type)) == NULL) {
|
||||||
|
VIR_ERROR(_("Missing backend %d"), pool->def->type);
|
||||||
|
virStoragePoolObjUnlock(pool);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Backends which do not support 'checkPool' are considered
|
||||||
|
* inactive by default.
|
||||||
|
*/
|
||||||
|
active = false;
|
||||||
|
if (backend->checkPool &&
|
||||||
|
backend->checkPool(pool, &active) < 0) {
|
||||||
|
virErrorPtr err = virGetLastError();
|
||||||
|
VIR_ERROR(_("Failed to initialize storage pool '%s': %s"),
|
||||||
|
pool->def->name, err ? err->message :
|
||||||
|
_("no error message found"));
|
||||||
|
virStoragePoolObjUnlock(pool);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* We can pass NULL as connection, most backends do not use
|
||||||
|
* it anyway, but if they do and fail, we want to log error and
|
||||||
|
* continue with other pools.
|
||||||
|
*/
|
||||||
|
if (active) {
|
||||||
|
virStoragePoolObjClearVols(pool);
|
||||||
|
if (backend->refreshPool(NULL, pool) < 0) {
|
||||||
|
virErrorPtr err = virGetLastError();
|
||||||
|
if (backend->stopPool)
|
||||||
|
backend->stopPool(NULL, pool);
|
||||||
|
VIR_ERROR(_("Failed to restart storage pool '%s': %s"),
|
||||||
|
pool->def->name, err ? err->message :
|
||||||
|
_("no error message found"));
|
||||||
|
virStoragePoolObjUnlock(pool);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pool->active = active;
|
||||||
|
virStoragePoolObjUnlock(pool);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
storageDriverAutostart(void)
|
storageDriverAutostart(void)
|
||||||
{
|
{
|
||||||
@ -95,23 +154,11 @@ storageDriverAutostart(void)
|
|||||||
|
|
||||||
virStoragePoolObjLock(pool);
|
virStoragePoolObjLock(pool);
|
||||||
if ((backend = virStorageBackendForType(pool->def->type)) == NULL) {
|
if ((backend = virStorageBackendForType(pool->def->type)) == NULL) {
|
||||||
VIR_ERROR(_("Missing backend %d"), pool->def->type);
|
|
||||||
virStoragePoolObjUnlock(pool);
|
virStoragePoolObjUnlock(pool);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (backend->checkPool &&
|
if (pool->autostart &&
|
||||||
backend->checkPool(pool, &started) < 0) {
|
|
||||||
virErrorPtr err = virGetLastError();
|
|
||||||
VIR_ERROR(_("Failed to initialize storage pool '%s': %s"),
|
|
||||||
pool->def->name, err ? err->message :
|
|
||||||
_("no error message found"));
|
|
||||||
virStoragePoolObjUnlock(pool);
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!started &&
|
|
||||||
pool->autostart &&
|
|
||||||
!virStoragePoolObjIsActive(pool)) {
|
!virStoragePoolObjIsActive(pool)) {
|
||||||
if (backend->startPool &&
|
if (backend->startPool &&
|
||||||
backend->startPool(conn, pool) < 0) {
|
backend->startPool(conn, pool) < 0) {
|
||||||
@ -213,6 +260,8 @@ storageStateInitialize(bool privileged,
|
|||||||
driver->autostartDir) < 0)
|
driver->autostartDir) < 0)
|
||||||
goto error;
|
goto error;
|
||||||
|
|
||||||
|
storagePoolUpdateAllState();
|
||||||
|
|
||||||
storageDriverUnlock();
|
storageDriverUnlock();
|
||||||
|
|
||||||
ret = 0;
|
ret = 0;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user