summaryrefslogtreecommitdiffstats
path: root/daemons
diff options
context:
space:
mode:
authorPetr Rockai <prockai@redhat.com>2011-07-25 17:59:50 +0000
committerPetr Rockai <prockai@redhat.com>2011-07-25 17:59:50 +0000
commit1d9c083fcca6456016acba76ed8e512955872b06 (patch)
tree0246c15820fda466f50fc315d4a45e7109eaa21c /daemons
parent059ee25951458128a119581c62703d83961b6097 (diff)
downloadlvm2-1d9c083fcca6456016acba76ed8e512955872b06.tar.gz
lvm2-1d9c083fcca6456016acba76ed8e512955872b06.tar.xz
lvm2-1d9c083fcca6456016acba76ed8e512955872b06.zip
lvmetad: Edit the MISSING_PV flags only after making a "reply" copy of the
metadata, which is then serialised and discarded. This fixes a couple of outstanding TODO items about handling the MISSING flags correctly.
Diffstat (limited to 'daemons')
-rw-r--r--daemons/lvmetad/lvmetad-core.c172
1 files changed, 86 insertions, 86 deletions
diff --git a/daemons/lvmetad/lvmetad-core.c b/daemons/lvmetad/lvmetad-core.c
index 4dfb0a27..eecc146c 100644
--- a/daemons/lvmetad/lvmetad-core.c
+++ b/daemons/lvmetad/lvmetad-core.c
@@ -65,6 +65,89 @@ void unlock_vg(lvmetad_state *s, const char *id) {
unlock_vgs(s);
}
+static struct config_node *pvs(struct config_node *vg)
+{
+ struct config_node *pv = find_config_node(vg, "metadata/physical_volumes");
+ if (pv)
+ pv = pv->child;
+ return pv;
+}
+
+/*
+ * TODO: This set_flag function is pretty generic and might make sense in a
+ * library here or there.
+ */
+static void set_flag(struct config_tree *cft, struct config_node *parent,
+ char *field, const char *flag, int want) {
+ struct config_value *value = NULL, *pred = NULL;
+ struct config_node *node = find_config_node(parent->child, field);
+ int found = 0;
+
+ if (node)
+ value = node->v;
+
+ while (value && value->type != CFG_EMPTY_ARRAY && strcmp(value->v.str, flag)) {
+ pred = value;
+ value = value->next;
+ }
+
+ if (value && want)
+ return;
+
+ if (!value && !want)
+ return;
+
+ if (value && !want) {
+ if (pred)
+ pred->next = value->next;
+ if (value == node->v)
+ node->v = value->next;
+ }
+
+ if (!value && want) {
+ if (!node) {
+ node = create_config_node(cft, field);
+ node->sib = parent->child;
+ node->v = create_config_value(cft);
+ node->v->type = CFG_EMPTY_ARRAY;
+ node->parent = parent;
+ parent->child = node;
+ }
+ struct config_value *new = create_config_value(cft);
+ new->type = CFG_STRING;
+ new->v.str = flag;
+ new->next = node->v;
+ node->v = new;
+ }
+}
+
+/* Either the "big" vgs lock, or a per-vg lock needs to be held before entering
+ * this function. */
+static int update_pv_status(lvmetad_state *s, struct config_tree *cft, struct config_node *vg, int act)
+{
+ int complete = 1;
+
+ lock_pvs(s);
+ struct config_node *pv = pvs(vg);
+ while (pv) {
+ const char *uuid = find_config_str(pv->child, "id", NULL);
+ int found = uuid ? (dm_hash_lookup(s->pvs, uuid) ? 1 : 0) : 0;
+ if (act)
+ set_flag(cft, pv, "status", "MISSING", !found);
+ if (!found) {
+ complete = 0;
+ if (!act) { // optimisation
+ unlock_pvs(s);
+ return complete;
+ }
+ }
+ pv = pv->sib;
+ }
+ unlock_pvs(s);
+
+ return complete;
+}
+
static response vg_by_uuid(lvmetad_state *s, request r)
{
const char *uuid = daemon_request_str(r, "uuid", "NONE");
@@ -92,14 +175,11 @@ static response vg_by_uuid(lvmetad_state *s, request r)
res.error = 0;
unlock_vg(s, uuid);
+ update_pv_status(s, cft, n, 1);
+
return res;
}
-/*
- * TODO: Accept different flag permutations? Or else, ensure fixed ordering of
- * flags in set_flag below (following the same order as we have in
- * lib/format_text/flags.c).
- */
static int compare_value(struct config_value *a, struct config_value *b)
{
if (a->type > b->type)
@@ -143,85 +223,6 @@ static int compare_config(struct config_node *a, struct config_node *b)
return result;
}
-/*
- * TODO: This set_flag function is pretty generic and might make sense in a
- * library here or there.
- */
-static void set_flag(struct config_tree *cft, struct config_node *parent,
- char *field, const char *flag, int want) {
- struct config_value *value = NULL, *pred = NULL;
- struct config_node *node = find_config_node(parent->child, field);
- int found = 0;
-
- if (node)
- value = node->v;
-
- while (value && value->type != CFG_EMPTY_ARRAY && strcmp(value->v.str, flag)) {
- pred = value;
- value = value->next;
- }
-
- if (value && want)
- return;
-
- if (!value && !want)
- return;
-
- if (value && !want) {
- if (pred)
- pred->next = value->next;
- if (value == node->v)
- node->v = value->next;
- }
-
- if (!value && want) {
- if (!node) {
- node = create_config_node(cft, field);
- node->sib = parent->child;
- node->v = create_config_value(cft);
- node->v->type = CFG_EMPTY_ARRAY;
- node->parent = parent;
- parent->child = node;
- }
- struct config_value *new = create_config_value(cft);
- new->type = CFG_STRING;
- new->v.str = flag;
- new->next = node->v;
- node->v = new;
- }
-}
-
-struct config_node *pvs(struct config_tree *vg)
-{
- struct config_node *pv = find_config_node(vg->root, "metadata/physical_volumes");
- if (pv)
- pv = pv->child;
- return pv;
-}
-
-/* Either the "big" vgs lock, or a per-vg lock needs to be held before entering
- * this function. */
-static int update_pv_status(lvmetad_state *s, struct config_tree *vg)
-{
- int complete = 1;
-
- lock_pvs(s);
- struct config_node *pv = pvs(vg);
- while (pv) {
- const char *uuid = find_config_str(pv->child, "id", NULL);
- int found = uuid ? (dm_hash_lookup(s->pvs, uuid) ? 1 : 0) : 0;
- // TODO: avoid the override here if MISSING came from the actual
- // metadata, as opposed from our manipulation...
- set_flag(vg, pv, "status", "MISSING", !found);
- if (!found)
- complete = 0;
- pv = pv->sib;
- }
- unlock_pvs(s);
-
- return complete;
-}
-
/* You need to be holding the pvid_map lock already to call this. */
int update_pvid_map(lvmetad_state *s, struct config_tree *vg, const char *vgid)
{
@@ -261,7 +262,6 @@ static int update_metadata(lvmetad_state *s, const char *_vgid, struct config_no
if (seq == haveseq) {
retval = 1;
- // TODO: disregard the MISSING_PV flag status in this comparison
if (compare_config(metadata, old->root))
retval = 0;
goto out;
@@ -337,7 +337,7 @@ static response pv_add(lvmetad_state *s, request r)
int complete = 0;
if (vgid) {
struct config_tree *cft = lock_vg(s, vgid);
- complete = update_pv_status(s, cft);
+ complete = update_pv_status(s, cft, cft->root, 0);
unlock_vg(s, vgid);
}