mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-18 10:04:20 +03:00
488 lines
8.6 KiB
C
488 lines
8.6 KiB
C
/*
|
|
* Copyright (C) 2001 Sistina Software (UK) Limited.
|
|
*
|
|
* This file is released under the GPL.
|
|
*/
|
|
|
|
#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 <ctype.h>
|
|
#include <errno.h>
|
|
#include <unistd.h>
|
|
#include <getopt.h>
|
|
#include <linux/kdev_t.h>
|
|
|
|
#define LINE_SIZE 1024
|
|
|
|
#define err(msg, x...) fprintf(stderr, msg "\n", ##x)
|
|
|
|
/*
|
|
* We have only very simple switches ATM.
|
|
*/
|
|
enum {
|
|
READ_ONLY = 0,
|
|
MINOR_ARG,
|
|
NUM_SWITCHES
|
|
};
|
|
|
|
static int _switches[NUM_SWITCHES];
|
|
static int _values[NUM_SWITCHES];
|
|
|
|
/*
|
|
* Commands
|
|
*/
|
|
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, const char *uuid)
|
|
{
|
|
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 (uuid && !dm_task_set_uuid(dmt, uuid))
|
|
goto out;
|
|
|
|
if (!_parse_file(dmt, file))
|
|
goto out;
|
|
|
|
if (_switches[READ_ONLY] && !dm_task_set_ro(dmt))
|
|
goto out;
|
|
|
|
if (_switches[MINOR_ARG] && !dm_task_set_minor(dmt, _values[MINOR_ARG]))
|
|
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], argv[3]);
|
|
}
|
|
|
|
static int _reload(int argc, char **argv)
|
|
{
|
|
return _load(DM_DEVICE_RELOAD, argv[1], argv[2], NULL);
|
|
}
|
|
|
|
static int _rename(int argc, char **argv)
|
|
{
|
|
int r = 0;
|
|
struct dm_task *dmt;
|
|
|
|
if (!(dmt = dm_task_create(DM_DEVICE_RENAME)))
|
|
return 0;
|
|
|
|
if (!dm_task_set_name(dmt, argv[1]))
|
|
goto out;
|
|
|
|
if (!dm_task_set_newname(dmt, argv[2]))
|
|
goto out;
|
|
|
|
if (!dm_task_run(dmt))
|
|
goto out;
|
|
|
|
r = 1;
|
|
|
|
out:
|
|
dm_task_destroy(dmt);
|
|
|
|
return r;
|
|
}
|
|
|
|
static int _version(int argc, char **argv)
|
|
{
|
|
int r = 0;
|
|
struct dm_task *dmt;
|
|
char version[80];
|
|
|
|
if (dm_get_library_version(version, sizeof(version)))
|
|
printf("Library version: %s\n", version);
|
|
|
|
if (!(dmt = dm_task_create(DM_DEVICE_VERSION)))
|
|
return 0;
|
|
|
|
if (!dm_task_run(dmt))
|
|
goto out;
|
|
|
|
if (!dm_task_get_driver_version(dmt, (char *) &version,
|
|
sizeof(version)))
|
|
goto out;
|
|
|
|
printf("Driver version: %s\n", version);
|
|
|
|
r = 1;
|
|
|
|
out:
|
|
dm_task_destroy(dmt);
|
|
|
|
return r;
|
|
}
|
|
|
|
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_all(int argc, char **argv)
|
|
{
|
|
return _simple(DM_DEVICE_REMOVE_ALL, "");
|
|
}
|
|
|
|
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 _wait(int argc, char **argv)
|
|
{
|
|
return _simple(DM_DEVICE_WAITEVENT, argv[1]);
|
|
}
|
|
|
|
static int _status(int argc, char **argv)
|
|
{
|
|
int r = 0;
|
|
struct dm_task *dmt;
|
|
void *next = NULL;
|
|
uint64_t start, length;
|
|
char *target_type = NULL;
|
|
char *params;
|
|
int cmd;
|
|
|
|
if (!strcmp(argv[0], "status"))
|
|
cmd = DM_DEVICE_STATUS;
|
|
else
|
|
cmd = DM_DEVICE_TABLE;
|
|
|
|
if (!(dmt = dm_task_create(cmd)))
|
|
return 0;
|
|
|
|
if (!dm_task_set_name(dmt, argv[1]))
|
|
goto out;
|
|
|
|
if (!dm_task_run(dmt))
|
|
goto out;
|
|
|
|
/* Fetch targets and print 'em */
|
|
do {
|
|
next = dm_get_next_target(dmt, next, &start, &length,
|
|
&target_type, ¶ms);
|
|
if (target_type) {
|
|
printf("%"PRIu64" %"PRIu64" %s %s\n",
|
|
start, length, target_type, params);
|
|
}
|
|
} while (next);
|
|
|
|
r = 1;
|
|
|
|
out:
|
|
dm_task_destroy(dmt);
|
|
return r;
|
|
|
|
}
|
|
|
|
static int _info(int argc, char **argv)
|
|
{
|
|
int r = 0;
|
|
const char *uuid;
|
|
|
|
/* 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("Device does not exist.\n");
|
|
r = 1;
|
|
goto out;
|
|
}
|
|
|
|
printf("Name: %s\n", argv[1]);
|
|
|
|
printf("State: %s\n",
|
|
info.suspended ? "SUSPENDED" : "ACTIVE");
|
|
|
|
if (info.open_count != -1)
|
|
printf("Open count: %d\n", info.open_count);
|
|
|
|
printf("Major, minor: %d, %d\n", info.major, info.minor);
|
|
|
|
if (info.target_count != -1)
|
|
printf("Number of targets: %d\n", info.target_count);
|
|
|
|
if ((uuid = dm_task_get_uuid(dmt)) && *uuid)
|
|
printf("UUID: %s\n", uuid);
|
|
|
|
r = 1;
|
|
|
|
out:
|
|
dm_task_destroy(dmt);
|
|
return r;
|
|
}
|
|
|
|
static int _deps(int argc, char **argv)
|
|
{
|
|
int r = 0, i;
|
|
struct dm_deps *deps;
|
|
|
|
/* remove <dev_name> */
|
|
struct dm_task *dmt;
|
|
struct dm_info info;
|
|
|
|
if (!(dmt = dm_task_create(DM_DEVICE_DEPS)))
|
|
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 (!(deps = dm_task_get_deps(dmt)))
|
|
goto out;
|
|
|
|
if (!info.exists) {
|
|
printf("Device does not exist.\n");
|
|
r = 1;
|
|
goto out;
|
|
}
|
|
|
|
printf("%d dependencies\t:", deps->count);
|
|
|
|
for (i = 0; i < deps->count; i++)
|
|
printf(" (%d, %d)",
|
|
(int) MAJOR(deps->device[i]),
|
|
(int) MINOR(deps->device[i]));
|
|
printf("\n");
|
|
|
|
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 min_args;
|
|
int max_args;
|
|
command_fn fn;
|
|
};
|
|
|
|
static struct command _commands[] = {
|
|
{"create", "<dev_name> <table_file> [<uuid>]", 2, 3, _create},
|
|
{"remove", "<dev_name>", 1, 1, _remove},
|
|
{"remove_all", "", 0, 0, _remove_all},
|
|
{"suspend", "<dev_name>", 1, 1, _suspend},
|
|
{"resume", "<dev_name>", 1, 1, _resume},
|
|
{"reload", "<dev_name> <table_file>", 2, 2, _reload},
|
|
{"info", "<dev_name>", 1, 1, _info},
|
|
{"deps", "<dev_name>", 1, 1, _deps},
|
|
{"rename", "<dev_name> <new_name>", 2, 2, _rename},
|
|
{"status", "<dev_name>", 1, 1, _status},
|
|
{"table", "<dev_name>", 1, 1, _status},
|
|
{"wait", "<dev_name>", 1, 1, _wait},
|
|
{"version", "", 0, 0, _version},
|
|
{NULL, NULL, 0, 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;
|
|
}
|
|
|
|
static 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;
|
|
}
|
|
|
|
static int _process_switches(int *argc, char ***argv)
|
|
{
|
|
int index;
|
|
char c;
|
|
|
|
static struct option long_options[] = {
|
|
{"read-only", 0, NULL, READ_ONLY},
|
|
{"minor", 1, NULL, MINOR_ARG},
|
|
{"", 0, NULL, 0}
|
|
};
|
|
|
|
/*
|
|
* Zero all the index counts.
|
|
*/
|
|
memset(&_switches, 0, sizeof(_switches));
|
|
memset(&_values, 0, sizeof(_values));
|
|
|
|
while ((c = getopt_long(*argc, *argv, "m:r",
|
|
long_options, &index)) != -1) {
|
|
if (c == 'r' || index == READ_ONLY)
|
|
_switches[READ_ONLY]++;
|
|
if (c == 'm' || index == MINOR_ARG) {
|
|
_switches[MINOR_ARG]++;
|
|
_values[MINOR_ARG] = atoi(optarg);
|
|
}
|
|
}
|
|
|
|
*argv += optind;
|
|
*argc -= optind;
|
|
return 1;
|
|
}
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
struct command *c;
|
|
|
|
if (!_process_switches(&argc, &argv)) {
|
|
fprintf(stderr, "Couldn't process command line switches.\n");
|
|
exit(1);
|
|
}
|
|
|
|
if (argc == 0) {
|
|
_usage(stderr);
|
|
exit(1);
|
|
}
|
|
|
|
if (!(c = _find_command(argv[0]))) {
|
|
fprintf(stderr, "Unknown command\n");
|
|
_usage(stderr);
|
|
exit(1);
|
|
}
|
|
|
|
if (argc < c->min_args + 1 || argc > c->max_args + 1) {
|
|
fprintf(stderr, "Incorrect number of arguments\n");
|
|
_usage(stderr);
|
|
exit(1);
|
|
}
|
|
|
|
if (!c->fn(argc, argv)) {
|
|
fprintf(stderr, "Command failed\n");
|
|
exit(1);
|
|
}
|
|
|
|
return 0;
|
|
}
|