mirror of
git://sourceware.org/git/lvm2.git
synced 2024-12-22 17:35:59 +03:00
25b733809a
Lots of changes/very little testing so far => there'll be bugs! Use 'vgcreate -M text' to create a volume group with its metadata stored in text files. Text format metadata changes should be reasonably atomic, with a (basic) automatic recovery mechanism if the system crashes while a change is in progress. Add a metadata section to lvm.conf to specify multiple directories if you want (recommended) to keep multiple copies of the metadata (eg on different filesystems). e.g. metadata { dirs = ["/etc/lvm/metadata1","/usr/local/lvm/metadata2"] } Plenty of refinements still in the pipeline.
161 lines
2.7 KiB
C
161 lines
2.7 KiB
C
/*
|
|
* Copyright (C) 2001 Sistina Software (UK) Limited.
|
|
*
|
|
* This file is released under the LGPL.
|
|
*/
|
|
|
|
#include "log.h"
|
|
#include "metadata.h"
|
|
#include "import-export.h"
|
|
#include "lvm-string.h"
|
|
|
|
/*
|
|
* Bitsets held in the 'status' flags get
|
|
* converted into arrays of strings.
|
|
*/
|
|
struct flag {
|
|
int mask;
|
|
char *description;
|
|
};
|
|
|
|
static struct flag _vg_flags[] = {
|
|
{EXPORTED_VG, "EXPORTED"},
|
|
{RESIZEABLE_VG, "RESIZEABLE"},
|
|
{PARTIAL_VG, "PARTIAL"},
|
|
{LVM_READ, "READ"},
|
|
{LVM_WRITE, "WRITE"},
|
|
{CLUSTERED, "CLUSTERED"},
|
|
{SHARED, "SHARED"},
|
|
{0, NULL}
|
|
};
|
|
|
|
static struct flag _pv_flags[] = {
|
|
{ALLOCATABLE_PV, "ALLOCATABLE"},
|
|
{EXPORTED_VG, "EXPORTED"},
|
|
{0, NULL}
|
|
};
|
|
|
|
static struct flag _lv_flags[] = {
|
|
{LVM_READ, "READ"},
|
|
{LVM_WRITE, "WRITE"},
|
|
{ALLOC_SIMPLE, "ALLOC_SIMPLE"},
|
|
{ALLOC_STRICT, "ALLOC_STRICT"},
|
|
{ALLOC_CONTIGUOUS, "ALLOC_CONTIGUOUS"},
|
|
{FIXED_MINOR, "FIXED_MINOR"},
|
|
{0, NULL}
|
|
};
|
|
|
|
static struct flag *_get_flags(int type)
|
|
{
|
|
switch (type) {
|
|
case VG_FLAGS:
|
|
return _vg_flags;
|
|
|
|
case PV_FLAGS:
|
|
return _pv_flags;
|
|
|
|
case LV_FLAGS:
|
|
return _lv_flags;
|
|
}
|
|
|
|
log_err("Unknown flag set requested.");
|
|
return NULL;
|
|
}
|
|
|
|
static int _emit(char **buffer, size_t * size, const char *fmt, ...)
|
|
{
|
|
size_t n;
|
|
va_list ap;
|
|
|
|
va_start(ap, fmt);
|
|
n = vsnprintf(*buffer, *size, fmt, ap);
|
|
va_end(ap);
|
|
|
|
if (n < 0 || (n == *size))
|
|
return 0;
|
|
|
|
*buffer += n;
|
|
*size -= n;
|
|
return 1;
|
|
}
|
|
|
|
/*
|
|
* Converts a bitset to an array of string values,
|
|
* using one of the tables defined at the top of
|
|
* the file.
|
|
*/
|
|
int print_flags(uint32_t status, int type, char *buffer, size_t size)
|
|
{
|
|
int f, first = 1;
|
|
struct flag *flags;
|
|
|
|
if (!(flags = _get_flags(type))) {
|
|
stack;
|
|
return 0;
|
|
}
|
|
|
|
if (!_emit(&buffer, &size, "["))
|
|
return 0;
|
|
|
|
for (f = 0; flags[f].mask; f++) {
|
|
if (status & flags[f].mask) {
|
|
if (!first) {
|
|
if (!_emit(&buffer, &size, ", "))
|
|
return 0;
|
|
|
|
} else
|
|
first = 0;
|
|
|
|
if (!_emit(&buffer, &size, "\"%s\"",
|
|
flags[f].description))
|
|
return 0;
|
|
|
|
status &= ~flags[f].mask;
|
|
}
|
|
}
|
|
|
|
if (!_emit(&buffer, &size, "]"))
|
|
return 0;
|
|
|
|
if (status)
|
|
log_error("Metadata inconsistency: Not all flags successfully "
|
|
"exported.");
|
|
|
|
return 1;
|
|
}
|
|
|
|
int read_flags(uint32_t * status, int type, struct config_value *cv)
|
|
{
|
|
int f;
|
|
uint32_t s = 0;
|
|
struct flag *flags;
|
|
|
|
if (!(flags = _get_flags(type))) {
|
|
stack;
|
|
return 0;
|
|
}
|
|
|
|
while (cv) {
|
|
if (cv->type != CFG_STRING) {
|
|
log_err("Status value is not a string.");
|
|
return 0;
|
|
}
|
|
|
|
for (f = 0; flags[f].description; f++)
|
|
if (!strcmp(flags[f].description, cv->v.str)) {
|
|
s |= flags[f].mask;
|
|
break;
|
|
}
|
|
|
|
if (!flags[f].description) {
|
|
log_err("Unknown status flag '%s'.", cv->v.str);
|
|
return 0;
|
|
}
|
|
|
|
cv = cv->next;
|
|
}
|
|
|
|
*status = s;
|
|
return 1;
|
|
}
|