mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-03 05:18:29 +03:00
fc63d04f42
o First pass at low level labelling switch. This allows us to register different label types (eg, lvm1, lvm2).
149 lines
2.2 KiB
C
149 lines
2.2 KiB
C
/*
|
|
* Copyright (C) 2001 Sistina Software
|
|
*
|
|
* This file is released under the LGPL.
|
|
*/
|
|
|
|
#include "label.h"
|
|
#include "list.h"
|
|
#include "dbg_malloc.h"
|
|
|
|
/*
|
|
* Internal labeller struct.
|
|
*/
|
|
struct labeller_i {
|
|
struct list list;
|
|
|
|
struct labeller *l;
|
|
char name[0];
|
|
};
|
|
|
|
static struct list _labellers;
|
|
|
|
|
|
static struct labeller_i *_alloc_li(const char *name, struct labeller *l)
|
|
{
|
|
struct labeller_i *li;
|
|
size_t len;
|
|
|
|
len = sizeof(*li) + strlen(name) + 1;
|
|
|
|
if (!(li = dbg_malloc(len))) {
|
|
log_err("Couldn't allocate memory for labeller list object.");
|
|
return NULL;
|
|
}
|
|
|
|
li->l = l;
|
|
strcpy(li->name, name);
|
|
|
|
return li;
|
|
}
|
|
|
|
static void _free_li(struct labeller_i *li)
|
|
{
|
|
dbg_free(li);
|
|
}
|
|
|
|
|
|
int label_init(void)
|
|
{
|
|
list_init(&_labellers);
|
|
}
|
|
|
|
void label_exit(void)
|
|
{
|
|
struct list *c, *n;
|
|
struct labeller_list *ll;
|
|
|
|
for (c = _labellers.n; c != &_labellers; c = n) {
|
|
n = c->n;
|
|
ll = list_item(c, struct labeller_list);
|
|
_free_li(c);
|
|
}
|
|
}
|
|
|
|
int label_register_handler(const char *name, struct labeller *handler)
|
|
{
|
|
struct labeller_i *li;
|
|
|
|
if (!(li = _alloc_li(name, handler))) {
|
|
stack;
|
|
return 0;
|
|
}
|
|
|
|
list_add(&_labellers, &li->list);
|
|
return 1;
|
|
}
|
|
|
|
struct labeller *label_get_handler(const char *name)
|
|
{
|
|
struct list *lih;
|
|
struct labeller_i *li;
|
|
|
|
list_iterate (lih, &_labellers) {
|
|
li = list_item(lih, struct labeller_i);
|
|
if (!strcmp(li->name, name))
|
|
return li->l;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static struct labeller *_find_labeller(const char *device)
|
|
{
|
|
struct list *lih;
|
|
struct labeller_i *li;
|
|
|
|
list_iterate (lih, &_labellers) {
|
|
li = list_item(lih, struct labeller_i);
|
|
if (li->l->ops->can_handle(device))
|
|
return li->l;
|
|
}
|
|
|
|
log_err("Could not find label on device '%s'.", device);
|
|
return NULL;
|
|
}
|
|
|
|
int label_remove(const char *device)
|
|
{
|
|
struct labeller *l;
|
|
|
|
if (!(l = _find_labeller(device))) {
|
|
stack;
|
|
return 0;
|
|
}
|
|
|
|
return l->ops->remove(device);
|
|
}
|
|
|
|
int label_read(const char *path, struct label **result)
|
|
{
|
|
struct labeller *l;
|
|
|
|
if (!(l = _find_labeller(device))) {
|
|
stack;
|
|
return 0;
|
|
}
|
|
|
|
return l->ops->read(device, label);
|
|
}
|
|
|
|
int label_verify(const char *path)
|
|
{
|
|
struct labeller *l;
|
|
|
|
if (!(l = _find_labeller(device))) {
|
|
stack;
|
|
return 0;
|
|
}
|
|
|
|
return l->ops->verify(device);
|
|
}
|
|
|
|
void label_free(struct label *l)
|
|
{
|
|
dbg_free(l->extra_info);
|
|
dbg_free(l);
|
|
}
|
|
|