1
0
mirror of git://sourceware.org/git/lvm2.git synced 2025-01-07 21:18:59 +03:00
lvm2/driver/device-mapper/dm-fs.c

332 lines
6.5 KiB
C
Raw Normal View History

/*
* 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";
2001-08-20 17:45:43 +04:00
const char *_control_name = "control";
static struct proc_dir_entry *_proc_dir;
2001-08-20 17:45:43 +04:00
static struct proc_dir_entry *_control;
static devfs_handle_t _dev_dir;
2001-08-20 17:45:43 +04:00
static int process_control(const char *b, const char *e, int minor,
struct dm_table **map);
static int process_table(const char *b, const char *e, int minor,
struct dm_table **map);
static int line_splitter(struct file *file, const char *buffer,
unsigned long count, void *data);
static int tok_cmp(const char *str, const char *b, const char *e);
2001-08-20 17:45:43 +04:00
typedef int (*process_fn)(const char *b, const char *e, int minor,
struct dm_table **map);
2001-08-20 17:45:43 +04:00
struct pf_data {
process_fn fn;
2001-08-20 17:45:43 +04:00
int minor;
struct dm_table *map;
2001-08-20 17:45:43 +04:00
};
2001-08-31 16:49:31 +04:00
int dm_fs_init(void)
{
2001-08-20 17:45:43 +04:00
struct pf_data *pfd = kmalloc(sizeof(*pfd), GFP_KERNEL);
2001-08-20 17:45:43 +04:00
if (!pfd)
return 0;
2001-08-20 17:45:43 +04:00
_dev_dir = devfs_mk_dir(0, _fs_dir, NULL);
2001-08-20 17:45:43 +04:00
if (!(_proc_dir = create_proc_entry(_fs_dir, S_IFDIR, &proc_root)))
goto fail;
if (!(_control = create_proc_entry(_control_name, S_IWUSR, _proc_dir)))
2001-08-20 17:45:43 +04:00
goto fail;
_control->write_proc = line_splitter;
pfd->fn = process_control;
2001-08-20 17:45:43 +04:00
pfd->minor = -1;
_control->data = pfd;
2001-08-22 19:02:55 +04:00
return 0;
2001-08-20 17:45:43 +04:00
fail:
2001-08-31 16:49:31 +04:00
dm_fs_exit();
2001-08-22 19:02:55 +04:00
return -ENOMEM;
}
2001-08-31 16:49:31 +04:00
void dm_fs_exit(void)
{
2001-08-20 17:45:43 +04:00
if (_control) {
remove_proc_entry(_control_name, _proc_dir);
_control = 0;
}
2001-08-20 17:45:43 +04:00
if (_proc_dir) {
remove_proc_entry(_fs_dir, &proc_root);
_proc_dir = 0;
}
2001-08-20 17:45:43 +04:00
if (_dev_dir)
devfs_unregister(_dev_dir);
}
int dm_fs_add(struct mapped_device *md)
{
struct pf_data *pfd = kmalloc(sizeof(*pfd), GFP_KERNEL);
if (!pfd)
return -ENOMEM;
pfd->fn = process_table;
pfd->minor = MINOR(md->dev);
pfd->map = 0;
if (!(md->pde = create_proc_entry(md->name, S_IRUGO | S_IWUSR,
_proc_dir))) {
kfree(pfd);
return -ENOMEM;
}
md->pde->write_proc = line_splitter;
md->pde->data = pfd;
md->devfs_entry =
devfs_register(_dev_dir, md->name, DEVFS_FL_CURRENT_OWNER,
MAJOR(md->dev), MINOR(md->dev),
S_IFBLK | S_IRUSR | S_IWUSR | S_IRGRP,
&dm_blk_dops, NULL);
if (!md->devfs_entry) {
kfree(pfd);
remove_proc_entry(md->name, _proc_dir);
md->pde = 0;
return -ENOMEM;
}
return 0;
}
int dm_fs_remove(struct mapped_device *md)
{
if (md->pde) {
struct pf_data *pfd = (struct pf_data *) md->pde->data;
if (pfd->map)
dm_table_destroy(pfd->map);
kfree(md->pde->data);
remove_proc_entry(md->name, _proc_dir);
md->pde = 0;
}
devfs_unregister(md->devfs_entry);
md->devfs_entry = 0;
return 0;
}
static int process_control(const char *b, const char *e, int minor,
struct dm_table **map)
{
2001-08-20 17:45:43 +04:00
const char *wb, *we;
char name[64];
2001-08-20 17:45:43 +04:00
int create = 0;
2001-08-20 17:45:43 +04:00
/*
* create <name> [minor]
* remove <name>
*/
if (get_word(b, e, &wb, &we))
2001-08-20 17:45:43 +04:00
return -EINVAL;
b = we;
if (!tok_cmp("create", wb, we))
2001-08-20 17:45:43 +04:00
create = 1;
else if (tok_cmp("remove", wb, we))
2001-08-20 17:45:43 +04:00
return -EINVAL;
if (get_word(b, e, &wb, &we))
2001-08-20 17:45:43 +04:00
return -EINVAL;
b = we;
tok_cpy(name, sizeof(name), wb, we);
if (!create)
return dm_remove(name);
else {
if (!get_word(b, e, &wb, &we)) {
minor = simple_strtol(wb, (char **) &we, 10);
2001-08-20 17:45:43 +04:00
if (we == wb)
return -EINVAL;
}
return dm_create(name, minor);
}
2001-08-20 17:45:43 +04:00
return -EINVAL;
}
static int process_table(const char *b, const char *e, int minor,
struct dm_table **map)
{
2001-08-20 17:45:43 +04:00
const char *wb, *we;
2001-08-29 17:58:48 +04:00
struct mapped_device *md = dm_find_by_minor(minor);
struct dm_table *table = *map;
void *context;
int r;
if (!md)
return -ENXIO;
if (get_word(b, e, &wb, &we))
2001-08-20 17:45:43 +04:00
return -EINVAL;
if (!tok_cmp("begin", b, e)) {
2001-08-20 17:45:43 +04:00
/* suspend the device if it's active */
dm_suspend(md);
2001-08-20 17:45:43 +04:00
/* start loading a table */
table = *map = dm_table_create();
} else if (!tok_cmp("end", b, e)) {
2001-08-20 17:45:43 +04:00
/* activate the device ... <evil chuckle> ... */
if (table) {
dm_table_complete(table);
dm_bind(md, table);
dm_activate(md);
}
2001-08-20 17:45:43 +04:00
} else {
/* add the new entry */
char target[64];
2001-08-31 16:49:31 +04:00
struct target_type *t;
offset_t start, size, high;
size_t len;
2001-08-20 17:45:43 +04:00
if (get_number(&b, e, &start))
return -EINVAL;
2001-08-20 17:45:43 +04:00
if (get_number(&b, e, &size))
return -EINVAL;
2001-08-20 17:45:43 +04:00
if (get_word(b, e, &wb, &we))
return -EINVAL;
2001-08-20 17:45:43 +04:00
len = we - wb;
if (len > sizeof(target))
return -EINVAL;
2001-08-20 17:45:43 +04:00
strncpy(target, wb, len);
target[len] = '\0';
if (!(t = dm_get_target(target)))
return -EINVAL;
2001-08-20 17:45:43 +04:00
/* check there isn't a gap */
if ((table->num_targets &&
start != table->highs[table->num_targets - 1] + 1) ||
(!table->num_targets && start)) {
WARN("gap in target ranges");
return -EINVAL;
}
2001-09-02 14:49:20 +04:00
if ((r = t->ctr(table, start, size, we, e, &context)))
return r;
2001-08-20 17:45:43 +04:00
2001-09-02 14:49:20 +04:00
high = start + (size - 1);
if ((r = dm_table_add_target(table, high, t, context))) {
t->dtr(table, context);
return r;
}
}
2001-08-20 17:45:43 +04:00
return 0;
2001-08-20 17:45:43 +04:00
}
static int line_splitter(struct file *file, const char *buffer,
unsigned long count, void *data)
2001-08-20 17:45:43 +04:00
{
int r;
2001-08-20 17:45:43 +04:00
const char *b = buffer, *e = buffer + count, *lb;
struct pf_data *pfd = (struct pf_data *) data;
while(b < e) {
b = eat_space(b, e);
2001-08-20 17:45:43 +04:00
if (b == e)
2001-08-22 18:30:30 +04:00
break;
2001-08-20 17:45:43 +04:00
lb = b;
while((b != e) && *b != '\n')
b++;
if ((r = pfd->fn(lb, b, pfd->minor, &pfd->map)))
return r;
}
2001-08-20 17:45:43 +04:00
return count;
}
static int tok_cmp(const char *str, const char *b, const char *e)
2001-08-20 17:45:43 +04:00
{
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;
2001-08-20 17:45:43 +04:00
return -1;
}