2e68c4e6ca
The cache's policy may have been established using the "default" alias, which is currently the "mq" policy but the default policy may change in the future. It is useful to know exactly which policy is being used. Add a 'real' member to the dm_cache_policy_type structure and have the "default" dm_cache_policy_type point to the real "mq" dm_cache_policy_type. Update dm_cache_policy_get_name() to check if real is set, if so report the name of the real policy (not the alias). Requested-by: Jonathan Brassow <jbrassow@redhat.com> Signed-off-by: Mike Snitzer <snitzer@redhat.com>
174 lines
3.6 KiB
C
174 lines
3.6 KiB
C
/*
|
|
* 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);
|
|
|
|
/*----------------------------------------------------------------*/
|