mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-04 09:18:36 +03:00
lib/cache/lvmetad: Refactor to use dm_config_tree in requests.
We were using daemon_send_simple until now, but it is no longer adequate, since we need to manipulate requests in a generic way (adding a validity token to each request), and the tree-based request interface is much more suitable for this.
This commit is contained in:
parent
ea14d5159c
commit
2276379a71
214
lib/cache/lvmetad.c
vendored
214
lib/cache/lvmetad.c
vendored
@ -23,6 +23,7 @@
|
|||||||
|
|
||||||
static int _using_lvmetad = 0;
|
static int _using_lvmetad = 0;
|
||||||
static daemon_handle _lvmetad;
|
static daemon_handle _lvmetad;
|
||||||
|
static const char *_lvmetad_token;
|
||||||
|
|
||||||
void lvmetad_init(void)
|
void lvmetad_init(void)
|
||||||
{
|
{
|
||||||
@ -36,6 +37,24 @@ void lvmetad_init(void)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static daemon_reply _lvmetad_send(const char *id, ...)
|
||||||
|
{
|
||||||
|
va_list ap;
|
||||||
|
va_start(ap, id);
|
||||||
|
daemon_reply repl;
|
||||||
|
daemon_request req = daemon_request_make(id);
|
||||||
|
|
||||||
|
// daemon_request_extend(req, "token", _lvmetad_token, NULL);
|
||||||
|
daemon_request_extend_v(req, ap);
|
||||||
|
|
||||||
|
repl = daemon_send(_lvmetad, req);
|
||||||
|
|
||||||
|
daemon_request_destroy(req);
|
||||||
|
|
||||||
|
va_end(ap);
|
||||||
|
return repl;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Helper; evaluate the reply from lvmetad, check for errors, print diagnostics
|
* Helper; evaluate the reply from lvmetad, check for errors, print diagnostics
|
||||||
* and return a summary success/failure exit code.
|
* and return a summary success/failure exit code.
|
||||||
@ -190,11 +209,11 @@ struct volume_group *lvmetad_vg_lookup(struct cmd_context *cmd, const char *vgna
|
|||||||
if (vgid) {
|
if (vgid) {
|
||||||
if (!id_write_format((const struct id*)vgid, uuid, sizeof(uuid)))
|
if (!id_write_format((const struct id*)vgid, uuid, sizeof(uuid)))
|
||||||
return_NULL;
|
return_NULL;
|
||||||
reply = daemon_send_simple(_lvmetad, "vg_lookup", "uuid = %s", uuid, NULL);
|
reply = _lvmetad_send("vg_lookup", "uuid = %s", uuid, NULL);
|
||||||
} else {
|
} else {
|
||||||
if (!vgname)
|
if (!vgname)
|
||||||
log_error(INTERNAL_ERROR "VG name required (VGID not available)");
|
log_error(INTERNAL_ERROR "VG name required (VGID not available)");
|
||||||
reply = daemon_send_simple(_lvmetad, "vg_lookup", "name = %s", vgname, NULL);
|
reply = _lvmetad_send("vg_lookup", "name = %s", vgname, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!strcmp(daemon_reply_str(reply, "response", ""), "OK")) {
|
if (!strcmp(daemon_reply_str(reply, "response", ""), "OK")) {
|
||||||
@ -264,9 +283,28 @@ static int _fixup_ignored(struct metadata_area *mda, void *baton) {
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
int lvmetad_vg_update(struct volume_group *vg)
|
static struct dm_config_tree *_export_vg_to_config_tree(struct volume_group *vg)
|
||||||
{
|
{
|
||||||
char *buf = NULL;
|
char *buf = NULL;
|
||||||
|
struct dm_config_tree *vgmeta;
|
||||||
|
|
||||||
|
if (!export_vg_to_buffer(vg, &buf)) {
|
||||||
|
log_error("Could not format VG metadata.");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!(vgmeta = dm_config_from_string(buf))) {
|
||||||
|
log_error("Error parsing VG metadata.");
|
||||||
|
dm_free(buf);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
dm_free(buf);
|
||||||
|
return vgmeta;
|
||||||
|
}
|
||||||
|
|
||||||
|
int lvmetad_vg_update(struct volume_group *vg)
|
||||||
|
{
|
||||||
daemon_reply reply;
|
daemon_reply reply;
|
||||||
struct dm_hash_node *n;
|
struct dm_hash_node *n;
|
||||||
struct metadata_area *mda;
|
struct metadata_area *mda;
|
||||||
@ -274,6 +312,7 @@ int lvmetad_vg_update(struct volume_group *vg)
|
|||||||
struct pv_list *pvl;
|
struct pv_list *pvl;
|
||||||
struct lvmcache_info *info;
|
struct lvmcache_info *info;
|
||||||
struct _fixup_baton baton;
|
struct _fixup_baton baton;
|
||||||
|
struct dm_config_tree *vgmeta;
|
||||||
|
|
||||||
if (!vg)
|
if (!vg)
|
||||||
return 0;
|
return 0;
|
||||||
@ -281,20 +320,12 @@ int lvmetad_vg_update(struct volume_group *vg)
|
|||||||
if (!_using_lvmetad || test_mode())
|
if (!_using_lvmetad || test_mode())
|
||||||
return 1; /* fake it */
|
return 1; /* fake it */
|
||||||
|
|
||||||
/* TODO. This is not entirely correct, since export_vg_to_buffer
|
if (!(vgmeta = _export_vg_to_config_tree(vg)))
|
||||||
* adds trailing nodes to the buffer. We may need to use
|
return_0;
|
||||||
* export_vg_to_config_tree and format the buffer ourselves. It
|
|
||||||
* does, however, work for now, since the garbage is well
|
|
||||||
* formatted and has no conflicting keys with the rest of the
|
|
||||||
* request. */
|
|
||||||
if (!export_vg_to_buffer(vg, &buf)) {
|
|
||||||
log_error("Could not format VG metadata.");
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
reply = daemon_send_simple(_lvmetad, "vg_update", "vgname = %s", vg->name,
|
reply = _lvmetad_send("vg_update", "vgname = %s", vg->name,
|
||||||
"metadata = %b", strchr(buf, '{'), NULL);
|
"metadata = %t", vgmeta, NULL);
|
||||||
dm_free(buf);
|
dm_config_destroy(vgmeta);
|
||||||
|
|
||||||
if (!_lvmetad_handle_reply(reply, "update VG", vg->name, NULL)) {
|
if (!_lvmetad_handle_reply(reply, "update VG", vg->name, NULL)) {
|
||||||
daemon_reply_destroy(reply);
|
daemon_reply_destroy(reply);
|
||||||
@ -344,8 +375,7 @@ int lvmetad_vg_remove(struct volume_group *vg)
|
|||||||
if (!id_write_format(&vg->id, uuid, sizeof(uuid)))
|
if (!id_write_format(&vg->id, uuid, sizeof(uuid)))
|
||||||
return_0;
|
return_0;
|
||||||
|
|
||||||
reply = daemon_send_simple(_lvmetad, "vg_remove", "uuid = %s", uuid, NULL);
|
reply = _lvmetad_send("vg_remove", "uuid = %s", uuid, NULL);
|
||||||
|
|
||||||
result = _lvmetad_handle_reply(reply, "remove VG", vg->name, NULL);
|
result = _lvmetad_handle_reply(reply, "remove VG", vg->name, NULL);
|
||||||
|
|
||||||
daemon_reply_destroy(reply);
|
daemon_reply_destroy(reply);
|
||||||
@ -366,8 +396,7 @@ int lvmetad_pv_lookup(struct cmd_context *cmd, struct id pvid, int *found)
|
|||||||
if (!id_write_format(&pvid, uuid, sizeof(uuid)))
|
if (!id_write_format(&pvid, uuid, sizeof(uuid)))
|
||||||
return_0;
|
return_0;
|
||||||
|
|
||||||
reply = daemon_send_simple(_lvmetad, "pv_lookup", "uuid = %s", uuid, NULL);
|
reply = _lvmetad_send("pv_lookup", "uuid = %s", uuid, NULL);
|
||||||
|
|
||||||
if (!_lvmetad_handle_reply(reply, "lookup PV", "", found))
|
if (!_lvmetad_handle_reply(reply, "lookup PV", "", found))
|
||||||
goto_out;
|
goto_out;
|
||||||
|
|
||||||
@ -397,8 +426,7 @@ int lvmetad_pv_lookup_by_dev(struct cmd_context *cmd, struct device *dev, int *f
|
|||||||
if (!_using_lvmetad)
|
if (!_using_lvmetad)
|
||||||
return_0;
|
return_0;
|
||||||
|
|
||||||
reply = daemon_send_simple(_lvmetad, "pv_lookup", "device = %d", dev->dev, NULL);
|
reply = _lvmetad_send("pv_lookup", "device = %d", dev->dev, NULL);
|
||||||
|
|
||||||
if (!_lvmetad_handle_reply(reply, "lookup PV", dev_name(dev), found))
|
if (!_lvmetad_handle_reply(reply, "lookup PV", dev_name(dev), found))
|
||||||
goto_out;
|
goto_out;
|
||||||
|
|
||||||
@ -425,8 +453,7 @@ int lvmetad_pv_list_to_lvmcache(struct cmd_context *cmd)
|
|||||||
if (!_using_lvmetad)
|
if (!_using_lvmetad)
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
reply = daemon_send_simple(_lvmetad, "pv_list", NULL);
|
reply = _lvmetad_send("pv_list", NULL);
|
||||||
|
|
||||||
if (!_lvmetad_handle_reply(reply, "list PVs", "", NULL)) {
|
if (!_lvmetad_handle_reply(reply, "list PVs", "", NULL)) {
|
||||||
daemon_reply_destroy(reply);
|
daemon_reply_destroy(reply);
|
||||||
return_0;
|
return_0;
|
||||||
@ -452,8 +479,7 @@ int lvmetad_vg_list_to_lvmcache(struct cmd_context *cmd)
|
|||||||
if (!_using_lvmetad)
|
if (!_using_lvmetad)
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
reply = daemon_send_simple(_lvmetad, "vg_list", NULL);
|
reply = _lvmetad_send("vg_list", NULL);
|
||||||
|
|
||||||
if (!_lvmetad_handle_reply(reply, "list VGs", "", NULL)) {
|
if (!_lvmetad_handle_reply(reply, "list VGs", "", NULL)) {
|
||||||
daemon_reply_destroy(reply);
|
daemon_reply_destroy(reply);
|
||||||
return_0;
|
return_0;
|
||||||
@ -476,66 +502,67 @@ int lvmetad_vg_list_to_lvmcache(struct cmd_context *cmd)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
struct _print_mda_baton {
|
struct _extract_mda_baton {
|
||||||
int i;
|
int i;
|
||||||
char *buffer;
|
struct dm_config_tree *cft;
|
||||||
|
struct dm_config_node *pre_sib;
|
||||||
};
|
};
|
||||||
|
|
||||||
static int _print_mda(struct metadata_area *mda, void *baton)
|
static int _extract_mda(struct metadata_area *mda, void *baton)
|
||||||
{
|
{
|
||||||
|
struct _extract_mda_baton *b = baton;
|
||||||
|
struct dm_config_node *cn;
|
||||||
int result = 0;
|
int result = 0;
|
||||||
struct _print_mda_baton *b = baton;
|
char id[32];
|
||||||
char *buf, *mda_txt;
|
|
||||||
|
|
||||||
if (!mda->ops->mda_export_text) /* do nothing */
|
if (!mda->ops->mda_export_text) /* do nothing */
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
buf = b->buffer;
|
dm_snprintf(id, 32, "mda%d", b->i);
|
||||||
mda_txt = mda->ops->mda_export_text(mda);
|
if (!(cn = make_config_node(b->cft, id, b->cft->root, b->pre_sib)))
|
||||||
if (!dm_asprintf(&b->buffer, "%s mda%i { %s }", b->buffer ?: "", b->i, mda_txt))
|
return 0;
|
||||||
goto_out;
|
if (!mda->ops->mda_export_text(mda, b->cft, cn))
|
||||||
|
return 0;
|
||||||
|
|
||||||
b->i ++;
|
b->i ++;
|
||||||
result = 1;
|
b->pre_sib = cn; /* for efficiency */
|
||||||
out:
|
|
||||||
dm_free(mda_txt);
|
return 1;
|
||||||
dm_free(buf);
|
|
||||||
return result;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int _print_da(struct disk_locn *da, void *baton)
|
static int _extract_da(struct disk_locn *da, void *baton)
|
||||||
{
|
{
|
||||||
struct _print_mda_baton *b;
|
struct _extract_mda_baton *b = baton;
|
||||||
char *buf;
|
struct dm_config_node *cn;
|
||||||
|
char id[32];
|
||||||
|
|
||||||
if (!da)
|
if (!da)
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
b = baton;
|
dm_snprintf(id, 32, "da%d", b->i);
|
||||||
buf = b->buffer;
|
if (!(cn = make_config_node(b->cft, id, b->cft->root, b->pre_sib)))
|
||||||
if (!dm_asprintf(&b->buffer, "%s da%i { offset = %" PRIu64
|
return 0;
|
||||||
" size = %" PRIu64 " }",
|
if (!config_make_nodes(b->cft, cn, NULL, "offset = %d", da->offset, "size = %d", da->size, NULL))
|
||||||
b->buffer ?: "", b->i, da->offset, da->size))
|
return 0;
|
||||||
{
|
|
||||||
dm_free(buf);
|
|
||||||
return_0;
|
|
||||||
}
|
|
||||||
b->i ++;
|
b->i ++;
|
||||||
dm_free(buf);
|
b->pre_sib = cn; /* for efficiency */
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static const char *_print_mdas(struct lvmcache_info *info)
|
static int _extract_mdas(struct lvmcache_info *info, struct dm_config_tree *cft,
|
||||||
|
struct dm_config_node *pre_sib)
|
||||||
{
|
{
|
||||||
struct _print_mda_baton baton = { .i = 0, .buffer = NULL };
|
struct _extract_mda_baton baton = { .i = 0, .cft = cft, .pre_sib = NULL };
|
||||||
|
|
||||||
if (!lvmcache_foreach_mda(info, &_print_mda, &baton))
|
if (!lvmcache_foreach_mda(info, &_extract_mda, &baton))
|
||||||
return NULL;
|
return 0;
|
||||||
baton.i = 0;
|
baton.i = 0;
|
||||||
if (!lvmcache_foreach_da(info, &_print_da, &baton))
|
if (!lvmcache_foreach_da(info, &_extract_da, &baton))
|
||||||
return NULL;
|
return 0;
|
||||||
|
|
||||||
return baton.buffer;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
int lvmetad_pv_found(struct id pvid, struct device *device, const struct format_type *fmt,
|
int lvmetad_pv_found(struct id pvid, struct device *device, const struct format_type *fmt,
|
||||||
@ -545,8 +572,7 @@ int lvmetad_pv_found(struct id pvid, struct device *device, const struct format_
|
|||||||
daemon_reply reply;
|
daemon_reply reply;
|
||||||
struct lvmcache_info *info;
|
struct lvmcache_info *info;
|
||||||
const char *mdas = NULL;
|
const char *mdas = NULL;
|
||||||
char *pvmeta;
|
struct dm_config_tree *pvmeta, *vgmeta;
|
||||||
char *buf = NULL;
|
|
||||||
const char *status;
|
const char *status;
|
||||||
int result;
|
int result;
|
||||||
|
|
||||||
@ -556,46 +582,45 @@ int lvmetad_pv_found(struct id pvid, struct device *device, const struct format_
|
|||||||
if (!id_write_format(&pvid, uuid, sizeof(uuid)))
|
if (!id_write_format(&pvid, uuid, sizeof(uuid)))
|
||||||
return_0;
|
return_0;
|
||||||
|
|
||||||
/* FIXME A more direct route would be much preferable. */
|
pvmeta = dm_config_create();
|
||||||
if ((info = lvmcache_info_from_pvid((const char *)&pvid, 0)))
|
if (!pvmeta)
|
||||||
mdas = _print_mdas(info);
|
return_0;
|
||||||
|
|
||||||
if (!dm_asprintf(&pvmeta,
|
info = lvmcache_info_from_pvid((const char *)&pvid, 0);
|
||||||
"{ device = %" PRIu64 "\n"
|
|
||||||
" dev_size = %" PRIu64 "\n"
|
if (!(pvmeta->root = make_config_node(pvmeta, "pv", NULL, NULL))) {
|
||||||
" format = \"%s\"\n"
|
dm_config_destroy(pvmeta);
|
||||||
" label_sector = %" PRIu64 "\n"
|
|
||||||
" id = \"%s\"\n"
|
|
||||||
" %s"
|
|
||||||
"}", device->dev,
|
|
||||||
info ? lvmcache_device_size(info) : 0,
|
|
||||||
fmt->name, label_sector, uuid, mdas ?: "")) {
|
|
||||||
dm_free((char *)mdas);
|
|
||||||
return_0;
|
return_0;
|
||||||
}
|
}
|
||||||
|
|
||||||
dm_free((char *)mdas);
|
if (!config_make_nodes(pvmeta, pvmeta->root, NULL,
|
||||||
|
"device = %d", device->dev,
|
||||||
|
"dev_size = %d", info ? lvmcache_device_size(info) : 0,
|
||||||
|
"format = %s", fmt->name,
|
||||||
|
"label_sector = %d", label_sector,
|
||||||
|
"id = %s", uuid,
|
||||||
|
NULL))
|
||||||
|
{
|
||||||
|
dm_config_destroy(pvmeta);
|
||||||
|
return_0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (info)
|
||||||
|
/* FIXME A more direct route would be much preferable. */
|
||||||
|
_extract_mdas(info, pvmeta, pvmeta->root);
|
||||||
|
|
||||||
if (vg) {
|
if (vg) {
|
||||||
/*
|
if (!(vgmeta = _export_vg_to_config_tree(vg))) {
|
||||||
* TODO. This is not entirely correct, since export_vg_to_buffer
|
dm_config_destroy(pvmeta);
|
||||||
* adds trailing garbage to the buffer. We may need to use
|
|
||||||
* export_vg_to_config_tree and format the buffer ourselves. It
|
|
||||||
* does, however, work for now, since the garbage is well
|
|
||||||
* formatted and has no conflicting keys with the rest of the
|
|
||||||
* request.
|
|
||||||
*/
|
|
||||||
if (!export_vg_to_buffer(vg, &buf)) {
|
|
||||||
dm_free(pvmeta);
|
|
||||||
return_0;
|
return_0;
|
||||||
}
|
}
|
||||||
|
|
||||||
reply = daemon_send_simple(_lvmetad,
|
reply = _lvmetad_send("pv_found",
|
||||||
"pv_found",
|
"pvmeta = %t", pvmeta,
|
||||||
"pvmeta = %b", pvmeta,
|
|
||||||
"vgname = %s", vg->name,
|
"vgname = %s", vg->name,
|
||||||
"metadata = %b", strchr(buf, '{'),
|
"metadata = %t", vgmeta,
|
||||||
NULL);
|
NULL);
|
||||||
|
dm_config_destroy(vgmeta);
|
||||||
} else {
|
} else {
|
||||||
if (handler) {
|
if (handler) {
|
||||||
log_error(INTERNAL_ERROR "Handler needs existing VG.");
|
log_error(INTERNAL_ERROR "Handler needs existing VG.");
|
||||||
@ -603,13 +628,10 @@ int lvmetad_pv_found(struct id pvid, struct device *device, const struct format_
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
/* There are no MDAs on this PV. */
|
/* There are no MDAs on this PV. */
|
||||||
reply = daemon_send_simple(_lvmetad,
|
reply = _lvmetad_send("pv_found", "pvmeta = %t", pvmeta, NULL);
|
||||||
"pv_found",
|
|
||||||
"pvmeta = %b", pvmeta,
|
|
||||||
NULL);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
dm_free(pvmeta);
|
dm_config_destroy(pvmeta);
|
||||||
|
|
||||||
result = _lvmetad_handle_reply(reply, "update PV", uuid, NULL);
|
result = _lvmetad_handle_reply(reply, "update PV", uuid, NULL);
|
||||||
|
|
||||||
|
2
lib/cache/lvmetad.h
vendored
2
lib/cache/lvmetad.h
vendored
@ -15,6 +15,8 @@
|
|||||||
#ifndef _LVM_METAD_H
|
#ifndef _LVM_METAD_H
|
||||||
#define _LVM_METAD_H
|
#define _LVM_METAD_H
|
||||||
|
|
||||||
|
#include "daemon-shared.h" // XXX
|
||||||
|
|
||||||
struct volume_group;
|
struct volume_group;
|
||||||
struct cmd_context;
|
struct cmd_context;
|
||||||
struct dm_config_tree;
|
struct dm_config_tree;
|
||||||
|
@ -1584,7 +1584,9 @@ static struct metadata_area_ops _metadata_text_file_backup_ops = {
|
|||||||
.vg_commit = _vg_commit_file_backup
|
.vg_commit = _vg_commit_file_backup
|
||||||
};
|
};
|
||||||
|
|
||||||
static char *_mda_export_text_raw(struct metadata_area *mda);
|
static int _mda_export_text_raw(struct metadata_area *mda,
|
||||||
|
struct dm_config_tree *cft,
|
||||||
|
struct dm_config_node *parent);
|
||||||
static int _mda_import_text_raw(struct lvmcache_info *info, const struct dm_config_node *cn);
|
static int _mda_import_text_raw(struct lvmcache_info *info, const struct dm_config_node *cn);
|
||||||
|
|
||||||
static struct metadata_area_ops _metadata_text_raw_ops = {
|
static struct metadata_area_ops _metadata_text_raw_ops = {
|
||||||
@ -1608,19 +1610,18 @@ static struct metadata_area_ops _metadata_text_raw_ops = {
|
|||||||
.mda_import_text = _mda_import_text_raw
|
.mda_import_text = _mda_import_text_raw
|
||||||
};
|
};
|
||||||
|
|
||||||
static char *_mda_export_text_raw(struct metadata_area *mda)
|
static int _mda_export_text_raw(struct metadata_area *mda,
|
||||||
|
struct dm_config_tree *cft,
|
||||||
|
struct dm_config_node *parent)
|
||||||
{
|
{
|
||||||
struct mda_context *mdc = (struct mda_context *) mda->metadata_locn;
|
struct mda_context *mdc = (struct mda_context *) mda->metadata_locn;
|
||||||
char *result;
|
|
||||||
|
|
||||||
dm_asprintf(&result,
|
return config_make_nodes(cft, parent, NULL,
|
||||||
"ignore = %d "
|
"ignore = %d", mda_is_ignored(mda),
|
||||||
"start = %" PRIu64" "
|
"start = %d", mdc->area.start,
|
||||||
"size = %" PRIu64 " "
|
"size = %d", mdc->area.size,
|
||||||
"free_sectors = %" PRIu64,
|
"free_sectors = %d", mdc->free_sectors,
|
||||||
mda_is_ignored(mda), mdc->area.start, mdc->area.size, mdc->free_sectors);
|
NULL) ? 1 : 0;
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static int _mda_import_text_raw(struct lvmcache_info *info, const struct dm_config_node *cn)
|
static int _mda_import_text_raw(struct lvmcache_info *info, const struct dm_config_node *cn)
|
||||||
|
@ -148,7 +148,8 @@ struct metadata_area_ops {
|
|||||||
struct metadata_area *mda2);
|
struct metadata_area *mda2);
|
||||||
|
|
||||||
struct device *(*mda_get_device)(struct metadata_area *mda);
|
struct device *(*mda_get_device)(struct metadata_area *mda);
|
||||||
char *(*mda_export_text)(struct metadata_area *mda);
|
int (*mda_export_text)(struct metadata_area *mda, struct dm_config_tree *cft,
|
||||||
|
struct dm_config_node *parent);
|
||||||
int (*mda_import_text)(struct lvmcache_info *info, const struct dm_config_node *cn);
|
int (*mda_import_text)(struct lvmcache_info *info, const struct dm_config_node *cn);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user