206 lines
6.7 KiB
C
206 lines
6.7 KiB
C
|
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
||
|
/* General filesystem caching backing cache interface
|
||
|
*
|
||
|
* Copyright (C) 2021 Red Hat, Inc. All Rights Reserved.
|
||
|
* Written by David Howells (dhowells@redhat.com)
|
||
|
*
|
||
|
* NOTE!!! See:
|
||
|
*
|
||
|
* Documentation/filesystems/caching/backend-api.rst
|
||
|
*
|
||
|
* for a description of the cache backend interface declared here.
|
||
|
*/
|
||
|
|
||
|
#ifndef _LINUX_FSCACHE_CACHE_H
|
||
|
#define _LINUX_FSCACHE_CACHE_H
|
||
|
|
||
|
#include <linux/fscache.h>
|
||
|
|
||
|
enum fscache_cache_trace;
|
||
|
enum fscache_cookie_trace;
|
||
|
enum fscache_access_trace;
|
||
|
|
||
|
enum fscache_cache_state {
|
||
|
FSCACHE_CACHE_IS_NOT_PRESENT, /* No cache is present for this name */
|
||
|
FSCACHE_CACHE_IS_PREPARING, /* A cache is preparing to come live */
|
||
|
FSCACHE_CACHE_IS_ACTIVE, /* Attached cache is active and can be used */
|
||
|
FSCACHE_CACHE_GOT_IOERROR, /* Attached cache stopped on I/O error */
|
||
|
FSCACHE_CACHE_IS_WITHDRAWN, /* Attached cache is being withdrawn */
|
||
|
#define NR__FSCACHE_CACHE_STATE (FSCACHE_CACHE_IS_WITHDRAWN + 1)
|
||
|
};
|
||
|
|
||
|
/*
|
||
|
* Cache cookie.
|
||
|
*/
|
||
|
struct fscache_cache {
|
||
|
const struct fscache_cache_ops *ops;
|
||
|
struct list_head cache_link; /* Link in cache list */
|
||
|
void *cache_priv; /* Private cache data (or NULL) */
|
||
|
refcount_t ref;
|
||
|
atomic_t n_volumes; /* Number of active volumes; */
|
||
|
atomic_t n_accesses; /* Number of in-progress accesses on the cache */
|
||
|
atomic_t object_count; /* no. of live objects in this cache */
|
||
|
unsigned int debug_id;
|
||
|
enum fscache_cache_state state;
|
||
|
char *name;
|
||
|
};
|
||
|
|
||
|
/*
|
||
|
* cache operations
|
||
|
*/
|
||
|
struct fscache_cache_ops {
|
||
|
/* name of cache provider */
|
||
|
const char *name;
|
||
|
|
||
|
/* Acquire a volume */
|
||
|
void (*acquire_volume)(struct fscache_volume *volume);
|
||
|
|
||
|
/* Free the cache's data attached to a volume */
|
||
|
void (*free_volume)(struct fscache_volume *volume);
|
||
|
|
||
|
/* Look up a cookie in the cache */
|
||
|
bool (*lookup_cookie)(struct fscache_cookie *cookie);
|
||
|
|
||
|
/* Withdraw an object without any cookie access counts held */
|
||
|
void (*withdraw_cookie)(struct fscache_cookie *cookie);
|
||
|
|
||
|
/* Change the size of a data object */
|
||
|
void (*resize_cookie)(struct netfs_cache_resources *cres,
|
||
|
loff_t new_size);
|
||
|
|
||
|
/* Invalidate an object */
|
||
|
bool (*invalidate_cookie)(struct fscache_cookie *cookie);
|
||
|
|
||
|
/* Begin an operation for the netfs lib */
|
||
|
bool (*begin_operation)(struct netfs_cache_resources *cres,
|
||
|
enum fscache_want_state want_state);
|
||
|
|
||
|
/* Prepare to write to a live cache object */
|
||
|
void (*prepare_to_write)(struct fscache_cookie *cookie);
|
||
|
};
|
||
|
|
||
|
extern struct workqueue_struct *fscache_wq;
|
||
|
extern wait_queue_head_t fscache_clearance_waiters;
|
||
|
|
||
|
/*
|
||
|
* out-of-line cache backend functions
|
||
|
*/
|
||
|
extern struct rw_semaphore fscache_addremove_sem;
|
||
|
extern struct fscache_cache *fscache_acquire_cache(const char *name);
|
||
|
extern void fscache_relinquish_cache(struct fscache_cache *cache);
|
||
|
extern int fscache_add_cache(struct fscache_cache *cache,
|
||
|
const struct fscache_cache_ops *ops,
|
||
|
void *cache_priv);
|
||
|
extern void fscache_withdraw_cache(struct fscache_cache *cache);
|
||
|
extern void fscache_withdraw_volume(struct fscache_volume *volume);
|
||
|
extern void fscache_withdraw_cookie(struct fscache_cookie *cookie);
|
||
|
|
||
|
extern void fscache_io_error(struct fscache_cache *cache);
|
||
|
|
||
|
extern void fscache_end_volume_access(struct fscache_volume *volume,
|
||
|
struct fscache_cookie *cookie,
|
||
|
enum fscache_access_trace why);
|
||
|
|
||
|
extern struct fscache_cookie *fscache_get_cookie(struct fscache_cookie *cookie,
|
||
|
enum fscache_cookie_trace where);
|
||
|
extern void fscache_put_cookie(struct fscache_cookie *cookie,
|
||
|
enum fscache_cookie_trace where);
|
||
|
extern void fscache_end_cookie_access(struct fscache_cookie *cookie,
|
||
|
enum fscache_access_trace why);
|
||
|
extern void fscache_cookie_lookup_negative(struct fscache_cookie *cookie);
|
||
|
extern void fscache_resume_after_invalidation(struct fscache_cookie *cookie);
|
||
|
extern void fscache_caching_failed(struct fscache_cookie *cookie);
|
||
|
extern bool fscache_wait_for_operation(struct netfs_cache_resources *cred,
|
||
|
enum fscache_want_state state);
|
||
|
|
||
|
/**
|
||
|
* fscache_cookie_state - Read the state of a cookie
|
||
|
* @cookie: The cookie to query
|
||
|
*
|
||
|
* Get the state of a cookie, imposing an ordering between the cookie contents
|
||
|
* and the state value. Paired with fscache_set_cookie_state().
|
||
|
*/
|
||
|
static inline
|
||
|
enum fscache_cookie_state fscache_cookie_state(struct fscache_cookie *cookie)
|
||
|
{
|
||
|
return smp_load_acquire(&cookie->state);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* fscache_get_key - Get a pointer to the cookie key
|
||
|
* @cookie: The cookie to query
|
||
|
*
|
||
|
* Return a pointer to the where a cookie's key is stored.
|
||
|
*/
|
||
|
static inline void *fscache_get_key(struct fscache_cookie *cookie)
|
||
|
{
|
||
|
if (cookie->key_len <= sizeof(cookie->inline_key))
|
||
|
return cookie->inline_key;
|
||
|
else
|
||
|
return cookie->key;
|
||
|
}
|
||
|
|
||
|
static inline struct fscache_cookie *fscache_cres_cookie(struct netfs_cache_resources *cres)
|
||
|
{
|
||
|
return cres->cache_priv;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* fscache_count_object - Tell fscache that an object has been added
|
||
|
* @cache: The cache to account to
|
||
|
*
|
||
|
* Tell fscache that an object has been added to the cache. This prevents the
|
||
|
* cache from tearing down the cache structure until the object is uncounted.
|
||
|
*/
|
||
|
static inline void fscache_count_object(struct fscache_cache *cache)
|
||
|
{
|
||
|
atomic_inc(&cache->object_count);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* fscache_uncount_object - Tell fscache that an object has been removed
|
||
|
* @cache: The cache to account to
|
||
|
*
|
||
|
* Tell fscache that an object has been removed from the cache and will no
|
||
|
* longer be accessed. After this point, the cache cookie may be destroyed.
|
||
|
*/
|
||
|
static inline void fscache_uncount_object(struct fscache_cache *cache)
|
||
|
{
|
||
|
if (atomic_dec_and_test(&cache->object_count))
|
||
|
wake_up_all(&fscache_clearance_waiters);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* fscache_wait_for_objects - Wait for all objects to be withdrawn
|
||
|
* @cache: The cache to query
|
||
|
*
|
||
|
* Wait for all extant objects in a cache to finish being withdrawn
|
||
|
* and go away.
|
||
|
*/
|
||
|
static inline void fscache_wait_for_objects(struct fscache_cache *cache)
|
||
|
{
|
||
|
wait_event(fscache_clearance_waiters,
|
||
|
atomic_read(&cache->object_count) == 0);
|
||
|
}
|
||
|
|
||
|
#ifdef CONFIG_FSCACHE_STATS
|
||
|
extern atomic_t fscache_n_read;
|
||
|
extern atomic_t fscache_n_write;
|
||
|
extern atomic_t fscache_n_no_write_space;
|
||
|
extern atomic_t fscache_n_no_create_space;
|
||
|
extern atomic_t fscache_n_culled;
|
||
|
#define fscache_count_read() atomic_inc(&fscache_n_read)
|
||
|
#define fscache_count_write() atomic_inc(&fscache_n_write)
|
||
|
#define fscache_count_no_write_space() atomic_inc(&fscache_n_no_write_space)
|
||
|
#define fscache_count_no_create_space() atomic_inc(&fscache_n_no_create_space)
|
||
|
#define fscache_count_culled() atomic_inc(&fscache_n_culled)
|
||
|
#else
|
||
|
#define fscache_count_read() do {} while(0)
|
||
|
#define fscache_count_write() do {} while(0)
|
||
|
#define fscache_count_no_write_space() do {} while(0)
|
||
|
#define fscache_count_no_create_space() do {} while(0)
|
||
|
#define fscache_count_culled() do {} while(0)
|
||
|
#endif
|
||
|
|
||
|
#endif /* _LINUX_FSCACHE_CACHE_H */
|