123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173 |
- /*
- * Copyright (C) 2012 Red Hat. All rights reserved.
- *
- * This file is released under the GPL.
- */
- #include "dm-cache-policy-internal.h"
- #include "dm.h"
- #include <linux/module.h>
- #include <linux/slab.h>
- /*----------------------------------------------------------------*/
- #define DM_MSG_PREFIX "cache-policy"
- static DEFINE_SPINLOCK(register_lock);
- static LIST_HEAD(register_list);
- static struct dm_cache_policy_type *__find_policy(const char *name)
- {
- struct dm_cache_policy_type *t;
- list_for_each_entry(t, ®ister_list, list)
- if (!strcmp(t->name, name))
- return t;
- return NULL;
- }
- static struct dm_cache_policy_type *__get_policy_once(const char *name)
- {
- struct dm_cache_policy_type *t = __find_policy(name);
- if (t && !try_module_get(t->owner)) {
- DMWARN("couldn't get module %s", name);
- t = ERR_PTR(-EINVAL);
- }
- return t;
- }
- static struct dm_cache_policy_type *get_policy_once(const char *name)
- {
- struct dm_cache_policy_type *t;
- spin_lock(®ister_lock);
- t = __get_policy_once(name);
- spin_unlock(®ister_lock);
- return t;
- }
- static struct dm_cache_policy_type *get_policy(const char *name)
- {
- struct dm_cache_policy_type *t;
- t = get_policy_once(name);
- if (IS_ERR(t))
- return NULL;
- if (t)
- return t;
- request_module("dm-cache-%s", name);
- t = get_policy_once(name);
- if (IS_ERR(t))
- return NULL;
- return t;
- }
- static void put_policy(struct dm_cache_policy_type *t)
- {
- module_put(t->owner);
- }
- int dm_cache_policy_register(struct dm_cache_policy_type *type)
- {
- int r;
- /* One size fits all for now */
- if (type->hint_size != 0 && type->hint_size != 4) {
- DMWARN("hint size must be 0 or 4 but %llu supplied.", (unsigned long long) type->hint_size);
- return -EINVAL;
- }
- spin_lock(®ister_lock);
- if (__find_policy(type->name)) {
- DMWARN("attempt to register policy under duplicate name %s", type->name);
- r = -EINVAL;
- } else {
- list_add(&type->list, ®ister_list);
- r = 0;
- }
- spin_unlock(®ister_lock);
- return r;
- }
- EXPORT_SYMBOL_GPL(dm_cache_policy_register);
- void dm_cache_policy_unregister(struct dm_cache_policy_type *type)
- {
- spin_lock(®ister_lock);
- list_del_init(&type->list);
- spin_unlock(®ister_lock);
- }
- EXPORT_SYMBOL_GPL(dm_cache_policy_unregister);
- struct dm_cache_policy *dm_cache_policy_create(const char *name,
- dm_cblock_t cache_size,
- sector_t origin_size,
- sector_t cache_block_size)
- {
- struct dm_cache_policy *p = NULL;
- struct dm_cache_policy_type *type;
- type = get_policy(name);
- if (!type) {
- DMWARN("unknown policy type");
- return ERR_PTR(-EINVAL);
- }
- p = type->create(cache_size, origin_size, cache_block_size);
- if (!p) {
- put_policy(type);
- return ERR_PTR(-ENOMEM);
- }
- p->private = type;
- return p;
- }
- EXPORT_SYMBOL_GPL(dm_cache_policy_create);
- void dm_cache_policy_destroy(struct dm_cache_policy *p)
- {
- struct dm_cache_policy_type *t = p->private;
- p->destroy(p);
- put_policy(t);
- }
- EXPORT_SYMBOL_GPL(dm_cache_policy_destroy);
- const char *dm_cache_policy_get_name(struct dm_cache_policy *p)
- {
- struct dm_cache_policy_type *t = p->private;
- /* if t->real is set then an alias was used (e.g. "default") */
- if (t->real)
- return t->real->name;
- return t->name;
- }
- EXPORT_SYMBOL_GPL(dm_cache_policy_get_name);
- const unsigned *dm_cache_policy_get_version(struct dm_cache_policy *p)
- {
- struct dm_cache_policy_type *t = p->private;
- return t->version;
- }
- EXPORT_SYMBOL_GPL(dm_cache_policy_get_version);
- size_t dm_cache_policy_get_hint_size(struct dm_cache_policy *p)
- {
- struct dm_cache_policy_type *t = p->private;
- return t->hint_size;
- }
- EXPORT_SYMBOL_GPL(dm_cache_policy_get_hint_size);
- /*----------------------------------------------------------------*/
|