mirror of
git://sourceware.org/git/lvm2.git
synced 2025-01-18 10:04:20 +03:00
o map loads ok now
o request function appears to work, but something is segfaulting when i mke2fs
This commit is contained in:
parent
0bdf99c171
commit
3cf3ab579e
@ -39,7 +39,7 @@ struct mapped_device;
|
|||||||
typedef unsigned int offset_t;
|
typedef unsigned int offset_t;
|
||||||
|
|
||||||
typedef int (*dm_ctr_fn)(offset_t b, offset_t e, struct mapped_device *md,
|
typedef int (*dm_ctr_fn)(offset_t b, offset_t e, struct mapped_device *md,
|
||||||
const char *context, void **result);
|
const char *cb, const char *ce, void **result);
|
||||||
typedef void (*dm_dtr_fn)(void *c);
|
typedef void (*dm_dtr_fn)(void *c);
|
||||||
typedef int (*dm_map_fn)(struct buffer_head *bh, void *context);
|
typedef int (*dm_map_fn)(struct buffer_head *bh, void *context);
|
||||||
|
|
||||||
|
@ -49,15 +49,14 @@ static struct proc_dir_entry *_control;
|
|||||||
|
|
||||||
static devfs_handle_t _dev_dir;
|
static devfs_handle_t _dev_dir;
|
||||||
|
|
||||||
static int _line_splitter(struct file *file, const char *buffer,
|
static int line_splitter(struct file *file, const char *buffer,
|
||||||
unsigned long count, void *data);
|
unsigned long count, void *data);
|
||||||
int _process_control(const char *b, const char *e, int minor);
|
static int process_control(const char *b, const char *e, int minor);
|
||||||
static int _process_table(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,
|
||||||
static int _get_word(const char *b, const char *e,
|
|
||||||
const char **wb, const char **we);
|
const char **wb, const char **we);
|
||||||
static int _tok_cmp(const char *str, const char *b, const char *e);
|
static int tok_cmp(const char *str, const char *b, const char *e);
|
||||||
static void _tok_cpy(char *dest, size_t max,
|
static void tok_cpy(char *dest, size_t max,
|
||||||
const char *b, const char *e);
|
const char *b, const char *e);
|
||||||
|
|
||||||
typedef int (*process_fn)(const char *b, const char *e, int minor);
|
typedef int (*process_fn)(const char *b, const char *e, int minor);
|
||||||
@ -82,9 +81,9 @@ int dm_init_fs()
|
|||||||
if (!(_control = create_proc_entry(_control_name, S_IWUSR, _proc_dir)))
|
if (!(_control = create_proc_entry(_control_name, S_IWUSR, _proc_dir)))
|
||||||
goto fail;
|
goto fail;
|
||||||
|
|
||||||
_control->write_proc = _line_splitter;
|
_control->write_proc = line_splitter;
|
||||||
|
|
||||||
pfd->fn = _process_control;
|
pfd->fn = process_control;
|
||||||
pfd->minor = -1;
|
pfd->minor = -1;
|
||||||
_control->data = pfd;
|
_control->data = pfd;
|
||||||
|
|
||||||
@ -118,7 +117,7 @@ int dm_fs_add(struct mapped_device *md)
|
|||||||
if (!pfd)
|
if (!pfd)
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
pfd->fn = _process_table;
|
pfd->fn = process_table;
|
||||||
pfd->minor = MINOR(md->dev);
|
pfd->minor = MINOR(md->dev);
|
||||||
|
|
||||||
if (!(md->pde = create_proc_entry(md->name, S_IRUGO | S_IWUSR,
|
if (!(md->pde = create_proc_entry(md->name, S_IRUGO | S_IWUSR,
|
||||||
@ -127,7 +126,7 @@ int dm_fs_add(struct mapped_device *md)
|
|||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
md->pde->write_proc = _line_splitter;
|
md->pde->write_proc = line_splitter;
|
||||||
md->pde->data = pfd;
|
md->pde->data = pfd;
|
||||||
|
|
||||||
md->devfs_entry =
|
md->devfs_entry =
|
||||||
@ -140,7 +139,7 @@ int dm_fs_add(struct mapped_device *md)
|
|||||||
kfree(pfd);
|
kfree(pfd);
|
||||||
remove_proc_entry(md->name, _proc_dir);
|
remove_proc_entry(md->name, _proc_dir);
|
||||||
md->pde = 0;
|
md->pde = 0;
|
||||||
return -ENOMEM; /* FIXME: better error ? */
|
return -ENOMEM;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -159,7 +158,7 @@ int dm_fs_remove(struct mapped_device *md)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int _process_control(const char *b, const char *e, int minor)
|
static int process_control(const char *b, const char *e, int minor)
|
||||||
{
|
{
|
||||||
const char *wb, *we;
|
const char *wb, *we;
|
||||||
char name[64];
|
char name[64];
|
||||||
@ -169,27 +168,27 @@ int _process_control(const char *b, const char *e, int minor)
|
|||||||
* create <name> [minor]
|
* create <name> [minor]
|
||||||
* remove <name>
|
* remove <name>
|
||||||
*/
|
*/
|
||||||
if (!_get_word(b, e, &wb, &we))
|
if (get_word(b, e, &wb, &we))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
b = we;
|
b = we;
|
||||||
|
|
||||||
if (!_tok_cmp("create", wb, we))
|
if (!tok_cmp("create", wb, we))
|
||||||
create = 1;
|
create = 1;
|
||||||
|
|
||||||
else if (_tok_cmp("remove", wb, we))
|
else if (tok_cmp("remove", wb, we))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
if (!_get_word(b, e, &wb, &we))
|
if (get_word(b, e, &wb, &we))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
b = we;
|
b = we;
|
||||||
|
|
||||||
_tok_cpy(name, sizeof(name), wb, we);
|
tok_cpy(name, sizeof(name), wb, we);
|
||||||
|
|
||||||
if (!create)
|
if (!create)
|
||||||
return dm_remove(name);
|
return dm_remove(name);
|
||||||
|
|
||||||
else {
|
else {
|
||||||
if (_get_word(b, e, &wb, &we)) {
|
if (!get_word(b, e, &wb, &we)) {
|
||||||
minor = simple_strtol(wb, (char **) &we, 10);
|
minor = simple_strtol(wb, (char **) &we, 10);
|
||||||
|
|
||||||
if (we == wb)
|
if (we == wb)
|
||||||
@ -202,27 +201,27 @@ int _process_control(const char *b, const char *e, int minor)
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int _process_table(const char *b, const char *e, int minor)
|
static int process_table(const char *b, const char *e, int minor)
|
||||||
{
|
{
|
||||||
const char *wb, *we;
|
const char *wb, *we;
|
||||||
struct mapped_device *md = dm_find_minor(minor);
|
struct mapped_device *md = dm_find_minor(minor);
|
||||||
void *context;
|
void *context;
|
||||||
int ret;
|
int r;
|
||||||
|
|
||||||
if (!md)
|
if (!md)
|
||||||
return -ENXIO;
|
return -ENXIO;
|
||||||
|
|
||||||
if (!_get_word(b, e, &wb, &we))
|
if (get_word(b, e, &wb, &we))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
if (!_tok_cmp("begin", b, e)) {
|
if (!tok_cmp("begin", b, e)) {
|
||||||
/* suspend the device if it's active */
|
/* suspend the device if it's active */
|
||||||
dm_suspend(md);
|
dm_suspend(md);
|
||||||
|
|
||||||
/* start loading a table */
|
/* start loading a table */
|
||||||
dm_start_table(md);
|
dm_start_table(md);
|
||||||
|
|
||||||
} else if (!_tok_cmp("end", b, e)) {
|
} else if (!tok_cmp("end", b, e)) {
|
||||||
/* activate the device ... <evil chuckle> ... */
|
/* activate the device ... <evil chuckle> ... */
|
||||||
dm_complete_table(md);
|
dm_complete_table(md);
|
||||||
dm_activate(md);
|
dm_activate(md);
|
||||||
@ -233,66 +232,61 @@ static int _process_table(const char *b, const char *e, int minor)
|
|||||||
char high_s[64], *ptr;
|
char high_s[64], *ptr;
|
||||||
char target[64];
|
char target[64];
|
||||||
struct target *t;
|
struct target *t;
|
||||||
offset_t high;
|
offset_t last = 0, high;
|
||||||
|
|
||||||
if (len > sizeof(high_s))
|
if (len > sizeof(high_s))
|
||||||
return 0;
|
return -EINVAL;
|
||||||
|
|
||||||
strncpy(high_s, wb, we - wb);
|
strncpy(high_s, wb, we - wb);
|
||||||
high_s[len] = '\0';
|
high_s[len] = '\0';
|
||||||
|
|
||||||
high = simple_strtol(high_s, &ptr, 10);
|
high = simple_strtol(high_s, &ptr, 10);
|
||||||
if (ptr == high_s)
|
if (ptr == high_s)
|
||||||
return 0;
|
return -EINVAL;
|
||||||
|
|
||||||
b = we;
|
b = we;
|
||||||
if (!_get_word(b, e, &wb, &we))
|
if (get_word(b, e, &wb, &we))
|
||||||
return 0;
|
return -EINVAL;
|
||||||
|
|
||||||
len = we - wb;
|
len = we - wb;
|
||||||
if (len > sizeof(target))
|
if (len > sizeof(target))
|
||||||
return 0;
|
return -EINVAL;
|
||||||
|
|
||||||
strncpy(target, wb, len);
|
strncpy(target, wb, len);
|
||||||
target[len] = '\0';
|
target[len] = '\0';
|
||||||
|
|
||||||
if (!(t = dm_get_target(target)))
|
if (!(t = dm_get_target(target)))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (md->num_targets)
|
||||||
|
last = md->highs[md->num_targets - 1] + 1;
|
||||||
|
|
||||||
|
if ((r = t->ctr(last, high, md, we, e, &context)))
|
||||||
|
return r;
|
||||||
|
|
||||||
|
if ((r = dm_add_entry(md, high, t->map, context)))
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
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 int get_word(const char *b, const char *e,
|
||||||
}
|
|
||||||
|
|
||||||
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)
|
const char **wb, const char **we)
|
||||||
{
|
{
|
||||||
b = _eat_space(b, e);
|
b = eat_space(b, e);
|
||||||
|
|
||||||
if (b == e)
|
if (b == e)
|
||||||
return 0;
|
return -EINVAL;
|
||||||
|
|
||||||
*wb = b;
|
*wb = b;
|
||||||
while(b != e && !isspace((int) *b))
|
while(b != e && !isspace((int) *b))
|
||||||
b++;
|
b++;
|
||||||
*we = b;
|
*we = b;
|
||||||
return 1;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int _line_splitter(struct file *file, const char *buffer,
|
static int line_splitter(struct file *file, const char *buffer,
|
||||||
unsigned long count, void *data)
|
unsigned long count, void *data)
|
||||||
{
|
{
|
||||||
int r;
|
int r;
|
||||||
@ -300,7 +294,7 @@ static int _line_splitter(struct file *file, const char *buffer,
|
|||||||
struct pf_data *pfd = (struct pf_data *) data;
|
struct pf_data *pfd = (struct pf_data *) data;
|
||||||
|
|
||||||
while(b < e) {
|
while(b < e) {
|
||||||
b = _eat_space(b, e);
|
b = eat_space(b, e);
|
||||||
if (b == e)
|
if (b == e)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@ -315,7 +309,7 @@ static int _line_splitter(struct file *file, const char *buffer,
|
|||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int _tok_cmp(const char *str, const char *b, const char *e)
|
static int tok_cmp(const char *str, const char *b, const char *e)
|
||||||
{
|
{
|
||||||
while (*str && b != e) {
|
while (*str && b != e) {
|
||||||
if (*str < *b)
|
if (*str < *b)
|
||||||
@ -336,7 +330,7 @@ static int _tok_cmp(const char *str, const char *b, const char *e)
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void _tok_cpy(char *dest, size_t max,
|
static void tok_cpy(char *dest, size_t max,
|
||||||
const char *b, const char *e)
|
const char *b, const char *e)
|
||||||
{
|
{
|
||||||
size_t len = e - b;
|
size_t len = e - b;
|
||||||
|
@ -27,20 +27,20 @@
|
|||||||
|
|
||||||
#include "dm.h"
|
#include "dm.h"
|
||||||
|
|
||||||
static int _alloc_targets(struct mapped_device *md, int num);
|
static int alloc_targets(struct mapped_device *md, int num);
|
||||||
|
|
||||||
static inline ulong _round_up(ulong n, ulong size)
|
static inline ulong round_up(ulong n, ulong size)
|
||||||
{
|
{
|
||||||
ulong r = n % size;
|
ulong r = n % size;
|
||||||
return n + (r ? (size - r) : 0);
|
return n + (r ? (size - r) : 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline ulong _div_up(ulong n, ulong size)
|
static inline ulong div_up(ulong n, ulong size)
|
||||||
{
|
{
|
||||||
return _round_up(n, size) / size;
|
return round_up(n, size) / size;
|
||||||
}
|
}
|
||||||
|
|
||||||
static offset_t _high(struct mapped_device *md, int l, int n)
|
static offset_t high(struct mapped_device *md, int l, int n)
|
||||||
{
|
{
|
||||||
while (1) {
|
while (1) {
|
||||||
if (n >= md->counts[l])
|
if (n >= md->counts[l])
|
||||||
@ -54,7 +54,7 @@ static offset_t _high(struct mapped_device *md, int l, int n)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static int _setup_btree_index(int l, struct mapped_device *md)
|
static int setup_btree_index(int l, struct mapped_device *md)
|
||||||
{
|
{
|
||||||
int n, c, cn;
|
int n, c, cn;
|
||||||
|
|
||||||
@ -62,22 +62,27 @@ static int _setup_btree_index(int l, struct mapped_device *md)
|
|||||||
offset_t *k = md->index[l] + (n * KEYS_PER_NODE);
|
offset_t *k = md->index[l] + (n * KEYS_PER_NODE);
|
||||||
|
|
||||||
for (c = 0; c < KEYS_PER_NODE; c++)
|
for (c = 0; c < KEYS_PER_NODE; c++)
|
||||||
k[c] = _high(md, l + 1, cn++);
|
k[c] = high(md, l + 1, cn++);
|
||||||
cn++;
|
cn++;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 1;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void dm_free_table(struct mapped_device *md)
|
void dm_free_table(struct mapped_device *md)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
for (i = 0; i < md->depth; i++)
|
for (i = 0; i < md->depth; i++) {
|
||||||
vfree(md->index[i]);
|
vfree(md->index[i]);
|
||||||
|
md->index[i] = 0;
|
||||||
|
}
|
||||||
|
|
||||||
vfree(md->targets);
|
vfree(md->targets);
|
||||||
vfree(md->contexts);
|
vfree(md->contexts);
|
||||||
|
|
||||||
|
md->targets = 0;
|
||||||
|
md->contexts = 0;
|
||||||
|
|
||||||
md->num_targets = 0;
|
md->num_targets = 0;
|
||||||
md->num_allocated = 0;
|
md->num_allocated = 0;
|
||||||
}
|
}
|
||||||
@ -88,7 +93,7 @@ int dm_start_table(struct mapped_device *md)
|
|||||||
set_bit(DM_LOADING, &md->state);
|
set_bit(DM_LOADING, &md->state);
|
||||||
|
|
||||||
dm_free_table(md);
|
dm_free_table(md);
|
||||||
if ((r = _alloc_targets(md, 2))) /* FIXME: increase once debugged 256 ? */
|
if ((r = alloc_targets(md, 2))) /* FIXME: increase once debugged 256 ? */
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
@ -98,7 +103,7 @@ int dm_add_entry(struct mapped_device *md, offset_t high,
|
|||||||
dm_map_fn target, void *context)
|
dm_map_fn target, void *context)
|
||||||
{
|
{
|
||||||
if (md->num_targets >= md->num_targets &&
|
if (md->num_targets >= md->num_targets &&
|
||||||
!_alloc_targets(md, md->num_allocated * 2))
|
alloc_targets(md, md->num_allocated * 2))
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
md->highs[md->num_targets] = high;
|
md->highs[md->num_targets] = high;
|
||||||
@ -116,14 +121,14 @@ int dm_complete_table(struct mapped_device *md)
|
|||||||
clear_bit(DM_LOADING, &md->state);
|
clear_bit(DM_LOADING, &md->state);
|
||||||
|
|
||||||
/* how many indexes will the btree have ? */
|
/* how many indexes will the btree have ? */
|
||||||
for (n = _div_up(md->num_targets, KEYS_PER_NODE), i = 1; n != 1; i++)
|
for (n = div_up(md->num_targets, KEYS_PER_NODE), i = 1; n != 1; i++)
|
||||||
n = _div_up(n, KEYS_PER_NODE + 1);
|
n = div_up(n, KEYS_PER_NODE + 1);
|
||||||
|
|
||||||
md->depth = i;
|
md->depth = i;
|
||||||
md->counts[md->depth - 1] = _div_up(md->num_targets, KEYS_PER_NODE);
|
md->counts[md->depth - 1] = div_up(md->num_targets, KEYS_PER_NODE);
|
||||||
|
|
||||||
while (--i)
|
while (--i)
|
||||||
md->counts[i - 1] = _div_up(md->counts[i], KEYS_PER_NODE + 1);
|
md->counts[i - 1] = div_up(md->counts[i], KEYS_PER_NODE + 1);
|
||||||
|
|
||||||
for (i = 0; i < md->depth; i++) {
|
for (i = 0; i < md->depth; i++) {
|
||||||
size_t s = NODE_SIZE * md->counts[i];
|
size_t s = NODE_SIZE * md->counts[i];
|
||||||
@ -136,13 +141,13 @@ int dm_complete_table(struct mapped_device *md)
|
|||||||
|
|
||||||
/* fill in higher levels */
|
/* fill in higher levels */
|
||||||
for (i = md->depth - 1; i; i--)
|
for (i = md->depth - 1; i; i--)
|
||||||
_setup_btree_index(i - 1, md);
|
setup_btree_index(i - 1, md);
|
||||||
|
|
||||||
set_bit(DM_LOADED, &md->state);
|
set_bit(DM_LOADED, &md->state);
|
||||||
return 1;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int _alloc_targets(struct mapped_device *md, int num)
|
static int alloc_targets(struct mapped_device *md, int num)
|
||||||
{
|
{
|
||||||
offset_t *n_highs;
|
offset_t *n_highs;
|
||||||
dm_map_fn *n_targets;
|
dm_map_fn *n_targets;
|
||||||
|
@ -25,6 +25,8 @@
|
|||||||
|
|
||||||
#include "dm.h"
|
#include "dm.h"
|
||||||
|
|
||||||
|
#include <linux/ctype.h>
|
||||||
|
|
||||||
static struct target *_targets;
|
static struct target *_targets;
|
||||||
static spinlock_t _lock = SPIN_LOCK_UNLOCKED;
|
static spinlock_t _lock = SPIN_LOCK_UNLOCKED;
|
||||||
|
|
||||||
@ -84,20 +86,20 @@ int register_map_target(const char *name, dm_ctr_fn ctr,
|
|||||||
*
|
*
|
||||||
* 'linear' target maps a linear range of a device
|
* 'linear' target maps a linear range of a device
|
||||||
*/
|
*/
|
||||||
int _io_err_ctr(offset_t b, offset_t e, struct mapped_device *md,
|
int io_err_ctr(offset_t b, offset_t e, struct mapped_device *md,
|
||||||
const char *context, void **result)
|
const char *cb, const char *ce, void **result)
|
||||||
{
|
{
|
||||||
/* this takes no arguments */
|
/* this takes no arguments */
|
||||||
*result = 0;
|
*result = 0;
|
||||||
return 1;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void _io_err_dtr(void *c)
|
void io_err_dtr(void *c)
|
||||||
{
|
{
|
||||||
/* empty */
|
/* empty */
|
||||||
}
|
}
|
||||||
|
|
||||||
int _io_err_map(struct buffer_head *bh, void *context)
|
int io_err_map(struct buffer_head *bh, void *context)
|
||||||
{
|
{
|
||||||
buffer_IO_error(bh);
|
buffer_IO_error(bh);
|
||||||
return 0;
|
return 0;
|
||||||
@ -109,38 +111,47 @@ struct linear_c {
|
|||||||
int offset; /* FIXME: we need a signed offset type */
|
int offset; /* FIXME: we need a signed offset type */
|
||||||
};
|
};
|
||||||
|
|
||||||
int _linear_ctr(offset_t b, offset_t e, struct mapped_device *md,
|
static int get_number(const char **b, const char *e, unsigned int *n)
|
||||||
const char *context, void **result)
|
{
|
||||||
|
char *ptr;
|
||||||
|
*b = eat_space(*b, e);
|
||||||
|
if (*b >= e)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
*n = simple_strtoul(*b, &ptr, 10);
|
||||||
|
if (ptr == *b)
|
||||||
|
return -EINVAL;
|
||||||
|
*b = ptr;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int linear_ctr(offset_t low, offset_t high, struct mapped_device *md,
|
||||||
|
const char *cb, const char *ce, void **result)
|
||||||
{
|
{
|
||||||
/* context string should be of the form:
|
/* context string should be of the form:
|
||||||
* <major> <minor> <offset>
|
* <major> <minor> <offset>
|
||||||
*/
|
*/
|
||||||
char *ptr = (char *) context;
|
|
||||||
struct linear_c *lc;
|
struct linear_c *lc;
|
||||||
int major, minor, start;
|
unsigned int major, minor, start;
|
||||||
|
int r;
|
||||||
|
|
||||||
/* FIXME: somewhat verbose */
|
if ((r = get_number(&cb, ce, &major)))
|
||||||
major = simple_strtol(context, &ptr, 10);
|
return r;
|
||||||
if (ptr == context)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
context = ptr;
|
if ((r = get_number(&cb, ce, &minor)))
|
||||||
minor = simple_strtol(context, &ptr, 10);
|
return r;
|
||||||
if (ptr == context)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
context = ptr;
|
if ((r = get_number(&cb, ce, &start)))
|
||||||
start = simple_strtoul(context, &ptr, 10);
|
return r;
|
||||||
if (ptr == context)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
if (!(lc = kmalloc(sizeof(lc), GFP_KERNEL))) {
|
if (!(lc = kmalloc(sizeof(lc), GFP_KERNEL))) {
|
||||||
WARN("couldn't allocate memory for linear context\n");
|
WARN("couldn't allocate memory for linear context\n");
|
||||||
return 0;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
|
||||||
lc->dev = MKDEV(major, minor);
|
lc->dev = MKDEV((int) major, (int) minor);
|
||||||
lc->offset = start - b;
|
lc->offset = (int) start - (int) low;
|
||||||
|
|
||||||
if (!dm_add_device(md, lc->dev)) {
|
if (!dm_add_device(md, lc->dev)) {
|
||||||
kfree(lc);
|
kfree(lc);
|
||||||
@ -148,15 +159,15 @@ int _linear_ctr(offset_t b, offset_t e, struct mapped_device *md,
|
|||||||
}
|
}
|
||||||
|
|
||||||
*result = lc;
|
*result = lc;
|
||||||
return 1;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void _linear_dtr(void *c)
|
void linear_dtr(void *c)
|
||||||
{
|
{
|
||||||
kfree(c);
|
kfree(c);
|
||||||
}
|
}
|
||||||
|
|
||||||
int _linear_map(struct buffer_head *bh, void *context)
|
int linear_map(struct buffer_head *bh, void *context)
|
||||||
{
|
{
|
||||||
struct linear_c *lc = (struct linear_c *) context;
|
struct linear_c *lc = (struct linear_c *) context;
|
||||||
|
|
||||||
@ -173,8 +184,8 @@ int dm_std_targets(void)
|
|||||||
if ((ret = register_map_target(n, \
|
if ((ret = register_map_target(n, \
|
||||||
fn ## _ctr, fn ## _dtr, fn ## _map) < 0)) return ret
|
fn ## _ctr, fn ## _dtr, fn ## _map) < 0)) return ret
|
||||||
|
|
||||||
xx("io-err", _io_err);
|
xx("io-err", io_err);
|
||||||
xx("linear", _linear);
|
xx("linear", linear);
|
||||||
#undef xx
|
#undef xx
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -36,8 +36,10 @@
|
|||||||
|
|
||||||
/* defines for blk.h */
|
/* defines for blk.h */
|
||||||
#define MAJOR_NR DM_BLK_MAJOR
|
#define MAJOR_NR DM_BLK_MAJOR
|
||||||
#define DEVICE_OFF(device)
|
#define DEVICE_NR(device) MINOR(device) /* has no partition bits */
|
||||||
#define LOCAL_END_REQUEST
|
#define DEVICE_NAME "device-mapper" /* name for messaging */
|
||||||
|
#define DEVICE_NO_RANDOM /* no entropy to contribute */
|
||||||
|
#define DEVICE_OFF(d) /* do-nothing */
|
||||||
|
|
||||||
#include <linux/blk.h>
|
#include <linux/blk.h>
|
||||||
|
|
||||||
@ -117,7 +119,7 @@
|
|||||||
#define DEFAULT_READ_AHEAD 64
|
#define DEFAULT_READ_AHEAD 64
|
||||||
|
|
||||||
const char *_name = "device-mapper";
|
const char *_name = "device-mapper";
|
||||||
int _version[3] = {1, 0, 0};
|
int _version[3] = {0, 1, 0};
|
||||||
|
|
||||||
#define rl down_read(&_dev_lock)
|
#define rl down_read(&_dev_lock)
|
||||||
#define ru up_read(&_dev_lock)
|
#define ru up_read(&_dev_lock)
|
||||||
@ -132,23 +134,23 @@ static int _block_size[MAX_DEVICES];
|
|||||||
static int _blksize_size[MAX_DEVICES];
|
static int _blksize_size[MAX_DEVICES];
|
||||||
static int _hardsect_size[MAX_DEVICES];
|
static int _hardsect_size[MAX_DEVICES];
|
||||||
|
|
||||||
static int _blk_open(struct inode *inode, struct file *file);
|
static int blk_open(struct inode *inode, struct file *file);
|
||||||
static int _blk_close(struct inode *inode, struct file *file);
|
static int blk_close(struct inode *inode, struct file *file);
|
||||||
static int _blk_ioctl(struct inode *inode, struct file *file,
|
static int blk_ioctl(struct inode *inode, struct file *file,
|
||||||
uint command, ulong a);
|
uint command, ulong a);
|
||||||
|
|
||||||
struct block_device_operations dm_blk_dops = {
|
struct block_device_operations dm_blk_dops = {
|
||||||
open: _blk_open,
|
open: blk_open,
|
||||||
release: _blk_close,
|
release: blk_close,
|
||||||
ioctl: _blk_ioctl
|
ioctl: blk_ioctl
|
||||||
};
|
};
|
||||||
|
|
||||||
static int _request_fn(request_queue_t *q, int rw, struct buffer_head *bh);
|
static int request(request_queue_t *q, int rw, struct buffer_head *bh);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* setup and teardown the driver
|
* setup and teardown the driver
|
||||||
*/
|
*/
|
||||||
static int _init(void)
|
static int init(void)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
|
|
||||||
@ -171,14 +173,14 @@ static int _init(void)
|
|||||||
return -EIO;
|
return -EIO;
|
||||||
}
|
}
|
||||||
|
|
||||||
blk_queue_make_request(BLK_DEFAULT_QUEUE(MAJOR_NR), _request_fn);
|
blk_queue_make_request(BLK_DEFAULT_QUEUE(MAJOR_NR), request);
|
||||||
|
|
||||||
printk(KERN_INFO "%s(%d, %d, %d) successfully initialised\n", _name,
|
printk(KERN_INFO "%s %d.%d.%d initialised\n", _name,
|
||||||
_version[0], _version[1], _version[2]);
|
_version[0], _version[1], _version[2]);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void _fin(void)
|
static void fin(void)
|
||||||
{
|
{
|
||||||
dm_fin_fs();
|
dm_fin_fs();
|
||||||
|
|
||||||
@ -190,14 +192,14 @@ static void _fin(void)
|
|||||||
blksize_size[MAJOR_NR] = 0;
|
blksize_size[MAJOR_NR] = 0;
|
||||||
hardsect_size[MAJOR_NR] = 0;
|
hardsect_size[MAJOR_NR] = 0;
|
||||||
|
|
||||||
printk(KERN_INFO "%s(%d, %d, %d) successfully finalised\n", _name,
|
printk(KERN_INFO "%s %d.%d.%d finalised\n", _name,
|
||||||
_version[0], _version[1], _version[2]);
|
_version[0], _version[1], _version[2]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* block device functions
|
* block device functions
|
||||||
*/
|
*/
|
||||||
static int _blk_open(struct inode *inode, struct file *file)
|
static int blk_open(struct inode *inode, struct file *file)
|
||||||
{
|
{
|
||||||
int minor = MINOR(inode->i_rdev);
|
int minor = MINOR(inode->i_rdev);
|
||||||
struct mapped_device *md;
|
struct mapped_device *md;
|
||||||
@ -220,7 +222,7 @@ static int _blk_open(struct inode *inode, struct file *file)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int _blk_close(struct inode *inode, struct file *file)
|
static int blk_close(struct inode *inode, struct file *file)
|
||||||
{
|
{
|
||||||
int minor = MINOR(inode->i_rdev);
|
int minor = MINOR(inode->i_rdev);
|
||||||
struct mapped_device *md;
|
struct mapped_device *md;
|
||||||
@ -243,7 +245,7 @@ static int _blk_close(struct inode *inode, struct file *file)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int _blk_ioctl(struct inode *inode, struct file *file,
|
static int blk_ioctl(struct inode *inode, struct file *file,
|
||||||
uint command, ulong a)
|
uint command, ulong a)
|
||||||
{
|
{
|
||||||
/* FIXME: check in the latest Rubini that all expected ioctl's
|
/* FIXME: check in the latest Rubini that all expected ioctl's
|
||||||
@ -290,7 +292,7 @@ static int _blk_ioctl(struct inode *inode, struct file *file,
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int _request_fn(request_queue_t *q, int rw, struct buffer_head *bh)
|
static int request(request_queue_t *q, int rw, struct buffer_head *bh)
|
||||||
{
|
{
|
||||||
struct mapped_device *md;
|
struct mapped_device *md;
|
||||||
offset_t *node;
|
offset_t *node;
|
||||||
@ -358,7 +360,7 @@ static inline int __any_old_dev(void)
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static struct mapped_device *_alloc_dev(int minor)
|
static struct mapped_device *alloc_dev(int minor)
|
||||||
{
|
{
|
||||||
struct mapped_device *md = kmalloc(sizeof(*md), GFP_KERNEL);
|
struct mapped_device *md = kmalloc(sizeof(*md), GFP_KERNEL);
|
||||||
memset(md, 0, sizeof(*md));
|
memset(md, 0, sizeof(*md));
|
||||||
@ -393,7 +395,7 @@ static inline struct mapped_device *__find_name(const char *name)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int _open_dev(struct dev_list *d)
|
static int open_dev(struct dev_list *d)
|
||||||
{
|
{
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
@ -408,13 +410,27 @@ static int _open_dev(struct dev_list *d)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void _close_dev(struct dev_list *d)
|
static void close_dev(struct dev_list *d)
|
||||||
{
|
{
|
||||||
blkdev_put(d->bd, BDEV_FILE);
|
blkdev_put(d->bd, BDEV_FILE);
|
||||||
bdput(d->bd);
|
bdput(d->bd);
|
||||||
d->bd = 0;
|
d->bd = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int __find_hardsect_size(struct mapped_device *md)
|
||||||
|
{
|
||||||
|
int r = INT_MAX, s;
|
||||||
|
struct dev_list *dl;
|
||||||
|
|
||||||
|
for (dl = md->devices; dl; dl = dl->next) {
|
||||||
|
s = get_hardsect_size(dl->dev);
|
||||||
|
if (s < r)
|
||||||
|
r = s;
|
||||||
|
}
|
||||||
|
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
struct mapped_device *dm_find_name(const char *name)
|
struct mapped_device *dm_find_name(const char *name)
|
||||||
{
|
{
|
||||||
struct mapped_device *md;
|
struct mapped_device *md;
|
||||||
@ -445,7 +461,7 @@ int dm_create(const char *name, int minor)
|
|||||||
if (minor >= MAX_DEVICES)
|
if (minor >= MAX_DEVICES)
|
||||||
return -ENXIO;
|
return -ENXIO;
|
||||||
|
|
||||||
if (!(md = _alloc_dev(minor)))
|
if (!(md = alloc_dev(minor)))
|
||||||
return -ENOMEM;
|
return -ENOMEM;
|
||||||
|
|
||||||
wl;
|
wl;
|
||||||
@ -485,7 +501,7 @@ int dm_remove(const char *name)
|
|||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
//dm_free_table(md);
|
dm_free_table(md);
|
||||||
for (d = md->devices; d; d = n) {
|
for (d = md->devices; d; d = n) {
|
||||||
n = d->next;
|
n = d->next;
|
||||||
kfree(d);
|
kfree(d);
|
||||||
@ -504,29 +520,48 @@ int dm_add_device(struct mapped_device *md, kdev_t dev)
|
|||||||
struct dev_list *d = kmalloc(sizeof(*d), GFP_KERNEL);
|
struct dev_list *d = kmalloc(sizeof(*d), GFP_KERNEL);
|
||||||
|
|
||||||
if (!d)
|
if (!d)
|
||||||
return 0;
|
return -EINVAL;
|
||||||
|
|
||||||
d->dev = dev;
|
d->dev = dev;
|
||||||
d->next = md->devices;
|
d->next = md->devices;
|
||||||
md->devices = d;
|
md->devices = d;
|
||||||
|
|
||||||
return 1;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int dm_activate(struct mapped_device *md)
|
int dm_activate(struct mapped_device *md)
|
||||||
{
|
{
|
||||||
int ret;
|
int ret, minor;
|
||||||
struct dev_list *d, *od;
|
struct dev_list *d, *od;
|
||||||
|
|
||||||
if (is_active(md))
|
wl;
|
||||||
return 1;
|
|
||||||
|
if (is_active(md)) {
|
||||||
|
wu;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!md->num_targets) {
|
||||||
|
wu;
|
||||||
|
return -ENXIO;
|
||||||
|
}
|
||||||
|
|
||||||
rl;
|
|
||||||
/* open all the devices */
|
/* open all the devices */
|
||||||
for (d = md->devices; d; d = d->next)
|
for (d = md->devices; d; d = d->next)
|
||||||
if ((ret = _open_dev(d)))
|
if ((ret = open_dev(d)))
|
||||||
goto bad;
|
goto bad;
|
||||||
ru;
|
|
||||||
|
minor = MINOR(md->dev);
|
||||||
|
|
||||||
|
_block_size[minor] = md->highs[md->num_targets - 1] + 1;
|
||||||
|
_blksize_size[minor] = BLOCK_SIZE; /* FIXME: this depends on
|
||||||
|
the mapping table */
|
||||||
|
_hardsect_size[minor] = __find_hardsect_size(md);
|
||||||
|
|
||||||
|
register_disk(NULL, md->dev, 1, &dm_blk_dops, _block_size[minor]);
|
||||||
|
|
||||||
|
set_bit(DM_ACTIVE, &md->state);
|
||||||
|
wu;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
@ -534,7 +569,7 @@ int dm_activate(struct mapped_device *md)
|
|||||||
|
|
||||||
od = d;
|
od = d;
|
||||||
for (d = md->devices; d != od; d = d->next)
|
for (d = md->devices; d != od; d = d->next)
|
||||||
_close_dev(d);
|
close_dev(d);
|
||||||
ru;
|
ru;
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
@ -548,17 +583,17 @@ void dm_suspend(struct mapped_device *md)
|
|||||||
|
|
||||||
/* close all the devices */
|
/* close all the devices */
|
||||||
for (d = md->devices; d; d = d->next)
|
for (d = md->devices; d; d = d->next)
|
||||||
_close_dev(d);
|
close_dev(d);
|
||||||
|
|
||||||
set_active(md, 0);
|
clear_bit(DM_ACTIVE, &md->state);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* module hooks
|
* module hooks
|
||||||
*/
|
*/
|
||||||
module_init(_init);
|
module_init(init);
|
||||||
module_exit(_fin);
|
module_exit(fin);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Local variables:
|
* Local variables:
|
||||||
|
@ -40,6 +40,7 @@
|
|||||||
#include <linux/compatmac.h>
|
#include <linux/compatmac.h>
|
||||||
#include <linux/cache.h>
|
#include <linux/cache.h>
|
||||||
#include <linux/devfs_fs_kernel.h>
|
#include <linux/devfs_fs_kernel.h>
|
||||||
|
#include <linux/ctype.h>
|
||||||
#include <linux/device-mapper.h>
|
#include <linux/device-mapper.h>
|
||||||
|
|
||||||
#define MAX_DEPTH 16
|
#define MAX_DEPTH 16
|
||||||
@ -134,12 +135,12 @@ static inline int is_active(struct mapped_device *md)
|
|||||||
return test_bit(DM_ACTIVE, &md->state);
|
return test_bit(DM_ACTIVE, &md->state);
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline void set_active(struct mapped_device *md, int set)
|
static inline const char *eat_space(const char *b, const char *e)
|
||||||
{
|
{
|
||||||
if (set)
|
while(b != e && isspace((int) *b))
|
||||||
set_bit(DM_ACTIVE, &md->state);
|
b++;
|
||||||
else
|
|
||||||
clear_bit(DM_ACTIVE,& md->state);
|
return b;
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
Loading…
x
Reference in New Issue
Block a user