11da177e4SLinus Torvalds /* 21da177e4SLinus Torvalds * Copyright (C) 2001 Sistina Software (UK) Limited 31da177e4SLinus Torvalds * 41da177e4SLinus Torvalds * This file is released under the GPL. 51da177e4SLinus Torvalds */ 61da177e4SLinus Torvalds 71da177e4SLinus Torvalds #include "dm.h" 81da177e4SLinus Torvalds 91da177e4SLinus Torvalds #include <linux/module.h> 101da177e4SLinus Torvalds #include <linux/init.h> 111da177e4SLinus Torvalds #include <linux/kmod.h> 121da177e4SLinus Torvalds #include <linux/bio.h> 131da177e4SLinus Torvalds #include <linux/slab.h> 141da177e4SLinus Torvalds 1572d94861SAlasdair G Kergon #define DM_MSG_PREFIX "target" 1672d94861SAlasdair G Kergon 171da177e4SLinus Torvalds static LIST_HEAD(_targets); 181da177e4SLinus Torvalds static DECLARE_RWSEM(_lock); 191da177e4SLinus Torvalds 201da177e4SLinus Torvalds #define DM_MOD_NAME_SIZE 32 211da177e4SLinus Torvalds 22*45194e4fSCheng Renquan static inline struct target_type *__find_target_type(const char *name) 231da177e4SLinus Torvalds { 24*45194e4fSCheng Renquan struct target_type *tt; 251da177e4SLinus Torvalds 26*45194e4fSCheng Renquan list_for_each_entry(tt, &_targets, list) 27*45194e4fSCheng Renquan if (!strcmp(name, tt->name)) 28*45194e4fSCheng Renquan return tt; 291da177e4SLinus Torvalds 301da177e4SLinus Torvalds return NULL; 311da177e4SLinus Torvalds } 321da177e4SLinus Torvalds 33*45194e4fSCheng Renquan static struct target_type *get_target_type(const char *name) 341da177e4SLinus Torvalds { 35*45194e4fSCheng Renquan struct target_type *tt; 361da177e4SLinus Torvalds 371da177e4SLinus Torvalds down_read(&_lock); 381da177e4SLinus Torvalds 39*45194e4fSCheng Renquan tt = __find_target_type(name); 40*45194e4fSCheng Renquan if (tt && !try_module_get(tt->module)) 41*45194e4fSCheng Renquan tt = NULL; 421da177e4SLinus Torvalds 431da177e4SLinus Torvalds up_read(&_lock); 44*45194e4fSCheng Renquan return tt; 451da177e4SLinus Torvalds } 461da177e4SLinus Torvalds 471da177e4SLinus Torvalds static void load_module(const char *name) 481da177e4SLinus Torvalds { 491da177e4SLinus Torvalds request_module("dm-%s", name); 501da177e4SLinus Torvalds } 511da177e4SLinus Torvalds 521da177e4SLinus Torvalds struct target_type *dm_get_target_type(const char *name) 531da177e4SLinus Torvalds { 54*45194e4fSCheng Renquan struct target_type *tt = get_target_type(name); 551da177e4SLinus Torvalds 56*45194e4fSCheng Renquan if (!tt) { 571da177e4SLinus Torvalds load_module(name); 58*45194e4fSCheng Renquan tt = get_target_type(name); 591da177e4SLinus Torvalds } 601da177e4SLinus Torvalds 61*45194e4fSCheng Renquan return tt; 621da177e4SLinus Torvalds } 631da177e4SLinus Torvalds 64*45194e4fSCheng Renquan void dm_put_target_type(struct target_type *tt) 651da177e4SLinus Torvalds { 661da177e4SLinus Torvalds down_read(&_lock); 67*45194e4fSCheng Renquan module_put(tt->module); 681da177e4SLinus Torvalds up_read(&_lock); 691da177e4SLinus Torvalds } 701da177e4SLinus Torvalds 711da177e4SLinus Torvalds int dm_target_iterate(void (*iter_func)(struct target_type *tt, 721da177e4SLinus Torvalds void *param), void *param) 731da177e4SLinus Torvalds { 74*45194e4fSCheng Renquan struct target_type *tt; 751da177e4SLinus Torvalds 761da177e4SLinus Torvalds down_read(&_lock); 77*45194e4fSCheng Renquan list_for_each_entry(tt, &_targets, list) 78*45194e4fSCheng Renquan iter_func(tt, param); 791da177e4SLinus Torvalds up_read(&_lock); 801da177e4SLinus Torvalds 811da177e4SLinus Torvalds return 0; 821da177e4SLinus Torvalds } 831da177e4SLinus Torvalds 84*45194e4fSCheng Renquan int dm_register_target(struct target_type *tt) 851da177e4SLinus Torvalds { 861da177e4SLinus Torvalds int rv = 0; 871da177e4SLinus Torvalds 881da177e4SLinus Torvalds down_write(&_lock); 89*45194e4fSCheng Renquan if (__find_target_type(tt->name)) 901da177e4SLinus Torvalds rv = -EEXIST; 911da177e4SLinus Torvalds else 92*45194e4fSCheng Renquan list_add(&tt->list, &_targets); 931da177e4SLinus Torvalds 941da177e4SLinus Torvalds up_write(&_lock); 951da177e4SLinus Torvalds return rv; 961da177e4SLinus Torvalds } 971da177e4SLinus Torvalds 98*45194e4fSCheng Renquan void dm_unregister_target(struct target_type *tt) 991da177e4SLinus Torvalds { 1001da177e4SLinus Torvalds down_write(&_lock); 101*45194e4fSCheng Renquan if (!__find_target_type(tt->name)) { 102*45194e4fSCheng Renquan DMCRIT("Unregistering unrecognised target: %s", tt->name); 10310d3bd09SMikulas Patocka BUG(); 1041da177e4SLinus Torvalds } 1051da177e4SLinus Torvalds 106*45194e4fSCheng Renquan list_del(&tt->list); 1071da177e4SLinus Torvalds 1081da177e4SLinus Torvalds up_write(&_lock); 1091da177e4SLinus Torvalds } 1101da177e4SLinus Torvalds 1111da177e4SLinus Torvalds /* 1121da177e4SLinus Torvalds * io-err: always fails an io, useful for bringing 1131da177e4SLinus Torvalds * up LVs that have holes in them. 1141da177e4SLinus Torvalds */ 115*45194e4fSCheng Renquan static int io_err_ctr(struct dm_target *tt, unsigned int argc, char **args) 1161da177e4SLinus Torvalds { 1171da177e4SLinus Torvalds return 0; 1181da177e4SLinus Torvalds } 1191da177e4SLinus Torvalds 120*45194e4fSCheng Renquan static void io_err_dtr(struct dm_target *tt) 1211da177e4SLinus Torvalds { 1221da177e4SLinus Torvalds /* empty */ 1231da177e4SLinus Torvalds } 1241da177e4SLinus Torvalds 125*45194e4fSCheng Renquan static int io_err_map(struct dm_target *tt, struct bio *bio, 1261da177e4SLinus Torvalds union map_info *map_context) 1271da177e4SLinus Torvalds { 1281da177e4SLinus Torvalds return -EIO; 1291da177e4SLinus Torvalds } 1301da177e4SLinus Torvalds 1311da177e4SLinus Torvalds static struct target_type error_target = { 1321da177e4SLinus Torvalds .name = "error", 1331da177e4SLinus Torvalds .version = {1, 0, 1}, 1341da177e4SLinus Torvalds .ctr = io_err_ctr, 1351da177e4SLinus Torvalds .dtr = io_err_dtr, 1361da177e4SLinus Torvalds .map = io_err_map, 1371da177e4SLinus Torvalds }; 1381da177e4SLinus Torvalds 1391da177e4SLinus Torvalds int __init dm_target_init(void) 1401da177e4SLinus Torvalds { 1411da177e4SLinus Torvalds return dm_register_target(&error_target); 1421da177e4SLinus Torvalds } 1431da177e4SLinus Torvalds 1441da177e4SLinus Torvalds void dm_target_exit(void) 1451da177e4SLinus Torvalds { 14610d3bd09SMikulas Patocka dm_unregister_target(&error_target); 1471da177e4SLinus Torvalds } 1481da177e4SLinus Torvalds 1491da177e4SLinus Torvalds EXPORT_SYMBOL(dm_register_target); 1501da177e4SLinus Torvalds EXPORT_SYMBOL(dm_unregister_target); 151