mirror of
git://sourceware.org/git/lvm2.git
synced 2024-12-21 13:34:40 +03:00
o Migration of device-mapper from LVM_WORK to it's own (public) repository.
Please use this one from now on.
This commit is contained in:
parent
0d3593f5c1
commit
19df7116c6
54
libdm/Makefile
Normal file
54
libdm/Makefile
Normal file
@ -0,0 +1,54 @@
|
||||
#
|
||||
# Copyright (C) 2001 Sistina Software (UK) Limited.
|
||||
#
|
||||
# This file is released under the LGPL.
|
||||
#
|
||||
|
||||
srcdir = .
|
||||
top_srcdir = ..
|
||||
VPATH = .
|
||||
SHELL = /bin/sh
|
||||
|
||||
CC = gcc
|
||||
RANLIB = ranlib
|
||||
SHELL = /bin/sh
|
||||
INSTALL = /usr/bin/install -c
|
||||
LN_S = ln -s
|
||||
|
||||
prefix = /usr
|
||||
libdir = ${prefix}/lib
|
||||
incdir = ${prefix}/include
|
||||
kernelsrcdir = /scratch/alphalinux/linux
|
||||
|
||||
OWNER=root
|
||||
GROUP=root
|
||||
|
||||
INCLUDES=-I. -I${kernelsrcdir}/include
|
||||
CFLAGS+=-Wall
|
||||
#CFLAGS+=-O2
|
||||
CFLAGS+=-g -fno-omit-frame-pointer
|
||||
#CFLAGS+=-pg
|
||||
#LD_FLAGS=-pg
|
||||
|
||||
libdevmapper.so: libdm.o
|
||||
$(CC) -shared -o libdevmapper.so libdm.o
|
||||
|
||||
SUFFIXES=
|
||||
SUFFIXES=.c .o .so
|
||||
|
||||
%.o: %.c
|
||||
$(CC) -c $(INCLUDES) $(CFLAGS) $< -o $@
|
||||
|
||||
install: libdevmapper.so
|
||||
$(INSTALL) -c -o $(OWNER) -g $(GROUP) -m 555 -s libdevmapper.so \
|
||||
$(libdir)
|
||||
$(INSTALL) -D -c -o $(OWNER) -g $(GROUP) -m 444 libdevmapper.h \
|
||||
$(incdir)/devmapper/libdevmapper.h
|
||||
|
||||
clean:
|
||||
$(RM) libdevmapper.so libdm.o
|
||||
|
||||
distclean: clean
|
||||
|
||||
.PHONY: install test clean distclean
|
||||
|
81
libdm/libdevmapper.h
Normal file
81
libdm/libdevmapper.h
Normal file
@ -0,0 +1,81 @@
|
||||
/*
|
||||
* Copyright (C) 2001 Sistina Software (UK) Limited.
|
||||
*
|
||||
* This file is released under the LGPL.
|
||||
*/
|
||||
|
||||
#ifndef LIB_DEVICE_MAPPER_H
|
||||
#define LIB_DEVICE_MAPPER_H
|
||||
|
||||
/*
|
||||
* Since it is quite laborious to build the ioctl
|
||||
* arguments for the device-mapper people are
|
||||
* encouraged to use this library.
|
||||
*
|
||||
* You will need to build a struct dm_task for
|
||||
* each ioctl command you want to execute.
|
||||
*/
|
||||
|
||||
|
||||
typedef void (*dm_log_fn)(int level, const char *file, int line,
|
||||
const char *f, ...);
|
||||
|
||||
/*
|
||||
* The library user may wish to register their own
|
||||
* logging function, by default errors go to
|
||||
* stderr.
|
||||
*/
|
||||
void dm_log_init(dm_log_fn fn);
|
||||
|
||||
enum {
|
||||
DM_DEVICE_CREATE,
|
||||
DM_DEVICE_RELOAD,
|
||||
DM_DEVICE_REMOVE,
|
||||
|
||||
DM_DEVICE_SUSPEND,
|
||||
DM_DEVICE_RESUME,
|
||||
|
||||
DM_DEVICE_INFO,
|
||||
};
|
||||
|
||||
|
||||
struct dm_task;
|
||||
|
||||
struct dm_task *dm_task_create(int type);
|
||||
void dm_task_destroy(struct dm_task *dmt);
|
||||
|
||||
int dm_task_set_name(struct dm_task *dmt, const char *name);
|
||||
|
||||
/*
|
||||
* Retrieve attributes after an info.
|
||||
*/
|
||||
struct dm_info {
|
||||
int exists;
|
||||
int suspended;
|
||||
unsigned int open_count;
|
||||
int minor; /* minor device number */
|
||||
unsigned int target_count;
|
||||
};
|
||||
|
||||
int dm_task_get_info(struct dm_task *dmt, struct dm_info *dmi);
|
||||
|
||||
/*
|
||||
* Use these to prepare for a create or reload.
|
||||
*/
|
||||
int dm_task_add_target(struct dm_task *dmt,
|
||||
unsigned long long start,
|
||||
unsigned long long size,
|
||||
const char *ttype,
|
||||
const char *params);
|
||||
|
||||
/*
|
||||
* Call this to actually run the ioctl.
|
||||
*/
|
||||
int dm_task_run(struct dm_task *dmt);
|
||||
|
||||
/*
|
||||
* Return the device-mapper directory
|
||||
*/
|
||||
const char *dm_dir(void);
|
||||
|
||||
#endif /* LIB_DEVICE_MAPPER_H */
|
421
libdm/libdm.c
Normal file
421
libdm/libdm.c
Normal file
@ -0,0 +1,421 @@
|
||||
/*
|
||||
* Copyright (C) 2001 Sistina Software (UK) Limited.
|
||||
*
|
||||
* This file is released under the LGPL.
|
||||
*/
|
||||
|
||||
#include "libdevmapper.h"
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <stdarg.h>
|
||||
#include <string.h>
|
||||
#include <unistd.h>
|
||||
#include <fcntl.h>
|
||||
#include <sys/stat.h>
|
||||
#include <sys/types.h>
|
||||
#include <sys/ioctl.h>
|
||||
#include <errno.h>
|
||||
|
||||
#include <linux/dm-ioctl.h>
|
||||
|
||||
#define DEVICE_MAPPER_CONTROL "/dev/device-mapper/control"
|
||||
#define ALIGNMENT sizeof(int)
|
||||
|
||||
/*
|
||||
* Library users can provide their own logging
|
||||
* function.
|
||||
*/
|
||||
static void _default_log(int level, const char *file, int line,
|
||||
const char *f, ...)
|
||||
{
|
||||
va_list ap;
|
||||
|
||||
//fprintf(stderr, "%s:%d ", file, line);
|
||||
|
||||
va_start(ap, f);
|
||||
vfprintf(stderr, f, ap);
|
||||
va_end(ap);
|
||||
|
||||
fprintf(stderr, "\n");
|
||||
}
|
||||
|
||||
static dm_log_fn _log = _default_log;
|
||||
|
||||
void dm_log_init(dm_log_fn fn)
|
||||
{
|
||||
_log = fn;
|
||||
}
|
||||
|
||||
#define log(msg, x...) _log(1, __FILE__, __LINE__, msg, ## x)
|
||||
|
||||
struct target {
|
||||
|
||||
unsigned long long start;
|
||||
unsigned long long length;
|
||||
char *type;
|
||||
char *params;
|
||||
|
||||
struct target *next;
|
||||
};
|
||||
|
||||
struct dm_task {
|
||||
int type;
|
||||
char *dev_name;
|
||||
|
||||
struct target *head, *tail;
|
||||
|
||||
struct dm_ioctl *dmi;
|
||||
};
|
||||
|
||||
struct dm_task *dm_task_create(int type)
|
||||
{
|
||||
struct dm_task *dmt = malloc(sizeof(*dmt));
|
||||
|
||||
if (!dmt)
|
||||
return NULL;
|
||||
|
||||
memset(dmt, 0, sizeof(*dmt));
|
||||
|
||||
dmt->type = type;
|
||||
return dmt;
|
||||
}
|
||||
|
||||
void dm_task_destroy(struct dm_task *dmt)
|
||||
{
|
||||
struct target *t, *n;
|
||||
|
||||
for (t = dmt->head; t; t = n) {
|
||||
n = t->next;
|
||||
free(t);
|
||||
}
|
||||
|
||||
if (dmt->dmi)
|
||||
free(dmt->dmi);
|
||||
|
||||
free(dmt);
|
||||
}
|
||||
|
||||
int dm_task_set_name(struct dm_task *dmt, const char *name)
|
||||
{
|
||||
if (dmt->dev_name)
|
||||
free(dmt->dev_name);
|
||||
|
||||
return (dmt->dev_name = strdup(name)) ? 1 : 0;
|
||||
}
|
||||
|
||||
int dm_task_get_info(struct dm_task *dmt, struct dm_info *info)
|
||||
{
|
||||
if (!dmt->dmi)
|
||||
return 0;
|
||||
|
||||
info->exists = dmt->dmi->exists;
|
||||
info->suspended = dmt->dmi->suspend;
|
||||
info->open_count = dmt->dmi->open_count;
|
||||
info->minor = dmt->dmi->minor;
|
||||
info->target_count = dmt->dmi->target_count;
|
||||
return 1;
|
||||
}
|
||||
|
||||
static struct target *_create_target(unsigned long long start,
|
||||
unsigned long long len,
|
||||
const char *type, const char *params)
|
||||
{
|
||||
struct target *t = malloc(sizeof(*t));
|
||||
|
||||
if (!t)
|
||||
return NULL;
|
||||
memset(t, 0, sizeof(*t));
|
||||
|
||||
if (!(t->params = strdup(params))) {
|
||||
log("Out of memory");
|
||||
goto bad;
|
||||
}
|
||||
|
||||
if (!(t->type = strdup(type))) {
|
||||
log("Out of memory");
|
||||
goto bad;
|
||||
}
|
||||
|
||||
t->start = start;
|
||||
t->length = len;
|
||||
return t;
|
||||
|
||||
bad:
|
||||
free(t->params);
|
||||
free(t->type);
|
||||
free(t);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
int dm_task_add_target(struct dm_task *dmt,
|
||||
unsigned long long start,
|
||||
unsigned long long size,
|
||||
const char *ttype,
|
||||
const char *params)
|
||||
{
|
||||
struct target *t = _create_target(start, size, ttype, params);
|
||||
|
||||
if (!t)
|
||||
return 0;
|
||||
|
||||
if (!dmt->head)
|
||||
dmt->head = dmt->tail = t;
|
||||
else {
|
||||
dmt->tail->next = t;
|
||||
dmt->tail = t;
|
||||
}
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
static void *_align(void *ptr, unsigned int align)
|
||||
{
|
||||
align--;
|
||||
return (void *) (((long) ptr + align) & ~align);
|
||||
}
|
||||
|
||||
static void *_add_target(struct target *t, void *out, void *end)
|
||||
{
|
||||
void *out_sp = out;
|
||||
struct dm_target_spec sp;
|
||||
int len;
|
||||
const char no_space[] = "Ran out of memory building ioctl parameter";
|
||||
|
||||
out += sizeof(struct dm_target_spec);
|
||||
if (out >= end) {
|
||||
log(no_space);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
sp.status = 0;
|
||||
sp.sector_start = t->start;
|
||||
sp.length = t->length;
|
||||
strncpy(sp.target_type, t->type, sizeof(sp.target_type));
|
||||
|
||||
len = strlen(t->params);
|
||||
|
||||
if ((out + len + 1) >= end) {
|
||||
log(no_space);
|
||||
|
||||
log("t->params= '%s'", t->params);
|
||||
return NULL;
|
||||
}
|
||||
strcpy((char *) out, t->params);
|
||||
out += len + 1;
|
||||
|
||||
/* align next block */
|
||||
out = _align(out, ALIGNMENT);
|
||||
|
||||
sp.next = out - out_sp;
|
||||
memcpy(out_sp, &sp, sizeof(sp));
|
||||
|
||||
return out;
|
||||
}
|
||||
|
||||
static struct dm_ioctl *_flatten(struct dm_task *dmt)
|
||||
{
|
||||
struct dm_ioctl *dmi;
|
||||
struct target *t;
|
||||
size_t len = sizeof(struct dm_ioctl);
|
||||
void *b, *e;
|
||||
int count = 0;
|
||||
|
||||
for (t = dmt->head; t; t = t->next) {
|
||||
len += sizeof(struct dm_target_spec);
|
||||
len += strlen(t->params) + 1 + ALIGNMENT;
|
||||
count++;
|
||||
}
|
||||
|
||||
if (!(dmi = malloc(len)))
|
||||
return NULL;
|
||||
|
||||
dmi->data_size = len;
|
||||
strncpy(dmi->name, dmt->dev_name, sizeof(dmi->name));
|
||||
dmi->suspend = (dmt->type == DM_DEVICE_SUSPEND) ? 1 : 0;
|
||||
dmi->open_count = 0;
|
||||
dmi->minor = -1;
|
||||
|
||||
dmi->target_count = count;
|
||||
|
||||
b = (void *) (dmi + 1);
|
||||
e = (void *) ((char *) dmi + len);
|
||||
|
||||
for (t = dmt->head; t; t = t->next)
|
||||
if (!(b = _add_target(t, b, e)))
|
||||
goto bad;
|
||||
|
||||
return dmi;
|
||||
|
||||
bad:
|
||||
free(dmi);
|
||||
return NULL;
|
||||
}
|
||||
|
||||
/*
|
||||
* FIXME: This function is copied straight from
|
||||
* LVM1 without an audit.
|
||||
*/
|
||||
static int __check_devfs(void)
|
||||
{
|
||||
int r = 0, len;
|
||||
char dir[PATH_MAX], line[512];
|
||||
char type[32];
|
||||
FILE *mounts = NULL;
|
||||
const char *dev_dir = DM_DIR;
|
||||
|
||||
/* trim the trailing slash off dev_dir, yuck */
|
||||
len = strlen(dev_dir) - 1;
|
||||
while(len && dev_dir[len] == '/')
|
||||
len--;
|
||||
|
||||
if (!(mounts = fopen("/proc/mounts", "r"))) {
|
||||
log("Unable to open /proc/mounts to determine "
|
||||
"if devfs is mounted");
|
||||
return 0;
|
||||
}
|
||||
|
||||
while (!feof(mounts)) {
|
||||
fgets(line, sizeof(line) - 1, mounts);
|
||||
if (sscanf(line, "%*s %s %s %*s", dir, type) != 2)
|
||||
continue;
|
||||
|
||||
if (!strcmp(type, "devfs") && !strncmp(dir, dev_dir, len)) {
|
||||
r = 1;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
fclose(mounts);
|
||||
return r;
|
||||
}
|
||||
|
||||
/*
|
||||
* Memo the result of __check_devfs.
|
||||
*/
|
||||
static int _check_devfs(void)
|
||||
{
|
||||
static int prev_result = -1;
|
||||
|
||||
if (prev_result >= 0)
|
||||
return prev_result;
|
||||
|
||||
return (prev_result = __check_devfs());
|
||||
}
|
||||
|
||||
static void _build_dev_path(char *buffer, size_t len, const char *dev_name)
|
||||
{
|
||||
snprintf(buffer, len, "/dev/%s/%s", DM_DIR, dev_name);
|
||||
}
|
||||
|
||||
static int _add_dev_node(const char *dev_name, dev_t dev)
|
||||
{
|
||||
char path[PATH_MAX];
|
||||
|
||||
if (_check_devfs())
|
||||
return 1;
|
||||
|
||||
_build_dev_path(path, sizeof(path), dev_name);
|
||||
|
||||
if (mknod(path, S_IFBLK | S_IRUSR | S_IWUSR | S_IRGRP, dev) < 0) {
|
||||
log("Unable to make device node for '%s'", dev_name);
|
||||
return 0;
|
||||
}
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
static int _rm_dev_node(const char *dev_name)
|
||||
{
|
||||
char path[PATH_MAX];
|
||||
|
||||
if (_check_devfs())
|
||||
return 1;
|
||||
|
||||
_build_dev_path(path, sizeof(path), dev_name);
|
||||
|
||||
if (unlink(path) < 0) {
|
||||
log("Unable to unlink device node for '%s'", dev_name);
|
||||
return 0;
|
||||
}
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
int dm_task_run(struct dm_task *dmt)
|
||||
{
|
||||
int fd = -1;
|
||||
struct dm_ioctl *dmi = _flatten(dmt);
|
||||
unsigned int command;
|
||||
|
||||
if (!dmi) {
|
||||
log("Couldn't create ioctl argument");
|
||||
return 0;
|
||||
}
|
||||
|
||||
if ((fd = open(DEVICE_MAPPER_CONTROL, O_RDWR)) < 0) {
|
||||
log("Couldn't open device-mapper control device");
|
||||
goto bad;
|
||||
}
|
||||
|
||||
switch (dmt->type) {
|
||||
case DM_DEVICE_CREATE:
|
||||
command = DM_CREATE;
|
||||
break;
|
||||
|
||||
case DM_DEVICE_RELOAD:
|
||||
command = DM_RELOAD;
|
||||
break;
|
||||
|
||||
case DM_DEVICE_REMOVE:
|
||||
command = DM_REMOVE;
|
||||
break;
|
||||
|
||||
case DM_DEVICE_SUSPEND:
|
||||
command = DM_SUSPEND;
|
||||
break;
|
||||
|
||||
case DM_DEVICE_RESUME:
|
||||
command = DM_SUSPEND;
|
||||
break;
|
||||
|
||||
case DM_DEVICE_INFO:
|
||||
command = DM_INFO;
|
||||
break;
|
||||
|
||||
default:
|
||||
log("Internal error: unknown device-mapper task %d",
|
||||
dmt->type);
|
||||
goto bad;
|
||||
}
|
||||
|
||||
if (ioctl(fd, command, dmi) < 0) {
|
||||
log("device-mapper ioctl cmd %d failed: %s", dmt->type,
|
||||
strerror(errno));
|
||||
goto bad;
|
||||
}
|
||||
|
||||
switch (dmt->type) {
|
||||
case DM_DEVICE_CREATE:
|
||||
_add_dev_node(dmt->dev_name, dmt->dmi->minor);
|
||||
break;
|
||||
|
||||
case DM_DEVICE_REMOVE:
|
||||
_rm_dev_node(dmt->dev_name);
|
||||
break;
|
||||
}
|
||||
|
||||
dmt->dmi = dmi;
|
||||
return 1;
|
||||
|
||||
bad:
|
||||
free(dmi);
|
||||
if (fd >= 0)
|
||||
close(fd);
|
||||
return 0;
|
||||
}
|
||||
|
||||
const char *dm_dir(void)
|
||||
{
|
||||
return DM_DIR;
|
||||
}
|
257
tools/dmsetup.c
Normal file
257
tools/dmsetup.c
Normal file
@ -0,0 +1,257 @@
|
||||
/*
|
||||
* Copyright (C) 2001 Sistina Software (UK) Limited.
|
||||
*
|
||||
* This file is released under the GPL.
|
||||
*/
|
||||
|
||||
#include "libdm.h"
|
||||
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
#include <stdarg.h>
|
||||
#include <string.h>
|
||||
#include <unistd.h>
|
||||
#include <fcntl.h>
|
||||
#include <sys/stat.h>
|
||||
#include <sys/types.h>
|
||||
#include <ctype.h>
|
||||
#include <errno.h>
|
||||
|
||||
#define LINE_SIZE 1024
|
||||
|
||||
#define err(msg, x...) fprintf(stderr, msg "\n", ##x)
|
||||
|
||||
static int _parse_file(struct dm_task *dmt, const char *file)
|
||||
{
|
||||
char buffer[LINE_SIZE], *ttype, *ptr, *comment;
|
||||
FILE *fp = fopen(file, "r");
|
||||
unsigned long long start, size;
|
||||
int r = 0, n, line = 0;
|
||||
|
||||
if (!fp) {
|
||||
err("Couldn't open '%s' for reading", file);
|
||||
return 0;
|
||||
}
|
||||
|
||||
while (fgets(buffer, sizeof(buffer), fp)) {
|
||||
line++;
|
||||
|
||||
/* trim trailing space */
|
||||
for (ptr = buffer + strlen(buffer) - 1; ptr >= buffer; ptr--)
|
||||
if (!isspace((int) *ptr))
|
||||
break;
|
||||
ptr++;
|
||||
*ptr = '\0';
|
||||
|
||||
/* trim leading space */
|
||||
for (ptr = buffer; *ptr && isspace((int) *ptr); ptr++)
|
||||
;
|
||||
|
||||
if (!*ptr || *ptr == '#')
|
||||
continue;
|
||||
|
||||
if (sscanf(ptr, "%llu %llu %as %n",
|
||||
&start, &size, &ttype, &n) < 3) {
|
||||
err("%s:%d Invalid format", file, line);
|
||||
goto out;
|
||||
}
|
||||
|
||||
ptr += n;
|
||||
if ((comment = strchr(ptr, (int) '#')))
|
||||
*comment = '\0';
|
||||
|
||||
if (!dm_task_add_target(dmt, start, size, ttype, ptr))
|
||||
goto out;
|
||||
|
||||
free(ttype);
|
||||
}
|
||||
r = 1;
|
||||
|
||||
out:
|
||||
fclose(fp);
|
||||
return r;
|
||||
}
|
||||
|
||||
static int _load(int task, const char *name, const char *file)
|
||||
{
|
||||
int r = 0;
|
||||
struct dm_task *dmt;
|
||||
|
||||
if (!(dmt = dm_task_create(task)))
|
||||
return 0;
|
||||
|
||||
if (!dm_task_set_name(dmt, name))
|
||||
goto out;
|
||||
|
||||
if (!_parse_file(dmt, file))
|
||||
goto out;
|
||||
|
||||
if (!dm_task_run(dmt))
|
||||
goto out;
|
||||
|
||||
r = 1;
|
||||
|
||||
out:
|
||||
dm_task_destroy(dmt);
|
||||
|
||||
return r;
|
||||
}
|
||||
|
||||
static int _create(int argc, char **argv)
|
||||
{
|
||||
return _load(DM_DEVICE_CREATE, argv[1], argv[2]);
|
||||
}
|
||||
|
||||
static int _reload(int argc, char **argv)
|
||||
{
|
||||
return _load(DM_DEVICE_RELOAD, argv[1], argv[2]);
|
||||
}
|
||||
|
||||
|
||||
static int _simple(int task, const char *name)
|
||||
{
|
||||
int r = 0;
|
||||
|
||||
/* remove <dev_name> */
|
||||
struct dm_task *dmt;
|
||||
|
||||
if (!(dmt = dm_task_create(task)))
|
||||
return 0;
|
||||
|
||||
if (!dm_task_set_name(dmt, name))
|
||||
goto out;
|
||||
|
||||
r = dm_task_run(dmt);
|
||||
|
||||
out:
|
||||
dm_task_destroy(dmt);
|
||||
return r;
|
||||
}
|
||||
|
||||
static int _remove(int argc, char **argv)
|
||||
{
|
||||
return _simple(DM_DEVICE_REMOVE, argv[1]);
|
||||
}
|
||||
|
||||
static int _suspend(int argc, char **argv)
|
||||
{
|
||||
return _simple(DM_DEVICE_SUSPEND, argv[1]);
|
||||
}
|
||||
|
||||
static int _resume(int argc, char **argv)
|
||||
{
|
||||
return _simple(DM_DEVICE_RESUME, argv[1]);
|
||||
}
|
||||
|
||||
static int _info(int argc, char **argv)
|
||||
{
|
||||
int r = 0;
|
||||
|
||||
/* remove <dev_name> */
|
||||
struct dm_task *dmt;
|
||||
struct dm_info info;
|
||||
|
||||
if (!(dmt = dm_task_create(DM_DEVICE_INFO)))
|
||||
return 0;
|
||||
|
||||
if (!dm_task_set_name(dmt, argv[1]))
|
||||
goto out;
|
||||
|
||||
if (!dm_task_run(dmt))
|
||||
goto out;
|
||||
|
||||
if (!dm_task_get_info(dmt, &info))
|
||||
goto out;
|
||||
|
||||
if (!info.exists) {
|
||||
printf("No such device.\n");
|
||||
r = 1;
|
||||
goto out;
|
||||
}
|
||||
|
||||
printf("%s\t", info.suspended ? "SUSPENDED" : "ACTIVE");
|
||||
printf("%d\t", info.open_count);
|
||||
printf("%d\t", info.minor);
|
||||
printf("%d\n", info.target_count);
|
||||
r = 1;
|
||||
|
||||
out:
|
||||
dm_task_destroy(dmt);
|
||||
return r;
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
* dispatch table
|
||||
*/
|
||||
typedef int (*command_fn)(int argc, char **argv);
|
||||
|
||||
struct command {
|
||||
char *name;
|
||||
char *help;
|
||||
int num_args;
|
||||
command_fn fn;
|
||||
};
|
||||
|
||||
static struct command _commands[] = {
|
||||
{"create", "<dev_name> <table_file>", 2, _create},
|
||||
{"remove", "<dev_name>", 1, _remove},
|
||||
{"suspend", "<dev_name>", 1, _suspend},
|
||||
{"resume", "<dev_name>", 1, _resume},
|
||||
{"reload", "<dev_name> <table_file>", 2, _reload},
|
||||
{"info", "<dev_name>", 1, _info},
|
||||
{NULL, NULL, 0, NULL}
|
||||
};
|
||||
|
||||
static void _usage(FILE *out)
|
||||
{
|
||||
int i;
|
||||
|
||||
fprintf(out, "usage:\n");
|
||||
for (i = 0; _commands[i].name; i++)
|
||||
fprintf(out, "\t%s %s\n",
|
||||
_commands[i].name, _commands[i].help);
|
||||
return;
|
||||
}
|
||||
|
||||
struct command *_find_command(const char *name)
|
||||
{
|
||||
int i;
|
||||
|
||||
for (i = 0; _commands[i].name; i++)
|
||||
if (!strcmp(_commands[i].name, name))
|
||||
return _commands + i;
|
||||
|
||||
return NULL;
|
||||
}
|
||||
|
||||
|
||||
int main(int argc, char **argv)
|
||||
{
|
||||
struct command *c;
|
||||
|
||||
if (argc < 2) {
|
||||
_usage(stderr);
|
||||
exit(1);
|
||||
}
|
||||
|
||||
if (!(c = _find_command(argv[1]))) {
|
||||
fprintf(stderr, "Unknown command\n");
|
||||
_usage(stderr);
|
||||
exit(1);
|
||||
}
|
||||
|
||||
if (argc != c->num_args + 2) {
|
||||
fprintf(stderr, "Incorrect number of arguments\n");
|
||||
_usage(stderr);
|
||||
exit(1);
|
||||
}
|
||||
|
||||
if (!c->fn(argc - 1, argv + 1)) {
|
||||
//fprintf(stderr, "Command failed\n");
|
||||
exit(1);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user