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 15*72d94861SAlasdair G Kergon #define DM_MSG_PREFIX "target" 16*72d94861SAlasdair G Kergon 171da177e4SLinus Torvalds struct tt_internal { 181da177e4SLinus Torvalds struct target_type tt; 191da177e4SLinus Torvalds 201da177e4SLinus Torvalds struct list_head list; 211da177e4SLinus Torvalds long use; 221da177e4SLinus Torvalds }; 231da177e4SLinus Torvalds 241da177e4SLinus Torvalds static LIST_HEAD(_targets); 251da177e4SLinus Torvalds static DECLARE_RWSEM(_lock); 261da177e4SLinus Torvalds 271da177e4SLinus Torvalds #define DM_MOD_NAME_SIZE 32 281da177e4SLinus Torvalds 291da177e4SLinus Torvalds static inline struct tt_internal *__find_target_type(const char *name) 301da177e4SLinus Torvalds { 311da177e4SLinus Torvalds struct tt_internal *ti; 321da177e4SLinus Torvalds 331da177e4SLinus Torvalds list_for_each_entry (ti, &_targets, list) 341da177e4SLinus Torvalds if (!strcmp(name, ti->tt.name)) 351da177e4SLinus Torvalds return ti; 361da177e4SLinus Torvalds 371da177e4SLinus Torvalds return NULL; 381da177e4SLinus Torvalds } 391da177e4SLinus Torvalds 401da177e4SLinus Torvalds static struct tt_internal *get_target_type(const char *name) 411da177e4SLinus Torvalds { 421da177e4SLinus Torvalds struct tt_internal *ti; 431da177e4SLinus Torvalds 441da177e4SLinus Torvalds down_read(&_lock); 451da177e4SLinus Torvalds 461da177e4SLinus Torvalds ti = __find_target_type(name); 471da177e4SLinus Torvalds if (ti) { 481da177e4SLinus Torvalds if ((ti->use == 0) && !try_module_get(ti->tt.module)) 491da177e4SLinus Torvalds ti = NULL; 501da177e4SLinus Torvalds else 511da177e4SLinus Torvalds ti->use++; 521da177e4SLinus Torvalds } 531da177e4SLinus Torvalds 541da177e4SLinus Torvalds up_read(&_lock); 551da177e4SLinus Torvalds return ti; 561da177e4SLinus Torvalds } 571da177e4SLinus Torvalds 581da177e4SLinus Torvalds static void load_module(const char *name) 591da177e4SLinus Torvalds { 601da177e4SLinus Torvalds request_module("dm-%s", name); 611da177e4SLinus Torvalds } 621da177e4SLinus Torvalds 631da177e4SLinus Torvalds struct target_type *dm_get_target_type(const char *name) 641da177e4SLinus Torvalds { 651da177e4SLinus Torvalds struct tt_internal *ti = get_target_type(name); 661da177e4SLinus Torvalds 671da177e4SLinus Torvalds if (!ti) { 681da177e4SLinus Torvalds load_module(name); 691da177e4SLinus Torvalds ti = get_target_type(name); 701da177e4SLinus Torvalds } 711da177e4SLinus Torvalds 721da177e4SLinus Torvalds return ti ? &ti->tt : NULL; 731da177e4SLinus Torvalds } 741da177e4SLinus Torvalds 751da177e4SLinus Torvalds void dm_put_target_type(struct target_type *t) 761da177e4SLinus Torvalds { 771da177e4SLinus Torvalds struct tt_internal *ti = (struct tt_internal *) t; 781da177e4SLinus Torvalds 791da177e4SLinus Torvalds down_read(&_lock); 801da177e4SLinus Torvalds if (--ti->use == 0) 811da177e4SLinus Torvalds module_put(ti->tt.module); 821da177e4SLinus Torvalds 83543cb2a4SEric Sesterhenn BUG_ON(ti->use < 0); 841da177e4SLinus Torvalds up_read(&_lock); 851da177e4SLinus Torvalds 861da177e4SLinus Torvalds return; 871da177e4SLinus Torvalds } 881da177e4SLinus Torvalds 891da177e4SLinus Torvalds static struct tt_internal *alloc_target(struct target_type *t) 901da177e4SLinus Torvalds { 911da177e4SLinus Torvalds struct tt_internal *ti = kmalloc(sizeof(*ti), GFP_KERNEL); 921da177e4SLinus Torvalds 931da177e4SLinus Torvalds if (ti) { 941da177e4SLinus Torvalds memset(ti, 0, sizeof(*ti)); 951da177e4SLinus Torvalds ti->tt = *t; 961da177e4SLinus Torvalds } 971da177e4SLinus Torvalds 981da177e4SLinus Torvalds return ti; 991da177e4SLinus Torvalds } 1001da177e4SLinus Torvalds 1011da177e4SLinus Torvalds 1021da177e4SLinus Torvalds int dm_target_iterate(void (*iter_func)(struct target_type *tt, 1031da177e4SLinus Torvalds void *param), void *param) 1041da177e4SLinus Torvalds { 1051da177e4SLinus Torvalds struct tt_internal *ti; 1061da177e4SLinus Torvalds 1071da177e4SLinus Torvalds down_read(&_lock); 1081da177e4SLinus Torvalds list_for_each_entry (ti, &_targets, list) 1091da177e4SLinus Torvalds iter_func(&ti->tt, param); 1101da177e4SLinus Torvalds up_read(&_lock); 1111da177e4SLinus Torvalds 1121da177e4SLinus Torvalds return 0; 1131da177e4SLinus Torvalds } 1141da177e4SLinus Torvalds 1151da177e4SLinus Torvalds int dm_register_target(struct target_type *t) 1161da177e4SLinus Torvalds { 1171da177e4SLinus Torvalds int rv = 0; 1181da177e4SLinus Torvalds struct tt_internal *ti = alloc_target(t); 1191da177e4SLinus Torvalds 1201da177e4SLinus Torvalds if (!ti) 1211da177e4SLinus Torvalds return -ENOMEM; 1221da177e4SLinus Torvalds 1231da177e4SLinus Torvalds down_write(&_lock); 1241da177e4SLinus Torvalds if (__find_target_type(t->name)) 1251da177e4SLinus Torvalds rv = -EEXIST; 1261da177e4SLinus Torvalds else 1271da177e4SLinus Torvalds list_add(&ti->list, &_targets); 1281da177e4SLinus Torvalds 1291da177e4SLinus Torvalds up_write(&_lock); 1301da177e4SLinus Torvalds if (rv) 1311da177e4SLinus Torvalds kfree(ti); 1321da177e4SLinus Torvalds return rv; 1331da177e4SLinus Torvalds } 1341da177e4SLinus Torvalds 1351da177e4SLinus Torvalds int dm_unregister_target(struct target_type *t) 1361da177e4SLinus Torvalds { 1371da177e4SLinus Torvalds struct tt_internal *ti; 1381da177e4SLinus Torvalds 1391da177e4SLinus Torvalds down_write(&_lock); 1401da177e4SLinus Torvalds if (!(ti = __find_target_type(t->name))) { 1411da177e4SLinus Torvalds up_write(&_lock); 1421da177e4SLinus Torvalds return -EINVAL; 1431da177e4SLinus Torvalds } 1441da177e4SLinus Torvalds 1451da177e4SLinus Torvalds if (ti->use) { 1461da177e4SLinus Torvalds up_write(&_lock); 1471da177e4SLinus Torvalds return -ETXTBSY; 1481da177e4SLinus Torvalds } 1491da177e4SLinus Torvalds 1501da177e4SLinus Torvalds list_del(&ti->list); 1511da177e4SLinus Torvalds kfree(ti); 1521da177e4SLinus Torvalds 1531da177e4SLinus Torvalds up_write(&_lock); 1541da177e4SLinus Torvalds return 0; 1551da177e4SLinus Torvalds } 1561da177e4SLinus Torvalds 1571da177e4SLinus Torvalds /* 1581da177e4SLinus Torvalds * io-err: always fails an io, useful for bringing 1591da177e4SLinus Torvalds * up LVs that have holes in them. 1601da177e4SLinus Torvalds */ 1611da177e4SLinus Torvalds static int io_err_ctr(struct dm_target *ti, unsigned int argc, char **args) 1621da177e4SLinus Torvalds { 1631da177e4SLinus Torvalds return 0; 1641da177e4SLinus Torvalds } 1651da177e4SLinus Torvalds 1661da177e4SLinus Torvalds static void io_err_dtr(struct dm_target *ti) 1671da177e4SLinus Torvalds { 1681da177e4SLinus Torvalds /* empty */ 1691da177e4SLinus Torvalds } 1701da177e4SLinus Torvalds 1711da177e4SLinus Torvalds static int io_err_map(struct dm_target *ti, struct bio *bio, 1721da177e4SLinus Torvalds union map_info *map_context) 1731da177e4SLinus Torvalds { 1741da177e4SLinus Torvalds return -EIO; 1751da177e4SLinus Torvalds } 1761da177e4SLinus Torvalds 1771da177e4SLinus Torvalds static struct target_type error_target = { 1781da177e4SLinus Torvalds .name = "error", 1791da177e4SLinus Torvalds .version = {1, 0, 1}, 1801da177e4SLinus Torvalds .ctr = io_err_ctr, 1811da177e4SLinus Torvalds .dtr = io_err_dtr, 1821da177e4SLinus Torvalds .map = io_err_map, 1831da177e4SLinus Torvalds }; 1841da177e4SLinus Torvalds 1851da177e4SLinus Torvalds int __init dm_target_init(void) 1861da177e4SLinus Torvalds { 1871da177e4SLinus Torvalds return dm_register_target(&error_target); 1881da177e4SLinus Torvalds } 1891da177e4SLinus Torvalds 1901da177e4SLinus Torvalds void dm_target_exit(void) 1911da177e4SLinus Torvalds { 1921da177e4SLinus Torvalds if (dm_unregister_target(&error_target)) 1931da177e4SLinus Torvalds DMWARN("error target unregistration failed"); 1941da177e4SLinus Torvalds } 1951da177e4SLinus Torvalds 1961da177e4SLinus Torvalds EXPORT_SYMBOL(dm_register_target); 1971da177e4SLinus Torvalds EXPORT_SYMBOL(dm_unregister_target); 198