2001-08-20 12:05:51 +04:00
/*
* device - mapper . 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 .
*/
/*
* Changelog
*
* 14 / 08 / 2001 - First Version [ Joe Thornber ]
*/
2001-08-21 18:47:42 +04:00
# include "dm.h"
2001-08-20 12:05:51 +04:00
/* defines for blk.h */
# define MAJOR_NR DM_BLK_MAJOR
2001-09-14 17:27:58 +04:00
# define DEVICE_NR(device) MINOR(device) /* has no partition bits */
# define DEVICE_NAME "device-mapper" /* name for messaging */
# define DEVICE_NO_RANDOM /* no entropy to contribute */
# define DEVICE_OFF(d) /* do-nothing */
2001-08-20 12:05:51 +04:00
# include <linux/blk.h>
2001-09-26 21:32:57 +04:00
# include <linux/blkpg.h>
2001-08-20 12:05:51 +04:00
2001-09-27 14:15:02 +04:00
/* we only need this for the lv_bmap struct definition, not happy */
# include <linux/lvm.h>
2001-08-20 12:05:51 +04:00
# define MAX_DEVICES 64
# define DEFAULT_READ_AHEAD 64
const char * _name = " device-mapper " ;
2001-09-26 18:32:07 +04:00
int _version [ 3 ] = { 0 , 1 , 0 } ;
2001-08-20 12:05:51 +04:00
2001-08-28 17:04:44 +04:00
struct io_hook {
2001-09-26 18:32:07 +04:00
struct mapped_device * md ;
2001-09-26 21:32:57 +04:00
struct target * target ;
2001-09-14 20:22:02 +04:00
int rw ;
2001-09-26 18:32:07 +04:00
void ( * end_io ) ( struct buffer_head * bh , int uptodate ) ;
2001-08-28 17:04:44 +04:00
void * context ;
} ;
2001-08-31 14:25:32 +04:00
kmem_cache_t * _io_hook_cache ;
2001-09-26 18:32:07 +04:00
# define rl down_read(&_dev_lock)
# define ru up_read(&_dev_lock)
# define wl down_write(&_dev_lock)
# define wu up_write(&_dev_lock)
2001-08-20 17:45:43 +04:00
struct rw_semaphore _dev_lock ;
2001-08-20 12:05:51 +04:00
static struct mapped_device * _devs [ MAX_DEVICES ] ;
/* block device arrays */
static int _block_size [ MAX_DEVICES ] ;
static int _blksize_size [ MAX_DEVICES ] ;
static int _hardsect_size [ MAX_DEVICES ] ;
2001-09-07 15:34:46 +04:00
const char * _fs_dir = " device-mapper " ;
static devfs_handle_t _dev_dir ;
2001-09-14 17:45:40 +04:00
static int request ( request_queue_t * q , int rw , struct buffer_head * bh ) ;
static int dm_user_bmap ( struct inode * inode , struct lv_bmap * lvb ) ;
2001-08-20 12:05:51 +04:00
/*
* setup and teardown the driver
*/
2001-08-31 19:13:33 +04:00
static int dm_init ( void )
2001-08-20 12:05:51 +04:00
{
2001-08-22 17:41:00 +04:00
int ret ;
2001-08-20 17:45:43 +04:00
init_rwsem ( & _dev_lock ) ;
2001-08-20 12:05:51 +04:00
2001-09-26 18:32:07 +04:00
if ( ! _io_hook_cache )
_io_hook_cache = kmem_cache_create ( " dm io hooks " ,
sizeof ( struct io_hook ) ,
0 , 0 , NULL , NULL ) ;
if ( ! _io_hook_cache )
2001-08-31 14:25:32 +04:00
return - ENOMEM ;
2001-09-26 18:32:07 +04:00
if ( ( ret = dm_fs_init ( ) ) | | ( ret = dm_target_init ( ) ) )
2001-08-22 17:41:00 +04:00
return ret ;
2001-08-20 12:05:51 +04:00
/* set up the arrays */
read_ahead [ MAJOR_NR ] = DEFAULT_READ_AHEAD ;
blk_size [ MAJOR_NR ] = _block_size ;
blksize_size [ MAJOR_NR ] = _blksize_size ;
hardsect_size [ MAJOR_NR ] = _hardsect_size ;
2001-08-22 19:33:08 +04:00
if ( devfs_register_blkdev ( MAJOR_NR , _name , & dm_blk_dops ) < 0 ) {
2001-08-20 12:05:51 +04:00
printk ( KERN_ERR " %s -- register_blkdev failed \n " , _name ) ;
return - EIO ;
}
2001-08-23 16:35:02 +04:00
blk_queue_make_request ( BLK_DEFAULT_QUEUE ( MAJOR_NR ) , request ) ;
2001-08-20 12:05:51 +04:00
2001-09-07 15:34:46 +04:00
_dev_dir = devfs_mk_dir ( 0 , _fs_dir , NULL ) ;
2001-08-23 16:35:02 +04:00
printk ( KERN_INFO " %s %d.%d.%d initialised \n " , _name ,
2001-08-20 12:05:51 +04:00
_version [ 0 ] , _version [ 1 ] , _version [ 2 ] ) ;
return 0 ;
}
2001-08-31 19:13:33 +04:00
static void dm_exit ( void )
2001-08-20 12:05:51 +04:00
{
2001-09-14 17:27:58 +04:00
if ( kmem_cache_destroy ( _io_hook_cache ) )
2001-08-31 14:25:32 +04:00
WARN ( " it looks like there are still some io_hooks allocated " ) ;
2001-09-26 18:32:07 +04:00
_io_hook_cache = 0 ;
2001-08-31 14:25:32 +04:00
2001-09-26 18:32:07 +04:00
dm_fs_exit ( ) ;
2001-08-22 17:41:00 +04:00
if ( devfs_unregister_blkdev ( MAJOR_NR , _name ) < 0 )
2001-08-20 12:05:51 +04:00
printk ( KERN_ERR " %s -- unregister_blkdev failed \n " , _name ) ;
read_ahead [ MAJOR_NR ] = 0 ;
blk_size [ MAJOR_NR ] = 0 ;
blksize_size [ MAJOR_NR ] = 0 ;
hardsect_size [ MAJOR_NR ] = 0 ;
2001-08-29 17:58:48 +04:00
printk ( KERN_INFO " %s %d.%d.%d cleaned up \n " , _name ,
2001-08-20 12:05:51 +04:00
_version [ 0 ] , _version [ 1 ] , _version [ 2 ] ) ;
}
/*
* block device functions
*/
2001-09-13 18:01:13 +04:00
static int dm_blk_open ( struct inode * inode , struct file * file )
2001-08-20 12:05:51 +04:00
{
int minor = MINOR ( inode - > i_rdev ) ;
2001-08-21 18:28:00 +04:00
struct mapped_device * md ;
2001-08-20 12:05:51 +04:00
if ( minor > = MAX_DEVICES )
return - ENXIO ;
2001-09-26 18:32:07 +04:00
wl ;
2001-08-21 18:28:00 +04:00
md = _devs [ minor ] ;
2001-08-20 12:05:51 +04:00
2001-08-21 18:28:00 +04:00
if ( ! md | | ! is_active ( md ) ) {
2001-09-26 18:32:07 +04:00
wu ;
2001-08-20 12:05:51 +04:00
return - ENXIO ;
}
2001-08-21 18:28:00 +04:00
md - > use_count + + ;
2001-09-26 18:32:07 +04:00
wu ;
2001-08-20 12:05:51 +04:00
2001-08-21 18:28:00 +04:00
MOD_INC_USE_COUNT ;
2001-08-20 12:05:51 +04:00
return 0 ;
}
2001-09-13 18:01:13 +04:00
static int dm_blk_close ( struct inode * inode , struct file * file )
2001-08-20 12:05:51 +04:00
{
int minor = MINOR ( inode - > i_rdev ) ;
2001-08-21 18:28:00 +04:00
struct mapped_device * md ;
2001-08-20 12:05:51 +04:00
if ( minor > = MAX_DEVICES )
return - ENXIO ;
2001-09-26 18:32:07 +04:00
wl ;
2001-08-21 18:28:00 +04:00
md = _devs [ minor ] ;
2001-08-23 20:45:43 +04:00
if ( ! md | | md - > use_count < 1 ) {
2001-08-20 12:05:51 +04:00
WARN ( " reference count in mapped_device incorrect " ) ;
2001-09-26 18:32:07 +04:00
wu ;
2001-08-20 12:05:51 +04:00
return - ENXIO ;
}
2001-08-21 18:28:00 +04:00
md - > use_count - - ;
2001-09-26 18:32:07 +04:00
wu ;
2001-08-20 12:05:51 +04:00
2001-08-20 19:59:22 +04:00
MOD_DEC_USE_COUNT ;
2001-08-20 12:05:51 +04:00
return 0 ;
}
2001-09-19 21:46:27 +04:00
/* In 512-byte units */
# define VOLUME_SIZE(minor) (_block_size[(minor)] >> 1)
2001-09-13 18:01:13 +04:00
static int dm_blk_ioctl ( struct inode * inode , struct file * file ,
2001-09-14 17:27:58 +04:00
uint command , ulong a )
2001-08-20 12:05:51 +04:00
{
int minor = MINOR ( inode - > i_rdev ) ;
long size ;
2001-09-13 18:01:13 +04:00
if ( minor > = MAX_DEVICES )
return - ENXIO ;
2001-08-20 12:05:51 +04:00
switch ( command ) {
2001-09-13 18:01:13 +04:00
case BLKSSZGET :
case BLKROGET :
case BLKROSET :
#if 0
case BLKELVSET :
case BLKELVGET :
# endif
return blk_ioctl ( inode - > i_dev , command , a ) ;
break ;
2001-08-20 12:05:51 +04:00
case BLKGETSIZE :
2001-09-13 18:01:13 +04:00
size = VOLUME_SIZE ( minor ) ;
2001-09-14 17:27:58 +04:00
if ( copy_to_user ( ( void * ) a , & size , sizeof ( long ) ) )
2001-08-20 12:05:51 +04:00
return - EFAULT ;
break ;
case BLKFLSBUF :
if ( ! capable ( CAP_SYS_ADMIN ) )
return - EACCES ;
fsync_dev ( inode - > i_rdev ) ;
invalidate_buffers ( inode - > i_rdev ) ;
return 0 ;
case BLKRAGET :
2001-09-14 17:27:58 +04:00
if ( copy_to_user
( ( void * ) a , & read_ahead [ MAJOR ( inode - > i_rdev ) ] ,
sizeof ( long ) ) )
2001-08-20 12:05:51 +04:00
return - EFAULT ;
return 0 ;
case BLKRASET :
if ( ! capable ( CAP_SYS_ADMIN ) )
return - EACCES ;
read_ahead [ MAJOR ( inode - > i_rdev ) ] = a ;
return 0 ;
case BLKRRPART :
return - EINVAL ;
2001-09-14 17:45:40 +04:00
case LV_BMAP :
2001-09-27 14:15:02 +04:00
return dm_user_bmap ( inode , ( struct lv_bmap * ) a ) ;
2001-08-20 12:05:51 +04:00
default :
2001-09-26 18:32:07 +04:00
WARN ( " unknown block ioctl %d " , command ) ;
2001-08-20 12:05:51 +04:00
return - EINVAL ;
}
return 0 ;
}
2001-08-31 20:36:56 +04:00
static inline struct io_hook * alloc_io_hook ( void )
2001-08-28 17:04:44 +04:00
{
2001-08-31 14:25:32 +04:00
return kmem_cache_alloc ( _io_hook_cache , GFP_NOIO ) ;
2001-08-28 17:04:44 +04:00
}
2001-08-31 20:36:56 +04:00
static inline void free_io_hook ( struct io_hook * ih )
2001-08-28 17:04:44 +04:00
{
2001-08-31 14:25:32 +04:00
kmem_cache_free ( _io_hook_cache , ih ) ;
2001-08-28 17:04:44 +04:00
}
2001-08-31 16:49:31 +04:00
/*
* FIXME : need to decide if deferred_io ' s need
* their own slab , I say no for now since they are
* only used when the device is suspended .
*/
2001-08-31 20:36:56 +04:00
static inline struct deferred_io * alloc_deferred ( void )
2001-08-28 18:05:22 +04:00
{
2001-09-26 18:32:07 +04:00
return kmalloc ( sizeof ( struct deferred_io ) , GFP_NOIO ) ;
2001-08-28 18:05:22 +04:00
}
2001-08-31 20:36:56 +04:00
static inline void free_deferred ( struct deferred_io * di )
2001-08-28 18:05:22 +04:00
{
kfree ( di ) ;
}
2001-09-26 18:32:07 +04:00
/*
* call a targets optional error function if
* an io failed .
*/
static inline int call_err_fn ( struct io_hook * ih , struct buffer_head * bh )
{
dm_err_fn err = ih - > target - > type - > err ;
if ( err )
return err ( bh , ih - > rw , ih - > target - > private ) ;
return 0 ;
}
2001-08-31 19:13:33 +04:00
/*
* bh - > b_end_io routine that decrements the
* pending count and then calls the original
* bh - > b_end_io fn .
*/
2001-08-28 17:04:44 +04:00
static void dec_pending ( struct buffer_head * bh , int uptodate )
{
struct io_hook * ih = bh - > b_private ;
2001-09-26 18:32:07 +04:00
if ( ! uptodate & & call_err_fn ( ih , bh ) )
return ;
2001-09-14 20:22:02 +04:00
2001-09-26 18:32:07 +04:00
if ( atomic_dec_and_test ( & ih - > md - > pending ) )
2001-08-28 17:04:44 +04:00
/* nudge anyone waiting on suspend queue */
2001-09-26 18:32:07 +04:00
wake_up ( & ih - > md - > wait ) ;
2001-08-28 17:04:44 +04:00
bh - > b_end_io = ih - > end_io ;
bh - > b_private = ih - > context ;
free_io_hook ( ih ) ;
bh - > b_end_io ( bh , uptodate ) ;
}
2001-08-31 19:13:33 +04:00
/*
* add the bh to the list of deferred io .
*/
2001-08-28 18:05:22 +04:00
static int queue_io ( struct mapped_device * md , struct buffer_head * bh , int rw )
2001-08-20 12:05:51 +04:00
{
2001-08-28 18:05:22 +04:00
struct deferred_io * di = alloc_deferred ( ) ;
2001-08-20 12:05:51 +04:00
2001-08-28 18:05:22 +04:00
if ( ! di )
return - ENOMEM ;
2001-08-20 12:05:51 +04:00
2001-09-26 18:32:07 +04:00
wl ;
2001-08-28 18:05:22 +04:00
if ( test_bit ( DM_ACTIVE , & md - > state ) ) {
2001-09-26 18:32:07 +04:00
wu ;
2001-08-28 18:05:22 +04:00
return 0 ;
}
2001-08-20 17:45:43 +04:00
2001-08-28 18:05:22 +04:00
di - > bh = bh ;
di - > rw = rw ;
di - > next = md - > deferred ;
md - > deferred = di ;
2001-09-26 18:32:07 +04:00
wu ;
2001-08-31 13:14:55 +04:00
2001-08-28 18:05:22 +04:00
return 1 ;
}
2001-08-20 12:05:51 +04:00
2001-08-31 19:13:33 +04:00
/*
* do the bh mapping for a given leaf
*/
2001-08-31 20:36:56 +04:00
static inline int __map_buffer ( struct mapped_device * md ,
2001-09-14 20:22:02 +04:00
struct buffer_head * bh , int rw , int leaf )
2001-08-28 18:05:22 +04:00
{
2001-09-26 18:32:07 +04:00
int r ;
2001-08-28 18:05:22 +04:00
dm_map_fn fn ;
void * context ;
2001-09-14 20:22:02 +04:00
struct io_hook * ih = NULL ;
2001-08-31 19:13:33 +04:00
struct target * ti = md - > map - > targets + leaf ;
2001-08-28 18:05:22 +04:00
2001-09-02 14:49:20 +04:00
fn = ti - > type - > map ;
2001-08-29 17:58:48 +04:00
context = ti - > private ;
2001-08-20 12:05:51 +04:00
2001-08-28 17:04:44 +04:00
ih = alloc_io_hook ( ) ;
if ( ! ih )
2001-08-28 18:05:22 +04:00
return 0 ;
2001-08-28 17:04:44 +04:00
2001-09-26 18:32:07 +04:00
ih - > md = md ;
2001-09-14 20:22:02 +04:00
ih - > rw = rw ;
ih - > target = ti ;
2001-08-28 17:04:44 +04:00
ih - > end_io = bh - > b_end_io ;
ih - > context = bh - > b_private ;
2001-09-14 20:22:02 +04:00
r = fn ( bh , rw , context ) ;
2001-08-28 17:04:44 +04:00
if ( r > 0 ) {
/* hook the end io request fn */
2001-09-26 18:32:07 +04:00
atomic_inc ( & md - > pending ) ;
2001-08-28 17:04:44 +04:00
bh - > b_end_io = dec_pending ;
bh - > b_private = ih ;
} else if ( r = = 0 )
/* we don't need to hook */
free_io_hook ( ih ) ;
else if ( r < 0 ) {
free_io_hook ( ih ) ;
2001-08-28 18:05:22 +04:00
return 0 ;
}
return 1 ;
}
2001-08-31 19:13:33 +04:00
/*
* search the btree for the correct target .
*/
2001-08-31 20:36:56 +04:00
static inline int __find_node ( struct dm_table * t , struct buffer_head * bh )
2001-08-29 17:58:48 +04:00
{
2001-09-04 14:17:28 +04:00
int l , n = 0 , k = 0 ;
2001-08-29 17:58:48 +04:00
offset_t * node ;
2001-08-31 19:13:33 +04:00
for ( l = 0 ; l < t - > depth ; l + + ) {
2001-09-04 14:17:28 +04:00
n = get_child ( n , k ) ;
node = get_node ( t , l , n ) ;
2001-08-29 17:58:48 +04:00
2001-09-04 14:17:28 +04:00
for ( k = 0 ; k < KEYS_PER_NODE ; k + + )
if ( node [ k ] > = bh - > b_rsector )
2001-08-29 17:58:48 +04:00
break ;
}
2001-09-04 14:17:28 +04:00
return ( KEYS_PER_NODE * n ) + k ;
2001-08-29 17:58:48 +04:00
}
2001-09-14 17:45:40 +04:00
static int request ( request_queue_t * q , int rw , struct buffer_head * bh )
2001-08-28 18:05:22 +04:00
{
struct mapped_device * md ;
2001-08-29 17:58:48 +04:00
int r , minor = MINOR ( bh - > b_rdev ) ;
2001-08-28 18:05:22 +04:00
if ( minor > = MAX_DEVICES )
2001-08-31 13:43:35 +04:00
goto bad_no_lock ;
2001-08-28 18:05:22 +04:00
2001-09-26 18:32:07 +04:00
rl ;
2001-08-28 18:05:22 +04:00
md = _devs [ minor ] ;
2001-08-31 19:13:33 +04:00
if ( ! md | | ! md - > map )
2001-08-28 17:04:44 +04:00
goto bad ;
2001-08-28 18:05:22 +04:00
/* if we're suspended we have to queue this io for later */
if ( ! test_bit ( DM_ACTIVE , & md - > state ) ) {
2001-09-26 18:32:07 +04:00
ru ;
2001-08-28 18:05:22 +04:00
r = queue_io ( md , bh , rw ) ;
2001-08-31 13:43:35 +04:00
if ( r < 0 )
goto bad_no_lock ;
2001-08-28 18:05:22 +04:00
2001-08-31 13:43:35 +04:00
else if ( r > 0 )
2001-09-26 18:32:07 +04:00
return 0 ; /* deferred successfully */
2001-08-28 18:05:22 +04:00
2001-09-26 18:32:07 +04:00
rl ; /* FIXME: there's still a race here */
2001-08-28 17:04:44 +04:00
}
2001-08-20 12:05:51 +04:00
2001-09-14 20:22:02 +04:00
if ( ! __map_buffer ( md , bh , rw , __find_node ( md - > map , bh ) ) )
2001-08-28 18:05:22 +04:00
goto bad ;
2001-09-26 18:32:07 +04:00
ru ;
2001-08-28 18:05:22 +04:00
return 1 ;
2001-08-28 17:04:44 +04:00
2001-09-26 18:32:07 +04:00
bad :
ru ;
2001-08-31 13:43:35 +04:00
2001-09-26 18:32:07 +04:00
bad_no_lock :
2001-08-28 17:04:44 +04:00
buffer_IO_error ( bh ) ;
return 0 ;
2001-08-20 12:05:51 +04:00
}
2001-09-27 14:15:02 +04:00
static int check_dev_size ( int minor , unsigned long block )
{
/* FIXME: check this */
unsigned long max_sector = ( _block_size [ minor ] < < 1 ) + 1 ;
unsigned long sector = ( block + 1 ) * ( _blksize_size [ minor ] > > 9 ) ;
return ( sector > max_sector ) ? 0 : 1 ;
}
/*
* creates a dummy buffer head and maps it ( for lilo ) .
*/
static int do_bmap ( kdev_t dev , unsigned long block ,
kdev_t * r_dev , unsigned long * r_block )
{
struct mapped_device * md ;
struct buffer_head bh ;
int minor = MINOR ( dev ) , r ;
struct target * t ;
rl ;
if ( ( minor > = MAX_DEVICES ) | | ! ( md = _devs [ minor ] ) | |
! test_bit ( DM_ACTIVE , & md - > state ) ) {
r = - ENXIO ;
goto out ;
}
if ( ! check_dev_size ( minor , block ) ) {
r = - EINVAL ;
goto out ;
}
/* setup dummy bh */
memset ( & bh , 0 , sizeof ( bh ) ) ;
bh . b_blocknr = block ;
bh . b_dev = bh . b_rdev = dev ;
bh . b_size = _blksize_size [ minor ] ;
bh . b_rsector = block * ( bh . b_size > > 9 ) ;
/* find target */
t = md - > map - > targets + __find_node ( md - > map , & bh ) ;
/* do the mapping */
r = t - > type - > map ( & bh , READ , t - > private ) ;
* r_dev = bh . b_rdev ;
* r_block = bh . b_rsector / ( bh . b_size > > 9 ) ;
out :
ru ;
return r ;
}
/*
* marshals arguments and results between user and
* kernel space .
*/
static int dm_user_bmap ( struct inode * inode , struct lv_bmap * lvb )
{
unsigned long block , r_block ;
kdev_t r_dev ;
int r ;
if ( get_user ( block , & lvb - > lv_block ) )
return - EFAULT ;
if ( ( r = do_bmap ( inode - > i_rdev , block , & r_dev , & r_block ) ) )
return r ;
if ( put_user ( kdev_t_to_nr ( r_dev ) , & lvb - > lv_dev ) | |
put_user ( r_block , & lvb - > lv_block ) )
return - EFAULT ;
return 0 ;
}
2001-08-31 19:13:33 +04:00
/*
* see if the device with a specific minor # is
* free .
*/
2001-08-20 12:05:51 +04:00
static inline int __specific_dev ( int minor )
{
if ( minor > MAX_DEVICES ) {
WARN ( " request for a mapped_device > than MAX_DEVICES " ) ;
return 0 ;
}
if ( ! _devs [ minor ] )
return minor ;
return - 1 ;
}
2001-08-31 19:13:33 +04:00
/*
* find the first free device .
*/
2001-08-20 12:05:51 +04:00
static inline int __any_old_dev ( void )
{
int i ;
for ( i = 0 ; i < MAX_DEVICES ; i + + )
if ( ! _devs [ i ] )
return i ;
return - 1 ;
}
2001-08-31 19:13:33 +04:00
/*
* allocate and initialise a blank device .
*/
2001-08-23 16:35:02 +04:00
static struct mapped_device * alloc_dev ( int minor )
2001-08-20 12:05:51 +04:00
{
2001-09-26 18:32:07 +04:00
struct mapped_device * md = kmalloc ( sizeof ( * md ) , GFP_KERNEL ) ;
2001-09-14 14:06:22 +04:00
if ( ! md )
return 0 ;
2001-09-14 17:27:58 +04:00
memset ( md , 0 , sizeof ( * md ) ) ;
2001-08-20 12:05:51 +04:00
2001-09-26 18:32:07 +04:00
wl ;
2001-08-20 12:05:51 +04:00
minor = ( minor < 0 ) ? __any_old_dev ( ) : __specific_dev ( minor ) ;
if ( minor < 0 ) {
WARN ( " no free devices available " ) ;
2001-09-26 18:32:07 +04:00
wu ;
2001-08-20 12:05:51 +04:00
kfree ( md ) ;
return 0 ;
}
md - > dev = MKDEV ( DM_BLK_MAJOR , minor ) ;
md - > name [ 0 ] = ' \0 ' ;
2001-08-21 18:28:00 +04:00
md - > state = 0 ;
2001-08-20 12:05:51 +04:00
2001-09-26 18:32:07 +04:00
init_waitqueue_head ( & md - > wait ) ;
2001-08-20 12:05:51 +04:00
_devs [ minor ] = md ;
2001-09-26 18:32:07 +04:00
wu ;
2001-08-20 12:05:51 +04:00
2001-08-21 18:28:00 +04:00
return md ;
2001-08-20 12:05:51 +04:00
}
2001-09-26 18:32:07 +04:00
/*
* open a device so we can use it as a map
* destination .
*/
static int open_dev ( struct dm_dev * d )
{
int err ;
if ( d - > bd )
BUG ( ) ;
if ( ! ( d - > bd = bdget ( kdev_t_to_nr ( d - > dev ) ) ) )
return - ENOMEM ;
if ( ( err = blkdev_get ( d - > bd , FMODE_READ | FMODE_WRITE , 0 , BDEV_FILE ) ) ) {
bdput ( d - > bd ) ;
return err ;
}
return 0 ;
}
/*
* close a device that we ' ve been using .
*/
static void close_dev ( struct dm_dev * d )
{
if ( ! d - > bd )
return ;
blkdev_put ( d - > bd , BDEV_FILE ) ;
bdput ( d - > bd ) ;
d - > bd = 0 ;
}
/*
2001-09-26 21:32:57 +04:00
* Close a list of devices .
2001-09-26 18:32:07 +04:00
*/
2001-09-26 21:32:57 +04:00
static void close_devices ( struct list_head * devices )
2001-09-26 18:32:07 +04:00
{
struct list_head * tmp ;
for ( tmp = devices - > next ; tmp ! = devices ; tmp = tmp - > next ) {
struct dm_dev * dd = list_entry ( tmp , struct dm_dev , list ) ;
2001-09-26 21:32:57 +04:00
close_dev ( dd ) ;
2001-09-26 18:32:07 +04:00
}
}
/*
2001-09-26 21:32:57 +04:00
* Open a list of devices .
2001-09-26 18:32:07 +04:00
*/
2001-09-26 21:32:57 +04:00
static int open_devices ( struct list_head * devices )
2001-09-26 18:32:07 +04:00
{
2001-09-26 21:32:57 +04:00
int r = 0 ;
2001-09-26 18:32:07 +04:00
struct list_head * tmp ;
for ( tmp = devices - > next ; tmp ! = devices ; tmp = tmp - > next ) {
struct dm_dev * dd = list_entry ( tmp , struct dm_dev , list ) ;
2001-09-26 21:32:57 +04:00
if ( ( r = open_dev ( dd ) ) )
goto bad ;
2001-09-26 18:32:07 +04:00
}
2001-09-26 21:32:57 +04:00
return 0 ;
bad :
close_devices ( devices ) ;
return r ;
2001-09-26 18:32:07 +04:00
}
2001-08-29 17:58:48 +04:00
struct mapped_device * dm_find_by_minor ( int minor )
2001-08-20 17:45:43 +04:00
{
2001-08-20 18:06:25 +04:00
struct mapped_device * md ;
2001-09-26 18:32:07 +04:00
rl ;
2001-08-20 18:06:25 +04:00
md = _devs [ minor ] ;
2001-09-26 18:32:07 +04:00
ru ;
2001-08-20 17:45:43 +04:00
2001-08-20 18:06:25 +04:00
return md ;
2001-08-20 12:05:51 +04:00
}
2001-09-07 15:34:46 +04:00
static int register_device ( struct mapped_device * md )
{
md - > devfs_entry =
2001-09-26 18:32:07 +04:00
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 ) ;
2001-09-07 15:34:46 +04:00
if ( ! md - > devfs_entry )
return - ENOMEM ;
return 0 ;
}
static int unregister_device ( struct mapped_device * md )
{
devfs_unregister ( md - > devfs_entry ) ;
return 0 ;
}
2001-08-31 19:13:33 +04:00
/*
* constructor for a new device
*/
2001-09-18 20:53:18 +04:00
struct mapped_device * dm_create ( const char * name , int minor )
2001-08-20 12:05:51 +04:00
{
2001-08-22 19:01:09 +04:00
int r ;
2001-08-20 18:06:25 +04:00
struct mapped_device * md ;
2001-08-20 12:05:51 +04:00
2001-08-20 18:06:25 +04:00
if ( minor > = MAX_DEVICES )
2001-09-18 20:53:18 +04:00
return ERR_PTR ( - ENXIO ) ;
2001-08-20 12:05:51 +04:00
2001-08-23 16:35:02 +04:00
if ( ! ( md = alloc_dev ( minor ) ) )
2001-09-18 20:53:18 +04:00
return ERR_PTR ( - ENXIO ) ;
2001-08-20 18:06:25 +04:00
2001-09-26 18:32:07 +04:00
wl ;
2001-08-20 12:05:51 +04:00
strcpy ( md - > name , name ) ;
2001-09-26 18:32:07 +04:00
_devs [ minor ] = md ;
2001-09-07 15:34:46 +04:00
if ( ( r = register_device ( md ) ) ) {
2001-09-26 18:32:07 +04:00
wu ;
2001-09-18 20:53:18 +04:00
return ERR_PTR ( r ) ;
2001-08-22 19:01:09 +04:00
}
2001-09-26 18:32:07 +04:00
wu ;
2001-09-14 19:35:06 +04:00
2001-09-18 20:53:18 +04:00
return md ;
2001-08-20 12:05:51 +04:00
}
2001-08-31 19:13:33 +04:00
/*
* destructor for the device . md - > map is
* deliberately not destroyed , dm - fs should manage
* table objects .
*/
2001-09-18 20:53:18 +04:00
int dm_remove ( struct mapped_device * md )
2001-08-20 12:05:51 +04:00
{
2001-08-22 19:01:09 +04:00
int minor , r ;
2001-08-20 12:05:51 +04:00
2001-09-26 18:32:07 +04:00
wl ;
2001-08-28 18:05:22 +04:00
if ( md - > use_count ) {
2001-09-26 18:32:07 +04:00
wu ;
2001-08-28 17:04:44 +04:00
return - EPERM ;
}
2001-09-07 15:34:46 +04:00
if ( ( r = unregister_device ( md ) ) ) {
2001-09-26 18:32:07 +04:00
wu ;
2001-08-22 19:01:09 +04:00
return r ;
}
2001-08-20 18:06:25 +04:00
minor = MINOR ( md - > dev ) ;
2001-08-20 17:45:43 +04:00
_devs [ minor ] = 0 ;
2001-09-26 18:32:07 +04:00
wu ;
2001-08-21 18:28:00 +04:00
2001-09-18 20:53:18 +04:00
kfree ( md ) ;
2001-09-14 19:35:06 +04:00
2001-08-20 12:05:51 +04:00
return 0 ;
}
2001-09-26 18:32:07 +04:00
/*
* the hardsect size for a mapped device is the
* smallest hard sect size from the devices it
* maps onto .
*/
2001-09-26 21:32:57 +04:00
static int __find_hardsect_size ( struct list_head * devices )
2001-09-26 18:32:07 +04:00
{
2001-09-26 21:32:57 +04:00
int result = INT_MAX , size ;
struct list_head * tmp ;
2001-09-26 18:32:07 +04:00
2001-09-26 21:32:57 +04:00
for ( tmp = devices - > next ; tmp ! = devices ; tmp = tmp - > next ) {
struct dm_dev * dd = list_entry ( tmp , struct dm_dev , list ) ;
size = get_hardsect_size ( dd - > dev ) ;
if ( size < result )
result = size ;
}
return result ;
2001-09-26 18:32:07 +04:00
}
2001-08-31 19:13:33 +04:00
/*
2001-09-07 15:34:46 +04:00
* Bind a table to the device .
2001-08-31 19:13:33 +04:00
*/
2001-09-07 15:34:46 +04:00
void __bind ( struct mapped_device * md , struct dm_table * t )
2001-08-31 19:13:33 +04:00
{
int minor = MINOR ( md - > dev ) ;
md - > map = t ;
2001-09-26 18:32:07 +04:00
/* in k */
2001-08-31 19:13:33 +04:00
_block_size [ minor ] = ( t - > highs [ t - > num_targets - 1 ] + 1 ) > > 1 ;
2001-09-26 18:32:07 +04:00
_blksize_size [ minor ] = BLOCK_SIZE ;
2001-09-26 23:48:20 +04:00
_hardsect_size [ minor ] = __find_hardsect_size ( & t - > devices ) ;
2001-08-31 19:13:33 +04:00
register_disk ( NULL , md - > dev , 1 , & dm_blk_dops , _block_size [ minor ] ) ;
2001-08-20 20:12:22 +04:00
}
2001-08-31 19:13:33 +04:00
/*
* requeue the deferred buffer_heads by calling
* generic_make_request .
*/
2001-08-28 18:05:22 +04:00
static void __flush_deferred_io ( struct mapped_device * md )
{
struct deferred_io * c , * n ;
for ( c = md - > deferred , md - > deferred = 0 ; c ; c = n ) {
n = c - > next ;
generic_make_request ( c - > rw , c - > bh ) ;
free_deferred ( c ) ;
}
}
2001-08-31 19:13:33 +04:00
/*
* make the device available for use , if was
* previously suspended rather than newly created
* then all queued io is flushed
*/
2001-09-07 15:34:46 +04:00
int dm_activate ( struct mapped_device * md , struct dm_table * table )
2001-08-20 20:12:22 +04:00
{
2001-09-26 18:32:07 +04:00
int r ;
2001-09-07 15:34:46 +04:00
/* check that the mapping has at least been loaded. */
if ( ! table - > num_targets )
return - EINVAL ;
2001-08-20 20:12:22 +04:00
2001-09-26 18:32:07 +04:00
wl ;
2001-08-23 16:35:02 +04:00
2001-09-07 15:34:46 +04:00
/* you must be deactivated first */
2001-08-23 16:35:02 +04:00
if ( is_active ( md ) ) {
2001-09-26 18:32:07 +04:00
wu ;
2001-09-07 15:34:46 +04:00
return - EPERM ;
2001-08-23 16:35:02 +04:00
}
2001-09-07 15:34:46 +04:00
__bind ( md , table ) ;
2001-09-26 23:48:20 +04:00
if ( ( r = open_devices ( & md - > map - > devices ) ) ) {
2001-09-26 18:32:07 +04:00
wu ;
return r ;
}
2001-08-23 16:35:02 +04:00
set_bit ( DM_ACTIVE , & md - > state ) ;
2001-08-28 18:05:22 +04:00
__flush_deferred_io ( md ) ;
2001-09-26 18:32:07 +04:00
wu ;
2001-08-20 20:12:22 +04:00
return 0 ;
2001-09-07 15:34:46 +04:00
}
/*
* Deactivate the device , the device must not be
* opened by anyone .
*/
int dm_deactivate ( struct mapped_device * md )
{
2001-09-26 18:32:07 +04:00
rl ;
2001-09-07 15:34:46 +04:00
if ( md - > use_count ) {
2001-09-26 18:32:07 +04:00
ru ;
2001-09-07 15:34:46 +04:00
return - EPERM ;
}
fsync_dev ( md - > dev ) ;
2001-08-20 20:12:22 +04:00
2001-09-26 18:32:07 +04:00
ru ;
2001-08-20 20:12:22 +04:00
2001-09-26 18:32:07 +04:00
wl ;
2001-09-07 15:34:46 +04:00
if ( md - > use_count ) {
/* drat, somebody got in quick ... */
2001-09-26 18:32:07 +04:00
wu ;
2001-09-07 15:34:46 +04:00
return - EPERM ;
}
2001-09-26 23:48:20 +04:00
close_devices ( & md - > map - > devices ) ;
2001-09-07 15:34:46 +04:00
md - > map = 0 ;
clear_bit ( DM_ACTIVE , & md - > state ) ;
2001-09-26 18:32:07 +04:00
wu ;
2001-09-07 15:34:46 +04:00
return 0 ;
2001-08-20 20:12:22 +04:00
}
2001-08-31 19:13:33 +04:00
/*
2001-09-07 15:34:46 +04:00
* We need to be able to change a mapping table
2001-08-31 19:13:33 +04:00
* under a mounted filesystem . for example we
* might want to move some data in the background .
* Before the table can be swapped with
* dm_bind_table , dm_suspend must be called to
* flush any in flight buffer_heads and ensure
* that any further io gets deferred .
*/
2001-08-20 20:12:22 +04:00
void dm_suspend ( struct mapped_device * md )
{
2001-08-28 17:04:44 +04:00
DECLARE_WAITQUEUE ( wait , current ) ;
2001-08-31 19:13:33 +04:00
2001-09-26 18:32:07 +04:00
wl ;
2001-09-07 15:34:46 +04:00
if ( ! is_active ( md ) ) {
2001-09-26 18:32:07 +04:00
wu ;
2001-08-20 20:12:22 +04:00
return ;
2001-09-07 15:34:46 +04:00
}
clear_bit ( DM_ACTIVE , & md - > state ) ;
2001-09-26 18:32:07 +04:00
wu ;
2001-08-20 20:12:22 +04:00
2001-08-28 17:04:44 +04:00
/* wait for all the pending io to flush */
2001-09-26 18:32:07 +04:00
add_wait_queue ( & md - > wait , & wait ) ;
2001-09-14 18:03:02 +04:00
current - > state = TASK_UNINTERRUPTIBLE ;
2001-08-28 17:04:44 +04:00
do {
2001-09-26 18:32:07 +04:00
wl ;
if ( ! atomic_read ( & md - > pending ) )
2001-08-28 17:04:44 +04:00
break ;
2001-09-26 18:32:07 +04:00
wu ;
2001-08-28 17:04:44 +04:00
schedule ( ) ;
} while ( 1 ) ;
current - > state = TASK_RUNNING ;
2001-09-26 21:32:57 +04:00
remove_wait_queue ( & md - > wait , & wait ) ;
2001-09-26 23:48:20 +04:00
close_devices ( & md - > map - > devices ) ;
2001-08-28 17:04:44 +04:00
2001-09-07 15:34:46 +04:00
md - > map = 0 ;
2001-09-26 18:32:07 +04:00
wu ;
2001-08-20 20:12:22 +04:00
}
2001-08-31 19:13:33 +04:00
struct block_device_operations dm_blk_dops = {
2001-09-14 17:27:58 +04:00
open : dm_blk_open ,
2001-09-13 18:01:13 +04:00
release : dm_blk_close ,
2001-09-14 17:27:58 +04:00
ioctl : dm_blk_ioctl
2001-08-31 19:13:33 +04:00
} ;
2001-08-20 12:05:51 +04:00
/*
* module hooks
*/
2001-08-31 19:13:33 +04:00
module_init ( dm_init ) ;
module_exit ( dm_exit ) ;
2001-08-20 12:05:51 +04:00
2001-09-07 15:34:46 +04:00
MODULE_DESCRIPTION ( " device-mapper driver " ) ;
MODULE_AUTHOR ( " Joe Thornber <thornber@btconnect.com> " ) ;
2001-08-20 12:05:51 +04:00
/*
* Local variables :
* c - file - style : " linux "
* End :
*/