1
0
mirror of git://sourceware.org/git/lvm2.git synced 2025-01-24 06:04:19 +03:00

319 lines
6.2 KiB
C

/*
* dm.c
*
* Copyright (C) 2001 Sistina Software
*
* This software is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License as
* published by the Free Software Foundation; either version 2, or (at
* your option) any later version.
*
* This software is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with GNU CC; see the file COPYING. If not, write to
* the Free Software Foundation, 59 Temple Place - Suite 330,
* Boston, MA 02111-1307, USA.
*/
/*
* procfs and devfs handling for device mapper
*
* Changelog
*
* 16/08/2001 - First version [Joe Thornber]
*/
#include "dm.h"
#include <linux/proc_fs.h>
#include <linux/ctype.h>
/*
* /dev/device-mapper/control is the control char device used to
* create/destroy mapping devices.
*
* When a mapping device called <name> is created it appears as
* /dev/device-mapper/<name>. In addition the interface to control the
* mapping will appear in /proc/device-mapper/<name>.
*/
const char *_fs_dir = "device-mapper";
const char *_control_name = "control";
static struct proc_dir_entry *_proc_dir;
static struct proc_dir_entry *_control;
static devfs_handle_t _dev_dir;
static int _line_splitter(struct file *file, const char *buffer,
unsigned long count, void *data);
int _process_control(const char *b, const char *e, int minor);
static int _process_table(const char *b, const char *e, int minor);
static const char *_eat_space(const char *b, const char *e);
static int _get_word(const char *b, const char *e,
const char **wb, const char **we);
static int _tok_cmp(const char *str, const char *b, const char *e);
static void _tok_cpy(char *dest, size_t max,
const char *b, const char *e);
typedef int (*process_fn)(const char *b, const char *e, int minor);
struct pf_data {
process_fn fn;
int minor;
};
int dm_init_fs()
{
struct pf_data *pfd = kmalloc(sizeof(*pfd), GFP_KERNEL);
if (!pfd)
return 0;
_dev_dir = devfs_mk_dir(0, _fs_dir, NULL);
if (!(_proc_dir = create_proc_entry(_fs_dir, S_IFDIR, &proc_root)))
goto fail;
if (!(_control = create_proc_entry(_control_name, S_IWUGO, _proc_dir)))
goto fail;
_control->write_proc = _line_splitter;
pfd->fn = _process_control;
pfd->minor = -1;
_control->data = pfd;
return 1;
fail:
dm_fin_fs();
return 0;
}
void dm_fin_fs(void)
{
if (_control) {
remove_proc_entry(_control_name, _proc_dir);
_control = 0;
}
if (_proc_dir) {
remove_proc_entry(_fs_dir, &proc_root);
_proc_dir = 0;
}
if (_dev_dir)
devfs_unregister(_dev_dir);
}
int _process_control(const char *b, const char *e, int minor)
{
const char *wb, *we;
char name[64];
int create = 0;
struct pf_data *pfd;
struct proc_dir_entry *pde;
/*
* create <name> [minor]
* remove <name>
*/
if (!_get_word(b, e, &wb, &we))
return -EINVAL;
b = we;
if (!_tok_cmp("create", wb, we))
create = 1;
else if (_tok_cmp("remove", wb, we))
return -EINVAL;
if (!_get_word(b, e, &wb, &we))
return -EINVAL;
b = we;
_tok_cpy(name, sizeof(name), wb, we);
if (create) {
if (_get_word(b, e, &wb, &we)) {
minor = simple_strtol(wb, (char **) &we, 10);
if (we == wb)
return -EINVAL;
}
/* FIXME: quick hack */
pfd = kmalloc(sizeof(*pfd), GFP_KERNEL);
if (!pfd)
return -ENOMEM;
pfd->fn = _process_table;
pfd->minor = minor;
if (!(pde = create_proc_entry(name, 0, _proc_dir)))
return -ENOMEM;
pde->write_proc = _line_splitter;
pde->data = pfd;
return dm_create(name, minor);
} else {
if (!_get_word(b, e, &wb, &we))
return -EINVAL;
_tok_cpy(name, sizeof(name), wb, we);
return dm_remove(name);
}
return -EINVAL;
}
static int _process_table(const char *b, const char *e, int minor)
{
const char *wb, *we;
struct mapped_device *md = dm_find_minor(minor);
void *context;
int ret;
if (!md)
return -ENXIO;
if (!_get_word(b, e, &wb, &we))
return -EINVAL;
if (!_tok_cmp("begin", b, e)) {
/* suspend the device if it's active */
dm_suspend(md);
/* start loading a table */
dm_start_table(md);
} else if (!_tok_cmp("end", b, e)) {
/* activate the device ... <evil chuckle> ... */
dm_complete_table(md);
dm_activate(md);
} else {
/* add the new entry */
int len = we - wb;
char high_s[64], *ptr;
char target[64];
struct target *t;
offset_t high;
if (len > sizeof(high_s))
return 0;
strncpy(high_s, wb, we - wb);
high_s[len] = '\0';
high = simple_strtol(high_s, &ptr, 10);
if (ptr == high_s)
return 0;
b = we;
if (!_get_word(b, e, &wb, &we))
return 0;
len = we - wb;
if (len > sizeof(target))
return 0;
strncpy(target, wb, len);
target[len] = '\0';
if (!(t = dm_get_target(target)))
return 0;
if ((ret = t->ctr(md->highs[md->num_targets - 1],
high, md, we, &context)))
return ret;
dm_add_entry(md, high, t->map, context);
}
return 1;
}
static const char *_eat_space(const char *b, const char *e)
{
while(b != e && isspace((int) *b))
b++;
return b;
}
static int _get_word(const char *b, const char *e,
const char **wb, const char **we)
{
b = _eat_space(b, e);
if (b == e)
return 0;
*wb = b;
while(b != e && !isspace((int) b))
b++;
*we = b;
return 1;
}
static int _line_splitter(struct file *file, const char *buffer,
unsigned long count, void *data)
{
const char *b = buffer, *e = buffer + count, *lb;
struct pf_data *pfd = (struct pf_data *) data;
while(b < e) {
b = _eat_space(b, e);
if (b == e)
return 0;
lb = b;
while((b != e) && *b != '\n')
b++;
if (!pfd->fn(lb, b, pfd->minor))
return lb - buffer;
}
return count;
}
static int _tok_cmp(const char *str, const char *b, const char *e)
{
while (*str && b != e) {
if (*str < *b)
return -1;
if (*str > *b)
return 1;
str++, b++;
}
if (!*str && b == e)
return 0;
if (*str)
return 1;
return -1;
}
static void _tok_cpy(char *dest, size_t max,
const char *b, const char *e)
{
size_t len = e - b;
if (len > --max)
len = max;
strncpy(dest, b, len);
dest[len] = '0';
}