blob: 64dd0b34fcf490cee3779e179e9b8c7c543b7e53 [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * Copyright (C) 2001 Sistina Software (UK) Limited
3 *
4 * This file is released under the GPL.
5 */
6
Mike Snitzer4cc96132016-05-12 16:28:10 -04007#include "dm-core.h"
Linus Torvalds1da177e2005-04-16 15:20:36 -07008
9#include <linux/module.h>
10#include <linux/init.h>
11#include <linux/kmod.h>
12#include <linux/bio.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070013
Alasdair G Kergon72d94862006-06-26 00:27:35 -070014#define DM_MSG_PREFIX "target"
15
Linus Torvalds1da177e2005-04-16 15:20:36 -070016static LIST_HEAD(_targets);
17static DECLARE_RWSEM(_lock);
18
Cheng Renquan45194e42009-04-02 19:55:28 +010019static inline struct target_type *__find_target_type(const char *name)
Linus Torvalds1da177e2005-04-16 15:20:36 -070020{
Cheng Renquan45194e42009-04-02 19:55:28 +010021 struct target_type *tt;
Linus Torvalds1da177e2005-04-16 15:20:36 -070022
Cheng Renquan45194e42009-04-02 19:55:28 +010023 list_for_each_entry(tt, &_targets, list)
24 if (!strcmp(name, tt->name))
25 return tt;
Linus Torvalds1da177e2005-04-16 15:20:36 -070026
27 return NULL;
28}
29
Cheng Renquan45194e42009-04-02 19:55:28 +010030static struct target_type *get_target_type(const char *name)
Linus Torvalds1da177e2005-04-16 15:20:36 -070031{
Cheng Renquan45194e42009-04-02 19:55:28 +010032 struct target_type *tt;
Linus Torvalds1da177e2005-04-16 15:20:36 -070033
34 down_read(&_lock);
35
Cheng Renquan45194e42009-04-02 19:55:28 +010036 tt = __find_target_type(name);
37 if (tt && !try_module_get(tt->module))
38 tt = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -070039
40 up_read(&_lock);
Cheng Renquan45194e42009-04-02 19:55:28 +010041 return tt;
Linus Torvalds1da177e2005-04-16 15:20:36 -070042}
43
44static void load_module(const char *name)
45{
46 request_module("dm-%s", name);
47}
48
49struct target_type *dm_get_target_type(const char *name)
50{
Cheng Renquan45194e42009-04-02 19:55:28 +010051 struct target_type *tt = get_target_type(name);
Linus Torvalds1da177e2005-04-16 15:20:36 -070052
Cheng Renquan45194e42009-04-02 19:55:28 +010053 if (!tt) {
Linus Torvalds1da177e2005-04-16 15:20:36 -070054 load_module(name);
Cheng Renquan45194e42009-04-02 19:55:28 +010055 tt = get_target_type(name);
Linus Torvalds1da177e2005-04-16 15:20:36 -070056 }
57
Cheng Renquan45194e42009-04-02 19:55:28 +010058 return tt;
Linus Torvalds1da177e2005-04-16 15:20:36 -070059}
60
Cheng Renquan45194e42009-04-02 19:55:28 +010061void dm_put_target_type(struct target_type *tt)
Linus Torvalds1da177e2005-04-16 15:20:36 -070062{
Linus Torvalds1da177e2005-04-16 15:20:36 -070063 down_read(&_lock);
Cheng Renquan45194e42009-04-02 19:55:28 +010064 module_put(tt->module);
Linus Torvalds1da177e2005-04-16 15:20:36 -070065 up_read(&_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -070066}
67
Linus Torvalds1da177e2005-04-16 15:20:36 -070068int dm_target_iterate(void (*iter_func)(struct target_type *tt,
69 void *param), void *param)
70{
Cheng Renquan45194e42009-04-02 19:55:28 +010071 struct target_type *tt;
Linus Torvalds1da177e2005-04-16 15:20:36 -070072
73 down_read(&_lock);
Cheng Renquan45194e42009-04-02 19:55:28 +010074 list_for_each_entry(tt, &_targets, list)
75 iter_func(tt, param);
Linus Torvalds1da177e2005-04-16 15:20:36 -070076 up_read(&_lock);
77
78 return 0;
79}
80
Cheng Renquan45194e42009-04-02 19:55:28 +010081int dm_register_target(struct target_type *tt)
Linus Torvalds1da177e2005-04-16 15:20:36 -070082{
83 int rv = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -070084
85 down_write(&_lock);
Cheng Renquan45194e42009-04-02 19:55:28 +010086 if (__find_target_type(tt->name))
Linus Torvalds1da177e2005-04-16 15:20:36 -070087 rv = -EEXIST;
88 else
Cheng Renquan45194e42009-04-02 19:55:28 +010089 list_add(&tt->list, &_targets);
Linus Torvalds1da177e2005-04-16 15:20:36 -070090
91 up_write(&_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -070092 return rv;
93}
94
Cheng Renquan45194e42009-04-02 19:55:28 +010095void dm_unregister_target(struct target_type *tt)
Linus Torvalds1da177e2005-04-16 15:20:36 -070096{
Linus Torvalds1da177e2005-04-16 15:20:36 -070097 down_write(&_lock);
Cheng Renquan45194e42009-04-02 19:55:28 +010098 if (!__find_target_type(tt->name)) {
99 DMCRIT("Unregistering unrecognised target: %s", tt->name);
Mikulas Patocka10d3bd02009-01-06 03:04:58 +0000100 BUG();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700101 }
102
Cheng Renquan45194e42009-04-02 19:55:28 +0100103 list_del(&tt->list);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700104
105 up_write(&_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700106}
107
108/*
109 * io-err: always fails an io, useful for bringing
110 * up LVs that have holes in them.
111 */
Cheng Renquan45194e42009-04-02 19:55:28 +0100112static int io_err_ctr(struct dm_target *tt, unsigned int argc, char **args)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700113{
Mike Snitzer38e1b252010-08-12 04:14:14 +0100114 /*
115 * Return error for discards instead of -EOPNOTSUPP
116 */
Alasdair G Kergon55a62ee2013-03-01 22:45:47 +0000117 tt->num_discard_bios = 1;
Mike Snitzer38e1b252010-08-12 04:14:14 +0100118
Linus Torvalds1da177e2005-04-16 15:20:36 -0700119 return 0;
120}
121
Cheng Renquan45194e42009-04-02 19:55:28 +0100122static void io_err_dtr(struct dm_target *tt)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700123{
124 /* empty */
125}
126
Mikulas Patocka7de3ee52012-12-21 20:23:41 +0000127static int io_err_map(struct dm_target *tt, struct bio *bio)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700128{
Christoph Hellwig846785e2017-06-03 09:38:02 +0200129 return DM_MAPIO_KILL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700130}
131
Mike Snitzere5863d92014-12-17 21:08:12 -0500132static int io_err_clone_and_map_rq(struct dm_target *ti, struct request *rq,
133 union map_info *map_context,
134 struct request **clone)
135{
Christoph Hellwig412445a2017-04-26 09:40:39 +0200136 return DM_MAPIO_KILL;
Mike Snitzere5863d92014-12-17 21:08:12 -0500137}
138
Yufen Yu5de719e2019-04-24 23:19:05 +0800139static void io_err_release_clone_rq(struct request *clone,
140 union map_info *map_context)
Mike Snitzere5863d92014-12-17 21:08:12 -0500141{
142}
143
Dan Williams817bf402017-04-12 13:37:44 -0700144static long io_err_dax_direct_access(struct dm_target *ti, pgoff_t pgoff,
145 long nr_pages, void **kaddr, pfn_t *pfn)
Mike Snitzerf8df1fd2016-06-24 17:09:35 -0400146{
147 return -EIO;
148}
149
Linus Torvalds1da177e2005-04-16 15:20:36 -0700150static struct target_type error_target = {
151 .name = "error",
Mike Snitzerf8df1fd2016-06-24 17:09:35 -0400152 .version = {1, 5, 0},
Mike Snitzerf083b092016-02-06 18:38:46 -0500153 .features = DM_TARGET_WILDCARD,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700154 .ctr = io_err_ctr,
155 .dtr = io_err_dtr,
156 .map = io_err_map,
Mike Snitzere5863d92014-12-17 21:08:12 -0500157 .clone_and_map_rq = io_err_clone_and_map_rq,
158 .release_clone_rq = io_err_release_clone_rq,
Dan Williams817bf402017-04-12 13:37:44 -0700159 .direct_access = io_err_dax_direct_access,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700160};
161
162int __init dm_target_init(void)
163{
164 return dm_register_target(&error_target);
165}
166
167void dm_target_exit(void)
168{
Mikulas Patocka10d3bd02009-01-06 03:04:58 +0000169 dm_unregister_target(&error_target);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700170}
171
172EXPORT_SYMBOL(dm_register_target);
173EXPORT_SYMBOL(dm_unregister_target);