Merge tag 'trace-printf-v6.13' of git://git.kernel.org/pub/scm/linux/kernel/git/trace...
[drm/drm-misc.git] / drivers / md / dm-cache-policy.c
blob9330d5748895fcf2b01dbe2e6701122c8bb91c7f
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3 * Copyright (C) 2012 Red Hat. All rights reserved.
5 * This file is released under the GPL.
6 */
8 #include "dm-cache-policy-internal.h"
9 #include "dm.h"
11 #include <linux/module.h>
12 #include <linux/slab.h>
14 /*----------------------------------------------------------------*/
16 #define DM_MSG_PREFIX "cache-policy"
18 static DEFINE_SPINLOCK(register_lock);
19 static LIST_HEAD(register_list);
21 static struct dm_cache_policy_type *__find_policy(const char *name)
23 struct dm_cache_policy_type *t;
25 list_for_each_entry(t, &register_list, list)
26 if (!strcmp(t->name, name))
27 return t;
29 return NULL;
32 static struct dm_cache_policy_type *__get_policy_once(const char *name)
34 struct dm_cache_policy_type *t = __find_policy(name);
36 if (t && !try_module_get(t->owner)) {
37 DMWARN("couldn't get module %s", name);
38 t = ERR_PTR(-EINVAL);
41 return t;
44 static struct dm_cache_policy_type *get_policy_once(const char *name)
46 struct dm_cache_policy_type *t;
48 spin_lock(&register_lock);
49 t = __get_policy_once(name);
50 spin_unlock(&register_lock);
52 return t;
55 static struct dm_cache_policy_type *get_policy(const char *name)
57 struct dm_cache_policy_type *t;
59 t = get_policy_once(name);
60 if (IS_ERR(t))
61 return NULL;
63 if (t)
64 return t;
66 request_module("dm-cache-%s", name);
68 t = get_policy_once(name);
69 if (IS_ERR(t))
70 return NULL;
72 return t;
75 static void put_policy(struct dm_cache_policy_type *t)
77 module_put(t->owner);
80 int dm_cache_policy_register(struct dm_cache_policy_type *type)
82 int r;
84 /* One size fits all for now */
85 if (type->hint_size != 0 && type->hint_size != 4) {
86 DMWARN("hint size must be 0 or 4 but %llu supplied.", (unsigned long long) type->hint_size);
87 return -EINVAL;
90 spin_lock(&register_lock);
91 if (__find_policy(type->name)) {
92 DMWARN("attempt to register policy under duplicate name %s", type->name);
93 r = -EINVAL;
94 } else {
95 list_add(&type->list, &register_list);
96 r = 0;
98 spin_unlock(&register_lock);
100 return r;
102 EXPORT_SYMBOL_GPL(dm_cache_policy_register);
104 void dm_cache_policy_unregister(struct dm_cache_policy_type *type)
106 spin_lock(&register_lock);
107 list_del_init(&type->list);
108 spin_unlock(&register_lock);
110 EXPORT_SYMBOL_GPL(dm_cache_policy_unregister);
112 struct dm_cache_policy *dm_cache_policy_create(const char *name,
113 dm_cblock_t cache_size,
114 sector_t origin_size,
115 sector_t cache_block_size)
117 struct dm_cache_policy *p = NULL;
118 struct dm_cache_policy_type *type;
120 type = get_policy(name);
121 if (!type) {
122 DMWARN("unknown policy type");
123 return ERR_PTR(-EINVAL);
126 p = type->create(cache_size, origin_size, cache_block_size);
127 if (!p) {
128 put_policy(type);
129 return ERR_PTR(-ENOMEM);
131 p->private = type;
133 return p;
135 EXPORT_SYMBOL_GPL(dm_cache_policy_create);
137 void dm_cache_policy_destroy(struct dm_cache_policy *p)
139 struct dm_cache_policy_type *t = p->private;
141 p->destroy(p);
142 put_policy(t);
144 EXPORT_SYMBOL_GPL(dm_cache_policy_destroy);
146 const char *dm_cache_policy_get_name(struct dm_cache_policy *p)
148 struct dm_cache_policy_type *t = p->private;
150 /* if t->real is set then an alias was used (e.g. "default") */
151 if (t->real)
152 return t->real->name;
154 return t->name;
156 EXPORT_SYMBOL_GPL(dm_cache_policy_get_name);
158 const unsigned int *dm_cache_policy_get_version(struct dm_cache_policy *p)
160 struct dm_cache_policy_type *t = p->private;
162 return t->version;
164 EXPORT_SYMBOL_GPL(dm_cache_policy_get_version);
166 size_t dm_cache_policy_get_hint_size(struct dm_cache_policy *p)
168 struct dm_cache_policy_type *t = p->private;
170 return t->hint_size;
172 EXPORT_SYMBOL_GPL(dm_cache_policy_get_hint_size);
174 /*----------------------------------------------------------------*/