/* netfs cookie management * * Copyright (C) 2004-2007 Red Hat, Inc. All Rights Reserved. * Written by David Howells (dhowells@redhat.com) * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version * 2 of the License, or (at your option) any later version. * * See Documentation/filesystems/caching/netfs-api.txt for more information on * the netfs API. */ #define FSCACHE_DEBUG_LEVEL COOKIE #include <linux/module.h> #include <linux/slab.h> #include "internal.h" struct kmem_cache *fscache_cookie_jar; static atomic_t fscache_object_debug_id = ATOMIC_INIT(0); static int fscache_acquire_non_index_cookie(struct fscache_cookie *cookie); static int fscache_alloc_object(struct fscache_cache *cache, struct fscache_cookie *cookie); static int fscache_attach_object(struct fscache_cookie *cookie, struct fscache_object *object); /* * initialise an cookie jar slab element prior to any use */ void fscache_cookie_init_once(void *_cookie) { struct fscache_cookie *cookie = _cookie; memset(cookie, 0, sizeof(*cookie)); spin_lock_init(&cookie->lock); INIT_HLIST_HEAD(&cookie->backing_objects); } /* * request a cookie to represent an object (index, datafile, xattr, etc) * - parent specifies the parent object * - the top level index cookie for each netfs is stored in the fscache_netfs * struct upon registration * - def points to the definition * - the netfs_data will be passed to the functions pointed to in *def * - all attached caches will be searched to see if they contain this object * - index objects aren't stored on disk until there's a dependent file that * needs storing * - other objects are stored in a selected cache immediately, and all the * indices forming the path to it are instantiated if necessary * - we never let on to the netfs about errors * - we may set a negative cookie pointer, but that's okay */ struct fscache_cookie *__fscache_acquire_cookie( struct fscache_cookie *parent, const struct fscache_cookie_def *def, void *netfs_data) { struct fscache_cookie *cookie; BUG_ON(!def); _enter("{%s},{%s},%p", parent ? (char *) parent->def->name : "<no-parent>", def->name, netfs_data); fscache_stat(&fscache_n_acquires); /* if there's no parent cookie, then we don't create one here either */ if (!parent) { fscache_stat(&fscache_n_acquires_null); _leave(" [no parent]"); return NULL; } /* validate the definition */ BUG_ON(!def->get_key); BUG_ON(!def->name[0]); BUG_ON(def->type == FSCACHE_COOKIE_TYPE_INDEX && parent->def->type != FSCACHE_COOKIE_TYPE_INDEX); /* allocate and initialise a cookie */ cookie = kmem_cache_alloc(fscache_cookie_jar, GFP_KERNEL); if (!cookie) { fscache_stat(&fscache_n_acquires_oom); _leave(" [ENOMEM]"); return NULL; } atomic_set(&cookie->usage, 1); atomic_set(&cookie->n_children, 0); atomic_inc(&parent->usage); atomic_inc(&parent->n_children); cookie->def = def; cookie->parent = parent; cookie->netfs_data = netfs_data; cookie->flags = 0; INIT_RADIX_TREE(&cookie->stores, GFP_NOFS); switch (cookie->def->type) { case FSCACHE_COOKIE_TYPE_INDEX: fscache_stat(&fscache_n_cookie_index); break; case FSCACHE_COOKIE_TYPE_DATAFILE: fscache_stat(&fscache_n_cookie_data); break; default: fscache_stat(&fscache_n_cookie_special); break; } /* if the object is an index then we need do nothing more here - we * create indices on disk when we need them as an index may exist in * multiple caches */ if (cookie->def->type != FSCACHE_COOKIE_TYPE_INDEX) { if (fscache_acquire_non_index_cookie(cookie) < 0) { atomic_dec(&parent->n_children); __fscache_cookie_put(cookie); fscache_stat(&fscache_n_acquires_nobufs); _leave(" = NULL"); return NULL; } } fscache_stat(&fscache_n_acquires_ok); _leave(" = %p", cookie); return cookie; } EXPORT_SYMBOL(__fscache_acquire_cookie); /* * acquire a non-index cookie * - this must make sure the index chain is instantiated and instantiate the * object representation too */ static int fscache_acquire_non_index_cookie(struct fscache_cookie *cookie) { struct fscache_object *object; struct fscache_cache *cache; uint64_t i_size; int ret; _enter(""); cookie->flags = 1 << FSCACHE_COOKIE_UNAVAILABLE; /* now we need to see whether the backing objects for this cookie yet * exist, if not there'll be nothing to search */ down_read(&fscache_addremove_sem); if (list_empty(&fscache_cache_list)) { up_read(&fscache_addremove_sem); _leave(" = 0 [no caches]"); return 0; } /* select a cache in which to store the object */ cache = fscache_select_cache_for_object(cookie->parent); if (!cache) { up_read(&fscache_addremove_sem); fscache_stat(&fscache_n_acquires_no_cache); _leave(" = -ENOMEDIUM [no cache]"); return -ENOMEDIUM; } _debug("cache %s", cache->tag->name); cookie->flags = (1 << FSCACHE_COOKIE_LOOKING_UP) | (1 << FSCACHE_COOKIE_CREATING) | (1 << FSCACHE_COOKIE_NO_DATA_YET); /* ask the cache to allocate objects for this cookie and its parent * chain */ ret = fscache_alloc_object(cache, cookie); if (ret < 0) { up_read(&fscache_addremove_sem); _leave(" = %d", ret); return ret; } /* pass on how big the object we're caching is supposed to be */ cookie->def->get_attr(cookie->netfs_data, &i_size); spin_lock(&cookie->lock); if (hlist_empty(&cookie->backing_objects)) { spin_unlock(&cookie->lock); goto unavailable; } object = hlist_entry(cookie->backing_objects.first, struct fscache_object, cookie_link); fscache_set_store_limit(object, i_size); /* initiate the process of looking up all the objects in the chain * (done by fscache_initialise_object()) */ fscache_enqueue_object(object); spin_unlock(&cookie->lock); /* we may be required to wait for lookup to complete at this point */ if (!fscache_defer_lookup) { _debug("non-deferred lookup %p", &cookie->flags); wait_on_bit(&cookie->flags, FSCACHE_COOKIE_LOOKING_UP, fscache_wait_bit, TASK_UNINTERRUPTIBLE); _debug("complete"); if (test_bit(FSCACHE_COOKIE_UNAVAILABLE, &cookie->flags)) goto unavailable; } up_read(&fscache_addremove_sem); _leave(" = 0 [deferred]"); return 0; unavailable: up_read(&fscache_addremove_sem); _leave(" = -ENOBUFS"); return -ENOBUFS; } /* * recursively allocate cache object records for a cookie/cache combination * - caller must be holding the addremove sem */ static int fscache_alloc_object(struct fscache_cache *cache, struct fscache_cookie *cookie) { struct fscache_object *object; struct hlist_node *_n; int ret; _enter("%p,%p{%s}", cache, cookie, cookie->def->name); spin_lock(&cookie->lock); hlist_for_each_entry(object, _n, &cookie->backing_objects, cookie_link) { if (object->cache == cache) goto object_already_extant; } spin_unlock(&cookie->lock); /* ask the cache to allocate an object (we may end up with duplicate * objects at this stage, but we sort that out later) */ object = cache->ops->alloc_object(cache, cookie); if (IS_ERR(object)) { fscache_stat(&fscache_n_object_no_alloc); ret = PTR_ERR(object); goto error; } fscache_stat(&fscache_n_object_alloc); object->debug_id = atomic_inc_return(&fscache_object_debug_id); _debug("ALLOC OBJ%x: %s {%lx}", object->debug_id, cookie->def->name, object->events); ret = fscache_alloc_object(cache, cookie->parent); if (ret < 0) goto error_put; /* only attach if we managed to allocate all we needed, otherwise * discard the object we just allocated and instead use the one * attached to the cookie */ if (fscache_attach_object(cookie, object) < 0) cache->ops->put_object(object); _leave(" = 0"); return 0; object_already_extant: ret = -ENOBUFS; if (object->state >= FSCACHE_OBJECT_DYING) { spin_unlock(&cookie->lock); goto error; } spin_unlock(&cookie->lock); _leave(" = 0 [found]"); return 0; error_put: cache->ops->put_object(object); error: _leave(" = %d", ret); return ret; } /* * attach a cache object to a cookie */ static int fscache_attach_object(struct fscache_cookie *cookie, struct fscache_object *object) { struct fscache_object *p; struct fscache_cache *cache = object->cache; struct hlist_node *_n; int ret; _enter("{%s},{OBJ%x}", cookie->def->name, object->debug_id); spin_lock(&cookie->lock); /* there may be multiple initial creations of this object, but we only * want one */ ret = -EEXIST; hlist_for_each_entry(p, _n, &cookie->backing_objects, cookie_link) { if (p->cache == object->cache) { if (p->state >= FSCACHE_OBJECT_DYING) ret = -ENOBUFS; goto cant_attach_object; } } /* pin the parent object */ spin_lock_nested(&cookie->parent->lock, 1); hlist_for_each_entry(p, _n, &cookie->parent->backing_objects, cookie_link) { if (p->cache == object->cache) { if (p->state >= FSCACHE_OBJECT_DYING) { ret = -ENOBUFS; spin_unlock(&cookie->parent->lock); goto cant_attach_object; } object->parent = p; spin_lock(&p->lock); p->n_children++; spin_unlock(&p->lock); break; } } spin_unlock(&cookie->parent->lock); /* attach to the cache's object list */ if (list_empty(&object->cache_link)) { spin_lock(&cache->object_list_lock); list_add(&object->cache_link, &cache->object_list); spin_unlock(&cache->object_list_lock); } /* attach to the cookie */ object->cookie = cookie; atomic_inc(&cookie->usage); hlist_add_head(&object->cookie_link, &cookie->backing_objects); ret = 0; cant_attach_object: spin_unlock(&cookie->lock); _leave(" = %d", ret); return ret; } /* * update the index entries backing a cookie */ void __fscache_update_cookie(struct fscache_cookie *cookie) { struct fscache_object *object; struct hlist_node *_p; fscache_stat(&fscache_n_updates); if (!cookie) { fscache_stat(&fscache_n_updates_null); _leave(" [no cookie]"); return; } _enter("{%s}", cookie->def->name); BUG_ON(!cookie->def->get_aux); spin_lock(&cookie->lock); /* update the index entry on disk in each cache backing this cookie */ hlist_for_each_entry(object, _p, &cookie->backing_objects, cookie_link) { fscache_raise_event(object, FSCACHE_OBJECT_EV_UPDATE); } spin_unlock(&cookie->lock); _leave(""); } EXPORT_SYMBOL(__fscache_update_cookie); /* * release a cookie back to the cache * - the object will be marked as recyclable on disk if retire is true * - all dependents of this cookie must have already been unregistered * (indices/files/pages) */ void __fscache_relinquish_cookie(struct fscache_cookie *cookie, int retire) { struct fscache_cache *cache; struct fscache_object *object; unsigned long event; fscache_stat(&fscache_n_relinquishes); if (!cookie) { fscache_stat(&fscache_n_relinquishes_null); _leave(" [no cookie]"); return; } _enter("%p{%s,%p},%d", cookie, cookie->def->name, cookie->netfs_data, retire); if (atomic_read(&cookie->n_children) != 0) { printk(KERN_ERR "FS-Cache: Cookie '%s' still has children\n", cookie->def->name); BUG(); } /* wait for the cookie to finish being instantiated (or to fail) */ if (test_bit(FSCACHE_COOKIE_CREATING, &cookie->flags)) { fscache_stat(&fscache_n_relinquishes_waitcrt); wait_on_bit(&cookie->flags, FSCACHE_COOKIE_CREATING, fscache_wait_bit, TASK_UNINTERRUPTIBLE); } event = retire ? FSCACHE_OBJECT_EV_RETIRE : FSCACHE_OBJECT_EV_RELEASE; /* detach pointers back to the netfs */ spin_lock(&cookie->lock); cookie->netfs_data = NULL; cookie->def = NULL; /* break links with all the active objects */ while (!hlist_empty(&cookie->backing_objects)) { object = hlist_entry(cookie->backing_objects.first, struct fscache_object, cookie_link); _debug("RELEASE OBJ%x", object->debug_id); /* detach each cache object from the object cookie */ spin_lock(&object->lock); hlist_del_init(&object->cookie_link); cache = object->cache; object->cookie = NULL; fscache_raise_event(object, event); spin_unlock(&object->lock); if (atomic_dec_and_test(&cookie->usage)) /* the cookie refcount shouldn't be reduced to 0 yet */ BUG(); } spin_unlock(&cookie->lock); if (cookie->parent) { ASSERTCMP(atomic_read(&cookie->parent->usage), >, 0); ASSERTCMP(atomic_read(&cookie->parent->n_children), >, 0); atomic_dec(&cookie->parent->n_children); } /* finally dispose of the cookie */ ASSERTCMP(atomic_read(&cookie->usage), >, 0); fscache_cookie_put(cookie); _leave(""); } EXPORT_SYMBOL(__fscache_relinquish_cookie); /* * destroy a cookie */ void __fscache_cookie_put(struct fscache_cookie *cookie) { struct fscache_cookie *parent; _enter("%p", cookie); for (;;) { _debug("FREE COOKIE %p", cookie); parent = cookie->parent; BUG_ON(!hlist_empty(&cookie->backing_objects)); kmem_cache_free(fscache_cookie_jar, cookie); if (!parent) break; cookie = parent; BUG_ON(atomic_read(&cookie->usage) <= 0); if (!atomic_dec_and_test(&cookie->usage)) break; } _leave(""); }