forked from mirrors/linux
		
	 86a3238c7b
			
		
	
	
		86a3238c7b
		
	
	
	
	
		
			
			Signed-off-by: Heinz Mauelshagen <heinzm@redhat.com> Signed-off-by: Mike Snitzer <snitzer@kernel.org>
		
			
				
	
	
		
			174 lines
		
	
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			174 lines
		
	
	
	
		
			3.7 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // SPDX-License-Identifier: GPL-2.0-only
 | |
| /*
 | |
|  * 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 int *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);
 | |
| 
 | |
| /*----------------------------------------------------------------*/
 |