123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216 |
- /*
- * Copyright (C) 2001-2002 Sistina Software (UK) Limited.
- * Copyright (C) 2006-2008 Red Hat GmbH
- *
- * This file is released under the GPL.
- */
- #include "dm-exception-store.h"
- #include <linux/mm.h>
- #include <linux/pagemap.h>
- #include <linux/vmalloc.h>
- #include <linux/slab.h>
- #define DM_MSG_PREFIX "snapshot exception stores"
- static LIST_HEAD(_exception_store_types);
- static DEFINE_SPINLOCK(_lock);
- static struct dm_exception_store_type *__find_exception_store_type(const char *name)
- {
- struct dm_exception_store_type *type;
- list_for_each_entry(type, &_exception_store_types, list)
- if (!strcmp(name, type->name))
- return type;
- return NULL;
- }
- static struct dm_exception_store_type *_get_exception_store_type(const char *name)
- {
- struct dm_exception_store_type *type;
- spin_lock(&_lock);
- type = __find_exception_store_type(name);
- if (type && !try_module_get(type->module))
- type = NULL;
- spin_unlock(&_lock);
- return type;
- }
- /*
- * get_type
- * @type_name
- *
- * Attempt to retrieve the dm_exception_store_type by name. If not already
- * available, attempt to load the appropriate module.
- *
- * Exstore modules are named "dm-exstore-" followed by the 'type_name'.
- * Modules may contain multiple types.
- * This function will first try the module "dm-exstore-<type_name>",
- * then truncate 'type_name' on the last '-' and try again.
- *
- * For example, if type_name was "clustered-shared", it would search
- * 'dm-exstore-clustered-shared' then 'dm-exstore-clustered'.
- *
- * 'dm-exception-store-<type_name>' is too long of a name in my
- * opinion, which is why I've chosen to have the files
- * containing exception store implementations be 'dm-exstore-<type_name>'.
- * If you want your module to be autoloaded, you will follow this
- * naming convention.
- *
- * Returns: dm_exception_store_type* on success, NULL on failure
- */
- static struct dm_exception_store_type *get_type(const char *type_name)
- {
- char *p, *type_name_dup;
- struct dm_exception_store_type *type;
- type = _get_exception_store_type(type_name);
- if (type)
- return type;
- type_name_dup = kstrdup(type_name, GFP_KERNEL);
- if (!type_name_dup) {
- DMERR("No memory left to attempt load for \"%s\"", type_name);
- return NULL;
- }
- while (request_module("dm-exstore-%s", type_name_dup) ||
- !(type = _get_exception_store_type(type_name))) {
- p = strrchr(type_name_dup, '-');
- if (!p)
- break;
- p[0] = '\0';
- }
- if (!type)
- DMWARN("Module for exstore type \"%s\" not found.", type_name);
- kfree(type_name_dup);
- return type;
- }
- static void put_type(struct dm_exception_store_type *type)
- {
- spin_lock(&_lock);
- module_put(type->module);
- spin_unlock(&_lock);
- }
- int dm_exception_store_type_register(struct dm_exception_store_type *type)
- {
- int r = 0;
- spin_lock(&_lock);
- if (!__find_exception_store_type(type->name))
- list_add(&type->list, &_exception_store_types);
- else
- r = -EEXIST;
- spin_unlock(&_lock);
- return r;
- }
- EXPORT_SYMBOL(dm_exception_store_type_register);
- int dm_exception_store_type_unregister(struct dm_exception_store_type *type)
- {
- spin_lock(&_lock);
- if (!__find_exception_store_type(type->name)) {
- spin_unlock(&_lock);
- return -EINVAL;
- }
- list_del(&type->list);
- spin_unlock(&_lock);
- return 0;
- }
- EXPORT_SYMBOL(dm_exception_store_type_unregister);
- int dm_exception_store_create(const char *type_name, struct dm_target *ti,
- chunk_t chunk_size, chunk_t chunk_mask,
- chunk_t chunk_shift, struct dm_dev *cow,
- struct dm_exception_store **store)
- {
- int r = 0;
- struct dm_exception_store_type *type;
- struct dm_exception_store *tmp_store;
- tmp_store = kmalloc(sizeof(*tmp_store), GFP_KERNEL);
- if (!tmp_store)
- return -ENOMEM;
- type = get_type(type_name);
- if (!type) {
- kfree(tmp_store);
- return -EINVAL;
- }
- tmp_store->type = type;
- tmp_store->ti = ti;
- tmp_store->chunk_size = chunk_size;
- tmp_store->chunk_mask = chunk_mask;
- tmp_store->chunk_shift = chunk_shift;
- tmp_store->cow = cow;
- r = type->ctr(tmp_store, 0, NULL);
- if (r) {
- put_type(type);
- kfree(tmp_store);
- return r;
- }
- *store = tmp_store;
- return 0;
- }
- EXPORT_SYMBOL(dm_exception_store_create);
- void dm_exception_store_destroy(struct dm_exception_store *store)
- {
- store->type->dtr(store);
- put_type(store->type);
- kfree(store);
- }
- EXPORT_SYMBOL(dm_exception_store_destroy);
- int dm_exception_store_init(void)
- {
- int r;
- r = dm_transient_snapshot_init();
- if (r) {
- DMERR("Unable to register transient exception store type.");
- goto transient_fail;
- }
- r = dm_persistent_snapshot_init();
- if (r) {
- DMERR("Unable to register persistent exception store type");
- goto persistent_fail;
- }
- return 0;
- persistent_fail:
- dm_persistent_snapshot_exit();
- transient_fail:
- return r;
- }
- void dm_exception_store_exit(void)
- {
- dm_persistent_snapshot_exit();
- dm_transient_snapshot_exit();
- }
|