libvirt/src/conf/virdomaincheckpointobjlist.c
Eric Blake 2795d85647 backup: Allow for lists of checkpoint objects
Create a new file for managing a list of checkpoint objects, borrowing
heavily from existing virDomainSnapshotObjList paradigms.

Note that while snapshots definitely have a use case for multiple
children to a single parent (create a base snapshot, create a child
snapshot, revert to the base, then create another child snapshot),
it's harder to predict how checkpoints will play out with reverting to
prior points in time. Thus, in initial use, given a list of
checkpoints, you never have more than one child, and we can treat the
most-recent leaf node as the parent of the next node creation, without
having to expose a notion of a current node in XML or public API.
However, as the snapshot machinery is already generic, it is easier to
reuse the generic machinery that tracks relations between domain
moments than it is to open-code a new list-management scheme just for
checkpoints (hence, we still have internal functions related to a
current checkpoint, even though that has no observable effect
externally, as well as the addition of a function to easily find the
lone leaf in the list to use as the current checkpoint).

Signed-off-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Daniel P. Berrangé <berrange@redhat.com>
2019-07-26 16:48:58 -05:00

244 lines
7.2 KiB
C

/*
* virdomaincheckpointobjlist.c: handle a tree of checkpoint objects
* (derived from virdomainsnapshotobjlist.c)
*
* Copyright (C) 2006-2019 Red Hat, Inc.
* Copyright (C) 2006-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, see
* <http://www.gnu.org/licenses/>.
*/
#include <config.h>
#include "internal.h"
#include "virdomaincheckpointobjlist.h"
#include "checkpoint_conf.h"
#include "virlog.h"
#include "virerror.h"
#include "datatypes.h"
#include "virstring.h"
#include "viralloc.h"
#define VIR_FROM_THIS VIR_FROM_DOMAIN_CHECKPOINT
VIR_LOG_INIT("conf.virdomaincheckpointobjlist");
struct _virDomainCheckpointObjList {
virDomainMomentObjListPtr base;
};
virDomainMomentObjPtr
virDomainCheckpointAssignDef(virDomainCheckpointObjListPtr checkpoints,
virDomainCheckpointDefPtr def)
{
return virDomainMomentAssignDef(checkpoints->base, &def->parent);
}
static bool
virDomainCheckpointFilter(virDomainMomentObjPtr obj ATTRIBUTE_UNUSED,
unsigned int flags)
{
/* For now, we have no further filters than what the common code handles. */
virCheckFlags(0, false);
return true;
}
virDomainCheckpointObjListPtr
virDomainCheckpointObjListNew(void)
{
virDomainCheckpointObjListPtr checkpoints;
if (VIR_ALLOC(checkpoints) < 0)
return NULL;
checkpoints->base = virDomainMomentObjListNew();
if (!checkpoints->base) {
VIR_FREE(checkpoints);
return NULL;
}
return checkpoints;
}
void
virDomainCheckpointObjListFree(virDomainCheckpointObjListPtr checkpoints)
{
if (!checkpoints)
return;
virDomainMomentObjListFree(checkpoints->base);
VIR_FREE(checkpoints);
}
static int
virDomainCheckpointObjListGetNames(virDomainCheckpointObjListPtr checkpoints,
virDomainMomentObjPtr from,
char **const names,
int maxnames,
unsigned int flags)
{
/* We intentionally chose our public flags to match the common flags */
verify(VIR_DOMAIN_CHECKPOINT_LIST_ROOTS ==
(int) VIR_DOMAIN_MOMENT_LIST_ROOTS);
verify(VIR_DOMAIN_CHECKPOINT_LIST_TOPOLOGICAL ==
(int) VIR_DOMAIN_MOMENT_LIST_TOPOLOGICAL);
verify(VIR_DOMAIN_CHECKPOINT_LIST_LEAVES ==
(int) VIR_DOMAIN_MOMENT_LIST_LEAVES);
verify(VIR_DOMAIN_CHECKPOINT_LIST_NO_LEAVES ==
(int) VIR_DOMAIN_MOMENT_LIST_NO_LEAVES);
return virDomainMomentObjListGetNames(checkpoints->base, from, names,
maxnames, flags,
virDomainCheckpointFilter, 0);
}
virDomainMomentObjPtr
virDomainCheckpointFindByName(virDomainCheckpointObjListPtr checkpoints,
const char *name)
{
return virDomainMomentFindByName(checkpoints->base, name);
}
/* Return the current checkpoint, or NULL */
virDomainMomentObjPtr
virDomainCheckpointGetCurrent(virDomainCheckpointObjListPtr checkpoints)
{
return virDomainMomentGetCurrent(checkpoints->base);
}
/* Return the current checkpoint's name, or NULL */
const char *
virDomainCheckpointGetCurrentName(virDomainCheckpointObjListPtr checkpoints)
{
return virDomainMomentGetCurrentName(checkpoints->base);
}
/* Update the current checkpoint, using NULL if no current remains */
void
virDomainCheckpointSetCurrent(virDomainCheckpointObjListPtr checkpoints,
virDomainMomentObjPtr checkpoint)
{
virDomainMomentSetCurrent(checkpoints->base, checkpoint);
}
/* Remove checkpoint from the list; return true if it was current */
bool
virDomainCheckpointObjListRemove(virDomainCheckpointObjListPtr checkpoints,
virDomainMomentObjPtr checkpoint)
{
return virDomainMomentObjListRemove(checkpoints->base, checkpoint);
}
/* Remove all checkpoints tracked in the list */
void
virDomainCheckpointObjListRemoveAll(virDomainCheckpointObjListPtr checkpoints)
{
return virDomainMomentObjListRemoveAll(checkpoints->base);
}
int
virDomainCheckpointForEach(virDomainCheckpointObjListPtr checkpoints,
virHashIterator iter,
void *data)
{
return virDomainMomentForEach(checkpoints->base, iter, data);
}
/* Populate parent link of a given checkpoint. */
void
virDomainCheckpointLinkParent(virDomainCheckpointObjListPtr checkpoints,
virDomainMomentObjPtr chk)
{
return virDomainMomentLinkParent(checkpoints->base, chk);
}
/* Populate parent link and child count of all checkpoints, with all
* assigned defs having relations starting as 0/NULL. Return 0 on
* success, -1 if a parent is missing or if a circular relationship
* was requested. Set leaf to the end of the chain, if there is
* exactly one such leaf. */
int
virDomainCheckpointUpdateRelations(virDomainCheckpointObjListPtr checkpoints,
virDomainMomentObjPtr *leaf)
{
int ret = virDomainMomentUpdateRelations(checkpoints->base);
if (ret == 0)
*leaf = virDomainMomentFindLeaf(checkpoints->base);
return ret;
}
int
virDomainCheckpointCheckCycles(virDomainCheckpointObjListPtr checkpoints,
virDomainCheckpointDefPtr def,
const char *domname)
{
return virDomainMomentCheckCycles(checkpoints->base, &def->parent, domname);
}
int
virDomainListCheckpoints(virDomainCheckpointObjListPtr checkpoints,
virDomainMomentObjPtr from,
virDomainPtr dom,
virDomainCheckpointPtr **chks,
unsigned int flags)
{
int count = virDomainCheckpointObjListGetNames(checkpoints, from, NULL,
0, flags);
virDomainCheckpointPtr *list = NULL;
char **names;
int ret = -1;
size_t i;
if (!chks || count < 0)
return count;
if (VIR_ALLOC_N(names, count) < 0 ||
VIR_ALLOC_N(list, count + 1) < 0)
goto cleanup;
if (virDomainCheckpointObjListGetNames(checkpoints, from, names, count,
flags) < 0)
goto cleanup;
for (i = 0; i < count; i++)
if ((list[i] = virGetDomainCheckpoint(dom, names[i])) == NULL)
goto cleanup;
ret = count;
*chks = list;
cleanup:
for (i = 0; i < count; i++)
VIR_FREE(names[i]);
VIR_FREE(names);
if (ret < 0 && list) {
for (i = 0; i < count; i++)
virObjectUnref(list[i]);
VIR_FREE(list);
}
return ret;
}