2001-08-20 12:03:02 +04:00
/*
* dm - target . 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 .
*/
/*
* 16 / 08 / 2001 - First Version [ Joe Thornber ]
*/
# include "dm.h"
2001-09-13 22:30:05 +04:00
# include <linux/kmod.h>
2001-08-20 12:03:02 +04:00
2001-09-13 22:30:05 +04:00
static LIST_HEAD ( dm_targets ) ;
static rwlock_t dm_targets_lock = RW_LOCK_UNLOCKED ;
2001-08-20 12:03:02 +04:00
2001-09-13 22:30:05 +04:00
# define DM_MOD_NAME_SIZE 32
2001-08-20 12:03:02 +04:00
2001-09-07 15:34:46 +04:00
struct target_type * dm_get_target_type ( const char * name )
2001-08-20 12:03:02 +04:00
{
2001-09-13 22:30:05 +04:00
struct list_head * tmp , * head ;
2001-08-31 16:49:31 +04:00
struct target_type * t ;
2001-09-13 22:30:05 +04:00
int try = 0 ;
/* Length check for strcat() below */
if ( strlen ( name ) > ( DM_MOD_NAME_SIZE - 4 ) )
return NULL ;
try_again :
read_lock ( & dm_targets_lock ) ;
tmp = head = & dm_targets ;
for ( ; ; ) {
tmp = tmp - > next ;
if ( tmp = = head )
break ;
t = list_entry ( tmp , struct target_type , list ) ;
if ( strcmp ( name , t - > name ) = = 0 ) {
if ( t - > use = = 0 & & t - > module )
__MOD_INC_USE_COUNT ( t - > module ) ;
t - > use + + ;
read_unlock ( & dm_targets_lock ) ;
return t ;
}
}
read_unlock ( & dm_targets_lock ) ;
if ( try + + = = 0 ) {
char module_name [ DM_MOD_NAME_SIZE ] = " dm- " ;
/* strcat() is only safe due to length check above */
strcat ( module_name , name ) ;
request_module ( module_name ) ;
goto try_again ;
}
2001-08-20 12:03:02 +04:00
2001-09-13 22:30:05 +04:00
return NULL ;
2001-08-20 12:03:02 +04:00
}
2001-09-13 22:30:05 +04:00
void dm_put_target_type ( struct target_type * t )
2001-08-20 12:03:02 +04:00
{
2001-09-13 22:30:05 +04:00
read_lock ( & dm_targets_lock ) ;
if ( - - t - > use = = 0 & & t - > module )
__MOD_DEC_USE_COUNT ( t - > module ) ;
if ( t - > use < 0 )
BUG ( ) ;
read_unlock ( & dm_targets_lock ) ;
}
2001-08-20 12:03:02 +04:00
2001-09-13 22:30:05 +04:00
int dm_register_target ( struct target_type * t )
{
struct list_head * tmp , * head ;
struct target_type * t2 ;
int rv = 0 ;
write_lock ( & dm_targets_lock ) ;
tmp = head = & dm_targets ;
for ( ; ; ) {
if ( tmp = = head )
break ;
t2 = list_entry ( tmp , struct target_type , list ) ;
if ( strcmp ( t - > name , t2 - > name ) ! = 0 )
continue ;
rv = - EEXIST ;
break ;
2001-08-20 12:03:02 +04:00
}
2001-09-13 22:30:05 +04:00
if ( rv = = 0 )
list_add ( & t - > list , & dm_targets ) ;
write_unlock ( & dm_targets_lock ) ;
return rv ;
}
2001-08-20 12:03:02 +04:00
2001-09-13 22:30:05 +04:00
int dm_unregister_target ( struct target_type * t )
{
int rv = - ETXTBSY ;
2001-08-20 12:03:02 +04:00
2001-09-13 22:30:05 +04:00
write_lock ( & dm_targets_lock ) ;
if ( t - > use = = 0 ) {
list_del ( & t - > list ) ;
rv = 0 ;
}
write_unlock ( & dm_targets_lock ) ;
2001-08-20 12:03:02 +04:00
2001-09-13 22:30:05 +04:00
return rv ;
2001-08-20 12:03:02 +04:00
}
/*
2001-08-31 16:49:31 +04:00
* io - err : always fails an io , useful for bringing
* up LV ' s that have holes in them .
2001-08-20 12:03:02 +04:00
*/
2001-09-20 23:25:58 +04:00
static void * io_err_ctr ( struct dm_table * t , offset_t b , offset_t l ,
struct text_region * args )
2001-08-20 12:03:02 +04:00
{
/* this takes no arguments */
2001-09-20 23:25:58 +04:00
return NULL ;
2001-08-20 12:03:02 +04:00
}
2001-08-31 20:36:56 +04:00
static void io_err_dtr ( struct dm_table * t , void * c )
2001-08-20 12:03:02 +04:00
{
/* empty */
}
2001-09-14 20:22:02 +04:00
static int io_err_map ( struct buffer_head * bh , int rw , void * context )
2001-08-20 12:03:02 +04:00
{
buffer_IO_error ( bh ) ;
return 0 ;
}
2001-09-13 22:30:05 +04:00
static struct target_type error_target = {
name : " error " ,
ctr : io_err_ctr ,
dtr : io_err_dtr ,
map : io_err_map
2001-08-20 12:03:02 +04:00
} ;
2001-08-31 16:49:31 +04:00
int dm_target_init ( void )
2001-08-20 12:03:02 +04:00
{
2001-09-13 22:30:05 +04:00
return dm_register_target ( & error_target ) ;
2001-08-20 12:03:02 +04:00
}
2001-08-31 20:36:56 +04:00
EXPORT_SYMBOL ( dm_register_target ) ;
2001-09-13 22:30:05 +04:00
EXPORT_SYMBOL ( dm_unregister_target ) ;