2001-08-20 12:03:02 +04:00
/*
* 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"
2001-08-21 19:24:02 +04:00
# include <linux/proc_fs.h>
# include <linux/ctype.h>
2001-08-20 12:03:02 +04:00
/*
* / 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 " ;
2001-08-20 12:03:02 +04:00
static struct proc_dir_entry * _proc_dir ;
2001-08-20 17:45:43 +04:00
static struct proc_dir_entry * _control ;
2001-08-20 12:03:02 +04:00
static devfs_handle_t _dev_dir ;
2001-08-20 17:45:43 +04:00
2001-08-23 16:35:02 +04:00
static int line_splitter ( struct file * file , const char * buffer ,
unsigned long count , void * data ) ;
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 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 ) ;
2001-08-20 17:45:43 +04:00
2001-08-21 19:24:02 +04:00
typedef int ( * process_fn ) ( const char * b , const char * e , int minor ) ;
2001-08-20 17:45:43 +04:00
struct pf_data {
2001-08-21 19:24:02 +04:00
process_fn fn ;
2001-08-20 17:45:43 +04:00
int minor ;
} ;
2001-08-20 12:03:02 +04:00
2001-08-31 16:49:31 +04:00
int dm_fs_init ( void )
2001-08-20 12:03:02 +04:00
{
2001-08-20 17:45:43 +04:00
struct pf_data * pfd = kmalloc ( sizeof ( * pfd ) , GFP_KERNEL ) ;
2001-08-20 12:03:02 +04:00
2001-08-20 17:45:43 +04:00
if ( ! pfd )
2001-08-20 12:03:02 +04:00
return 0 ;
2001-08-20 17:45:43 +04:00
_dev_dir = devfs_mk_dir ( 0 , _fs_dir , NULL ) ;
2001-08-20 12:03:02 +04:00
2001-08-20 17:45:43 +04:00
if ( ! ( _proc_dir = create_proc_entry ( _fs_dir , S_IFDIR , & proc_root ) ) )
goto fail ;
2001-08-20 12:03:02 +04:00
2001-08-23 00:10:06 +04:00
if ( ! ( _control = create_proc_entry ( _control_name , S_IWUSR , _proc_dir ) ) )
2001-08-20 17:45:43 +04:00
goto fail ;
2001-08-20 12:03:02 +04:00
2001-08-23 16:35:02 +04:00
_control - > write_proc = line_splitter ;
2001-08-20 12:03:02 +04:00
2001-08-23 16:35:02 +04:00
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 12:03:02 +04:00
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-20 12:03:02 +04:00
}
2001-08-31 16:49:31 +04:00
void dm_fs_exit ( void )
2001-08-20 12:03:02 +04:00
{
2001-08-20 17:45:43 +04:00
if ( _control ) {
remove_proc_entry ( _control_name , _proc_dir ) ;
_control = 0 ;
}
2001-08-20 12:03:02 +04:00
2001-08-20 17:45:43 +04:00
if ( _proc_dir ) {
remove_proc_entry ( _fs_dir , & proc_root ) ;
_proc_dir = 0 ;
}
2001-08-20 12:03:02 +04:00
2001-08-20 17:45:43 +04:00
if ( _dev_dir )
devfs_unregister ( _dev_dir ) ;
}
2001-08-20 12:03:02 +04:00
2001-08-22 19:01:09 +04:00
int dm_fs_add ( struct mapped_device * md )
{
struct pf_data * pfd = kmalloc ( sizeof ( * pfd ) , GFP_KERNEL ) ;
if ( ! pfd )
return - ENOMEM ;
2001-08-23 16:35:02 +04:00
pfd - > fn = process_table ;
2001-08-22 19:01:09 +04:00
pfd - > minor = MINOR ( md - > dev ) ;
2001-08-28 18:56:47 +04:00
if ( ! ( md - > pde = create_proc_entry ( md - > name , S_IRUGO | S_IWUSR ,
2001-08-23 00:10:06 +04:00
_proc_dir ) ) ) {
2001-08-22 19:01:09 +04:00
kfree ( pfd ) ;
return - ENOMEM ;
}
2001-08-23 16:35:02 +04:00
md - > pde - > write_proc = line_splitter ;
2001-08-22 19:01:09 +04:00
md - > pde - > data = pfd ;
2001-08-22 19:33:08 +04:00
md - > devfs_entry =
2001-08-23 00:10:06 +04:00
devfs_register ( _dev_dir , md - > name , DEVFS_FL_CURRENT_OWNER ,
2001-08-22 19:33:08 +04:00
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 ;
2001-08-23 16:35:02 +04:00
return - ENOMEM ;
2001-08-22 19:33:08 +04:00
}
2001-08-22 19:01:09 +04:00
return 0 ;
}
int dm_fs_remove ( struct mapped_device * md )
{
if ( md - > pde ) {
kfree ( md - > pde - > data ) ;
remove_proc_entry ( md - > name , _proc_dir ) ;
2001-08-22 19:33:08 +04:00
md - > pde = 0 ;
2001-08-22 19:01:09 +04:00
}
2001-08-22 19:33:08 +04:00
devfs_unregister ( md - > devfs_entry ) ;
md - > devfs_entry = 0 ;
2001-08-22 19:01:09 +04:00
return 0 ;
}
2001-08-23 16:35:02 +04:00
static int process_control ( const char * b , const char * e , int minor )
2001-08-20 12:03:02 +04:00
{
2001-08-20 17:45:43 +04:00
const char * wb , * we ;
2001-08-21 19:24:02 +04:00
char name [ 64 ] ;
2001-08-20 17:45:43 +04:00
int create = 0 ;
2001-08-20 12:03:02 +04:00
2001-08-20 17:45:43 +04:00
/*
* create < name > [ minor ]
* remove < name >
*/
2001-08-23 16:35:02 +04:00
if ( get_word ( b , e , & wb , & we ) )
2001-08-20 17:45:43 +04:00
return - EINVAL ;
b = we ;
2001-08-20 12:03:02 +04:00
2001-08-23 16:35:02 +04:00
if ( ! tok_cmp ( " create " , wb , we ) )
2001-08-20 17:45:43 +04:00
create = 1 ;
2001-08-20 12:03:02 +04:00
2001-08-23 16:35:02 +04:00
else if ( tok_cmp ( " remove " , wb , we ) )
2001-08-20 17:45:43 +04:00
return - EINVAL ;
2001-08-20 12:03:02 +04:00
2001-08-23 16:35:02 +04:00
if ( get_word ( b , e , & wb , & we ) )
2001-08-20 17:45:43 +04:00
return - EINVAL ;
b = we ;
2001-08-20 12:03:02 +04:00
2001-08-23 16:35:02 +04:00
tok_cpy ( name , sizeof ( name ) , wb , we ) ;
2001-08-20 12:03:02 +04:00
2001-08-22 19:01:09 +04:00
if ( ! create )
return dm_remove ( name ) ;
else {
2001-08-23 16:35:02 +04:00
if ( ! get_word ( b , e , & wb , & we ) ) {
2001-08-21 19:24:02 +04:00
minor = simple_strtol ( wb , ( char * * ) & we , 10 ) ;
2001-08-20 12:03:02 +04:00
2001-08-20 17:45:43 +04:00
if ( we = = wb )
return - EINVAL ;
}
2001-08-21 19:24:02 +04:00
return dm_create ( name , minor ) ;
2001-08-20 12:03:02 +04:00
}
2001-08-20 17:45:43 +04:00
return - EINVAL ;
2001-08-20 12:03:02 +04:00
}
2001-08-23 16:35:02 +04:00
static int process_table ( const char * b , const char * e , int minor )
2001-08-20 12:03:02 +04:00
{
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 ) ;
2001-08-21 19:24:02 +04:00
void * context ;
2001-08-23 16:35:02 +04:00
int r ;
2001-08-20 12:03:02 +04:00
if ( ! md )
return - ENXIO ;
2001-08-23 16:35:02 +04:00
if ( get_word ( b , e , & wb , & we ) )
2001-08-20 17:45:43 +04:00
return - EINVAL ;
2001-08-23 16:35:02 +04:00
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 12:03:02 +04:00
2001-08-20 17:45:43 +04:00
/* start loading a table */
2001-08-29 17:58:48 +04:00
dm_table_start ( md ) ;
2001-08-20 12:03:02 +04:00
2001-08-23 16:35:02 +04:00
} else if ( ! tok_cmp ( " end " , b , e ) ) {
2001-08-20 17:45:43 +04:00
/* activate the device ... <evil chuckle> ... */
2001-08-29 17:58:48 +04:00
dm_table_complete ( md ) ;
2001-08-20 17:45:43 +04:00
dm_activate ( md ) ;
2001-08-20 12:03:02 +04:00
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 ;
2001-08-28 18:56:47 +04:00
offset_t start , size , high ;
size_t len ;
2001-08-20 17:45:43 +04:00
2001-08-28 18:56:47 +04:00
if ( get_number ( & b , e , & start ) )
2001-08-23 16:35:02 +04:00
return - EINVAL ;
2001-08-20 17:45:43 +04:00
2001-08-28 18:56:47 +04:00
if ( get_number ( & b , e , & size ) )
2001-08-23 16:35:02 +04:00
return - EINVAL ;
2001-08-20 17:45:43 +04:00
2001-08-23 16:35:02 +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 ) )
2001-08-23 16:35:02 +04:00
return - EINVAL ;
2001-08-20 17:45:43 +04:00
strncpy ( target , wb , len ) ;
target [ len ] = ' \0 ' ;
if ( ! ( t = dm_get_target ( target ) ) )
2001-08-23 16:35:02 +04:00
return - EINVAL ;
2001-08-20 17:45:43 +04:00
2001-08-28 18:56:47 +04:00
/* check there isn't a gap */
if ( ( md - > num_targets & &
start ! = md - > highs [ md - > num_targets - 1 ] + 1 ) | |
( ! md - > num_targets & & start ) ) {
WARN ( " gap in target ranges " ) ;
return - EINVAL ;
}
2001-08-20 12:03:02 +04:00
2001-08-28 18:56:47 +04:00
high = start + ( size - 1 ) ;
if ( ( r = t - > ctr ( start , high , md , we , e , & context ) ) )
2001-08-23 16:35:02 +04:00
return r ;
2001-08-20 17:45:43 +04:00
2001-08-29 17:58:48 +04:00
if ( ( r = dm_table_add_entry ( md , high , t - > map , context ) ) )
2001-08-23 16:35:02 +04:00
return r ;
}
2001-08-20 17:45:43 +04:00
2001-08-23 16:35:02 +04:00
return 0 ;
2001-08-20 17:45:43 +04:00
}
2001-08-20 12:03:02 +04:00
2001-08-23 16:35:02 +04:00
static int get_word ( const char * b , const char * e ,
const char * * wb , const char * * we )
2001-08-20 17:45:43 +04:00
{
2001-08-23 16:35:02 +04:00
b = eat_space ( b , e ) ;
2001-08-20 17:45:43 +04:00
if ( b = = e )
2001-08-23 16:35:02 +04:00
return - EINVAL ;
2001-08-20 17:45:43 +04:00
* wb = b ;
2001-08-22 18:13:26 +04:00
while ( b ! = e & & ! isspace ( ( int ) * b ) )
2001-08-20 17:45:43 +04:00
b + + ;
2001-08-22 17:52:26 +04:00
* we = b ;
2001-08-23 16:35:02 +04:00
return 0 ;
2001-08-20 17:45:43 +04:00
}
2001-08-23 16:35:02 +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
{
2001-08-22 18:13:26 +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 ) {
2001-08-23 16:35:02 +04:00
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 12:03:02 +04:00
2001-08-20 17:45:43 +04:00
lb = b ;
while ( ( b ! = e ) & & * b ! = ' \n ' )
b + + ;
2001-08-22 18:13:26 +04:00
if ( ( r = pfd - > fn ( lb , b , pfd - > minor ) ) )
return r ;
2001-08-20 12:03:02 +04:00
}
2001-08-20 17:45:43 +04:00
return count ;
}
2001-08-23 16:35:02 +04:00
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 12:03:02 +04:00
2001-08-20 17:45:43 +04:00
return - 1 ;
2001-08-20 12:03:02 +04:00
}
2001-08-21 19:24:02 +04:00
2001-08-23 16:35:02 +04:00
static void tok_cpy ( char * dest , size_t max ,
const char * b , const char * e )
2001-08-21 19:24:02 +04:00
{
size_t len = e - b ;
if ( len > - - max )
len = max ;
strncpy ( dest , b , len ) ;
2001-08-22 18:30:30 +04:00
dest [ len ] = ' \0 ' ;
2001-08-21 19:24:02 +04:00
}