/*
* 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
* .
*/
#include
#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;
}