size_t split;
};
+#define container_of(ptr, type, member) ({ \
+ const typeof( ((type *)0)->member ) *__mptr = (ptr); \
+ (type *)( (char *)__mptr - offsetof(type,member) );})
+
+#define list_entry(ptr, type, member) \
+ container_of(ptr, type, member)
+
struct list_head {
struct list_head *next;
struct list_head *prev;
struct fuse_session *se;
struct node_table name_table;
struct node_table id_table;
+ struct list_head lru_table;
fuse_ino_t ctr;
unsigned int generation;
unsigned int hidectr;
int open_count;
struct timespec stat_updated;
struct timespec mtime;
- struct timespec forget_time;
off_t size;
struct lock *locks;
unsigned int is_hidden : 1;
char inline_name[32];
};
+struct node_lru {
+ struct node node;
+ struct list_head lru;
+ struct timespec forget_time;
+};
+
struct fuse_dh {
pthread_mutex_t lock;
struct fuse *fuse;
return head->next == head;
}
-#ifdef FUSE_NODE_SLAB
static void list_add(struct list_head *new, struct list_head *prev,
struct list_head *next)
{
prev->next = next;
}
+#ifdef FUSE_NODE_SLAB
static struct node_slab *list_to_slab(struct list_head *head)
{
return (struct node_slab *) head;
return (struct node_slab *) (((uintptr_t) node) & ~((uintptr_t) f->pagesize - 1));
}
+static inline int lru_enabled(struct fuse *f)
+{
+ return f->conf.remember > 0;
+}
+
+static struct node_lru *node_lru(struct node *node)
+{
+ return (struct node_lru *) node;
+}
+
+static size_t get_node_size(struct fuse *f)
+{
+ if (lru_enabled(f))
+ return sizeof(struct node_lru);
+ else
+ return sizeof(struct node);
+}
+
static int alloc_slab(struct fuse *f)
{
void *mem;
char *start;
size_t num;
size_t i;
+ size_t node_size = get_node_size(f);
mem = mmap(NULL, f->pagesize, PROT_READ | PROT_WRITE,
MAP_PRIVATE | MAP_ANONYMOUS, -1, 0);
slab = mem;
init_list_head(&slab->freelist);
slab->used = 0;
- num = (f->pagesize - sizeof(struct node_slab)) / sizeof(struct node);
+ num = (f->pagesize - sizeof(struct node_slab)) / node_size;
- start = (char *) mem + f->pagesize - num * sizeof(struct node);
+ start = (char *) mem + f->pagesize - num * node_size;
for (i = 0; i < num; i++) {
struct list_head *n;
- n = (struct list_head *) (start + i * sizeof(struct node));
+ n = (struct list_head *) (start + i * node_size);
list_add_tail(n, &slab->freelist);
}
list_add_tail(&slab->list, &f->partial_slabs);
#else
static struct node *alloc_node(struct fuse *f)
{
- (void) f;
- return (struct node *) calloc(1, sizeof(struct node));
+ return (struct node *) calloc(1, get_node_size(f));
}
static void free_node_mem(struct fuse *f, struct node *node)
static double diff_timespec(const struct timespec *t1,
const struct timespec *t2);
+static void remove_node_lru(struct node *node)
+{
+ struct node_lru *lnode = node_lru(node);
+ list_del(&lnode->lru);
+ init_list_head(&lnode->lru);
+}
+
+static void set_forget_time(struct fuse *f, struct node *node)
+{
+ struct node_lru *lnode = node_lru(node);
+
+ list_del(&lnode->lru);
+ list_add_tail(&lnode->lru, &f->lru_table);
+ curr_time(&lnode->forget_time);
+}
+
static void free_node(struct fuse *f, struct node *node)
{
if (node->name != node->inline_name)
assert(node->treelock == 0);
assert(!node->name);
+ if (lru_enabled(f))
+ remove_node_lru(node);
unhash_id(f, node);
free_node(f, node);
}
goto out_err;
}
hash_id(f, node);
+ if (lru_enabled(f)) {
+ struct node_lru *lnode = node_lru(node);
+ init_list_head(&lnode->lru);
+ }
+ } else if (lru_enabled(f) && node->nlookup == 1) {
+ remove_node_lru(node);
}
node->nlookup ++;
out_err:
if (!node->nlookup) {
unhash_name(f, node);
unref_node(f, node);
- } else if (node->nlookup == 1 && f->conf.remember &&
- f->conf.remember != -1) {
- curr_time(&node->forget_time);
+ } else if (lru_enabled(f) && node->nlookup == 1) {
+ set_forget_time(f, node);
}
pthread_mutex_unlock(&f->lock);
}
int fuse_clean_cache(struct fuse *f)
{
- int i;
- struct node *node, *next;
+ struct node_lru *lnode;
+ struct list_head *curr, *next;
+ struct node *node;
struct timespec now;
- static int next_clean;
pthread_mutex_lock(&f->lock);
- next_clean = clean_delay(f);
curr_time(&now);
- for (i = 0; i < f->name_table.size; ++i) {
- for (node = f->name_table.array[i]; node; node = next) {
- double age;
- next = node->name_next;
+ for (curr = f->lru_table.next; curr != &f->lru_table; curr = next) {
+ double age;
- if (node->nodeid == FUSE_ROOT_ID)
- continue;
+ next = curr->next;
+ lnode = list_entry(curr, struct node_lru, lru);
+ node = &lnode->node;
- /* Don't forget active directories */
- if (node->refctr > 1)
- continue;
+ age = diff_timespec(&now, &lnode->forget_time);
+ if (age <= f->conf.remember)
+ break;
- /*
- * Only want to try the forget after the lookup count
- * has been reduced to 1 and the time to keep the node
- * around has expired
- */
- if (node->nlookup != 1)
- continue;
+ assert(node->nlookup == 1);
- age = diff_timespec(&now, &node->forget_time);
- if (age > f->conf.remember) {
- node->nlookup = 0;
- unhash_name(f, node);
- unref_node(f, node);
- }
- }
+ /* Don't forget active directories */
+ if (node->refctr > 1)
+ continue;
+
+ node->nlookup = 0;
+ unhash_name(f, node);
+ unref_node(f, node);
}
pthread_mutex_unlock(&f->lock);
- return next_clean;
+
+ return clean_delay(f);
}
static struct fuse_lowlevel_ops fuse_path_ops = {
if (!f)
return -1;
- if (f->conf.remember && f->conf.remember != -1)
+ if (lru_enabled(f))
return fuse_session_loop_remember(f);
return fuse_session_loop(f->se);
int fuse_start_cleanup_thread(struct fuse *f)
{
- if (f->conf.remember && f->conf.remember != -1)
+ if (lru_enabled(f))
return fuse_start_thread(&f->prune_thread, fuse_prune_nodes, f);
return 0;
void fuse_stop_cleanup_thread(struct fuse *f)
{
- if (f->conf.remember && f->conf.remember != -1) {
+ if (lru_enabled(f)) {
pthread_mutex_lock(&f->lock);
pthread_cancel(f->prune_thread);
pthread_mutex_unlock(&f->lock);
f->pagesize = getpagesize();
init_list_head(&f->partial_slabs);
init_list_head(&f->full_slabs);
+ init_list_head(&f->lru_table);
if (fuse_opt_parse(args, &f->conf, fuse_lib_opts,
fuse_lib_opt_proc) == -1)