mirror of
git://sourceware.org/git/lvm2.git
synced 2024-12-27 07:22:49 +03:00
c2ed40a74f
o Text format now has a description and time field at the top level. o archiving and backup set the description appropriately. eg, for an archive: description = "Created *before* executing 'lvextend test_vg/lvol0 -l +1'." creation_time = 1013166332 for a backup: description = "Created *after* executing 'lvextend test_vg/lvol0 -l +1'." creation_time = 1013166332 This is preparing the way for a simple vgcfgundo command.
227 lines
4.2 KiB
C
227 lines
4.2 KiB
C
/*
|
|
* Copyright (C) 2001 Sistina Software (UK) Limited.
|
|
*
|
|
* This file is released under the LGPL.
|
|
*/
|
|
|
|
#include "format-text.h"
|
|
#include "import-export.h"
|
|
|
|
#include "lvm-file.h"
|
|
#include "log.h"
|
|
#include "pool.h"
|
|
#include "config.h"
|
|
#include "hash.h"
|
|
#include "dbg_malloc.h"
|
|
|
|
#include <unistd.h>
|
|
#include <sys/types.h>
|
|
#include <sys/file.h>
|
|
#include <limits.h>
|
|
|
|
/*
|
|
* NOTE: Currently there can be only one vg per file.
|
|
*/
|
|
|
|
struct text_c {
|
|
char *path;
|
|
char *desc;
|
|
struct uuid_map *um;
|
|
};
|
|
|
|
static void _not_written(const char *cmd)
|
|
{
|
|
log_err("The text format is lacking an implementation for '%s'", cmd);
|
|
}
|
|
|
|
static struct list *_get_vgs(struct format_instance *fi)
|
|
{
|
|
_not_written("_get_vgs");
|
|
return NULL;
|
|
}
|
|
|
|
static struct list *_get_pvs(struct format_instance *fi)
|
|
{
|
|
_not_written("_get_vgs");
|
|
return NULL;
|
|
}
|
|
|
|
static struct physical_volume *_pv_read(struct format_instance *fi,
|
|
const char *pv_name)
|
|
{
|
|
_not_written("_get_vgs");
|
|
return NULL;
|
|
}
|
|
|
|
static int _pv_setup(struct format_instance *fi, struct physical_volume *pv,
|
|
struct volume_group *vg)
|
|
{
|
|
_not_written("_get_vgs");
|
|
return 0;
|
|
}
|
|
|
|
static int _pv_write(struct format_instance *fi, struct physical_volume *pv)
|
|
{
|
|
_not_written("_get_vgs");
|
|
return 0;
|
|
}
|
|
|
|
static int _vg_setup(struct format_instance *fi, struct volume_group *vg)
|
|
{
|
|
_not_written("_get_vgs");
|
|
return 0;
|
|
}
|
|
|
|
static struct volume_group *_vg_read(struct format_instance *fi,
|
|
const char *vg_name)
|
|
{
|
|
struct text_c *tc = (struct text_c *) fi->private;
|
|
struct volume_group *vg;
|
|
|
|
if (!(vg = text_vg_import(fi->cmd, tc->path, tc->um))) {
|
|
stack;
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* Currently you can only have a single volume group per
|
|
* text file (this restriction may remain). We need to
|
|
* check that it contains the correct volume group.
|
|
*/
|
|
if (strcmp(vg_name, vg->name)) {
|
|
pool_free(fi->cmd->mem, vg);
|
|
log_err("'%s' does not contain volume group '%s'.",
|
|
tc->path, vg_name);
|
|
return NULL;
|
|
}
|
|
|
|
return vg;
|
|
}
|
|
|
|
static int _vg_write(struct format_instance *fi, struct volume_group *vg)
|
|
{
|
|
struct text_c *tc = (struct text_c *) fi->private;
|
|
|
|
FILE *fp;
|
|
int fd;
|
|
char *slash;
|
|
char temp_file[PATH_MAX], temp_dir[PATH_MAX];
|
|
|
|
slash = rindex(tc->path, '/');
|
|
|
|
if (slash == 0)
|
|
strcpy(temp_dir, ".");
|
|
else if (slash - tc->path < PATH_MAX) {
|
|
strncpy(temp_dir, tc->path, slash - tc->path);
|
|
temp_dir[slash - tc->path] = '\0';
|
|
|
|
} else {
|
|
log_error("Text format failed to determine directory.");
|
|
return 0;
|
|
}
|
|
|
|
if (!create_temp_name(temp_dir, temp_file, sizeof(temp_file), &fd)) {
|
|
log_err("Couldn't create temporary text file name.");
|
|
return 0;
|
|
}
|
|
|
|
if (!(fp = fdopen(fd, "w"))) {
|
|
log_sys_error("fdopen", temp_file);
|
|
close(fd);
|
|
return 0;
|
|
}
|
|
|
|
if (!text_vg_export(fp, vg, tc->desc)) {
|
|
log_error("Failed to write metadata to %s.", temp_file);
|
|
fclose(fp);
|
|
return 0;
|
|
}
|
|
|
|
if (fclose(fp)) {
|
|
log_sys_error("fclose", tc->path);
|
|
return 0;
|
|
}
|
|
|
|
if (rename(temp_file, tc->path)) {
|
|
log_error("%s: rename to %s failed: %s", temp_file, tc->path,
|
|
strerror(errno));
|
|
return 0;
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static void _destroy(struct format_instance *fi)
|
|
{
|
|
struct text_c *tc = (struct text_c *) fi->private;
|
|
|
|
dbg_free(tc->path);
|
|
dbg_free(tc);
|
|
dbg_free(fi);
|
|
}
|
|
|
|
static struct format_handler _text_handler = {
|
|
get_vgs: _get_vgs,
|
|
get_pvs: _get_pvs,
|
|
pv_read: _pv_read,
|
|
pv_setup: _pv_setup,
|
|
pv_write: _pv_write,
|
|
vg_setup: _vg_setup,
|
|
vg_read: _vg_read,
|
|
vg_write: _vg_write,
|
|
destroy: _destroy
|
|
};
|
|
|
|
struct format_instance *text_format_create(struct cmd_context *cmd,
|
|
const char *file,
|
|
struct uuid_map *um,
|
|
const char *desc)
|
|
{
|
|
struct format_instance *fi;
|
|
char *path, *d;
|
|
struct text_c *tc;
|
|
|
|
if (!(fi = dbg_malloc(sizeof(*fi)))) {
|
|
stack;
|
|
goto no_mem;
|
|
}
|
|
|
|
if (!(path = dbg_strdup(file))) {
|
|
stack;
|
|
goto no_mem;
|
|
}
|
|
|
|
if (!(d = dbg_strdup(desc))) {
|
|
stack;
|
|
goto no_mem;
|
|
}
|
|
|
|
if (!(tc = dbg_malloc(sizeof(*tc)))) {
|
|
stack;
|
|
goto no_mem;
|
|
}
|
|
|
|
tc->path = path;
|
|
tc->desc = d;
|
|
tc->um = um;
|
|
|
|
fi->cmd = cmd;
|
|
fi->ops = &_text_handler;
|
|
fi->private = tc;
|
|
|
|
return fi;
|
|
|
|
no_mem:
|
|
if (fi)
|
|
dbg_free(fi);
|
|
|
|
if (path)
|
|
dbg_free(path);
|
|
|
|
if (d)
|
|
dbg_free(path);
|
|
|
|
log_err("Couldn't allocate text format object.");
|
|
return NULL;
|
|
}
|